Skip to content

Commit

Permalink
fix: logs and header parsing
Browse files Browse the repository at this point in the history
  • Loading branch information
krzko committed Oct 1, 2024
1 parent 98d8751 commit 05f8bc3
Show file tree
Hide file tree
Showing 3 changed files with 215 additions and 125 deletions.
145 changes: 107 additions & 38 deletions internal/cli/logs.go
Original file line number Diff line number Diff line change
@@ -1,73 +1,142 @@
package cli

import (
"errors"
"fmt"
"strings"
"time"

"github.com/krzko/otelgen/internal/logs"
"github.com/urfave/cli/v2"
"go.uber.org/zap"
"go.uber.org/zap/zapcore"
)

func genLogsCommand() *cli.Command {
return &cli.Command{
Name: "logs",
Usage: "Generate logs",
Aliases: []string{"l"},
Flags: []cli.Flag{
&cli.IntFlag{
Name: "number",
Aliases: []string{"n"},
Usage: "number of log events to generate",
Value: 10,
Subcommands: []*cli.Command{
{
Name: "single",
Usage: "generate a single log event",
Aliases: []string{"s"},
Action: func(c *cli.Context) error {
return generateLogs(c, true)
},
},
&cli.IntFlag{
Name: "workers",
Aliases: []string{"w"},
Usage: "number of workers (goroutines) to run",
Value: 1,
},
&cli.StringFlag{
Name: "severity-text",
Aliases: []string{"st"},
Usage: "Severity text of the log (e.g., Trace, Debug, Info, Warn, Error, Fatal)",
Value: "Info",
},
&cli.IntFlag{
Name: "severity-number",
Aliases: []string{"sn"},
Usage: "Severity number of the log, range from 1 to 24 (inclusive)",
Value: 9,
{
Name: "multi",
Usage: "generate multiple logs",
Aliases: []string{"m"},
Flags: []cli.Flag{
&cli.IntFlag{
Name: "number",
Aliases: []string{"n"},
Usage: "number of log events to generate",
Value: 0, // Default to 0, which means indefinite
},
&cli.IntFlag{
Name: "workers",
Aliases: []string{"w"},
Usage: "number of workers (goroutines) to run",
Value: 1,
},
&cli.IntFlag{
Name: "duration",
Aliases: []string{"d"},
Usage: "duration in seconds for how long to generate logs",
},
},
Action: func(c *cli.Context) error {
return generateLogs(c, false)
},
},
},
Action: func(c *cli.Context) error {
return generateLogs(c)
},
}
}

func generateLogs(c *cli.Context) error {
func generateLogs(c *cli.Context, isSingle bool) error {
if c.String("otel-exporter-otlp-endpoint") == "" {
return errors.New("'otel-exporter-otlp-endpoint' must be set")
}

logsCfg := &logs.Config{
WorkerCount: c.Int("workers"),
NumLogs: c.Int("number"),
ServiceName: c.String("service-name"),
Endpoint: c.String("otel-exporter-otlp-endpoint"),
Insecure: c.Bool("insecure"),
UseHTTP: c.String("protocol") == "http",
Rate: c.Float64("rate"),
TotalDuration: c.Duration("duration"),
SeverityText: c.String("severity-text"),
SeverityNumber: int32(c.Int("severity-number")),
Endpoint: c.String("otel-exporter-otlp-endpoint"),
ServiceName: c.String("service-name"),
Insecure: c.Bool("insecure"),
UseHTTP: c.String("protocol") == "http",
}

// Handle single log generation
if isSingle {
logsCfg.NumLogs = 1
logsCfg.WorkerCount = 1
} else {
logsCfg.NumLogs = c.Int("number")
logsCfg.WorkerCount = c.Int("workers")
logsCfg.TotalDuration = time.Duration(c.Int("duration") * int(time.Second))
logsCfg.Rate = c.Float64("rate")

// If neither `NumLogs` nor `TotalDuration` is set, default to indefinite generation
if logsCfg.NumLogs == 0 && logsCfg.TotalDuration == 0 {
logsCfg.NumLogs = 0 // Indefinite
logsCfg.TotalDuration = 0
}
}

logger, err := zap.NewDevelopment()
// Parse headers
headers := make(map[string]string)
for _, h := range c.StringSlice("header") {
kv := strings.SplitN(h, "=", 2)
if len(kv) != 2 {
return fmt.Errorf("header format must be 'key=value'")
}
headers[kv[0]] = kv[1]
}
logsCfg.Headers = headers

// Set up logger without stack trace for warnings
logger, err := newCustomLogger()
if err != nil {
return fmt.Errorf("failed to create logger: %w", err)
}

// Run the log generation
if err := logs.Run(logsCfg, logger); err != nil {
logger.Error("failed to run logs generation", zap.Error(err))
return err
}

return nil
}

func newCustomLogger() (*zap.Logger, error) {
cfg := zap.Config{
Level: zap.NewAtomicLevelAt(zap.DebugLevel),
Development: true,
Sampling: nil,
Encoding: "json", // or "console" if you prefer
EncoderConfig: zapcore.EncoderConfig{
MessageKey: "message",
LevelKey: "level",
TimeKey: "time",
NameKey: "logger",
CallerKey: "caller",
StacktraceKey: "stacktrace", // This will hold stack trace information
LineEnding: zapcore.DefaultLineEnding,
EncodeLevel: zapcore.CapitalLevelEncoder,
EncodeTime: zapcore.ISO8601TimeEncoder,
EncodeCaller: zapcore.ShortCallerEncoder,
},
OutputPaths: []string{"stdout"},
ErrorOutputPaths: []string{"stderr"},
}

// Disable stacktrace for warnings and below
cfg.EncoderConfig.StacktraceKey = ""
cfg.Level = zap.NewAtomicLevelAt(zap.DebugLevel)

return cfg.Build()
}
39 changes: 39 additions & 0 deletions internal/logs/config.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
package logs

import (
"flag"
"fmt"
"strings"
"time"
)

type Config struct {
WorkerCount int
NumLogs int
Rate float64
TotalDuration time.Duration
ServiceName string

// OTLP config
Endpoint string
Insecure bool
UseHTTP bool
Headers HeaderValue
}

type HeaderValue map[string]string

var _ flag.Value = (*HeaderValue)(nil)

func (v *HeaderValue) String() string {
return ""
}

func (v *HeaderValue) Set(s string) error {
kv := strings.SplitN(s, "=", 2)
if len(kv) != 2 {
return fmt.Errorf("value should be of the format key=value")
}
(*v)[kv[0]] = kv[1]
return nil
}
Loading

0 comments on commit 05f8bc3

Please sign in to comment.