|  | package toml | 
|  |  | 
|  | import ( | 
|  | "strings" | 
|  | "testing" | 
|  | ) | 
|  |  | 
|  | func testFlow(t *testing.T, input string, expectedFlow []token) { | 
|  | ch := lexToml(strings.NewReader(input)) | 
|  | for _, expected := range expectedFlow { | 
|  | token := <-ch | 
|  | if token != expected { | 
|  | t.Log("While testing: ", input) | 
|  | t.Log("compared (got)", token, "to (expected)", expected) | 
|  | t.Log("\tvalue:", token.val, "<->", expected.val) | 
|  | t.Log("\tvalue as bytes:", []byte(token.val), "<->", []byte(expected.val)) | 
|  | t.Log("\ttype:", token.typ.String(), "<->", expected.typ.String()) | 
|  | t.Log("\tline:", token.Line, "<->", expected.Line) | 
|  | t.Log("\tcolumn:", token.Col, "<->", expected.Col) | 
|  | t.Log("compared", token, "to", expected) | 
|  | t.FailNow() | 
|  | } | 
|  | } | 
|  |  | 
|  | tok, ok := <-ch | 
|  | if ok { | 
|  | t.Log("channel is not closed!") | 
|  | t.Log(len(ch)+1, "tokens remaining:") | 
|  |  | 
|  | t.Log("token ->", tok) | 
|  | for token := range ch { | 
|  | t.Log("token ->", token) | 
|  | } | 
|  | t.FailNow() | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestValidKeyGroup(t *testing.T) { | 
|  | testFlow(t, "[hello world]", []token{ | 
|  | {Position{1, 1}, tokenLeftBracket, "["}, | 
|  | {Position{1, 2}, tokenKeyGroup, "hello world"}, | 
|  | {Position{1, 13}, tokenRightBracket, "]"}, | 
|  | {Position{1, 14}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestNestedQuotedUnicodeKeyGroup(t *testing.T) { | 
|  | testFlow(t, `[ j . "ʞ" . l ]`, []token{ | 
|  | {Position{1, 1}, tokenLeftBracket, "["}, | 
|  | {Position{1, 2}, tokenKeyGroup, ` j . "ʞ" . l `}, | 
|  | {Position{1, 15}, tokenRightBracket, "]"}, | 
|  | {Position{1, 16}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestUnclosedKeyGroup(t *testing.T) { | 
|  | testFlow(t, "[hello world", []token{ | 
|  | {Position{1, 1}, tokenLeftBracket, "["}, | 
|  | {Position{1, 2}, tokenError, "unclosed table key"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestComment(t *testing.T) { | 
|  | testFlow(t, "# blahblah", []token{ | 
|  | {Position{1, 11}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyGroupComment(t *testing.T) { | 
|  | testFlow(t, "[hello world] # blahblah", []token{ | 
|  | {Position{1, 1}, tokenLeftBracket, "["}, | 
|  | {Position{1, 2}, tokenKeyGroup, "hello world"}, | 
|  | {Position{1, 13}, tokenRightBracket, "]"}, | 
|  | {Position{1, 25}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestMultipleKeyGroupsComment(t *testing.T) { | 
|  | testFlow(t, "[hello world] # blahblah\n[test]", []token{ | 
|  | {Position{1, 1}, tokenLeftBracket, "["}, | 
|  | {Position{1, 2}, tokenKeyGroup, "hello world"}, | 
|  | {Position{1, 13}, tokenRightBracket, "]"}, | 
|  | {Position{2, 1}, tokenLeftBracket, "["}, | 
|  | {Position{2, 2}, tokenKeyGroup, "test"}, | 
|  | {Position{2, 6}, tokenRightBracket, "]"}, | 
|  | {Position{2, 7}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestSimpleWindowsCRLF(t *testing.T) { | 
|  | testFlow(t, "a=4\r\nb=2", []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 2}, tokenEqual, "="}, | 
|  | {Position{1, 3}, tokenInteger, "4"}, | 
|  | {Position{2, 1}, tokenKey, "b"}, | 
|  | {Position{2, 2}, tokenEqual, "="}, | 
|  | {Position{2, 3}, tokenInteger, "2"}, | 
|  | {Position{2, 4}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestBasicKey(t *testing.T) { | 
|  | testFlow(t, "hello", []token{ | 
|  | {Position{1, 1}, tokenKey, "hello"}, | 
|  | {Position{1, 6}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestBasicKeyWithUnderscore(t *testing.T) { | 
|  | testFlow(t, "hello_hello", []token{ | 
|  | {Position{1, 1}, tokenKey, "hello_hello"}, | 
|  | {Position{1, 12}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestBasicKeyWithDash(t *testing.T) { | 
|  | testFlow(t, "hello-world", []token{ | 
|  | {Position{1, 1}, tokenKey, "hello-world"}, | 
|  | {Position{1, 12}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestBasicKeyWithUppercaseMix(t *testing.T) { | 
|  | testFlow(t, "helloHELLOHello", []token{ | 
|  | {Position{1, 1}, tokenKey, "helloHELLOHello"}, | 
|  | {Position{1, 16}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestBasicKeyWithInternationalCharacters(t *testing.T) { | 
|  | testFlow(t, "héllÖ", []token{ | 
|  | {Position{1, 1}, tokenKey, "héllÖ"}, | 
|  | {Position{1, 6}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestBasicKeyAndEqual(t *testing.T) { | 
|  | testFlow(t, "hello =", []token{ | 
|  | {Position{1, 1}, tokenKey, "hello"}, | 
|  | {Position{1, 7}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyWithSharpAndEqual(t *testing.T) { | 
|  | testFlow(t, "key#name = 5", []token{ | 
|  | {Position{1, 1}, tokenError, "keys cannot contain # character"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyWithSymbolsAndEqual(t *testing.T) { | 
|  | testFlow(t, "~!@$^&*()_+-`1234567890[]\\|/?><.,;:' = 5", []token{ | 
|  | {Position{1, 1}, tokenError, "keys cannot contain ~ character"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyEqualStringEscape(t *testing.T) { | 
|  | testFlow(t, `foo = "hello\""`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenString, "hello\""}, | 
|  | {Position{1, 16}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyEqualStringUnfinished(t *testing.T) { | 
|  | testFlow(t, `foo = "bar`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenError, "unclosed string"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyEqualString(t *testing.T) { | 
|  | testFlow(t, `foo = "bar"`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenString, "bar"}, | 
|  | {Position{1, 12}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyEqualTrue(t *testing.T) { | 
|  | testFlow(t, "foo = true", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenTrue, "true"}, | 
|  | {Position{1, 11}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyEqualFalse(t *testing.T) { | 
|  | testFlow(t, "foo = false", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenFalse, "false"}, | 
|  | {Position{1, 12}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestArrayNestedString(t *testing.T) { | 
|  | testFlow(t, `a = [ ["hello", "world"] ]`, []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenLeftBracket, "["}, | 
|  | {Position{1, 7}, tokenLeftBracket, "["}, | 
|  | {Position{1, 9}, tokenString, "hello"}, | 
|  | {Position{1, 15}, tokenComma, ","}, | 
|  | {Position{1, 18}, tokenString, "world"}, | 
|  | {Position{1, 24}, tokenRightBracket, "]"}, | 
|  | {Position{1, 26}, tokenRightBracket, "]"}, | 
|  | {Position{1, 27}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestArrayNestedInts(t *testing.T) { | 
|  | testFlow(t, "a = [ [42, 21], [10] ]", []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenLeftBracket, "["}, | 
|  | {Position{1, 7}, tokenLeftBracket, "["}, | 
|  | {Position{1, 8}, tokenInteger, "42"}, | 
|  | {Position{1, 10}, tokenComma, ","}, | 
|  | {Position{1, 12}, tokenInteger, "21"}, | 
|  | {Position{1, 14}, tokenRightBracket, "]"}, | 
|  | {Position{1, 15}, tokenComma, ","}, | 
|  | {Position{1, 17}, tokenLeftBracket, "["}, | 
|  | {Position{1, 18}, tokenInteger, "10"}, | 
|  | {Position{1, 20}, tokenRightBracket, "]"}, | 
|  | {Position{1, 22}, tokenRightBracket, "]"}, | 
|  | {Position{1, 23}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestArrayInts(t *testing.T) { | 
|  | testFlow(t, "a = [ 42, 21, 10, ]", []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenLeftBracket, "["}, | 
|  | {Position{1, 7}, tokenInteger, "42"}, | 
|  | {Position{1, 9}, tokenComma, ","}, | 
|  | {Position{1, 11}, tokenInteger, "21"}, | 
|  | {Position{1, 13}, tokenComma, ","}, | 
|  | {Position{1, 15}, tokenInteger, "10"}, | 
|  | {Position{1, 17}, tokenComma, ","}, | 
|  | {Position{1, 19}, tokenRightBracket, "]"}, | 
|  | {Position{1, 20}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestMultilineArrayComments(t *testing.T) { | 
|  | testFlow(t, "a = [1, # wow\n2, # such items\n3, # so array\n]", []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenLeftBracket, "["}, | 
|  | {Position{1, 6}, tokenInteger, "1"}, | 
|  | {Position{1, 7}, tokenComma, ","}, | 
|  | {Position{2, 1}, tokenInteger, "2"}, | 
|  | {Position{2, 2}, tokenComma, ","}, | 
|  | {Position{3, 1}, tokenInteger, "3"}, | 
|  | {Position{3, 2}, tokenComma, ","}, | 
|  | {Position{4, 1}, tokenRightBracket, "]"}, | 
|  | {Position{4, 2}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestNestedArraysComment(t *testing.T) { | 
|  | toml := ` | 
|  | someArray = [ | 
|  | # does not work | 
|  | ["entry1"] | 
|  | ]` | 
|  | testFlow(t, toml, []token{ | 
|  | {Position{2, 1}, tokenKey, "someArray"}, | 
|  | {Position{2, 11}, tokenEqual, "="}, | 
|  | {Position{2, 13}, tokenLeftBracket, "["}, | 
|  | {Position{4, 1}, tokenLeftBracket, "["}, | 
|  | {Position{4, 3}, tokenString, "entry1"}, | 
|  | {Position{4, 10}, tokenRightBracket, "]"}, | 
|  | {Position{5, 1}, tokenRightBracket, "]"}, | 
|  | {Position{5, 2}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyEqualArrayBools(t *testing.T) { | 
|  | testFlow(t, "foo = [true, false, true]", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenLeftBracket, "["}, | 
|  | {Position{1, 8}, tokenTrue, "true"}, | 
|  | {Position{1, 12}, tokenComma, ","}, | 
|  | {Position{1, 14}, tokenFalse, "false"}, | 
|  | {Position{1, 19}, tokenComma, ","}, | 
|  | {Position{1, 21}, tokenTrue, "true"}, | 
|  | {Position{1, 25}, tokenRightBracket, "]"}, | 
|  | {Position{1, 26}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyEqualArrayBoolsWithComments(t *testing.T) { | 
|  | testFlow(t, "foo = [true, false, true] # YEAH", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenLeftBracket, "["}, | 
|  | {Position{1, 8}, tokenTrue, "true"}, | 
|  | {Position{1, 12}, tokenComma, ","}, | 
|  | {Position{1, 14}, tokenFalse, "false"}, | 
|  | {Position{1, 19}, tokenComma, ","}, | 
|  | {Position{1, 21}, tokenTrue, "true"}, | 
|  | {Position{1, 25}, tokenRightBracket, "]"}, | 
|  | {Position{1, 33}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestDateRegexp(t *testing.T) { | 
|  | if dateRegexp.FindString("1979-05-27T07:32:00Z") == "" { | 
|  | t.Error("basic lexing") | 
|  | } | 
|  | if dateRegexp.FindString("1979-05-27T00:32:00-07:00") == "" { | 
|  | t.Error("offset lexing") | 
|  | } | 
|  | if dateRegexp.FindString("1979-05-27T00:32:00.999999-07:00") == "" { | 
|  | t.Error("nano precision lexing") | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestKeyEqualDate(t *testing.T) { | 
|  | testFlow(t, "foo = 1979-05-27T07:32:00Z", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenDate, "1979-05-27T07:32:00Z"}, | 
|  | {Position{1, 27}, tokenEOF, ""}, | 
|  | }) | 
|  | testFlow(t, "foo = 1979-05-27T00:32:00-07:00", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenDate, "1979-05-27T00:32:00-07:00"}, | 
|  | {Position{1, 32}, tokenEOF, ""}, | 
|  | }) | 
|  | testFlow(t, "foo = 1979-05-27T00:32:00.999999-07:00", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenDate, "1979-05-27T00:32:00.999999-07:00"}, | 
|  | {Position{1, 39}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestFloatEndingWithDot(t *testing.T) { | 
|  | testFlow(t, "foo = 42.", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenError, "float cannot end with a dot"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestFloatWithTwoDots(t *testing.T) { | 
|  | testFlow(t, "foo = 4.2.", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenError, "cannot have two dots in one float"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestFloatWithExponent1(t *testing.T) { | 
|  | testFlow(t, "a = 5e+22", []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenFloat, "5e+22"}, | 
|  | {Position{1, 10}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestFloatWithExponent2(t *testing.T) { | 
|  | testFlow(t, "a = 5E+22", []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenFloat, "5E+22"}, | 
|  | {Position{1, 10}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestFloatWithExponent3(t *testing.T) { | 
|  | testFlow(t, "a = -5e+22", []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenFloat, "-5e+22"}, | 
|  | {Position{1, 11}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestFloatWithExponent4(t *testing.T) { | 
|  | testFlow(t, "a = -5e-22", []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenFloat, "-5e-22"}, | 
|  | {Position{1, 11}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestFloatWithExponent5(t *testing.T) { | 
|  | testFlow(t, "a = 6.626e-34", []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenFloat, "6.626e-34"}, | 
|  | {Position{1, 14}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestInvalidEsquapeSequence(t *testing.T) { | 
|  | testFlow(t, `foo = "\x"`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenError, "invalid escape sequence: \\x"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestNestedArrays(t *testing.T) { | 
|  | testFlow(t, "foo = [[[]]]", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenLeftBracket, "["}, | 
|  | {Position{1, 8}, tokenLeftBracket, "["}, | 
|  | {Position{1, 9}, tokenLeftBracket, "["}, | 
|  | {Position{1, 10}, tokenRightBracket, "]"}, | 
|  | {Position{1, 11}, tokenRightBracket, "]"}, | 
|  | {Position{1, 12}, tokenRightBracket, "]"}, | 
|  | {Position{1, 13}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyEqualNumber(t *testing.T) { | 
|  | testFlow(t, "foo = 42", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenInteger, "42"}, | 
|  | {Position{1, 9}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = +42", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenInteger, "+42"}, | 
|  | {Position{1, 10}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = -42", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenInteger, "-42"}, | 
|  | {Position{1, 10}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = 4.2", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenFloat, "4.2"}, | 
|  | {Position{1, 10}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = +4.2", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenFloat, "+4.2"}, | 
|  | {Position{1, 11}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = -4.2", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenFloat, "-4.2"}, | 
|  | {Position{1, 11}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = 1_000", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenInteger, "1_000"}, | 
|  | {Position{1, 12}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = 5_349_221", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenInteger, "5_349_221"}, | 
|  | {Position{1, 16}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = 1_2_3_4_5", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenInteger, "1_2_3_4_5"}, | 
|  | {Position{1, 16}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "flt8 = 9_224_617.445_991_228_313", []token{ | 
|  | {Position{1, 1}, tokenKey, "flt8"}, | 
|  | {Position{1, 6}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenFloat, "9_224_617.445_991_228_313"}, | 
|  | {Position{1, 33}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = +", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenError, "no digit in that number"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestMultiline(t *testing.T) { | 
|  | testFlow(t, "foo = 42\nbar=21", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 7}, tokenInteger, "42"}, | 
|  | {Position{2, 1}, tokenKey, "bar"}, | 
|  | {Position{2, 4}, tokenEqual, "="}, | 
|  | {Position{2, 5}, tokenInteger, "21"}, | 
|  | {Position{2, 7}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyEqualStringUnicodeEscape(t *testing.T) { | 
|  | testFlow(t, `foo = "hello \u2665"`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenString, "hello ♥"}, | 
|  | {Position{1, 21}, tokenEOF, ""}, | 
|  | }) | 
|  | testFlow(t, `foo = "hello \U000003B4"`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenString, "hello δ"}, | 
|  | {Position{1, 25}, tokenEOF, ""}, | 
|  | }) | 
|  | testFlow(t, `foo = "\u2"`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenError, "unfinished unicode escape"}, | 
|  | }) | 
|  | testFlow(t, `foo = "\U2"`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenError, "unfinished unicode escape"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyEqualStringNoEscape(t *testing.T) { | 
|  | testFlow(t, "foo = \"hello \u0002\"", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenError, "unescaped control character U+0002"}, | 
|  | }) | 
|  | testFlow(t, "foo = \"hello \u001F\"", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenError, "unescaped control character U+001F"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestLiteralString(t *testing.T) { | 
|  | testFlow(t, `foo = 'C:\Users\nodejs\templates'`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenString, `C:\Users\nodejs\templates`}, | 
|  | {Position{1, 34}, tokenEOF, ""}, | 
|  | }) | 
|  | testFlow(t, `foo = '\\ServerX\admin$\system32\'`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenString, `\\ServerX\admin$\system32\`}, | 
|  | {Position{1, 35}, tokenEOF, ""}, | 
|  | }) | 
|  | testFlow(t, `foo = 'Tom "Dubs" Preston-Werner'`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenString, `Tom "Dubs" Preston-Werner`}, | 
|  | {Position{1, 34}, tokenEOF, ""}, | 
|  | }) | 
|  | testFlow(t, `foo = '<\i\c*\s*>'`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenString, `<\i\c*\s*>`}, | 
|  | {Position{1, 19}, tokenEOF, ""}, | 
|  | }) | 
|  | testFlow(t, `foo = 'C:\Users\nodejs\unfinis`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenError, "unclosed string"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestMultilineLiteralString(t *testing.T) { | 
|  | testFlow(t, `foo = '''hello 'literal' world'''`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 10}, tokenString, `hello 'literal' world`}, | 
|  | {Position{1, 34}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = '''\nhello\n'literal'\nworld'''", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{2, 1}, tokenString, "hello\n'literal'\nworld"}, | 
|  | {Position{4, 9}, tokenEOF, ""}, | 
|  | }) | 
|  | testFlow(t, "foo = '''\r\nhello\r\n'literal'\r\nworld'''", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{2, 1}, tokenString, "hello\r\n'literal'\r\nworld"}, | 
|  | {Position{4, 9}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestMultilineString(t *testing.T) { | 
|  | testFlow(t, `foo = """hello "literal" world"""`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 10}, tokenString, `hello "literal" world`}, | 
|  | {Position{1, 34}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = \"\"\"\r\nhello\\\r\n\"literal\"\\\nworld\"\"\"", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{2, 1}, tokenString, "hello\"literal\"world"}, | 
|  | {Position{4, 9}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "foo = \"\"\"\\\n    \\\n    \\\n    hello\\\nmultiline\\\nworld\"\"\"", []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 10}, tokenString, "hellomultilineworld"}, | 
|  | {Position{6, 9}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "key2 = \"\"\"\nThe quick brown \\\n\n\n  fox jumps over \\\n    the lazy dog.\"\"\"", []token{ | 
|  | {Position{1, 1}, tokenKey, "key2"}, | 
|  | {Position{1, 6}, tokenEqual, "="}, | 
|  | {Position{2, 1}, tokenString, "The quick brown fox jumps over the lazy dog."}, | 
|  | {Position{6, 21}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "key2 = \"\"\"\\\n       The quick brown \\\n       fox jumps over \\\n       the lazy dog.\\\n       \"\"\"", []token{ | 
|  | {Position{1, 1}, tokenKey, "key2"}, | 
|  | {Position{1, 6}, tokenEqual, "="}, | 
|  | {Position{1, 11}, tokenString, "The quick brown fox jumps over the lazy dog."}, | 
|  | {Position{5, 11}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, `key2 = "Roses are red\nViolets are blue"`, []token{ | 
|  | {Position{1, 1}, tokenKey, "key2"}, | 
|  | {Position{1, 6}, tokenEqual, "="}, | 
|  | {Position{1, 9}, tokenString, "Roses are red\nViolets are blue"}, | 
|  | {Position{1, 41}, tokenEOF, ""}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, "key2 = \"\"\"\nRoses are red\nViolets are blue\"\"\"", []token{ | 
|  | {Position{1, 1}, tokenKey, "key2"}, | 
|  | {Position{1, 6}, tokenEqual, "="}, | 
|  | {Position{2, 1}, tokenString, "Roses are red\nViolets are blue"}, | 
|  | {Position{3, 20}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestUnicodeString(t *testing.T) { | 
|  | testFlow(t, `foo = "hello ♥ world"`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenString, "hello ♥ world"}, | 
|  | {Position{1, 22}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  | func TestEscapeInString(t *testing.T) { | 
|  | testFlow(t, `foo = "\b\f\/"`, []token{ | 
|  | {Position{1, 1}, tokenKey, "foo"}, | 
|  | {Position{1, 5}, tokenEqual, "="}, | 
|  | {Position{1, 8}, tokenString, "\b\f/"}, | 
|  | {Position{1, 15}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyGroupArray(t *testing.T) { | 
|  | testFlow(t, "[[foo]]", []token{ | 
|  | {Position{1, 1}, tokenDoubleLeftBracket, "[["}, | 
|  | {Position{1, 3}, tokenKeyGroupArray, "foo"}, | 
|  | {Position{1, 6}, tokenDoubleRightBracket, "]]"}, | 
|  | {Position{1, 8}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestQuotedKey(t *testing.T) { | 
|  | testFlow(t, "\"a b\" = 42", []token{ | 
|  | {Position{1, 1}, tokenKey, "\"a b\""}, | 
|  | {Position{1, 7}, tokenEqual, "="}, | 
|  | {Position{1, 9}, tokenInteger, "42"}, | 
|  | {Position{1, 11}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestKeyNewline(t *testing.T) { | 
|  | testFlow(t, "a\n= 4", []token{ | 
|  | {Position{1, 1}, tokenError, "keys cannot contain new lines"}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestInvalidFloat(t *testing.T) { | 
|  | testFlow(t, "a=7e1_", []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 2}, tokenEqual, "="}, | 
|  | {Position{1, 3}, tokenFloat, "7e1_"}, | 
|  | {Position{1, 7}, tokenEOF, ""}, | 
|  | }) | 
|  | } | 
|  |  | 
|  | func TestLexUnknownRvalue(t *testing.T) { | 
|  | testFlow(t, `a = !b`, []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenError, "no value can start with !"}, | 
|  | }) | 
|  |  | 
|  | testFlow(t, `a = \b`, []token{ | 
|  | {Position{1, 1}, tokenKey, "a"}, | 
|  | {Position{1, 3}, tokenEqual, "="}, | 
|  | {Position{1, 5}, tokenError, `no value can start with \`}, | 
|  | }) | 
|  | } |