2017-10-20 17:00:53 -07:00
|
|
|
// Copyright 2017 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.
|
|
|
|
|
|
|
|
@TestOn('node')
|
2018-11-15 15:16:24 -08:00
|
|
|
@Tags(['node'])
|
2017-10-20 17:00:53 -07:00
|
|
|
|
|
|
|
import 'dart:async';
|
|
|
|
|
|
|
|
import 'package:js/js.dart';
|
2018-06-13 16:57:28 -07:00
|
|
|
import 'package:path/path.dart' as p;
|
2017-10-20 17:00:53 -07:00
|
|
|
import 'package:test/test.dart';
|
|
|
|
|
|
|
|
import 'package:sass/src/io.dart';
|
|
|
|
import 'package:sass/src/value/number.dart';
|
|
|
|
|
|
|
|
import '../ensure_npm_package.dart';
|
|
|
|
import '../hybrid.dart';
|
|
|
|
import 'api.dart';
|
|
|
|
import 'utils.dart';
|
|
|
|
|
|
|
|
String sassPath;
|
|
|
|
|
|
|
|
void main() {
|
|
|
|
setUpAll(ensureNpmPackage);
|
|
|
|
useSandbox();
|
|
|
|
|
|
|
|
setUp(() async {
|
|
|
|
sassPath = p.join(sandbox, 'test.scss');
|
|
|
|
await writeTextFile(sassPath, 'a {b: c}');
|
|
|
|
});
|
|
|
|
|
|
|
|
test("can import a file by contents", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop(
|
2018-11-15 15:16:24 -08:00
|
|
|
(_, __) => NodeImporterResult(contents: 'a {b: c}')))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("imports cascade through importers", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(data: "@import 'foo'", importer: [
|
2017-10-20 17:00:53 -07:00
|
|
|
allowInterop((url, __) {
|
|
|
|
if (url != "foo") return null;
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '@import "bar"');
|
2017-10-20 17:00:53 -07:00
|
|
|
}),
|
|
|
|
allowInterop((url, __) {
|
|
|
|
if (url != "bar") return null;
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '@import "baz"');
|
2017-10-20 17:00:53 -07:00
|
|
|
}),
|
|
|
|
allowInterop((url, __) {
|
|
|
|
if (url != "baz") return null;
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: 'a {b: c}');
|
2017-10-20 17:00:53 -07:00
|
|
|
})
|
|
|
|
])),
|
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("an empty object means an empty file", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
2018-11-15 15:16:24 -08:00
|
|
|
importer: allowInterop((_, __) => NodeImporterResult()))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace(''));
|
|
|
|
});
|
|
|
|
|
|
|
|
group("import precedence:", () {
|
|
|
|
group("in sandbox dir", () {
|
2018-04-20 17:55:13 -07:00
|
|
|
setUp(runTestInSandbox);
|
2017-10-20 17:00:53 -07:00
|
|
|
|
|
|
|
test("relative file is #1", () async {
|
|
|
|
var subDir = p.join(sandbox, 'sub');
|
|
|
|
await createDirectory(subDir);
|
|
|
|
await writeTextFile(p.join(subDir, 'test.scss'), 'x {y: z}');
|
|
|
|
|
|
|
|
var basePath = p.join(subDir, 'base.scss');
|
|
|
|
await writeTextFile(basePath, '@import "test"');
|
|
|
|
|
2019-05-20 15:34:57 -07:00
|
|
|
expect(
|
|
|
|
renderSync(RenderOptions(
|
|
|
|
file: basePath,
|
|
|
|
importer: allowInterop(
|
|
|
|
(_, __) => NodeImporterResult(contents: "q {r: s}")))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('x { y: z; }'));
|
|
|
|
});
|
|
|
|
|
2019-05-20 15:34:57 -07:00
|
|
|
test("importer is #2", () async {
|
|
|
|
expect(
|
|
|
|
renderSync(RenderOptions(
|
|
|
|
data: '@import "test"',
|
|
|
|
importer: allowInterop(
|
|
|
|
(_, __) => NodeImporterResult(contents: "x {y: z}")))),
|
|
|
|
equalsIgnoringWhitespace('x { y: z; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("CWD is #3", () async {
|
2017-10-20 17:00:53 -07:00
|
|
|
var subDir = p.join(sandbox, 'sub');
|
|
|
|
await createDirectory(subDir);
|
|
|
|
await writeTextFile(p.join(subDir, 'test.scss'), 'x {y: z}');
|
|
|
|
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(
|
|
|
|
RenderOptions(data: '@import "test"', includePaths: [subDir])),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2019-05-20 15:34:57 -07:00
|
|
|
test("include path is #4", () async {
|
|
|
|
var subDir = p.join(sandbox, 'sub');
|
|
|
|
await createDirectory(subDir);
|
|
|
|
await writeTextFile(p.join(subDir, 'test.scss'), 'x {y: z}');
|
|
|
|
|
|
|
|
withSassPath([subDir], () {
|
|
|
|
expect(
|
|
|
|
renderSync(
|
|
|
|
RenderOptions(data: '@import "test"', includePaths: [sandbox])),
|
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
2017-10-20 17:00:53 -07:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
group("with a file redirect", () {
|
|
|
|
test("imports the chosen file", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
2018-11-15 15:16:24 -08:00
|
|
|
importer:
|
|
|
|
allowInterop((_, __) => NodeImporterResult(file: sassPath)))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("supports the indented syntax", () async {
|
|
|
|
await writeTextFile(p.join(sandbox, 'target.sass'), 'a\n b: c');
|
|
|
|
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
2018-11-15 15:16:24 -08:00
|
|
|
importer: allowInterop((_, __) =>
|
|
|
|
NodeImporterResult(file: p.join(sandbox, 'target.sass'))))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
|
2018-08-10 15:58:15 -07:00
|
|
|
test("supports plain CSS", () async {
|
|
|
|
// An import in plain CSS is only ever interpreted as a plain CSS import.
|
|
|
|
await writeTextFile(p.join(sandbox, 'target.css'), "@import 'bar'");
|
|
|
|
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2018-08-10 15:58:15 -07:00
|
|
|
data: "@import 'foo'",
|
2018-11-15 15:16:24 -08:00
|
|
|
importer: allowInterop((_, __) =>
|
|
|
|
NodeImporterResult(file: p.join(sandbox, 'target.css'))))),
|
2018-08-10 15:58:15 -07:00
|
|
|
equalsIgnoringWhitespace('@import "bar";'));
|
|
|
|
});
|
|
|
|
|
2017-10-20 17:00:53 -07:00
|
|
|
test("supports partials", () async {
|
|
|
|
await writeTextFile(p.join(sandbox, '_target.scss'), 'a {b: c}');
|
|
|
|
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
2018-11-15 15:16:24 -08:00
|
|
|
importer: allowInterop((_, __) =>
|
|
|
|
NodeImporterResult(file: p.join(sandbox, 'target.scss'))))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("may be extensionless", () async {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop((_, __) =>
|
2018-11-15 15:16:24 -08:00
|
|
|
NodeImporterResult(file: p.withoutExtension(sassPath))))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("is resolved relative to the base file", () async {
|
|
|
|
var basePath = p.join(sandbox, 'base.scss');
|
|
|
|
await writeTextFile(basePath, '@import "foo"');
|
|
|
|
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
file: basePath,
|
|
|
|
importer: allowInterop(
|
2018-11-15 15:16:24 -08:00
|
|
|
(_, __) => NodeImporterResult(file: 'test.scss')))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("puts the absolute path in includedFiles", () async {
|
|
|
|
var basePath = p.join(sandbox, 'base.scss');
|
|
|
|
await writeTextFile(basePath, '@import "foo"');
|
|
|
|
|
2018-11-15 15:16:24 -08:00
|
|
|
var result = sass.renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
file: basePath,
|
2018-11-15 15:16:24 -08:00
|
|
|
importer: allowInterop((_, __) => NodeImporterResult(file: 'test'))));
|
2017-10-20 17:00:53 -07:00
|
|
|
expect(result.stats.includedFiles, equals([basePath, sassPath]));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("is resolved relative to include paths", () async {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
|
|
|
includePaths: [sandbox],
|
2018-11-15 15:16:24 -08:00
|
|
|
importer:
|
|
|
|
allowInterop((_, __) => NodeImporterResult(file: 'test')))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("relative to the base file takes precedence over include paths",
|
|
|
|
() async {
|
|
|
|
var basePath = p.join(sandbox, 'base.scss');
|
|
|
|
await writeTextFile(basePath, '@import "foo"');
|
|
|
|
|
|
|
|
var subDir = p.join(sandbox, 'sub');
|
|
|
|
await createDirectory(subDir);
|
|
|
|
await writeTextFile(p.join(subDir, 'test.scss'), 'x {y: z}');
|
|
|
|
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
file: basePath,
|
|
|
|
includePaths: [subDir],
|
2018-11-15 15:16:24 -08:00
|
|
|
importer:
|
|
|
|
allowInterop((_, __) => NodeImporterResult(file: 'test')))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
group("in the sandbox directory", () {
|
|
|
|
String oldWorkingDirectory;
|
|
|
|
setUp(() {
|
|
|
|
oldWorkingDirectory = currentPath;
|
|
|
|
chdir(sandbox);
|
|
|
|
});
|
|
|
|
|
|
|
|
tearDown(() => chdir(oldWorkingDirectory));
|
|
|
|
|
|
|
|
test("is resolved relative to the CWD", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop(
|
2018-11-15 15:16:24 -08:00
|
|
|
(_, __) => NodeImporterResult(file: 'test.scss')))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("file-relative takes precedence over the CWD", () async {
|
|
|
|
await createDirectory(p.join(sandbox, 'sub'));
|
|
|
|
var basePath = p.join(sandbox, 'sub', 'base.scss');
|
|
|
|
await writeTextFile(basePath, '@import "foo"');
|
|
|
|
await writeTextFile(p.join(sandbox, 'sub', 'test.scss'), 'x {y: z}');
|
|
|
|
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
file: basePath,
|
|
|
|
importer: allowInterop(
|
2018-11-15 15:16:24 -08:00
|
|
|
(_, __) => NodeImporterResult(file: 'test.scss')))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('x { y: z; }'));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("the CWD takes precedence over include paths", () async {
|
|
|
|
var basePath = p.join(sandbox, 'base.scss');
|
|
|
|
await writeTextFile(basePath, '@import "test"');
|
|
|
|
var subDir = p.join(sandbox, 'sub');
|
|
|
|
await createDirectory(subDir);
|
|
|
|
await writeTextFile(p.join(subDir, 'test.scss'), 'x {y: z}');
|
|
|
|
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
file: basePath,
|
|
|
|
includePaths: [subDir],
|
|
|
|
importer: allowInterop(
|
2018-11-15 15:16:24 -08:00
|
|
|
(_, __) => NodeImporterResult(file: 'test.scss')))),
|
2017-10-20 17:00:53 -07:00
|
|
|
equalsIgnoringWhitespace('a { b: c; }'));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
group("the imported URL", () {
|
|
|
|
test("is the exact imported text", () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop(expectAsync2((url, _) {
|
|
|
|
expect(url, equals('foo'));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
2018-03-11 15:46:59 -07:00
|
|
|
// Regression test for #246.
|
|
|
|
test("doesn't remove ./", () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2018-03-11 15:46:59 -07:00
|
|
|
data: "@import './foo'",
|
|
|
|
importer: allowInterop(expectAsync2((url, _) {
|
|
|
|
expect(url, equals('./foo'));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2018-03-11 15:46:59 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
2017-10-20 17:00:53 -07:00
|
|
|
test("isn't resolved relative to the current file", () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo/bar'",
|
|
|
|
importer: allowInterop(expectAsync2((url, _) {
|
|
|
|
if (url == 'foo/bar') {
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: "@import 'baz'");
|
2017-10-20 17:00:53 -07:00
|
|
|
} else {
|
|
|
|
expect(url, equals('baz'));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: "");
|
2017-10-20 17:00:53 -07:00
|
|
|
}
|
|
|
|
}, count: 2))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("is added to includedFiles", () {
|
2018-11-15 15:16:24 -08:00
|
|
|
var result = sass.renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop(expectAsync2((_, __) {
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
expect(result.stats.includedFiles, equals(['foo']));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
group("the previous URL", () {
|
|
|
|
test("is an absolute path for stylesheets from the filesystem", () async {
|
|
|
|
var importPath = p.join(sandbox, 'import.scss');
|
|
|
|
await writeTextFile(importPath, "@import 'foo'");
|
|
|
|
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
file: importPath,
|
|
|
|
importer: allowInterop(expectAsync2((_, prev) {
|
|
|
|
expect(prev, equals(p.absolute(importPath)));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("is an absolute path for stylesheets redirected to the filesystem",
|
|
|
|
() async {
|
|
|
|
var import1Path = p.join(sandbox, 'import1.scss');
|
|
|
|
await writeTextFile(import1Path, "@import 'foo'");
|
|
|
|
|
|
|
|
var import2Path = p.join(sandbox, 'import2.scss');
|
|
|
|
await writeTextFile(import2Path, "@import 'baz'");
|
|
|
|
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
file: import1Path,
|
|
|
|
importer: allowInterop(expectAsync2((url, prev) {
|
|
|
|
if (url == 'foo') {
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(file: 'import2');
|
2017-10-20 17:00:53 -07:00
|
|
|
} else {
|
|
|
|
expect(url, equals('baz'));
|
|
|
|
expect(prev, equals(import2Path));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: "");
|
2017-10-20 17:00:53 -07:00
|
|
|
}
|
|
|
|
}, count: 2))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test('is "stdin" for string stylesheets', () async {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: '@import "foo"',
|
|
|
|
importer: allowInterop(expectAsync2((_, prev) {
|
|
|
|
expect(prev, equals('stdin'));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("is the imported string for imports from importers", () async {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(data: '@import "foo"', importer: [
|
2017-10-20 17:00:53 -07:00
|
|
|
allowInterop(expectAsync2((url, _) {
|
|
|
|
if (url != "foo") return null;
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '@import "bar"');
|
2017-10-20 17:00:53 -07:00
|
|
|
}, count: 2)),
|
|
|
|
allowInterop(expectAsync2((url, prev) {
|
|
|
|
expect(url, equals("bar"));
|
|
|
|
expect(prev, equals("foo"));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))
|
|
|
|
]));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
group("this", () {
|
|
|
|
test('includes default option values', () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: '@import "foo"',
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
|
|
|
var options = this_.options;
|
|
|
|
expect(options.includePaths, equals(p.current));
|
|
|
|
expect(options.precision, equals(SassNumber.precision));
|
|
|
|
expect(options.style, equals(1));
|
|
|
|
expect(options.indentType, equals(0));
|
|
|
|
expect(options.indentWidth, equals(2));
|
|
|
|
expect(options.linefeed, equals('\n'));
|
|
|
|
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test('includes the data when rendering via data', () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: '@import "foo"',
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
|
|
|
expect(this_.options.data, equals('@import "foo"'));
|
|
|
|
expect(this_.options.file, isNull);
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test('includes the filename when rendering via file', () async {
|
|
|
|
await writeTextFile(sassPath, '@import "foo"');
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
file: sassPath,
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
|
|
|
expect(this_.options.data, isNull);
|
|
|
|
expect(this_.options.file, equals(sassPath));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test('includes other include paths', () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: '@import "foo"',
|
|
|
|
includePaths: [sandbox],
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
2018-12-17 15:30:17 -08:00
|
|
|
expect(this_.options.includePaths,
|
|
|
|
equals("${p.current}${isWindows ? ';' : ':'}$sandbox"));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
group('can override', () {
|
|
|
|
test('indentWidth', () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: '@import "foo"',
|
|
|
|
indentWidth: 5,
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
|
|
|
expect(this_.options.indentWidth, equals(5));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test('indentType', () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: '@import "foo"',
|
|
|
|
indentType: 'tab',
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
|
|
|
expect(this_.options.indentType, equals(1));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test('linefeed', () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: '@import "foo"',
|
|
|
|
linefeed: 'cr',
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
|
|
|
expect(this_.options.linefeed, equals('\r'));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
test('has a circular reference', () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: '@import "foo"',
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
|
|
|
expect(this_.options.context, same(this_));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
group("includes render stats with", () {
|
|
|
|
test('a start time', () {
|
2018-11-15 15:16:24 -08:00
|
|
|
var start = DateTime.now();
|
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: '@import "foo"',
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
|
|
|
expect(this_.options.result.stats.start,
|
|
|
|
greaterThanOrEqualTo(start.millisecondsSinceEpoch));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test('a data entry', () {
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: '@import "foo"',
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
|
|
|
expect(this_.options.result.stats.entry, equals('data'));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
|
|
|
|
test('a file entry', () async {
|
|
|
|
await writeTextFile(sassPath, '@import "foo"');
|
2018-11-15 15:16:24 -08:00
|
|
|
renderSync(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
file: sassPath,
|
|
|
|
importer: allowInteropCaptureThis(
|
|
|
|
expectAsync3((RenderContext this_, _, __) {
|
|
|
|
expect(this_.options.result.stats.entry, equals(sassPath));
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: '');
|
2017-10-20 17:00:53 -07:00
|
|
|
}))));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
group("gracefully handles an error when", () {
|
|
|
|
test("an importer redirects to a non-existent file", () {
|
2018-11-15 15:16:24 -08:00
|
|
|
var error = renderSyncError(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop(
|
2018-11-15 15:16:24 -08:00
|
|
|
(_, __) => NodeImporterResult(file: '_does_not_exist'))));
|
2017-10-20 17:00:53 -07:00
|
|
|
expect(
|
|
|
|
error,
|
|
|
|
toStringAndMessageEqual("Can't find stylesheet to import.\n"
|
2019-01-28 20:42:32 -05:00
|
|
|
" ╷\n"
|
|
|
|
"1 │ @import 'foo'\n"
|
|
|
|
" │ ^^^^^\n"
|
|
|
|
" ╵\n"
|
2017-10-20 17:00:53 -07:00
|
|
|
" stdin 1:9 root stylesheet"));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("an error is returned", () {
|
2018-11-15 15:16:24 -08:00
|
|
|
var error = renderSyncError(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'",
|
2018-11-15 15:16:24 -08:00
|
|
|
importer: allowInterop((_, __) => JSError("oh no"))));
|
2017-10-20 17:00:53 -07:00
|
|
|
|
|
|
|
expect(
|
|
|
|
error,
|
|
|
|
toStringAndMessageEqual("oh no\n"
|
2019-01-28 20:42:32 -05:00
|
|
|
" ╷\n"
|
|
|
|
"1 │ @import 'foo'\n"
|
|
|
|
" │ ^^^^^\n"
|
|
|
|
" ╵\n"
|
2017-10-20 17:00:53 -07:00
|
|
|
" stdin 1:9 root stylesheet"));
|
|
|
|
});
|
|
|
|
|
|
|
|
// TODO(nweiz): Test returning an error subclass when dart-lang/sdk#31168 is
|
|
|
|
// fixed.
|
|
|
|
|
|
|
|
test("null is returned", () {
|
2018-11-15 15:16:24 -08:00
|
|
|
var error = renderSyncError(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'", importer: allowInterop((_, __) => null)));
|
|
|
|
expect(
|
|
|
|
error,
|
|
|
|
toStringAndMessageEqual("Can't find stylesheet to import.\n"
|
2019-01-28 20:42:32 -05:00
|
|
|
" ╷\n"
|
|
|
|
"1 │ @import 'foo'\n"
|
|
|
|
" │ ^^^^^\n"
|
|
|
|
" ╵\n"
|
2017-10-20 17:00:53 -07:00
|
|
|
" stdin 1:9 root stylesheet"));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("undefined is returned", () {
|
2018-11-15 15:16:24 -08:00
|
|
|
var error = renderSyncError(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'", importer: allowInterop((_, __) => undefined)));
|
|
|
|
expect(
|
|
|
|
error,
|
|
|
|
toStringAndMessageEqual("Can't find stylesheet to import.\n"
|
2019-01-28 20:42:32 -05:00
|
|
|
" ╷\n"
|
|
|
|
"1 │ @import 'foo'\n"
|
|
|
|
" │ ^^^^^\n"
|
|
|
|
" ╵\n"
|
2017-10-20 17:00:53 -07:00
|
|
|
" stdin 1:9 root stylesheet"));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("an unrecognized value is returned", () {
|
2018-11-15 15:16:24 -08:00
|
|
|
var error = renderSyncError(RenderOptions(
|
2017-10-20 17:00:53 -07:00
|
|
|
data: "@import 'foo'", importer: allowInterop((_, __) => 10)));
|
|
|
|
expect(
|
|
|
|
error,
|
|
|
|
toStringAndMessageEqual("Can't find stylesheet to import.\n"
|
2019-01-28 20:42:32 -05:00
|
|
|
" ╷\n"
|
|
|
|
"1 │ @import 'foo'\n"
|
|
|
|
" │ ^^^^^\n"
|
|
|
|
" ╵\n"
|
2017-10-20 17:00:53 -07:00
|
|
|
" stdin 1:9 root stylesheet"));
|
|
|
|
});
|
|
|
|
});
|
2017-11-29 18:13:48 -08:00
|
|
|
|
|
|
|
group("render()", () {
|
|
|
|
test("supports asynchronous importers", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
render(RenderOptions(
|
2017-11-29 18:13:48 -08:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop((_, __, done) {
|
2019-05-31 15:45:27 +01:00
|
|
|
Timer(Duration.zero, () {
|
2018-11-15 15:16:24 -08:00
|
|
|
done(NodeImporterResult(contents: 'a {b: c}'));
|
2017-11-29 18:13:48 -08:00
|
|
|
});
|
|
|
|
}))),
|
|
|
|
completion(equalsIgnoringWhitespace('a { b: c; }')));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("supports asynchronous errors", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderError(RenderOptions(
|
2017-11-29 18:13:48 -08:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop((_, __, done) {
|
2019-05-31 15:45:27 +01:00
|
|
|
Timer(Duration.zero, () {
|
2018-11-15 15:16:24 -08:00
|
|
|
done(JSError('oh no'));
|
2017-11-29 18:13:48 -08:00
|
|
|
});
|
|
|
|
}))),
|
|
|
|
completion(toStringAndMessageEqual("oh no\n"
|
2019-01-28 20:42:32 -05:00
|
|
|
" ╷\n"
|
|
|
|
"1 │ @import 'foo'\n"
|
|
|
|
" │ ^^^^^\n"
|
|
|
|
" ╵\n"
|
2017-11-29 18:13:48 -08:00
|
|
|
" stdin 1:9 root stylesheet")));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("supports synchronous importers", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
render(RenderOptions(
|
2017-11-29 18:13:48 -08:00
|
|
|
data: "@import 'foo'",
|
2018-11-15 15:16:24 -08:00
|
|
|
importer: allowInterop(
|
|
|
|
(_, __, ___) => NodeImporterResult(contents: 'a {b: c}')))),
|
2017-11-29 18:13:48 -08:00
|
|
|
completion(equalsIgnoringWhitespace('a { b: c; }')));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("supports synchronous null returns", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderError(RenderOptions(
|
2017-11-29 18:13:48 -08:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop((_, __, ___) => jsNull))),
|
|
|
|
completion(
|
|
|
|
toStringAndMessageEqual("Can't find stylesheet to import.\n"
|
2019-01-28 20:42:32 -05:00
|
|
|
" ╷\n"
|
|
|
|
"1 │ @import 'foo'\n"
|
|
|
|
" │ ^^^^^\n"
|
|
|
|
" ╵\n"
|
2017-11-29 18:13:48 -08:00
|
|
|
" stdin 1:9 root stylesheet")));
|
|
|
|
});
|
2017-12-01 14:28:26 -08:00
|
|
|
|
|
|
|
group("with fibers", () {
|
|
|
|
setUpAll(() {
|
|
|
|
try {
|
|
|
|
fiber;
|
|
|
|
} catch (_) {
|
|
|
|
throw "Can't load fibers package.\n"
|
2018-06-25 16:39:35 -07:00
|
|
|
"Run pub run grinder before-test.";
|
2017-12-01 14:28:26 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
test("supports asynchronous importers", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
render(RenderOptions(
|
2017-12-01 14:28:26 -08:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop((_, __, done) {
|
2019-05-31 15:45:27 +01:00
|
|
|
Timer(Duration.zero, () {
|
2018-11-15 15:16:24 -08:00
|
|
|
done(NodeImporterResult(contents: 'a {b: c}'));
|
2017-12-01 14:28:26 -08:00
|
|
|
});
|
|
|
|
}),
|
|
|
|
fiber: fiber)),
|
|
|
|
completion(equalsIgnoringWhitespace('a { b: c; }')));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("supports synchronous calls to done", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
render(RenderOptions(
|
2017-12-01 14:28:26 -08:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop((_, __, done) {
|
2018-11-15 15:16:24 -08:00
|
|
|
done(NodeImporterResult(contents: 'a {b: c}'));
|
2017-12-01 14:28:26 -08:00
|
|
|
}),
|
|
|
|
fiber: fiber)),
|
|
|
|
completion(equalsIgnoringWhitespace('a { b: c; }')));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("supports synchronous importers", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
render(RenderOptions(
|
2017-12-01 14:28:26 -08:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop((_, __, ___) {
|
2018-11-15 15:16:24 -08:00
|
|
|
return NodeImporterResult(contents: 'a {b: c}');
|
2017-12-01 14:28:26 -08:00
|
|
|
}),
|
|
|
|
fiber: fiber)),
|
|
|
|
completion(equalsIgnoringWhitespace('a { b: c; }')));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("supports asynchronous errors", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderError(RenderOptions(
|
2017-12-01 14:28:26 -08:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop((_, __, done) {
|
2019-05-31 15:45:27 +01:00
|
|
|
Timer(Duration.zero, () {
|
2018-11-15 15:16:24 -08:00
|
|
|
done(JSError('oh no'));
|
2017-12-01 14:28:26 -08:00
|
|
|
});
|
|
|
|
}),
|
|
|
|
fiber: fiber)),
|
|
|
|
completion(toStringAndMessageEqual("oh no\n"
|
2019-01-28 20:42:32 -05:00
|
|
|
" ╷\n"
|
|
|
|
"1 │ @import 'foo'\n"
|
|
|
|
" │ ^^^^^\n"
|
|
|
|
" ╵\n"
|
2017-12-01 14:28:26 -08:00
|
|
|
" stdin 1:9 root stylesheet")));
|
|
|
|
});
|
|
|
|
|
|
|
|
test("supports synchronous null returns", () {
|
|
|
|
expect(
|
2018-11-15 15:16:24 -08:00
|
|
|
renderError(RenderOptions(
|
2017-12-01 14:28:26 -08:00
|
|
|
data: "@import 'foo'",
|
|
|
|
importer: allowInterop((_, __, ___) => jsNull),
|
|
|
|
fiber: fiber)),
|
|
|
|
completion(
|
|
|
|
toStringAndMessageEqual("Can't find stylesheet to import.\n"
|
2019-01-28 20:42:32 -05:00
|
|
|
" ╷\n"
|
|
|
|
"1 │ @import 'foo'\n"
|
|
|
|
" │ ^^^^^\n"
|
|
|
|
" ╵\n"
|
2017-12-01 14:28:26 -08:00
|
|
|
" stdin 1:9 root stylesheet")));
|
|
|
|
});
|
|
|
|
});
|
2018-09-19 15:28:47 -04:00
|
|
|
});
|
2017-10-20 17:00:53 -07:00
|
|
|
}
|