Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Package Importer] Dart Implementation #2130

Merged
merged 122 commits into from
Feb 6, 2024
Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
122 commits
Select commit Hold shift + click to select a range
5e570c5
Export nodePackageImporter
jamesnw Oct 25, 2023
d01b7ac
Initial Node Package Importer
jamesnw Oct 25, 2023
c01c999
Merge branch 'sass:main' into feature.package-importer
jamesnw Oct 26, 2023
2360346
Add case for NodePackageImporter so embedded host compiles
jamesnw Oct 26, 2023
f1ce3a1
Hook up embedded
jamesnw Oct 27, 2023
2609525
Resolving package root, return absolute paths
jamesnw Oct 27, 2023
b0b8dab
Add recursive parent directory lookup for resolving package root
jamesnw Oct 30, 2023
b7f50f4
FilesystemImporter constructor should be internal
jamesnw Oct 30, 2023
6895b57
Start package exports resolve implementation
jamesnw Nov 1, 2023
880ad5a
Exports algorithm
jamesnw Nov 1, 2023
02f6bcc
Exports parsing
jamesnw Nov 2, 2023
c1929f0
Move subprocedures inside class, make private
jamesnw Nov 2, 2023
ef994ab
Hook up package importers for legacy
jamesnw Nov 2, 2023
99990b1
Handle entrypoint cases
jamesnw Nov 3, 2023
8518c67
Add node package importer to async
jamesnw Nov 3, 2023
aed5ea4
Add logs for debugging tests
jamesnw Nov 3, 2023
e22bfef
Use platform-specific separators
jamesnw Nov 3, 2023
bc2d0a7
Handling parsing windows paths to file name
jamesnw Nov 3, 2023
7718e8a
Add logs for Windows troubleshooting
jamesnw Nov 3, 2023
1bcfb4c
Only prepend on non-Windows
jamesnw Nov 4, 2023
7dec6d9
Merge branch 'main' of https://github.com/sass/dart-sass into feature…
jamesnw Nov 4, 2023
e84ff35
Add wildcard expansion
jamesnw Nov 6, 2023
d5763c4
Disable wildcard matching
jamesnw Nov 6, 2023
d24da96
Revert to resolvePackageRoot working on Posix
jamesnw Nov 6, 2023
535481d
Windows pathing
jamesnw Nov 6, 2023
ada3782
Move parentDir logic to io
jamesnw Nov 7, 2023
60f9892
Try separate pathing for windows
jamesnw Nov 7, 2023
51e5bba
Enable wildcards
jamesnw Nov 7, 2023
c391990
Document functions
jamesnw Nov 7, 2023
787d261
Revert incorrect @internal, add CacheImporter.only
jamesnw Nov 8, 2023
851fa6f
Fix legacy importer order bug
jamesnw Nov 8, 2023
99d7ad2
Remove debug print
jamesnw Nov 8, 2023
e63fb1b
Don't throw if pkg not found, load package.json with windows-friendly…
jamesnw Nov 8, 2023
ce5caea
Add logging statements
jamesnw Nov 8, 2023
69ec0ab
Add logging statements, v2
jamesnw Nov 8, 2023
c0e85a8
Add logging statements, v3
jamesnw Nov 8, 2023
f1936d9
Add logging statements, v4
jamesnw Nov 9, 2023
8da2895
Uri.file > Uri.directory
jamesnw Nov 9, 2023
1688531
Convert path to Uri to filestring
jamesnw Nov 9, 2023
a162636
Use file path
jamesnw Nov 9, 2023
e44215b
More filepath usage
jamesnw Nov 9, 2023
e69ce0c
Inspect entry values
jamesnw Nov 9, 2023
afc32bb
Remove duplicated separators
jamesnw Nov 9, 2023
aa7c5f4
Remove scheme from containing url
jamesnw Nov 9, 2023
0549d2f
Remove log statements
jamesnw Nov 9, 2023
757309e
Documentation comments
jamesnw Nov 13, 2023
75de009
Use FilesystemImporter.canonicalize for file: URLs
jamesnw Nov 16, 2023
ece578e
Handle invalid URL
jamesnw Nov 17, 2023
c22c8ca
Create FilesystemImporter cwd singleton
jamesnw Nov 17, 2023
091ed1f
Add missed FilesystemImporter cwd's
jamesnw Nov 17, 2023
e0e42b6
Undo accidental self reference
jamesnw Nov 17, 2023
6175ce9
Switch NodePackageImporter type to JSSymbol
jamesnw Nov 17, 2023
c0a47cd
Change entry point logic
jamesnw Nov 22, 2023
d3431ea
Address review
jamesnw Dec 1, 2023
4cdd953
Remove separate _filesystemImporter fields for FilesystemImporter.cwd
jamesnw Dec 1, 2023
e847921
Remove unneeded import
jamesnw Dec 1, 2023
5e637de
Sync
jamesnw Dec 1, 2023
2575b86
Address review
jamesnw Dec 1, 2023
f0bd0c0
Use switch expression
jamesnw Dec 1, 2023
3916b23
More review
jamesnw Dec 1, 2023
b4f4342
Handle invalid json, update _resolvePackageRootValues
jamesnw Dec 1, 2023
fadd73a
Use null for empty subpath
jamesnw Dec 1, 2023
ac58863
List the multiple matches in error
jamesnw Dec 1, 2023
6f8aacc
Refactor multiple match logic
jamesnw Dec 1, 2023
2f49a34
Refactor _nodePackageExportsResolve
jamesnw Dec 4, 2023
b4015ee
Adjust _packageNameAndSubpath comment
jamesnw Dec 4, 2023
0571ae7
Adjust _resolvePackateRoot
jamesnw Dec 4, 2023
e18ca15
Throw error if exports has both path and non-path keys
jamesnw Dec 4, 2023
d95d7fc
Reduce indent
jamesnw Dec 4, 2023
123b693
Clean up file paths
jamesnw Dec 4, 2023
b1b78ad
Revert to naive concat for Windows tests
jamesnw Dec 4, 2023
afbfade
Another Windows attempt
jamesnw Dec 4, 2023
6167fd8
Try parsing containing URL for windows
jamesnw Dec 4, 2023
e3fc592
Remove unneeded parentDir
jamesnw Dec 5, 2023
c9c6473
Link to Node spec
jamesnw Dec 8, 2023
7f6d15a
Clean up errors
jamesnw Dec 8, 2023
e5f795a
Merge branch 'main' of github.com:oddbird/dart-sass into feature.pack…
jamesnw Dec 11, 2023
0676529
Merge branch 'main' of https://github.com/sass/dart-sass into feature…
jamesnw Dec 11, 2023
3ead339
Merge branch 'main' into feature.package-importer
jgerigmeyer Dec 15, 2023
f3f43d4
Move to NodePackageImporter as class
jamesnw Dec 19, 2023
485a71f
Fix embedded param
jamesnw Dec 19, 2023
ed58590
Switch posix url context for bare import specifiers
jamesnw Dec 20, 2023
da3694a
Try using containingURL directly
jgerigmeyer Dec 20, 2023
ed7fe62
temp try logging path for windows
jgerigmeyer Dec 20, 2023
c8406a4
More temp code for Windows debugging
jgerigmeyer Dec 20, 2023
ade6fe2
Try using baseUrl.path
jgerigmeyer Dec 20, 2023
5481ca6
manual attempt to coerce windows path
jgerigmeyer Dec 20, 2023
b106be1
Try skipping Uri.directory
jgerigmeyer Dec 20, 2023
02e8dd1
Merge branch 'main' of https://github.com/sass/dart-sass into feature…
jamesnw Jan 3, 2024
0d2f859
Remove Internal from class
jamesnw Jan 3, 2024
b686cc1
Progress on path/uri cleanup
jamesnw Jan 4, 2024
038b2f4
Use paths in _exportsToCheck
jamesnw Jan 4, 2024
924612a
Update docs
jamesnw Jan 4, 2024
31b59f4
Update exportsToCheck, packageNameAndSubpath
jamesnw Jan 5, 2024
aaadfc7
More review updates
jamesnw Jan 5, 2024
f0985f3
Handle paths in _nodePackageExportsResolve
jamesnw Jan 5, 2024
e1329cc
Remove unneeded fromUri
jamesnw Jan 5, 2024
c3b1c8b
Review updates
jamesnw Jan 5, 2024
15e33b9
Address review
jgerigmeyer Jan 5, 2024
397b789
Add missing file.
jgerigmeyer Jan 5, 2024
069b272
update copyrights
jgerigmeyer Jan 5, 2024
aa56d07
Address review
jamesnw Jan 5, 2024
e0c712e
Switch recursive resolvePackateRoot to while
jamesnw Jan 8, 2024
915d6d9
Differentiate path and directory
jamesnw Jan 8, 2024
a3ddb06
Update package name handling
jamesnw Jan 8, 2024
89e4dba
Convert URL targets to native paths
jamesnw Jan 8, 2024
e5c1773
Ensure 'as Object' is not null
jamesnw Jan 8, 2024
8441b3b
Merge branch 'main' of https://github.com/sass/dart-sass into feature…
jamesnw Jan 17, 2024
e04b109
Merge branch 'main' of https://github.com/sass/dart-sass into feature…
jamesnw Jan 18, 2024
4820ecc
Update errors and paths
jamesnw Jan 19, 2024
521b1cf
Review cleanup
jamesnw Jan 19, 2024
0d4f2c6
Don't reject invalid node package names
jamesnw Jan 19, 2024
0434343
Check for no entryPointPath or file system in constructor
jamesnw Jan 19, 2024
1cb845c
Private and document valid extensions
jamesnw Jan 19, 2024
125acf3
Move absolute path logic into constructor
jamesnw Jan 19, 2024
0a4aa2f
Remove unused import
jamesnw Jan 19, 2024
8e60938
Address review
jamesnw Jan 20, 2024
7f2132a
Expose Node Package Importer directly to JS
jamesnw Jan 20, 2024
c9f9bc9
typo
jgerigmeyer Jan 24, 2024
7779cbc
Merge branch 'main' into feature.package-importer
jgerigmeyer Jan 24, 2024
2f878b0
Update entry point to directory
jamesnw Feb 1, 2024
53472fd
Use parent directory of containing URL
jamesnw Feb 2, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions lib/src/embedded/dispatcher.dart
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import 'package:native_synchronization/mailbox.dart';
import 'package:path/path.dart' as p;
import 'package:protobuf/protobuf.dart';
import 'package:sass/sass.dart' as sass;
import 'package:sass/src/importer/node_package.dart';

