mirror of
https://github.com/mikefarah/yq.git
synced 2024-11-12 05:38:04 +00:00
Added tonumber support #71
This commit is contained in:
parent
6e65d44a98
commit
d113344abf
2
pkg/yqlib/doc/operators/headers/to_number.md
Normal file
2
pkg/yqlib/doc/operators/headers/to_number.md
Normal file
@ -0,0 +1,2 @@
|
||||
# To Number
|
||||
Parses the input as a number. yq will try to parse values as an int first, failing that it will try float. Values that already ints or floats will be left alone.
|
49
pkg/yqlib/doc/operators/to_number.md
Normal file
49
pkg/yqlib/doc/operators/to_number.md
Normal file
@ -0,0 +1,49 @@
|
||||
# To Number
|
||||
Parses the input as a number. yq will try to parse values as an int first, failing that it will try float. Values that already ints or floats will be left alone.
|
||||
|
||||
## Converts strings to numbers
|
||||
Given a sample.yml file of:
|
||||
```yaml
|
||||
- "3"
|
||||
- "3.1"
|
||||
- "-1e3"
|
||||
```
|
||||
then
|
||||
```bash
|
||||
yq '.[] | to_number' sample.yml
|
||||
```
|
||||
will output
|
||||
```yaml
|
||||
3
|
||||
3.1
|
||||
-1e3
|
||||
```
|
||||
|
||||
## Doesn't change numbers
|
||||
Given a sample.yml file of:
|
||||
```yaml
|
||||
- 3
|
||||
- 3.1
|
||||
- -1e3
|
||||
```
|
||||
then
|
||||
```bash
|
||||
yq '.[] | to_number' sample.yml
|
||||
```
|
||||
will output
|
||||
```yaml
|
||||
3
|
||||
3.1
|
||||
-1e3
|
||||
```
|
||||
|
||||
## Cannot convert null
|
||||
Running
|
||||
```bash
|
||||
yq --null-input '.a.b | to_number'
|
||||
```
|
||||
will output
|
||||
```bash
|
||||
Error: cannot convert node value [null] at path a.b of tag !!null to number
|
||||
```
|
||||
|
@ -34,6 +34,7 @@ var participleYqRules = []*participleYqRule{
|
||||
simpleOp("line", lineOpType),
|
||||
simpleOp("column", columnOpType),
|
||||
simpleOp("eval", evalOpType),
|
||||
simpleOp("to_?number", toNumberOpType),
|
||||
|
||||
{"MapValues", `map_?values`, opToken(mapValuesOpType), 0},
|
||||
simpleOp("map", mapOpType),
|
||||
|
@ -167,6 +167,7 @@ var valueOpType = &operationType{Type: "VALUE", NumArgs: 0, Precedence: 50, Hand
|
||||
var referenceOpType = &operationType{Type: "REF", NumArgs: 0, Precedence: 50, Handler: referenceOperator}
|
||||
var envOpType = &operationType{Type: "ENV", NumArgs: 0, Precedence: 50, Handler: envOperator}
|
||||
var notOpType = &operationType{Type: "NOT", NumArgs: 0, Precedence: 50, Handler: notOperator}
|
||||
var toNumberOpType = &operationType{Type: "TO_NUMBER", NumArgs: 0, Precedence: 50, Handler: toNumberOperator}
|
||||
var emptyOpType = &operationType{Type: "EMPTY", Precedence: 50, Handler: emptyOperator}
|
||||
|
||||
var envsubstOpType = &operationType{Type: "ENVSUBST", NumArgs: 0, Precedence: 50, Handler: envsubstOperator}
|
||||
|
56
pkg/yqlib/operator_to_number.go
Normal file
56
pkg/yqlib/operator_to_number.go
Normal file
@ -0,0 +1,56 @@
|
||||
package yqlib
|
||||
|
||||
import (
|
||||
"container/list"
|
||||
"fmt"
|
||||
"strconv"
|
||||
|
||||
yaml "gopkg.in/yaml.v3"
|
||||
)
|
||||
|
||||
func tryConvertToNumber(value string) (string, bool) {
|
||||
// try a int first
|
||||
_, _, err := parseInt64(value)
|
||||
if err == nil {
|
||||
return "!!int", true
|
||||
}
|
||||
// try float
|
||||
_, floatErr := strconv.ParseFloat(value, 64)
|
||||
|
||||
if floatErr == nil {
|
||||
return "!!float", true
|
||||
}
|
||||
return "", false
|
||||
|
||||
}
|
||||
|
||||
func toNumberOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
||||
log.Debugf("ToNumberOperator")
|
||||
|
||||
var results = list.New()
|
||||
|
||||
for el := context.MatchingNodes.Front(); el != nil; el = el.Next() {
|
||||
candidate := el.Value.(*CandidateNode)
|
||||
if candidate.Node.Kind != yaml.ScalarNode {
|
||||
return Context{}, fmt.Errorf("cannot convert node at path %v of tag %v to number", candidate.GetNicePath(), candidate.GetNiceTag())
|
||||
}
|
||||
|
||||
if candidate.Node.Tag == "!!int" || candidate.Node.Tag == "!!float" {
|
||||
// it already is a number!
|
||||
results.PushBack(candidate)
|
||||
} else {
|
||||
tag, converted := tryConvertToNumber(candidate.Node.Value)
|
||||
if converted {
|
||||
node := &yaml.Node{Kind: yaml.ScalarNode, Value: candidate.Node.Value, Tag: tag}
|
||||
|
||||
result := candidate.CreateReplacement(node)
|
||||
results.PushBack(result)
|
||||
} else {
|
||||
return Context{}, fmt.Errorf("cannot convert node value [%v] at path %v of tag %v to number", candidate.Node.Value, candidate.GetNicePath(), candidate.GetNiceTag())
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
return context.ChildContext(results), nil
|
||||
}
|
51
pkg/yqlib/operator_to_number_test.go
Normal file
51
pkg/yqlib/operator_to_number_test.go
Normal file
@ -0,0 +1,51 @@
|
||||
package yqlib
|
||||
|
||||
import (
|
||||
"testing"
|
||||
)
|
||||
|
||||
var toNumberScenarios = []expressionScenario{
|
||||
{
|
||||
description: "Converts strings to numbers",
|
||||
document: `["3", "3.1", "-1e3"]`,
|
||||
expression: `.[] | to_number`,
|
||||
expected: []string{
|
||||
"D0, P[0], (!!int)::3\n",
|
||||
"D0, P[1], (!!float)::3.1\n",
|
||||
"D0, P[2], (!!float)::-1e3\n",
|
||||
},
|
||||
},
|
||||
{
|
||||
skipDoc: true,
|
||||
description: "Converts strings to numbers, with tonumber because jq",
|
||||
document: `["3", "3.1", "-1e3"]`,
|
||||
expression: `.[] | tonumber`,
|
||||
expected: []string{
|
||||
"D0, P[0], (!!int)::3\n",
|
||||
"D0, P[1], (!!float)::3.1\n",
|
||||
"D0, P[2], (!!float)::-1e3\n",
|
||||
},
|
||||
},
|
||||
{
|
||||
description: "Doesn't change numbers",
|
||||
document: `[3, 3.1, -1e3]`,
|
||||
expression: `.[] | to_number`,
|
||||
expected: []string{
|
||||
"D0, P[0], (!!int)::3\n",
|
||||
"D0, P[1], (!!float)::3.1\n",
|
||||
"D0, P[2], (!!float)::-1e3\n",
|
||||
},
|
||||
},
|
||||
{
|
||||
description: "Cannot convert null",
|
||||
expression: `.a.b | to_number`,
|
||||
expectedError: "cannot convert node value [null] at path a.b of tag !!null to number",
|
||||
},
|
||||
}
|
||||
|
||||
func TestToNumberOperatorScenarios(t *testing.T) {
|
||||
for _, tt := range toNumberScenarios {
|
||||
testScenario(t, &tt)
|
||||
}
|
||||
documentOperatorScenarios(t, "to_number", toNumberScenarios)
|
||||
}
|
@ -249,3 +249,4 @@ yamld
|
||||
yqlib
|
||||
yuin
|
||||
zabbix
|
||||
tonumber
|
Loading…
Reference in New Issue
Block a user