yq/pkg/yqlib/printer.go

205 lines
5.7 KiB
Go
Raw Permalink Normal View History

2020-11-03 23:48:43 +00:00
package yqlib
import (
"bufio"
"container/list"
2021-07-25 08:08:33 +00:00
"fmt"
2020-11-03 23:48:43 +00:00
"io"
2021-07-19 09:52:51 +00:00
"strings"
2020-11-03 23:48:43 +00:00
2020-11-29 09:25:47 +00:00
yaml "gopkg.in/yaml.v3"
2020-11-03 23:48:43 +00:00
)
type Printer interface {
PrintResults(matchingNodes *list.List) error
2020-11-30 05:35:21 +00:00
PrintedAnything() bool
2021-07-18 02:28:46 +00:00
//e.g. when given a front-matter doc, like jekyll
SetAppendix(reader io.Reader)
2020-11-03 23:48:43 +00:00
}
2021-07-25 08:08:33 +00:00
type PrinterOutputFormat uint32
const (
YamlOutputFormat = 1 << iota
JsonOutputFormat
PropsOutputFormat
)
func OutputFormatFromString(format string) (PrinterOutputFormat, error) {
switch format {
2021-08-20 05:46:33 +00:00
case "yaml", "y":
2021-07-25 08:08:33 +00:00
return YamlOutputFormat, nil
2021-08-20 05:46:33 +00:00
case "json", "j":
2021-07-25 08:08:33 +00:00
return JsonOutputFormat, nil
2021-08-20 05:46:33 +00:00
case "props", "p":
2021-07-25 08:08:33 +00:00
return PropsOutputFormat, nil
default:
return 0, fmt.Errorf("Unknown fromat '%v' please use [yaml|json|props]", format)
}
}
2020-11-03 23:48:43 +00:00
type resultsPrinter struct {
2021-07-25 08:08:33 +00:00
outputFormat PrinterOutputFormat
2020-11-03 23:48:43 +00:00
unwrapScalar bool
colorsEnabled bool
indent int
printDocSeparators bool
2020-11-13 02:19:54 +00:00
writer io.Writer
firstTimePrinting bool
2020-11-16 01:09:57 +00:00
previousDocIndex uint
2020-12-15 03:33:50 +00:00
previousFileIndex int
2020-11-30 05:35:21 +00:00
printedMatches bool
2021-01-12 23:04:52 +00:00
treeNavigator DataTreeNavigator
2021-07-18 02:28:46 +00:00
appendixReader io.Reader
2020-11-03 23:48:43 +00:00
}
2021-07-25 08:08:33 +00:00
func NewPrinter(writer io.Writer, outputFormat PrinterOutputFormat, unwrapScalar bool, colorsEnabled bool, indent int, printDocSeparators bool) Printer {
2020-11-13 02:19:54 +00:00
return &resultsPrinter{
writer: writer,
2021-07-25 08:08:33 +00:00
outputFormat: outputFormat,
2020-11-13 02:19:54 +00:00
unwrapScalar: unwrapScalar,
colorsEnabled: colorsEnabled,
indent: indent,
2021-07-25 08:08:33 +00:00
printDocSeparators: outputFormat == YamlOutputFormat && printDocSeparators,
2020-11-13 02:19:54 +00:00
firstTimePrinting: true,
2021-01-12 23:04:52 +00:00
treeNavigator: NewDataTreeNavigator(),
2020-11-13 02:19:54 +00:00
}
2020-11-03 23:48:43 +00:00
}
2021-07-18 02:28:46 +00:00
func (p *resultsPrinter) SetAppendix(reader io.Reader) {
p.appendixReader = reader
}
2020-11-30 05:35:21 +00:00
func (p *resultsPrinter) PrintedAnything() bool {
return p.printedMatches
}
2020-11-03 23:48:43 +00:00
func (p *resultsPrinter) printNode(node *yaml.Node, writer io.Writer) error {
2020-11-30 05:35:21 +00:00
p.printedMatches = p.printedMatches || (node.Tag != "!!null" &&
(node.Tag != "!!bool" || node.Value != "false"))
2020-11-03 23:48:43 +00:00
var encoder Encoder
2021-07-25 08:08:33 +00:00
if node.Kind == yaml.ScalarNode && p.unwrapScalar && p.outputFormat == YamlOutputFormat {
2020-11-03 23:48:43 +00:00
return p.writeString(writer, node.Value+"\n")
}
2021-07-25 08:08:33 +00:00
if p.outputFormat == JsonOutputFormat {
2020-11-03 23:48:43 +00:00
encoder = NewJsonEncoder(writer, p.indent)
2021-07-25 08:08:33 +00:00
} else if p.outputFormat == PropsOutputFormat {
encoder = NewPropertiesEncoder(writer)
2020-11-03 23:48:43 +00:00
} else {
encoder = NewYamlEncoder(writer, p.indent, p.colorsEnabled)
}
return encoder.Encode(node)
}
func (p *resultsPrinter) writeString(writer io.Writer, txt string) error {
_, errorWriting := writer.Write([]byte(txt))
return errorWriting
}
2020-11-29 09:25:47 +00:00
func (p *resultsPrinter) safelyFlush(writer *bufio.Writer) {
if err := writer.Flush(); err != nil {
log.Error("Error flushing writer!")
log.Error(err.Error())
}
}
func (p *resultsPrinter) PrintResults(matchingNodes *list.List) error {
2020-11-16 01:09:57 +00:00
log.Debug("PrintResults for %v matches", matchingNodes.Len())
2021-07-25 08:08:33 +00:00
if p.outputFormat != YamlOutputFormat {
explodeOp := Operation{OperationType: explodeOpType}
2021-01-12 23:18:53 +00:00
explodeNode := ExpressionNode{Operation: &explodeOp}
context, err := p.treeNavigator.GetMatchingNodes(Context{MatchingNodes: matchingNodes}, &explodeNode)
2020-11-06 03:37:01 +00:00
if err != nil {
return err
}
matchingNodes = context.MatchingNodes
2020-11-06 03:37:01 +00:00
}
2020-11-03 23:48:43 +00:00
2020-11-13 02:19:54 +00:00
bufferedWriter := bufio.NewWriter(p.writer)
2020-11-29 09:25:47 +00:00
defer p.safelyFlush(bufferedWriter)
2020-11-03 23:48:43 +00:00
if matchingNodes.Len() == 0 {
log.Debug("no matching results, nothing to print")
return nil
}
2020-11-16 01:09:57 +00:00
if p.firstTimePrinting {
2020-12-15 03:33:50 +00:00
node := matchingNodes.Front().Value.(*CandidateNode)
p.previousDocIndex = node.Document
p.previousFileIndex = node.FileIndex
2020-11-16 01:09:57 +00:00
p.firstTimePrinting = false
}
2020-11-03 23:48:43 +00:00
for el := matchingNodes.Front(); el != nil; el = el.Next() {
mappedDoc := el.Value.(*CandidateNode)
2020-11-16 01:09:57 +00:00
log.Debug("-- print sep logic: p.firstTimePrinting: %v, previousDocIndex: %v, mappedDoc.Document: %v, printDocSeparators: %v", p.firstTimePrinting, p.previousDocIndex, mappedDoc.Document, p.printDocSeparators)
commentStartsWithSeparator := strings.Contains(mappedDoc.Node.HeadComment, "$yqLeadingContent$\n$yqDocSeperator$")
if (p.previousDocIndex != mappedDoc.Document || p.previousFileIndex != mappedDoc.FileIndex) && p.printDocSeparators && !commentStartsWithSeparator {
2020-11-16 01:09:57 +00:00
log.Debug("-- writing doc sep")
2020-11-13 03:07:11 +00:00
if err := p.writeString(bufferedWriter, "---\n"); err != nil {
return err
}
2020-11-03 23:48:43 +00:00
}
if strings.Contains(mappedDoc.Node.HeadComment, "$yqLeadingContent$") {
log.Debug("headcommentwas %v", mappedDoc.Node.HeadComment)
log.Debug("finished headcomment")
reader := bufio.NewReader(strings.NewReader(mappedDoc.Node.HeadComment))
mappedDoc.Node.HeadComment = ""
for {
readline, errReading := reader.ReadString('\n')
if errReading != nil && errReading != io.EOF {
return errReading
}
if strings.Contains(readline, "$yqLeadingContent$") {
// skip this
} else if strings.Contains(readline, "$yqDocSeperator$") {
if p.printDocSeparators {
if err := p.writeString(bufferedWriter, "---\n"); err != nil {
return err
}
}
2021-07-25 08:08:33 +00:00
} else if p.outputFormat == YamlOutputFormat {
if err := p.writeString(bufferedWriter, readline); err != nil {
return err
}
}
if errReading == io.EOF {
if readline != "" {
// the last comment we read didn't have a new line, put one in
if err := p.writeString(bufferedWriter, "\n"); err != nil {
return err
}
}
break
}
2021-07-19 09:58:47 +00:00
}
2021-07-19 09:52:51 +00:00
}
2021-07-25 08:08:33 +00:00
2020-11-03 23:48:43 +00:00
if err := p.printNode(mappedDoc.Node, bufferedWriter); err != nil {
return err
}
2020-11-16 01:09:57 +00:00
p.previousDocIndex = mappedDoc.Document
2020-11-03 23:48:43 +00:00
}
2021-07-25 08:08:33 +00:00
if p.appendixReader != nil && p.outputFormat == YamlOutputFormat {
2021-07-18 02:28:46 +00:00
log.Debug("Piping appendix reader...")
betterReader := bufio.NewReader(p.appendixReader)
_, err := io.Copy(bufferedWriter, betterReader)
if err != nil {
return err
}
}
2020-11-03 23:48:43 +00:00
return nil
}