2551 lines
		
	
	
		
			103 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			2551 lines
		
	
	
		
			103 KiB
		
	
	
	
		
			Go
		
	
	
	
// Copyright The OpenTelemetry Authors
 | 
						|
// SPDX-License-Identifier: Apache-2.0
 | 
						|
 | 
						|
package graph
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"strings"
 | 
						|
	"sync"
 | 
						|
	"testing"
 | 
						|
 | 
						|
	"github.com/stretchr/testify/assert"
 | 
						|
	"github.com/stretchr/testify/require"
 | 
						|
	"gonum.org/v1/gonum/graph/simple"
 | 
						|
 | 
						|
	"go.opentelemetry.io/collector/component"
 | 
						|
	"go.opentelemetry.io/collector/component/componenttest"
 | 
						|
	"go.opentelemetry.io/collector/connector"
 | 
						|
	"go.opentelemetry.io/collector/connector/connectortest"
 | 
						|
	"go.opentelemetry.io/collector/consumer"
 | 
						|
	"go.opentelemetry.io/collector/consumer/consumertest"
 | 
						|
	"go.opentelemetry.io/collector/exporter"
 | 
						|
	"go.opentelemetry.io/collector/exporter/exportertest"
 | 
						|
	"go.opentelemetry.io/collector/internal/testdata"
 | 
						|
	"go.opentelemetry.io/collector/processor"
 | 
						|
	"go.opentelemetry.io/collector/processor/processortest"
 | 
						|
	"go.opentelemetry.io/collector/receiver"
 | 
						|
	"go.opentelemetry.io/collector/receiver/receivertest"
 | 
						|
	"go.opentelemetry.io/collector/service/internal/servicetelemetry"
 | 
						|
	"go.opentelemetry.io/collector/service/internal/status"
 | 
						|
	"go.opentelemetry.io/collector/service/internal/testcomponents"
 | 
						|
	"go.opentelemetry.io/collector/service/pipelines"
 | 
						|
)
 | 
						|
 | 
						|
var _ component.Component = &testNode{}
 | 
						|
 | 
						|
type testNode struct {
 | 
						|
	id          component.ID
 | 
						|
	startErr    error
 | 
						|
	shutdownErr error
 | 
						|
}
 | 
						|
 | 
						|
// ID satisfies the graph.Node interface, allowing
 | 
						|
// testNode to be used in a simple.DirectedGraph
 | 
						|
func (n *testNode) ID() int64 {
 | 
						|
	return int64(newNodeID(n.id.String()))
 | 
						|
}
 | 
						|
 | 
						|
