2023-04-19 08:28:09 +03:00
|
|
|
// Package yamlw implements output writers for YAML.
|
|
|
|
package yamlw
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"io"
|
|
|
|
|
|
|
|
"github.com/fatih/color"
|
|
|
|
goccy "github.com/goccy/go-yaml"
|
|
|
|
"github.com/goccy/go-yaml/lexer"
|
|
|
|
"github.com/neilotoole/sq/libsq/core/errz"
|
|
|
|
|
|
|
|
"github.com/goccy/go-yaml/printer"
|
|
|
|
"github.com/neilotoole/sq/cli/output"
|
|
|
|
)
|
|
|
|
|
2023-05-19 17:24:18 +03:00
|
|
|
// MarshalToString renders v to a string.
|
|
|
|
func MarshalToString(pr *output.Printing, v any) (string, error) {
|
|
|
|
p := newPrinter(pr)
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
if err := writeYAML(buf, p, v); err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return buf.String(), nil
|
|
|
|
}
|
|
|
|
|
2023-04-19 08:28:09 +03:00
|
|
|
// writeYAML prints a YAML representation of v to out, using specs
|
2023-04-22 06:36:32 +03:00
|
|
|
// from pr.
|
2023-05-05 17:32:50 +03:00
|
|
|
func writeYAML(out io.Writer, p printer.Printer, v any) error {
|
2023-04-19 08:28:09 +03:00
|
|
|
b, err := goccy.Marshal(v)
|
|
|
|
if err != nil {
|
|
|
|
return errz.Err(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
tokens := lexer.Tokenize(string(b))
|
|
|
|
|
|
|
|
_, err = out.Write([]byte(p.PrintTokens(tokens) + "\n"))
|
|
|
|
return errz.Err(err)
|
|
|
|
}
|
|
|
|
|
2023-04-22 06:36:32 +03:00
|
|
|
func newPrinter(pr *output.Printing) printer.Printer {
|
2023-04-19 08:28:09 +03:00
|
|
|
var p printer.Printer
|
|
|
|
p.LineNumber = false
|
2023-05-11 05:03:45 +03:00
|
|
|
if pr.IsMonochrome() {
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
2023-04-19 08:28:09 +03:00
|
|
|
p.Bool = func() *printer.Property {
|
|
|
|
return &printer.Property{
|
2023-04-22 06:36:32 +03:00
|
|
|
Prefix: formatColor(pr.Bool),
|
2023-04-19 08:28:09 +03:00
|
|
|
Suffix: reset,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
p.Number = func() *printer.Property {
|
|
|
|
return &printer.Property{
|
2023-04-22 06:36:32 +03:00
|
|
|
Prefix: formatColor(pr.Number),
|
2023-04-19 08:28:09 +03:00
|
|
|
Suffix: reset,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
p.MapKey = func() *printer.Property {
|
|
|
|
return &printer.Property{
|
2023-04-22 06:36:32 +03:00
|
|
|
Prefix: formatColor(pr.Key),
|
2023-04-19 08:28:09 +03:00
|
|
|
Suffix: reset,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
p.Anchor = func() *printer.Property {
|
|
|
|
return &printer.Property{
|
2023-04-22 06:36:32 +03:00
|
|
|
Prefix: formatColor(pr.Faint),
|
2023-04-19 08:28:09 +03:00
|
|
|
Suffix: reset,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
p.Alias = func() *printer.Property {
|
|
|
|
return &printer.Property{
|
2023-04-22 06:36:32 +03:00
|
|
|
Prefix: formatColor(pr.Faint),
|
2023-04-19 08:28:09 +03:00
|
|
|
Suffix: reset,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
p.String = func() *printer.Property {
|
|
|
|
return &printer.Property{
|
2023-04-22 06:36:32 +03:00
|
|
|
Prefix: formatColor(pr.String),
|
2023-04-19 08:28:09 +03:00
|
|
|
Suffix: reset,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
|
|
|
const reset = "\x1b[0m"
|
|
|
|
|
|
|
|
// formatColor is a hack to extract the escape chars from
|
|
|
|
// a color.
|
|
|
|
func formatColor(c *color.Color) string {
|
|
|
|
if c == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make a copy because the pkg-level color.NoColor could be false.
|
|
|
|
c2 := *c
|
|
|
|
c2.EnableColor()
|
|
|
|
|
|
|
|
b := []byte(c2.Sprint(" "))
|
|
|
|
i := bytes.IndexByte(b, ' ')
|
|
|
|
if i <= 0 {
|
|
|
|
// Shouldn't happen
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
return string(b[:i])
|
|
|
|
}
|