-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathtologfmt.go
87 lines (82 loc) · 2.02 KB
/
tologfmt.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
package main
import (
"context"
"fmt"
"github.com/ainvaltin/nu-plugin"
"github.com/ainvaltin/nu-plugin/types"
"github.com/oderwat/nu_plugin_logfmt/logfmt"
)
func toLogFmt() *nu.Command {
return &nu.Command{
Signature: nu.PluginSignature{
Name: "to logfmt",
Category: "Formats",
Desc: `Convert Nushell Value to 'logfmt' format.`,
SearchTerms: []string{"logfmt", "slog", "logging"},
InputOutputTypes: []nu.InOutTypes{{types.Any(), types.String()}},
AllowMissingExamples: true,
},
Examples: nu.Examples{
{
Description: `Convert an Nu record to a logmt string`,
Example: `{ "msg": "Hello World!", "Lang": { "Go": true, "Rust": false } } | to logfmt`,
Result: &nu.Value{Value: `msg="Hello World!" Lang.Go=true Lang.Rust=false`},
},
},
OnRun: toLogFmtHandler,
}
}
func toLogFmtHandler(ctx context.Context, call *nu.ExecCommand) error {
switch in := call.Input.(type) {
case nil:
return nil
case nu.Value:
v, err := toLogfmtValue(in)
if err != nil {
return err
}
return call.ReturnValue(ctx, v)
case <-chan nu.Value:
out, err := call.ReturnListStream(ctx)
if err != nil {
return err
}
defer close(out)
for v := range in {
v, err := toLogfmtValue(v)
if err != nil {
return err
}
out <- v
}
return nil
default:
return fmt.Errorf("unsupported input type %T", call.Input)
}
}
func toLogfmtValue(v nu.Value) (nu.Value, error) {
var buf []byte
if data, ok := fromValue(v).(map[string]interface{}); ok {
buf = []byte(logfmt.Encode(data))
} else if data, ok := fromValue(v).([]interface{}); ok {
buf = []byte(logfmt.Encode(data))
}
return nu.Value{Value: string(buf)}, nil
}
func fromValue(v nu.Value) any {
switch vt := v.Value.(type) {
case []nu.Value:
lst := make([]any, len(vt))
for i := 0; i < len(vt); i++ {
lst[i] = fromValue(vt[i])
}
return lst
case nu.Record:
rec := map[string]any{}
for k, v := range vt {
rec[k] = fromValue(v)
}
return rec
}
return v.Value
}