Files
esbuild/internal/css_parser/css_parser_test.go
2020-09-28 22:50:45 -07:00

325 lines
17 KiB
Go

package css_parser
import (
"testing"
"github.com/evanw/esbuild/internal/config"
"github.com/evanw/esbuild/internal/css_printer"
"github.com/evanw/esbuild/internal/logger"
"github.com/evanw/esbuild/internal/test"
)
func assertEqual(t *testing.T, a interface{}, b interface{}) {
t.Helper()
if a != b {
t.Fatalf("%s != %s", a, b)
}
}
func expectParseError(t *testing.T, contents string, expected string) {
t.Helper()
t.Run(contents, func(t *testing.T) {
t.Helper()
log := logger.NewDeferLog()
Parse(log, test.SourceForTest(contents), config.Options{})
msgs := log.Done()
text := ""
for _, msg := range msgs {
text += msg.String(logger.StderrOptions{}, logger.TerminalInfo{})
}
test.AssertEqual(t, text, expected)
})
}
func expectPrintedCommon(t *testing.T, contents string, expected string, options config.Options) {
t.Helper()
t.Run(contents, func(t *testing.T) {
t.Helper()
log := logger.NewDeferLog()
tree := Parse(log, test.SourceForTest(contents), options)
msgs := log.Done()
text := ""
for _, msg := range msgs {
text += msg.String(logger.StderrOptions{}, logger.TerminalInfo{})
}
assertEqual(t, text, "")
css := css_printer.Print(tree, css_printer.Options{})
assertEqual(t, string(css), expected)
})
}
func expectPrinted(t *testing.T, contents string, expected string) {
t.Helper()
expectPrintedCommon(t, contents, expected, config.Options{})
}
func expectPrintedMangle(t *testing.T, contents string, expected string) {
t.Helper()
expectPrintedCommon(t, contents, expected, config.Options{
MangleSyntax: true,
})
}
func TestString(t *testing.T) {
expectPrinted(t, "a:after { content: 'a\\\rb' }", "a:after {\n content: \"ab\";\n}\n")
expectPrinted(t, "a:after { content: 'a\\\nb' }", "a:after {\n content: \"ab\";\n}\n")
expectPrinted(t, "a:after { content: 'a\\\fb' }", "a:after {\n content: \"ab\";\n}\n")
expectPrinted(t, "a:after { content: 'a\\\r\nb' }", "a:after {\n content: \"ab\";\n}\n")
expectPrinted(t, "a:after { content: 'a\\62 c' }", "a:after {\n content: \"abc\";\n}\n")
expectParseError(t, "a:after { content: '\r' }",
`<stdin>: error: Unterminated string token
<stdin>: error: Expected "}" but found end of file
<stdin>: error: Unterminated string token
`)
expectParseError(t, "a:after { content: '\n' }",
`<stdin>: error: Unterminated string token
<stdin>: error: Expected "}" but found end of file
<stdin>: error: Unterminated string token
`)
expectParseError(t, "a:after { content: '\f' }",
`<stdin>: error: Unterminated string token
<stdin>: error: Expected "}" but found end of file
<stdin>: error: Unterminated string token
`)
expectParseError(t, "a:after { content: '\r\n' }",
`<stdin>: error: Unterminated string token
<stdin>: error: Expected "}" but found end of file
<stdin>: error: Unterminated string token
`)
expectPrinted(t, "a:after { content: '\\1010101' }", "a:after {\n content: \"\U001010101\";\n}\n")
expectPrinted(t, "a:after { content: '\\invalid' }", "a:after {\n content: \"invalid\";\n}\n")
}
func TestDeclaration(t *testing.T) {
expectPrinted(t, ".decl {}", ".decl {\n}\n")
expectPrinted(t, ".decl { a: b }", ".decl {\n a: b;\n}\n")
expectPrinted(t, ".decl { a: b; }", ".decl {\n a: b;\n}\n")
expectPrinted(t, ".decl { a: b; c: d }", ".decl {\n a: b;\n c: d;\n}\n")
expectPrinted(t, ".decl { a: b; c: d; }", ".decl {\n a: b;\n c: d;\n}\n")
expectParseError(t, ".decl { a { b: c; } }", "<stdin>: error: Expected \":\" but found \"{\"\n")
expectPrinted(t, ".decl { & a { b: c; } }", ".decl {\n & a {\n b: c;\n }\n}\n")
}
func TestSelector(t *testing.T) {
expectPrinted(t, "a{}", "a {\n}\n")
expectPrinted(t, "a {}", "a {\n}\n")
expectPrinted(t, "a b {}", "a b {\n}\n")
expectPrinted(t, "[b]{}", "[b] {\n}\n")
expectPrinted(t, "[b] {}", "[b] {\n}\n")
expectPrinted(t, "a[b] {}", "a[b] {\n}\n")
expectPrinted(t, "a [b] {}", "a [b] {\n}\n")
expectParseError(t, "[] {}", "<stdin>: error: Expected identifier but found \"]\"\n")
expectParseError(t, "[b {}", "<stdin>: error: Expected \"]\" but found \"{\"\n")
expectParseError(t, "[b]] {}", "<stdin>: error: Unexpected \"]\"\n")
expectParseError(t, "a[b {}", "<stdin>: error: Expected \"]\" but found \"{\"\n")
expectParseError(t, "a[b]] {}", "<stdin>: error: Unexpected \"]\"\n")
expectPrinted(t, "[|b]{}", "[b] {\n}\n") // "[|b]" is equivalent to "[b]"
expectPrinted(t, "[*|b]{}", "[*|b] {\n}\n")
expectPrinted(t, "[a|b]{}", "[a|b] {\n}\n")
expectPrinted(t, "[a|b|=\"c\"]{}", "[a|b|=\"c\"] {\n}\n")
expectPrinted(t, "[a|b |= \"c\"]{}", "[a|b|=\"c\"] {\n}\n")
expectParseError(t, "[a||b] {}", "<stdin>: error: Expected identifier but found \"|\"\n")
expectParseError(t, "[* | b] {}", "<stdin>: error: Expected \"|\" but found whitespace\n")
expectParseError(t, "[a | b] {}", "<stdin>: error: Expected \"=\" but found whitespace\n")
expectPrinted(t, "[b=\"c\"] {}", "[b=\"c\"] {\n}\n")
expectPrinted(t, "[b~=\"c\"] {}", "[b~=\"c\"] {\n}\n")
expectPrinted(t, "[b^=\"c\"] {}", "[b^=\"c\"] {\n}\n")
expectPrinted(t, "[b$=\"c\"] {}", "[b$=\"c\"] {\n}\n")
expectPrinted(t, "[b*=\"c\"] {}", "[b*=\"c\"] {\n}\n")
expectPrinted(t, "[b|=\"c\"] {}", "[b|=\"c\"] {\n}\n")
expectParseError(t, "[b?=\"c\"] {}", "<stdin>: error: Expected \"]\" but found \"?\"\n")
expectPrinted(t, "[b = \"c\"] {}", "[b=\"c\"] {\n}\n")
expectPrinted(t, "[b ~= \"c\"] {}", "[b~=\"c\"] {\n}\n")
expectPrinted(t, "[b ^= \"c\"] {}", "[b^=\"c\"] {\n}\n")
expectPrinted(t, "[b $= \"c\"] {}", "[b$=\"c\"] {\n}\n")
expectPrinted(t, "[b *= \"c\"] {}", "[b*=\"c\"] {\n}\n")
expectPrinted(t, "[b |= \"c\"] {}", "[b|=\"c\"] {\n}\n")
expectParseError(t, "[b ?= \"c\"] {}", "<stdin>: error: Expected \"]\" but found \"?\"\n")
expectPrinted(t, "[b = \"c\" i] {}", "[b=\"c\" i] {\n}\n")
expectPrinted(t, "[b = \"c\" I] {}", "[b=\"c\" I] {\n}\n")
expectParseError(t, "[b i] {}", "<stdin>: error: Expected \"]\" but found \"i\"\n<stdin>: error: Unexpected \"]\"\n")
expectParseError(t, "[b I] {}", "<stdin>: error: Expected \"]\" but found \"I\"\n<stdin>: error: Unexpected \"]\"\n")
expectPrinted(t, "|b {}", "|b {\n}\n")
expectPrinted(t, "|* {}", "|* {\n}\n")
expectPrinted(t, "a|b {}", "a|b {\n}\n")
expectPrinted(t, "a|* {}", "a|* {\n}\n")
expectPrinted(t, "*|b {}", "*|b {\n}\n")
expectPrinted(t, "*|* {}", "*|* {\n}\n")
expectParseError(t, "a||b {}", "<stdin>: error: Expected identifier but found \"|\"\n")
expectPrinted(t, "a+b {}", "a + b {\n}\n")
expectPrinted(t, "a>b {}", "a > b {\n}\n")
expectPrinted(t, "a+b {}", "a + b {\n}\n")
expectPrinted(t, "a~b {}", "a ~ b {\n}\n")
expectPrinted(t, "a + b {}", "a + b {\n}\n")
expectPrinted(t, "a > b {}", "a > b {\n}\n")
expectPrinted(t, "a + b {}", "a + b {\n}\n")
expectPrinted(t, "a ~ b {}", "a ~ b {\n}\n")
expectPrinted(t, "::b {}", "::b {\n}\n")
expectPrinted(t, "*::b {}", "*::b {\n}\n")
expectPrinted(t, "a::b {}", "a::b {\n}\n")
expectPrinted(t, "::b(c) {}", "::b(c) {\n}\n")
expectPrinted(t, "*::b(c) {}", "*::b(c) {\n}\n")
expectPrinted(t, "a::b(c) {}", "a::b(c) {\n}\n")
expectPrinted(t, "a:b:c {}", "a:b:c {\n}\n")
expectPrinted(t, "a:b(:c) {}", "a:b(:c) {\n}\n")
}
func TestNestedSelector(t *testing.T) {
expectPrinted(t, "& {}", "& {\n}\n")
expectPrinted(t, "& b {}", "& b {\n}\n")
expectPrinted(t, "&:b {}", "&:b {\n}\n")
expectPrinted(t, "&* {}", "&* {\n}\n")
expectPrinted(t, "&|b {}", "&|b {\n}\n")
expectPrinted(t, "&*|b {}", "&*|b {\n}\n")
expectPrinted(t, "&a|b {}", "&a|b {\n}\n")
expectPrinted(t, "&[a] {}", "&[a] {\n}\n")
expectPrinted(t, "a { & {} }", "a {\n & {\n }\n}\n")
expectPrinted(t, "a { & b {} }", "a {\n & b {\n }\n}\n")
expectPrinted(t, "a { &:b {} }", "a {\n &:b {\n }\n}\n")
expectPrinted(t, "a { &* {} }", "a {\n &* {\n }\n}\n")
expectPrinted(t, "a { &|b {} }", "a {\n &|b {\n }\n}\n")
expectPrinted(t, "a { &*|b {} }", "a {\n &*|b {\n }\n}\n")
expectPrinted(t, "a { &a|b {} }", "a {\n &a|b {\n }\n}\n")
expectPrinted(t, "a { &[b] {} }", "a {\n &[b] {\n }\n}\n")
}
func TestAtRule(t *testing.T) {
expectPrinted(t, "@unknown;", "@unknown;\n")
expectPrinted(t, "@unknown{}", "@unknown {}\n")
expectPrinted(t, "@unknown x;", "@unknown x;\n")
expectPrinted(t, "@unknown{\na: b;\nc: d;\n}", "@unknown { a: b; c: d; }\n")
expectParseError(t, "@unknown", "<stdin>: error: Expected \"{\" but found end of file\n")
expectParseError(t, "@", "<stdin>: error: Unexpected \"@\"\n<stdin>: error: Expected \"{\" but found end of file\n")
expectParseError(t, "@;", "<stdin>: error: Unexpected \"@\"\n<stdin>: error: Expected \"{\" but found end of file\n")
expectParseError(t, "@{}", "<stdin>: error: Unexpected \"@\"\n")
}
func TestAtCharset(t *testing.T) {
expectPrinted(t, "@charset \"UTF-8\";", "@charset \"UTF-8\";\n")
expectPrinted(t, "@charset 'UTF-8';", "@charset \"UTF-8\";\n")
expectParseError(t, "@charset \"US-ASCII\";", "<stdin>: warning: \"UTF-8\" will be used instead of unsupported charset \"US-ASCII\"\n")
expectParseError(t, "@charset;", "<stdin>: error: Expected whitespace but found \";\"\n")
expectParseError(t, "@charset ;", "<stdin>: error: Expected string token but found \";\"\n")
expectParseError(t, "@charset\"UTF-8\";", "<stdin>: error: Expected whitespace but found \"\\\"UTF-8\\\"\"\n")
expectParseError(t, "@charset \"UTF-8\"", "<stdin>: error: Expected \";\" but found end of file\n")
expectParseError(t, "@charset url(UTF-8);", "<stdin>: error: Expected string token but found \"url(UTF-8)\"\n")
expectParseError(t, "@charset url(\"UTF-8\");", "<stdin>: error: Expected string token but found \"url(\"\n")
expectParseError(t, "@charset \"UTF-8\" ", "<stdin>: error: Expected \";\" but found whitespace\n")
expectParseError(t, "@charset \"UTF-8\"{}", "<stdin>: error: Expected \";\" but found \"{\"\n")
}
func TestAtNamespace(t *testing.T) {
expectPrinted(t, "@namespace\"http://www.com\";", "@namespace \"http://www.com\";\n")
expectPrinted(t, "@namespace \"http://www.com\";", "@namespace \"http://www.com\";\n")
expectPrinted(t, "@namespace \"http://www.com\" ;", "@namespace \"http://www.com\";\n")
expectPrinted(t, "@namespace url();", "@namespace \"\";\n")
expectPrinted(t, "@namespace url(http://www.com);", "@namespace \"http://www.com\";\n")
expectPrinted(t, "@namespace url(http://www.com) ;", "@namespace \"http://www.com\";\n")
expectPrinted(t, "@namespace url(\"http://www.com\");", "@namespace \"http://www.com\";\n")
expectPrinted(t, "@namespace url(\"http://www.com\") ;", "@namespace \"http://www.com\";\n")
expectPrinted(t, "@namespace ns\"http://www.com\";", "@namespace ns \"http://www.com\";\n")
expectPrinted(t, "@namespace ns \"http://www.com\";", "@namespace ns \"http://www.com\";\n")
expectPrinted(t, "@namespace ns \"http://www.com\" ;", "@namespace ns \"http://www.com\";\n")
expectPrinted(t, "@namespace ns url();", "@namespace ns \"\";\n")
expectPrinted(t, "@namespace ns url(http://www.com);", "@namespace ns \"http://www.com\";\n")
expectPrinted(t, "@namespace ns url(http://www.com) ;", "@namespace ns \"http://www.com\";\n")
expectPrinted(t, "@namespace ns url(\"http://www.com\");", "@namespace ns \"http://www.com\";\n")
expectPrinted(t, "@namespace ns url(\"http://www.com\") ;", "@namespace ns \"http://www.com\";\n")
expectParseError(t, "@namespace;", "<stdin>: error: Expected URL token but found \";\"\n")
expectParseError(t, "@namespace \"http://www.com\"", "<stdin>: error: Expected \";\" but found end of file\n")
expectParseError(t, "@namespace url(\"http://www.com\";", "<stdin>: error: Expected \")\" but found \";\"\n")
expectParseError(t, "@namespace noturl(\"http://www.com\");", "<stdin>: error: Expected URL token but found \"noturl(\"\n")
expectParseError(t, "@namespace url(", `<stdin>: error: Expected URL token but found bad URL token
<stdin>: error: Expected ")" to end URL token
<stdin>: error: Expected ";" but found end of file
`)
expectParseError(t, "@namespace ns;", "<stdin>: error: Expected URL token but found \";\"\n")
expectParseError(t, "@namespace ns \"http://www.com\"", "<stdin>: error: Expected \";\" but found end of file\n")
expectParseError(t, "@namespace ns url(\"http://www.com\";", "<stdin>: error: Expected \")\" but found \";\"\n")
expectParseError(t, "@namespace ns noturl(\"http://www.com\");", "<stdin>: error: Expected URL token but found \"noturl(\"\n")
expectParseError(t, "@namespace ns url(", `<stdin>: error: Expected URL token but found bad URL token
<stdin>: error: Expected ")" to end URL token
<stdin>: error: Expected ";" but found end of file
`)
expectParseError(t, "@namespace \"http://www.com\" {}", `<stdin>: error: Expected ";"
<stdin>: error: Unexpected "{"
`)
}
func TestAtImport(t *testing.T) {
expectPrinted(t, "@import\"foo.css\";", "@import \"foo.css\";\n")
expectPrinted(t, "@import \"foo.css\";", "@import \"foo.css\";\n")
expectPrinted(t, "@import \"foo.css\" ;", "@import \"foo.css\";\n")
expectPrinted(t, "@import url();", "@import \"\";\n")
expectPrinted(t, "@import url(foo.css);", "@import \"foo.css\";\n")
expectPrinted(t, "@import url(foo.css) ;", "@import \"foo.css\";\n")
expectPrinted(t, "@import url(\"foo.css\");", "@import \"foo.css\";\n")
expectPrinted(t, "@import url(\"foo.css\") ;", "@import \"foo.css\";\n")
expectParseError(t, "@import;", "<stdin>: error: Expected URL token but found \";\"\n")
expectParseError(t, "@import ;", "<stdin>: error: Expected URL token but found \";\"\n")
expectParseError(t, "@import \"foo.css\"", "<stdin>: error: Expected \";\" but found end of file\n")
expectParseError(t, "@import url(\"foo.css\";", "<stdin>: error: Expected \")\" but found \";\"\n")
expectParseError(t, "@import noturl(\"foo.css\");", "<stdin>: error: Expected URL token but found \"noturl(\"\n")
expectParseError(t, "@import url(", `<stdin>: error: Expected URL token but found bad URL token
<stdin>: error: Expected ")" to end URL token
<stdin>: error: Expected ";" but found end of file
`)
expectParseError(t, "@import \"foo.css\" {}", `<stdin>: error: Expected ";"
<stdin>: error: Unexpected "{"
`)
}
func TestAtKeyframes(t *testing.T) {
expectPrinted(t, "@keyframes name{}", "@keyframes name {\n}\n")
expectPrinted(t, "@keyframes name {}", "@keyframes name {\n}\n")
expectPrinted(t, "@keyframes name{0%,50%{color:red}25%,75%{color:blue}}",
"@keyframes name {\n 0%, 50% {\n color: red;\n }\n 25%, 75% {\n color: blue;\n }\n}\n")
expectPrinted(t, "@keyframes name { 0%, 50% { color: red } 25%, 75% { color: blue } }",
"@keyframes name {\n 0%, 50% {\n color: red;\n }\n 25%, 75% {\n color: blue;\n }\n}\n")
expectPrinted(t, "@keyframes name{from{color:red}to{color:blue}}",
"@keyframes name {\n from {\n color: red;\n }\n to {\n color: blue;\n }\n}\n")
expectPrinted(t, "@keyframes name { from { color: red } to { color: blue } }",
"@keyframes name {\n from {\n color: red;\n }\n to {\n color: blue;\n }\n}\n")
expectPrinted(t, "@keyframes name { from { color: red } }", "@keyframes name {\n from {\n color: red;\n }\n}\n")
expectPrinted(t, "@keyframes name { 100% { color: red } }", "@keyframes name {\n 100% {\n color: red;\n }\n}\n")
expectPrintedMangle(t, "@keyframes name { from { color: red } }", "@keyframes name {\n 0% {\n color: red;\n }\n}\n")
expectPrintedMangle(t, "@keyframes name { 100% { color: red } }", "@keyframes name {\n to {\n color: red;\n }\n}\n")
expectPrinted(t, "@-webkit-keyframes name {}", "@-webkit-keyframes name {\n}\n")
expectPrinted(t, "@-moz-keyframes name {}", "@-moz-keyframes name {\n}\n")
expectPrinted(t, "@-ms-keyframes name {}", "@-ms-keyframes name {\n}\n")
expectPrinted(t, "@-o-keyframes name {}", "@-o-keyframes name {\n}\n")
expectParseError(t, "@keyframes {}", "<stdin>: error: Expected identifier but found \"{\"\n")
expectParseError(t, "@keyframes 'name' {}", "<stdin>: error: Expected identifier but found \"'name'\"\n")
expectParseError(t, "@keyframes name { 0% 100% {} }", "<stdin>: error: Expected \",\" but found \"100%\"\n")
expectParseError(t, "@keyframes name { {} 0% {} }", "<stdin>: error: Expected percentage but found \"{\"\n")
expectParseError(t, "@keyframes name { 100 {} }", "<stdin>: error: Expected percentage but found \"100\"\n")
expectParseError(t, "@keyframes name { into {} }", "<stdin>: error: Expected percentage but found \"into\"\n")
expectParseError(t, "@keyframes name { 1,2 {} }", "<stdin>: error: Expected percentage but found \"1\"\n<stdin>: error: Expected percentage but found \"2\"\n")
expectParseError(t, "@keyframes name { 1, 2 {} }", "<stdin>: error: Expected percentage but found \"1\"\n<stdin>: error: Expected percentage but found \"2\"\n")
expectParseError(t, "@keyframes name { 1 ,2 {} }", "<stdin>: error: Expected percentage but found \"1\"\n<stdin>: error: Expected percentage but found \"2\"\n")
expectParseError(t, "@keyframes name { 1%,,2% {} }", "<stdin>: error: Expected percentage but found \",\"\n")
}