// Copyright 2019 Google Inc. Use of this source code is governed by an // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. import 'package:test/test.dart'; import 'package:sass_embedded/src/embedded_sass.pb.dart'; import 'package:sass_embedded/src/utils.dart'; import 'embedded_process.dart'; import 'utils.dart'; final _true = Value()..singleton = SingletonValue.TRUE; final _false = Value()..singleton = SingletonValue.FALSE; final _null = Value()..singleton = SingletonValue.NULL; late EmbeddedProcess _process; void main() { setUp(() async { _process = await EmbeddedProcess.start(); }); group("emits a protocol error", () { test("for an empty signature", () async { _process.inbound.add(compileString("a {b: c}", functions: [r""])); await expectParamsError( _process, 0, 'CompileRequest.global_functions: "" is missing "("'); await _process.kill(); }); test("for a signature with just a name", () async { _process.inbound.add(compileString("a {b: c}", functions: [r"foo"])); await expectParamsError( _process, 0, 'CompileRequest.global_functions: "foo" is missing "("'); await _process.kill(); }); test("for a signature without a closing paren", () async { _process.inbound.add(compileString("a {b: c}", functions: [r"foo($bar"])); await expectParamsError(_process, 0, 'CompileRequest.global_functions: "foo(\$bar" doesn\'t end with ")"'); await _process.kill(); }); test("for a signature with text after the closing paren", () async { _process.inbound.add(compileString("a {b: c}", functions: [r"foo() "])); await expectParamsError(_process, 0, 'CompileRequest.global_functions: "foo() " doesn\'t end with ")"'); await _process.kill(); }); test("for a signature with invalid arguments", () async { _process.inbound.add(compileString("a {b: c}", functions: [r"foo($)"])); await expectParamsError( _process, 0, 'CompileRequest.global_functions: Error: Expected identifier.\n' ' ╷\n' '1 │ @function foo(\$) {\n' ' │ ^\n' ' ╵\n' ' - 1:16 root stylesheet'); await _process.kill(); }); }); group("includes in FunctionCallRequest", () { var compilationId = 1234; late OutboundMessage_FunctionCallRequest request; setUp(() async { _process.inbound.add(compileString("a {b: foo()}", id: compilationId, functions: ["foo()"])); request = getFunctionCallRequest(await _process.outbound.next); }); test("the same compilationId as the compilation", () async { expect(request.compilationId, equals(compilationId)); await _process.kill(); }); test("the function name", () async { expect(request.name, equals("foo")); await _process.kill(); }); group("arguments", () { test("that are empty", () async { _process.inbound .add(compileString("a {b: foo()}", functions: ["foo()"])); var request = getFunctionCallRequest(await _process.outbound.next); expect(request.arguments, isEmpty); await _process.kill(); }); test("by position", () async { _process.inbound.add(compileString("a {b: foo(true, null, false)}", functions: [r"foo($arg1, $arg2, $arg3)"])); var request = getFunctionCallRequest(await _process.outbound.next); expect(request.arguments, equals([_true, _null, _false])); await _process.kill(); }); test("by name", () async { _process.inbound.add(compileString( r"a {b: foo($arg3: true, $arg1: null, $arg2: false)}", functions: [r"foo($arg1, $arg2, $arg3)"])); var request = getFunctionCallRequest(await _process.outbound.next); expect(request.arguments, equals([_null, _false, _true])); await _process.kill(); }); test("by position and name", () async { _process.inbound.add(compileString( r"a {b: foo(true, $arg3: null, $arg2: false)}", functions: [r"foo($arg1, $arg2, $arg3)"])); var request = getFunctionCallRequest(await _process.outbound.next); expect(request.arguments, equals([_true, _false, _null])); await _process.kill(); }); test("from defaults", () async { _process.inbound.add(compileString(r"a {b: foo(1, $arg3: 2)}", functions: [r"foo($arg1: null, $arg2: true, $arg3: false)"])); var request = getFunctionCallRequest(await _process.outbound.next); expect( request.arguments, equals([ Value()..number = (Value_Number()..value = 1.0), _true, Value()..number = (Value_Number()..value = 2.0) ])); await _process.kill(); }); group("from argument lists", () { test("with no named arguments", () async { _process.inbound.add(compileString("a {b: foo(true, false, null)}", functions: [r"foo($arg, $args...)"])); var request = getFunctionCallRequest(await _process.outbound.next); expect( request.arguments, equals([ _true, Value() ..argumentList = (Value_ArgumentList() ..id = 1 ..separator = ListSeparator.COMMA ..contents.addAll([_false, _null])) ])); await _process.kill(); }); test("with named arguments", () async { _process.inbound.add(compileString(r"a {b: foo(true, $arg: false)}", functions: [r"foo($args...)"])); var request = getFunctionCallRequest(await _process.outbound.next); expect( request.arguments, equals([ Value() ..argumentList = (Value_ArgumentList() ..id = 1 ..separator = ListSeparator.COMMA ..contents.addAll([_true]) ..keywords.addAll({"arg": _false})) ])); await _process.kill(); }); test("throws if named arguments are unused", () async { _process.inbound.add(compileString(r"a {b: foo($arg: false)}", functions: [r"foo($args...)"])); var request = getFunctionCallRequest(await _process.outbound.next); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = _true)); var failure = getCompileFailure(await _process.outbound.next); expect(failure.message, equals(r"No argument named $arg.")); await _process.kill(); }); test("doesn't throw if named arguments are used", () async { _process.inbound.add(compileString(r"a {b: foo($arg: false)}", functions: [r"foo($args...)"])); var request = getFunctionCallRequest(await _process.outbound.next); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..accessedArgumentLists .add(request.arguments.first.argumentList.id) ..success = _true)); await expectLater(_process.outbound, emits(isSuccess(equals("a {\n b: true;\n}")))); await _process.kill(); }); }); }); }); test("returns the result as a SassScript value", () async { _process.inbound .add(compileString("a {b: foo() + 2px}", functions: [r"foo()"])); var request = getFunctionCallRequest(await _process.outbound.next); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = (Value() ..number = (Value_Number() ..value = 1 ..numerators.add("px"))))); await expectLater( _process.outbound, emits(isSuccess(equals("a {\n b: 3px;\n}")))); await _process.kill(); }); group("calls a first-class function", () { test("defined in the compiler and passed to and from the host", () async { _process.inbound.add(compileString(r""" @use "sass:math"; @use "sass:meta"; a {b: call(foo(meta.get-function("abs", $module: "math")), -1)} """, functions: [r"foo($arg)"])); var request = getFunctionCallRequest(await _process.outbound.next); var value = request.arguments.single; expect(value.hasCompilerFunction(), isTrue); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = value)); await expectLater( _process.outbound, emits(isSuccess(equals("a {\n b: 1;\n}")))); await _process.kill(); }); test("defined in the host", () async { var compilationId = 1234; _process.inbound.add(compileString("a {b: call(foo(), true)}", id: compilationId, functions: [r"foo()"])); var hostFunctionId = 5678; var request = getFunctionCallRequest(await _process.outbound.next); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = (Value() ..hostFunction = (Value_HostFunction() ..id = hostFunctionId ..signature = r"bar($arg)")))); request = getFunctionCallRequest(await _process.outbound.next); expect(request.compilationId, equals(compilationId)); expect(request.functionId, equals(hostFunctionId)); expect(request.arguments, equals([_true])); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = _false)); await expectLater( _process.outbound, emits(isSuccess(equals("a {\n b: false;\n}")))); await _process.kill(); }); test("defined in the host and passed to and from the host", () async { var compilationId = 1234; _process.inbound.add(compileString( r""" $function: get-host-function(); $function: round-trip($function); a {b: call($function, true)} """, id: compilationId, functions: [r"get-host-function()", r"round-trip($function)"])); var hostFunctionId = 5678; var request = getFunctionCallRequest(await _process.outbound.next); expect(request.name, equals("get-host-function")); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = (Value() ..hostFunction = (Value_HostFunction() ..id = hostFunctionId ..signature = r"bar($arg)")))); request = getFunctionCallRequest(await _process.outbound.next); expect(request.name, equals("round-trip")); var value = request.arguments.single; expect(value.hasCompilerFunction(), isTrue); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = value)); request = getFunctionCallRequest(await _process.outbound.next); expect(request.compilationId, equals(compilationId)); expect(request.functionId, equals(hostFunctionId)); expect(request.arguments, equals([_true])); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = _false)); await expectLater( _process.outbound, emits(isSuccess(equals("a {\n b: false;\n}")))); await _process.kill(); }); }); group("serializes to protocol buffers", () { group("a string that's", () { group("quoted", () { test("and empty", () async { var value = (await _protofy('""')).string; expect(value.text, isEmpty); expect(value.quoted, isTrue); }); test("and non-empty", () async { var value = (await _protofy('"foo bar"')).string; expect(value.text, equals("foo bar")); expect(value.quoted, isTrue); }); }); group("unquoted", () { test("and empty", () async { var value = (await _protofy('unquote("")')).string; expect(value.text, isEmpty); expect(value.quoted, isFalse); }); test("and non-empty", () async { var value = (await _protofy('"foo bar"')).string; expect(value.text, equals("foo bar")); expect(value.quoted, isTrue); }); }); }); group("a number", () { group("that's unitless", () { test("and an integer", () async { var value = (await _protofy('1')).number; expect(value.value, equals(1.0)); expect(value.numerators, isEmpty); expect(value.denominators, isEmpty); }); test("and a float", () async { var value = (await _protofy('1.5')).number; expect(value.value, equals(1.5)); expect(value.numerators, isEmpty); expect(value.denominators, isEmpty); }); }); test("with one numerator", () async { var value = (await _protofy('1em')).number; expect(value.value, equals(1.0)); expect(value.numerators, ["em"]); expect(value.denominators, isEmpty); }); test("with multiple numerators", () async { var value = (await _protofy('1em * 1px * 1foo')).number; expect(value.value, equals(1.0)); expect(value.numerators, unorderedEquals(["em", "px", "foo"])); expect(value.denominators, isEmpty); }); test("with one denominator", () async { var value = (await _protofy('math.div(1,1em)')).number; expect(value.value, equals(1.0)); expect(value.numerators, isEmpty); expect(value.denominators, ["em"]); }); test("with multiple denominators", () async { var value = (await _protofy('math.div(math.div(math.div(1, 1em), 1px), 1foo)')) .number; expect(value.value, equals(1.0)); expect(value.numerators, isEmpty); expect(value.denominators, unorderedEquals(["em", "px", "foo"])); }); test("with numerators and denominators", () async { var value = (await _protofy('1em * math.div(math.div(1px, 1s), 1foo)')).number; expect(value.value, equals(1.0)); expect(value.numerators, unorderedEquals(["em", "px"])); expect(value.denominators, unorderedEquals(["s", "foo"])); }); }); group("a color that's", () { group("rgb", () { group("without alpha:", () { test("black", () async { expect(await _protofy('#000'), _rgb(0, 0, 0, 1.0)); }); test("white", () async { expect(await _protofy('#fff'), equals(_rgb(255, 255, 255, 1.0))); }); test("in the middle", () async { expect(await _protofy('#abc'), equals(_rgb(0xaa, 0xbb, 0xcc, 1.0))); }); }); group("with alpha", () { test("0", () async { expect(await _protofy('rgb(10, 20, 30, 0)'), equals(_rgb(10, 20, 30, 0.0))); }); test("1", () async { expect(await _protofy('rgb(10, 20, 30, 1)'), equals(_rgb(10, 20, 30, 1.0))); }); test("between 0 and 1", () async { expect(await _protofy('rgb(10, 20, 30, 0.123)'), equals(_rgb(10, 20, 30, 0.123))); }); }); }); group("hsl", () { group("without alpha:", () { group("hue", () { test("0", () async { expect(await _protofy('hsl(0, 50%, 50%)'), _hsl(0, 50, 50, 1.0)); }); test("360", () async { expect( await _protofy('hsl(360, 50%, 50%)'), _hsl(0, 50, 50, 1.0)); }); test("below 0", () async { expect(await _protofy('hsl(-100, 50%, 50%)'), _hsl(260, 50, 50, 1.0)); }); test("between 0 and 360", () async { expect( await _protofy('hsl(100, 50%, 50%)'), _hsl(100, 50, 50, 1.0)); }); test("above 360", () async { expect( await _protofy('hsl(560, 50%, 50%)'), _hsl(200, 50, 50, 1.0)); }); }); group("saturation", () { test("0", () async { expect(await _protofy('hsl(0, 0%, 50%)'), _hsl(0, 0, 50, 1.0)); }); test("100", () async { expect( await _protofy('hsl(0, 100%, 50%)'), _hsl(0, 100, 50, 1.0)); }); test("in the middle", () async { expect(await _protofy('hsl(0, 42%, 50%)'), _hsl(0, 42, 50, 1.0)); }); }); group("lightness", () { test("0", () async { expect(await _protofy('hsl(0, 50%, 0%)'), _hsl(0, 50, 0, 1.0)); }); test("100", () async { expect( await _protofy('hsl(0, 50%, 100%)'), _hsl(0, 50, 100, 1.0)); }); test("in the middle", () async { expect(await _protofy('hsl(0, 50%, 42%)'), _hsl(0, 50, 42, 1.0)); }); }); }); group("with alpha", () { test("0", () async { expect(await _protofy('hsl(10, 20%, 30%, 0)'), equals(_hsl(10, 20, 30, 0.0))); }); test("1", () async { expect(await _protofy('hsl(10, 20%, 30%, 1)'), equals(_hsl(10, 20, 30, 1.0))); }); test("between 0 and 1", () async { expect(await _protofy('hsl(10, 20%, 30%, 0.123)'), equals(_hsl(10, 20, 30, 0.123))); }); }); }); }); group("a list", () { group("with no elements", () { group("with brackets", () { test("with unknown separator", () async { var list = (await _protofy("[]")).list; expect(list.contents, isEmpty); expect(list.hasBrackets, isTrue); expect(list.separator, equals(ListSeparator.UNDECIDED)); }); test("with a comma separator", () async { var list = (await _protofy(r"list.join([], [], $separator: comma)")).list; expect(list.contents, isEmpty); expect(list.hasBrackets, isTrue); expect(list.separator, equals(ListSeparator.COMMA)); }); test("with a space separator", () async { var list = (await _protofy(r"list.join([], [], $separator: space)")).list; expect(list.contents, isEmpty); expect(list.hasBrackets, isTrue); expect(list.separator, equals(ListSeparator.SPACE)); }); test("with a slash separator", () async { var list = (await _protofy(r"list.join([], [], $separator: slash)")).list; expect(list.contents, isEmpty); expect(list.hasBrackets, isTrue); expect(list.separator, equals(ListSeparator.SLASH)); }); }); group("without brackets", () { test("with unknown separator", () async { var list = (await _protofy("()")).list; expect(list.contents, isEmpty); expect(list.hasBrackets, isFalse); expect(list.separator, equals(ListSeparator.UNDECIDED)); }); test("with a comma separator", () async { var list = (await _protofy(r"list.join((), (), $separator: comma)")).list; expect(list.contents, isEmpty); expect(list.hasBrackets, isFalse); expect(list.separator, equals(ListSeparator.COMMA)); }); test("with a space separator", () async { var list = (await _protofy(r"list.join((), (), $separator: space)")).list; expect(list.contents, isEmpty); expect(list.hasBrackets, isFalse); expect(list.separator, equals(ListSeparator.SPACE)); }); test("with a slash separator", () async { var list = (await _protofy(r"list.join((), (), $separator: slash)")).list; expect(list.contents, isEmpty); expect(list.hasBrackets, isFalse); expect(list.separator, equals(ListSeparator.SLASH)); }); }); }); group("with one element", () { group("with brackets", () { test("with unknown separator", () async { var list = (await _protofy("[true]")).list; expect(list.contents, equals([_true])); expect(list.hasBrackets, isTrue); expect(list.separator, equals(ListSeparator.UNDECIDED)); }); test("with a comma separator", () async { var list = (await _protofy(r"[true,]")).list; expect(list.contents, equals([_true])); expect(list.hasBrackets, isTrue); expect(list.separator, equals(ListSeparator.COMMA)); }); test("with a space separator", () async { var list = (await _protofy(r"list.join([true], [], $separator: space)")) .list; expect(list.contents, equals([_true])); expect(list.hasBrackets, isTrue); expect(list.separator, equals(ListSeparator.SPACE)); }); test("with a slash separator", () async { var list = (await _protofy(r"list.join([true], [], $separator: slash)")) .list; expect(list.contents, equals([_true])); expect(list.hasBrackets, isTrue); expect(list.separator, equals(ListSeparator.SLASH)); }); }); group("without brackets", () { test("with a comma separator", () async { var list = (await _protofy(r"(true,)")).list; expect(list.contents, equals([_true])); expect(list.hasBrackets, isFalse); expect(list.separator, equals(ListSeparator.COMMA)); }); test("with a space separator", () async { var list = (await _protofy(r"list.join(true, (), $separator: space)")) .list; expect(list.contents, equals([_true])); expect(list.hasBrackets, isFalse); expect(list.separator, equals(ListSeparator.SPACE)); }); test("with a slash separator", () async { var list = (await _protofy(r"list.join(true, (), $separator: slash)")) .list; expect(list.contents, equals([_true])); expect(list.hasBrackets, isFalse); expect(list.separator, equals(ListSeparator.SLASH)); }); }); }); group("with multiple elements", () { group("with brackets", () { test("with a comma separator", () async { var list = (await _protofy(r"[true, null, false]")).list; expect(list.contents, equals([_true, _null, _false])); expect(list.hasBrackets, isTrue); expect(list.separator, equals(ListSeparator.COMMA)); }); test("with a space separator", () async { var list = (await _protofy(r"[true null false]")).list; expect(list.contents, equals([_true, _null, _false])); expect(list.hasBrackets, isTrue); expect(list.separator, equals(ListSeparator.SPACE)); }); }); group("without brackets", () { test("with a comma separator", () async { var list = (await _protofy(r"true, null, false")).list; expect(list.contents, equals([_true, _null, _false])); expect(list.hasBrackets, isFalse); expect(list.separator, equals(ListSeparator.COMMA)); }); test("with a space separator", () async { var list = (await _protofy(r"true null false")).list; expect(list.contents, equals([_true, _null, _false])); expect(list.hasBrackets, isFalse); expect(list.separator, equals(ListSeparator.SPACE)); }); test("with a slash separator", () async { var list = (await _protofy(r"list.slash(true, null, false)")).list; expect(list.contents, equals([_true, _null, _false])); expect(list.hasBrackets, isFalse); expect(list.separator, equals(ListSeparator.SLASH)); }); }); }); }); group("an argument list", () { test("that's empty", () async { var list = (await _protofy(r"capture-args()")).argumentList; expect(list.contents, isEmpty); expect(list.keywords, isEmpty); expect(list.separator, equals(ListSeparator.COMMA)); }); test("with arguments", () async { var list = (await _protofy(r"capture-args(true, null, false)")).argumentList; expect(list.contents, [_true, _null, _false]); expect(list.keywords, isEmpty); expect(list.separator, equals(ListSeparator.COMMA)); }); test("with a space separator", () async { var list = (await _protofy(r"capture-args(true null false...)")).argumentList; expect(list.contents, [_true, _null, _false]); expect(list.keywords, isEmpty); expect(list.separator, equals(ListSeparator.SPACE)); }); test("with a slash separator", () async { var list = (await _protofy(r"capture-args(list.slash(true, null, false)...)")) .argumentList; expect(list.contents, [_true, _null, _false]); expect(list.keywords, isEmpty); expect(list.separator, equals(ListSeparator.SLASH)); }); test("with keywords", () async { var list = (await _protofy(r"capture-args($arg1: true, $arg2: false)")) .argumentList; expect(list.contents, isEmpty); expect(list.keywords, equals({"arg1": _true, "arg2": _false})); expect(list.separator, equals(ListSeparator.COMMA)); }); }); group("a map", () { test("with no elements", () async { expect((await _protofy("map.remove((1: 2), 1)")).map.entries, isEmpty); }); test("with one element", () async { expect( (await _protofy("(true: false)")).map.entries, equals([ Value_Map_Entry() ..key = _true ..value = _false ])); }); test("with multiple elements", () async { expect( (await _protofy("(true: false, 1: 2, a: b)")).map.entries, equals([ Value_Map_Entry() ..key = _true ..value = _false, Value_Map_Entry() ..key = (Value()..number = (Value_Number()..value = 1.0)) ..value = (Value()..number = (Value_Number()..value = 2.0)), Value_Map_Entry() ..key = (Value() ..string = (Value_String() ..text = "a" ..quoted = false)) ..value = (Value() ..string = (Value_String() ..text = "b" ..quoted = false)) ])); }); }); test("true", () async { expect((await _protofy("true")), equals(_true)); }); test("false", () async { expect((await _protofy("false")), equals(_false)); }); test("true", () async { expect((await _protofy("null")), equals(_null)); }); }); group("deserializes from protocol buffer", () { group("a string that's", () { group("quoted", () { test("and empty", () async { expect( await _deprotofy(Value() ..string = (Value_String() ..text = "" ..quoted = true)), '""'); }); test("and non-empty", () async { expect( await _deprotofy(Value() ..string = (Value_String() ..text = "foo bar" ..quoted = true)), '"foo bar"'); }); }); group("unquoted", () { test("and empty", () async { // We can't use [_deprotofy] here because a property with an empty // value won't render at all. await _assertRoundTrips(Value() ..string = (Value_String() ..text = "" ..quoted = false)); }); test("and non-empty", () async { expect( await _deprotofy(Value() ..string = (Value_String() ..text = "foo bar" ..quoted = false)), "foo bar"); }); }); }); group("a number", () { group("that's unitless", () { test("and an integer", () async { expect( await _deprotofy(Value()..number = (Value_Number()..value = 1.0)), "1"); }); test("and a float", () async { expect( await _deprotofy(Value()..number = (Value_Number()..value = 1.5)), "1.5"); }); }); test("with one numerator", () async { expect( await _deprotofy(Value() ..number = (Value_Number() ..value = 1 ..numerators.add("em"))), "1em"); }); test("with multiple numerators", () async { expect( await _deprotofy( Value() ..number = (Value_Number() ..value = 1 ..numerators.addAll(["em", "px", "foo"])), inspect: true), "1em*px*foo"); }); test("with one denominator", () async { expect( await _deprotofy( Value() ..number = (Value_Number() ..value = 1 ..denominators.add("em")), inspect: true), "1em^-1"); }); test("with multiple denominators", () async { expect( await _deprotofy( Value() ..number = (Value_Number() ..value = 1 ..denominators.addAll(["em", "px", "foo"])), inspect: true), "1(em*px*foo)^-1"); }); test("with numerators and denominators", () async { expect( await _deprotofy( Value() ..number = (Value_Number() ..value = 1 ..numerators.addAll(["em", "px"]) ..denominators.addAll(["s", "foo"])), inspect: true), "1em*px/s*foo"); }); }); group("a color that's", () { group("rgb", () { group("without alpha:", () { test("black", () async { expect(await _deprotofy(_rgb(0, 0, 0, 1.0)), equals('black')); }); test("white", () async { expect(await _deprotofy(_rgb(255, 255, 255, 1.0)), equals('white')); }); test("in the middle", () async { expect(await _deprotofy(_rgb(0xaa, 0xbb, 0xcc, 1.0)), equals('#aabbcc')); }); }); group("with alpha", () { test("0", () async { expect(await _deprotofy(_rgb(10, 20, 30, 0.0)), equals('rgba(10, 20, 30, 0)')); }); test("between 0 and 1", () async { expect(await _deprotofy(_rgb(10, 20, 30, 0.123)), equals('rgba(10, 20, 30, 0.123)')); }); }); }); group("hsl", () { group("without alpha:", () { group("hue", () { test("0", () async { expect(await _deprotofy(_hsl(0, 50, 50, 1.0)), "#bf4040"); }); test("360", () async { expect(await _deprotofy(_hsl(360, 50, 50, 1.0)), "#bf4040"); }); test("below 0", () async { expect(await _deprotofy(_hsl(-100, 50, 50, 1.0)), "#6a40bf"); }); test("between 0 and 360", () async { expect(await _deprotofy(_hsl(100, 50, 50, 1.0)), "#6abf40"); }); test("above 360", () async { expect(await _deprotofy(_hsl(560, 50, 50, 1.0)), "#4095bf"); }); }); group("saturation", () { test("0", () async { expect(await _deprotofy(_hsl(0, 0, 50, 1.0)), "gray"); }); test("100", () async { expect(await _deprotofy(_hsl(0, 100, 50, 1.0)), "red"); }); test("in the middle", () async { expect(await _deprotofy(_hsl(0, 42, 50, 1.0)), "#b54a4a"); }); }); group("lightness", () { test("0", () async { expect(await _deprotofy(_hsl(0, 50, 0, 1.0)), "black"); }); test("100", () async { expect(await _deprotofy(_hsl(0, 50, 100, 1.0)), "white"); }); test("in the middle", () async { expect(await _deprotofy(_hsl(0, 50, 42, 1.0)), "#a13636"); }); }); }); group("with alpha", () { test("0", () async { expect( await _deprotofy(_hsl(10, 20, 30, 0.0)), "rgba(92, 66, 61, 0)"); }); test("between 0 and 1", () async { expect(await _deprotofy(_hsl(10, 20, 30, 0.123)), "rgba(92, 66, 61, 0.123)"); }); }); }); }); group("a list", () { group("with no elements", () { group("with brackets", () { group("with unknown separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..hasBrackets = true ..separator = ListSeparator.UNDECIDED), "[]"); }); group("with a comma separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..hasBrackets = true ..separator = ListSeparator.COMMA), "[]"); }); group("with a space separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..hasBrackets = true ..separator = ListSeparator.SPACE), "[]"); }); group("with a slash separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..hasBrackets = true ..separator = ListSeparator.SLASH), "[]"); }); }); group("without brackets", () { group("with unknown separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..hasBrackets = false ..separator = ListSeparator.UNDECIDED), "()", inspect: true); }); group("with a comma separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..hasBrackets = false ..separator = ListSeparator.COMMA), "()", inspect: true); }); group("with a space separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..hasBrackets = false ..separator = ListSeparator.SPACE), "()", inspect: true); }); group("with a slash separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..hasBrackets = false ..separator = ListSeparator.SLASH), "()", inspect: true); }); }); }); group("with one element", () { group("with brackets", () { group("with unknown separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..contents.add(_true) ..hasBrackets = true ..separator = ListSeparator.UNDECIDED), "[true]"); }); test("with a comma separator", () async { expect( await _deprotofy( Value() ..list = (Value_List() ..contents.add(_true) ..hasBrackets = true ..separator = ListSeparator.COMMA), inspect: true), "[true,]"); }); group("with a space separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..contents.add(_true) ..hasBrackets = true ..separator = ListSeparator.SPACE), "[true]"); }); group("with a slash separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..contents.add(_true) ..hasBrackets = true ..separator = ListSeparator.SLASH), "[true]"); }); }); group("without brackets", () { group("with unknown separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..contents.add(_true) ..hasBrackets = false ..separator = ListSeparator.UNDECIDED), "true"); }); test("with a comma separator", () async { expect( await _deprotofy( Value() ..list = (Value_List() ..contents.add(_true) ..hasBrackets = false ..separator = ListSeparator.COMMA), inspect: true), "(true,)"); }); group("with a space separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..contents.add(_true) ..hasBrackets = false ..separator = ListSeparator.SPACE), "true"); }); group("with a slash separator", () { _testSerializationAndRoundTrip( Value() ..list = (Value_List() ..contents.add(_true) ..hasBrackets = false ..separator = ListSeparator.SLASH), "true"); }); }); }); group("with multiple elements", () { group("with brackets", () { test("with a comma separator", () async { expect( await _deprotofy( Value() ..list = (Value_List() ..contents.addAll([_true, _null, _false]) ..hasBrackets = true ..separator = ListSeparator.COMMA), inspect: true), "[true, null, false]"); }); test("with a space separator", () async { expect( await _deprotofy( Value() ..list = (Value_List() ..contents.addAll([_true, _null, _false]) ..hasBrackets = true ..separator = ListSeparator.SPACE), inspect: true), "[true null false]"); }); test("with a slash separator", () async { expect( await _deprotofy( Value() ..list = (Value_List() ..contents.addAll([_true, _null, _false]) ..hasBrackets = true ..separator = ListSeparator.SLASH), inspect: true), "[true / null / false]"); }); }); group("without brackets", () { test("with a comma separator", () async { expect( await _deprotofy( Value() ..list = (Value_List() ..contents.addAll([_true, _null, _false]) ..hasBrackets = false ..separator = ListSeparator.COMMA), inspect: true), "true, null, false"); }); test("with a space separator", () async { expect( await _deprotofy( Value() ..list = (Value_List() ..contents.addAll([_true, _null, _false]) ..hasBrackets = false ..separator = ListSeparator.SPACE), inspect: true), "true null false"); }); test("with a slash separator", () async { expect( await _deprotofy( Value() ..list = (Value_List() ..contents.addAll([_true, _null, _false]) ..hasBrackets = false ..separator = ListSeparator.SLASH), inspect: true), "true / null / false"); }); }); }); }); group("an argument list", () { test("with no elements", () async { expect( await _roundTrip(Value() ..argumentList = (Value_ArgumentList()..separator = ListSeparator.UNDECIDED)), equals(Value() ..argumentList = (Value_ArgumentList() ..id = 1 ..separator = ListSeparator.UNDECIDED))); }); test("with comma separator", () async { expect( await _roundTrip(Value() ..argumentList = (Value_ArgumentList() ..contents.addAll([_true, _false, _null]) ..separator = ListSeparator.COMMA)), equals(Value() ..argumentList = (Value_ArgumentList() ..id = 1 ..contents.addAll([_true, _false, _null]) ..separator = ListSeparator.COMMA))); }); test("with space separator", () async { expect( await _roundTrip(Value() ..argumentList = (Value_ArgumentList() ..contents.addAll([_true, _false, _null]) ..separator = ListSeparator.SPACE)), equals(Value() ..argumentList = (Value_ArgumentList() ..id = 1 ..contents.addAll([_true, _false, _null]) ..separator = ListSeparator.SPACE))); }); test("with slash separator", () async { expect( await _roundTrip(Value() ..argumentList = (Value_ArgumentList() ..contents.addAll([_true, _false, _null]) ..separator = ListSeparator.SLASH)), equals(Value() ..argumentList = (Value_ArgumentList() ..id = 1 ..contents.addAll([_true, _false, _null]) ..separator = ListSeparator.SLASH))); }); test("with keywords", () async { expect( await _roundTrip(Value() ..argumentList = (Value_ArgumentList() ..keywords.addAll({"arg1": _true, "arg2": _false}) ..separator = ListSeparator.COMMA)), equals(Value() ..argumentList = (Value_ArgumentList() ..id = 1 ..keywords.addAll({"arg1": _true, "arg2": _false}) ..separator = ListSeparator.COMMA))); }); }); group("a map", () { group("with no elements", () { _testSerializationAndRoundTrip(Value()..map = Value_Map(), "()", inspect: true); }); test("with one element", () async { expect( await _deprotofy( Value() ..map = (Value_Map() ..entries.add(Value_Map_Entry() ..key = _true ..value = _false)), inspect: true), "(true: false)"); }); test("with multiple elements", () async { expect( await _deprotofy( Value() ..map = (Value_Map() ..entries.addAll([ Value_Map_Entry() ..key = _true ..value = _false, Value_Map_Entry() ..key = (Value()..number = (Value_Number()..value = 1.0)) ..value = (Value()..number = (Value_Number()..value = 2.0)), Value_Map_Entry() ..key = (Value() ..string = (Value_String() ..text = "a" ..quoted = false)) ..value = (Value() ..string = (Value_String() ..text = "b" ..quoted = false)) ])), inspect: true), "(true: false, 1: 2, a: b)"); }); }); test("true", () async { expect(await _deprotofy(_true), equals("true")); }); test("false", () async { expect(await _deprotofy(_false), equals("false")); }); test("null", () async { expect(await _deprotofy(_null, inspect: true), equals("null")); }); group("and rejects", () { group("a color", () { test("with red above 255", () async { await _expectDeprotofyError(_rgb(256, 0, 0, 1.0), "RgbColor.red must be between 0 and 255, was 256"); }); test("with green above 255", () async { await _expectDeprotofyError(_rgb(0, 256, 0, 1.0), "RgbColor.green must be between 0 and 255, was 256"); }); test("with blue above 255", () async { await _expectDeprotofyError(_rgb(0, 0, 256, 1.0), "RgbColor.blue must be between 0 and 255, was 256"); }); test("with RGB alpha below 0", () async { await _expectDeprotofyError(_rgb(0, 0, 0, -0.1), "RgbColor.alpha must be between 0 and 1, was -0.1"); }); test("with RGB alpha above 1", () async { await _expectDeprotofyError(_rgb(0, 0, 0, 1.1), "RgbColor.alpha must be between 0 and 1, was 1.1"); }); test("with saturation below 0", () async { await _expectDeprotofyError(_hsl(0, -0.1, 0, 1.0), "HslColor.saturation must be between 0 and 100, was -0.1"); }); test("with saturation above 100", () async { await _expectDeprotofyError( _hsl(0, 100.1, 0, 1.0), "HslColor.saturation must be between 0 and 100, was " "100.1"); }); test("with lightness below 0", () async { await _expectDeprotofyError(_hsl(0, 0, -0.1, 1.0), "HslColor.lightness must be between 0 and 100, was -0.1"); }); test("with lightness above 100", () async { await _expectDeprotofyError( _hsl(0, 0, 100.1, 1.0), "HslColor.lightness must be between 0 and 100, was " "100.1"); }); test("with HSL alpha below 0", () async { await _expectDeprotofyError(_hsl(0, 0, 0, -0.1), "HslColor.alpha must be between 0 and 1, was -0.1"); }); test("with HSL alpha above 1", () async { await _expectDeprotofyError(_hsl(0, 0, 0, 1.1), "HslColor.alpha must be between 0 and 1, was 1.1"); }); }); test("a list with multiple elements and an unknown separator", () async { await _expectDeprotofyError( Value() ..list = (Value_List() ..contents.addAll([_true, _false]) ..separator = ListSeparator.UNDECIDED), endsWith("can't have an undecided separator because it has 2 " "elements")); }); test("an arglist with an unknown id", () async { await _expectDeprotofyError( Value()..argumentList = (Value_ArgumentList()..id = 1), equals( "Value.ArgumentList.id 1 doesn't match any known argument lists")); }); }); }); } /// Evaluates [sassScript] in the compiler, passes it to a custom function, and /// returns the protocol buffer result. Future _protofy(String sassScript) async { _process.inbound.add(compileString(""" @use 'sass:list'; @use 'sass:map'; @use 'sass:math'; @use 'sass:meta'; @function capture-args(\$args...) { \$_: meta.keywords(\$args); @return \$args; } \$_: foo(($sassScript)); """, functions: [r"foo($arg)"])); var request = getFunctionCallRequest(await _process.outbound.next); expect(_process.kill(), completes); return request.arguments.single; } /// Defines two tests: one that asserts that [value] is serialized to the CSS /// value [expected], and one that asserts that it survives a round trip in the /// same protocol buffer format. /// /// This is necessary for values that can be serialized but also have metadata /// that's not visible in the serialized form. void _testSerializationAndRoundTrip(Value value, String expected, {bool inspect = false}) { test("is serialized correctly", () async => expect(await _deprotofy(value, inspect: inspect), expected)); test("preserves metadata", () => _assertRoundTrips(value)); } /// Sends [value] to the compiler and returns its string serialization. /// /// If [inspect] is true, this returns the value as serialized by the /// `meta.inspect()` function. Future _deprotofy(Value value, {bool inspect = false}) async { _process.inbound.add(compileString( inspect ? "a {b: inspect(foo())}" : "a {b: foo()}", functions: [r"foo()"])); var request = getFunctionCallRequest(await _process.outbound.next); expect(request.arguments, isEmpty); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = value)); var success = await getCompileSuccess(await _process.outbound.next); expect(_process.kill(), completes); return RegExp(r" b: (.*);").firstMatch(success.css)![1]!; } /// Asserts that [value] causes a parameter error with a message matching /// [message] when deserializing it from a protocol buffer. Future _expectDeprotofyError(Value value, message) async { _process.inbound.add(compileString("a {b: foo()}", functions: [r"foo()"])); var request = getFunctionCallRequest(await _process.outbound.next); expect(request.arguments, isEmpty); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = value)); await expectParamsError(_process, errorId, message); await _process.kill(); } /// Sends [value] to the compiler to convert to a native Sass value, then sends /// it back out to the host as a protocol buffer and asserts the two buffers are /// identical. /// /// Generally [_deprotofy] should be used instead unless there are details about /// the internal structure of the value that won't show up in its string /// representation. Future _assertRoundTrips(Value value) async => expect(await _roundTrip(value), equals(value)); /// Sends [value] to the compiler to convert to a native Sass value, then sends /// it back out to the host as a protocol buffer and returns the result. Future _roundTrip(Value value) async { _process.inbound.add(compileString(""" \$_: outbound(inbound()); """, functions: ["inbound()", r"outbound($arg)"])); var request = getFunctionCallRequest(await _process.outbound.next); expect(request.arguments, isEmpty); _process.inbound.add(InboundMessage() ..functionCallResponse = (InboundMessage_FunctionCallResponse() ..id = request.id ..success = value)); request = getFunctionCallRequest(await _process.outbound.next); expect(_process.kill(), completes); return request.arguments.single; } /// Returns a [Value] that's an RGB color with the given fields. Value _rgb(int red, int green, int blue, double alpha) => Value() ..rgbColor = (Value_RgbColor() ..red = red ..green = green ..blue = blue ..alpha = alpha); /// Returns a [Value] that's an HSL color with the given fields. Value _hsl(num hue, num saturation, num lightness, double alpha) => Value() ..hslColor = (Value_HslColor() ..hue = hue * 1.0 ..saturation = saturation * 1.0 ..lightness = lightness * 1.0 ..alpha = alpha);