yq/yq.go

705 lines
21 KiB
Go
Raw Permalink Normal View History

2015-09-26 22:15:49 +00:00
package main
import (
2018-06-12 05:33:59 +00:00
"bufio"
2015-09-28 02:00:38 +00:00
"fmt"
2018-06-12 05:33:59 +00:00
"io"
2015-09-28 02:00:38 +00:00
"io/ioutil"
"os"
"reflect"
2015-09-28 02:00:38 +00:00
"strconv"
"strings"
2018-06-15 06:11:13 +00:00
errors "github.com/pkg/errors"
yaml "gopkg.in/mikefarah/yaml.v2"
logging "gopkg.in/op/go-logging.v1"
cobra "gopkg.in/spf13/cobra.v0"
2015-09-26 22:15:49 +00:00
)
2015-10-06 05:07:09 +00:00
var trimOutput = true
var writeInplace = false
var writeScript = ""
2015-10-10 23:00:22 +00:00
var outputToJSON = false
var overwriteFlag = false
var allowEmptyFlag = false
2018-07-07 05:26:56 +00:00
var appendFlag = false
2017-04-11 23:16:54 +00:00
var verbose = false
var version = false
2018-06-20 01:45:51 +00:00
var docIndex = "0"
2017-12-17 22:11:08 +00:00
var log = logging.MustGetLogger("yq")
2015-10-06 05:07:09 +00:00
2015-09-26 22:15:49 +00:00
func main() {
cmd := newCommandCLI()
if err := cmd.Execute(); err != nil {
log.Error(err.Error())
os.Exit(1)
}
}
2017-04-11 23:16:54 +00:00
func newCommandCLI() *cobra.Command {
yaml.DefaultMapType = reflect.TypeOf(yaml.MapSlice{})
var rootCmd = &cobra.Command{
2019-05-14 17:28:06 +00:00
Use: "yq",
Short: "yq is a lightweight and portable command-line YAML processor.",
Long: `yq is a lightweight and portable command-line YAML processor. It aims to be the jq or sed of yaml files.`,
RunE: func(cmd *cobra.Command, args []string) error {
if version {
cmd.Print(GetVersionDisplay())
return nil
}
cmd.Println(cmd.UsageString())
return nil
},
PersistentPreRun: func(cmd *cobra.Command, args []string) {
var format = logging.MustStringFormatter(
`%{color}%{time:15:04:05} %{shortfunc} [%{level:.4s}]%{color:reset} %{message}`,
)
var backend = logging.AddModuleLevel(
logging.NewBackendFormatter(logging.NewLogBackend(os.Stderr, "", 0), format))
if verbose {
backend.SetLevel(logging.DEBUG, "")
} else {
backend.SetLevel(logging.ERROR, "")
}
logging.SetBackend(backend)
},
}
2015-10-13 10:42:36 +00:00
rootCmd.PersistentFlags().BoolVarP(&trimOutput, "trim", "t", true, "trim yaml output")
2017-04-11 23:16:54 +00:00
rootCmd.PersistentFlags().BoolVarP(&verbose, "verbose", "v", false, "verbose mode")
rootCmd.Flags().BoolVarP(&version, "version", "V", false, "Print version information and quit")
rootCmd.AddCommand(
createReadCmd(),
createWriteCmd(),
2018-11-18 15:35:28 +00:00
createPrefixCmd(),
createDeleteCmd(),
createNewCmd(),
createMergeCmd(),
)
rootCmd.SetOutput(os.Stdout)
return rootCmd
2015-10-13 10:42:36 +00:00
}
func createReadCmd() *cobra.Command {
2018-06-12 05:33:59 +00:00
var cmdRead = &cobra.Command{
2015-10-06 05:07:09 +00:00
Use: "read [yaml_file] [path]",
Aliases: []string{"r"},
Short: "yq r [--doc/-d index] sample.yaml a.b.c",
2015-10-06 05:07:09 +00:00
Example: `
2017-12-17 22:11:08 +00:00
yq read things.yaml a.b.c
yq r - a.b.c (reads from stdin)
yq r things.yaml a.*.c
2018-06-12 23:13:58 +00:00
yq r -d1 things.yaml a.array[0].blah
2017-12-17 22:11:08 +00:00
yq r things.yaml a.array[*].blah
yq r -- things.yaml --key-starting-with-dashes
2015-10-13 10:42:36 +00:00
`,
2015-10-06 05:07:09 +00:00
Long: "Outputs the value of the given path in the yaml file to STDOUT",
RunE: readProperty,
2015-09-28 02:00:38 +00:00
}
2018-07-08 11:47:01 +00:00
cmdRead.PersistentFlags().StringVarP(&docIndex, "doc", "d", "0", "process document index number (0 based, * for all documents)")
cmdRead.PersistentFlags().BoolVarP(&outputToJSON, "tojson", "j", false, "output as json")
2018-06-12 05:33:59 +00:00
return cmdRead
2015-10-13 10:42:36 +00:00
}
2015-09-26 23:38:44 +00:00
2015-10-13 10:42:36 +00:00
func createWriteCmd() *cobra.Command {
2015-10-06 05:07:09 +00:00
var cmdWrite = &cobra.Command{
Use: "write [yaml_file] [path] [value]",
Aliases: []string{"w"},
Short: "yq w [--inplace/-i] [--script/-s script_file] [--doc/-d index] sample.yaml a.b.c newValue",
2015-10-06 05:07:09 +00:00
Example: `
2017-12-17 22:11:08 +00:00
yq write things.yaml a.b.c cat
yq write --inplace -- things.yaml a.b.c --cat
2017-12-17 22:11:08 +00:00
yq w -i things.yaml a.b.c cat
yq w --script update_script.yaml things.yaml
yq w -i -s update_script.yaml things.yaml
yq w --doc 2 things.yaml a.b.d[+] foo
yq w -d2 things.yaml a.b.d[+] foo
2015-10-13 10:42:36 +00:00
`,
2015-10-06 05:07:09 +00:00
Long: `Updates the yaml file w.r.t the given path and value.
Outputs to STDOUT unless the inplace flag is used, in which case the file is updated instead.
Append value to array adds the value to the end of array.
Update Scripts:
Note that you can give an update script to perform more sophisticated updated. Update script
format is a yaml map where the key is the path and the value is..well the value. e.g.:
---
a.b.c: true,
a.b.e:
- name: bob
`,
RunE: writeProperty,
2015-10-06 05:07:09 +00:00
}
cmdWrite.PersistentFlags().BoolVarP(&writeInplace, "inplace", "i", false, "update the yaml file inplace")
cmdWrite.PersistentFlags().StringVarP(&writeScript, "script", "s", "", "yaml script for updating yaml")
2018-06-20 01:45:51 +00:00
cmdWrite.PersistentFlags().StringVarP(&docIndex, "doc", "d", "0", "process document index number (0 based, * for all documents)")
2015-10-13 10:42:36 +00:00
return cmdWrite
2015-10-06 05:07:09 +00:00
}
2015-09-29 06:29:32 +00:00
2018-11-18 15:35:28 +00:00
func createPrefixCmd() *cobra.Command {
var cmdWrite = &cobra.Command{
Use: "prefix [yaml_file] [path]",
Aliases: []string{"p"},
Short: "yq p [--inplace/-i] [--doc/-d index] sample.yaml a.b.c",
Example: `
yq prefix things.yaml a.b.c
yq prefix --inplace things.yaml a.b.c
yq prefix --inplace -- things.yaml --key-starting-with-dash
2018-11-18 15:35:28 +00:00
yq p -i things.yaml a.b.c
yq p --doc 2 things.yaml a.b.d
yq p -d2 things.yaml a.b.d
`,
Long: `Prefixes w.r.t to the yaml file at the given path.
Outputs to STDOUT unless the inplace flag is used, in which case the file is updated instead.
`,
RunE: prefixProperty,
}
cmdWrite.PersistentFlags().BoolVarP(&writeInplace, "inplace", "i", false, "update the yaml file inplace")
cmdWrite.PersistentFlags().StringVarP(&docIndex, "doc", "d", "0", "process document index number (0 based, * for all documents)")
return cmdWrite
}
func createDeleteCmd() *cobra.Command {
var cmdDelete = &cobra.Command{
Use: "delete [yaml_file] [path]",
Aliases: []string{"d"},
Short: "yq d [--inplace/-i] [--doc/-d index] sample.yaml a.b.c",
Example: `
yq delete things.yaml a.b.c
yq delete --inplace things.yaml a.b.c
yq delete --inplace -- things.yaml --key-starting-with-dash
yq d -i things.yaml a.b.c
yq d things.yaml a.b.c
`,
Long: `Deletes the given path from the YAML file.
Outputs to STDOUT unless the inplace flag is used, in which case the file is updated instead.
`,
RunE: deleteProperty,
}
cmdDelete.PersistentFlags().BoolVarP(&writeInplace, "inplace", "i", false, "update the yaml file inplace")
2018-06-20 01:45:51 +00:00
cmdDelete.PersistentFlags().StringVarP(&docIndex, "doc", "d", "0", "process document index number (0 based, * for all documents)")
return cmdDelete
}
func createNewCmd() *cobra.Command {
var cmdNew = &cobra.Command{
Use: "new [path] [value]",
Aliases: []string{"n"},
Short: "yq n [--script/-s script_file] a.b.c newValue",
Example: `
2017-12-17 22:11:08 +00:00
yq new a.b.c cat
yq n a.b.c cat
yq n -- --key-starting-with-dash cat
2017-12-17 22:11:08 +00:00
yq n --script create_script.yaml
`,
Long: `Creates a new yaml w.r.t the given path and value.
Outputs to STDOUT
Create Scripts:
Note that you can give a create script to perform more sophisticated yaml. This follows the same format as the update script.
`,
RunE: newProperty,
}
cmdNew.PersistentFlags().StringVarP(&writeScript, "script", "s", "", "yaml script for updating yaml")
return cmdNew
}
func createMergeCmd() *cobra.Command {
var cmdMerge = &cobra.Command{
Use: "merge [initial_yaml_file] [additional_yaml_file]...",
Aliases: []string{"m"},
2018-07-07 05:26:56 +00:00
Short: "yq m [--inplace/-i] [--doc/-d index] [--overwrite/-x] [--append/-a] sample.yaml sample2.yaml",
Example: `
2017-12-17 22:11:08 +00:00
yq merge things.yaml other.yaml
yq merge --inplace things.yaml other.yaml
yq m -i things.yaml other.yaml
yq m --overwrite things.yaml other.yaml
yq m -i -x things.yaml other.yaml
2018-07-07 05:26:56 +00:00
yq m -i -a things.yaml other.yaml
`,
Long: `Updates the yaml file by adding/updating the path(s) and value(s) from additional yaml file(s).
Outputs to STDOUT unless the inplace flag is used, in which case the file is updated instead.
If overwrite flag is set then existing values will be overwritten using the values from each additional yaml file.
2018-07-07 05:26:56 +00:00
If append flag is set then existing arrays will be merged with the arrays from each additional yaml file.
Note that if you set both flags only overwrite will take effect.
`,
RunE: mergeProperties,
}
cmdMerge.PersistentFlags().BoolVarP(&writeInplace, "inplace", "i", false, "update the yaml file inplace")
cmdMerge.PersistentFlags().BoolVarP(&overwriteFlag, "overwrite", "x", false, "update the yaml file by overwriting existing values")
2018-07-07 05:26:56 +00:00
cmdMerge.PersistentFlags().BoolVarP(&appendFlag, "append", "a", false, "update the yaml file by appending array values")
cmdMerge.PersistentFlags().BoolVarP(&allowEmptyFlag, "allow-empty", "e", false, "allow empty yaml files")
2018-06-20 01:45:51 +00:00
cmdMerge.PersistentFlags().StringVarP(&docIndex, "doc", "d", "0", "process document index number (0 based, * for all documents)")
return cmdMerge
}
func readProperty(cmd *cobra.Command, args []string) error {
2017-08-03 07:30:07 +00:00
var path = ""
if len(args) < 1 {
2018-07-08 11:47:01 +00:00
return errors.New("Must provide filename")
} else if len(args) > 1 {
2017-08-03 07:30:07 +00:00
path = args[1]
}
2018-07-08 11:47:01 +00:00
var updateAll, docIndexInt, errorParsingDocIndex = parseDocumentIndex()
if errorParsingDocIndex != nil {
return errorParsingDocIndex
2018-06-20 01:45:51 +00:00
}
2018-07-08 11:47:01 +00:00
var mappedDocs []interface{}
var dataBucket interface{}
var currentIndex = 0
var errorReadingStream = readStream(args[0], func(decoder *yaml.Decoder) error {
for {
errorReading := decoder.Decode(&dataBucket)
if errorReading == io.EOF {
log.Debugf("done %v / %v", currentIndex, docIndexInt)
if !updateAll && currentIndex <= docIndexInt {
2019-01-20 22:33:14 +00:00
return fmt.Errorf("asked to process document index %v but there are only %v document(s)", docIndex, currentIndex)
2018-07-08 11:47:01 +00:00
}
return nil
}
2018-07-08 11:57:56 +00:00
log.Debugf("processing %v - requested index %v", currentIndex, docIndexInt)
2018-07-08 11:47:01 +00:00
if updateAll || currentIndex == docIndexInt {
2018-07-08 11:57:56 +00:00
log.Debugf("reading %v in index %v", path, currentIndex)
2018-07-08 11:47:01 +00:00
mappedDoc, errorParsing := readPath(dataBucket, path)
log.Debugf("%v", mappedDoc)
if errorParsing != nil {
return errors.Wrapf(errorParsing, "Error reading path in document index %v", currentIndex)
}
mappedDocs = append(mappedDocs, mappedDoc)
}
currentIndex = currentIndex + 1
}
})
if errorReadingStream != nil {
return errorReadingStream
}
2018-07-08 11:47:01 +00:00
if !updateAll {
dataBucket = mappedDocs[0]
} else {
dataBucket = mappedDocs
2015-09-29 06:29:32 +00:00
}
2018-07-08 11:47:01 +00:00
dataStr, err := toString(dataBucket)
if err != nil {
return err
}
cmd.Println(dataStr)
return nil
}
func readPath(dataBucket interface{}, path string) (interface{}, error) {
if path == "" {
log.Debug("no path")
return dataBucket, nil
}
2017-08-03 07:30:07 +00:00
var paths = parsePath(path)
2018-07-08 11:47:01 +00:00
return recurse(dataBucket, paths[0], paths[1:])
}
func newProperty(cmd *cobra.Command, args []string) error {
updatedData, err := newYaml(args)
if err != nil {
return err
}
dataStr, err := toString(updatedData)
if err != nil {
return err
}
cmd.Println(dataStr)
return nil
}
func newYaml(args []string) (interface{}, error) {
2018-06-12 23:13:58 +00:00
var writeCommands, writeCommandsError = readWriteCommands(args, 2, "Must provide <path_to_update> <value>")
if writeCommandsError != nil {
return nil, writeCommandsError
}
2018-06-15 06:21:18 +00:00
var dataBucket interface{}
2017-08-08 07:04:30 +00:00
var isArray = strings.HasPrefix(writeCommands[0].Key.(string), "[")
if isArray {
2018-06-15 06:21:18 +00:00
dataBucket = make([]interface{}, 0)
2017-08-08 07:04:30 +00:00
} else {
2018-06-15 06:21:18 +00:00
dataBucket = make(yaml.MapSlice, 0)
2017-08-08 07:04:30 +00:00
}
2018-06-15 06:21:18 +00:00
for _, entry := range writeCommands {
path := entry.Key.(string)
value := entry.Value
log.Debugf("setting %v to %v", path, value)
var paths = parsePath(path)
dataBucket = updatedChildValue(dataBucket, paths, value)
}
return dataBucket, nil
}
2018-06-20 01:45:51 +00:00
func parseDocumentIndex() (bool, int, error) {
if docIndex == "*" {
return true, -1, nil
}
docIndexInt64, err := strconv.ParseInt(docIndex, 10, 32)
if err != nil {
return false, -1, errors.Wrapf(err, "Document index %v is not a integer or *", docIndex)
}
return false, int(docIndexInt64), nil
}
2018-06-15 06:40:52 +00:00
type updateDataFn func(dataBucket interface{}, currentIndex int) (interface{}, error)
2018-06-14 23:43:20 +00:00
func mapYamlDecoder(updateData updateDataFn, encoder *yaml.Encoder) yamlDecoderFn {
return func(decoder *yaml.Decoder) error {
var dataBucket interface{}
var errorReading error
var errorWriting error
2018-06-15 06:40:52 +00:00
var errorUpdating error
var currentIndex = 0
2018-06-20 01:45:51 +00:00
var updateAll, docIndexInt, errorParsingDocIndex = parseDocumentIndex()
if errorParsingDocIndex != nil {
return errorParsingDocIndex
}
for {
log.Debugf("Read doc %v", currentIndex)
errorReading = decoder.Decode(&dataBucket)
if errorReading == io.EOF {
2018-07-08 11:47:01 +00:00
if !updateAll && currentIndex <= docIndexInt {
2019-01-20 22:33:14 +00:00
return fmt.Errorf("asked to process document index %v but there are only %v document(s)", docIndex, currentIndex)
}
return nil
} else if errorReading != nil {
2018-06-15 06:11:13 +00:00
return errors.Wrapf(errorReading, "Error reading document at index %v, %v", currentIndex, errorReading)
}
2018-06-15 06:40:52 +00:00
dataBucket, errorUpdating = updateData(dataBucket, currentIndex)
if errorUpdating != nil {
return errors.Wrapf(errorUpdating, "Error updating document at index %v", currentIndex)
}
errorWriting = encoder.Encode(dataBucket)
if errorWriting != nil {
2018-06-15 06:11:13 +00:00
return errors.Wrapf(errorWriting, "Error writing document at index %v, %v", currentIndex, errorWriting)
}
currentIndex = currentIndex + 1
}
}
}
func writeProperty(cmd *cobra.Command, args []string) error {
var writeCommands, writeCommandsError = readWriteCommands(args, 3, "Must provide <filename> <path_to_update> <value>")
if writeCommandsError != nil {
return writeCommandsError
2017-04-11 23:16:54 +00:00
}
2018-06-20 01:45:51 +00:00
var updateAll, docIndexInt, errorParsingDocIndex = parseDocumentIndex()
if errorParsingDocIndex != nil {
return errorParsingDocIndex
}
2018-06-15 06:40:52 +00:00
var updateData = func(dataBucket interface{}, currentIndex int) (interface{}, error) {
2018-06-20 01:45:51 +00:00
if updateAll || currentIndex == docIndexInt {
2018-06-14 23:43:20 +00:00
log.Debugf("Updating doc %v", currentIndex)
for _, entry := range writeCommands {
path := entry.Key.(string)
value := entry.Value
log.Debugf("setting %v to %v", path, value)
var paths = parsePath(path)
dataBucket = updatedChildValue(dataBucket, paths, value)
}
}
2018-06-15 06:40:52 +00:00
return dataBucket, nil
2018-06-14 23:43:20 +00:00
}
return readAndUpdate(cmd.OutOrStdout(), args[0], updateData)
}
2018-11-18 15:35:28 +00:00
func prefixProperty(cmd *cobra.Command, args []string) error {
if len(args) != 2 {
return errors.New("Must provide <filename> <prefixed_path>")
}
var updateAll, docIndexInt, errorParsingDocIndex = parseDocumentIndex()
if errorParsingDocIndex != nil {
return errorParsingDocIndex
}
var paths = parsePath(args[1])
// Inverse order
for i := len(paths)/2 - 1; i >= 0; i-- {
opp := len(paths) - 1 - i
paths[i], paths[opp] = paths[opp], paths[i]
}
var updateData = func(dataBucket interface{}, currentIndex int) (interface{}, error) {
if updateAll || currentIndex == docIndexInt {
log.Debugf("Prefixing %v to doc %v", paths, currentIndex)
var mapDataBucket = dataBucket
for _, key := range paths {
2018-11-19 22:47:17 +00:00
singlePath := []string{key}
mapDataBucket = updatedChildValue(nil, singlePath, mapDataBucket)
2018-11-18 15:35:28 +00:00
}
return mapDataBucket, nil
}
return dataBucket, nil
}
return readAndUpdate(cmd.OutOrStdout(), args[0], updateData)
}
2018-06-14 23:43:20 +00:00
func readAndUpdate(stdOut io.Writer, inputFile string, updateData updateDataFn) error {
var destination io.Writer
var destinationName string
if writeInplace {
info, err := os.Stat(inputFile)
if err != nil {
return err
}
tempFile, err := ioutil.TempFile("", "temp")
if err != nil {
return err
}
destinationName = tempFile.Name()
err = os.Chmod(destinationName, info.Mode())
if err != nil {
return err
}
destination = tempFile
defer func() {
safelyCloseFile(tempFile)
safelyRenameFile(tempFile.Name(), inputFile)
}()
} else {
2018-06-14 23:43:20 +00:00
var writer = bufio.NewWriter(stdOut)
destination = writer
destinationName = "Stdout"
defer safelyFlush(writer)
}
var encoder = yaml.NewEncoder(destination)
log.Debugf("Writing to %v from %v", destinationName, inputFile)
2018-06-14 23:43:20 +00:00
return readStream(inputFile, mapYamlDecoder(updateData, encoder))
}
func deleteProperty(cmd *cobra.Command, args []string) error {
if len(args) < 2 {
2018-06-14 23:43:20 +00:00
return errors.New("Must provide <filename> <path_to_delete>")
}
2018-06-14 23:43:20 +00:00
var deletePath = args[1]
var paths = parsePath(deletePath)
2018-06-20 01:45:51 +00:00
var updateAll, docIndexInt, errorParsingDocIndex = parseDocumentIndex()
if errorParsingDocIndex != nil {
return errorParsingDocIndex
}
2018-06-15 06:40:52 +00:00
var updateData = func(dataBucket interface{}, currentIndex int) (interface{}, error) {
2018-06-20 01:45:51 +00:00
if updateAll || currentIndex == docIndexInt {
2018-06-15 06:40:52 +00:00
log.Debugf("Deleting path in doc %v", currentIndex)
2019-05-14 01:20:41 +00:00
return deleteChildValue(dataBucket, paths)
2018-06-14 23:43:20 +00:00
}
2018-06-15 06:40:52 +00:00
return dataBucket, nil
}
2018-06-14 23:43:20 +00:00
return readAndUpdate(cmd.OutOrStdout(), args[0], updateData)
}
func mergeProperties(cmd *cobra.Command, args []string) error {
if len(args) < 2 {
return errors.New("Must provide at least 2 yaml files")
}
2018-06-15 06:40:52 +00:00
var input = args[0]
var filesToMerge = args[1:]
2018-06-20 01:45:51 +00:00
var updateAll, docIndexInt, errorParsingDocIndex = parseDocumentIndex()
if errorParsingDocIndex != nil {
return errorParsingDocIndex
}
2018-06-15 06:40:52 +00:00
var updateData = func(dataBucket interface{}, currentIndex int) (interface{}, error) {
2018-06-20 01:45:51 +00:00
if updateAll || currentIndex == docIndexInt {
2018-06-15 06:40:52 +00:00
log.Debugf("Merging doc %v", currentIndex)
var mergedData map[interface{}]interface{}
2018-07-07 05:26:56 +00:00
// merge only works for maps, so put everything in a temporary
// map
var mapDataBucket = make(map[interface{}]interface{})
mapDataBucket["root"] = dataBucket
if err := merge(&mergedData, mapDataBucket, overwriteFlag, appendFlag); err != nil {
2018-06-15 06:40:52 +00:00
return nil, err
}
for _, f := range filesToMerge {
var fileToMerge interface{}
if err := readData(f, 0, &fileToMerge); err != nil {
if allowEmptyFlag && err == io.EOF {
continue
}
2018-06-15 06:40:52 +00:00
return nil, err
}
2018-07-07 05:26:56 +00:00
mapDataBucket["root"] = fileToMerge
if err := merge(&mergedData, mapDataBucket, overwriteFlag, appendFlag); err != nil {
2018-06-15 06:40:52 +00:00
return nil, err
}
}
2018-07-07 05:26:56 +00:00
return mergedData["root"], nil
}
2018-06-15 06:40:52 +00:00
return dataBucket, nil
}
2018-06-15 06:40:52 +00:00
yaml.DefaultMapType = reflect.TypeOf(map[interface{}]interface{}{})
defer func() { yaml.DefaultMapType = reflect.TypeOf(yaml.MapSlice{}) }()
return readAndUpdate(cmd.OutOrStdout(), input, updateData)
}
2018-06-12 23:13:58 +00:00
func readWriteCommands(args []string, expectedArgs int, badArgsMessage string) (yaml.MapSlice, error) {
var writeCommands yaml.MapSlice
if writeScript != "" {
if err := readData(writeScript, 0, &writeCommands); err != nil {
return nil, err
}
2018-06-12 23:13:58 +00:00
} else if len(args) < expectedArgs {
return nil, errors.New(badArgsMessage)
} else {
writeCommands = make(yaml.MapSlice, 1)
2018-06-12 23:13:58 +00:00
writeCommands[0] = yaml.MapItem{Key: args[expectedArgs-2], Value: parseValue(args[expectedArgs-1])}
2015-09-29 06:29:32 +00:00
}
2018-06-12 23:13:58 +00:00
return writeCommands, nil
}
2015-09-29 06:29:32 +00:00
func parseValue(argument string) interface{} {
var value, err interface{}
var inQuotes = len(argument) > 0 && argument[0] == '"'
2015-10-03 07:25:13 +00:00
if !inQuotes {
value, err = strconv.ParseFloat(argument, 64)
if err == nil {
return value
}
value, err = strconv.ParseBool(argument)
if err == nil {
return value
}
if argument == "[]" {
return make([]interface{}, 0)
}
2015-10-03 07:25:13 +00:00
return argument
}
2015-10-03 07:25:13 +00:00
return argument[1 : len(argument)-1]
}
func toString(context interface{}) (string, error) {
2015-10-10 23:00:22 +00:00
if outputToJSON {
return jsonToString(context)
2015-10-10 23:00:22 +00:00
}
return yamlToString(context)
}
func yamlToString(context interface{}) (string, error) {
2019-01-20 22:33:14 +00:00
switch context := context.(type) {
case string:
2019-01-20 22:33:14 +00:00
return context, nil
default:
return marshalContext(context)
}
}
func marshalContext(context interface{}) (string, error) {
2015-09-29 01:05:28 +00:00
out, err := yaml.Marshal(context)
2015-09-29 01:05:28 +00:00
if err != nil {
2018-06-15 06:11:13 +00:00
return "", errors.Wrap(err, "error printing yaml")
2015-09-29 01:05:28 +00:00
}
2015-09-29 06:29:32 +00:00
outStr := string(out)
// trim the trailing new line as it's easier for a script to add
// it in if required than to remove it
2015-10-06 05:07:09 +00:00
if trimOutput {
return strings.Trim(outStr, "\n "), nil
2017-08-03 07:30:07 +00:00
}
return outStr, nil
2017-08-03 07:30:07 +00:00
}
func safelyRenameFile(from string, to string) {
2018-06-27 02:06:31 +00:00
if renameError := os.Rename(from, to); renameError != nil {
log.Warningf("Error renaming from %v to %v, attemting to copy contents", from, to)
log.Warning(renameError.Error())
// can't do this rename when running in docker to a file targeted in a mounted volume,
// so gracefully degrade to copying the entire contents.
if copyError := copyFileContents(from, to); copyError != nil {
log.Errorf("Failed copying from %v to %v", from, to)
log.Errorf(copyError.Error())
}
}
}
// thanks https://stackoverflow.com/questions/21060945/simple-way-to-copy-a-file-in-golang
func copyFileContents(src, dst string) (err error) {
2018-08-06 06:24:06 +00:00
in, err := os.Open(src) // nolint gosec
2018-06-27 02:06:31 +00:00
if err != nil {
return err
}
defer safelyCloseFile(in)
out, err := os.Create(dst)
if err != nil {
return err
}
defer safelyCloseFile(out)
if _, err = io.Copy(out, in); err != nil {
return err
}
2018-06-27 02:06:31 +00:00
return out.Sync()
}
func safelyFlush(writer *bufio.Writer) {
if err := writer.Flush(); err != nil {
log.Error("Error flushing writer!")
log.Error(err.Error())
}
}
2018-06-12 05:33:59 +00:00
func safelyCloseFile(file *os.File) {
err := file.Close()
if err != nil {
log.Error("Error closing file!")
log.Error(err.Error())
2018-06-12 05:33:59 +00:00
}
}
type yamlDecoderFn func(*yaml.Decoder) error
func readStream(filename string, yamlDecoder yamlDecoderFn) error {
if filename == "" {
return errors.New("Must provide filename")
2015-09-28 02:00:38 +00:00
}
2015-10-05 04:48:34 +00:00
2018-06-12 05:33:59 +00:00
var stream io.Reader
if filename == "-" {
2018-06-12 05:33:59 +00:00
stream = bufio.NewReader(os.Stdin)
2015-10-05 04:48:34 +00:00
} else {
2018-08-06 06:24:06 +00:00
file, err := os.Open(filename) // nolint gosec
2018-06-12 05:33:59 +00:00
if err != nil {
return err
}
defer safelyCloseFile(file)
stream = file
2015-10-05 04:48:34 +00:00
}
return yamlDecoder(yaml.NewDecoder(stream))
}
2015-09-26 22:20:42 +00:00
func readData(filename string, indexToRead int, parsedData interface{}) error {
return readStream(filename, func(decoder *yaml.Decoder) error {
for currentIndex := 0; currentIndex < indexToRead; currentIndex++ {
errorSkipping := decoder.Decode(parsedData)
if errorSkipping != nil {
2018-06-15 06:11:13 +00:00
return errors.Wrapf(errorSkipping, "Error processing document at index %v, %v", currentIndex, errorSkipping)
}
2018-06-12 05:33:59 +00:00
}
return decoder.Decode(parsedData)
})
2015-10-06 05:39:19 +00:00
}