mirror of
https://github.com/containers/skopeo.git
synced 2025-05-02 13:13:24 +00:00
> go get github.com/containers/image/v5@main > make vendor Signed-off-by: Miloslav Trmač <mitr@redhat.com>
208 lines
5.4 KiB
Go
208 lines
5.4 KiB
Go
package client
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
"strings"
|
|
|
|
"github.com/go-openapi/runtime"
|
|
"github.com/go-openapi/strfmt"
|
|
"go.opentelemetry.io/otel"
|
|
"go.opentelemetry.io/otel/attribute"
|
|
"go.opentelemetry.io/otel/codes"
|
|
"go.opentelemetry.io/otel/propagation"
|
|
semconv "go.opentelemetry.io/otel/semconv/v1.12.0"
|
|
"go.opentelemetry.io/otel/trace"
|
|
)
|
|
|
|
const (
|
|
instrumentationVersion = "1.0.0"
|
|
tracerName = "go-openapi"
|
|
)
|
|
|
|
type config struct {
|
|
Tracer trace.Tracer
|
|
Propagator propagation.TextMapPropagator
|
|
SpanStartOptions []trace.SpanStartOption
|
|
SpanNameFormatter func(*runtime.ClientOperation) string
|
|
TracerProvider trace.TracerProvider
|
|
}
|
|
|
|
type OpenTelemetryOpt interface {
|
|
apply(*config)
|
|
}
|
|
|
|
type optionFunc func(*config)
|
|
|
|
func (o optionFunc) apply(c *config) {
|
|
o(c)
|
|
}
|
|
|
|
// WithTracerProvider specifies a tracer provider to use for creating a tracer.
|
|
// If none is specified, the global provider is used.
|
|
func WithTracerProvider(provider trace.TracerProvider) OpenTelemetryOpt {
|
|
return optionFunc(func(c *config) {
|
|
if provider != nil {
|
|
c.TracerProvider = provider
|
|
}
|
|
})
|
|
}
|
|
|
|
// WithPropagators configures specific propagators. If this
|
|
// option isn't specified, then the global TextMapPropagator is used.
|
|
func WithPropagators(ps propagation.TextMapPropagator) OpenTelemetryOpt {
|
|
return optionFunc(func(c *config) {
|
|
if ps != nil {
|
|
c.Propagator = ps
|
|
}
|
|
})
|
|
}
|
|
|
|
// WithSpanOptions configures an additional set of
|
|
// trace.SpanOptions, which are applied to each new span.
|
|
func WithSpanOptions(opts ...trace.SpanStartOption) OpenTelemetryOpt {
|
|
return optionFunc(func(c *config) {
|
|
c.SpanStartOptions = append(c.SpanStartOptions, opts...)
|
|
})
|
|
}
|
|
|
|
// WithSpanNameFormatter takes a function that will be called on every
|
|
// request and the returned string will become the Span Name.
|
|
func WithSpanNameFormatter(f func(op *runtime.ClientOperation) string) OpenTelemetryOpt {
|
|
return optionFunc(func(c *config) {
|
|
c.SpanNameFormatter = f
|
|
})
|
|
}
|
|
|
|
func defaultTransportFormatter(op *runtime.ClientOperation) string {
|
|
if op.ID != "" {
|
|
return op.ID
|
|
}
|
|
|
|
return fmt.Sprintf("%s_%s", strings.ToLower(op.Method), op.PathPattern)
|
|
}
|
|
|
|
type openTelemetryTransport struct {
|
|
transport runtime.ClientTransport
|
|
host string
|
|
tracer trace.Tracer
|
|
config *config
|
|
}
|
|
|
|
func newOpenTelemetryTransport(transport runtime.ClientTransport, host string, opts []OpenTelemetryOpt) *openTelemetryTransport {
|
|
tr := &openTelemetryTransport{
|
|
transport: transport,
|
|
host: host,
|
|
}
|
|
|
|
defaultOpts := []OpenTelemetryOpt{
|
|
WithSpanOptions(trace.WithSpanKind(trace.SpanKindClient)),
|
|
WithSpanNameFormatter(defaultTransportFormatter),
|
|
WithPropagators(otel.GetTextMapPropagator()),
|
|
WithTracerProvider(otel.GetTracerProvider()),
|
|
}
|
|
|
|
c := newConfig(append(defaultOpts, opts...)...)
|
|
tr.config = c
|
|
|
|
return tr
|
|
}
|
|
|
|
func (t *openTelemetryTransport) Submit(op *runtime.ClientOperation) (interface{}, error) {
|
|
if op.Context == nil {
|
|
return t.transport.Submit(op)
|
|
}
|
|
|
|
params := op.Params
|
|
reader := op.Reader
|
|
|
|
var span trace.Span
|
|
defer func() {
|
|
if span != nil {
|
|
span.End()
|
|
}
|
|
}()
|
|
|
|
op.Params = runtime.ClientRequestWriterFunc(func(req runtime.ClientRequest, reg strfmt.Registry) error {
|
|
span = t.newOpenTelemetrySpan(op, req.GetHeaderParams())
|
|
return params.WriteToRequest(req, reg)
|
|
})
|
|
|
|
op.Reader = runtime.ClientResponseReaderFunc(func(response runtime.ClientResponse, consumer runtime.Consumer) (interface{}, error) {
|
|
if span != nil {
|
|
statusCode := response.Code()
|
|
span.SetAttributes(attribute.Int(string(semconv.HTTPStatusCodeKey), statusCode))
|
|
span.SetStatus(semconv.SpanStatusFromHTTPStatusCodeAndSpanKind(statusCode, trace.SpanKindClient))
|
|
}
|
|
|
|
return reader.ReadResponse(response, consumer)
|
|
})
|
|
|
|
submit, err := t.transport.Submit(op)
|
|
if err != nil && span != nil {
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
}
|
|
|
|
return submit, err
|
|
}
|
|
|
|
func (t *openTelemetryTransport) newOpenTelemetrySpan(op *runtime.ClientOperation, header http.Header) trace.Span {
|
|
ctx := op.Context
|
|
|
|
tracer := t.tracer
|
|
if tracer == nil {
|
|
if span := trace.SpanFromContext(ctx); span.SpanContext().IsValid() {
|
|
tracer = newTracer(span.TracerProvider())
|
|
} else {
|
|
tracer = newTracer(otel.GetTracerProvider())
|
|
}
|
|
}
|
|
|
|
ctx, span := tracer.Start(ctx, t.config.SpanNameFormatter(op), t.config.SpanStartOptions...)
|
|
|
|
var scheme string
|
|
if len(op.Schemes) > 0 {
|
|
scheme = op.Schemes[0]
|
|
}
|
|
|
|
span.SetAttributes(
|
|
attribute.String("net.peer.name", t.host),
|
|
attribute.String(string(semconv.HTTPRouteKey), op.PathPattern),
|
|
attribute.String(string(semconv.HTTPMethodKey), op.Method),
|
|
attribute.String("span.kind", trace.SpanKindClient.String()),
|
|
attribute.String("http.scheme", scheme),
|
|
)
|
|
|
|
carrier := propagation.HeaderCarrier(header)
|
|
t.config.Propagator.Inject(ctx, carrier)
|
|
|
|
return span
|
|
}
|
|
|
|
func newTracer(tp trace.TracerProvider) trace.Tracer {
|
|
return tp.Tracer(tracerName, trace.WithInstrumentationVersion(version()))
|
|
}
|
|
|
|
func newConfig(opts ...OpenTelemetryOpt) *config {
|
|
c := &config{
|
|
Propagator: otel.GetTextMapPropagator(),
|
|
}
|
|
|
|
for _, opt := range opts {
|
|
opt.apply(c)
|
|
}
|
|
|
|
// Tracer is only initialized if manually specified. Otherwise, can be passed with the tracing context.
|
|
if c.TracerProvider != nil {
|
|
c.Tracer = newTracer(c.TracerProvider)
|
|
}
|
|
|
|
return c
|
|
}
|
|
|
|
// Version is the current release version of the go-runtime instrumentation.
|
|
func version() string {
|
|
return instrumentationVersion
|
|
}
|