5
5
- manual_instrumentation
6
6
weight : 30
7
7
description : Manual instrumentation for OpenTelemetry Go
8
- cSpell:ignore : fatalf otlptrace sdktrace sighup
8
+ cSpell:ignore : fatalf logr logrus otelslog otlplog otlploghttp sdktrace sighup
9
9
---
10
10
11
11
{{% docs/languages/instrumentation-intro %}}
@@ -37,7 +37,6 @@ import (
37
37
" log"
38
38
39
39
" go.opentelemetry.io/otel"
40
- " go.opentelemetry.io/otel/exporters/otlp/otlptrace"
41
40
" go.opentelemetry.io/otel/sdk/resource"
42
41
sdktrace " go.opentelemetry.io/otel/sdk/trace"
43
42
semconv " go.opentelemetry.io/otel/semconv/v1.21.0"
@@ -876,7 +875,161 @@ meterProvider := metric.NewMeterProvider(
876
875
877
876
## Logs
878
877
879
- The logs API is currently unstable, documentation TBA.
878
+ Logs are distinct from metrics and traces in that ** there is no user-facing
879
+ OpenTelemetry logs API** . Instead, there is tooling to bridge logs from existing
880
+ popular log packages (such as slog, logrus, zap, logr) into the OpenTelemetry
881
+ ecosystem. For rationale behind this design decision, see
882
+ [ Logging specification] ( /docs/specs/otel/logs/ ) .
883
+
884
+ The two typical workflows discussed below each cater to different application
885
+ requirements.
886
+
887
+ ### Direct-to-Collector
888
+
889
+ ** Status** : [ Experimental] ( /docs/specs/otel/document-status/ )
890
+
891
+ In the direct-to-Collector workflow, logs are emitted directly from an
892
+ application to a collector using a network protocol (e.g. OTLP). This workflow
893
+ is simple to set up as it doesn't require any additional log forwarding
894
+ components, and allows an application to easily emit structured logs that
895
+ conform to the [ log data model] [ log data model ] . However, the overhead required
896
+ for applications to queue and export logs to a network location may not be
897
+ suitable for all applications.
898
+
899
+ To use this workflow:
900
+
901
+ - Configure the OpenTelemetry [ Log SDK] ( #logs-sdk ) to export log records to
902
+ desired target destination (the [ collector] [ opentelemetry collector ] or
903
+ other).
904
+ - Use an appropriate [ Log Bridge] ( #log-bridge ) .
905
+
906
+ #### Logs SDK
907
+
908
+ The logs SDK dictates how logs are processed when using the
909
+ [ direct-to-Collector] ( #direct-to-collector ) workflow. No log SDK is needed when
910
+ using the [ log forwarding] ( #via-file-or-stdout ) workflow.
911
+
912
+ The typical log SDK configuration installs a batching log record processor with
913
+ an OTLP exporter.
914
+
915
+ To enable [ logs] ( /docs/concepts/signals/logs/ ) in your app, you'll need to have
916
+ an initialized [ ` LoggerProvider ` ] ( /docs/concepts/signals/logs/#logger-provider )
917
+ that will let you use a [ Log Bridge] ( #log-bridge ) .
918
+
919
+ If a ` LoggerProvider ` is not created, the OpenTelemetry APIs for logs will use a
920
+ no-op implementation and fail to generate data. Therefore, you have to modify
921
+ the source code to include the SDK initialization code using the following
922
+ packages:
923
+
924
+ - [ ` go.opentelemetry.io/otel ` ] [ ]
925
+ - [ ` go.opentelemetry.io/otel/sdk/log ` ] [ ]
926
+ - [ ` go.opentelemetry.io/otel/sdk/resource ` ] [ ]
927
+ - [ ` go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp ` ] [ ]
928
+
929
+ Ensure you have the right Go modules installed:
930
+
931
+ ``` sh
932
+ go get go.opentelemetry.io/otel \
933
+ go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp \
934
+ go.opentelemetry.io/otel/sdk \
935
+ go.opentelemetry.io/otel/sdk/log
936
+ ```
937
+
938
+ Then initialize a logger provider:
939
+
940
+ ``` go
941
+ package main
942
+
943
+ import (
944
+ " context"
945
+ " fmt"
946
+
947
+ " go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp"
948
+ " go.opentelemetry.io/otel/log/global"
949
+ " go.opentelemetry.io/otel/sdk/log"
950
+ " go.opentelemetry.io/otel/sdk/resource"
951
+ semconv " go.opentelemetry.io/otel/semconv/v1.21.0"
952
+ )
953
+
954
+ func main () {
955
+ ctx := context.Background ()
956
+
957
+ // Create resource.
958
+ res , err := newResource ()
959
+ if err != nil {
960
+ panic (err)
961
+ }
962
+
963
+ // Create a logger provider.
964
+ // You can pass this instance directly when creating bridges.
965
+ loggerProvider , err := newLoggerProvider (ctx, res)
966
+ if err != nil {
967
+ panic (err)
968
+ }
969
+
970
+ // Handle shutdown properly so nothing leaks.
971
+ defer func () {
972
+ if err := loggerProvider.Shutdown (ctx); err != nil {
973
+ fmt.Println (err)
974
+ }
975
+ }()
976
+
977
+ // Register as global logger provider so that it can be accessed global.LoggerProvider.
978
+ // Most log bridges use the global logger provider as default.
979
+ // If the global logger provider is not set then a no-op implementation
980
+ // is used, which fails to generate data.
981
+ global.SetLoggerProvider (loggerProvider)
982
+ }
983
+
984
+ func newResource () (*resource .Resource , error ) {
985
+ return resource.Merge (resource.Default (),
986
+ resource.NewWithAttributes (semconv.SchemaURL ,
987
+ semconv.ServiceName (" my-service" ),
988
+ semconv.ServiceVersion (" 0.1.0" ),
989
+ ))
990
+ }
991
+
992
+ func newLoggerProvider (ctx context .Context , res *resource .Resource ) (*log .LoggerProvider , error ) {
993
+ exporter , err := otlploghttp.New (ctx)
994
+ if err != nil {
995
+ return nil , err
996
+ }
997
+ processor := log.NewBatchProcessor (exporter)
998
+ provider := log.NewLoggerProvider (
999
+ log.WithResource (res),
1000
+ log.WithProcessor (processor),
1001
+ )
1002
+ return provider, nil
1003
+ }
1004
+ ```
1005
+
1006
+ Now that a ` LoggerProvider ` is configured, you can use it to set up a
1007
+ [ Log Bridge] ( #log-bridge ) .
1008
+
1009
+ #### Log Bridge
1010
+
1011
+ A log bridge is a component that bridges logs from an existing log package into
1012
+ the OpenTelemetry [ Log SDK] ( #logs-sdk ) using the [ Logs Bridge
1013
+ API] [ logs bridge API ] . Log bridges are available for various popular Go log
1014
+ packages:
1015
+
1016
+ - [ logrus bridge] [ otellogrus ]
1017
+ - [ slog bridge] [ otelslog ]
1018
+
1019
+ The links above contain full usage and installation documentation.
1020
+
1021
+ ### Via file or stdout
1022
+
1023
+ In the file or stdout workflow, logs are written to files or standout output.
1024
+ Another component (e.g. FluentBit) is responsible for reading / tailing the
1025
+ logs, parsing them to more structured format, and forwarding them a target, such
1026
+ as the collector. This workflow may be preferable in situations where
1027
+ application requirements do not permit additional overhead from
1028
+ [ direct-to-Collector] ( #direct-to-collector ) . However, it requires that all log
1029
+ fields required down stream are encoded into the logs, and that the component
1030
+ reading the logs parse the data into the [ log data model] [ log data model ] . The
1031
+ installation and configuration of log forwarding components is outside the scope
1032
+ of this document.
880
1033
881
1034
## Next Steps
882
1035
@@ -887,11 +1040,21 @@ telemetry backends.
887
1040
[ opentelemetry specification ] : /docs/specs/otel/
888
1041
[ trace semantic conventions ] : /docs/specs/semconv/general/trace/
889
1042
[ instrumentation library ] : ../libraries/
1043
+ [ opentelemetry collector] :
1044
+ https://github.com/open-telemetry/opentelemetry-collector
1045
+ [ logs bridge API ] : /docs/specs/otel/logs/bridge-api
1046
+ [ log data model ] : /docs/specs/otel/logs/data-model
1047
+ [ otellogrus ] : https://pkg.go.dev/go.opentelemetry.io/contrib/bridges/otellogrus
1048
+ [ otelslog ] : https://pkg.go.dev/go.opentelemetry.io/contrib/bridges/otelslog
890
1049
[ `go.opentelemetry.io/otel` ] : https://pkg.go.dev/go.opentelemetry.io/otel
891
1050
[ ` go.opentelemetry.io/otel/exporters/stdout/stdoutmetric ` ] :
892
1051
https://pkg.go.dev/go.opentelemetry.io/otel/exporters/stdout/stdoutmetric
893
1052
[ ` go.opentelemetry.io/otel/metric ` ] :
894
1053
https://pkg.go.dev/go.opentelemetry.io/otel/metric
1054
+ [ ` go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp ` ] :
1055
+ https://pkg.go.dev/go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp
1056
+ [ ` go.opentelemetry.io/otel/sdk/log ` ] :
1057
+ https://pkg.go.dev/go.opentelemetry.io/otel/sdk/log
895
1058
[ ` go.opentelemetry.io/otel/sdk/metric ` ] :
896
1059
https://pkg.go.dev/go.opentelemetry.io/otel/sdk/metric
897
1060
[ ` go.opentelemetry.io/otel/sdk/resource ` ] :
0 commit comments