mirror of
https://github.com/open-telemetry/opentelemetry-go.git
synced 2025-01-22 03:38:42 +02:00
38e76efe99
* Add a split protocol driver This is a wrapper around two other protocol drivers, so it makes it possible to send traces using a different protocol than the one used for metrics. * Add an example and tests for multi GRPC endpoint driver * Update changelog * Document the split driver
146 lines
4.4 KiB
Go
146 lines
4.4 KiB
Go
// Copyright The OpenTelemetry 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 otlp // import "go.opentelemetry.io/otel/exporters/otlp"
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
|
|
metricsdk "go.opentelemetry.io/otel/sdk/export/metric"
|
|
tracesdk "go.opentelemetry.io/otel/sdk/export/trace"
|
|
)
|
|
|
|
// ProtocolDriver is an interface used by OTLP exporter. It's
|
|
// responsible for connecting to and disconnecting from the collector,
|
|
// and for transforming traces and metrics into wire format and
|
|
// transmitting them to the collector.
|
|
type ProtocolDriver interface {
|
|
// Start should establish connection(s) to endpoint(s). It is
|
|
// called just once by the exporter, so the implementation
|
|
// does not need to worry about idempotence and locking.
|
|
Start(ctx context.Context) error
|
|
// Stop should close the connections. The function is called
|
|
// only once by the exporter, so the implementation does not
|
|
// need to worry about idempotence, but it may be called
|
|
// concurrently with ExportMetrics or ExportTraces, so proper
|
|
// locking is required. The function serves as a
|
|
// synchronization point - after the function returns, the
|
|
// process of closing connections is assumed to be finished.
|
|
Stop(ctx context.Context) error
|
|
// ExportMetrics should transform the passed metrics to the
|
|
// wire format and send it to the collector. May be called
|
|
// concurrently with ExportTraces, so the manager needs to
|
|
// take this into account by doing proper locking.
|
|
ExportMetrics(ctx context.Context, cps metricsdk.CheckpointSet, selector metricsdk.ExportKindSelector) error
|
|
// ExportTraces should transform the passed traces to the wire
|
|
// format and send it to the collector. May be called
|
|
// concurrently with ExportMetrics, so the manager needs to
|
|
// take this into account by doing proper locking.
|
|
ExportTraces(ctx context.Context, ss []*tracesdk.SpanSnapshot) error
|
|
}
|
|
|
|
// SplitConfig is used to configure a split driver.
|
|
type SplitConfig struct {
|
|
// ForMetrics driver will be used for sending metrics to the
|
|
// collector.
|
|
ForMetrics ProtocolDriver
|
|
// ForTraces driver will be used for sending spans to the
|
|
// collector.
|
|
ForTraces ProtocolDriver
|
|
}
|
|
|
|
type splitDriver struct {
|
|
metric ProtocolDriver
|
|
trace ProtocolDriver
|
|
}
|
|
|
|
var _ ProtocolDriver = (*splitDriver)(nil)
|
|
|
|
// NewSplitDriver creates a protocol driver which contains two other
|
|
// protocol drivers and will forward traces to one of them and metrics
|
|
// to another.
|
|
func NewSplitDriver(cfg SplitConfig) ProtocolDriver {
|
|
return &splitDriver{
|
|
metric: cfg.ForMetrics,
|
|
trace: cfg.ForTraces,
|
|
}
|
|
}
|
|
|
|
// Start implements ProtocolDriver. It starts both drivers at the same
|
|
// time.
|
|
func (d *splitDriver) Start(ctx context.Context) error {
|
|
wg := sync.WaitGroup{}
|
|
wg.Add(2)
|
|
var (
|
|
metricErr error
|
|
traceErr error
|
|
)
|
|
go func() {
|
|
defer wg.Done()
|
|
metricErr = d.metric.Start(ctx)
|
|
}()
|
|
go func() {
|
|
defer wg.Done()
|
|
traceErr = d.trace.Start(ctx)
|
|
}()
|
|
wg.Wait()
|
|
if metricErr != nil {
|
|
return metricErr
|
|
}
|
|
if traceErr != nil {
|
|
return traceErr
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Stop implements ProtocolDriver. It stops both drivers at the same
|
|
// time.
|
|
func (d *splitDriver) Stop(ctx context.Context) error {
|
|
wg := sync.WaitGroup{}
|
|
wg.Add(2)
|
|
var (
|
|
metricErr error
|
|
traceErr error
|
|
)
|
|
go func() {
|
|
defer wg.Done()
|
|
metricErr = d.metric.Stop(ctx)
|
|
}()
|
|
go func() {
|
|
defer wg.Done()
|
|
traceErr = d.trace.Stop(ctx)
|
|
}()
|
|
wg.Wait()
|
|
if metricErr != nil {
|
|
return metricErr
|
|
}
|
|
if traceErr != nil {
|
|
return traceErr
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ExportMetrics implements ProtocolDriver. It forwards the call to
|
|
// the driver used for sending metrics.
|
|
func (d *splitDriver) ExportMetrics(ctx context.Context, cps metricsdk.CheckpointSet, selector metricsdk.ExportKindSelector) error {
|
|
return d.metric.ExportMetrics(ctx, cps, selector)
|
|
}
|
|
|
|
// ExportTraces implements ProtocolDriver. It forwards the call to the
|
|
// driver used for sending spans.
|
|
func (d *splitDriver) ExportTraces(ctx context.Context, ss []*tracesdk.SpanSnapshot) error {
|
|
return d.trace.ExportTraces(ctx, ss)
|
|
}
|