xgo/internal/scanner_test.go
2019-12-20 11:40:38 -08:00

265 lines
6.1 KiB
Go
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

package internal_test
import (
"fmt"
"math/rand"
"strings"
"testing"
"time"
"github.com/d5/tengo/internal"
"github.com/d5/tengo/internal/require"
"github.com/d5/tengo/internal/token"
)
var testFileSet = internal.NewFileSet()
type scanResult struct {
Token token.Token
Literal string
Line int
Column int
}
func TestScanner_Scan(t *testing.T) {
var testCases = [...]struct {
token token.Token
literal string
}{
{token.Comment, "/* a comment */"},
{token.Comment, "// a comment \n"},
{token.Comment, "/*\r*/"},
{token.Comment, "/**\r/*/"},
{token.Comment, "/**\r\r/*/"},
{token.Comment, "//\r\n"},
{token.Ident, "foobar"},
{token.Ident, "a۰۱۸"},
{token.Ident, "foo६४"},
{token.Ident, "bar"},
{token.Ident, "ŝ"},
{token.Ident, "ŝfoo"},
{token.Int, "0"},
{token.Int, "1"},
{token.Int, "123456789012345678890"},
{token.Int, "01234567"},
{token.Int, "0xcafebabe"},
{token.Float, "0."},
{token.Float, ".0"},
{token.Float, "3.14159265"},
{token.Float, "1e0"},
{token.Float, "1e+100"},
{token.Float, "1e-100"},
{token.Float, "2.71828e-1000"},
{token.Char, "'a'"},
{token.Char, "'\\000'"},
{token.Char, "'\\xFF'"},
{token.Char, "'\\uff16'"},
{token.Char, "'\\U0000ff16'"},
{token.String, "`foobar`"},
{token.String, "`" + `foo
bar` +
"`",
},
{token.String, "`\r`"},
{token.String, "`foo\r\nbar`"},
{token.Add, "+"},
{token.Sub, "-"},
{token.Mul, "*"},
{token.Quo, "/"},
{token.Rem, "%"},
{token.And, "&"},
{token.Or, "|"},
{token.Xor, "^"},
{token.Shl, "<<"},
{token.Shr, ">>"},
{token.AndNot, "&^"},
{token.AddAssign, "+="},
{token.SubAssign, "-="},
{token.MulAssign, "*="},
{token.QuoAssign, "/="},
{token.RemAssign, "%="},
{token.AndAssign, "&="},
{token.OrAssign, "|="},
{token.XorAssign, "^="},
{token.ShlAssign, "<<="},
{token.ShrAssign, ">>="},
{token.AndNotAssign, "&^="},
{token.LAnd, "&&"},
{token.LOr, "||"},
{token.Inc, "++"},
{token.Dec, "--"},
{token.Equal, "=="},
{token.Less, "<"},
{token.Greater, ">"},
{token.Assign, "="},
{token.Not, "!"},
{token.NotEqual, "!="},
{token.LessEq, "<="},
{token.GreaterEq, ">="},
{token.Define, ":="},
{token.Ellipsis, "..."},
{token.LParen, "("},
{token.LBrack, "["},
{token.LBrace, "{"},
{token.Comma, ","},
{token.Period, "."},
{token.RParen, ")"},
{token.RBrack, "]"},
{token.RBrace, "}"},
{token.Semicolon, ";"},
{token.Colon, ":"},
{token.Break, "break"},
{token.Continue, "continue"},
{token.Else, "else"},
{token.For, "for"},
{token.Func, "func"},
{token.If, "if"},
{token.Return, "return"},
{token.Export, "export"},
}
// combine
var lines []string
var lineSum int
lineNos := make([]int, len(testCases))
columnNos := make([]int, len(testCases))
for i, tc := range testCases {
// add 0-2 lines before each test case
emptyLines := rand.Intn(3)
for j := 0; j < emptyLines; j++ {
lines = append(lines, strings.Repeat(" ", rand.Intn(10)))
}
// add test case line with some whitespaces around it
emptyColumns := rand.Intn(10)
lines = append(lines, fmt.Sprintf("%s%s%s",
strings.Repeat(" ", emptyColumns),
tc.literal,
strings.Repeat(" ", rand.Intn(10))))
lineNos[i] = lineSum + emptyLines + 1
lineSum += emptyLines + countLines(tc.literal)
columnNos[i] = emptyColumns + 1
}
// expected results
var expected []scanResult
var expectedSkipComments []scanResult
for i, tc := range testCases {
// expected literal
var expectedLiteral string
switch tc.token {
case token.Comment:
// strip CRs in comments
expectedLiteral = string(internal.StripCR([]byte(tc.literal),
tc.literal[1] == '*'))
//-style comment literal doesn't contain newline
if expectedLiteral[1] == '/' {
expectedLiteral = expectedLiteral[:len(expectedLiteral)-1]
}
case token.Ident:
expectedLiteral = tc.literal
case token.Semicolon:
expectedLiteral = ";"
default:
if tc.token.IsLiteral() {
// strip CRs in raw string
expectedLiteral = tc.literal
if expectedLiteral[0] == '`' {
expectedLiteral = string(internal.StripCR(
[]byte(expectedLiteral), false))
}
} else if tc.token.IsKeyword() {
expectedLiteral = tc.literal
}
}
res := scanResult{
Token: tc.token,
Literal: expectedLiteral,
Line: lineNos[i],
Column: columnNos[i],
}
expected = append(expected, res)
if tc.token != token.Comment {
expectedSkipComments = append(expectedSkipComments, res)
}
}
scanExpect(t, strings.Join(lines, "\n"),
internal.ScanComments|internal.DontInsertSemis, expected...)
scanExpect(t, strings.Join(lines, "\n"),
internal.DontInsertSemis, expectedSkipComments...)
}
func TestStripCR(t *testing.T) {
for _, tc := range []struct {
input string
expect string
}{
{"//\n", "//\n"},
{"//\r\n", "//\n"},
{"//\r\r\r\n", "//\n"},
{"//\r*\r/\r\n", "//*/\n"},
{"/**/", "/**/"},
{"/*\r/*/", "/*/*/"},
{"/*\r*/", "/**/"},
{"/**\r/*/", "/**\r/*/"},
{"/*\r/\r*\r/*/", "/*/*\r/*/"},
{"/*\r\r\r\r*/", "/**/"},
} {
actual := string(internal.StripCR([]byte(tc.input),
len(tc.input) >= 2 && tc.input[1] == '*'))
require.Equal(t, tc.expect, actual)
}
}
func scanExpect(
t *testing.T,
input string,
mode internal.ScanMode,
expected ...scanResult,
) {
testFile := testFileSet.AddFile("test", -1, len(input))
s := internal.NewScanner(
testFile,
[]byte(input),
func(_ internal.SourceFilePos, msg string) { require.Fail(t, msg) },
mode)
for idx, e := range expected {
tok, literal, pos := s.Scan()
filePos := testFile.Position(pos)
require.Equal(t, e.Token, tok, "[%d] expected: %s, actual: %s",
idx, e.Token.String(), tok.String())
require.Equal(t, e.Literal, literal)
require.Equal(t, e.Line, filePos.Line)
require.Equal(t, e.Column, filePos.Column)
}
tok, _, _ := s.Scan()
require.Equal(t, token.EOF, tok, "more tokens left")
require.Equal(t, 0, s.ErrorCount())
}
func countLines(s string) int {
if s == "" {
return 0
}
n := 1
for i := 0; i < len(s); i++ {
if s[i] == '\n' {
n++
}
}
return n
}
func init() {
rand.Seed(time.Now().UnixNano())
}