Skip to content

Commit 92ab7df

Browse files
authored
[mdatagen] Support logs (#12659)
<!--Ex. Fixing a bug - Describe the bug and how this fixes the issue. Ex. Adding a feature - Explain what this achieves.--> ### Description This PR introduces the foundational changes necessary for supporting `logs` data in the `mdatagen` tool. #### mdatagen files for logs This update includes the generation of `generated_logs.go` and `generated_logs_test.go`. These files are specifically for receiver and scraper components that support `logs` data, enabling initial log handling capabilities. #### Introducing LogsBuilder This PR introduced a `LogsBuilder` similar to the existing `MetricsBuilder`. It provides a structured way to manage log data with the following functions: 1. `Emit(...ResourceLogsOption)` Similar to `Emit` function in MetricsBuilder 2. `EmitForResource(...ResourceLogsOption)` Similar to `EmitForResource` function in MetricsBuilder 3. `AppendLogRecord(plog.LogRecord)` This function appends a log record to an internal buffer. The buffered log records are used to construct a `ScopeLog` when the `Emit()` or `EmitForResource()` functions are called. Scope name and version are automatically generated. #### Next steps * Add more test cases to LogsBuilder (e.g. reading test configs) * Add LogsBuilderConfig to read logs property in `metadata.yml` to send structured logs * Update receivers in contrib to use LogsBuilder. #### Example usage: ``` lb := NewLogsBuilder(settings) res := pcommon.NewResource() res.Attributes().PutStr("region", "us-west-1") // append the first log record lr := plog.NewLogRecord() lr.SetTimestamp(pcommon.NewTimestampFromTime(time.Now())) lr.Attributes().PutStr("type", "log") lr.Body().SetStr("the first log record") // append the second log record lr2 := plog.NewLogRecord() lr2.SetTimestamp(pcommon.NewTimestampFromTime(time.Now())) lr2.Attributes().PutStr("type", "event") lr2.Body().SetStr("the second log record") lb.AppendLogRecord(lr) lb.AppendLogRecord(lr2) logs := lb.Emit(WithLogsResource(res)) ``` #### Example output: ``` resourceLogs: - resource: attributes: - key: region value: stringValue: us-west-1 scopeLogs: - logRecords: - attributes: - key: type value: stringValue: log body: stringValue: the first log record spanId: "" timeUnixNano: "1742291226022739000" traceId: "" - attributes: - key: type value: stringValue: event body: stringValue: the second log record spanId: "" timeUnixNano: "1742291226022739000" traceId: "" scope: name: github.com/open-telemetry/opentelemetry-collector-contrib/receiver/awscloudwatchreceiver version: latest ``` <!-- Issue number if applicable --> ### Link to tracking issue Part of #12571 <!--Describe what testing was performed and which tests were added.--> ### Testing Added <!--Describe the documentation added.--> ### Documentation Added <!--Please delete paragraphs that you did not use before submitting.-->
1 parent 63a22b2 commit 92ab7df

File tree

14 files changed

+707
-2
lines changed

14 files changed

+707
-2
lines changed

.chloggen/mdatagen-logs.yaml

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
# Use this changelog template to create an entry for release notes.
2+
3+
# One of 'breaking', 'deprecation', 'new_component', 'enhancement', 'bug_fix'
4+
change_type: enhancement
5+
6+
# The name of the component, or a single word describing the area of concern, (e.g. otlpreceiver)
7+
component: cmd/mdatagen
8+
9+
# A brief description of the change. Surround your text with quotes ("") if it needs to start with a backtick (`).
10+
note: Add the foundational changes necessary for supporting logs data in `mdatagen`
11+
12+
# One or more tracking issues or pull requests related to the change
13+
issues: [12571]
14+
15+
# (Optional) One or more lines of additional information to render under the primary note.
16+
# These lines will be padded with 2 spaces and then inserted directly into the document.
17+
# Use pipe (|) for multiline entries.
18+
subtext:
19+
20+
# Optional: The change log or logs in which this entry should be included.
21+
# e.g. '[user]' or '[user, api]'
22+
# Include 'user' if the change is relevant to end users.
23+
# Include 'api' if there is a change to a library API.
24+
# Default: '[user]'
25+
change_logs: [api]

cmd/mdatagen/internal/command.go

+6
Original file line numberDiff line numberDiff line change
@@ -183,6 +183,12 @@ func run(ymlPath string) error {
183183
toGenerate[filepath.Join(tmplDir, "metrics_test.go.tmpl")] = filepath.Join(codeDir, "generated_metrics_test.go")
184184
}
185185

186+
if md.supportsSignal("logs") &&
187+
(md.Status.Class == "receiver" || md.Status.Class == "scraper") {
188+
toGenerate[filepath.Join(tmplDir, "logs.go.tmpl")] = filepath.Join(codeDir, "generated_logs.go")
189+
toGenerate[filepath.Join(tmplDir, "logs_test.go.tmpl")] = filepath.Join(codeDir, "generated_logs_test.go")
190+
}
191+
186192
// If at least one file to generate, will need the codeDir
187193
if len(toGenerate) > 0 {
188194
if err = os.MkdirAll(codeDir, 0o700); err != nil {

cmd/mdatagen/internal/embedded_templates_test.go

+2
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,8 @@ func TestEnsureTemplatesLoaded(t *testing.T) {
2525
path.Join(rootDir, "documentation.md.tmpl"): {},
2626
path.Join(rootDir, "metrics.go.tmpl"): {},
2727
path.Join(rootDir, "metrics_test.go.tmpl"): {},
28+
path.Join(rootDir, "logs.go.tmpl"): {},
29+
path.Join(rootDir, "logs_test.go.tmpl"): {},
2830
path.Join(rootDir, "resource.go.tmpl"): {},
2931
path.Join(rootDir, "resource_test.go.tmpl"): {},
3032
path.Join(rootDir, "config.go.tmpl"): {},

cmd/mdatagen/internal/metadata.go

+4
Original file line numberDiff line numberDiff line change
@@ -136,6 +136,10 @@ func (md *Metadata) validateAttributes(usedAttrs map[AttributeName]bool) error {
136136
}
137137

138138
func (md *Metadata) supportsSignal(signal string) bool {
139+
if md.Status == nil {
140+
return false
141+
}
142+
139143
for _, signals := range md.Status.Stability {
140144
for _, s := range signals {
141145
if s == signal {

cmd/mdatagen/internal/metadata_test.go

+5
Original file line numberDiff line numberDiff line change
@@ -158,6 +158,11 @@ func TestValidateMetricDuplicates(t *testing.T) {
158158
}
159159
}
160160

161+
func TestSupportsSignal(t *testing.T) {
162+
md := Metadata{}
163+
assert.False(t, md.supportsSignal("logs"))
164+
}
165+
161166
func contains(r string, rs []string) bool {
162167
for _, s := range rs {
163168
if s == r {

cmd/mdatagen/internal/samplereceiver/internal/metadata/generated_logs.go

+95
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

cmd/mdatagen/internal/samplereceiver/internal/metadata/generated_logs_test.go

+75
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,105 @@
1+
// Code generated by mdatagen. DO NOT EDIT.
2+
3+
package {{ .Package }}
4+
5+
import (
6+
"go.opentelemetry.io/collector/component"
7+
"go.opentelemetry.io/collector/pdata/pcommon"
8+
"go.opentelemetry.io/collector/pdata/plog"
9+
{{- if or isReceiver isScraper }}
10+
"go.opentelemetry.io/collector/{{ .Status.Class }}"
11+
{{- end }}
12+
{{- if .SemConvVersion }}
13+
conventions "go.opentelemetry.io/collector/semconv/v{{ .SemConvVersion }}"
14+
{{- end }}
15+
)
16+
17+
// LogsBuilder provides an interface for scrapers to report logs while taking care of all the transformations
18+
// required to produce log representation defined in metadata and user config.
19+
type LogsBuilder struct {
20+
logsBuffer plog.Logs
21+
logRecordsBuffer plog.LogRecordSlice
22+
buildInfo component.BuildInfo // contains version information.
23+
}
24+
25+
// LogBuilderOption applies changes to default logs builder.
26+
type LogBuilderOption interface {
27+
apply(*LogsBuilder)
28+
}
29+
30+
{{- if isReceiver }}
31+
func NewLogsBuilder(settings {{ .Status.Class }}.Settings) *LogsBuilder {
32+
lb := &LogsBuilder{
33+
logsBuffer: plog.NewLogs(),
34+
logRecordsBuffer: plog.NewLogRecordSlice(),
35+
buildInfo: settings.BuildInfo,
36+
}
37+
38+
return lb
39+
}
40+
{{- end }}
41+
42+
{{- if .ResourceAttributes }}
43+
// NewResourceBuilder returns a new resource builder that should be used to build a resource associated with for the emitted logs.
44+
func (lb *LogsBuilder) NewResourceBuilder() *ResourceBuilder {
45+
return NewResourceBuilder(ResourceAttributesConfig{})
46+
}
47+
{{- end }}
48+
49+
// ResourceLogsOption applies changes to provided resource logs.
50+
type ResourceLogsOption interface {
51+
apply(plog.ResourceLogs)
52+
}
53+
54+
type resourceLogsOptionFunc func(plog.ResourceLogs)
55+
56+
func (rlof resourceLogsOptionFunc) apply(rl plog.ResourceLogs) {
57+
rlof(rl)
58+
}
59+
60+
// WithLogsResource sets the provided resource on the emitted ResourceLogs.
61+
// It's recommended to use ResourceBuilder to create the resource.
62+
func WithLogsResource(res pcommon.Resource) ResourceLogsOption {
63+
return resourceLogsOptionFunc(func(rl plog.ResourceLogs) {
64+
res.CopyTo(rl.Resource())
65+
})
66+
}
67+
68+
// AppendLogRecord adds a log record to the logs builder.
69+
func (lb *LogsBuilder) AppendLogRecord(lr plog.LogRecord) {
70+
lr.MoveTo(lb.logRecordsBuffer.AppendEmpty())
71+
}
72+
73+
// EmitForResource saves all the generated logs under a new resource and updates the internal state to be ready for
74+
// recording another set of log records as part of another resource. This function can be helpful when one scraper
75+
// needs to emit logs from several resources. Otherwise calling this function is not required,
76+
// just `Emit` function can be called instead.
77+
// Resource attributes should be provided as ResourceLogsOption arguments.
78+
func (lb *LogsBuilder) EmitForResource(options ...ResourceLogsOption) {
79+
rl := lb.logsBuffer.ResourceLogs().AppendEmpty()
80+
{{- if .SemConvVersion }}
81+
rl.SetSchemaUrl(conventions.SchemaURL)
82+
{{- end }}
83+
ils := rl.ScopeLogs().AppendEmpty()
84+
ils.Scope().SetName(ScopeName)
85+
ils.Scope().SetVersion(lb.buildInfo.Version)
86+
87+
for _, op := range options {
88+
op.apply(rl)
89+
}
90+
91+
if lb.logRecordsBuffer.Len() > 0 {
92+
lb.logRecordsBuffer.MoveAndAppendTo(ils.LogRecords())
93+
lb.logRecordsBuffer = plog.NewLogRecordSlice()
94+
}
95+
}
96+
97+
// Emit returns all the logs accumulated by the logs builder and updates the internal state to be ready for
98+
// recording another set of logs. This function will be responsible for applying all the transformations required to
99+
// produce logs representation defined in metadata and user config.
100+
func (lb *LogsBuilder) Emit(options ...ResourceLogsOption) plog.Logs {
101+
lb.EmitForResource(options...)
102+
logs := lb.logsBuffer
103+
lb.logsBuffer = plog.NewLogs()
104+
return logs
105+
}

0 commit comments

Comments
 (0)