-
Notifications
You must be signed in to change notification settings - Fork 209
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: configurable mock destination for pipeline testing (#2012)
This new exporter mocks a destination, and allows one to simulate various back-pressure scenarios, like long time to get a response for the export, and simulating some export requests being rejected. It is useful for pipeline development and testing
- Loading branch information
Showing
22 changed files
with
1,114 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
include ../../Makefile.Common |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
# Mock Destination Exporter | ||
|
||
This exporter can be used for development and testing. | ||
It allows you to mock a specific behavior of a destination exporter. | ||
|
||
## Configuration | ||
|
||
The following configuration options are available: | ||
|
||
- `response_duration` can be used to set the duration of time until the export response is returned. can be used to simulate slow receivers (due to errors, network issues, etc). | ||
- `reject_fraction` number from 0 to 1 that determines the fraction of exports that mocks a rejection of the export request. | ||
|
||
Example: | ||
|
||
```yaml | ||
│ mockdestination: | ||
│ reject_fraction: 0.5 | ||
│ response_duration: 500ms | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,33 @@ | ||
package mockdestinationexporter | ||
|
||
import ( | ||
"fmt" | ||
"time" | ||
|
||
"go.opentelemetry.io/collector/component" | ||
) | ||
|
||
// Config contains the main configuration options for the mockdestination exporter | ||
type Config struct { | ||
|
||
// ResponseDuration is the amount of time the exporter will wait before returning a response. | ||
// It can be used to simulate loaded and slow destinations. | ||
ResponseDuration time.Duration `mapstructure:"response_duration"` | ||
|
||
// RejectFraction is the fraction of exports that will randomly be rejected. | ||
// Set to 0 to disable rejection, and to 1 to reject all exports. | ||
// Can be used to simulate destinations that are back-pressuring the collector. | ||
RejectFraction float64 `mapstructure:"reject_fraction"` | ||
} | ||
|
||
func (c *Config) Validate() error { | ||
if c.ResponseDuration < 0 { | ||
return fmt.Errorf("response_duration must be a non-negative duration") | ||
} | ||
if c.RejectFraction < 0 || c.RejectFraction > 1 { | ||
return fmt.Errorf("reject_fraction must be a fraction between 0 and 1") | ||
} | ||
return nil | ||
} | ||
|
||
var _ component.Config = (*Config)(nil) |
15 changes: 15 additions & 0 deletions
15
collector/exporters/mockdestinationexporter/config_test.go
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
package mockdestinationexporter | ||
|
||
import ( | ||
"testing" | ||
|
||
"github.com/stretchr/testify/assert" | ||
"go.opentelemetry.io/collector/confmap" | ||
) | ||
|
||
func TestUnmarshalDefaultConfig(t *testing.T) { | ||
factory := NewFactory() | ||
cfg := factory.CreateDefaultConfig() | ||
assert.NoError(t, confmap.New().Unmarshal(&cfg)) | ||
assert.Equal(t, factory.CreateDefaultConfig(), cfg) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,62 @@ | ||
package mockdestinationexporter | ||
|
||
import ( | ||
"context" | ||
"errors" | ||
"math/rand/v2" | ||
"time" | ||
|
||
"go.opentelemetry.io/collector/exporter" | ||
|
||
"go.opentelemetry.io/collector/consumer" | ||
"go.opentelemetry.io/collector/pdata/plog" | ||
"go.opentelemetry.io/collector/pdata/pmetric" | ||
"go.opentelemetry.io/collector/pdata/ptrace" | ||
"go.uber.org/zap" | ||
) | ||
|
||
type MockDestinationExporter struct { | ||
config *Config | ||
logger *zap.Logger | ||
} | ||
|
||
func NewMockDestinationExporter(config *Config, | ||
params exporter.Settings) (*MockDestinationExporter, error) { | ||
|
||
if config == nil { | ||
return nil, errors.New("mock destination exporter config is nil") | ||
} | ||
|
||
logger := params.Logger | ||
|
||
mockDestinationExporter := &MockDestinationExporter{ | ||
config: config, | ||
logger: logger, | ||
} | ||
return mockDestinationExporter, nil | ||
} | ||
|
||
func (e *MockDestinationExporter) Capabilities() consumer.Capabilities { | ||
return consumer.Capabilities{MutatesData: false} | ||
} | ||
|
||
func (e *MockDestinationExporter) ConsumeTraces(ctx context.Context, traces ptrace.Traces) error { | ||
return e.mockExport(ctx) | ||
} | ||
|
||
func (e *MockDestinationExporter) ConsumeMetrics(ctx context.Context, metrics pmetric.Metrics) error { | ||
return e.mockExport(ctx) | ||
} | ||
|
||
func (e *MockDestinationExporter) ConsumeLogs(ctx context.Context, logs plog.Logs) error { | ||
return e.mockExport(ctx) | ||
} | ||
|
||
func (e *MockDestinationExporter) mockExport(context.Context) error { | ||
// not taking care of ctx cancel and shutdown as this is a dummy exporter and not used in production | ||
<-time.After(e.config.ResponseDuration) | ||
if rand.Float64() < e.config.RejectFraction { | ||
return errors.New("export rejected by mock destination") | ||
} | ||
return nil | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,85 @@ | ||
package mockdestinationexporter | ||
|
||
import ( | ||
"context" | ||
"time" | ||
|
||
"github.com/open-telemetry/opentelemetry-collector-contrib/odigos/exporter/mockdestinationexporter/internal/metadata" | ||
"go.opentelemetry.io/collector/exporter" | ||
|
||
"go.opentelemetry.io/collector/component" | ||
"go.opentelemetry.io/collector/exporter/exporterhelper" | ||
) | ||
|
||
// NewFactory creates a factory for GCS exporter. | ||
func NewFactory() exporter.Factory { | ||
return exporter.NewFactory( | ||
metadata.Type, | ||
createDefaultConfig, | ||
exporter.WithLogs(createLogsExporter, component.StabilityLevelBeta), | ||
exporter.WithTraces(createTracesExporter, component.StabilityLevelBeta), | ||
exporter.WithMetrics(createMetricsExporter, component.StabilityLevelBeta)) | ||
} | ||
|
||
func createDefaultConfig() component.Config { | ||
return &Config{ | ||
ResponseDuration: time.Millisecond * 100, | ||
RejectFraction: 0, | ||
} | ||
} | ||
|
||
func createLogsExporter( | ||
ctx context.Context, | ||
set exporter.Settings, | ||
cfg component.Config) (exporter.Logs, error) { | ||
|
||
pCfg := cfg.(*Config) | ||
gcsExporter, err := NewMockDestinationExporter(pCfg, set) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
return exporterhelper.NewLogsExporter( | ||
ctx, | ||
set, | ||
cfg, | ||
gcsExporter.ConsumeLogs) | ||
} | ||
|
||
func createTracesExporter( | ||
ctx context.Context, | ||
set exporter.Settings, | ||
cfg component.Config) (exporter.Traces, error) { | ||
|
||
pCfg := cfg.(*Config) | ||
gcsExporter, err := NewMockDestinationExporter(pCfg, set) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
return exporterhelper.NewTracesExporter( | ||
ctx, | ||
set, | ||
cfg, | ||
gcsExporter.ConsumeTraces, | ||
) | ||
} | ||
|
||
func createMetricsExporter( | ||
ctx context.Context, | ||
set exporter.Settings, | ||
cfg component.Config) (exporter.Metrics, error) { | ||
|
||
pCfg := cfg.(*Config) | ||
gcsExporter, err := NewMockDestinationExporter(pCfg, set) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
return exporterhelper.NewMetricsExporter( | ||
ctx, | ||
set, | ||
cfg, | ||
gcsExporter.ConsumeMetrics, | ||
) | ||
} |
150 changes: 150 additions & 0 deletions
150
collector/exporters/mockdestinationexporter/generated_component_test.go
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
Oops, something went wrong.