2023-06-08 22:46:07 +02:00
|
|
|
/*
|
|
|
|
Copyright 2023 Docker Compose CLI authors
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package tracing
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/docker/cli/cli/command"
|
|
|
|
"github.com/docker/cli/cli/context/store"
|
2024-03-12 14:47:41 +01:00
|
|
|
"github.com/docker/compose/v2/internal/memnet"
|
2023-06-08 22:46:07 +02:00
|
|
|
"go.opentelemetry.io/otel/exporters/otlp/otlptrace"
|
|
|
|
"go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc"
|
|
|
|
"google.golang.org/grpc"
|
|
|
|
"google.golang.org/grpc/credentials/insecure"
|
|
|
|
)
|
|
|
|
|
|
|
|
const otelConfigFieldName = "otel"
|
|
|
|
|
|
|
|
// traceClientFromDockerContext creates a gRPC OTLP client based on metadata
|
|
|
|
// from the active Docker CLI context.
|
|
|
|
func traceClientFromDockerContext(dockerCli command.Cli, otelEnv envMap) (otlptrace.Client, error) {
|
|
|
|
// attempt to extract an OTEL config from the Docker context to enable
|
|
|
|
// automatic integration with Docker Desktop;
|
|
|
|
cfg, err := ConfigFromDockerContext(dockerCli.ContextStore(), dockerCli.CurrentContext())
|
|
|
|
if err != nil {
|
2023-09-26 00:57:12 +02:00
|
|
|
return nil, fmt.Errorf("loading otel config from docker context metadata: %w", err)
|
2023-06-08 22:46:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.Endpoint == "" {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// HACK: unfortunately _all_ public OTEL initialization functions
|
|
|
|
// implicitly read from the OS env, so temporarily unset them all and
|
|
|
|
// restore afterwards
|
|
|
|
defer func() {
|
|
|
|
for k, v := range otelEnv {
|
|
|
|
if err := os.Setenv(k, v); err != nil {
|
2023-09-26 00:57:12 +02:00
|
|
|
panic(fmt.Errorf("restoring env for %q: %w", k, err))
|
2023-06-08 22:46:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
for k := range otelEnv {
|
|
|
|
if err := os.Unsetenv(k); err != nil {
|
2023-09-26 00:57:12 +02:00
|
|
|
return nil, fmt.Errorf("stashing env for %q: %w", k, err)
|
2023-06-08 22:46:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-21 14:21:43 +01:00
|
|
|
dialCtx, cancel := context.WithTimeout(context.Background(), 100*time.Millisecond)
|
2023-06-08 22:46:07 +02:00
|
|
|
defer cancel()
|
|
|
|
conn, err := grpc.DialContext(
|
|
|
|
dialCtx,
|
|
|
|
cfg.Endpoint,
|
2024-03-12 14:47:41 +01:00
|
|
|
grpc.WithContextDialer(memnet.DialEndpoint),
|
|
|
|
// this dial is restricted to using a local Unix socket / named pipe,
|
|
|
|
// so there is no need for TLS
|
2023-06-08 22:46:07 +02:00
|
|
|
grpc.WithTransportCredentials(insecure.NewCredentials()),
|
|
|
|
)
|
|
|
|
if err != nil {
|
2023-09-26 00:57:12 +02:00
|
|
|
return nil, fmt.Errorf("initializing otel connection from docker context metadata: %w", err)
|
2023-06-08 22:46:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
client := otlptracegrpc.NewClient(otlptracegrpc.WithGRPCConn(conn))
|
|
|
|
return client, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConfigFromDockerContext inspects extra metadata included as part of the
|
|
|
|
// specified Docker context to try and extract a valid OTLP client configuration.
|
|
|
|
func ConfigFromDockerContext(st store.Store, name string) (OTLPConfig, error) {
|
|
|
|
meta, err := st.GetMetadata(name)
|
|
|
|
if err != nil {
|
|
|
|
return OTLPConfig{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var otelCfg interface{}
|
|
|
|
switch m := meta.Metadata.(type) {
|
|
|
|
case command.DockerContext:
|
|
|
|
otelCfg = m.AdditionalFields[otelConfigFieldName]
|
|
|
|
case map[string]interface{}:
|
|
|
|
otelCfg = m[otelConfigFieldName]
|
|
|
|
}
|
2023-06-15 18:43:15 +02:00
|
|
|
if otelCfg == nil {
|
|
|
|
return OTLPConfig{}, nil
|
|
|
|
}
|
|
|
|
|
2023-06-08 22:46:07 +02:00
|
|
|
otelMap, ok := otelCfg.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
return OTLPConfig{}, fmt.Errorf(
|
|
|
|
"unexpected type for field %q: %T (expected: %T)",
|
|
|
|
otelConfigFieldName,
|
|
|
|
otelCfg,
|
|
|
|
otelMap,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// keys from https://opentelemetry.io/docs/concepts/sdk-configuration/otlp-exporter-configuration/
|
|
|
|
cfg := OTLPConfig{
|
|
|
|
Endpoint: valueOrDefault[string](otelMap, "OTEL_EXPORTER_OTLP_ENDPOINT"),
|
|
|
|
}
|
|
|
|
return cfg, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// valueOrDefault returns the type-cast value at the specified key in the map
|
|
|
|
// if present and the correct type; otherwise, it returns the default value for
|
|
|
|
// T.
|
|
|
|
func valueOrDefault[T any](m map[string]interface{}, key string) T {
|
|
|
|
if v, ok := m[key].(T); ok {
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
return *new(T)
|
|
|
|
}
|