2021-10-22 01:00:47 +00:00
|
|
|
package yqlib
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"bytes"
|
|
|
|
"container/list"
|
2023-03-01 02:19:06 +00:00
|
|
|
"errors"
|
2021-10-22 03:53:39 +00:00
|
|
|
"regexp"
|
2021-10-22 01:37:47 +00:00
|
|
|
"strings"
|
2021-10-22 01:00:47 +00:00
|
|
|
)
|
|
|
|
|
2022-01-15 00:57:59 +00:00
|
|
|
func configureEncoder(format PrinterOutputFormat, indent int) Encoder {
|
|
|
|
switch format {
|
2022-02-07 00:55:55 +00:00
|
|
|
case JSONOutputFormat:
|
2022-11-10 08:21:18 +00:00
|
|
|
return NewJSONEncoder(indent, false, false)
|
2022-01-15 00:57:59 +00:00
|
|
|
case PropsOutputFormat:
|
2022-06-25 02:22:03 +00:00
|
|
|
return NewPropertiesEncoder(true)
|
2022-02-07 00:55:55 +00:00
|
|
|
case CSVOutputFormat:
|
2022-01-15 00:57:59 +00:00
|
|
|
return NewCsvEncoder(',')
|
2022-02-07 00:55:55 +00:00
|
|
|
case TSVOutputFormat:
|
2022-01-15 00:57:59 +00:00
|
|
|
return NewCsvEncoder('\t')
|
|
|
|
case YamlOutputFormat:
|
2022-10-28 03:16:46 +00:00
|
|
|
return NewYamlEncoder(indent, false, ConfiguredYamlPreferences)
|
2022-02-07 00:55:55 +00:00
|
|
|
case XMLOutputFormat:
|
2022-10-25 03:27:16 +00:00
|
|
|
return NewXMLEncoder(indent, ConfiguredXMLPreferences)
|
2022-02-22 22:26:35 +00:00
|
|
|
case Base64OutputFormat:
|
|
|
|
return NewBase64Encoder()
|
2023-01-23 00:37:18 +00:00
|
|
|
case UriOutputFormat:
|
|
|
|
return NewUriEncoder()
|
2023-02-02 01:22:52 +00:00
|
|
|
case ShOutputFormat:
|
|
|
|
return NewShEncoder()
|
2022-01-15 00:57:59 +00:00
|
|
|
}
|
|
|
|
panic("invalid encoder")
|
|
|
|
}
|
|
|
|
|
|
|
|
func encodeToString(candidate *CandidateNode, prefs encoderPreferences) (string, error) {
|
2021-10-22 01:00:47 +00:00
|
|
|
var output bytes.Buffer
|
2021-10-24 00:35:40 +00:00
|
|
|
log.Debug("printing with indent: %v", prefs.indent)
|
2021-10-29 03:14:39 +00:00
|
|
|
|
2022-01-15 00:57:59 +00:00
|
|
|
encoder := configureEncoder(prefs.format, prefs.indent)
|
2023-03-01 02:19:06 +00:00
|
|
|
if encoder == nil {
|
|
|
|
return "", errors.New("no support for output format")
|
|
|
|
}
|
2022-01-15 00:57:59 +00:00
|
|
|
|
|
|
|
printer := NewPrinter(encoder, NewSinglePrinterWriter(bufio.NewWriter(&output)))
|
2021-10-22 03:53:39 +00:00
|
|
|
err := printer.PrintResults(candidate.AsList())
|
2021-10-22 01:00:47 +00:00
|
|
|
return output.String(), err
|
|
|
|
}
|
|
|
|
|
|
|
|
type encoderPreferences struct {
|
|
|
|
format PrinterOutputFormat
|
2021-10-24 00:35:40 +00:00
|
|
|
indent int
|
2021-10-22 01:00:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* encodes object as yaml string */
|
|
|
|
|
|
|
|
func encodeOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
|
|
|
preferences := expressionNode.Operation.Preferences.(encoderPreferences)
|
|
|
|
var results = list.New()
|
2021-10-22 03:53:39 +00:00
|
|
|
|
|
|
|
hasOnlyOneNewLine := regexp.MustCompile("[^\n].*\n$")
|
2021-10-22 04:21:01 +00:00
|
|
|
endWithNewLine := regexp.MustCompile(".*\n$")
|
2021-10-22 03:53:39 +00:00
|
|
|
chomper := regexp.MustCompile("\n+$")
|
|
|
|
|
2021-10-22 01:00:47 +00:00
|
|
|
for el := context.MatchingNodes.Front(); el != nil; el = el.Next() {
|
|
|
|
candidate := el.Value.(*CandidateNode)
|
2022-01-15 00:57:59 +00:00
|
|
|
stringValue, err := encodeToString(candidate, preferences)
|
2021-10-22 03:53:39 +00:00
|
|
|
|
2021-10-22 01:00:47 +00:00
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
|
|
|
|
2021-10-22 03:53:39 +00:00
|
|
|
// remove trailing newlines if needed.
|
|
|
|
// check if we originally decoded this path, and the original thing had a single line.
|
|
|
|
originalList := context.GetVariable("decoded: " + candidate.GetKey())
|
|
|
|
if originalList != nil && originalList.Len() > 0 && hasOnlyOneNewLine.MatchString(stringValue) {
|
|
|
|
|
|
|
|
original := originalList.Front().Value.(*CandidateNode)
|
2023-04-09 01:14:51 +00:00
|
|
|
originalNode := original.unwrapDocument()
|
2023-03-16 02:39:36 +00:00
|
|
|
// original block did not have a newline at the end, get rid of this one too
|
2021-10-22 04:21:01 +00:00
|
|
|
if !endWithNewLine.MatchString(originalNode.Value) {
|
2021-10-22 03:53:39 +00:00
|
|
|
stringValue = chomper.ReplaceAllString(stringValue, "")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-16 02:39:36 +00:00
|
|
|
// dont print a newline when printing json on a single line.
|
2022-02-07 00:55:55 +00:00
|
|
|
if (preferences.format == JSONOutputFormat && preferences.indent == 0) ||
|
|
|
|
preferences.format == CSVOutputFormat ||
|
|
|
|
preferences.format == TSVOutputFormat {
|
2021-10-24 00:35:40 +00:00
|
|
|
stringValue = chomper.ReplaceAllString(stringValue, "")
|
|
|
|
}
|
|
|
|
|
2023-04-09 01:14:51 +00:00
|
|
|
results.PushBack(candidate.CreateReplacement(ScalarNode, "!!str", stringValue))
|
2021-10-22 01:00:47 +00:00
|
|
|
}
|
|
|
|
return context.ChildContext(results), nil
|
|
|
|
}
|
2021-10-22 01:37:47 +00:00
|
|
|
|
2021-12-21 04:02:07 +00:00
|
|
|
type decoderPreferences struct {
|
|
|
|
format InputFormat
|
|
|
|
}
|
|
|
|
|
2023-03-01 02:19:06 +00:00
|
|
|
func createDecoder(format InputFormat) Decoder {
|
2021-12-21 04:02:07 +00:00
|
|
|
var decoder Decoder
|
2023-03-01 02:19:06 +00:00
|
|
|
switch format {
|
|
|
|
case JsonInputFormat:
|
|
|
|
decoder = NewJSONDecoder()
|
2021-12-21 04:02:07 +00:00
|
|
|
case YamlInputFormat:
|
2022-10-28 03:16:46 +00:00
|
|
|
decoder = NewYamlDecoder(ConfiguredYamlPreferences)
|
2022-02-07 00:55:55 +00:00
|
|
|
case XMLInputFormat:
|
2022-10-25 03:27:16 +00:00
|
|
|
decoder = NewXMLDecoder(ConfiguredXMLPreferences)
|
2022-02-22 22:26:35 +00:00
|
|
|
case Base64InputFormat:
|
|
|
|
decoder = NewBase64Decoder()
|
2022-03-28 08:48:30 +00:00
|
|
|
case PropertiesInputFormat:
|
|
|
|
decoder = NewPropertiesDecoder()
|
2022-08-01 00:28:34 +00:00
|
|
|
case CSVObjectInputFormat:
|
|
|
|
decoder = NewCSVObjectDecoder(',')
|
|
|
|
case TSVObjectInputFormat:
|
|
|
|
decoder = NewCSVObjectDecoder('\t')
|
2023-01-23 00:37:18 +00:00
|
|
|
case UriInputFormat:
|
|
|
|
decoder = NewUriDecoder()
|
2021-12-21 04:02:07 +00:00
|
|
|
}
|
2023-03-01 02:19:06 +00:00
|
|
|
return decoder
|
|
|
|
}
|
|
|
|
|
|
|
|
/* takes a string and decodes it back into an object */
|
|
|
|
func decodeOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
|
|
|
|
|
|
|
preferences := expressionNode.Operation.Preferences.(decoderPreferences)
|
|
|
|
|
|
|
|
decoder := createDecoder(preferences.format)
|
|
|
|
if decoder == nil {
|
|
|
|
return Context{}, errors.New("no support for input format")
|
|
|
|
}
|
2021-12-21 04:02:07 +00:00
|
|
|
|
2021-10-22 01:37:47 +00:00
|
|
|
var results = list.New()
|
|
|
|
for el := context.MatchingNodes.Front(); el != nil; el = el.Next() {
|
|
|
|
candidate := el.Value.(*CandidateNode)
|
2021-10-22 03:53:39 +00:00
|
|
|
|
|
|
|
context.SetVariable("decoded: "+candidate.GetKey(), candidate.AsList())
|
|
|
|
|
2023-04-09 01:14:51 +00:00
|
|
|
log.Debugf("got: [%v]", candidate.Value)
|
2021-12-21 04:02:07 +00:00
|
|
|
|
2023-04-09 01:14:51 +00:00
|
|
|
err := decoder.Init(strings.NewReader(candidate.unwrapDocument().Value))
|
2022-10-28 03:16:46 +00:00
|
|
|
if err != nil {
|
|
|
|
return Context{}, err
|
|
|
|
}
|
2021-12-21 04:02:07 +00:00
|
|
|
|
2022-10-28 03:16:46 +00:00
|
|
|
decodedNode, errorReading := decoder.Decode()
|
2021-10-22 01:37:47 +00:00
|
|
|
if errorReading != nil {
|
|
|
|
return Context{}, errorReading
|
|
|
|
}
|
|
|
|
//first node is a doc
|
2023-04-09 01:14:51 +00:00
|
|
|
node := decodedNode.unwrapDocument()
|
|
|
|
node.Key = candidate.Key
|
|
|
|
node.Parent = candidate.Parent
|
|
|
|
node.Document = candidate.Document
|
|
|
|
node.FileIndex = candidate.FileIndex
|
|
|
|
node.Filename = candidate.Filename
|
|
|
|
|
|
|
|
results.PushBack(node)
|
2021-10-22 01:37:47 +00:00
|
|
|
}
|
|
|
|
return context.ChildContext(results), nil
|
|
|
|
}
|