2024-02-16 17:09:58 +02:00
|
|
|
// Copyright The OpenTelemetry Authors
|
2024-02-29 08:05:28 +02:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
package log // import "go.opentelemetry.io/otel/log"
|
|
|
|
|
2024-02-25 20:48:32 +02:00
|
|
|
import (
|
|
|
|
"slices"
|
|
|
|
"time"
|
|
|
|
)
|
2024-02-16 17:09:58 +02:00
|
|
|
|
2024-02-19 15:08:41 +02:00
|
|
|
// attributesInlineCount is the number of attributes that are efficiently
|
|
|
|
// stored in an array within a Record. This value is borrowed from slog which
|
|
|
|
// performed a quantitative survey of log library use and found this value to
|
|
|
|
// cover 95% of all use-cases (https://go.dev/blog/slog#performance).
|
|
|
|
const attributesInlineCount = 5
|
|
|
|
|
2024-02-16 17:09:58 +02:00
|
|
|
// Record represents a log record.
|
2024-02-19 15:08:41 +02:00
|
|
|
type Record struct {
|
2024-09-26 08:32:36 +02:00
|
|
|
// Ensure forward compatibility by explicitly making this not comparable.
|
|
|
|
noCmp [0]func() //nolint: unused // This is indeed used.
|
|
|
|
|
2024-02-19 15:08:41 +02:00
|
|
|
timestamp time.Time
|
|
|
|
observedTimestamp time.Time
|
|
|
|
severity Severity
|
|
|
|
severityText string
|
|
|
|
body Value
|
|
|
|
|
|
|
|
// The fields below are for optimizing the implementation of Attributes and
|
|
|
|
// AddAttributes. This design is borrowed from the slog Record type:
|
|
|
|
// https://cs.opensource.google/go/go/+/refs/tags/go1.22.0:src/log/slog/record.go;l=20
|
|
|
|
|
|
|
|
// Allocation optimization: an inline array sized to hold
|
|
|
|
// the majority of log calls (based on examination of open-source
|
|
|
|
// code). It holds the start of the list of attributes.
|
|
|
|
front [attributesInlineCount]KeyValue
|
|
|
|
|
|
|
|
// The number of attributes in front.
|
|
|
|
nFront int
|
|
|
|
|
|
|
|
// The list of attributes except for those in front.
|
|
|
|
// Invariants:
|
|
|
|
// - len(back) > 0 if nFront == len(front)
|
|
|
|
// - Unused array elements are zero-ed. Used to detect mistakes.
|
|
|
|
back []KeyValue
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// Timestamp returns the time when the log record occurred.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) Timestamp() time.Time {
|
|
|
|
return r.timestamp
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// SetTimestamp sets the time when the log record occurred.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) SetTimestamp(t time.Time) {
|
|
|
|
r.timestamp = t
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// ObservedTimestamp returns the time when the log record was observed.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) ObservedTimestamp() time.Time {
|
|
|
|
return r.observedTimestamp
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// SetObservedTimestamp sets the time when the log record was observed.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) SetObservedTimestamp(t time.Time) {
|
|
|
|
r.observedTimestamp = t
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// Severity returns the [Severity] of the log record.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) Severity() Severity {
|
|
|
|
return r.severity
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// SetSeverity sets the [Severity] level of the log record.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) SetSeverity(level Severity) {
|
|
|
|
r.severity = level
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// SeverityText returns severity (also known as log level) text. This is the
|
|
|
|
// original string representation of the severity as it is known at the source.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) SeverityText() string {
|
|
|
|
return r.severityText
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// SetSeverityText sets severity (also known as log level) text. This is the
|
|
|
|
// original string representation of the severity as it is known at the source.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) SetSeverityText(text string) {
|
|
|
|
r.severityText = text
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// Body returns the body of the log record.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) Body() Value {
|
|
|
|
return r.body
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// SetBody sets the body of the log record.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) SetBody(v Value) {
|
|
|
|
r.body = v
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// WalkAttributes walks all attributes the log record holds by calling f for
|
|
|
|
// each on each [KeyValue] in the [Record]. Iteration stops if f returns false.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) WalkAttributes(f func(KeyValue) bool) {
|
|
|
|
for i := 0; i < r.nFront; i++ {
|
|
|
|
if !f(r.front[i]) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, a := range r.back {
|
|
|
|
if !f(a) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-02-16 17:09:58 +02:00
|
|
|
|
|
|
|
// AddAttributes adds attributes to the log record.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) AddAttributes(attrs ...KeyValue) {
|
|
|
|
var i int
|
|
|
|
for i = 0; i < len(attrs) && r.nFront < len(r.front); i++ {
|
|
|
|
a := attrs[i]
|
|
|
|
r.front[r.nFront] = a
|
|
|
|
r.nFront++
|
|
|
|
}
|
|
|
|
|
2024-02-25 20:48:32 +02:00
|
|
|
r.back = slices.Grow(r.back, len(attrs[i:]))
|
2024-02-19 15:08:41 +02:00
|
|
|
r.back = append(r.back, attrs[i:]...)
|
|
|
|
}
|
|
|
|
|
2024-02-16 17:09:58 +02:00
|
|
|
// AttributesLen returns the number of attributes in the log record.
|
2024-02-19 15:08:41 +02:00
|
|
|
func (r *Record) AttributesLen() int {
|
|
|
|
return r.nFront + len(r.back)
|
|
|
|
}
|