Skip to content

Commit

Permalink
Allow processing of a stream of separate JSON elements. (#2)
Browse files Browse the repository at this point in the history
Signed-off-by: Springcomp <[email protected]>
  • Loading branch information
springcomp authored Mar 25, 2023
1 parent 725b6ea commit db0c35f
Show file tree
Hide file tree
Showing 2 changed files with 49 additions and 26 deletions.
64 changes: 38 additions & 26 deletions jp.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package main
import (
"encoding/json"
"fmt"
"io"
"io/ioutil"
"os"

Expand Down Expand Up @@ -43,6 +44,10 @@ func main() {
Usage: "If the final result is a string, it will be printed without quotes.",
EnvVar: "JP_UNQUOTED",
},
cli.BoolFlag{
Name: "stream, s",
Usage: "Parse JSON elements until the input stream is exhausted (rather than just the first JSON).",
},
cli.BoolFlag{
Name: "ast",
Usage: "Only print the AST of the parsed expression. Do not rely on this output, only useful for debugging purposes.",
Expand Down Expand Up @@ -106,7 +111,6 @@ func runMain(c *cli.Context) int {
fmt.Printf("%s\n", parsed)
return 0
}
var input interface{}
var jsonParser *json.Decoder
if c.String("filename") != "" {
f, err := os.Open(c.String("filename"))
Expand All @@ -118,36 +122,44 @@ func runMain(c *cli.Context) int {
} else {
jsonParser = json.NewDecoder(os.Stdin)
}
if err := jsonParser.Decode(&input); err != nil {
errMsg("Error parsing input json: %s\n", err)
return 2
}
result, err := jmespath.Search(expression, input)
if err != nil {
if syntaxError, ok := err.(jmespath.SyntaxError); ok {
return errMsg("%s\n%s\n",
syntaxError,
syntaxError.HighlightLocation())
for {
var input interface{}
if err := jsonParser.Decode(&input); err == io.EOF {
break
} else if err != nil {
errMsg("Error parsing input json: %s\n", err)
return 2
}
return errMsg("Error evaluating JMESPath expression: %s", err)
}
converted, isString := result.(string)
if c.Bool("unquoted") && isString {
os.Stdout.WriteString(converted)
} else {
var toJSON []byte
if c.Bool("compact") {
toJSON, err = jsoncolor.Marshal(result)
result, err := jmespath.Search(expression, input)
if err != nil {
if syntaxError, ok := err.(jmespath.SyntaxError); ok {
return errMsg("%s\n%s\n",
syntaxError,
syntaxError.HighlightLocation())
}
return errMsg("Error evaluating JMESPath expression: %s", err)
}
converted, isString := result.(string)
if c.Bool("unquoted") && isString {
os.Stdout.WriteString(converted)
} else {
toJSON, err = jsoncolor.MarshalIndent(result, "", " ")
var toJSON []byte
if c.Bool("compact") {
toJSON, err = json.Marshal(result)
} else {
toJSON, err = json.MarshalIndent(result, "", " ")
}
if err != nil {
errMsg("Error marshalling result to JSON: %s\n", err)
return 3
}
os.Stdout.Write(toJSON)
}
if err != nil {
errMsg("Error marshalling result to JSON: %s\n", err)
return 3
os.Stdout.WriteString("\n")
if !c.Bool("stream") {
break
}
os.Stdout.Write(toJSON)
}
os.Stdout.WriteString("\n")
return 0
}

Expand Down
11 changes: 11 additions & 0 deletions test/cases/search.bats
Original file line number Diff line number Diff line change
Expand Up @@ -69,3 +69,14 @@ ASTField {
[ "$status" -eq 0 ]
[ "$output" == "12345" ]
}

@test "Ignores subsequent data" {
output=$(echo '{"foo": "bar"}blah' | ./jp foo)
[ "$output" == "\"bar\"" ]
}

@test "Processes subsequent data in stream mode" {
output=$(echo '{"foo": "bar"}{"foo": "x"}' | ./jp -s foo)
echo "$output"
[ "$output" == $'\"bar\"\n\"x\"' ]
}

0 comments on commit db0c35f

Please sign in to comment.