diff --git a/.drone/drone.yml b/.drone/drone.yml index 8f43369108..fae58d3202 100644 --- a/.drone/drone.yml +++ b/.drone/drone.yml @@ -8,7 +8,7 @@ steps: - commands: - apt-get update -y && apt-get install -y libsystemd-dev - make lint - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Lint trigger: event: @@ -23,7 +23,7 @@ platform: steps: - commands: - make GO_TAGS="nodocker" test - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Run Go tests trigger: event: @@ -38,7 +38,7 @@ platform: steps: - commands: - K8S_USE_DOCKER_NETWORK=1 make test - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Run Go tests volumes: - name: docker @@ -61,7 +61,7 @@ platform: steps: - commands: - go test -tags="nodocker,nonetwork" ./... - image: grafana/agent-build-image:0.33.0-windows + image: grafana/agent-build-image:0.40.2-windows name: Run Go tests trigger: ref: @@ -76,7 +76,7 @@ platform: steps: - commands: - make agent-image - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build container volumes: - name: docker @@ -102,7 +102,7 @@ platform: steps: - commands: - '& "C:/Program Files/git/bin/bash.exe" ./tools/ci/docker-containers-windows agent' - image: grafana/agent-build-image:0.33.0-windows + image: grafana/agent-build-image:0.40.2-windows name: Build container volumes: - name: docker @@ -129,7 +129,7 @@ steps: - make generate-ui - GO_TAGS="builtinassets promtail_journal_enabled" GOOS=linux GOARCH=amd64 GOARM= make agent - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build trigger: event: @@ -146,7 +146,7 @@ steps: - make generate-ui - GO_TAGS="builtinassets promtail_journal_enabled" GOOS=linux GOARCH=arm64 GOARM= make agent - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build trigger: event: @@ -163,7 +163,7 @@ steps: - make generate-ui - GO_TAGS="builtinassets promtail_journal_enabled" GOOS=linux GOARCH=ppc64le GOARM= make agent - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build trigger: event: @@ -180,7 +180,7 @@ steps: - make generate-ui - GO_TAGS="builtinassets promtail_journal_enabled" GOOS=linux GOARCH=s390x GOARM= make agent - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build trigger: event: @@ -196,7 +196,7 @@ steps: - commands: - make generate-ui - GO_TAGS="builtinassets" GOOS=darwin GOARCH=amd64 GOARM= make agent - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build trigger: event: @@ -212,7 +212,7 @@ steps: - commands: - make generate-ui - GO_TAGS="builtinassets" GOOS=darwin GOARCH=arm64 GOARM= make agent - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build trigger: event: @@ -228,7 +228,7 @@ steps: - commands: - make generate-ui - GO_TAGS="builtinassets" GOOS=windows GOARCH=amd64 GOARM= make agent - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build trigger: event: @@ -244,7 +244,7 @@ steps: - commands: - make generate-ui - GO_TAGS="builtinassets" GOOS=freebsd GOARCH=amd64 GOARM= make agent - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build trigger: event: @@ -261,7 +261,7 @@ steps: - make generate-ui - GO_TAGS="builtinassets promtail_journal_enabled" GOOS=linux GOARCH=amd64 GOARM= GOEXPERIMENT=boringcrypto make agent-boringcrypto - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build trigger: event: @@ -278,7 +278,7 @@ steps: - make generate-ui - GO_TAGS="builtinassets promtail_journal_enabled" GOOS=linux GOARCH=arm64 GOARM= GOEXPERIMENT=boringcrypto make agent-boringcrypto - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Build trigger: event: @@ -295,7 +295,7 @@ steps: - make generate-ui - GO_TAGS="builtinassets" GOOS=windows GOARCH=amd64 GOARM= GOEXPERIMENT=cngcrypto make agent-windows-boringcrypto - image: grafana/agent-build-image:0.33.0-boringcrypto + image: grafana/agent-build-image:0.40.2-boringcrypto name: Build trigger: event: @@ -311,7 +311,7 @@ steps: - commands: - DOCKER_OPTS="" make dist/grafana-agent-linux-amd64 - DOCKER_OPTS="" make test-packages - image: grafana/agent-build-image:0.33.0 + image: grafana/agent-build-image:0.40.2 name: Test Linux system packages volumes: - name: docker @@ -407,6 +407,6 @@ kind: secret name: updater_private_key --- kind: signature -hmac: 509aa746729e5eaf86e4cbb02b07f125399aabefcc3bf5b1693ea2cca2eaa4e1 +hmac: 59c741cd4e3cd3f555cbf0165da386b269a7f54987fe5a2aba621edc6ebb09a5 ... diff --git a/CHANGELOG.md b/CHANGELOG.md index 223d3eae7b..3bde8a8576 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -16,6 +16,9 @@ Main (unreleased) - Add support for importing directories as single module to `import.git`. (@wildum) +- Improve converter diagnostic output by including a Footer and removing lower + level diagnostics when a configuration fails to generate. (@erikbaranowski) + ### Features - Added a new CLI flag `--stability.level` which defines the minimum stability @@ -29,10 +32,14 @@ Main (unreleased) - Fix a bug where structured metadata and parsed field are not passed further in `loki.source.api` (@marchellodev) +- Change `import.git` to use Git pulls rather than fetches to fix scenarios where the local code did not get updated. (@mattdurham) + ### Other changes - Clustering for Grafana Agent in Flow mode has graduated from beta to stable. +- Upgrade to Go 1.22.1 (@thampiotr) + v0.40.2 (2024-03-05) -------------------- @@ -49,7 +56,6 @@ v0.40.2 (2024-03-05) - Fix an issue where Loki could reject a batch of logs when structured metadata feature is used. (@thampiotr) -======= - Fix a duplicate metrics registration panic when recreating static mode metric instance's write handler. (@rfratto, @hainenber) diff --git a/README.md b/README.md index 5993571eef..87ba44ba8d 100644 --- a/README.md +++ b/README.md @@ -1,9 +1,8 @@
-[Grafana Alloy][] is a vendor-neutral, batteries-included telemetry collector with -configuration inspired by [Terraform][]. It is designed to be flexible, -performant, and compatible with multiple ecosystems such as Prometheus and -OpenTelemetry. +[Grafana Alloy][] is an OpenTelemetry Collector distribution with configuration +inspired by [Terraform][]. It is designed to be flexible, performant, and +compatible with multiple ecosystems such as Prometheus and OpenTelemetry. Grafana Alloy is based around **components**. Components are wired together to form programmable observability **pipelines** for telemetry collection, diff --git a/build-image/Dockerfile b/build-image/Dockerfile index 05125a2632..4e8ae49b39 100644 --- a/build-image/Dockerfile +++ b/build-image/Dockerfile @@ -4,7 +4,7 @@ # default when running `docker buildx build` or when DOCKER_BUILDKIT=1 is set # in environment variables. -# NOTE: The GO_RUNTIME is used to switch between the default google go runtime and mcr.microsoft.com/oss/go/microsoft/golang:1.22.1-bullseye which is a microsoft +# NOTE: The GO_RUNTIME is used to switch between the default Google go runtime and mcr.microsoft.com/oss/go/microsoft/golang:1.22.1-bullseye which is a Microsoft # fork of go that allows using windows crypto instead of boring crypto. Details at https://github.com/microsoft/go/tree/microsoft/main/eng/doc/fips ARG GO_RUNTIME=mustoverride diff --git a/build-image/windows/Dockerfile b/build-image/windows/Dockerfile index ddd3448e2c..3827b073f9 100644 --- a/build-image/windows/Dockerfile +++ b/build-image/windows/Dockerfile @@ -1,4 +1,4 @@ -FROM library/golang:1.22.0-windowsservercore-1809 +FROM library/golang:1.22.1-windowsservercore-1809 SHELL ["powershell", "-command"] diff --git a/cmd/grafana-agent/Dockerfile b/cmd/grafana-agent/Dockerfile index 633c3ebed0..e7cf42a67f 100644 --- a/cmd/grafana-agent/Dockerfile +++ b/cmd/grafana-agent/Dockerfile @@ -4,7 +4,7 @@ # default when running `docker buildx build` or when DOCKER_BUILDKIT=1 is set # in environment variables. -FROM --platform=$BUILDPLATFORM grafana/agent-build-image:0.33.0 as build +FROM --platform=$BUILDPLATFORM grafana/agent-build-image:0.40.2 as build ARG BUILDPLATFORM ARG TARGETPLATFORM ARG TARGETOS diff --git a/cmd/grafana-agent/Dockerfile.windows b/cmd/grafana-agent/Dockerfile.windows index 23a0423db8..66b3d87e5f 100644 --- a/cmd/grafana-agent/Dockerfile.windows +++ b/cmd/grafana-agent/Dockerfile.windows @@ -1,4 +1,4 @@ -FROM grafana/agent-build-image:0.33.0-windows as builder +FROM grafana/agent-build-image:0.40.2-windows as builder ARG VERSION ARG RELEASE_BUILD=1 @@ -10,7 +10,12 @@ SHELL ["cmd", "/S", "/C"] # Creating new layers can be really slow on Windows so we clean up any caches # we can before moving on to the next step. RUN ""C:\Program Files\git\bin\bash.exe" -c "RELEASE_BUILD=${RELEASE_BUILD} VERSION=${VERSION} make generate-ui && rm -rf web/ui/node_modules && yarn cache clean --all"" -RUN ""C:\Program Files\git\bin\bash.exe" -c "RELEASE_BUILD=${RELEASE_BUILD} VERSION=${VERSION} GO_TAGS='builtinassets' make agent && go clean -cache -modcache"" + +RUN ""C:\Program Files\git\bin\bash.exe" -c "RELEASE_BUILD=${RELEASE_BUILD} VERSION=${VERSION} GO_TAGS='builtinassets' make agent"" +# In this case, we're separating the clean command from make agent to avoid an issue where access to some mod cache +# files is denied immediately after make agent, for example: +# "go: remove C:\go\pkg\mod\golang.org\toolchain@v0.0.1-go1.22.1.windows-amd64\bin\go.exe: Access is denied." +RUN ""C:\Program Files\git\bin\bash.exe" -c "go clean -cache -modcache"" # Use the smallest container possible for the final image FROM mcr.microsoft.com/windows/nanoserver:1809 diff --git a/docs/Makefile b/docs/Makefile index e233b21be6..b0c97d4d33 100644 --- a/docs/Makefile +++ b/docs/Makefile @@ -11,10 +11,10 @@ include docs.mk docs: check-cloudwatch-integration check-cloudwatch-integration: - $(PODMAN) run -v $(shell git rev-parse --show-toplevel):/repo -v $(shell pwd):/docs -w /repo golang:1.21-bullseye go run internal/static/integrations/cloudwatch_exporter/docs/doc.go check /docs/sources/reference/components/prometheus.exporter.cloudwatch.md + $(PODMAN) run -v $(shell git rev-parse --show-toplevel):/repo -v $(shell pwd):/docs -w /repo golang:1.22.1-bullseye go run internal/static/integrations/cloudwatch_exporter/docs/doc.go check /docs/sources/reference/components/prometheus.exporter.cloudwatch.md generate-cloudwatch-integration: - $(PODMAN) run -v $(shell git rev-parse --show-toplevel):/repo -v $(shell pwd):/docs -w /repo golang:1.21-bullseye go run internal/static/integrations/cloudwatch_exporter/docs/doc.go generate + $(PODMAN) run -v $(shell git rev-parse --show-toplevel):/repo -v $(shell pwd):/docs -w /repo golang:1.22.1-bullseye go run internal/static/integrations/cloudwatch_exporter/docs/doc.go generate sources/assets/hierarchy.svg: sources/operator/hierarchy.dot cat $< | $(PODMAN) run --rm -i nshine/dot dot -Tsvg > $@ diff --git a/docs/sources/_index.md b/docs/sources/_index.md index 55e8817108..a751d5bbed 100644 --- a/docs/sources/_index.md +++ b/docs/sources/_index.md @@ -12,7 +12,7 @@ cascade: # {{% param "PRODUCT_NAME" %}} -{{< param "PRODUCT_NAME" >}} is a vendor-neutral, batteries-included telemetry collector with configuration inspired by [Terraform][]. +{{< param "PRODUCT_NAME" >}} is an OpenTelemetry Collector distribution with configuration inspired by [Terraform][]. It is designed to be flexible, performant, and compatible with multiple ecosystems such as Prometheus and OpenTelemetry. {{< param "PRODUCT_NAME" >}} is based around **components**. Components are wired together to form programmable observability **pipelines** for telemetry collection, processing, and delivery. diff --git a/docs/sources/_index.md.t b/docs/sources/_index.md.t index 55e8817108..a751d5bbed 100644 --- a/docs/sources/_index.md.t +++ b/docs/sources/_index.md.t @@ -12,7 +12,7 @@ cascade: # {{% param "PRODUCT_NAME" %}} -{{< param "PRODUCT_NAME" >}} is a vendor-neutral, batteries-included telemetry collector with configuration inspired by [Terraform][]. +{{< param "PRODUCT_NAME" >}} is an OpenTelemetry Collector distribution with configuration inspired by [Terraform][]. It is designed to be flexible, performant, and compatible with multiple ecosystems such as Prometheus and OpenTelemetry. {{< param "PRODUCT_NAME" >}} is based around **components**. Components are wired together to form programmable observability **pipelines** for telemetry collection, processing, and delivery. diff --git a/docs/sources/reference/components/otelcol.auth.oauth2.md b/docs/sources/reference/components/otelcol.auth.oauth2.md index 28e7cc8e20..fd5f858f46 100644 --- a/docs/sources/reference/components/otelcol.auth.oauth2.md +++ b/docs/sources/reference/components/otelcol.auth.oauth2.md @@ -32,17 +32,26 @@ otelcol.auth.oauth2 "LABEL" { ## Arguments -Name | Type | Description | Default | Required -------------------|---------------------|------------------------------------------------------------|---------|--------- -`client_id` | `string` | The client identifier issued to the client. | | yes -`client_secret` | `secret` | The secret string associated with the client identifier. | | yes -`token_url` | `string` | The server endpoint URL from which to get tokens. | | yes -`endpoint_params` | `map(list(string))` | Additional parameters that are sent to the token endpoint. | `{}` | no -`scopes` | `list(string)` | Requested permissions associated for the client. | `[]` | no -`timeout` | `duration` | The timeout on the client connecting to `token_url`. | `"0s"` | no +Name | Type | Description | Default | Required +-------------------- | ------------------- | ---------------------------------------------------------------------------------- | ------- | -------- +`client_id` | `string` | The client identifier issued to the client. | | no +`client_id_file` | `string` | The file path to retrieve the client identifier issued to the client. | | no +`client_secret` | `secret` | The secret string associated with the client identifier. | | no +`client_secret_file` | `secret` | The file path to retrieve the secret string associated with the client identifier. | | no +`token_url` | `string` | The server endpoint URL from which to get tokens. | | yes +`endpoint_params` | `map(list(string))` | Additional parameters that are sent to the token endpoint. | `{}` | no +`scopes` | `list(string)` | Requested permissions associated for the client. | `[]` | no +`timeout` | `duration` | The timeout on the client connecting to `token_url`. | `"0s"` | no The `timeout` argument is used both for requesting initial tokens and for refreshing tokens. `"0s"` implies no timeout. +At least one of the `client_id` and `client_id_file` pair of arguments must be +set. In case both are set, `client_id_file` takes precedence. + +Similarly, at least one of the `client_secret` and `client_secret_file` pair of +arguments must be set. In case both are set, `client_secret_file` also takes +precedence. + ## Blocks The following blocks are supported inside the definition of diff --git a/go.mod b/go.mod index 6a989840fa..f017127088 100644 --- a/go.mod +++ b/go.mod @@ -1,6 +1,12 @@ module github.com/grafana/agent -go 1.21.0 +go 1.22.1 + +retract ( + v1.3.191 // Published accidentally + v1.2.99 // Published accidentally + v1.2.99-rc1 // Published accidentally +) require ( cloud.google.com/go/pubsub v1.33.0 diff --git a/internal/cmd/integration-tests/configs/otel-metrics-gen/Dockerfile b/internal/cmd/integration-tests/configs/otel-metrics-gen/Dockerfile index bc0c2cf3a9..ef2867ff2e 100644 --- a/internal/cmd/integration-tests/configs/otel-metrics-gen/Dockerfile +++ b/internal/cmd/integration-tests/configs/otel-metrics-gen/Dockerfile @@ -1,4 +1,4 @@ -FROM golang:1.21 as build +FROM golang:1.22.1 as build WORKDIR /app/ COPY go.mod go.sum ./ COPY syntax/go.mod syntax/go.sum ./syntax/ diff --git a/internal/cmd/integration-tests/configs/prom-gen/Dockerfile b/internal/cmd/integration-tests/configs/prom-gen/Dockerfile index 875b7bad7e..d0c118b495 100644 --- a/internal/cmd/integration-tests/configs/prom-gen/Dockerfile +++ b/internal/cmd/integration-tests/configs/prom-gen/Dockerfile @@ -1,4 +1,4 @@ -FROM golang:1.21 as build +FROM golang:1.22.1 as build WORKDIR /app/ COPY go.mod go.sum ./ COPY syntax/go.mod syntax/go.sum ./syntax/ diff --git a/internal/component/otelcol/auth/oauth2/oauth2.go b/internal/component/otelcol/auth/oauth2/oauth2.go index 95f88c5072..b90164711b 100644 --- a/internal/component/otelcol/auth/oauth2/oauth2.go +++ b/internal/component/otelcol/auth/oauth2/oauth2.go @@ -31,13 +31,15 @@ func init() { // Arguments configures the otelcol.auth.oauth2 component. type Arguments struct { - ClientID string `river:"client_id,attr"` - ClientSecret rivertypes.Secret `river:"client_secret,attr"` - TokenURL string `river:"token_url,attr"` - EndpointParams url.Values `river:"endpoint_params,attr,optional"` - Scopes []string `river:"scopes,attr,optional"` - TLSSetting otelcol.TLSClientArguments `river:"tls,block,optional"` - Timeout time.Duration `river:"timeout,attr,optional"` + ClientID string `river:"client_id,attr,optional"` + ClientIDFile string `river:"client_id_file,attr,optional"` + ClientSecret rivertypes.Secret `river:"client_secret,attr,optional"` + ClientSecretFile string `river:"client_secret_file,attr,optional"` + TokenURL string `river:"token_url,attr"` + EndpointParams url.Values `river:"endpoint_params,attr,optional"` + Scopes []string `river:"scopes,attr,optional"` + TLSSetting otelcol.TLSClientArguments `river:"tls,block,optional"` + Timeout time.Duration `river:"timeout,attr,optional"` } var _ auth.Arguments = Arguments{} @@ -45,13 +47,15 @@ var _ auth.Arguments = Arguments{} // Convert implements auth.Arguments. func (args Arguments) Convert() (otelcomponent.Config, error) { return &oauth2clientauthextension.Config{ - ClientID: args.ClientID, - ClientSecret: configopaque.String(args.ClientSecret), - TokenURL: args.TokenURL, - EndpointParams: args.EndpointParams, - Scopes: args.Scopes, - TLSSetting: *args.TLSSetting.Convert(), - Timeout: args.Timeout, + ClientID: args.ClientID, + ClientIDFile: args.ClientIDFile, + ClientSecret: configopaque.String(args.ClientSecret), + ClientSecretFile: args.ClientSecretFile, + TokenURL: args.TokenURL, + EndpointParams: args.EndpointParams, + Scopes: args.Scopes, + TLSSetting: *args.TLSSetting.Convert(), + Timeout: args.Timeout, }, nil } diff --git a/internal/component/otelcol/processor/transform/transform.go b/internal/component/otelcol/processor/transform/transform.go index 708ce7cdc4..aabae21e4c 100644 --- a/internal/component/otelcol/processor/transform/transform.go +++ b/internal/component/otelcol/processor/transform/transform.go @@ -53,9 +53,9 @@ func (c *ContextID) UnmarshalText(text []byte) error { } } -type contextStatementsSlice []contextStatements +type ContextStatementsSlice []ContextStatements -type contextStatements struct { +type ContextStatements struct { Context ContextID `river:"context,attr"` Statements []string `river:"statements,attr"` } @@ -64,9 +64,9 @@ type contextStatements struct { type Arguments struct { // ErrorMode determines how the processor reacts to errors that occur while processing a statement. ErrorMode ottl.ErrorMode `river:"error_mode,attr,optional"` - TraceStatements contextStatementsSlice `river:"trace_statements,block,optional"` - MetricStatements contextStatementsSlice `river:"metric_statements,block,optional"` - LogStatements contextStatementsSlice `river:"log_statements,block,optional"` + TraceStatements ContextStatementsSlice `river:"trace_statements,block,optional"` + MetricStatements ContextStatementsSlice `river:"metric_statements,block,optional"` + LogStatements ContextStatementsSlice `river:"log_statements,block,optional"` // Output configures where to send processed data. Required. Output *otelcol.ConsumerArguments `river:"output,block"` @@ -95,7 +95,7 @@ func (args *Arguments) Validate() error { return otelArgs.Validate() } -func (stmts *contextStatementsSlice) convert() []interface{} { +func (stmts *ContextStatementsSlice) convert() []interface{} { if stmts == nil { return nil } @@ -112,7 +112,7 @@ func (stmts *contextStatementsSlice) convert() []interface{} { return res } -func (args *contextStatements) convert() map[string]interface{} { +func (args *ContextStatements) convert() map[string]interface{} { if args == nil { return nil } diff --git a/internal/component/registry.go b/internal/component/registry.go index a7c0aca910..b382719bcd 100644 --- a/internal/component/registry.go +++ b/internal/component/registry.go @@ -124,7 +124,7 @@ type Registration struct { // sure the user is not accidentally using a component that is not yet stable - users // need to explicitly enable less-than-stable components via, for example, a command-line flag. // If a component is not stable enough, an attempt to create it via the controller will fail. - // The default stability level is Experimental. + // This field must be set to a non-zero value. Stability featuregate.Stability // An example Arguments value that the registered component expects to diff --git a/internal/converter/diag/diagnostics.go b/internal/converter/diag/diagnostics.go index 6c3fcfba45..94f416e6dd 100644 --- a/internal/converter/diag/diagnostics.go +++ b/internal/converter/diag/diagnostics.go @@ -46,12 +46,12 @@ func (ds Diagnostics) Error() string { return sb.String() } -func (ds Diagnostics) GenerateReport(writer io.Writer, reportType string) error { +func (ds Diagnostics) GenerateReport(writer io.Writer, reportType string, bypassErrors bool) error { switch reportType { case Text: - return generateTextReport(writer, ds) + return generateTextReport(writer, ds, bypassErrors) default: - return fmt.Errorf("Invalid diagnostic report type %q", reportType) + return fmt.Errorf("invalid diagnostic report type %q", reportType) } } @@ -66,3 +66,12 @@ func (ds *Diagnostics) RemoveDiagsBySeverity(severity Severity) { *ds = newDiags } + +func (ds *Diagnostics) HasSeverityLevel(severity Severity) bool { + for _, diag := range *ds { + if diag.Severity == severity { + return true + } + } + return false +} diff --git a/internal/converter/diag/report.go b/internal/converter/diag/report.go index 89d2a0e2b7..a41ed02ecd 100644 --- a/internal/converter/diag/report.go +++ b/internal/converter/diag/report.go @@ -6,10 +6,23 @@ import ( const Text = ".txt" -// generateTextReport generates a text report for the diagnostics. -func generateTextReport(writer io.Writer, ds Diagnostics) error { - content := ds.Error() +const criticalErrorFooter = ` + +A configuration file was not generated due to critical issues. Refer to the critical messages for more information.` + +const errorFooter = ` + +A configuration file was not generated due to errors. Refer to the error messages for more information. + +You can bypass the errors by using the --bypass-errors flag. Bypassing errors isn't recommended for production environments.` + +const successFooter = ` +A configuration file was generated successfully.` + +// generateTextReport generates a text report for the diagnostics. +func generateTextReport(writer io.Writer, ds Diagnostics, bypassErrors bool) error { + content := getContent(ds, bypassErrors) _, err := writer.Write([]byte(content)) if err != nil { return err @@ -17,3 +30,23 @@ func generateTextReport(writer io.Writer, ds Diagnostics) error { return nil } + +// getContent returns the formatted content for the report based on the diagnostics and bypassErrors. +func getContent(ds Diagnostics, bypassErrors bool) string { + var content string + switch { + case ds.HasSeverityLevel(SeverityLevelCritical): + content = criticalErrorFooter + ds.RemoveDiagsBySeverity(SeverityLevelInfo) + ds.RemoveDiagsBySeverity(SeverityLevelWarn) + ds.RemoveDiagsBySeverity(SeverityLevelError) + case ds.HasSeverityLevel(SeverityLevelError) && !bypassErrors: + content = errorFooter + ds.RemoveDiagsBySeverity(SeverityLevelInfo) + ds.RemoveDiagsBySeverity(SeverityLevelWarn) + default: + content = successFooter + } + + return ds.Error() + content +} diff --git a/internal/converter/diag/report_test.go b/internal/converter/diag/report_test.go new file mode 100644 index 0000000000..4bca751594 --- /dev/null +++ b/internal/converter/diag/report_test.go @@ -0,0 +1,81 @@ +package diag + +import ( + "bytes" + "testing" + + "github.com/stretchr/testify/require" +) + +func TestDiagReporting(t *testing.T) { + var ( + criticalDiagnostic = Diagnostic{ + Severity: SeverityLevelCritical, + Summary: "this is a critical diag", + } + errorDiagnostic = Diagnostic{ + Severity: SeverityLevelError, + Summary: "this is an error diag", + } + warnDiagnostic = Diagnostic{ + Severity: SeverityLevelWarn, + Summary: "this is a warn diag", + } + infoDiagnostic = Diagnostic{ + Severity: SeverityLevelInfo, + Summary: "this is an info diag", + } + ) + + tt := []struct { + name string + diags Diagnostics + bypassErrors bool + expectedMessage string + }{ + { + name: "Empty", + diags: Diagnostics{}, + expectedMessage: successFooter, + }, + { + name: "Critical", + diags: Diagnostics{criticalDiagnostic, errorDiagnostic, warnDiagnostic, infoDiagnostic}, + expectedMessage: `(Critical) this is a critical diag` + criticalErrorFooter, + }, + { + name: "Error", + diags: Diagnostics{errorDiagnostic, warnDiagnostic, infoDiagnostic}, + expectedMessage: `(Error) this is an error diag` + errorFooter, + }, + { + name: "Bypass Error", + diags: Diagnostics{errorDiagnostic, warnDiagnostic, infoDiagnostic}, + bypassErrors: true, + expectedMessage: `(Error) this is an error diag +(Warning) this is a warn diag +(Info) this is an info diag` + successFooter, + }, + { + name: "Warn", + diags: Diagnostics{warnDiagnostic, infoDiagnostic}, + expectedMessage: `(Warning) this is a warn diag +(Info) this is an info diag` + successFooter, + }, + { + name: "Info", + diags: Diagnostics{infoDiagnostic}, + expectedMessage: `(Info) this is an info diag` + successFooter, + }, + } + + for _, tc := range tt { + t.Run(tc.name, func(t *testing.T) { + var buf bytes.Buffer + err := generateTextReport(&buf, tc.diags, tc.bypassErrors) + require.NoError(t, err) + + require.Equal(t, tc.expectedMessage, buf.String()) + }) + } +} diff --git a/internal/converter/internal/otelcolconvert/converter_attributesprocessor.go b/internal/converter/internal/otelcolconvert/converter_attributesprocessor.go new file mode 100644 index 0000000000..c9b9486b26 --- /dev/null +++ b/internal/converter/internal/otelcolconvert/converter_attributesprocessor.go @@ -0,0 +1,85 @@ +package otelcolconvert + +import ( + "fmt" + + "github.com/grafana/agent/internal/component/otelcol" + "github.com/grafana/agent/internal/component/otelcol/processor/attributes" + "github.com/grafana/agent/internal/converter/diag" + "github.com/grafana/agent/internal/converter/internal/common" + "github.com/open-telemetry/opentelemetry-collector-contrib/processor/attributesprocessor" + "go.opentelemetry.io/collector/component" +) + +func init() { + converters = append(converters, attributesProcessorConverter{}) +} + +type attributesProcessorConverter struct{} + +func (attributesProcessorConverter) Factory() component.Factory { + return attributesprocessor.NewFactory() +} + +func (attributesProcessorConverter) InputComponentName() string { + return "otelcol.processor.attributes" +} + +func (attributesProcessorConverter) ConvertAndAppend(state *state, id component.InstanceID, cfg component.Config) diag.Diagnostics { + var diags diag.Diagnostics + + label := state.FlowComponentLabel() + + args := toAttributesProcessor(state, id, cfg.(*attributesprocessor.Config)) + block := common.NewBlockWithOverride([]string{"otelcol", "processor", "attributes"}, label, args) + + diags.Add( + diag.SeverityLevelInfo, + fmt.Sprintf("Converted %s into %s", stringifyInstanceID(id), stringifyBlock(block)), + ) + + state.Body().AppendBlock(block) + return diags +} + +func toAttributesProcessor(state *state, id component.InstanceID, cfg *attributesprocessor.Config) *attributes.Arguments { + var ( + nextMetrics = state.Next(id, component.DataTypeMetrics) + nextTraces = state.Next(id, component.DataTypeTraces) + nextLogs = state.Next(id, component.DataTypeLogs) + ) + + return &attributes.Arguments{ + Match: toMatchConfig(cfg), + Actions: toAttrActionKeyValue(encodeMapslice(cfg.Actions)), + Output: &otelcol.ConsumerArguments{ + Metrics: toTokenizedConsumers(nextMetrics), + Logs: toTokenizedConsumers(nextLogs), + Traces: toTokenizedConsumers(nextTraces)}, + } +} + +func toMatchConfig(cfg *attributesprocessor.Config) otelcol.MatchConfig { + return otelcol.MatchConfig{ + Include: toMatchProperties(encodeMapstruct(cfg.Include)), + Exclude: toMatchProperties(encodeMapstruct(cfg.Exclude)), + } +} + +func toAttrActionKeyValue(cfg []map[string]any) []otelcol.AttrActionKeyValue { + result := make([]otelcol.AttrActionKeyValue, 0) + + for _, action := range cfg { + result = append(result, otelcol.AttrActionKeyValue{ + Key: action["key"].(string), + Value: action["value"], + RegexPattern: action["pattern"].(string), + FromAttribute: action["from_attribute"].(string), + FromContext: action["from_context"].(string), + ConvertedType: action["converted_type"].(string), + Action: encodeString(action["action"]), + }) + } + + return result +} diff --git a/internal/converter/internal/otelcolconvert/converter_jaegerremotesamplingextension.go b/internal/converter/internal/otelcolconvert/converter_jaegerremotesamplingextension.go new file mode 100644 index 0000000000..2076a7290d --- /dev/null +++ b/internal/converter/internal/otelcolconvert/converter_jaegerremotesamplingextension.go @@ -0,0 +1,73 @@ +package otelcolconvert + +import ( + "fmt" + + "github.com/grafana/agent/internal/component/otelcol/extension/jaeger_remote_sampling" + "github.com/grafana/agent/internal/converter/diag" + "github.com/grafana/agent/internal/converter/internal/common" + "github.com/open-telemetry/opentelemetry-collector-contrib/extension/jaegerremotesampling" + "go.opentelemetry.io/collector/component" +) + +func init() { + converters = append(converters, jaegerRemoteSamplingExtensionConverter{}) +} + +type jaegerRemoteSamplingExtensionConverter struct{} + +func (jaegerRemoteSamplingExtensionConverter) Factory() component.Factory { + return jaegerremotesampling.NewFactory() +} + +func (jaegerRemoteSamplingExtensionConverter) InputComponentName() string { + return "otelcol.extension.jaeger_remote_sampling" +} + +func (jaegerRemoteSamplingExtensionConverter) ConvertAndAppend(state *state, id component.InstanceID, cfg component.Config) diag.Diagnostics { + var diags diag.Diagnostics + + label := state.FlowComponentLabel() + + args := toJaegerRemoteSamplingExtension(cfg.(*jaegerremotesampling.Config)) + block := common.NewBlockWithOverride([]string{"otelcol", "extension", "jaeger_remote_sampling"}, label, args) + + diags.Add( + diag.SeverityLevelInfo, + fmt.Sprintf("Converted %s into %s", stringifyInstanceID(id), stringifyBlock(block)), + ) + + state.Body().AppendBlock(block) + return diags +} + +func toJaegerRemoteSamplingExtension(cfg *jaegerremotesampling.Config) *jaeger_remote_sampling.Arguments { + if cfg == nil { + return nil + } + + var grpc *jaeger_remote_sampling.GRPCServerArguments + if cfg.GRPCServerSettings != nil { + grpc = (*jaeger_remote_sampling.GRPCServerArguments)(toGRPCServerArguments(cfg.GRPCServerSettings)) + } + var http *jaeger_remote_sampling.HTTPServerArguments + if cfg.HTTPServerSettings != nil { + http = (*jaeger_remote_sampling.HTTPServerArguments)(toHTTPServerArguments(cfg.HTTPServerSettings)) + } + var remote *jaeger_remote_sampling.GRPCClientArguments + if cfg.Source.Remote != nil { + r := toGRPCClientArguments(*cfg.Source.Remote) + remote = (*jaeger_remote_sampling.GRPCClientArguments)(&r) + } + + return &jaeger_remote_sampling.Arguments{ + GRPC: grpc, + HTTP: http, + Source: jaeger_remote_sampling.ArgumentsSource{ + Content: "", + Remote: remote, + File: cfg.Source.File, + ReloadInterval: cfg.Source.ReloadInterval, + }, + } +} diff --git a/internal/converter/internal/otelcolconvert/converter_oauth2clientauthextension.go b/internal/converter/internal/otelcolconvert/converter_oauth2clientauthextension.go new file mode 100644 index 0000000000..14ba01ea91 --- /dev/null +++ b/internal/converter/internal/otelcolconvert/converter_oauth2clientauthextension.go @@ -0,0 +1,53 @@ +package otelcolconvert + +import ( + "fmt" + + "github.com/grafana/agent/internal/component/otelcol/auth/oauth2" + "github.com/grafana/agent/internal/converter/diag" + "github.com/grafana/agent/internal/converter/internal/common" + "github.com/grafana/river/rivertypes" + "github.com/open-telemetry/opentelemetry-collector-contrib/extension/oauth2clientauthextension" + "go.opentelemetry.io/collector/component" +) + +func init() { + converters = append(converters, oauth2ClientAuthExtensionConverter{}) +} + +type oauth2ClientAuthExtensionConverter struct{} + +func (oauth2ClientAuthExtensionConverter) Factory() component.Factory { + return oauth2clientauthextension.NewFactory() +} + +func (oauth2ClientAuthExtensionConverter) InputComponentName() string { return "otelcol.auth.oauth2" } + +func (oauth2ClientAuthExtensionConverter) ConvertAndAppend(state *state, id component.InstanceID, cfg component.Config) diag.Diagnostics { + var diags diag.Diagnostics + + label := state.FlowComponentLabel() + + args := toOAuth2ClientAuthExtension(cfg.(*oauth2clientauthextension.Config)) + block := common.NewBlockWithOverride([]string{"otelcol", "auth", "oauth2"}, label, args) + + diags.Add( + diag.SeverityLevelInfo, + fmt.Sprintf("Converted %s into %s", stringifyInstanceID(id), stringifyBlock(block)), + ) + + state.Body().AppendBlock(block) + return diags +} + +func toOAuth2ClientAuthExtension(cfg *oauth2clientauthextension.Config) *oauth2.Arguments { + return &oauth2.Arguments{ + ClientID: cfg.ClientID, + ClientSecret: rivertypes.Secret(cfg.ClientSecret), + TokenURL: cfg.TokenURL, + EndpointParams: cfg.EndpointParams, + Scopes: cfg.Scopes, + TLSSetting: toTLSClientArguments(cfg.TLSSetting), + Timeout: cfg.Timeout, + } +} diff --git a/internal/converter/internal/otelcolconvert/converter_transformprocessor.go b/internal/converter/internal/otelcolconvert/converter_transformprocessor.go new file mode 100644 index 0000000000..694046bb21 --- /dev/null +++ b/internal/converter/internal/otelcolconvert/converter_transformprocessor.go @@ -0,0 +1,75 @@ +package otelcolconvert + +import ( + "fmt" + + "github.com/grafana/agent/internal/component/otelcol" + "github.com/grafana/agent/internal/component/otelcol/processor/transform" + "github.com/grafana/agent/internal/converter/diag" + "github.com/grafana/agent/internal/converter/internal/common" + "github.com/open-telemetry/opentelemetry-collector-contrib/processor/transformprocessor" + "go.opentelemetry.io/collector/component" +) + +func init() { + converters = append(converters, transformProcessorConverter{}) +} + +type transformProcessorConverter struct{} + +func (transformProcessorConverter) Factory() component.Factory { + return transformprocessor.NewFactory() +} + +func (transformProcessorConverter) InputComponentName() string { + return "otelcol.processor.transform" +} + +func (transformProcessorConverter) ConvertAndAppend(state *state, id component.InstanceID, cfg component.Config) diag.Diagnostics { + var diags diag.Diagnostics + + label := state.FlowComponentLabel() + + args := toTransformProcessor(state, id, cfg.(*transformprocessor.Config)) + block := common.NewBlockWithOverride([]string{"otelcol", "processor", "transform"}, label, args) + + diags.Add( + diag.SeverityLevelInfo, + fmt.Sprintf("Converted %s into %s", stringifyInstanceID(id), stringifyBlock(block)), + ) + + state.Body().AppendBlock(block) + return diags +} + +func toTransformProcessor(state *state, id component.InstanceID, cfg *transformprocessor.Config) *transform.Arguments { + var ( + nextMetrics = state.Next(id, component.DataTypeMetrics) + nextLogs = state.Next(id, component.DataTypeLogs) + nextTraces = state.Next(id, component.DataTypeTraces) + ) + + return &transform.Arguments{ + ErrorMode: cfg.ErrorMode, + TraceStatements: toContextStatements(encodeMapslice(cfg.TraceStatements)), + MetricStatements: toContextStatements(encodeMapslice(cfg.MetricStatements)), + LogStatements: toContextStatements(encodeMapslice(cfg.LogStatements)), + Output: &otelcol.ConsumerArguments{ + Metrics: toTokenizedConsumers(nextMetrics), + Logs: toTokenizedConsumers(nextLogs), + Traces: toTokenizedConsumers(nextTraces), + }, + } +} + +func toContextStatements(in []map[string]any) []transform.ContextStatements { + res := make([]transform.ContextStatements, 0, len(in)) + for _, s := range in { + res = append(res, transform.ContextStatements{ + Context: transform.ContextID(encodeString(s["context"])), + Statements: s["statements"].([]string), + }) + } + + return res +} diff --git a/internal/converter/internal/otelcolconvert/testdata/attributes.river b/internal/converter/internal/otelcolconvert/testdata/attributes.river new file mode 100644 index 0000000000..493640814c --- /dev/null +++ b/internal/converter/internal/otelcolconvert/testdata/attributes.river @@ -0,0 +1,64 @@ +otelcol.receiver.otlp "default" { + grpc { } + + http { } + + output { + metrics = [otelcol.processor.attributes.default_example.input] + logs = [otelcol.processor.attributes.default_example.input] + traces = [otelcol.processor.attributes.default_example.input] + } +} + +otelcol.processor.attributes "default_example" { + action { + key = "db.table" + action = "delete" + } + + action { + key = "redacted_span" + value = true + action = "upsert" + } + + action { + key = "copy_key" + from_attribute = "key_original" + action = "update" + } + + action { + key = "account_id" + value = 2245 + action = "insert" + } + + action { + key = "account_password" + action = "delete" + } + + action { + key = "account_email" + action = "hash" + } + + action { + key = "http.status_code" + converted_type = "int" + action = "convert" + } + + output { + metrics = [otelcol.exporter.otlp.default.input] + logs = [otelcol.exporter.otlp.default.input] + traces = [otelcol.exporter.otlp.default.input] + } +} + +otelcol.exporter.otlp "default" { + client { + endpoint = "database:4317" + } +} diff --git a/internal/converter/internal/otelcolconvert/testdata/attributes.yaml b/internal/converter/internal/otelcolconvert/testdata/attributes.yaml new file mode 100644 index 0000000000..dc9cfcd6e7 --- /dev/null +++ b/internal/converter/internal/otelcolconvert/testdata/attributes.yaml @@ -0,0 +1,51 @@ +receivers: + otlp: + protocols: + grpc: + http: + +exporters: + otlp: + # Our defaults have drifted from upstream, so we explicitly set our + # defaults below (balancer_name and queue_size). + endpoint: database:4317 + balancer_name: pick_first + sending_queue: + queue_size: 5000 + +processors: + attributes/example: + actions: + - key: db.table + action: delete + - key: redacted_span + value: true + action: upsert + - key: copy_key + from_attribute: key_original + action: update + - key: account_id + value: 2245 + action: insert + - key: account_password + action: delete + - key: account_email + action: hash + - key: http.status_code + action: convert + converted_type: int + +service: + pipelines: + metrics: + receivers: [otlp] + processors: [attributes/example] + exporters: [otlp] + logs: + receivers: [otlp] + processors: [attributes/example] + exporters: [otlp] + traces: + receivers: [otlp] + processors: [attributes/example] + exporters: [otlp] diff --git a/internal/converter/internal/otelcolconvert/testdata/jaegerremotesampling.river b/internal/converter/internal/otelcolconvert/testdata/jaegerremotesampling.river new file mode 100644 index 0000000000..5d1efebc69 --- /dev/null +++ b/internal/converter/internal/otelcolconvert/testdata/jaegerremotesampling.river @@ -0,0 +1,38 @@ +otelcol.extension.jaeger_remote_sampling "default" { + grpc { } + + http { } + + source { + remote { + endpoint = "jaeger-collector:14250" + } + reload_interval = "30s" + } +} + +otelcol.receiver.jaeger "default" { + protocols { + grpc { } + + thrift_http { } + + thrift_binary { + max_packet_size = "63KiB488B" + } + + thrift_compact { + max_packet_size = "63KiB488B" + } + } + + output { + traces = [otelcol.exporter.otlp.default.input] + } +} + +otelcol.exporter.otlp "default" { + client { + endpoint = "database:4317" + } +} diff --git a/internal/converter/internal/otelcolconvert/testdata/jaegerremotesampling.yaml b/internal/converter/internal/otelcolconvert/testdata/jaegerremotesampling.yaml new file mode 100644 index 0000000000..d85b388771 --- /dev/null +++ b/internal/converter/internal/otelcolconvert/testdata/jaegerremotesampling.yaml @@ -0,0 +1,43 @@ +extensions: + jaegerremotesampling: + # Our defaults have drifted from upstream so we explicitly set our defaults + # below by adding the 0.0.0.0 prefix for http.endpoint and grpc.endpoint. + http: + endpoint: "0.0.0.0:5778" + grpc: + endpoint: "0.0.0.0:14250" + source: + reload_interval: 30s + remote: + endpoint: jaeger-collector:14250 + # Our defaults have drifted from upstream so we explicitly set our + # defaults below for the remote block that is used as GRPC client + # arguments (balancer_name, compression, write_buffer_size). + balancer_name: pick_first + compression: "gzip" + write_buffer_size: 524288 # 512 * 1024 + +receivers: + jaeger: + protocols: + grpc: + thrift_binary: + thrift_compact: + thrift_http: + +exporters: + otlp: + # Our defaults have drifted from upstream, so we explicitly set our + # defaults below (balancer_name and queue_size). + endpoint: database:4317 + balancer_name: pick_first + sending_queue: + queue_size: 5000 + +service: + extensions: [jaegerremotesampling] + pipelines: + traces: + receivers: [jaeger] + processors: [] + exporters: [otlp] diff --git a/internal/converter/internal/otelcolconvert/testdata/oauth2.river b/internal/converter/internal/otelcolconvert/testdata/oauth2.river new file mode 100644 index 0000000000..9a125399bc --- /dev/null +++ b/internal/converter/internal/otelcolconvert/testdata/oauth2.river @@ -0,0 +1,44 @@ +otelcol.auth.oauth2 "default" { + client_id = "someclientid" + client_secret = "someclientsecret" + token_url = "https://example.com/oauth2/default/v1/token" + endpoint_params = { + audience = ["someaudience"], + } + scopes = ["api.metrics"] + + tls { + ca_file = "/var/lib/mycert.pem" + cert_file = "certfile" + key_file = "keyfile" + insecure = true + } + timeout = "2s" +} + +otelcol.receiver.otlp "default" { + grpc { } + + output { + metrics = [otelcol.exporter.otlp.default_withauth.input, otelcol.exporter.otlphttp.default_noauth.input] + logs = [otelcol.exporter.otlp.default_withauth.input, otelcol.exporter.otlphttp.default_noauth.input] + traces = [otelcol.exporter.otlp.default_withauth.input, otelcol.exporter.otlphttp.default_noauth.input] + } +} + +otelcol.exporter.otlp "default_withauth" { + client { + endpoint = "database:4317" + + tls { + ca_file = "/tmp/certs/ca.pem" + } + auth = otelcol.auth.oauth2.default.handler + } +} + +otelcol.exporter.otlphttp "default_noauth" { + client { + endpoint = "database:4318" + } +} diff --git a/internal/converter/internal/otelcolconvert/testdata/oauth2.yaml b/internal/converter/internal/otelcolconvert/testdata/oauth2.yaml new file mode 100644 index 0000000000..d337d40bca --- /dev/null +++ b/internal/converter/internal/otelcolconvert/testdata/oauth2.yaml @@ -0,0 +1,61 @@ +extensions: + oauth2client/noop: # this extension is not defined in services and shouldn't be converted + client_id: dummyclientid + client_secret: dummyclientsecret + token_url: https://example.com/oauth2/default/v1/token + oauth2client: + client_id: someclientid + client_secret: someclientsecret + endpoint_params: + audience: someaudience + token_url: https://example.com/oauth2/default/v1/token + scopes: ["api.metrics"] + # tls settings for the token client + tls: + insecure: true + ca_file: /var/lib/mycert.pem + cert_file: certfile + key_file: keyfile + # timeout for the token client + timeout: 2s + +receivers: + otlp: + protocols: + grpc: + +exporters: + otlphttp/noauth: + # Our defaults have drifted from upstream, so we explicitly set our + # defaults below for queue_size. + endpoint: database:4318 + sending_queue: + queue_size: 5000 + + otlp/withauth: + tls: + ca_file: /tmp/certs/ca.pem + auth: + authenticator: oauth2client + # Our defaults have drifted from upstream, so we explicitly set our + # defaults below (balancer_name and queue_size). + endpoint: database:4317 + balancer_name: pick_first + sending_queue: + queue_size: 5000 + +service: + extensions: [oauth2client] + pipelines: + metrics: + receivers: [otlp] + processors: [] + exporters: [otlp/withauth, otlphttp/noauth] + logs: + receivers: [otlp] + processors: [] + exporters: [otlp/withauth, otlphttp/noauth] + traces: + receivers: [otlp] + processors: [] + exporters: [otlp/withauth, otlphttp/noauth] diff --git a/internal/converter/internal/otelcolconvert/testdata/transform.river b/internal/converter/internal/otelcolconvert/testdata/transform.river new file mode 100644 index 0000000000..5b7902f04f --- /dev/null +++ b/internal/converter/internal/otelcolconvert/testdata/transform.river @@ -0,0 +1,62 @@ +otelcol.receiver.otlp "default" { + grpc { } + + http { } + + output { + metrics = [otelcol.processor.transform.default.input] + logs = [otelcol.processor.transform.default.input] + traces = [otelcol.processor.transform.default.input] + } +} + +otelcol.processor.transform "default" { + error_mode = "ignore" + + trace_statements { + context = "resource" + statements = ["keep_keys(attributes, [\"service.name\", \"service.namespace\", \"cloud.region\", \"process.command_line\"])", "replace_pattern(attributes[\"process.command_line\"], \"password\\\\=[^\\\\s]*(\\\\s?)\", \"password=***\")", "limit(attributes, 100, [])", "truncate_all(attributes, 4096)"] + } + + trace_statements { + context = "span" + statements = ["set(status.code, 1) where attributes[\"http.path\"] == \"/health\"", "set(name, attributes[\"http.route\"])", "replace_match(attributes[\"http.target\"], \"/user/*/list/*\", \"/user/{userId}/list/{listId}\")", "limit(attributes, 100, [])", "truncate_all(attributes, 4096)"] + } + + metric_statements { + context = "resource" + statements = ["keep_keys(attributes, [\"host.name\"])", "truncate_all(attributes, 4096)"] + } + + metric_statements { + context = "metric" + statements = ["set(description, \"Sum\") where type == \"Sum\""] + } + + metric_statements { + context = "datapoint" + statements = ["limit(attributes, 100, [\"host.name\"])", "truncate_all(attributes, 4096)", "convert_sum_to_gauge() where metric.name == \"system.processes.count\"", "convert_gauge_to_sum(\"cumulative\", false) where metric.name == \"prometheus_metric\""] + } + + log_statements { + context = "resource" + statements = ["keep_keys(attributes, [\"service.name\", \"service.namespace\", \"cloud.region\"])"] + } + + log_statements { + context = "log" + statements = ["set(severity_text, \"FAIL\") where body == \"request failed\"", "replace_all_matches(attributes, \"/user/*/list/*\", \"/user/{userId}/list/{listId}\")", "replace_all_patterns(attributes, \"value\", \"/account/\\\\d{4}\", \"/account/{accountId}\")", "set(body, attributes[\"http.route\"])"] + } + + output { + metrics = [otelcol.exporter.otlp.default.input] + logs = [otelcol.exporter.otlp.default.input] + traces = [otelcol.exporter.otlp.default.input] + } +} + +otelcol.exporter.otlp "default" { + client { + endpoint = "database:4317" + } +} diff --git a/internal/converter/internal/otelcolconvert/testdata/transform.yaml b/internal/converter/internal/otelcolconvert/testdata/transform.yaml new file mode 100644 index 0000000000..4bd271d264 --- /dev/null +++ b/internal/converter/internal/otelcolconvert/testdata/transform.yaml @@ -0,0 +1,75 @@ +receivers: + otlp: + protocols: + grpc: + http: + +processors: + transform: + error_mode: ignore + trace_statements: + - context: resource + statements: + - keep_keys(attributes, ["service.name", "service.namespace", "cloud.region", "process.command_line"]) + - replace_pattern(attributes["process.command_line"], "password\\=[^\\s]*(\\s?)", "password=***") + - limit(attributes, 100, []) + - truncate_all(attributes, 4096) + - context: span + statements: + - set(status.code, 1) where attributes["http.path"] == "/health" + - set(name, attributes["http.route"]) + - replace_match(attributes["http.target"], "/user/*/list/*", "/user/{userId}/list/{listId}") + - limit(attributes, 100, []) + - truncate_all(attributes, 4096) + + metric_statements: + - context: resource + statements: + - keep_keys(attributes, ["host.name"]) + - truncate_all(attributes, 4096) + - context: metric + statements: + - set(description, "Sum") where type == "Sum" + - context: datapoint + statements: + - limit(attributes, 100, ["host.name"]) + - truncate_all(attributes, 4096) + - convert_sum_to_gauge() where metric.name == "system.processes.count" + - convert_gauge_to_sum("cumulative", false) where metric.name == "prometheus_metric" + + log_statements: + - context: resource + statements: + - keep_keys(attributes, ["service.name", "service.namespace", "cloud.region"]) + - context: log + statements: + - set(severity_text, "FAIL") where body == "request failed" + - replace_all_matches(attributes, "/user/*/list/*", "/user/{userId}/list/{listId}") + - replace_all_patterns(attributes, "value", "/account/\\d{4}", "/account/{accountId}") + - set(body, attributes["http.route"]) + + +exporters: + otlp: + # Our defaults have drifted from upstream, so we explicitly set our + # defaults below (balancer_name and queue_size). + endpoint: database:4317 + balancer_name: pick_first + sending_queue: + queue_size: 5000 + +service: + pipelines: + metrics: + receivers: [otlp] + processors: [transform] + exporters: [otlp] + logs: + receivers: [otlp] + processors: [transform] + exporters: [otlp] + traces: + receivers: [otlp] + processors: [transform] + exporters: [otlp] + diff --git a/internal/flow/flow_services_test.go b/internal/flow/flow_services_test.go index 86e375132f..80404b80f4 100644 --- a/internal/flow/flow_services_test.go +++ b/internal/flow/flow_services_test.go @@ -63,6 +63,7 @@ func TestServices_Configurable(t *testing.T) { return service.Definition{ Name: "fake", ConfigType: ServiceOptions{}, + Stability: featuregate.StabilityBeta, } }, @@ -117,6 +118,7 @@ func TestServices_Configurable_Optional(t *testing.T) { return service.Definition{ Name: "fake", ConfigType: ServiceOptions{}, + Stability: featuregate.StabilityBeta, } }, diff --git a/internal/flow/import_test.go b/internal/flow/import_test.go index 89fddafc46..691d9246b5 100644 --- a/internal/flow/import_test.go +++ b/internal/flow/import_test.go @@ -4,6 +4,7 @@ import ( "context" "io/fs" "os" + "os/exec" "path/filepath" "strings" "sync" @@ -250,6 +251,101 @@ func TestImportError(t *testing.T) { } } +func TestPullUpdating(t *testing.T) { + // Previously we used fetch instead of pull, which would set the FETCH_HEAD but not HEAD + // This caused changes not to propagate if there were changes, since HEAD was pinned to whatever it was on the initial download. + // Switching to pull removes this problem at the expense of network bandwidth. + // Tried switching to FETCH_HEAD but FETCH_HEAD is only set on fetch and not initial repo clone so we would need to + // remember to always call fetch after clone. + // + // This test ensures we can pull the correct values down if they update no matter what, it works by creating a local + // file based git repo then committing a file, running the component, then updating the file in the repo. + testRepo := t.TempDir() + + contents := `declare "add" { + argument "a" {} + argument "b" {} + + export "sum" { + value = argument.a.value + argument.b.value + } +}` + main := ` +import.git "testImport" { + repository = "` + testRepo + `" + path = "math.river" + pull_frequency = "5s" +} + +testImport.add "cc" { + a = 1 + b = 1 +} +` + init := exec.Command("git", "init", testRepo) + err := init.Run() + require.NoError(t, err) + math := filepath.Join(testRepo, "math.river") + err = os.WriteFile(math, []byte(contents), 0666) + require.NoError(t, err) + add := exec.Command("git", "add", ".") + add.Dir = testRepo + err = add.Run() + require.NoError(t, err) + commit := exec.Command("git", "commit", "-m \"test\"") + commit.Dir = testRepo + err = commit.Run() + require.NoError(t, err) + + defer verifyNoGoroutineLeaks(t) + ctrl, f := setup(t, main) + err = ctrl.LoadSource(f, nil) + require.NoError(t, err) + ctx, cancel := context.WithCancel(context.Background()) + + var wg sync.WaitGroup + defer func() { + cancel() + wg.Wait() + }() + + wg.Add(1) + go func() { + defer wg.Done() + ctrl.Run(ctx) + }() + + // Check for initial condition + require.Eventually(t, func() bool { + export := getExport[map[string]interface{}](t, ctrl, "", "testImport.add.cc") + return export["sum"] == 2 + }, 3*time.Second, 10*time.Millisecond) + + contentsMore := `declare "add" { + argument "a" {} + argument "b" {} + + export "sum" { + value = argument.a.value + argument.b.value + 1 + } +}` + err = os.WriteFile(math, []byte(contentsMore), 0666) + require.NoError(t, err) + add2 := exec.Command("git", "add", ".") + add2.Dir = testRepo + add2.Run() + + commit2 := exec.Command("git", "commit", "-m \"test2\"") + commit2.Dir = testRepo + commit2.Run() + + // Check for final condition. + require.Eventually(t, func() bool { + export := getExport[map[string]interface{}](t, ctrl, "", "testImport.add.cc") + return export["sum"] == 3 + }, 20*time.Second, 1*time.Millisecond) +} + func testConfig(t *testing.T, config string, reloadConfig string, update func()) { defer verifyNoGoroutineLeaks(t) ctrl, f := setup(t, config) diff --git a/internal/flow/internal/controller/loader.go b/internal/flow/internal/controller/loader.go index 43e102963e..8921d5ff18 100644 --- a/internal/flow/internal/controller/loader.go +++ b/internal/flow/internal/controller/loader.go @@ -10,6 +10,7 @@ import ( "time" "github.com/go-kit/log" + "github.com/grafana/agent/internal/featuregate" "github.com/grafana/agent/internal/flow/internal/dag" "github.com/grafana/agent/internal/flow/internal/worker" "github.com/grafana/agent/internal/flow/logging/level" @@ -441,6 +442,19 @@ func (l *Loader) populateServiceNodes(g *dag.Graph, serviceBlocks []*ast.BlockSt node := g.GetByID(blockID).(*ServiceNode) + // Don't permit configuring services that have a lower stability level than + // what is currently enabled. + nodeStability := node.Service().Definition().Stability + if err := featuregate.CheckAllowed(nodeStability, l.globals.MinStability, fmt.Sprintf("block %q", blockID)); err != nil { + diags.Add(diag.Diagnostic{ + Severity: diag.SeverityLevelError, + Message: err.Error(), + StartPos: ast.StartPos(block).Position(), + EndPos: ast.EndPos(block).Position(), + }) + continue + } + // Blocks assigned to services are reset to nil in the previous loop. // // If the block is non-nil, it means that there was a duplicate block diff --git a/internal/flow/internal/controller/loader_test.go b/internal/flow/internal/controller/loader_test.go index 672e2dce67..398cd5cae0 100644 --- a/internal/flow/internal/controller/loader_test.go +++ b/internal/flow/internal/controller/loader_test.go @@ -1,6 +1,7 @@ package controller_test import ( + "context" "errors" "os" "strings" @@ -11,6 +12,7 @@ import ( "github.com/grafana/agent/internal/flow/internal/controller" "github.com/grafana/agent/internal/flow/internal/dag" "github.com/grafana/agent/internal/flow/logging" + "github.com/grafana/agent/internal/service" "github.com/grafana/river/ast" "github.com/grafana/river/diag" "github.com/grafana/river/parser" @@ -316,6 +318,60 @@ func TestLoader(t *testing.T) { }) } +func TestLoader_Services(t *testing.T) { + testFile := ` + testsvc { } + ` + + testService := &fakeService{ + DefinitionFunc: func() service.Definition { + return service.Definition{ + Name: "testsvc", + ConfigType: struct { + Name string `river:"name,attr,optional"` + }{}, + Stability: featuregate.StabilityBeta, + } + }, + } + + newLoaderOptionsWithStability := func(stability featuregate.Stability) controller.LoaderOptions { + l, _ := logging.New(os.Stderr, logging.DefaultOptions) + return controller.LoaderOptions{ + ComponentGlobals: controller.ComponentGlobals{ + Logger: l, + TraceProvider: noop.NewTracerProvider(), + DataPath: t.TempDir(), + MinStability: stability, + OnBlockNodeUpdate: func(cn controller.BlockNode) { /* no-op */ }, + Registerer: prometheus.NewRegistry(), + NewModuleController: func(id string) controller.ModuleController { + return nil + }, + }, + Services: []service.Service{testService}, + } + } + + t.Run("Load with service at correct stability level", func(t *testing.T) { + l := controller.NewLoader(newLoaderOptionsWithStability(featuregate.StabilityBeta)) + diags := applyFromContent(t, l, []byte(testFile), nil, nil) + require.NoError(t, diags.ErrorOrNil()) + }) + + t.Run("Load with service below minimum stabilty level", func(t *testing.T) { + l := controller.NewLoader(newLoaderOptionsWithStability(featuregate.StabilityStable)) + diags := applyFromContent(t, l, []byte(testFile), nil, nil) + require.ErrorContains(t, diags.ErrorOrNil(), `block "testsvc" is at stability level "beta", which is below the minimum allowed stability level "stable"`) + }) + + t.Run("Load with undefined minimum stability level", func(t *testing.T) { + l := controller.NewLoader(newLoaderOptionsWithStability(featuregate.StabilityUndefined)) + diags := applyFromContent(t, l, []byte(testFile), nil, nil) + require.ErrorContains(t, diags.ErrorOrNil(), `stability levels must be defined: got "beta" as stability of block "testsvc" and