Skip to content

Commit 702a7ee

Browse files
authoredApr 6, 2023
Fix the performance of selector span expansion (#1929)
Instead of calling `SourceFile.getText()`, which creates string copies of a substantial subset of the text of the file every time, this directly accesses the file's underlying code units without doing any copies. Closes #1913
1 parent 4ddd8f5 commit 702a7ee

File tree

5 files changed

+26
-20
lines changed

5 files changed

+26
-20
lines changed
 

‎CHANGELOG.md

+2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@
22

33
* **Potentially breaking change:** Drop support for End-of-Life Node.js 12.
44

5+
* Fix remaining cases for the performance regression introduced in 1.59.0.
6+
57
### Embedded Sass
68

79
* The JS embedded host now loads files from the working directory when using the

‎lib/src/interpolation_map.dart

+16-16
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,6 @@ import 'dart:math' as math;
66

77
import 'package:charcode/charcode.dart';
88
import 'package:source_span/source_span.dart';
9-
import 'package:string_scanner/string_scanner.dart';
109

1110
import 'ast/sass.dart';
1211
import 'util/character.dart';
@@ -125,21 +124,21 @@ class InterpolationMap {
125124
/// comment before the expression, but since it's only used for error
126125
/// reporting that's probably fine.
127126
int _expandInterpolationSpanLeft(FileLocation start) {
128-
var source = start.file.getText(0, start.offset);
127+
var source = start.file.codeUnits;
129128
var i = start.offset - 1;
130-
while (true) {
131-
var prev = source.codeUnitAt(i--);
129+
while (i >= 0) {
130+
var prev = source[i--];
132131
if (prev == $lbrace) {
133-
if (source.codeUnitAt(i) == $hash) break;
132+
if (source[i] == $hash) break;
134133
} else if (prev == $slash) {
135-
var second = source.codeUnitAt(i--);
134+
var second = source[i--];
136135
if (second == $asterisk) {
137136
while (true) {
138-
var char = source.codeUnitAt(i--);
137+
var char = source[i--];
139138
if (char != $asterisk) continue;
140139

141140
do {
142-
char = source.codeUnitAt(i--);
141+
char = source[i--];
143142
} while (char == $asterisk);
144143
if (char == $slash) break;
145144
}
@@ -153,28 +152,29 @@ class InterpolationMap {
153152
/// Given the end of a [FileSpan] covering an interpolated expression, returns
154153
/// the offset of the interpolation's closing `}`.
155154
int _expandInterpolationSpanRight(FileLocation end) {
156-
var scanner = StringScanner(end.file.getText(end.offset));
157-
while (true) {
158-
var next = scanner.readChar();
155+
var source = end.file.codeUnits;
156+
var i = end.offset;
157+
while (i < source.length) {
158+
var next = source[i++];
159159
if (next == $rbrace) break;
160160
if (next == $slash) {
161-
var second = scanner.readChar();
161+
var second = source[i++];
162162
if (second == $slash) {
163-
while (!isNewline(scanner.readChar())) {}
163+
while (!isNewline(source[i++])) {}
164164
} else if (second == $asterisk) {
165165
while (true) {
166-
var char = scanner.readChar();
166+
var char = source[i++];
167167
if (char != $asterisk) continue;
168168

169169
do {
170-
char = scanner.readChar();
170+
char = source[i++];
171171
} while (char == $asterisk);
172172
if (char == $slash) break;
173173
}
174174
}
175175
}
176176
}
177177

178-
return end.offset + scanner.position;
178+
return i;
179179
}
180180
}

‎pkg/sass_api/CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
## 6.2.0
2+
3+
* No user-visible changes.
4+
15
## 6.1.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: 6.1.0
5+
version: 6.2.0
66
description: Additional APIs for Dart Sass.
77
homepage: https://github.com/sass/dart-sass
88

99
environment:
1010
sdk: ">=2.17.0 <3.0.0"
1111

1212
dependencies:
13-
sass: 1.60.0
13+
sass: 1.61.0
1414

1515
dev_dependencies:
1616
dartdoc: ^5.0.0

‎pubspec.yaml

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

@@ -23,7 +23,7 @@ dependencies:
2323
path: ^1.8.0
2424
pub_semver: ^2.0.0
2525
source_maps: ^0.10.10
26-
source_span: ^1.8.1
26+
source_span: ^1.10.0
2727
stack_trace: ^1.10.0
2828
stream_transform: ^2.0.0
2929
string_scanner: ^1.1.0

0 commit comments

Comments
 (0)
Please sign in to comment.