Skip to content

Commit cee843b

Browse files
nex3ntkme
andauthored
Improve case canonicalization performance (#2552)
Closes #2548 Co-authored-by: なつき <[email protected]>
1 parent bce1f4c commit cee843b

11 files changed

+77
-14
lines changed

CHANGELOG.md

+5
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,8 @@
1+
## 1.86.1
2+
3+
* Improve the performance of `file:` URL case canonicalization on Windows and
4+
Mac OS.
5+
16
## 1.86.0
27

38
* Add support for `%` as an expression in its own right. It will still be parsed

lib/src/io.dart

+23-5
Original file line numberDiff line numberDiff line change
@@ -30,9 +30,8 @@ String canonicalize(String path) => _couldBeCaseInsensitive
3030

3131
/// Returns `path` with the case updated to match the path's case on disk.
3232
///
33-
/// This only updates `path`'s basename. It always returns `path` as-is on
34-
/// operating systems other than Windows or Mac OS, since they almost never use
35-
/// case-insensitive filesystems.
33+
/// This always returns `path` as-is on operating systems other than Windows or
34+
/// Mac OS, since they almost never use case-insensitive filesystems.
3635
String _realCasePath(String path) {
3736
// TODO(nweiz): Use an SDK function for this when dart-lang/sdk#35370 and/or
3837
// nodejs/node#24942 are fixed, or at least use FFI functions.
@@ -47,14 +46,33 @@ String _realCasePath(String path) {
4746
}
4847
}
4948

50-
String helper(String path) {
49+
String helper(String path, [String? realPath]) {
5150
var dirname = p.dirname(path);
5251
if (dirname == path) return path;
5352

5453
return _realCaseCache.putIfAbsent(path, () {
54+
// If the path isn't a symlink, we can use the libraries' `realpath()`
55+
// functions to get its actual basename much more efficiently than listing
56+
// all its siblings.
57+
if (!linkExists(path)) {
58+
// Don't recompute the real path if it was already computed for a child
59+
// and we haven't seen any symlinks between that child and this directory.
60+
String realPathNonNull;
61+
try {
62+
realPathNonNull = realPath ?? realpath(path);
63+
} on FileSystemException {
64+
// If we can't get the realpath, that probably means the file doesn't
65+
// exist. Rather than throwing an error about symlink resolution,
66+
// return the non-existent path and let it throw whatever use-time
67+
// error it's going to throw.
68+
return path;
69+
}
70+
return p.join(helper(dirname, p.dirname(realPathNonNull)),
71+
p.basename(realPathNonNull));
72+
}
73+
5574
var realDirname = helper(dirname);
5675
var basename = p.basename(path);
57-
5876
try {
5977
var matches = listDir(realDirname)
6078
.where(

lib/src/io/interface.dart

+7
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,9 @@ bool fileExists(String path) => throw '';
6161
/// Returns whether a dir at [path] exists.
6262
bool dirExists(String path) => throw '';
6363

64+
/// Returns whether a symbolic link at [path] exists.
65+
bool linkExists(String path) => throw '';
66+
6467
/// Ensures that a directory exists at [path], creating it and its ancestors if
6568
/// necessary.
6669
void ensureDir(String path) => throw '';
@@ -71,6 +74,10 @@ void ensureDir(String path) => throw '';
7174
/// beneath [path] as well.
7275
Iterable<String> listDir(String path, {bool recursive = false}) => throw '';
7376

77+
/// Returns the resolved physical path of [path] on disk, with symbolic links
78+
/// resolved and with the same case as the physical file.
79+
String realpath(String path) => throw '';
80+
7481
/// Returns the modification time of the file at [path].
7582
DateTime modificationTime(String path) => throw '';
7683

lib/src/io/js.dart

+22
Original file line numberDiff line numberDiff line change
@@ -181,6 +181,21 @@ bool dirExists(String path) {
181181
});
182182
}
183183

184+
bool linkExists(String path) {
185+
if (!isNodeJs) {
186+
throw UnsupportedError("linkExists() is only supported on Node.js");
187+
}
188+
return _systemErrorToFileSystemException(() {
189+
try {
190+
return fs.lstatSync(path).isSymbolicLink();
191+
} catch (error) {
192+
var systemError = error as JsSystemError;
193+
if (systemError.code == 'ENOENT') return false;
194+
rethrow;
195+
}
196+
});
197+
}
198+
184199
void ensureDir(String path) {
185200
if (!isNodeJs) {
186201
throw UnsupportedError("ensureDir() is only supported on Node.js");
@@ -220,6 +235,13 @@ Iterable<String> listDir(String path, {bool recursive = false}) {
220235
});
221236
}
222237

238+
String realpath(String path) {
239+
if (!isNodeJs) {
240+
throw UnsupportedError("listDir() is only supported on Node.js");
241+
}
242+
return _systemErrorToFileSystemException(() => fs.realpathSync.native(path));
243+
}
244+
223245
DateTime modificationTime(String path) {
224246
if (!isNodeJs) {
225247
throw UnsupportedError("modificationTime() is only supported on Node.js");

lib/src/io/vm.dart

+5
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,8 @@ bool fileExists(String path) => io.File(path).existsSync();
8080

8181
bool dirExists(String path) => io.Directory(path).existsSync();
8282

83+
bool linkExists(String path) => io.Link(path).existsSync();
84+
8385
void ensureDir(String path) => io.Directory(path).createSync(recursive: true);
8486

8587
Iterable<String> listDir(String path, {bool recursive = false}) =>
@@ -88,6 +90,9 @@ Iterable<String> listDir(String path, {bool recursive = false}) =>
8890
.whereType<io.File>()
8991
.map((entity) => entity.path);
9092

93+
// The `File()` method works with directories as well.
94+
String realpath(String path) => io.File(path).resolveSymbolicLinksSync();
95+
9196
DateTime modificationTime(String path) {
9297
var stat = io.FileStat.statSync(path);
9398
if (stat.type == io.FileSystemEntityType.notFound) {

lib/src/js/legacy.dart

+1-4
Original file line numberDiff line numberDiff line change
@@ -473,10 +473,7 @@ RenderResult _newRenderResult(
473473
var end = DateTime.now();
474474

475475
var css = result.css;
476-
// TODO(nweiz): Get rid of this cast once pulyaevskiy/node-interop#109 is
477-
// released.
478-
// ignore: prefer_void_to_null
479-
Uint8List? sourceMapBytes = undefined as Null;
476+
Uint8List? sourceMapBytes = undefined;
480477
if (_enableSourceMaps(options)) {
481478
var sourceMapOption = options.sourceMap;
482479
var sourceMapPath =

pkg/sass-parser/CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
## 0.4.17
2+
3+
* No user-visible changes.
4+
15
## 0.4.16
26

37
* Use union types rather than base classes for Sass nodes wherever possible.

pkg/sass-parser/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "sass-parser",
3-
"version": "0.4.16",
3+
"version": "0.4.17",
44
"description": "A PostCSS-compatible wrapper of the official Sass parser",
55
"repository": "sass/sass",
66
"author": "Google Inc.",

pkg/sass_api/CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
## 15.3.1
2+
3+
* No user-visible changes.
4+
15
## 15.3.0
26

37
* No user-visible changes.

pkg/sass_api/pubspec.yaml

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,15 +2,15 @@ name: sass_api
22
# Note: Every time we add a new Sass AST node, we need to bump the *major*
33
# version because it's a breaking change for anyone who's implementing the
44
# visitor interface(s).
5-
version: 15.3.0
5+
version: 15.3.1
66
description: Additional APIs for Dart Sass.
77
homepage: https://github.com/sass/dart-sass
88

99
environment:
1010
sdk: ">=3.6.0 <4.0.0"
1111

1212
dependencies:
13-
sass: 1.86.0
13+
sass: 1.86.1
1414

1515
dev_dependencies:
1616
dartdoc: ^8.0.14

pubspec.yaml

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
name: sass
2-
version: 1.86.0
2+
version: 1.86.1
33
description: A Sass implementation in Dart.
44
homepage: https://github.com/sass/dart-sass
55

@@ -21,7 +21,7 @@ dependencies:
2121
js: ^0.6.3
2222
meta: ^1.3.0
2323
native_synchronization: ^0.3.0
24-
node_interop: ^2.1.0
24+
node_interop: ^2.2.0
2525
package_config: ^2.0.0
2626
path: ^1.8.0
2727
pool: ^1.5.1
@@ -53,3 +53,4 @@ dev_dependencies:
5353
test_process: ^2.0.0
5454
yaml: ^3.1.0
5555
cli_util: ^0.4.0
56+

0 commit comments

Comments
 (0)