mirror of
https://github.com/mikefarah/yq.git
synced 2025-01-13 20:15:57 +00:00
wip
This commit is contained in:
parent
bff4c9e586
commit
d912d7d178
@ -1,4 +1,8 @@
|
|||||||
- name: bob
|
# welcome! # cat
|
||||||
age: 23
|
# bob
|
||||||
- name: tim
|
---
|
||||||
age: 17
|
# bob
|
||||||
|
|
||||||
|
a: cat # meow
|
||||||
|
|
||||||
|
# have a great day
|
@ -1,3 +1,2 @@
|
|||||||
#b1
|
c:
|
||||||
b: 2
|
d: hamster
|
||||||
#b2
|
|
@ -71,14 +71,15 @@ func (e *allAtOnceEvaluator) EvaluateFiles(expression string, filenames []string
|
|||||||
|
|
||||||
if allDocuments.Len() == 0 {
|
if allDocuments.Len() == 0 {
|
||||||
candidateNode := &CandidateNode{
|
candidateNode := &CandidateNode{
|
||||||
Document: 0,
|
Document: 0,
|
||||||
Filename: "",
|
Filename: "",
|
||||||
Node: &yaml.Node{Kind: yaml.DocumentNode, HeadComment: firstFileLeadingContent, Content: []*yaml.Node{{Tag: "!!null", Kind: yaml.ScalarNode}}},
|
Node: &yaml.Node{Kind: yaml.DocumentNode, Content: []*yaml.Node{{Tag: "!!null", Kind: yaml.ScalarNode}}},
|
||||||
FileIndex: 0,
|
FileIndex: 0,
|
||||||
|
LeadingContent: firstFileLeadingContent,
|
||||||
}
|
}
|
||||||
allDocuments.PushBack(candidateNode)
|
allDocuments.PushBack(candidateNode)
|
||||||
} else {
|
} else {
|
||||||
allDocuments.Front().Value.(*CandidateNode).Node.HeadComment = firstFileLeadingContent
|
allDocuments.Front().Value.(*CandidateNode).LeadingContent = firstFileLeadingContent
|
||||||
}
|
}
|
||||||
|
|
||||||
matches, err := e.EvaluateCandidateNodes(expression, allDocuments)
|
matches, err := e.EvaluateCandidateNodes(expression, allDocuments)
|
||||||
|
@ -9,10 +9,13 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type CandidateNode struct {
|
type CandidateNode struct {
|
||||||
Node *yaml.Node // the actual node
|
Node *yaml.Node // the actual node
|
||||||
Parent *CandidateNode // parent node
|
Parent *CandidateNode // parent node
|
||||||
Path []interface{} /// the path we took to get to this node
|
|
||||||
Document uint // the document index of this node
|
LeadingContent string
|
||||||
|
|
||||||
|
Path []interface{} /// the path we took to get to this node
|
||||||
|
Document uint // the document index of this node
|
||||||
Filename string
|
Filename string
|
||||||
FileIndex int
|
FileIndex int
|
||||||
// when performing op against all nodes given, this will treat all the nodes as one
|
// when performing op against all nodes given, this will treat all the nodes as one
|
||||||
|
@ -139,6 +139,26 @@ will output
|
|||||||
welcome!
|
welcome!
|
||||||
```
|
```
|
||||||
|
|
||||||
|
##
|
||||||
|
Given a sample.yml file of:
|
||||||
|
```yaml
|
||||||
|
# welcome!
|
||||||
|
---
|
||||||
|
# bob
|
||||||
|
a: cat # meow
|
||||||
|
|
||||||
|
# have a great day
|
||||||
|
```
|
||||||
|
then
|
||||||
|
```bash
|
||||||
|
yq eval 'headComment' sample.yml
|
||||||
|
```
|
||||||
|
will output
|
||||||
|
```yaml
|
||||||
|
welcome!
|
||||||
|
bob
|
||||||
|
```
|
||||||
|
|
||||||
## Get foot comment
|
## Get foot comment
|
||||||
Given a sample.yml file of:
|
Given a sample.yml file of:
|
||||||
```yaml
|
```yaml
|
||||||
@ -147,6 +167,7 @@ Given a sample.yml file of:
|
|||||||
a: cat # meow
|
a: cat # meow
|
||||||
|
|
||||||
# have a great day
|
# have a great day
|
||||||
|
# no really
|
||||||
```
|
```
|
||||||
then
|
then
|
||||||
```bash
|
```bash
|
||||||
@ -155,5 +176,6 @@ yq eval '. | footComment' sample.yml
|
|||||||
will output
|
will output
|
||||||
```yaml
|
```yaml
|
||||||
have a great day
|
have a great day
|
||||||
|
no really
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -1,8 +1,12 @@
|
|||||||
package yqlib
|
package yqlib
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
// "bufio"
|
||||||
|
// "bytes"
|
||||||
|
"bufio"
|
||||||
|
"bytes"
|
||||||
"container/list"
|
"container/list"
|
||||||
"strings"
|
"regexp"
|
||||||
|
|
||||||
yaml "gopkg.in/yaml.v3"
|
yaml "gopkg.in/yaml.v3"
|
||||||
)
|
)
|
||||||
@ -57,6 +61,7 @@ func assignCommentsOperator(d *dataTreeNavigator, context Context, expressionNod
|
|||||||
}
|
}
|
||||||
if preferences.HeadComment {
|
if preferences.HeadComment {
|
||||||
candidate.Node.HeadComment = comment
|
candidate.Node.HeadComment = comment
|
||||||
|
candidate.LeadingContent = "" // clobber the leading content, if there was any.
|
||||||
}
|
}
|
||||||
if preferences.FootComment {
|
if preferences.FootComment {
|
||||||
candidate.Node.FootComment = comment
|
candidate.Node.FootComment = comment
|
||||||
@ -68,6 +73,9 @@ func assignCommentsOperator(d *dataTreeNavigator, context Context, expressionNod
|
|||||||
|
|
||||||
func getCommentsOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
func getCommentsOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
||||||
preferences := expressionNode.Operation.Preferences.(commentOpPreferences)
|
preferences := expressionNode.Operation.Preferences.(commentOpPreferences)
|
||||||
|
var startCommentCharaterRegExp = regexp.MustCompile(`^# `)
|
||||||
|
var subsequentCommentCharaterRegExp = regexp.MustCompile(`\n# `)
|
||||||
|
|
||||||
log.Debugf("GetComments operator!")
|
log.Debugf("GetComments operator!")
|
||||||
var results = list.New()
|
var results = list.New()
|
||||||
|
|
||||||
@ -76,12 +84,25 @@ func getCommentsOperator(d *dataTreeNavigator, context Context, expressionNode *
|
|||||||
comment := ""
|
comment := ""
|
||||||
if preferences.LineComment {
|
if preferences.LineComment {
|
||||||
comment = candidate.Node.LineComment
|
comment = candidate.Node.LineComment
|
||||||
|
} else if preferences.HeadComment && candidate.LeadingContent != "" {
|
||||||
|
var chompRegexp = regexp.MustCompile(`\n$`)
|
||||||
|
var output bytes.Buffer
|
||||||
|
var writer = bufio.NewWriter(&output)
|
||||||
|
if err := processLeadingContent(candidate, writer, false, YamlOutputFormat); err != nil {
|
||||||
|
return Context{}, err
|
||||||
|
}
|
||||||
|
if err := writer.Flush(); err != nil {
|
||||||
|
return Context{}, err
|
||||||
|
}
|
||||||
|
comment = output.String()
|
||||||
|
comment = chompRegexp.ReplaceAllString(comment, "")
|
||||||
} else if preferences.HeadComment {
|
} else if preferences.HeadComment {
|
||||||
comment = candidate.Node.HeadComment
|
comment = candidate.Node.HeadComment
|
||||||
} else if preferences.FootComment {
|
} else if preferences.FootComment {
|
||||||
comment = candidate.Node.FootComment
|
comment = candidate.Node.FootComment
|
||||||
}
|
}
|
||||||
comment = strings.Replace(comment, "# ", "", 1)
|
comment = startCommentCharaterRegExp.ReplaceAllString(comment, "")
|
||||||
|
comment = subsequentCommentCharaterRegExp.ReplaceAllString(comment, "\n")
|
||||||
|
|
||||||
node := &yaml.Node{Kind: yaml.ScalarNode, Value: comment, Tag: "!!str"}
|
node := &yaml.Node{Kind: yaml.ScalarNode, Value: comment, Tag: "!!str"}
|
||||||
result := candidate.CreateChild(nil, node)
|
result := candidate.CreateChild(nil, node)
|
||||||
|
@ -112,13 +112,22 @@ var commentOperatorScenarios = []expressionScenario{
|
|||||||
"D0, P[], (!!str)::welcome!\n",
|
"D0, P[], (!!str)::welcome!\n",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
skipDoc: false,
|
||||||
|
dontFormatInputForDoc: true,
|
||||||
|
document: "# welcome!\n---\n# bob\na: cat # meow\n\n# have a great day",
|
||||||
|
expression: `headComment`,
|
||||||
|
expected: []string{
|
||||||
|
"D0, P[], (!!str)::|-\n welcome!\n bob\n",
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
description: "Get foot comment",
|
description: "Get foot comment",
|
||||||
dontFormatInputForDoc: true,
|
dontFormatInputForDoc: true,
|
||||||
document: "# welcome!\n\na: cat # meow\n\n# have a great day",
|
document: "# welcome!\n\na: cat # meow\n\n# have a great day\n# no really",
|
||||||
expression: `. | footComment`,
|
expression: `. | footComment`,
|
||||||
expected: []string{
|
expected: []string{
|
||||||
"D0, P[], (!!str)::have a great day\n",
|
"D0, P[], (!!str)::have a great day\nno really\n",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -26,6 +26,20 @@ type expressionScenario struct {
|
|||||||
dontFormatInputForDoc bool // dont format input doc for documentation generation
|
dontFormatInputForDoc bool // dont format input doc for documentation generation
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func readDocumentWithLeadingContent(content string, fakefilename string, fakeFileIndex int) (*list.List, error) {
|
||||||
|
reader, firstFileLeadingContent, err := processReadStream(bufio.NewReader(strings.NewReader(content)))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
inputs, err := readDocuments(reader, fakefilename, fakeFileIndex)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
inputs.Front().Value.(*CandidateNode).LeadingContent = firstFileLeadingContent
|
||||||
|
return inputs, nil
|
||||||
|
}
|
||||||
|
|
||||||
func testScenario(t *testing.T, s *expressionScenario) {
|
func testScenario(t *testing.T, s *expressionScenario) {
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
@ -37,15 +51,17 @@ func testScenario(t *testing.T, s *expressionScenario) {
|
|||||||
inputs := list.New()
|
inputs := list.New()
|
||||||
|
|
||||||
if s.document != "" {
|
if s.document != "" {
|
||||||
inputs, err = readDocuments(strings.NewReader(s.document), "sample.yml", 0)
|
inputs, err = readDocumentWithLeadingContent(s.document, "sample.yml", 0)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err, s.document, s.expression)
|
t.Error(err, s.document, s.expression)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if s.document2 != "" {
|
if s.document2 != "" {
|
||||||
moreInputs, err := readDocuments(strings.NewReader(s.document2), "another.yml", 1)
|
moreInputs, err := readDocumentWithLeadingContent(s.document2, "another.yml", 1)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err, s.document, s.expression)
|
t.Error(err, s.document2, s.expression)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
inputs.PushBackList(moreInputs)
|
inputs.PushBackList(moreInputs)
|
||||||
@ -227,13 +243,14 @@ func documentOutput(t *testing.T, w *bufio.Writer, s expressionScenario, formatt
|
|||||||
inputs := list.New()
|
inputs := list.New()
|
||||||
|
|
||||||
if s.document != "" {
|
if s.document != "" {
|
||||||
inputs, err = readDocuments(strings.NewReader(formattedDoc), "sample.yml", 0)
|
|
||||||
|
inputs, err = readDocumentWithLeadingContent(formattedDoc, "sample.yml", 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err, s.document, s.expression)
|
t.Error(err, s.document, s.expression)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if s.document2 != "" {
|
if s.document2 != "" {
|
||||||
moreInputs, err := readDocuments(strings.NewReader(formattedDoc2), "another.yml", 1)
|
moreInputs, err := readDocumentWithLeadingContent(formattedDoc2, "another.yml", 1)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err, s.document, s.expression)
|
t.Error(err, s.document, s.expression)
|
||||||
return
|
return
|
||||||
|
@ -84,7 +84,7 @@ func (p *resultsPrinter) printNode(node *yaml.Node, writer io.Writer) error {
|
|||||||
|
|
||||||
var encoder Encoder
|
var encoder Encoder
|
||||||
if node.Kind == yaml.ScalarNode && p.unwrapScalar && p.outputFormat == YamlOutputFormat {
|
if node.Kind == yaml.ScalarNode && p.unwrapScalar && p.outputFormat == YamlOutputFormat {
|
||||||
return p.writeString(writer, node.Value+"\n")
|
return writeString(writer, node.Value+"\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
if p.outputFormat == JsonOutputFormat {
|
if p.outputFormat == JsonOutputFormat {
|
||||||
@ -97,54 +97,6 @@ func (p *resultsPrinter) printNode(node *yaml.Node, writer io.Writer) error {
|
|||||||
return encoder.Encode(node)
|
return encoder.Encode(node)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *resultsPrinter) writeString(writer io.Writer, txt string) error {
|
|
||||||
_, errorWriting := writer.Write([]byte(txt))
|
|
||||||
return errorWriting
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *resultsPrinter) processLeadingContent(mappedDoc *CandidateNode, writer io.Writer) error {
|
|
||||||
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(writer, "---\n"); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if p.outputFormat == YamlOutputFormat {
|
|
||||||
if err := p.writeString(writer, 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(writer, "\n"); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *resultsPrinter) PrintResults(matchingNodes *list.List) error {
|
func (p *resultsPrinter) PrintResults(matchingNodes *list.List) error {
|
||||||
log.Debug("PrintResults for %v matches", matchingNodes.Len())
|
log.Debug("PrintResults for %v matches", matchingNodes.Len())
|
||||||
|
|
||||||
@ -180,16 +132,16 @@ func (p *resultsPrinter) PrintResults(matchingNodes *list.List) error {
|
|||||||
return errorWriting
|
return errorWriting
|
||||||
}
|
}
|
||||||
|
|
||||||
commentStartsWithSeparator := strings.Contains(mappedDoc.Node.HeadComment, "$yqLeadingContent$\n$yqDocSeperator$")
|
commentStartsWithSeparator := strings.Contains(mappedDoc.LeadingContent, "$yqLeadingContent$\n$yqDocSeperator$")
|
||||||
|
|
||||||
if (p.previousDocIndex != mappedDoc.Document || p.previousFileIndex != mappedDoc.FileIndex) && p.printDocSeparators && !commentStartsWithSeparator {
|
if (p.previousDocIndex != mappedDoc.Document || p.previousFileIndex != mappedDoc.FileIndex) && p.printDocSeparators && !commentStartsWithSeparator {
|
||||||
log.Debug("-- writing doc sep")
|
log.Debug("-- writing doc sep")
|
||||||
if err := p.writeString(writer, "---\n"); err != nil {
|
if err := writeString(writer, "---\n"); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := p.processLeadingContent(mappedDoc, writer); err != nil {
|
if err := processLeadingContent(mappedDoc, writer, p.printDocSeparators, p.outputFormat); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -33,10 +33,11 @@ func (s *streamEvaluator) EvaluateNew(expression string, printer Printer, leadin
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
candidateNode := &CandidateNode{
|
candidateNode := &CandidateNode{
|
||||||
Document: 0,
|
Document: 0,
|
||||||
Filename: "",
|
Filename: "",
|
||||||
Node: &yaml.Node{Kind: yaml.DocumentNode, HeadComment: leadingContent, Content: []*yaml.Node{{Tag: "!!null", Kind: yaml.ScalarNode}}},
|
Node: &yaml.Node{Kind: yaml.DocumentNode, Content: []*yaml.Node{{Tag: "!!null", Kind: yaml.ScalarNode}}},
|
||||||
FileIndex: 0,
|
FileIndex: 0,
|
||||||
|
LeadingContent: leadingContent,
|
||||||
}
|
}
|
||||||
inputList := list.New()
|
inputList := list.New()
|
||||||
inputList.PushBack(candidateNode)
|
inputList.PushBack(candidateNode)
|
||||||
@ -100,15 +101,16 @@ func (s *streamEvaluator) Evaluate(filename string, reader io.Reader, node *Expr
|
|||||||
} else if errorReading != nil {
|
} else if errorReading != nil {
|
||||||
return currentIndex, errorReading
|
return currentIndex, errorReading
|
||||||
}
|
}
|
||||||
if currentIndex == 0 {
|
|
||||||
dataBucket.HeadComment = leadingContent
|
|
||||||
}
|
|
||||||
candidateNode := &CandidateNode{
|
candidateNode := &CandidateNode{
|
||||||
Document: currentIndex,
|
Document: currentIndex,
|
||||||
Filename: filename,
|
Filename: filename,
|
||||||
Node: &dataBucket,
|
Node: &dataBucket,
|
||||||
FileIndex: s.fileIndex,
|
FileIndex: s.fileIndex,
|
||||||
}
|
}
|
||||||
|
if currentIndex == 0 {
|
||||||
|
candidateNode.LeadingContent = leadingContent
|
||||||
|
}
|
||||||
inputList := list.New()
|
inputList := list.New()
|
||||||
inputList.PushBack(candidateNode)
|
inputList.PushBack(candidateNode)
|
||||||
|
|
||||||
|
@ -31,6 +31,54 @@ func readStream(filename string, leadingContentPreProcessing bool) (io.Reader, s
|
|||||||
return processReadStream(reader)
|
return processReadStream(reader)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func writeString(writer io.Writer, txt string) error {
|
||||||
|
_, errorWriting := writer.Write([]byte(txt))
|
||||||
|
return errorWriting
|
||||||
|
}
|
||||||
|
|
||||||
|
func processLeadingContent(mappedDoc *CandidateNode, writer io.Writer, printDocSeparators bool, outputFormat PrinterOutputFormat) error {
|
||||||
|
if strings.Contains(mappedDoc.LeadingContent, "$yqLeadingContent$") {
|
||||||
|
log.Debug("headcommentwas %v", mappedDoc.LeadingContent)
|
||||||
|
log.Debug("finished headcomment")
|
||||||
|
reader := bufio.NewReader(strings.NewReader(mappedDoc.LeadingContent))
|
||||||
|
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 printDocSeparators {
|
||||||
|
if err := writeString(writer, "---\n"); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if outputFormat == YamlOutputFormat {
|
||||||
|
if err := writeString(writer, 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 := writeString(writer, "\n"); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func processReadStream(reader *bufio.Reader) (io.Reader, string, error) {
|
func processReadStream(reader *bufio.Reader) (io.Reader, string, error) {
|
||||||
var commentLineRegEx = regexp.MustCompile(`^\s*#`)
|
var commentLineRegEx = regexp.MustCompile(`^\s*#`)
|
||||||
var sb strings.Builder
|
var sb strings.Builder
|
||||||
|
Loading…
Reference in New Issue
Block a user