func (n *testNode) Start(ctx context.Context, _ component.Host) error {
 | 
						|
	if n.startErr != nil {
 | 
						|
		return n.startErr
 | 
						|
	}
 | 
						|
	if cwo, ok := ctx.(*contextWithOrder); ok {
 | 
						|
		cwo.record(n.id)
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (n *testNode) Shutdown(ctx context.Context) error {
 | 
						|
	if n.shutdownErr != nil {
 | 
						|
		return n.shutdownErr
 | 
						|
	}
 | 
						|
	if cwo, ok := ctx.(*contextWithOrder); ok {
 | 
						|
		cwo.record(n.id)
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
type contextWithOrder struct {
 | 
						|
	context.Context
 | 
						|
	sync.Mutex
 | 
						|
	next  int
 | 
						|
	order map[component.ID]int
 | 
						|
}
 | 
						|
 | 
						|
func (c *contextWithOrder) record(id component.ID) {
 | 
						|
	c.Lock()
 | 
						|
	c.order[id] = c.next
 | 
						|
	c.next++
 | 
						|
	c.Unlock()
 | 
						|
}
 | 
						|
 | 
						|
func TestGraphStartStop(t *testing.T) {
 | 
						|
	testCases := []struct {
 | 
						|
		name  string
 | 
						|
		edges [][2]component.ID
 | 
						|
	}{
 | 
						|
		{
 | 
						|
			name: "single",
 | 
						|
			edges: [][2]component.ID{
 | 
						|
				{component.NewIDWithName("r", "1"), component.NewIDWithName("p", "1")},
 | 
						|
				{component.NewIDWithName("r", "2"), component.NewIDWithName("p", "1")},
 | 
						|
				{component.NewIDWithName("p", "1"), component.NewIDWithName("p", "2")},
 | 
						|
				{component.NewIDWithName("p", "2"), component.NewIDWithName("e", "1")},
 | 
						|
				{component.NewIDWithName("p", "1"), component.NewIDWithName("e", "2")},
 | 
						|
			},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "multi",
 | 
						|
			edges: [][2]component.ID{
 | 
						|
				// Pipeline 1
 | 
						|
				{component.NewIDWithName("r", "1"), component.NewIDWithName("p", "1")},
 | 
						|
				{component.NewIDWithName("r", "2"), component.NewIDWithName("p", "1")},
 | 
						|
				{component.NewIDWithName("p", "1"), component.NewIDWithName("p", "2")},
 | 
						|
				{component.NewIDWithName("p", "2"), component.NewIDWithName("e", "1")},
 | 
						|
				{component.NewIDWithName("p", "1"), component.NewIDWithName("e", "2")},
 | 
						|
 | 
						|
				// Pipeline 2, shares r1 and e2
 | 
						|
				{component.NewIDWithName("r", "1"), component.NewIDWithName("p", "3")},
 | 
						|
				{component.NewIDWithName("p", "3"), component.NewIDWithName("e", "2")},
 | 
						|
			},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "connected",
 | 
						|
			edges: [][2]component.ID{
 | 
						|
				// Pipeline 1
 | 
						|
				{component.NewIDWithName("r", "1"), component.NewIDWithName("p", "1")},
 | 
						|
				{component.NewIDWithName("r", "2"), component.NewIDWithName("p", "1")},
 | 
						|
				{component.NewIDWithName("p", "1"), component.NewIDWithName("p", "2")},
 | 
						|
				{component.NewIDWithName("p", "2"), component.NewIDWithName("e", "1")},
 | 
						|
				{component.NewIDWithName("p", "1"), component.NewIDWithName("c", "1")},
 | 
						|
 | 
						|
				// Pipeline 2, shares r1 and c1
 | 
						|
				{component.NewIDWithName("r", "1"), component.NewIDWithName("p", "3")},
 | 
						|
				{component.NewIDWithName("p", "3"), component.NewIDWithName("c", "1")},
 | 
						|
 | 
						|
				// Pipeline 3, emits to e2 and c2
 | 
						|
				{component.NewIDWithName("c", "1"), component.NewIDWithName("e", "2")},
 | 
						|
				{component.NewIDWithName("c", "1"), component.NewIDWithName("c", "2")},
 | 
						|
 | 
						|
				// Pipeline 4, also emits to e2
 | 
						|
				{component.NewIDWithName("c", "2"), component.NewIDWithName("e", "2")},
 | 
						|
			},
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	for _, tt := range testCases {
 | 
						|
		t.Run(tt.name, func(t *testing.T) {
 | 
						|
			ctx := &contextWithOrder{
 | 
						|
				Context: context.Background(),
 | 
						|
				order:   map[component.ID]int{},
 | 
						|
			}
 | 
						|
 | 
						|
			pg := &Graph{componentGraph: simple.NewDirectedGraph()}
 | 
						|
			pg.telemetry = servicetelemetry.NewNopTelemetrySettings()
 | 
						|
			pg.instanceIDs = make(map[int64]*component.InstanceID)
 | 
						|
 | 
						|
			for _, edge := range tt.edges {
 | 
						|
				f, t := &testNode{id: edge[0]}, &testNode{id: edge[1]}
 | 
						|
				pg.instanceIDs[f.ID()] = &component.InstanceID{}
 | 
						|
				pg.instanceIDs[t.ID()] = &component.InstanceID{}
 | 
						|
				pg.componentGraph.SetEdge(simple.Edge{F: f, T: t})
 | 
						|
			}
 | 
						|
 | 
						|
			require.NoError(t, pg.StartAll(ctx, componenttest.NewNopHost()))
 | 
						|
			for _, edge := range tt.edges {
 | 
						|
				assert.Greater(t, ctx.order[edge[0]], ctx.order[edge[1]])
 | 
						|
			}
 | 
						|
 | 
						|
			ctx.order = map[component.ID]int{}
 | 
						|
			require.NoError(t, pg.ShutdownAll(ctx))
 | 
						|
			for _, edge := range tt.edges {
 | 
						|
				assert.Less(t, ctx.order[edge[0]], ctx.order[edge[1]])
 | 
						|
			}
 | 
						|
		})
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func TestGraphStartStopCycle(t *testing.T) {
 | 
						|
	pg := &Graph{componentGraph: simple.NewDirectedGraph()}
 | 
						|
 | 
						|
	r1 := &testNode{id: component.NewIDWithName("r", "1")}
 | 
						|
	p1 := &testNode{id: component.NewIDWithName("p", "1")}
 | 
						|
	c1 := &testNode{id: component.NewIDWithName("c", "1")}
 | 
						|
	e1 := &testNode{id: component.NewIDWithName("e", "1")}
 | 
						|
 | 
						|
	pg.instanceIDs = map[int64]*component.InstanceID{
 | 
						|
		r1.ID(): {},
 | 
						|
		p1.ID(): {},
 | 
						|
		c1.ID(): {},
 | 
						|
		e1.ID(): {},
 | 
						|
	}
 | 
						|
 | 
						|
	pg.componentGraph.SetEdge(simple.Edge{F: r1, T: p1})
 | 
						|
	pg.componentGraph.SetEdge(simple.Edge{F: p1, T: c1})
 | 
						|
	pg.componentGraph.SetEdge(simple.Edge{F: c1, T: e1})
 | 
						|
	pg.componentGraph.SetEdge(simple.Edge{F: c1, T: p1}) // loop back
 | 
						|
 | 
						|
	err := pg.StartAll(context.Background(), componenttest.NewNopHost())
 | 
						|
	assert.Error(t, err)
 | 
						|
	assert.Contains(t, err.Error(), `topo: no topological ordering: cyclic components`)
 | 
						|
 | 
						|
	err = pg.ShutdownAll(context.Background())
 | 
						|
	assert.Error(t, err)
 | 
						|
	assert.Contains(t, err.Error(), `topo: no topological ordering: cyclic components`)
 | 
						|
}
 | 
						|
 | 
						|
func TestGraphStartStopComponentError(t *testing.T) {
 | 
						|
	pg := &Graph{componentGraph: simple.NewDirectedGraph()}
 | 
						|
	pg.telemetry = servicetelemetry.NewNopTelemetrySettings()
 | 
						|
	r1 := &testNode{
 | 
						|
		id:       component.NewIDWithName("r", "1"),
 | 
						|
		startErr: errors.New("foo"),
 | 
						|
	}
 | 
						|
	e1 := &testNode{
 | 
						|
		id:          component.NewIDWithName("e", "1"),
 | 
						|
		shutdownErr: errors.New("bar"),
 | 
						|
	}
 | 
						|
	pg.instanceIDs = map[int64]*component.InstanceID{
 | 
						|
		r1.ID(): {},
 | 
						|
		e1.ID(): {},
 | 
						|
	}
 | 
						|
	pg.componentGraph.SetEdge(simple.Edge{
 | 
						|
		F: r1,
 | 
						|
		T: e1,
 | 
						|
	})
 | 
						|
	assert.EqualError(t, pg.StartAll(context.Background(), componenttest.NewNopHost()), "foo")
 | 
						|
	assert.EqualError(t, pg.ShutdownAll(context.Background()), "bar")
 | 
						|
}
 | 
						|
 | 
						|
func TestConnectorPipelinesGraph(t *testing.T) {
 | 
						|
	tests := []struct {
 | 
						|
		name                string
 | 
						|
		pipelineConfigs     pipelines.Config
 | 
						|
		expectedPerExporter int // requires symmetry in pipelines
 | 
						|
	}{
 | 
						|
		{
 | 
						|
			name: "pipelines_simple.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_simple_mutate.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_simple_multi_proc.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor"), component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor"), component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor"), component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_simple_no_proc.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_multi.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver"), component.NewIDWithName("examplereceiver", "1")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate"), component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter"), component.NewIDWithName("exampleexporter", "1")},
 | 
						|
				},
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver"), component.NewIDWithName("examplereceiver", "1")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate"), component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter"), component.NewIDWithName("exampleexporter", "1")},
 | 
						|
				},
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver"), component.NewIDWithName("examplereceiver", "1")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate"), component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter"), component.NewIDWithName("exampleexporter", "1")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 2,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_multi_no_proc.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver"), component.NewIDWithName("examplereceiver", "1")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter"), component.NewIDWithName("exampleexporter", "1")},
 | 
						|
				},
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver"), component.NewIDWithName("examplereceiver", "1")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter"), component.NewIDWithName("exampleexporter", "1")},
 | 
						|
				},
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver"), component.NewIDWithName("examplereceiver", "1")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter"), component.NewIDWithName("exampleexporter", "1")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 2,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "multi_pipeline_receivers_and_exporters.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "1"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "1"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "1"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 2,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_simple_traces.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")}, // mutate propagates upstream to connector
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_simple_metrics.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")}, // mutate propagates upstream to connector
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_simple_logs.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("logs", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")}, // mutate propagates upstream to connector
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_fork_merge_traces.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "type0"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "merge")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "type1"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")}, // mutate propagates upstream to connector
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "merge")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "merge")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 2,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_fork_merge_metrics.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "type0"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "merge")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "type1"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")}, // mutate propagates upstream to connector
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "merge")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "merge")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 2,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_fork_merge_logs.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("logs", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "type0"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "merge")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "type1"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")}, // mutate propagates upstream to connector
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "merge")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "merge")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 2,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_translate_from_traces.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleconnector")},
 | 
						|
				},
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("exampleconnector")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("exampleconnector")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_translate_from_metrics.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleconnector")},
 | 
						|
				},
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("exampleconnector")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("exampleconnector")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_translate_from_logs.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleconnector")},
 | 
						|
				},
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("exampleconnector")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("exampleconnector")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_matrix.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")}, // mutate propagates upstream to connector
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")}, // mutate propagates upstream to connector
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewIDWithName("exampleprocessor", "mutate")}, // mutate propagates upstream to connector
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 3,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_lanes.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Exporters: []component.ID{component.NewID("mockforward")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("mockforward")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Exporters: []component.ID{component.NewID("mockforward")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("mockforward")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Exporters: []component.ID{component.NewID("mockforward")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("mockforward")},
 | 
						|
					Exporters: []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 1,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_mutate_traces.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out0"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "middle"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out1"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 2,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_mutate_metrics.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out0"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "middle"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out1"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 2,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "pipelines_conn_mutate_logs.yaml",
 | 
						|
			pipelineConfigs: pipelines.Config{
 | 
						|
				component.NewIDWithName("logs", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("examplereceiver")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "out0"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "middle"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "inherit_mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("exampleconnector", "mutate")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "out1"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("exampleconnector", "mutate")},
 | 
						|
					Processors: []component.ID{component.NewID("exampleprocessor")},
 | 
						|
					Exporters:  []component.ID{component.NewID("exampleexporter")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expectedPerExporter: 2,
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	for _, test := range tests {
 | 
						|
		t.Run(test.name, func(t *testing.T) {
 | 
						|
			// Build the pipeline
 | 
						|
			set := Settings{
 | 
						|
				Telemetry: servicetelemetry.NewNopTelemetrySettings(),
 | 
						|
				BuildInfo: component.NewDefaultBuildInfo(),
 | 
						|
				ReceiverBuilder: receiver.NewBuilder(
 | 
						|
					map[component.ID]component.Config{
 | 
						|
						component.NewID("examplereceiver"):              testcomponents.ExampleReceiverFactory.CreateDefaultConfig(),
 | 
						|
						component.NewIDWithName("examplereceiver", "1"): testcomponents.ExampleReceiverFactory.CreateDefaultConfig(),
 | 
						|
					},
 | 
						|
					map[component.Type]receiver.Factory{
 | 
						|
						testcomponents.ExampleReceiverFactory.Type(): testcomponents.ExampleReceiverFactory,
 | 
						|
					},
 | 
						|
				),
 | 
						|
				ProcessorBuilder: processor.NewBuilder(
 | 
						|
					map[component.ID]component.Config{
 | 
						|
						component.NewID("exampleprocessor"):                   testcomponents.ExampleProcessorFactory.CreateDefaultConfig(),
 | 
						|
						component.NewIDWithName("exampleprocessor", "mutate"): testcomponents.ExampleProcessorFactory.CreateDefaultConfig(),
 | 
						|
					},
 | 
						|
					map[component.Type]processor.Factory{
 | 
						|
						testcomponents.ExampleProcessorFactory.Type(): testcomponents.ExampleProcessorFactory,
 | 
						|
					},
 | 
						|
				),
 | 
						|
				ExporterBuilder: exporter.NewBuilder(
 | 
						|
					map[component.ID]component.Config{
 | 
						|
						component.NewID("exampleexporter"):              testcomponents.ExampleExporterFactory.CreateDefaultConfig(),
 | 
						|
						component.NewIDWithName("exampleexporter", "1"): testcomponents.ExampleExporterFactory.CreateDefaultConfig(),
 | 
						|
					},
 | 
						|
					map[component.Type]exporter.Factory{
 | 
						|
						testcomponents.ExampleExporterFactory.Type(): testcomponents.ExampleExporterFactory,
 | 
						|
					},
 | 
						|
				),
 | 
						|
				ConnectorBuilder: connector.NewBuilder(
 | 
						|
					map[component.ID]component.Config{
 | 
						|
						component.NewID("exampleconnector"):                           testcomponents.ExampleConnectorFactory.CreateDefaultConfig(),
 | 
						|
						component.NewIDWithName("exampleconnector", "merge"):          testcomponents.ExampleConnectorFactory.CreateDefaultConfig(),
 | 
						|
						component.NewIDWithName("exampleconnector", "mutate"):         testcomponents.ExampleConnectorFactory.CreateDefaultConfig(),
 | 
						|
						component.NewIDWithName("exampleconnector", "inherit_mutate"): testcomponents.ExampleConnectorFactory.CreateDefaultConfig(),
 | 
						|
						component.NewID("mockforward"):                                testcomponents.MockForwardConnectorFactory.CreateDefaultConfig(),
 | 
						|
					},
 | 
						|
					map[component.Type]connector.Factory{
 | 
						|
						testcomponents.ExampleConnectorFactory.Type():     testcomponents.ExampleConnectorFactory,
 | 
						|
						testcomponents.MockForwardConnectorFactory.Type(): testcomponents.MockForwardConnectorFactory,
 | 
						|
					},
 | 
						|
				),
 | 
						|
				PipelineConfigs: test.pipelineConfigs,
 | 
						|
			}
 | 
						|
 | 
						|
			pg, err := Build(context.Background(), set)
 | 
						|
			require.NoError(t, err)
 | 
						|
 | 
						|
			assert.Equal(t, len(test.pipelineConfigs), len(pg.pipelines))
 | 
						|
 | 
						|
			assert.NoError(t, pg.StartAll(context.Background(), componenttest.NewNopHost()))
 | 
						|
 | 
						|
			mutatingPipelines := make(map[component.ID]bool, len(test.pipelineConfigs))
 | 
						|
 | 
						|
			// Check each pipeline individually, ensuring that all components are started
 | 
						|
			// and that they have observed no signals yet.
 | 
						|
			for pipelineID, pipelineCfg := range test.pipelineConfigs {
 | 
						|
				pipeline, ok := pg.pipelines[pipelineID]
 | 
						|
				require.True(t, ok, "expected to find pipeline: %s", pipelineID.String())
 | 
						|
 | 
						|
				// Determine independently if the capabilities node should report MutateData as true
 | 
						|
				var expectMutatesData bool
 | 
						|
				for _, expr := range pipelineCfg.Exporters {
 | 
						|
					if strings.Contains(expr.Name(), "mutate") {
 | 
						|
						expectMutatesData = true
 | 
						|
					}
 | 
						|
				}
 | 
						|
				for _, proc := range pipelineCfg.Processors {
 | 
						|
					if proc.Name() == "mutate" {
 | 
						|
						expectMutatesData = true
 | 
						|
					}
 | 
						|
				}
 | 
						|
				assert.Equal(t, expectMutatesData, pipeline.capabilitiesNode.getConsumer().Capabilities().MutatesData)
 | 
						|
				mutatingPipelines[pipelineID] = expectMutatesData
 | 
						|
 | 
						|
				expectedReceivers, expectedExporters := expectedInstances(test.pipelineConfigs, pipelineID)
 | 
						|
				require.Equal(t, expectedReceivers, len(pipeline.receivers))
 | 
						|
				require.Equal(t, len(pipelineCfg.Processors), len(pipeline.processors))
 | 
						|
				require.Equal(t, expectedExporters, len(pipeline.exporters))
 | 
						|
 | 
						|
				for _, n := range pipeline.exporters {
 | 
						|
					switch c := n.(type) {
 | 
						|
					case *exporterNode:
 | 
						|
						e := c.Component.(*testcomponents.ExampleExporter)
 | 
						|
						require.True(t, e.Started())
 | 
						|
						require.Equal(t, 0, len(e.Traces))
 | 
						|
						require.Equal(t, 0, len(e.Metrics))
 | 
						|
						require.Equal(t, 0, len(e.Logs))
 | 
						|
					case *connectorNode:
 | 
						|
						// connector needs to be unwrapped to access component as ExampleConnector
 | 
						|
						switch ct := c.Component.(type) {
 | 
						|
						case connector.Traces:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Started())
 | 
						|
						case connector.Metrics:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Started())
 | 
						|
						case connector.Logs:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Started())
 | 
						|
						}
 | 
						|
					default:
 | 
						|
						require.Fail(t, fmt.Sprintf("unexpected type %T", c))
 | 
						|
					}
 | 
						|
				}
 | 
						|
 | 
						|
				for _, n := range pipeline.processors {
 | 
						|
					require.True(t, n.Component.(*testcomponents.ExampleProcessor).Started())
 | 
						|
				}
 | 
						|
 | 
						|
				for _, n := range pipeline.receivers {
 | 
						|
					switch c := n.(type) {
 | 
						|
					case *receiverNode:
 | 
						|
						require.True(t, c.Component.(*testcomponents.ExampleReceiver).Started())
 | 
						|
					case *connectorNode:
 | 
						|
						// connector needs to be unwrapped to access component as ExampleConnector
 | 
						|
						switch ct := c.Component.(type) {
 | 
						|
						case connector.Traces:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Started())
 | 
						|
						case connector.Metrics:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Started())
 | 
						|
						case connector.Logs:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Started())
 | 
						|
						}
 | 
						|
					default:
 | 
						|
						require.Fail(t, fmt.Sprintf("unexpected type %T", c))
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			// Check that connectors are correctly inheriting mutability from downstream pipelines
 | 
						|
			for expPipelineID, expPipeline := range pg.pipelines {
 | 
						|
				for _, exp := range expPipeline.exporters {
 | 
						|
					expConn, ok := exp.(*connectorNode)
 | 
						|
					if !ok {
 | 
						|
						continue
 | 
						|
					}
 | 
						|
					if expConn.getConsumer().Capabilities().MutatesData {
 | 
						|
						continue
 | 
						|
					}
 | 
						|
					// find all the pipelines of the same type where this connector is a receiver
 | 
						|
					var inheritMutatesData bool
 | 
						|
					for recPipelineID, recPipeline := range pg.pipelines {
 | 
						|
						if recPipelineID == expPipelineID || recPipelineID.Type() != expPipelineID.Type() {
 | 
						|
							continue
 | 
						|
						}
 | 
						|
						for _, rec := range recPipeline.receivers {
 | 
						|
							recConn, ok := rec.(*connectorNode)
 | 
						|
							if !ok || recConn.ID() != expConn.ID() {
 | 
						|
								continue
 | 
						|
							}
 | 
						|
							inheritMutatesData = inheritMutatesData || mutatingPipelines[recPipelineID]
 | 
						|
						}
 | 
						|
					}
 | 
						|
					assert.Equal(t, inheritMutatesData, expConn.getConsumer().Capabilities().MutatesData)
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			// Push data into the pipelines. The list of receivers is retrieved directly from the overall
 | 
						|
			// component graph because we do not want to duplicate signal inputs to receivers that are
 | 
						|
			// shared between pipelines. The `allReceivers` function also excludes connectors, which we do
 | 
						|
			// not want to directly inject with signals.
 | 
						|
			allReceivers := pg.getReceivers()
 | 
						|
			for _, c := range allReceivers[component.DataTypeTraces] {
 | 
						|
				tracesReceiver := c.(*testcomponents.ExampleReceiver)
 | 
						|
				assert.NoError(t, tracesReceiver.ConsumeTraces(context.Background(), testdata.GenerateTraces(1)))
 | 
						|
			}
 | 
						|
			for _, c := range allReceivers[component.DataTypeMetrics] {
 | 
						|
				metricsReceiver := c.(*testcomponents.ExampleReceiver)
 | 
						|
				assert.NoError(t, metricsReceiver.ConsumeMetrics(context.Background(), testdata.GenerateMetrics(1)))
 | 
						|
			}
 | 
						|
			for _, c := range allReceivers[component.DataTypeLogs] {
 | 
						|
				logsReceiver := c.(*testcomponents.ExampleReceiver)
 | 
						|
				assert.NoError(t, logsReceiver.ConsumeLogs(context.Background(), testdata.GenerateLogs(1)))
 | 
						|
			}
 | 
						|
 | 
						|
			// Shut down the entire component graph
 | 
						|
			assert.NoError(t, pg.ShutdownAll(context.Background()))
 | 
						|
 | 
						|
			// Check each pipeline individually, ensuring that all components are stopped.
 | 
						|
			for pipelineID := range test.pipelineConfigs {
 | 
						|
				pipeline, ok := pg.pipelines[pipelineID]
 | 
						|
				require.True(t, ok, "expected to find pipeline: %s", pipelineID.String())
 | 
						|
 | 
						|
				for _, n := range pipeline.receivers {
 | 
						|
					switch c := n.(type) {
 | 
						|
					case *receiverNode:
 | 
						|
						require.True(t, c.Component.(*testcomponents.ExampleReceiver).Stopped())
 | 
						|
					case *connectorNode:
 | 
						|
						// connector needs to be unwrapped to access component as ExampleConnector
 | 
						|
						switch ct := c.Component.(type) {
 | 
						|
						case connector.Traces:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Stopped())
 | 
						|
						case connector.Metrics:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Stopped())
 | 
						|
						case connector.Logs:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Stopped())
 | 
						|
						}
 | 
						|
					default:
 | 
						|
						require.Fail(t, fmt.Sprintf("unexpected type %T", c))
 | 
						|
					}
 | 
						|
				}
 | 
						|
 | 
						|
				for _, n := range pipeline.processors {
 | 
						|
					require.True(t, n.Component.(*testcomponents.ExampleProcessor).Stopped())
 | 
						|
				}
 | 
						|
 | 
						|
				for _, n := range pipeline.exporters {
 | 
						|
					switch c := n.(type) {
 | 
						|
					case *exporterNode:
 | 
						|
						e := c.Component.(*testcomponents.ExampleExporter)
 | 
						|
						require.True(t, e.Stopped())
 | 
						|
					case *connectorNode:
 | 
						|
						// connector needs to be unwrapped to access component as ExampleConnector
 | 
						|
						switch ct := c.Component.(type) {
 | 
						|
						case connector.Traces:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Stopped())
 | 
						|
						case connector.Metrics:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Stopped())
 | 
						|
						case connector.Logs:
 | 
						|
							require.True(t, ct.(*testcomponents.ExampleConnector).Stopped())
 | 
						|
						}
 | 
						|
					default:
 | 
						|
						require.Fail(t, fmt.Sprintf("unexpected type %T", c))
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			// Get the list of exporters directly from the overall component graph. Like receivers,
 | 
						|
			// exclude connectors and validate each exporter once regardless of sharing between pipelines.
 | 
						|
			allExporters := pg.GetExporters()
 | 
						|
			for _, e := range allExporters[component.DataTypeTraces] {
 | 
						|
				tracesExporter := e.(*testcomponents.ExampleExporter)
 | 
						|
				assert.Equal(t, test.expectedPerExporter, len(tracesExporter.Traces))
 | 
						|
				expected := testdata.GenerateTraces(1)
 | 
						|
				if len(allExporters[component.DataTypeTraces]) > 1 {
 | 
						|
					expected.MarkReadOnly() // multiple read-only exporters should get read-only pdata
 | 
						|
				}
 | 
						|
				for i := 0; i < test.expectedPerExporter; i++ {
 | 
						|
					assert.EqualValues(t, expected, tracesExporter.Traces[0])
 | 
						|
				}
 | 
						|
			}
 | 
						|
			for _, e := range allExporters[component.DataTypeMetrics] {
 | 
						|
				metricsExporter := e.(*testcomponents.ExampleExporter)
 | 
						|
				assert.Equal(t, test.expectedPerExporter, len(metricsExporter.Metrics))
 | 
						|
				expected := testdata.GenerateMetrics(1)
 | 
						|
				if len(allExporters[component.DataTypeMetrics]) > 1 {
 | 
						|
					expected.MarkReadOnly() // multiple read-only exporters should get read-only pdata
 | 
						|
				}
 | 
						|
				for i := 0; i < test.expectedPerExporter; i++ {
 | 
						|
					assert.EqualValues(t, expected, metricsExporter.Metrics[0])
 | 
						|
				}
 | 
						|
			}
 | 
						|
			for _, e := range allExporters[component.DataTypeLogs] {
 | 
						|
				logsExporter := e.(*testcomponents.ExampleExporter)
 | 
						|
				assert.Equal(t, test.expectedPerExporter, len(logsExporter.Logs))
 | 
						|
				expected := testdata.GenerateLogs(1)
 | 
						|
				if len(allExporters[component.DataTypeLogs]) > 1 {
 | 
						|
					expected.MarkReadOnly() // multiple read-only exporters should get read-only pdata
 | 
						|
				}
 | 
						|
				for i := 0; i < test.expectedPerExporter; i++ {
 | 
						|
					assert.EqualValues(t, expected, logsExporter.Logs[0])
 | 
						|
				}
 | 
						|
			}
 | 
						|
		})
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func TestConnectorRouter(t *testing.T) {
 | 
						|
	rcvrID := component.NewID("examplereceiver")
 | 
						|
	routeTracesID := component.NewIDWithName("examplerouter", "traces")
 | 
						|
	routeMetricsID := component.NewIDWithName("examplerouter", "metrics")
 | 
						|
	routeLogsID := component.NewIDWithName("examplerouter", "logs")
 | 
						|
	expRightID := component.NewIDWithName("exampleexporter", "right")
 | 
						|
	expLeftID := component.NewIDWithName("exampleexporter", "left")
 | 
						|
 | 
						|
	tracesInID := component.NewIDWithName("traces", "in")
 | 
						|
	tracesRightID := component.NewIDWithName("traces", "right")
 | 
						|
	tracesLeftID := component.NewIDWithName("traces", "left")
 | 
						|
 | 
						|
	metricsInID := component.NewIDWithName("metrics", "in")
 | 
						|
	metricsRightID := component.NewIDWithName("metrics", "right")
 | 
						|
	metricsLeftID := component.NewIDWithName("metrics", "left")
 | 
						|
 | 
						|
	logsInID := component.NewIDWithName("logs", "in")
 | 
						|
	logsRightID := component.NewIDWithName("logs", "right")
 | 
						|
	logsLeftID := component.NewIDWithName("logs", "left")
 | 
						|
 | 
						|
	ctx := context.Background()
 | 
						|
	set := Settings{
 | 
						|
		Telemetry: servicetelemetry.NewNopTelemetrySettings(),
 | 
						|
		BuildInfo: component.NewDefaultBuildInfo(),
 | 
						|
		ReceiverBuilder: receiver.NewBuilder(
 | 
						|
			map[component.ID]component.Config{
 | 
						|
				rcvrID: testcomponents.ExampleReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			map[component.Type]receiver.Factory{
 | 
						|
				testcomponents.ExampleReceiverFactory.Type(): testcomponents.ExampleReceiverFactory,
 | 
						|
			},
 | 
						|
		),
 | 
						|
		ExporterBuilder: exporter.NewBuilder(
 | 
						|
			map[component.ID]component.Config{
 | 
						|
				expRightID: testcomponents.ExampleExporterFactory.CreateDefaultConfig(),
 | 
						|
				expLeftID:  testcomponents.ExampleExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			map[component.Type]exporter.Factory{
 | 
						|
				testcomponents.ExampleExporterFactory.Type(): testcomponents.ExampleExporterFactory,
 | 
						|
			},
 | 
						|
		),
 | 
						|
		ConnectorBuilder: connector.NewBuilder(
 | 
						|
			map[component.ID]component.Config{
 | 
						|
				routeTracesID: testcomponents.ExampleRouterConfig{
 | 
						|
					Traces: &testcomponents.LeftRightConfig{
 | 
						|
						Right: tracesRightID,
 | 
						|
						Left:  tracesLeftID,
 | 
						|
					},
 | 
						|
				},
 | 
						|
				routeMetricsID: testcomponents.ExampleRouterConfig{
 | 
						|
					Metrics: &testcomponents.LeftRightConfig{
 | 
						|
						Right: metricsRightID,
 | 
						|
						Left:  metricsLeftID,
 | 
						|
					},
 | 
						|
				},
 | 
						|
				routeLogsID: testcomponents.ExampleRouterConfig{
 | 
						|
					Logs: &testcomponents.LeftRightConfig{
 | 
						|
						Right: logsRightID,
 | 
						|
						Left:  logsLeftID,
 | 
						|
					},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			map[component.Type]connector.Factory{
 | 
						|
				testcomponents.ExampleRouterFactory.Type(): testcomponents.ExampleRouterFactory,
 | 
						|
			},
 | 
						|
		),
 | 
						|
		PipelineConfigs: pipelines.Config{
 | 
						|
			tracesInID: {
 | 
						|
				Receivers: []component.ID{rcvrID},
 | 
						|
				Exporters: []component.ID{routeTracesID},
 | 
						|
			},
 | 
						|
			tracesRightID: {
 | 
						|
				Receivers: []component.ID{routeTracesID},
 | 
						|
				Exporters: []component.ID{expRightID},
 | 
						|
			},
 | 
						|
			tracesLeftID: {
 | 
						|
				Receivers: []component.ID{routeTracesID},
 | 
						|
				Exporters: []component.ID{expLeftID},
 | 
						|
			},
 | 
						|
			metricsInID: {
 | 
						|
				Receivers: []component.ID{rcvrID},
 | 
						|
				Exporters: []component.ID{routeMetricsID},
 | 
						|
			},
 | 
						|
			metricsRightID: {
 | 
						|
				Receivers: []component.ID{routeMetricsID},
 | 
						|
				Exporters: []component.ID{expRightID},
 | 
						|
			},
 | 
						|
			metricsLeftID: {
 | 
						|
				Receivers: []component.ID{routeMetricsID},
 | 
						|
				Exporters: []component.ID{expLeftID},
 | 
						|
			},
 | 
						|
			logsInID: {
 | 
						|
				Receivers: []component.ID{rcvrID},
 | 
						|
				Exporters: []component.ID{routeLogsID},
 | 
						|
			},
 | 
						|
			logsRightID: {
 | 
						|
				Receivers: []component.ID{routeLogsID},
 | 
						|
				Exporters: []component.ID{expRightID},
 | 
						|
			},
 | 
						|
			logsLeftID: {
 | 
						|
				Receivers: []component.ID{routeLogsID},
 | 
						|
				Exporters: []component.ID{expLeftID},
 | 
						|
			},
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	pg, err := Build(ctx, set)
 | 
						|
	require.NoError(t, err)
 | 
						|
 | 
						|
	allReceivers := pg.getReceivers()
 | 
						|
	allExporters := pg.GetExporters()
 | 
						|
 | 
						|
	assert.Equal(t, len(set.PipelineConfigs), len(pg.pipelines))
 | 
						|
 | 
						|
	// Get a handle for the traces receiver and both exporters
 | 
						|
	tracesReceiver := allReceivers[component.DataTypeTraces][rcvrID].(*testcomponents.ExampleReceiver)
 | 
						|
	tracesRight := allExporters[component.DataTypeTraces][expRightID].(*testcomponents.ExampleExporter)
 | 
						|
	tracesLeft := allExporters[component.DataTypeTraces][expLeftID].(*testcomponents.ExampleExporter)
 | 
						|
 | 
						|
	// Consume 1, validate it went right
 | 
						|
	assert.NoError(t, tracesReceiver.ConsumeTraces(ctx, testdata.GenerateTraces(1)))
 | 
						|
	assert.Equal(t, 1, len(tracesRight.Traces))
 | 
						|
	assert.Equal(t, 0, len(tracesLeft.Traces))
 | 
						|
 | 
						|
	// Consume 1, validate it went left
 | 
						|
	assert.NoError(t, tracesReceiver.ConsumeTraces(ctx, testdata.GenerateTraces(1)))
 | 
						|
	assert.Equal(t, 1, len(tracesRight.Traces))
 | 
						|
	assert.Equal(t, 1, len(tracesLeft.Traces))
 | 
						|
 | 
						|
	// Consume 3, validate 2 went right, 1 went left
 | 
						|
	assert.NoError(t, tracesReceiver.ConsumeTraces(ctx, testdata.GenerateTraces(1)))
 | 
						|
	assert.NoError(t, tracesReceiver.ConsumeTraces(ctx, testdata.GenerateTraces(1)))
 | 
						|
	assert.NoError(t, tracesReceiver.ConsumeTraces(ctx, testdata.GenerateTraces(1)))
 | 
						|
	assert.Equal(t, 3, len(tracesRight.Traces))
 | 
						|
	assert.Equal(t, 2, len(tracesLeft.Traces))
 | 
						|
 | 
						|
	// Get a handle for the metrics receiver and both exporters
 | 
						|
	metricsReceiver := allReceivers[component.DataTypeMetrics][rcvrID].(*testcomponents.ExampleReceiver)
 | 
						|
	metricsRight := allExporters[component.DataTypeMetrics][expRightID].(*testcomponents.ExampleExporter)
 | 
						|
	metricsLeft := allExporters[component.DataTypeMetrics][expLeftID].(*testcomponents.ExampleExporter)
 | 
						|
 | 
						|
	// Consume 1, validate it went right
 | 
						|
	assert.NoError(t, metricsReceiver.ConsumeMetrics(ctx, testdata.GenerateMetrics(1)))
 | 
						|
	assert.Equal(t, 1, len(metricsRight.Metrics))
 | 
						|
	assert.Equal(t, 0, len(metricsLeft.Metrics))
 | 
						|
 | 
						|
	// Consume 1, validate it went left
 | 
						|
	assert.NoError(t, metricsReceiver.ConsumeMetrics(ctx, testdata.GenerateMetrics(1)))
 | 
						|
	assert.Equal(t, 1, len(metricsRight.Metrics))
 | 
						|
	assert.Equal(t, 1, len(metricsLeft.Metrics))
 | 
						|
 | 
						|
	// Consume 3, validate 2 went right, 1 went left
 | 
						|
	assert.NoError(t, metricsReceiver.ConsumeMetrics(ctx, testdata.GenerateMetrics(1)))
 | 
						|
	assert.NoError(t, metricsReceiver.ConsumeMetrics(ctx, testdata.GenerateMetrics(1)))
 | 
						|
	assert.NoError(t, metricsReceiver.ConsumeMetrics(ctx, testdata.GenerateMetrics(1)))
 | 
						|
	assert.Equal(t, 3, len(metricsRight.Metrics))
 | 
						|
	assert.Equal(t, 2, len(metricsLeft.Metrics))
 | 
						|
 | 
						|
	// Get a handle for the logs receiver and both exporters
 | 
						|
	logsReceiver := allReceivers[component.DataTypeLogs][rcvrID].(*testcomponents.ExampleReceiver)
 | 
						|
	logsRight := allExporters[component.DataTypeLogs][expRightID].(*testcomponents.ExampleExporter)
 | 
						|
	logsLeft := allExporters[component.DataTypeLogs][expLeftID].(*testcomponents.ExampleExporter)
 | 
						|
 | 
						|
	// Consume 1, validate it went right
 | 
						|
	assert.NoError(t, logsReceiver.ConsumeLogs(ctx, testdata.GenerateLogs(1)))
 | 
						|
	assert.Equal(t, 1, len(logsRight.Logs))
 | 
						|
	assert.Equal(t, 0, len(logsLeft.Logs))
 | 
						|
 | 
						|
	// Consume 1, validate it went left
 | 
						|
	assert.NoError(t, logsReceiver.ConsumeLogs(ctx, testdata.GenerateLogs(1)))
 | 
						|
	assert.Equal(t, 1, len(logsRight.Logs))
 | 
						|
	assert.Equal(t, 1, len(logsLeft.Logs))
 | 
						|
 | 
						|
	// Consume 3, validate 2 went right, 1 went left
 | 
						|
	assert.NoError(t, logsReceiver.ConsumeLogs(ctx, testdata.GenerateLogs(1)))
 | 
						|
	assert.NoError(t, logsReceiver.ConsumeLogs(ctx, testdata.GenerateLogs(1)))
 | 
						|
	assert.NoError(t, logsReceiver.ConsumeLogs(ctx, testdata.GenerateLogs(1)))
 | 
						|
	assert.Equal(t, 3, len(logsRight.Logs))
 | 
						|
	assert.Equal(t, 2, len(logsLeft.Logs))
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
func TestGraphBuildErrors(t *testing.T) {
 | 
						|
	nopReceiverFactory := receivertest.NewNopFactory()
 | 
						|
	nopProcessorFactory := processortest.NewNopFactory()
 | 
						|
	nopExporterFactory := exportertest.NewNopFactory()
 | 
						|
	nopConnectorFactory := connectortest.NewNopFactory()
 | 
						|
	mfConnectorFactory := testcomponents.MockForwardConnectorFactory
 | 
						|
	badReceiverFactory := newBadReceiverFactory()
 | 
						|
	badProcessorFactory := newBadProcessorFactory()
 | 
						|
	badExporterFactory := newBadExporterFactory()
 | 
						|
	badConnectorFactory := newBadConnectorFactory()
 | 
						|
 | 
						|
	tests := []struct {
 | 
						|
		name          string
 | 
						|
		receiverCfgs  map[component.ID]component.Config
 | 
						|
		processorCfgs map[component.ID]component.Config
 | 
						|
		exporterCfgs  map[component.ID]component.Config
 | 
						|
		connectorCfgs map[component.ID]component.Config
 | 
						|
		pipelineCfgs  pipelines.Config
 | 
						|
		expected      string
 | 
						|
	}{
 | 
						|
		{
 | 
						|
			name: "not_supported_exporter_logs",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): badExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"bf\" exporter for data type \"logs\": telemetry type is not supported",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_exporter_metrics",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): badExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"bf\" exporter for data type \"metrics\": telemetry type is not supported",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_exporter_traces",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): badExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"bf\" exporter for data type \"traces\": telemetry type is not supported",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_processor_logs",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): badProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"bf\" processor, in pipeline \"logs\": telemetry type is not supported",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_processor_metrics",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): badProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"bf\" processor, in pipeline \"metrics\": telemetry type is not supported",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_processor_traces",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): badProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"bf\" processor, in pipeline \"traces\": telemetry type is not supported",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_receiver_logs",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): badReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"bf\" receiver for data type \"logs\": telemetry type is not supported",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_receiver_metrics",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): badReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"bf\" receiver for data type \"metrics\": telemetry type is not supported",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_receiver_traces",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): badReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"bf\" receiver for data type \"traces\": telemetry type is not supported",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_connector_traces_traces.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "connector \"bf\" used as exporter in traces pipeline but not used in any supported receiver pipeline",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_connector_traces_metrics.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "connector \"bf\" used as exporter in traces pipeline but not used in any supported receiver pipeline",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_connector_traces_logs.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "connector \"bf\" used as exporter in traces pipeline but not used in any supported receiver pipeline",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_connector_metrics_traces.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "connector \"bf\" used as exporter in metrics pipeline but not used in any supported receiver pipeline",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_connector_metrics_metrics.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "connector \"bf\" used as exporter in metrics pipeline but not used in any supported receiver pipeline",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_connector_metrics_logs.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "connector \"bf\" used as exporter in metrics pipeline but not used in any supported receiver pipeline",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_connector_logs_traces.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("logs", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "connector \"bf\" used as exporter in logs pipeline but not used in any supported receiver pipeline",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_connector_logs_metrics.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("logs", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "connector \"bf\" used as exporter in logs pipeline but not used in any supported receiver pipeline",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_supported_connector_logs_logs.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("bf"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("logs", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("bf")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("bf")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "connector \"bf\" used as exporter in logs pipeline but not used in any supported receiver pipeline",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "orphaned-connector-use-as-exporter",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewIDWithName("nop", "conn"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `connector "nop/conn" used as exporter in metrics pipeline but not used in any supported receiver pipeline`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "orphaned-connector-use-as-receiver",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewIDWithName("nop", "conn"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `connector "nop/conn" used as receiver in traces pipeline but not used in any supported exporter pipeline`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "partially-orphaned-connector-use-as-exporter",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("mockforward"): mfConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("mockforward")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("mockforward")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("mockforward")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `connector "mockforward" used as exporter in metrics pipeline but not used in any supported receiver pipeline`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "partially-orphaned-connector-use-as-receiver",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("mockforward"): mfConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("mockforward")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("mockforward")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("mockforward")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `connector "mockforward" used as receiver in traces pipeline but not used in any supported exporter pipeline`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_allowed_simple_cycle_traces.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewIDWithName("nop", "conn"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `cycle detected: ` +
 | 
						|
				`connector "nop/conn" (traces to traces) -> ` +
 | 
						|
				`processor "nop" in pipeline "traces" -> ` +
 | 
						|
				`connector "nop/conn" (traces to traces)`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_allowed_simple_cycle_metrics.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewIDWithName("nop", "conn"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `cycle detected: ` +
 | 
						|
				`connector "nop/conn" (metrics to metrics) -> ` +
 | 
						|
				`processor "nop" in pipeline "metrics" -> ` +
 | 
						|
				`connector "nop/conn" (metrics to metrics)`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_allowed_simple_cycle_logs.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewIDWithName("nop", "conn"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `cycle detected: ` +
 | 
						|
				`connector "nop/conn" (logs to logs) -> ` +
 | 
						|
				`processor "nop" in pipeline "logs" -> ` +
 | 
						|
				`connector "nop/conn" (logs to logs)`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_allowed_deep_cycle_traces.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewIDWithName("nop", "conn"):  nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewIDWithName("nop", "conn1"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewIDWithName("nop", "conn2"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "1"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn1")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "2"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn1")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn2"), component.NewIDWithName("nop", "conn")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn2")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `cycle detected: ` +
 | 
						|
				`connector "nop/conn1" (traces to traces) -> ` +
 | 
						|
				`processor "nop" in pipeline "traces/2" -> ` +
 | 
						|
				`connector "nop/conn" (traces to traces) -> ` +
 | 
						|
				`processor "nop" in pipeline "traces/1" -> ` +
 | 
						|
				`connector "nop/conn1" (traces to traces)`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_allowed_deep_cycle_metrics.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewIDWithName("nop", "conn"):  nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewIDWithName("nop", "conn1"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewIDWithName("nop", "conn2"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("metrics", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "1"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn1")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "2"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn1")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn2"), component.NewIDWithName("nop", "conn")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn2")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `cycle detected: ` +
 | 
						|
				`connector "nop/conn1" (metrics to metrics) -> ` +
 | 
						|
				`processor "nop" in pipeline "metrics/2" -> ` +
 | 
						|
				`connector "nop/conn" (metrics to metrics) -> ` +
 | 
						|
				`processor "nop" in pipeline "metrics/1" -> ` +
 | 
						|
				`connector "nop/conn1" (metrics to metrics)`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_allowed_deep_cycle_logs.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewIDWithName("nop", "conn"):  nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewIDWithName("nop", "conn1"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewIDWithName("nop", "conn2"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("logs", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "1"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn1")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "2"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn1")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "conn2"), component.NewIDWithName("nop", "conn")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "out"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "conn2")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `cycle detected: ` +
 | 
						|
				`connector "nop/conn1" (logs to logs) -> ` +
 | 
						|
				`processor "nop" in pipeline "logs/2" -> ` +
 | 
						|
				`connector "nop/conn" (logs to logs) -> ` +
 | 
						|
				`processor "nop" in pipeline "logs/1" -> ` +
 | 
						|
				`connector "nop/conn1" (logs to logs)`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "not_allowed_deep_cycle_multi_signal.yaml",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewIDWithName("nop", "fork"):      nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewIDWithName("nop", "count"):     nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewIDWithName("nop", "forkagain"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewIDWithName("nop", "rawlog"):    nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "fork")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "copy1"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "fork")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "count")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "copy2"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "fork")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "forkagain")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "copy2a"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "forkagain")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "count")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "copy2b"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "forkagain")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "rawlog")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("metrics", "count"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "count")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("logs", "raw"): {
 | 
						|
					Receivers:  []component.ID{component.NewIDWithName("nop", "rawlog")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewIDWithName("nop", "fork")}, // cannot loop back to "nop/fork"
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: `cycle detected: ` +
 | 
						|
				`connector "nop/rawlog" (traces to logs) -> ` +
 | 
						|
				`processor "nop" in pipeline "logs/raw" -> ` +
 | 
						|
				`connector "nop/fork" (logs to traces) -> ` +
 | 
						|
				`processor "nop" in pipeline "traces/copy2" -> ` +
 | 
						|
				`connector "nop/forkagain" (traces to traces) -> ` +
 | 
						|
				`processor "nop" in pipeline "traces/copy2b" -> ` +
 | 
						|
				`connector "nop/rawlog" (traces to logs)`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "unknown_exporter_config",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop"), component.NewIDWithName("nop", "1")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"nop/1\" exporter for data type \"traces\": exporter \"nop/1\" is not configured",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "unknown_exporter_factory",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("unknown"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("traces"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("unknown")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"unknown\" exporter for data type \"traces\": exporter factory not available for: \"unknown\"",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "unknown_processor_config",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("nop"), component.NewIDWithName("nop", "1")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"nop/1\" processor, in pipeline \"metrics\": processor \"nop/1\" is not configured",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "unknown_processor_factory",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			processorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("unknown"): nopProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("metrics"): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("unknown")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"unknown\" processor, in pipeline \"metrics\": processor factory not available for: \"unknown\"",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "unknown_receiver_config",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop"), component.NewIDWithName("nop", "1")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"nop/1\" receiver for data type \"logs\": receiver \"nop/1\" is not configured",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "unknown_receiver_factory",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("unknown"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewID("logs"): {
 | 
						|
					Receivers: []component.ID{component.NewID("unknown")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "failed to create \"unknown\" receiver for data type \"logs\": receiver factory not available for: \"unknown\"",
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "unknown_connector_factory",
 | 
						|
			receiverCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			exporterCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("nop"): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			connectorCfgs: map[component.ID]component.Config{
 | 
						|
				component.NewID("unknown"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			pipelineCfgs: pipelines.Config{
 | 
						|
				component.NewIDWithName("traces", "in"): {
 | 
						|
					Receivers: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters: []component.ID{component.NewID("unknown")},
 | 
						|
				},
 | 
						|
				component.NewIDWithName("traces", "out"): {
 | 
						|
					Receivers: []component.ID{component.NewID("unknown")},
 | 
						|
					Exporters: []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			},
 | 
						|
			expected: "connector factory not available for: \"unknown\"",
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	for _, test := range tests {
 | 
						|
		t.Run(test.name, func(t *testing.T) {
 | 
						|
			set := Settings{
 | 
						|
				BuildInfo: component.NewDefaultBuildInfo(),
 | 
						|
				Telemetry: servicetelemetry.NewNopTelemetrySettings(),
 | 
						|
				ReceiverBuilder: receiver.NewBuilder(
 | 
						|
					test.receiverCfgs,
 | 
						|
					map[component.Type]receiver.Factory{
 | 
						|
						nopReceiverFactory.Type(): nopReceiverFactory,
 | 
						|
						badReceiverFactory.Type(): badReceiverFactory,
 | 
						|
					}),
 | 
						|
				ProcessorBuilder: processor.NewBuilder(
 | 
						|
					test.processorCfgs,
 | 
						|
					map[component.Type]processor.Factory{
 | 
						|
						nopProcessorFactory.Type(): nopProcessorFactory,
 | 
						|
						badProcessorFactory.Type(): badProcessorFactory,
 | 
						|
					}),
 | 
						|
				ExporterBuilder: exporter.NewBuilder(
 | 
						|
					test.exporterCfgs,
 | 
						|
					map[component.Type]exporter.Factory{
 | 
						|
						nopExporterFactory.Type(): nopExporterFactory,
 | 
						|
						badExporterFactory.Type(): badExporterFactory,
 | 
						|
					}),
 | 
						|
				ConnectorBuilder: connector.NewBuilder(
 | 
						|
					test.connectorCfgs,
 | 
						|
					map[component.Type]connector.Factory{
 | 
						|
						nopConnectorFactory.Type(): nopConnectorFactory,
 | 
						|
						badConnectorFactory.Type(): badConnectorFactory,
 | 
						|
						mfConnectorFactory.Type():  mfConnectorFactory,
 | 
						|
					}),
 | 
						|
				PipelineConfigs: test.pipelineCfgs,
 | 
						|
			}
 | 
						|
			_, err := Build(context.Background(), set)
 | 
						|
			assert.EqualError(t, err, test.expected)
 | 
						|
		})
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// This includes all tests from the previous implmentation, plus a new one
 | 
						|
// relevant only to the new graph-based implementation.
 | 
						|
func TestGraphFailToStartAndShutdown(t *testing.T) {
 | 
						|
	errReceiverFactory := newErrReceiverFactory()
 | 
						|
	errProcessorFactory := newErrProcessorFactory()
 | 
						|
	errExporterFactory := newErrExporterFactory()
 | 
						|
	errConnectorFactory := newErrConnectorFactory()
 | 
						|
	nopReceiverFactory := receivertest.NewNopFactory()
 | 
						|
	nopProcessorFactory := processortest.NewNopFactory()
 | 
						|
	nopExporterFactory := exportertest.NewNopFactory()
 | 
						|
	nopConnectorFactory := connectortest.NewNopFactory()
 | 
						|
 | 
						|
	set := Settings{
 | 
						|
		Telemetry: servicetelemetry.NewNopTelemetrySettings(),
 | 
						|
		BuildInfo: component.NewDefaultBuildInfo(),
 | 
						|
		ReceiverBuilder: receiver.NewBuilder(
 | 
						|
			map[component.ID]component.Config{
 | 
						|
				component.NewID(nopReceiverFactory.Type()): nopReceiverFactory.CreateDefaultConfig(),
 | 
						|
				component.NewID(errReceiverFactory.Type()): errReceiverFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			map[component.Type]receiver.Factory{
 | 
						|
				nopReceiverFactory.Type(): nopReceiverFactory,
 | 
						|
				errReceiverFactory.Type(): errReceiverFactory,
 | 
						|
			}),
 | 
						|
		ProcessorBuilder: processor.NewBuilder(
 | 
						|
			map[component.ID]component.Config{
 | 
						|
				component.NewID(nopProcessorFactory.Type()): nopProcessorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewID(errProcessorFactory.Type()): errProcessorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			map[component.Type]processor.Factory{
 | 
						|
				nopProcessorFactory.Type(): nopProcessorFactory,
 | 
						|
				errProcessorFactory.Type(): errProcessorFactory,
 | 
						|
			}),
 | 
						|
		ExporterBuilder: exporter.NewBuilder(
 | 
						|
			map[component.ID]component.Config{
 | 
						|
				component.NewID(nopExporterFactory.Type()): nopExporterFactory.CreateDefaultConfig(),
 | 
						|
				component.NewID(errExporterFactory.Type()): errExporterFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			map[component.Type]exporter.Factory{
 | 
						|
				nopExporterFactory.Type(): nopExporterFactory,
 | 
						|
				errExporterFactory.Type(): errExporterFactory,
 | 
						|
			}),
 | 
						|
		ConnectorBuilder: connector.NewBuilder(
 | 
						|
			map[component.ID]component.Config{
 | 
						|
				component.NewIDWithName(nopConnectorFactory.Type(), "conn"): nopConnectorFactory.CreateDefaultConfig(),
 | 
						|
				component.NewIDWithName(errConnectorFactory.Type(), "conn"): errConnectorFactory.CreateDefaultConfig(),
 | 
						|
			},
 | 
						|
			map[component.Type]connector.Factory{
 | 
						|
				nopConnectorFactory.Type(): nopConnectorFactory,
 | 
						|
				errConnectorFactory.Type(): errConnectorFactory,
 | 
						|
			}),
 | 
						|
	}
 | 
						|
 | 
						|
	dataTypes := []component.DataType{component.DataTypeTraces, component.DataTypeMetrics, component.DataTypeLogs}
 | 
						|
	for _, dt := range dataTypes {
 | 
						|
		t.Run(string(dt)+"/receiver", func(t *testing.T) {
 | 
						|
			set.PipelineConfigs = pipelines.Config{
 | 
						|
				component.NewID(dt): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop"), component.NewID("err")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			}
 | 
						|
			pipelines, err := Build(context.Background(), set)
 | 
						|
			assert.NoError(t, err)
 | 
						|
			assert.Error(t, pipelines.StartAll(context.Background(), componenttest.NewNopHost()))
 | 
						|
			assert.Error(t, pipelines.ShutdownAll(context.Background()))
 | 
						|
		})
 | 
						|
 | 
						|
		t.Run(string(dt)+"/processor", func(t *testing.T) {
 | 
						|
			set.PipelineConfigs = pipelines.Config{
 | 
						|
				component.NewID(dt): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("nop"), component.NewID("err")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
				},
 | 
						|
			}
 | 
						|
			pipelines, err := Build(context.Background(), set)
 | 
						|
			assert.NoError(t, err)
 | 
						|
			assert.Error(t, pipelines.StartAll(context.Background(), componenttest.NewNopHost()))
 | 
						|
			assert.Error(t, pipelines.ShutdownAll(context.Background()))
 | 
						|
		})
 | 
						|
 | 
						|
		t.Run(string(dt)+"/exporter", func(t *testing.T) {
 | 
						|
			set.PipelineConfigs = pipelines.Config{
 | 
						|
				component.NewID(dt): {
 | 
						|
					Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
					Processors: []component.ID{component.NewID("nop")},
 | 
						|
					Exporters:  []component.ID{component.NewID("nop"), component.NewID("err")},
 | 
						|
				},
 | 
						|
			}
 | 
						|
			pipelines, err := Build(context.Background(), set)
 | 
						|
			assert.NoError(t, err)
 | 
						|
			assert.Error(t, pipelines.StartAll(context.Background(), componenttest.NewNopHost()))
 | 
						|
			assert.Error(t, pipelines.ShutdownAll(context.Background()))
 | 
						|
		})
 | 
						|
 | 
						|
		for _, dt2 := range dataTypes {
 | 
						|
			t.Run(string(dt)+"/"+string(dt2)+"/connector", func(t *testing.T) {
 | 
						|
				set.PipelineConfigs = pipelines.Config{
 | 
						|
					component.NewIDWithName(dt, "in"): {
 | 
						|
						Receivers:  []component.ID{component.NewID("nop")},
 | 
						|
						Processors: []component.ID{component.NewID("nop")},
 | 
						|
						Exporters:  []component.ID{component.NewID("nop"), component.NewIDWithName("err", "conn")},
 | 
						|
					},
 | 
						|
					component.NewIDWithName(dt2, "out"): {
 | 
						|
						Receivers:  []component.ID{component.NewID("nop"), component.NewIDWithName("err", "conn")},
 | 
						|
						Processors: []component.ID{component.NewID("nop")},
 | 
						|
						Exporters:  []component.ID{component.NewID("nop")},
 | 
						|
					},
 | 
						|
				}
 | 
						|
				pipelines, err := Build(context.Background(), set)
 | 
						|
				assert.NoError(t, err)
 | 
						|
				assert.Error(t, pipelines.StartAll(context.Background(), componenttest.NewNopHost()))
 | 
						|
				assert.Error(t, pipelines.ShutdownAll(context.Background()))
 | 
						|
			})
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func TestStatusReportedOnStartupShutdown(t *testing.T) {
 | 
						|
 | 
						|
	rNoErr := &testNode{id: component.NewIDWithName("r-no-err", "1")}
 | 
						|
	rStErr := &testNode{id: component.NewIDWithName("r-st-err", "1"), startErr: assert.AnError}
 | 
						|
	rSdErr := &testNode{id: component.NewIDWithName("r-sd-err", "1"), shutdownErr: assert.AnError}
 | 
						|
 | 
						|
	eNoErr := &testNode{id: component.NewIDWithName("e-no-err", "1")}
 | 
						|
	eStErr := &testNode{id: component.NewIDWithName("e-st-err", "1"), startErr: assert.AnError}
 | 
						|
	eSdErr := &testNode{id: component.NewIDWithName("e-sd-err", "1"), shutdownErr: assert.AnError}
 | 
						|
 | 
						|
	instanceIDs := map[*testNode]*component.InstanceID{
 | 
						|
		rNoErr: {ID: rNoErr.id},
 | 
						|
		rStErr: {ID: rStErr.id},
 | 
						|
		rSdErr: {ID: rSdErr.id},
 | 
						|
		eNoErr: {ID: eNoErr.id},
 | 
						|
		eStErr: {ID: eStErr.id},
 | 
						|
		eSdErr: {ID: eSdErr.id},
 | 
						|
	}
 | 
						|
 | 
						|
	// compare two maps of status events ignoring timestamp
 | 
						|
	assertEqualStatuses := func(t *testing.T, evMap1, evMap2 map[*component.InstanceID][]*component.StatusEvent) {
 | 
						|
		assert.Equal(t, len(evMap1), len(evMap2))
 | 
						|
		for id, evts1 := range evMap1 {
 | 
						|
			evts2 := evMap2[id]
 | 
						|
			assert.Equal(t, len(evts1), len(evts2))
 | 
						|
			for i := 0; i < len(evts1); i++ {
 | 
						|
				ev1 := evts1[i]
 | 
						|
				ev2 := evts2[i]
 | 
						|
				assert.Equal(t, ev1.Status(), ev2.Status())
 | 
						|
				assert.Equal(t, ev1.Err(), ev2.Err())
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	for _, tc := range []struct {
 | 
						|
		name             string
 | 
						|
		edge             [2]*testNode
 | 
						|
		expectedStatuses map[*component.InstanceID][]*component.StatusEvent
 | 
						|
		startupErr       error
 | 
						|
		shutdownErr      error
 | 
						|
	}{
 | 
						|
		{
 | 
						|
			name: "successful startup/shutdown",
 | 
						|
			edge: [2]*testNode{rNoErr, eNoErr},
 | 
						|
			expectedStatuses: map[*component.InstanceID][]*component.StatusEvent{
 | 
						|
				instanceIDs[rNoErr]: {
 | 
						|
					component.NewStatusEvent(component.StatusStarting),
 | 
						|
					component.NewStatusEvent(component.StatusOK),
 | 
						|
					component.NewStatusEvent(component.StatusStopping),
 | 
						|
					component.NewStatusEvent(component.StatusStopped),
 | 
						|
				},
 | 
						|
				instanceIDs[eNoErr]: {
 | 
						|
					component.NewStatusEvent(component.StatusStarting),
 | 
						|
					component.NewStatusEvent(component.StatusOK),
 | 
						|
					component.NewStatusEvent(component.StatusStopping),
 | 
						|
					component.NewStatusEvent(component.StatusStopped),
 | 
						|
				},
 | 
						|
			},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "early startup error",
 | 
						|
			edge: [2]*testNode{rNoErr, eStErr},
 | 
						|
			expectedStatuses: map[*component.InstanceID][]*component.StatusEvent{
 | 
						|
				instanceIDs[eStErr]: {
 | 
						|
					component.NewStatusEvent(component.StatusStarting),
 | 
						|
					component.NewPermanentErrorEvent(assert.AnError),
 | 
						|
				},
 | 
						|
			},
 | 
						|
			startupErr: assert.AnError,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "late startup error",
 | 
						|
			edge: [2]*testNode{rStErr, eNoErr},
 | 
						|
			expectedStatuses: map[*component.InstanceID][]*component.StatusEvent{
 | 
						|
				instanceIDs[rStErr]: {
 | 
						|
					component.NewStatusEvent(component.StatusStarting),
 | 
						|
					component.NewPermanentErrorEvent(assert.AnError),
 | 
						|
				},
 | 
						|
				instanceIDs[eNoErr]: {
 | 
						|
					component.NewStatusEvent(component.StatusStarting),
 | 
						|
					component.NewStatusEvent(component.StatusOK),
 | 
						|
					component.NewStatusEvent(component.StatusStopping),
 | 
						|
					component.NewStatusEvent(component.StatusStopped),
 | 
						|
				},
 | 
						|
			},
 | 
						|
			startupErr: assert.AnError,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "early shutdown error",
 | 
						|
			edge: [2]*testNode{rSdErr, eNoErr},
 | 
						|
			expectedStatuses: map[*component.InstanceID][]*component.StatusEvent{
 | 
						|
				instanceIDs[rSdErr]: {
 | 
						|
					component.NewStatusEvent(component.StatusStarting),
 | 
						|
					component.NewStatusEvent(component.StatusOK),
 | 
						|
					component.NewStatusEvent(component.StatusStopping),
 | 
						|
					component.NewPermanentErrorEvent(assert.AnError),
 | 
						|
				},
 | 
						|
				instanceIDs[eNoErr]: {
 | 
						|
					component.NewStatusEvent(component.StatusStarting),
 | 
						|
					component.NewStatusEvent(component.StatusOK),
 | 
						|
					component.NewStatusEvent(component.StatusStopping),
 | 
						|
					component.NewStatusEvent(component.StatusStopped),
 | 
						|
				},
 | 
						|
			},
 | 
						|
			shutdownErr: assert.AnError,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name: "late shutdown error",
 | 
						|
			edge: [2]*testNode{rNoErr, eSdErr},
 | 
						|
			expectedStatuses: map[*component.InstanceID][]*component.StatusEvent{
 | 
						|
				instanceIDs[rNoErr]: {
 | 
						|
					component.NewStatusEvent(component.StatusStarting),
 | 
						|
					component.NewStatusEvent(component.StatusOK),
 | 
						|
					component.NewStatusEvent(component.StatusStopping),
 | 
						|
					component.NewStatusEvent(component.StatusStopped),
 | 
						|
				},
 | 
						|
				instanceIDs[eSdErr]: {
 | 
						|
					component.NewStatusEvent(component.StatusStarting),
 | 
						|
					component.NewStatusEvent(component.StatusOK),
 | 
						|
					component.NewStatusEvent(component.StatusStopping),
 | 
						|
					component.NewPermanentErrorEvent(assert.AnError),
 | 
						|
				},
 | 
						|
			},
 | 
						|
			shutdownErr: assert.AnError,
 | 
						|
		},
 | 
						|
	} {
 | 
						|
		t.Run(tc.name, func(t *testing.T) {
 | 
						|
			pg := &Graph{componentGraph: simple.NewDirectedGraph()}
 | 
						|
			pg.telemetry = servicetelemetry.NewNopTelemetrySettings()
 | 
						|
 | 
						|
			actualStatuses := make(map[*component.InstanceID][]*component.StatusEvent)
 | 
						|
			rep := status.NewReporter(func(id *component.InstanceID, ev *component.StatusEvent) {
 | 
						|
				actualStatuses[id] = append(actualStatuses[id], ev)
 | 
						|
			}, func(err error) {
 | 
						|
			})
 | 
						|
 | 
						|
			pg.telemetry.Status = rep
 | 
						|
			rep.Ready()
 | 
						|
 | 
						|
			e0, e1 := tc.edge[0], tc.edge[1]
 | 
						|
			pg.instanceIDs = map[int64]*component.InstanceID{
 | 
						|
				e0.ID(): instanceIDs[e0],
 | 
						|
				e1.ID(): instanceIDs[e1],
 | 
						|
			}
 | 
						|
			pg.componentGraph.SetEdge(simple.Edge{F: e0, T: e1})
 | 
						|
 | 
						|
			assert.Equal(t, tc.startupErr, pg.StartAll(context.Background(), componenttest.NewNopHost()))
 | 
						|
			assert.Equal(t, tc.shutdownErr, pg.ShutdownAll(context.Background()))
 | 
						|
			assertEqualStatuses(t, tc.expectedStatuses, actualStatuses)
 | 
						|
		})
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (g *Graph) getReceivers() map[component.DataType]map[component.ID]component.Component {
 | 
						|
	receiversMap := make(map[component.DataType]map[component.ID]component.Component)
 | 
						|
	receiversMap[component.DataTypeTraces] = make(map[component.ID]component.Component)
 | 
						|
	receiversMap[component.DataTypeMetrics] = make(map[component.ID]component.Component)
 | 
						|
	receiversMap[component.DataTypeLogs] = make(map[component.ID]component.Component)
 | 
						|
 | 
						|
	for _, pg := range g.pipelines {
 | 
						|
		for _, rcvrNode := range pg.receivers {
 | 
						|
			rcvrOrConnNode := g.componentGraph.Node(rcvrNode.ID())
 | 
						|
			rcvrNode, ok := rcvrOrConnNode.(*receiverNode)
 | 
						|
			if !ok {
 | 
						|
				continue
 | 
						|
			}
 | 
						|
			receiversMap[rcvrNode.pipelineType][rcvrNode.componentID] = rcvrNode.Component
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return receiversMap
 | 
						|
}
 | 
						|
 | 
						|
// Calculates the expected number of receiver and exporter instances in the specified pipeline.
 | 
						|
//
 | 
						|
// Expect one instance of each receiver and exporter, unless it is a connector.
 | 
						|
//
 | 
						|
// For connectors:
 | 
						|
// - Let E equal the number of pipeline types in which the connector is used as an exporter.
 | 
						|
// - Let R equal the number of pipeline types in which the connector is used as a receiver.
 | 
						|
//
 | 
						|
// Within the graph as a whole, we expect E*R instances, i.e. one per combination of data types.
 | 
						|
//
 | 
						|
// However, within an individual pipeline, we expect:
 | 
						|
// - E instances of the connector as a receiver.
 | 
						|
// - R instances of the connector as an exporter.
 | 
						|
func expectedInstances(m pipelines.Config, pID component.ID) (int, int) {
 | 
						|
	var r, e int
 | 
						|
	for _, rID := range m[pID].Receivers {
 | 
						|
		if rID.Type() != "exampleconnector" {
 | 
						|
			r++
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		// This is a connector. Count the pipeline types where it is an exporter.
 | 
						|
		typeMap := map[component.DataType]bool{}
 | 
						|
		for pID, pCfg := range m {
 | 
						|
			for _, eID := range pCfg.Exporters {
 | 
						|
				if eID == rID {
 | 
						|
					typeMap[pID.Type()] = true
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		r += len(typeMap)
 | 
						|
	}
 | 
						|
	for _, eID := range m[pID].Exporters {
 | 
						|
		if eID.Type() != "exampleconnector" {
 | 
						|
			e++
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		// This is a connector. Count the pipeline types where it is a receiver.
 | 
						|
		typeMap := map[component.DataType]bool{}
 | 
						|
		for pID, pCfg := range m {
 | 
						|
			for _, rID := range pCfg.Receivers {
 | 
						|
				if rID == eID {
 | 
						|
					typeMap[pID.Type()] = true
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		e += len(typeMap)
 | 
						|
	}
 | 
						|
	return r, e
 | 
						|
}
 | 
						|
 | 
						|
func newBadReceiverFactory() receiver.Factory {
 | 
						|
	return receiver.NewFactory("bf", func() component.Config {
 | 
						|
		return &struct{}{}
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
func newBadProcessorFactory() processor.Factory {
 | 
						|
	return processor.NewFactory("bf", func() component.Config {
 | 
						|
		return &struct{}{}
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
func newBadExporterFactory() exporter.Factory {
 | 
						|
	return exporter.NewFactory("bf", func() component.Config {
 | 
						|
		return &struct{}{}
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
func newBadConnectorFactory() connector.Factory {
 | 
						|
	return connector.NewFactory("bf", func() component.Config {
 | 
						|
		return &struct{}{}
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
func newErrReceiverFactory() receiver.Factory {
 | 
						|
	return receiver.NewFactory("err",
 | 
						|
		func() component.Config { return &struct{}{} },
 | 
						|
		receiver.WithTraces(func(context.Context, receiver.CreateSettings, component.Config, consumer.Traces) (receiver.Traces, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUndefined),
 | 
						|
		receiver.WithLogs(func(context.Context, receiver.CreateSettings, component.Config, consumer.Logs) (receiver.Logs, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUndefined),
 | 
						|
		receiver.WithMetrics(func(context.Context, receiver.CreateSettings, component.Config, consumer.Metrics) (receiver.Metrics, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUndefined),
 | 
						|
	)
 | 
						|
}
 | 
						|
 | 
						|
func newErrProcessorFactory() processor.Factory {
 | 
						|
	return processor.NewFactory("err",
 | 
						|
		func() component.Config { return &struct{}{} },
 | 
						|
		processor.WithTraces(func(context.Context, processor.CreateSettings, component.Config, consumer.Traces) (processor.Traces, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUndefined),
 | 
						|
		processor.WithLogs(func(context.Context, processor.CreateSettings, component.Config, consumer.Logs) (processor.Logs, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUndefined),
 | 
						|
		processor.WithMetrics(func(context.Context, processor.CreateSettings, component.Config, consumer.Metrics) (processor.Metrics, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUndefined),
 | 
						|
	)
 | 
						|
}
 | 
						|
 | 
						|
func newErrExporterFactory() exporter.Factory {
 | 
						|
	return exporter.NewFactory("err",
 | 
						|
		func() component.Config { return &struct{}{} },
 | 
						|
		exporter.WithTraces(func(context.Context, exporter.CreateSettings, component.Config) (exporter.Traces, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUndefined),
 | 
						|
		exporter.WithLogs(func(context.Context, exporter.CreateSettings, component.Config) (exporter.Logs, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUndefined),
 | 
						|
		exporter.WithMetrics(func(context.Context, exporter.CreateSettings, component.Config) (exporter.Metrics, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUndefined),
 | 
						|
	)
 | 
						|
}
 | 
						|
 | 
						|
func newErrConnectorFactory() connector.Factory {
 | 
						|
	return connector.NewFactory("err", func() component.Config {
 | 
						|
		return &struct{}{}
 | 
						|
	},
 | 
						|
		connector.WithTracesToTraces(func(context.Context, connector.CreateSettings, component.Config, consumer.Traces) (connector.Traces, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUnmaintained),
 | 
						|
		connector.WithTracesToMetrics(func(context.Context, connector.CreateSettings, component.Config, consumer.Metrics) (connector.Traces, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUnmaintained),
 | 
						|
		connector.WithTracesToLogs(func(context.Context, connector.CreateSettings, component.Config, consumer.Logs) (connector.Traces, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUnmaintained),
 | 
						|
 | 
						|
		connector.WithMetricsToTraces(func(context.Context, connector.CreateSettings, component.Config, consumer.Traces) (connector.Metrics, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUnmaintained),
 | 
						|
		connector.WithMetricsToMetrics(func(context.Context, connector.CreateSettings, component.Config, consumer.Metrics) (connector.Metrics, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUnmaintained),
 | 
						|
		connector.WithMetricsToLogs(func(context.Context, connector.CreateSettings, component.Config, consumer.Logs) (connector.Metrics, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUnmaintained),
 | 
						|
 | 
						|
		connector.WithLogsToTraces(func(context.Context, connector.CreateSettings, component.Config, consumer.Traces) (connector.Logs, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUnmaintained),
 | 
						|
		connector.WithLogsToMetrics(func(context.Context, connector.CreateSettings, component.Config, consumer.Metrics) (connector.Logs, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUnmaintained),
 | 
						|
		connector.WithLogsToLogs(func(context.Context, connector.CreateSettings, component.Config, consumer.Logs) (connector.Logs, error) {
 | 
						|
			return &errComponent{}, nil
 | 
						|
		}, component.StabilityLevelUnmaintained),
 | 
						|
	)
 | 
						|
}
 | 
						|
 | 
						|
type errComponent struct {
 | 
						|
	consumertest.Consumer
 | 
						|
}
 | 
						|
 | 
						|
func (e errComponent) Capabilities() consumer.Capabilities {
 | 
						|
	return consumer.Capabilities{MutatesData: false}
 | 
						|
}
 | 
						|
 | 
						|
func (e errComponent) Start(context.Context, component.Host) error {
 | 
						|
	return errors.New("my error")
 | 
						|
}
 | 
						|
 | 
						|
func (e errComponent) Shutdown(context.Context) error {
 | 
						|
	return errors.New("my error")
 | 
						|
}
 |