Skip to content

Commit c41a8b4

Browse files
authored
feat(lambda-promtail): add relabeling support for log entries (#15600)
1 parent ccee7f9 commit c41a8b4

File tree

6 files changed

+172
-8
lines changed

6 files changed

+172
-8
lines changed

‎docs/sources/send-data/lambda-promtail/_index.md

+110-3
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,13 @@
11
---
2-
title: Lambda Promtail client
2+
title: Lambda Promtail client
33
menuTitle: Lambda Promtail
44
description: Configuring the Lambda Promtail client to send logs to Loki.
5-
aliases:
5+
aliases:
66
- ../clients/lambda-promtail/
77
weight: 700
88
---
99

10-
# Lambda Promtail client
10+
# Lambda Promtail client
1111

1212
Grafana Loki includes [Terraform](https://www.terraform.io/) and [CloudFormation](https://aws.amazon.com/cloudformation/) for shipping Cloudwatch, Cloudtrail, VPC Flow Logs and loadbalancer logs to Loki via a [lambda function](https://aws.amazon.com/lambda/). This is done via [lambda-promtail](https://github.com/grafana/loki/blob/main/tools/lambda-promtail) which processes cloudwatch events and propagates them to Loki (or a Promtail instance) via the push-api [scrape config]({{< relref "../../send-data/promtail/configuration#loki_push_api" >}}).
1313

@@ -161,6 +161,113 @@ Incoming logs can have seven special labels assigned to them which can be used i
161161
- `__aws_s3_log_lb`: The name of the loadbalancer.
162162
- `__aws_s3_log_lb_owner`: The Account ID of the loadbalancer owner.
163163

164+
## Relabeling Configuration
165+
166+
Lambda-promtail supports Prometheus-style relabeling through the `RELABEL_CONFIGS` environment variable. This allows you to modify, keep, or drop labels before sending logs to Loki. The configuration is provided as a JSON array of relabel configurations. The relabeling functionality follows the same principles as Prometheus relabeling - for a detailed explanation of how relabeling works, see [How relabeling in Prometheus works](https://grafana.com/blog/2022/03/21/how-relabeling-in-prometheus-works/).
167+
168+
Example configurations:
169+
170+
1. Rename a label and capture regex groups:
171+
```json
172+
{
173+
"RELABEL_CONFIGS": [
174+
{
175+
"source_labels": ["__aws_log_type"],
176+
"target_label": "log_type",
177+
"action": "replace",
178+
"regex": "(.*)",
179+
"replacement": "${1}"
180+
}
181+
]
182+
}
183+
```
184+
185+
2. Keep only specific log types (useful for filtering):
186+
```json
187+
{
188+
"RELABEL_CONFIGS": [
189+
{
190+
"source_labels": ["__aws_log_type"],
191+
"regex": "s3_.*",
192+
"action": "keep"
193+
}
194+
]
195+
}
196+
```
197+
198+
3. Drop internal AWS labels (cleanup):
199+
```json
200+
{
201+
"RELABEL_CONFIGS": [
202+
{
203+
"regex": "__aws_.*",
204+
"action": "labeldrop"
205+
}
206+
]
207+
}
208+
```
209+
210+
4. Multiple relabeling rules (combining different actions):
211+
```json
212+
{
213+
"RELABEL_CONFIGS": [
214+
{
215+
"source_labels": ["__aws_log_type"],
216+
"target_label": "log_type",
217+
"action": "replace",
218+
"regex": "(.*)",
219+
"replacement": "${1}"
220+
},
221+
{
222+
"source_labels": ["__aws_s3_log_lb"],
223+
"target_label": "loadbalancer",
224+
"action": "replace"
225+
},
226+
{
227+
"regex": "__aws_.*",
228+
"action": "labeldrop"
229+
}
230+
]
231+
}
232+
```
233+
234+
### Supported Actions
235+
236+
The following actions are supported, matching Prometheus relabeling capabilities:
237+
238+
- `replace`: Replace a label value with a new value using regex capture groups
239+
- `keep`: Keep entries where labels match the regex (useful for filtering)
240+
- `drop`: Drop entries where labels match the regex (useful for excluding)
241+
- `hashmod`: Set a label to the modulus of a hash of labels (useful for sharding)
242+
- `labelmap`: Copy labels to other labels based on regex matching
243+
- `labeldrop`: Remove labels matching the regex pattern
244+
- `labelkeep`: Keep only labels matching the regex pattern
245+
- `lowercase`: Convert label values to lowercase
246+
- `uppercase`: Convert label values to uppercase
247+
248+
### Configuration Fields
249+
250+
Each relabel configuration supports these fields (all fields are optional except for `action`):
251+
252+
- `source_labels`: List of label names to use as input for the action
253+
- `separator`: String to join source label values (default: ";")
254+
- `target_label`: Label to modify (required for replace and hashmod actions)
255+
- `regex`: Regular expression to match against (defaults to "(.+)" for most actions)
256+
- `replacement`: Replacement pattern for matched regex, supports ${1}, ${2}, etc. for capture groups
257+
- `modulus`: Modulus for hashmod action
258+
- `action`: One of the supported actions listed above
259+
260+
### Important Notes
261+
262+
1. Relabeling is applied after merging extra labels and dropping labels specified by `DROP_LABELS`.
263+
2. If all labels are removed after relabeling, the log entry will be dropped entirely.
264+
3. The relabeling configuration follows the same format as Prometheus's relabel_configs, making it familiar for users of Prometheus.
265+
4. Relabeling rules are processed in order, and each rule can affect the input of subsequent rules.
266+
5. Regular expressions in the `regex` field support full RE2 syntax.
267+
6. For the `replace` action, if the `regex` doesn't match, the target label remains unchanged.
268+
269+
For more details about how relabeling works and advanced use cases, refer to the [Prometheus relabeling blog post](https://grafana.com/blog/2022/03/21/how-relabeling-in-prometheus-works/).
270+
164271
## Limitations
165272

166273
### Promtail labels

‎tools/lambda-promtail/lambda-promtail/eventbridge.go

+1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import (
44
"context"
55
"encoding/json"
66
"fmt"
7+
78
"github.com/aws/aws-lambda-go/events"
89
"github.com/go-kit/log"
910
)

‎tools/lambda-promtail/lambda-promtail/eventbridge_test.go

+3-2
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,12 @@ package main
33
import (
44
"context"
55
"encoding/json"
6+
"os"
7+
"testing"
8+
69
"github.com/aws/aws-lambda-go/events"
710
"github.com/go-kit/log"
811
"github.com/stretchr/testify/require"
9-
"os"
10-
"testing"
1112
)
1213

1314
type testPromtailClient struct{}

‎tools/lambda-promtail/lambda-promtail/main.go

+51-1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,8 @@ import (
1313
"github.com/go-kit/log/level"
1414
"github.com/grafana/dskit/backoff"
1515
"github.com/prometheus/common/model"
16+
prommodel "github.com/prometheus/prometheus/model/labels"
17+
"github.com/prometheus/prometheus/model/relabel"
1618

1719
"github.com/aws/aws-lambda-go/events"
1820
"github.com/aws/aws-lambda-go/lambda"
@@ -38,6 +40,7 @@ var (
3840
dropLabels []model.LabelName
3941
skipTlsVerify bool
4042
printLogLine bool
43+
relabelConfigs []*relabel.Config
4144
)
4245

4346
func setupArguments() {
@@ -106,14 +109,21 @@ func setupArguments() {
106109
printLogLine = false
107110
}
108111
s3Clients = make(map[string]*s3.Client)
112+
113+
// Parse relabel configs from environment variable
114+
if relabelConfigsRaw := os.Getenv("RELABEL_CONFIGS"); relabelConfigsRaw != "" {
115+
if err := json.Unmarshal([]byte(relabelConfigsRaw), &relabelConfigs); err != nil {
116+
panic(fmt.Errorf("failed to parse RELABEL_CONFIGS: %v", err))
117+
}
118+
}
109119
}
110120

111121
func parseExtraLabels(extraLabelsRaw string, omitPrefix bool) (model.LabelSet, error) {
112122
prefix := "__extra_"
113123
if omitPrefix {
114124
prefix = ""
115125
}
116-
var extractedLabels = model.LabelSet{}
126+
extractedLabels := model.LabelSet{}
117127
extraLabelsSplit := strings.Split(extraLabelsRaw, ",")
118128

119129
if len(extraLabelsRaw) < 1 {
@@ -151,13 +161,53 @@ func getDropLabels() ([]model.LabelName, error) {
151161
return result, nil
152162
}
153163

164+
func applyRelabelConfigs(labels model.LabelSet) model.LabelSet {
165+
if len(relabelConfigs) == 0 {
166+
return labels
167+
}
168+
169+
// Convert model.LabelSet to prommodel.Labels
170+
promLabels := make([]prommodel.Label, 0, len(labels))
171+
for name, value := range labels {
172+
promLabels = append(promLabels, prommodel.Label{
173+
Name: string(name),
174+
Value: string(value),
175+
})
176+
}
177+
178+
// Sort labels as required by Process
179+
promLabels = prommodel.New(promLabels...)
180+
181+
// Apply relabeling
182+
processedLabels, keep := relabel.Process(promLabels, relabelConfigs...)
183+
if !keep {
184+
return model.LabelSet{}
185+
}
186+
187+
// Convert back to model.LabelSet
188+
result := make(model.LabelSet)
189+
for _, l := range processedLabels {
190+
result[model.LabelName(l.Name)] = model.LabelValue(l.Value)
191+
}
192+
193+
return result
194+
}
195+
154196
func applyLabels(labels model.LabelSet) model.LabelSet {
155197
finalLabels := labels.Merge(extraLabels)
156198

157199
for _, dropLabel := range dropLabels {
158200
delete(finalLabels, dropLabel)
159201
}
160202

203+
// Apply relabeling after merging extra labels and dropping labels
204+
finalLabels = applyRelabelConfigs(finalLabels)
205+
206+
// Skip entries with no labels after relabeling
207+
if len(finalLabels) == 0 {
208+
return nil
209+
}
210+
161211
return finalLabels
162212
}
163213

‎tools/lambda-promtail/lambda-promtail/promtail.go

+5
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,11 @@ func newBatch(ctx context.Context, pClient Client, entries ...entry) (*batch, er
5858
}
5959

6060
func (b *batch) add(ctx context.Context, e entry) error {
61+
// Skip entries with no labels (filtered out by relabeling)
62+
if e.labels == nil {
63+
return nil
64+
}
65+
6166
labels := labelsMapToString(e.labels, reservedLabelTenantID)
6267
stream, ok := b.streams[labels]
6368
if !ok {

‎tools/lambda-promtail/lambda-promtail/s3.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -290,8 +290,8 @@ func processS3Event(ctx context.Context, ev *events.S3Event, pc Client, log *log
290290
}
291291
obj, err := s3Client.GetObject(ctx,
292292
&s3.GetObjectInput{
293-
Bucket: aws.String(labels["bucket"]),
294-
Key: aws.String(labels["key"]),
293+
Bucket: aws.String(labels["bucket"]),
294+
Key: aws.String(labels["key"]),
295295
})
296296
if err != nil {
297297
return fmt.Errorf("failed to get object %s from bucket %s, %s", labels["key"], labels["bucket"], err)

0 commit comments

Comments
 (0)