2021-06-11 22:25:56 +02:00
|
|
|
// 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.
|
|
|
|
|
2021-10-29 18:34:37 +02:00
|
|
|
package otlpconfig // import "go.opentelemetry.io/otel/exporters/otlp/otlpmetric/internal/otlpconfig"
|
2021-06-11 22:25:56 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"fmt"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"google.golang.org/grpc"
|
2021-12-03 23:59:07 +02:00
|
|
|
"google.golang.org/grpc/backoff"
|
2021-06-11 22:25:56 +02:00
|
|
|
"google.golang.org/grpc/credentials"
|
2021-12-03 23:59:07 +02:00
|
|
|
"google.golang.org/grpc/encoding/gzip"
|
|
|
|
|
|
|
|
"go.opentelemetry.io/otel/exporters/otlp/otlpmetric/internal/retry"
|
2021-06-11 22:25:56 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2021-06-22 20:45:17 +02:00
|
|
|
// DefaultMaxAttempts describes how many times the driver
|
|
|
|
// should retry the sending of the payload in case of a
|
|
|
|
// retryable error.
|
|
|
|
DefaultMaxAttempts int = 5
|
2021-06-11 22:25:56 +02:00
|
|
|
// DefaultMetricsPath is a default URL path for endpoint that
|
|
|
|
// receives metrics.
|
|
|
|
DefaultMetricsPath string = "/v1/metrics"
|
2021-06-22 20:45:17 +02:00
|
|
|
// DefaultBackoff is a default base backoff time used in the
|
|
|
|
// exponential backoff strategy.
|
|
|
|
DefaultBackoff time.Duration = 300 * time.Millisecond
|
2021-06-11 22:25:56 +02:00
|
|
|
// DefaultTimeout is a default max waiting time for the backend to process
|
|
|
|
// each span or metrics batch.
|
|
|
|
DefaultTimeout time.Duration = 10 * time.Second
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
SignalConfig struct {
|
|
|
|
Endpoint string
|
|
|
|
Insecure bool
|
|
|
|
TLSCfg *tls.Config
|
|
|
|
Headers map[string]string
|
|
|
|
Compression Compression
|
|
|
|
Timeout time.Duration
|
|
|
|
URLPath string
|
|
|
|
|
|
|
|
// gRPC configurations
|
|
|
|
GRPCCredentials credentials.TransportCredentials
|
|
|
|
}
|
|
|
|
|
|
|
|
Config struct {
|
|
|
|
// Signal specific configurations
|
|
|
|
Metrics SignalConfig
|
|
|
|
|
2021-12-03 23:59:07 +02:00
|
|
|
RetryConfig retry.Config
|
2021-06-22 20:45:17 +02:00
|
|
|
|
2021-06-11 22:25:56 +02:00
|
|
|
// gRPC configurations
|
|
|
|
ReconnectionPeriod time.Duration
|
|
|
|
ServiceConfig string
|
|
|
|
DialOptions []grpc.DialOption
|
2021-10-11 21:23:12 +02:00
|
|
|
GRPCConn *grpc.ClientConn
|
2021-06-11 22:25:56 +02:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
func NewDefaultConfig() Config {
|
|
|
|
c := Config{
|
|
|
|
Metrics: SignalConfig{
|
|
|
|
Endpoint: fmt.Sprintf("%s:%d", DefaultCollectorHost, DefaultCollectorPort),
|
|
|
|
URLPath: DefaultMetricsPath,
|
|
|
|
Compression: NoCompression,
|
|
|
|
Timeout: DefaultTimeout,
|
|
|
|
},
|
2021-12-03 23:59:07 +02:00
|
|
|
RetryConfig: retry.DefaultConfig,
|
2021-06-11 22:25:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
2021-12-03 23:59:07 +02:00
|
|
|
// NewGRPCConfig returns a new Config with all settings applied from opts and
|
|
|
|
// any unset setting using the default gRPC config values.
|
|
|
|
func NewGRPCConfig(opts ...GRPCOption) Config {
|
|
|
|
cfg := NewDefaultConfig()
|
|
|
|
ApplyGRPCEnvConfigs(&cfg)
|
|
|
|
for _, opt := range opts {
|
|
|
|
opt.ApplyGRPCOption(&cfg)
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.ServiceConfig != "" {
|
|
|
|
cfg.DialOptions = append(cfg.DialOptions, grpc.WithDefaultServiceConfig(cfg.ServiceConfig))
|
|
|
|
}
|
|
|
|
if cfg.Metrics.GRPCCredentials != nil {
|
|
|
|
cfg.DialOptions = append(cfg.DialOptions, grpc.WithTransportCredentials(cfg.Metrics.GRPCCredentials))
|
|
|
|
} else if cfg.Metrics.Insecure {
|
|
|
|
cfg.DialOptions = append(cfg.DialOptions, grpc.WithInsecure())
|
|
|
|
}
|
|
|
|
if cfg.Metrics.Compression == GzipCompression {
|
|
|
|
cfg.DialOptions = append(cfg.DialOptions, grpc.WithDefaultCallOptions(grpc.UseCompressor(gzip.Name)))
|
|
|
|
}
|
|
|
|
if len(cfg.DialOptions) != 0 {
|
|
|
|
cfg.DialOptions = append(cfg.DialOptions, cfg.DialOptions...)
|
|
|
|
}
|
|
|
|
if cfg.ReconnectionPeriod != 0 {
|
|
|
|
p := grpc.ConnectParams{
|
|
|
|
Backoff: backoff.DefaultConfig,
|
|
|
|
MinConnectTimeout: cfg.ReconnectionPeriod,
|
|
|
|
}
|
|
|
|
cfg.DialOptions = append(cfg.DialOptions, grpc.WithConnectParams(p))
|
|
|
|
}
|
|
|
|
|
|
|
|
return cfg
|
|
|
|
}
|
|
|
|
|
2021-06-11 22:25:56 +02:00
|
|
|
type (
|
|
|
|
// GenericOption applies an option to the HTTP or gRPC driver.
|
|
|
|
GenericOption interface {
|
|
|
|
ApplyHTTPOption(*Config)
|
|
|
|
ApplyGRPCOption(*Config)
|
|
|
|
|
|
|
|
// A private method to prevent users implementing the
|
|
|
|
// interface and so future additions to it will not
|
|
|
|
// violate compatibility.
|
|
|
|
private()
|
|
|
|
}
|
|
|
|
|
|
|
|
// HTTPOption applies an option to the HTTP driver.
|
|
|
|
HTTPOption interface {
|
|
|
|
ApplyHTTPOption(*Config)
|
|
|
|
|
|
|
|
// A private method to prevent users implementing the
|
|
|
|
// interface and so future additions to it will not
|
|
|
|
// violate compatibility.
|
|
|
|
private()
|
|
|
|
}
|
|
|
|
|
|
|
|
// GRPCOption applies an option to the gRPC driver.
|
|
|
|
GRPCOption interface {
|
|
|
|
ApplyGRPCOption(*Config)
|
|
|
|
|
|
|
|
// A private method to prevent users implementing the
|
|
|
|
// interface and so future additions to it will not
|
|
|
|
// violate compatibility.
|
|
|
|
private()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
// genericOption is an option that applies the same logic
|
|
|
|
// for both gRPC and HTTP.
|
|
|
|
type genericOption struct {
|
|
|
|
fn func(*Config)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *genericOption) ApplyGRPCOption(cfg *Config) {
|
|
|
|
g.fn(cfg)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *genericOption) ApplyHTTPOption(cfg *Config) {
|
|
|
|
g.fn(cfg)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (genericOption) private() {}
|
|
|
|
|
|
|
|
func newGenericOption(fn func(cfg *Config)) GenericOption {
|
|
|
|
return &genericOption{fn: fn}
|
|
|
|
}
|
|
|
|
|
|
|
|
// splitOption is an option that applies different logics
|
|
|
|
// for gRPC and HTTP.
|
|
|
|
type splitOption struct {
|
|
|
|
httpFn func(*Config)
|
|
|
|
grpcFn func(*Config)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *splitOption) ApplyGRPCOption(cfg *Config) {
|
|
|
|
g.grpcFn(cfg)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *splitOption) ApplyHTTPOption(cfg *Config) {
|
|
|
|
g.httpFn(cfg)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (splitOption) private() {}
|
|
|
|
|
|
|
|
func newSplitOption(httpFn func(cfg *Config), grpcFn func(cfg *Config)) GenericOption {
|
|
|
|
return &splitOption{httpFn: httpFn, grpcFn: grpcFn}
|
|
|
|
}
|
|
|
|
|
|
|
|
// httpOption is an option that is only applied to the HTTP driver.
|
|
|
|
type httpOption struct {
|
|
|
|
fn func(*Config)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *httpOption) ApplyHTTPOption(cfg *Config) {
|
|
|
|
h.fn(cfg)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (httpOption) private() {}
|
|
|
|
|
|
|
|
func NewHTTPOption(fn func(cfg *Config)) HTTPOption {
|
|
|
|
return &httpOption{fn: fn}
|
|
|
|
}
|
|
|
|
|
|
|
|
// grpcOption is an option that is only applied to the gRPC driver.
|
|
|
|
type grpcOption struct {
|
|
|
|
fn func(*Config)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *grpcOption) ApplyGRPCOption(cfg *Config) {
|
|
|
|
h.fn(cfg)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (grpcOption) private() {}
|
|
|
|
|
|
|
|
func NewGRPCOption(fn func(cfg *Config)) GRPCOption {
|
|
|
|
return &grpcOption{fn: fn}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generic Options
|
|
|
|
|
|
|
|
func WithEndpoint(endpoint string) GenericOption {
|
|
|
|
return newGenericOption(func(cfg *Config) {
|
|
|
|
cfg.Metrics.Endpoint = endpoint
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithCompression(compression Compression) GenericOption {
|
|
|
|
return newGenericOption(func(cfg *Config) {
|
|
|
|
cfg.Metrics.Compression = compression
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithURLPath(urlPath string) GenericOption {
|
|
|
|
return newGenericOption(func(cfg *Config) {
|
|
|
|
cfg.Metrics.URLPath = urlPath
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-12-03 23:59:07 +02:00
|
|
|
func WithRetry(rc retry.Config) GenericOption {
|
2021-06-11 22:25:56 +02:00
|
|
|
return newGenericOption(func(cfg *Config) {
|
2021-12-03 23:59:07 +02:00
|
|
|
cfg.RetryConfig = rc
|
2021-06-11 22:25:56 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithTLSClientConfig(tlsCfg *tls.Config) GenericOption {
|
|
|
|
return newSplitOption(func(cfg *Config) {
|
|
|
|
cfg.Metrics.TLSCfg = tlsCfg.Clone()
|
|
|
|
}, func(cfg *Config) {
|
|
|
|
cfg.Metrics.GRPCCredentials = credentials.NewTLS(tlsCfg)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithInsecure() GenericOption {
|
|
|
|
return newGenericOption(func(cfg *Config) {
|
|
|
|
cfg.Metrics.Insecure = true
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithSecure() GenericOption {
|
|
|
|
return newGenericOption(func(cfg *Config) {
|
|
|
|
cfg.Metrics.Insecure = false
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithHeaders(headers map[string]string) GenericOption {
|
|
|
|
return newGenericOption(func(cfg *Config) {
|
|
|
|
cfg.Metrics.Headers = headers
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithTimeout(duration time.Duration) GenericOption {
|
|
|
|
return newGenericOption(func(cfg *Config) {
|
|
|
|
cfg.Metrics.Timeout = duration
|
|
|
|
})
|
|
|
|
}
|