diff --git a/ast/ast.go b/ast/ast.go index 1ad8f15b..53ad0daa 100644 --- a/ast/ast.go +++ b/ast/ast.go @@ -301,12 +301,12 @@ func checkLineBreak(t *token.Token) bool { } // Due to the way that comment parsing works its assumed that when a null value does not have new line in origin // it was squashed therefore difference is ignored. - //foo: + // foo: // bar: // # comment // baz: 1 - //becomes - //foo: + // becomes + // foo: // bar: null # comment // // baz: 1 diff --git a/token/token.go b/token/token.go index b880bd28..21464cf6 100644 --- a/token/token.go +++ b/token/token.go @@ -1,6 +1,7 @@ package token import ( + "errors" "fmt" "strconv" "strings" @@ -539,15 +540,35 @@ type NumberValue struct { } func ToNumber(value string) *NumberValue { - if len(value) == 0 { + num, err := toNumber(value) + if err != nil { return nil } + return num +} + +func isNumber(value string) bool { + num, err := toNumber(value) + if err != nil { + var numErr *strconv.NumError + if errors.As(err, &numErr) && errors.Is(numErr.Err, strconv.ErrRange) { + return true + } + return false + } + return num != nil +} + +func toNumber(value string) (*NumberValue, error) { + if len(value) == 0 { + return nil, nil + } if strings.HasPrefix(value, "_") { - return nil + return nil, nil } dotCount := strings.Count(value, ".") if dotCount > 1 { - return nil + return nil, nil } isNegative := strings.HasPrefix(value, "-") @@ -589,19 +610,19 @@ func ToNumber(value string) *NumberValue { if typ == NumberTypeFloat { f, err := strconv.ParseFloat(text, 64) if err != nil { - return nil + return nil, err } v = f } else if isNegative { i, err := strconv.ParseInt(text, base, 64) if err != nil { - return nil + return nil, err } v = i } else { u, err := strconv.ParseUint(text, base, 64) if err != nil { - return nil + return nil, err } v = u } @@ -610,7 +631,7 @@ func ToNumber(value string) *NumberValue { Type: typ, Value: v, Text: text, - } + }, nil } // This is a subset of the formats permitted by the regular expression @@ -635,7 +656,7 @@ func isTimestamp(value string) bool { return false } -// IsNeedQuoted whether need quote for passed string or not +// IsNeedQuoted checks whether the value needs quote for passed string or not func IsNeedQuoted(value string) bool { if value == "" { return true @@ -643,7 +664,7 @@ func IsNeedQuoted(value string) bool { if _, exists := reservedEncKeywordMap[value]; exists { return true } - if num := ToNumber(value); num != nil { + if isNumber(value) { return true } first := value[0] diff --git a/token/token_test.go b/token/token_test.go index e5e5ba8f..3b339b4b 100644 --- a/token/token_test.go +++ b/token/token_test.go @@ -81,6 +81,10 @@ func TestIsNeedQuoted(t *testing.T) { "", "true", "1.234", + "0b11111111111111111111111111111111111111111111111111111111111111111", + "0o7777777777777777777777777777777777777777", + "999999999999999999999999999999999999999999", + "0xffffffffffffffffffffffffffffffffffffffff", "1:1", "2001-12-15T02:59:43.1Z", "2001-12-14t21:59:43.10-05:00",