import '../value/function.dart';
import '../value/mixin.dart';
Expand Down Expand Up @@ -223,6 +224,11 @@ final class Dispatcher {
case InboundMessage_CompileRequest_Importer_Importer.notSet:
_checkNoNonCanonicalScheme(importer);
return null;

case InboundMessage_CompileRequest_Importer_Importer.nodePackageImporter:
var entryPointURL =
Uri.parse(importer.nodePackageImporter.entryPointUrl);
return NodePackageImporterInternal(entryPointURL);
}
}

Expand Down
2 changes: 2 additions & 0 deletions lib/src/importer/filesystem.dart
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,8 @@ class FilesystemImporter extends Importer {
final String _loadPath;

/// Creates an importer that loads files relative to [loadPath].
/// @nodoc
@internal
jamesnw marked this conversation as resolved.
Show resolved Hide resolved
FilesystemImporter(String loadPath) : _loadPath = p.absolute(loadPath);

Uri? canonicalize(Uri url) {
Expand Down
195 changes: 195 additions & 0 deletions lib/src/importer/node_package.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,195 @@
// Copyright 2023 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 '../importer.dart';
import './utils.dart';
import 'dart:convert';
import '../io.dart';
import 'package:path/path.dart' as p;

/// A filesystem importer to use for load implementation details, and for
/// canonicalizing paths not defined in package.json.
///
final _filesystemImporter = FilesystemImporter('.');

// An importer that resolves `pkg:` URLs using the Node resolution algorithm.
class NodePackageImporterInternal extends Importer {
jamesnw marked this conversation as resolved.
Show resolved Hide resolved
final Uri entryPointURL;
jamesnw marked this conversation as resolved.
Show resolved Hide resolved
// Creates an importer with the associated entry point url
NodePackageImporterInternal(this.entryPointURL);
jamesnw marked this conversation as resolved.
Show resolved Hide resolved

@override
Uri? canonicalize(Uri url) {
if (url.scheme != 'pkg') return null;
if (url.path.startsWith('/')) {
throw "pkg: URL $url must not be an absolute path";
}
if (url.path.isEmpty) {
throw "pkg: URL $url must not have an empty path";
}
if (url.userInfo != '' || url.hasPort || url.hasQuery || url.hasFragment) {
jamesnw marked this conversation as resolved.
Show resolved Hide resolved
throw "Invalid URL $url";
}
// TODO(jamesnw) `containingUrl` seems to be always null
var baseURL =
containingUrl?.scheme == 'file:' ? containingUrl! : entryPointURL;

var (packageName, subpath) = packageNameAndSubpath(url.path);
var packageRoot = resolvePackageRoot(packageName, baseURL);
if (packageRoot == null) {
throw "Node Package '$packageName' could not be found.";
}

// Attempt to resolve using conditional exports
var jsonString = readFile(packageRoot.path + '/package.json');
var packageManifest = jsonDecode(jsonString) as Map<String, dynamic>;
jamesnw marked this conversation as resolved.
Show resolved Hide resolved
jamesnw marked this conversation as resolved.
Show resolved Hide resolved

var resolved = resolvePackageExports(
packageRoot, subpath, packageManifest, packageName);

if (resolved != null &&
resolved.scheme == 'file:' &&
['scss', 'sass', 'css'].contains(p.extension(resolved.path))) {
return resolved;
} else if (resolved != null) {
throw "The export for $subpath in $packageName is not a valid Sass file.";
}
// If no subpath, attempt to resolve `sass` or `style` key in package.json,
// then `index` file at package root, resolved for file extensions and
// partials.
if (subpath == '') {
jamesnw marked this conversation as resolved.
Show resolved Hide resolved
return resolvePackageRootValues(packageRoot.path, packageManifest);
}

// If there is a subpath, attempt to resolve the path relative to the package root, and
// resolve for file extensions and partials.
return _filesystemImporter
.canonicalize(Uri.file("${packageRoot.path}/$subpath"));
}

@override
ImporterResult? load(Uri url) => _filesystemImporter.load(url);
}

// Takes a string, `path`, and returns a tuple with the package name and the
// subpath if it is present.
(String, String) packageNameAndSubpath(String path) {
var parts = path.split('/');
var name = parts.removeAt(0);
if (name.startsWith('@')) {
name = '$name/${parts.removeAt(0)}';
}
return (name, parts.isNotEmpty ? parts.join('/') : '');
}

// Takes a string, `packageName`, and an absolute URL `baseURL`, and returns an
// absolute URL to the root directory for the most proximate installed
// `packageName`.
Uri? resolvePackageRoot(String packageName, Uri baseURL) {
var baseDirectory = p.dirname(baseURL.toString());

Uri? recurseUpFrom(String entry) {
var potentialPackage = p.joinAll([entry, 'node_modules', packageName]);

if (dirExists(potentialPackage)) {
return Uri.file(potentialPackage);
}
List<String> parentDirectoryParts = List.from(Uri.parse(entry).pathSegments)
..removeLast();

if (parentDirectoryParts.isEmpty) return null;

return recurseUpFrom(p.joinAll(parentDirectoryParts));
}

return recurseUpFrom(baseDirectory);
}

// Takes a string `packagePath`, which is the root directory for a package, and
// `packageManifest`, which is the contents of that package's `package.json`
// file, and returns a file URL.
Uri? resolvePackageRootValues(
String packageRoot, Map<String, dynamic> packageManifest) {
var extensions = ['.scss', '.sass', '.css'];

var sassValue = packageManifest['sass'] as String?;
if (sassValue != null && extensions.contains(p.extension(sassValue))) {
return Uri.file('$packageRoot/$sassValue');
}
var styleValue = packageManifest['style'] as String?;
if (styleValue != null && extensions.contains(p.extension(styleValue))) {
return Uri.file('$packageRoot/$styleValue');
}

var result = resolveImportPath('$packageRoot/index');
if (result != null) return Uri.file(result);
return null;
}

// Takes a package.json value `packageManifest`, a directory URL `packageRoot`
// and a relative URL path `subpath`. It returns a file URL or null.
// `packageName` is used for error reporting only.
Uri? resolvePackageExports(Uri packageRoot, String subpath,
Map<String, dynamic> packageManifest, String packageName) {
if (packageManifest['exports'] == null) return null;
var exports = packageManifest['exports'] as Map<String, dynamic>;
var subpathVariants = exportLoadPaths(subpath);
var resolvedPaths =
nodePackageExportsResolve(packageRoot, subpathVariants, exports);

if (resolvedPaths.length == 1) return resolvedPaths.first;
if (resolvedPaths.length > 1) {
throw "Unable to determine which of multiple potential"
"resolutions found for $subpath in $packageName should be used.";
}
if (p.extension(subpath).isNotEmpty) return null;

var subpathIndexVariants = exportLoadPaths("$subpath/index");

var resolvedIndexpaths =
nodePackageExportsResolve(packageRoot, subpathIndexVariants, exports);

if (resolvedIndexpaths.length == 1) return resolvedPaths.first;
if (resolvedIndexpaths.length > 1) {
throw "Unable to determine which of multiple potential"
"resolutions found for $subpath in $packageName should be used.";
}

return null;
}

// Takes a package.json value `packageManifest`, a directory URL `packageRoot`
// and a list of relative URL paths `subpathVariants`. It returns a list of all
// subpaths present in the package Manifest exports.
List<Uri> nodePackageExportsResolve(Uri packageRoot,
List<String> subpathVariants, Map<String, dynamic> exports) {
// TODO implement
return [];
}

// Given a string `subpath`, returns a list of all possible variations with
// extensions and partials.
List<String> exportLoadPaths(String subpath) {
List<String> paths = [];

if (['scss', 'sass', 'css'].any((ext) => subpath.endsWith(ext))) {
paths.add(subpath);
} else {
paths.addAll([
'$subpath.scss',
'$subpath.sass',
'$subpath.css',
]);
}
var basename = Uri.parse(subpath).pathSegments.last;
if (!basename.startsWith('_')) {
List<String> prefixedPaths = [];
for (final path in paths) {
prefixedPaths.add('_$path');
}
paths.addAll(prefixedPaths);
}

return paths;
}
1 change: 1 addition & 0 deletions lib/src/js.dart
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ void main() {
silent: JSLogger(
warn: allowInteropNamed('sass.Logger.silent.warn', (_, __) {}),
debug: allowInteropNamed('sass.Logger.silent.debug', (_, __) {})));
exports.nodePackageImporter = nodePackageImporter;

exports.info =
"dart-sass\t${const String.fromEnvironment('version')}\t(Sass Compiler)\t"
Expand Down
20 changes: 20 additions & 0 deletions lib/src/js/compile.dart
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
import 'package:cli_pkg/js.dart';
import 'package:node_interop/js.dart';
import 'package:node_interop/util.dart' hide futureToPromise;
import 'package:path/path.dart' as p;
import 'package:term_glyph/term_glyph.dart' as glyph;

import '../../sass.dart';
Expand All @@ -13,6 +14,7 @@ import '../importer/js_to_dart/async.dart';
import '../importer/js_to_dart/async_file.dart';
import '../importer/js_to_dart/file.dart';
import '../importer/js_to_dart/sync.dart';
import '../importer/node_package.dart';
import '../io.dart';
import '../logger/js_to_dart.dart';
import '../util/nullable.dart';
Expand Down Expand Up @@ -207,6 +209,16 @@ AsyncImporter _parseAsyncImporter(Object? importer) {

/// Converts [importer] into a synchronous [Importer].
Importer _parseImporter(Object? importer) {
if (importer == nodePackageImporter) {
// TODO(jamesnw) Is this the correct check?
if (isBrowser) {
jsThrow(JsError(
"The Node Package Importer can not be used without a filesystem."));
}
// TODO(jamesnw) Can we get an actual filename for parity? Is it needed?
Uri entryPointURL = Uri.parse(p.absolute('./index.js'));
return NodePackageImporterInternal(entryPointURL);
}
if (importer == null) jsThrow(JsError("Importers may not be null."));

importer as JSImporter;
Expand Down Expand Up @@ -321,3 +333,11 @@ List<AsyncCallable> _parseFunctions(Object? functions, {bool asynch = false}) {
});
return result;
}

const NodePackageImporter nodePackageImporter = NodePackageImporter();

class NodePackageImporter {
// ignore: unused_field
final String _nodePackageImporterBrand = '';
const NodePackageImporter();
}
2 changes: 2 additions & 0 deletions lib/src/js/exports.dart
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import '../value.dart' as value;
import 'legacy/types.dart';
import 'logger.dart';
import 'reflection.dart';
import 'compile.dart';

@JS()
class Exports {
Expand All @@ -22,6 +23,7 @@ class Exports {
external set info(String info);
external set Exception(JSClass function);
external set Logger(LoggerNamespace namespace);
external set nodePackageImporter(NodePackageImporter nodePackageImporter);

// Value APIs
external set Value(JSClass function);
Expand Down
Loading