diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index 4ee23575b..04b7e00f4 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -126,24 +126,40 @@ jobs: working-directory: sass-spec sass_spec_js_embedded: - name: 'JS API Tests | Embedded | Node ${{ matrix.node-version }} | ${{ matrix.os }}' + name: "JS API Tests | Embedded ${{ matrix.js && 'Pure JS' || 'Dart' }} | Node ${{ matrix.node-version }} | ${{ matrix.os }}" runs-on: ${{ matrix.os }} if: "github.event_name != 'pull_request' || !contains(github.event.pull_request.body, 'skip sass-embedded')" strategy: fail-fast: false matrix: + js: [true, false] os: [ubuntu-latest, windows-latest, macos-latest] node-version: ['lts/*'] include: # Test older LTS versions - - os: ubuntu-latest + - js: true + os: ubuntu-latest dart_channel: stable node-version: lts/-1 - - os: ubuntu-latest + - js: true + os: ubuntu-latest dart_channel: stable node-version: lts/-2 - - os: ubuntu-latest + - js: true + os: ubuntu-latest + dart_channel: stable + node-version: lts/-3 + - js: false + os: ubuntu-latest + dart_channel: stable + node-version: lts/-1 + - js: false + os: ubuntu-latest + dart_channel: stable + node-version: lts/-2 + - js: false + os: ubuntu-latest dart_channel: stable node-version: lts/-3 @@ -166,19 +182,13 @@ jobs: - name: Initialize embedded host run: | npm install - npm run init -- --compiler-path=.. --language-path=../build/language + npm run init -- --compiler-path=.. --language-path=../build/language --compiler-js=${{ matrix.js && 'true' || 'false' }} npm run compile - mv {`pwd`/,dist/}lib/src/vendor/dart-sass working-directory: embedded-host-node - name: Version info - run: | - path=embedded-host-node/dist/lib/src/vendor/dart-sass/sass - if [[ -f "$path.cmd" ]]; then "./$path.cmd" --version - elif [[ -f "$path.bat" ]]; then "./$path.bat" --version - elif [[ -f "$path.exe" ]]; then "./$path.exe" --version - else "./$path" --version - fi + run: node dist/bin/sass.js --version + working-directory: embedded-host-node - name: Run tests run: npm run js-api-spec -- --sassPackage ../embedded-host-node --sassSassRepo ../build/language diff --git a/README.md b/README.md index de3edaad5..c78cd282d 100644 --- a/README.md +++ b/README.md @@ -434,9 +434,7 @@ an API for users to invoke Sass and define custom functions and importers. * `sass --embedded --version` prints `versionResponse` with `id = 0` in JSON and exits. -The `--embedded` command-line flag is not available when you install Dart Sass -as an [npm package]. No other command-line flags are supported with -`--embedded`. +No other command-line flags are supported with `--embedded`. [npm package]: #from-npm diff --git a/bin/sass.dart b/bin/sass.dart index 78ac31370..c6284901c 100644 --- a/bin/sass.dart +++ b/bin/sass.dart @@ -18,8 +18,7 @@ import 'package:sass/src/io.dart'; import 'package:sass/src/stylesheet_graph.dart'; import 'package:sass/src/utils.dart'; import 'package:sass/src/embedded/executable.dart' - // Never load the embedded protocol when compiling to JS. - if (dart.library.js) 'package:sass/src/embedded/unavailable.dart' + if (dart.library.js) 'package:sass/src/embedded/js/executable.dart' as embedded; Future main(List args) async { diff --git a/lib/src/embedded/compilation_dispatcher.dart b/lib/src/embedded/compilation_dispatcher.dart index 83456445e..4e497ef88 100644 --- a/lib/src/embedded/compilation_dispatcher.dart +++ b/lib/src/embedded/compilation_dispatcher.dart @@ -3,17 +3,17 @@ // https://opensource.org/licenses/MIT. import 'dart:convert'; -import 'dart:io'; -import 'dart:isolate'; +import 'dart:io' if (dart.library.js) 'js/io.dart'; +import 'dart:isolate' if (dart.library.js) 'js/isolate.dart'; import 'dart:typed_data'; -import 'package:native_synchronization/mailbox.dart'; import 'package:path/path.dart' as p; import 'package:protobuf/protobuf.dart'; import 'package:pub_semver/pub_semver.dart'; import 'package:sass/sass.dart' as sass; import 'package:sass/src/importer/node_package.dart' as npi; +import '../io.dart' show FileSystemException; import '../logger.dart'; import '../value/function.dart'; import '../value/mixin.dart'; @@ -23,6 +23,7 @@ import 'host_callable.dart'; import 'importer/file.dart'; import 'importer/host.dart'; import 'logger.dart'; +import 'sync_receive_port.dart'; import 'util/proto_extensions.dart'; import 'utils.dart'; @@ -35,8 +36,8 @@ final _outboundRequestId = 0; /// A class that dispatches messages to and from the host for a single /// compilation. final class CompilationDispatcher { - /// The mailbox for receiving messages from the host. - final Mailbox _mailbox; + /// The synchronous receive port for receiving messages from the host. + final SyncReceivePort _receivePort; /// The send port for sending messages to the host. final SendPort _sendPort; @@ -52,8 +53,8 @@ final class CompilationDispatcher { late Uint8List _compilationIdVarint; /// Creates a [CompilationDispatcher] that receives encoded protocol buffers - /// through [_mailbox] and sends them through [_sendPort]. - CompilationDispatcher(this._mailbox, this._sendPort); + /// through [_receivePort] and sends them through [_sendPort]. + CompilationDispatcher(this._receivePort, this._sendPort); /// Listens for incoming `CompileRequests` and runs their compilations. void listen() { @@ -366,14 +367,14 @@ final class CompilationDispatcher { message.writeToCodedBufferWriter(protobufWriter); // Add one additional byte to the beginning to indicate whether or not the - // compilation has finished (1) or encountered a fatal error (2), so the - // [IsolateDispatcher] knows whether to treat this isolate as inactive or - // close out entirely. + // compilation has finished (1) or encountered a fatal error (exitCode), so + // the [IsolateDispatcher] knows whether to treat this isolate as inactive + // or close out entirely. var packet = Uint8List( 1 + _compilationIdVarint.length + protobufWriter.lengthInBytes); packet[0] = switch (message.whichMessage()) { OutboundMessage_Message.compileResponse => 1, - OutboundMessage_Message.error => 2, + OutboundMessage_Message.error => exitCode, _ => 0 }; packet.setAll(1, _compilationIdVarint); @@ -384,9 +385,9 @@ final class CompilationDispatcher { /// Receive a packet from the host. Uint8List _receive() { try { - return _mailbox.take(); + return _receivePort.receive(); } on StateError catch (_) { - // The [_mailbox] has been closed, exit the current isolate immediately + // The [SyncReceivePort] has been closed, exit the current isolate immediately // to avoid bubble the error up as [SassException] during [_sendRequest]. Isolate.exit(); } diff --git a/lib/src/embedded/concurrency.dart b/lib/src/embedded/concurrency.dart new file mode 100644 index 000000000..66ca26081 --- /dev/null +++ b/lib/src/embedded/concurrency.dart @@ -0,0 +1,10 @@ +// Copyright 2024 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 'dart:ffi'; + +/// More than MaxMutatorThreadCount isolates in the same isolate group +/// can deadlock the Dart VM. +/// See https://github.com/sass/dart-sass/pull/2019 +int get concurrencyLimit => sizeOf() <= 4 ? 7 : 15; diff --git a/lib/src/embedded/executable.dart b/lib/src/embedded/executable.dart index 9248713ae..5c1ff52ca 100644 --- a/lib/src/embedded/executable.dart +++ b/lib/src/embedded/executable.dart @@ -3,35 +3,19 @@ // https://opensource.org/licenses/MIT. import 'dart:io'; -import 'dart:convert'; import 'package:stream_channel/stream_channel.dart'; import 'isolate_dispatcher.dart'; +import 'options.dart'; import 'util/length_delimited_transformer.dart'; void main(List args) { - switch (args) { - case ["--version", ...]: - var response = IsolateDispatcher.versionResponse(); - response.id = 0; - stdout.writeln( - JsonEncoder.withIndent(" ").convert(response.toProto3Json())); - return; - - case [_, ...]: - stderr.writeln( - "sass --embedded is not intended to be executed with additional " - "arguments.\n" - "See https://github.com/sass/dart-sass#embedded-dart-sass for " - "details."); - // USAGE error from https://bit.ly/2poTt90 - exitCode = 64; - return; + if (parseOptions(args)) { + IsolateDispatcher( + StreamChannel.withGuarantees(stdin, stdout, allowSinkErrors: false) + .transform(lengthDelimited), + gracefulShutdown: false) + .listen(); } - - IsolateDispatcher( - StreamChannel.withGuarantees(stdin, stdout, allowSinkErrors: false) - .transform(lengthDelimited)) - .listen(); } diff --git a/lib/src/embedded/isolate_dispatcher.dart b/lib/src/embedded/isolate_dispatcher.dart index fe79f034c..d2082f9c0 100644 --- a/lib/src/embedded/isolate_dispatcher.dart +++ b/lib/src/embedded/isolate_dispatcher.dart @@ -3,19 +3,17 @@ // https://opensource.org/licenses/MIT. import 'dart:async'; -import 'dart:ffi'; -import 'dart:io'; -import 'dart:isolate'; +import 'dart:io' if (dart.library.js) 'js/io.dart'; import 'dart:typed_data'; -import 'package:native_synchronization/mailbox.dart'; import 'package:pool/pool.dart'; import 'package:protobuf/protobuf.dart'; import 'package:stream_channel/stream_channel.dart'; -import 'compilation_dispatcher.dart'; +import 'concurrency.dart' if (dart.library.js) 'js/concurrency.dart'; import 'embedded_sass.pb.dart'; -import 'reusable_isolate.dart'; +import 'isolate_main.dart' if (dart.library.js) 'js/isolate_main.dart'; +import 'reusable_isolate.dart' if (dart.library.js) 'js/reusable_isolate.dart'; import 'util/proto_extensions.dart'; import 'utils.dart'; @@ -25,6 +23,10 @@ class IsolateDispatcher { /// The channel of encoded protocol buffers, connected to the host. final StreamChannel _channel; + /// Whether to wait for all worker isolates to exit before exiting the main + /// isolate or not. + final bool _gracefulShutdown; + /// All isolates that have been spawned to dispatch to. /// /// Only used for cleaning up the process when the underlying channel closes. @@ -38,16 +40,13 @@ class IsolateDispatcher { /// A pool controlling how many isolates (and thus concurrent compilations) /// may be live at once. - /// - /// More than MaxMutatorThreadCount isolates in the same isolate group - /// can deadlock the Dart VM. - /// See https://github.com/sass/dart-sass/pull/2019 - final _isolatePool = Pool(sizeOf() <= 4 ? 7 : 15); + final _isolatePool = Pool(concurrencyLimit); /// Whether [_channel] has been closed or not. var _closed = false; - IsolateDispatcher(this._channel); + IsolateDispatcher(this._channel, {bool gracefulShutdown = true}) + : _gracefulShutdown = gracefulShutdown; void listen() { _channel.stream.listen((packet) async { @@ -96,8 +95,12 @@ class IsolateDispatcher { }, onError: (Object error, StackTrace stackTrace) { _handleError(error, stackTrace); }, onDone: () { - _closed = true; - _allIsolates.stream.listen((isolate) => isolate.kill()); + if (_gracefulShutdown) { + _closed = true; + _allIsolates.stream.listen((isolate) => isolate.kill()); + } else { + exit(exitCode); + } }); } @@ -112,7 +115,7 @@ class IsolateDispatcher { isolate = _inactiveIsolates.first; _inactiveIsolates.remove(isolate); } else { - var future = ReusableIsolate.spawn(_isolateMain, + var future = ReusableIsolate.spawn(isolateMain, onError: (Object error, StackTrace stackTrace) { _handleError(error, stackTrace); }); @@ -124,11 +127,11 @@ class IsolateDispatcher { var fullBuffer = message as Uint8List; // The first byte of messages from isolates indicates whether the entire - // compilation is finished (1) or if it encountered an error (2). Sending - // this as part of the message buffer rather than a separate message - // avoids a race condition where the host might send a new compilation - // request with the same ID as one that just finished before the - // [IsolateDispatcher] receives word that the isolate with that ID is + // compilation is finished (1) or if it encountered an error (exitCode). + // Sending this as part of the message buffer rather than a separate + // message avoids a race condition where the host might send a new + // compilation request with the same ID as one that just finished before + // the [IsolateDispatcher] receives word that the isolate with that ID is // done. See sass/dart-sass#2004. var category = fullBuffer[0]; var packet = Uint8List.sublistView(fullBuffer, 1); @@ -142,9 +145,14 @@ class IsolateDispatcher { _inactiveIsolates.add(isolate); resource.release(); _channel.sink.add(packet); - case 2: + default: _channel.sink.add(packet); - exit(exitCode); + exitCode = category; + if (_gracefulShutdown) { + _channel.sink.close(); + } else { + exit(exitCode); + } } }); @@ -168,7 +176,11 @@ class IsolateDispatcher { {int? compilationId, int? messageId}) { sendError(compilationId ?? errorId, handleError(error, stackTrace, messageId: messageId)); - _channel.sink.close(); + if (_gracefulShutdown) { + _channel.sink.close(); + } else { + exit(exitCode); + } } /// Sends [message] to the host. @@ -179,7 +191,3 @@ class IsolateDispatcher { void sendError(int compilationId, ProtocolError error) => _send(compilationId, OutboundMessage()..error = error); } - -void _isolateMain(Mailbox mailbox, SendPort sendPort) { - CompilationDispatcher(mailbox, sendPort).listen(); -} diff --git a/lib/src/embedded/isolate_main.dart b/lib/src/embedded/isolate_main.dart new file mode 100644 index 000000000..070992a14 --- /dev/null +++ b/lib/src/embedded/isolate_main.dart @@ -0,0 +1,12 @@ +// Copyright 2024 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 'dart:isolate' show SendPort; + +import 'compilation_dispatcher.dart'; +import 'sync_receive_port.dart'; + +void isolateMain(SyncReceivePort receivePort, SendPort sendPort) { + CompilationDispatcher(receivePort, sendPort).listen(); +} diff --git a/lib/src/embedded/js/concurrency.dart b/lib/src/embedded/js/concurrency.dart new file mode 100644 index 000000000..57205479f --- /dev/null +++ b/lib/src/embedded/js/concurrency.dart @@ -0,0 +1,10 @@ +// Copyright 2024 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 'dart:js_interop'; + +@JS('os.cpus') +external JSArray _cpus(); + +int get concurrencyLimit => _cpus().length; diff --git a/lib/src/embedded/js/executable.dart b/lib/src/embedded/js/executable.dart new file mode 100644 index 000000000..724d10c3c --- /dev/null +++ b/lib/src/embedded/js/executable.dart @@ -0,0 +1,27 @@ +// Copyright 2024 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:stream_channel/stream_channel.dart'; + +import '../isolate_dispatcher.dart'; +import '../isolate_main.dart'; +import '../options.dart'; +import '../util/length_delimited_transformer.dart'; +import 'io.dart'; +import 'sync_receive_port.dart'; +import 'worker_threads.dart'; + +void main(List args) { + if (parseOptions(args)) { + if (isMainThread) { + IsolateDispatcher(StreamChannel.withGuarantees(stdin, stdout, + allowSinkErrors: false) + .transform(lengthDelimited)) + .listen(); + } else { + var port = workerData! as MessagePort; + isolateMain(JSSyncReceivePort(port), JSSendPort(port)); + } + } +} diff --git a/lib/src/embedded/js/io.dart b/lib/src/embedded/js/io.dart new file mode 100644 index 000000000..981a0fb65 --- /dev/null +++ b/lib/src/embedded/js/io.dart @@ -0,0 +1,66 @@ +// Copyright 2024 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 'dart:async'; +import 'dart:js_interop'; +import 'dart:typed_data'; + +@JS('process.exitCode') +external int? get _exitCode; +int get exitCode => _exitCode ?? 0; + +@JS('process.exitCode') +external set exitCode(int code); + +@JS('process.exit') +external void exit([int code]); + +@JS() +extension type _ReadStream(JSObject _) implements JSObject { + external void destroy(); + external void on(String type, JSFunction listener); +} + +@JS('process.stdin') +external _ReadStream get _stdin; + +@JS() +extension type _WriteStream(JSObject _) implements JSObject { + external void write(JSUint8Array chunk); +} + +@JS('process.stdout') +external _WriteStream get _stdout; + +Stream> get stdin { + var controller = StreamController( + onCancel: () { + _stdin.destroy(); + }, + sync: true); + _stdin.on( + 'data', + (JSUint8Array chunk) { + controller.sink.add(chunk.toDart); + }.toJS); + _stdin.on( + 'end', + () { + controller.sink.close(); + }.toJS); + _stdin.on( + 'error', + (JSObject e) { + controller.sink.addError(e); + }.toJS); + return controller.stream; +} + +StreamSink> get stdout { + var controller = StreamController(sync: true); + controller.stream.listen((buffer) { + _stdout.write(buffer.toJS); + }); + return controller.sink; +} diff --git a/lib/src/embedded/js/isolate.dart b/lib/src/embedded/js/isolate.dart new file mode 100644 index 000000000..1db935114 --- /dev/null +++ b/lib/src/embedded/js/isolate.dart @@ -0,0 +1,17 @@ +// Copyright 2024 Google LLC. 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 'dart:isolate' show SendPort; +export 'dart:isolate' show SendPort; + +import 'io.dart' as io; + +abstract class Isolate { + static Never exit([SendPort? finalMessagePort, Object? message]) { + if (message != null) { + finalMessagePort?.send(message); + } + io.exit(io.exitCode) as Never; + } +} diff --git a/lib/src/embedded/js/isolate_main.dart b/lib/src/embedded/js/isolate_main.dart new file mode 100644 index 000000000..1224877fd --- /dev/null +++ b/lib/src/embedded/js/isolate_main.dart @@ -0,0 +1,14 @@ +// Copyright 2024 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 'dart:js_interop'; + +import 'js.dart'; + +@JS('process.argv') +external JSArray get _argv; + +(String, JSArray) isolateMain() { + return ((_argv[1]! as JSString).toDart, _argv.slice(2) as JSArray); +} diff --git a/lib/src/embedded/js/js.dart b/lib/src/embedded/js/js.dart new file mode 100644 index 000000000..040cbf0b2 --- /dev/null +++ b/lib/src/embedded/js/js.dart @@ -0,0 +1,13 @@ +// Copyright 2024 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 'dart:js_interop'; + +extension JSTypedArrayExtension on JSTypedArray { + external JSArrayBuffer get buffer; +} + +extension JSArrayExtension on JSArray { + external JSArray slice([int start, int end]); +} diff --git a/lib/src/embedded/js/reusable_isolate.dart b/lib/src/embedded/js/reusable_isolate.dart new file mode 100644 index 000000000..e80173c84 --- /dev/null +++ b/lib/src/embedded/js/reusable_isolate.dart @@ -0,0 +1,103 @@ +// Copyright 2024 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 'dart:async'; +import 'dart:js_interop'; +import 'dart:typed_data'; + +import 'js.dart'; +import 'sync_message_port.dart'; +import 'worker_threads.dart'; + +/// The entrypoint for a [ReusableIsolate]. +/// +/// This must return a Record of filename and argv for creating the Worker. +typedef ReusableIsolateEntryPoint = (String, JSArray) Function(); + +class ReusableIsolate { + /// The worker. + final Worker _worker; + + /// The [MessagePort] used to receive messages to the [Worker]. + final MessagePort _receivePort; + + /// The [SyncMessagePort] used to send to the [Worker]. + final SyncMessagePort _sendPort; + + /// The subscription to [_receivePort]. + final StreamSubscription _subscription; + + /// Whether the current isolate has been borrowed. + bool _borrowed = false; + + ReusableIsolate._( + this._worker, this._sendPort, this._receivePort, this._subscription); + + /// Spawns a [ReusableIsolate] that runs the the entrypoint script. + static Future spawn(ReusableIsolateEntryPoint entryPoint, + {Function? onError}) async { + var (filename, argv) = entryPoint(); + var channel = SyncMessagePort.createChannel(); + var worker = Worker( + filename, + WorkerOptions( + workerData: channel.port2, + transferList: [channel.port2].toJS, + argv: argv)); + var controller = StreamController(sync: true); + var sendPort = SyncMessagePort(channel.port1); + var receivePort = channel.port1; + receivePort.on( + 'message', + ((JSUint8Array buffer) { + controller.add(buffer.toDart); + }).toJS); + return ReusableIsolate._(worker, sendPort, receivePort, + controller.stream.listen(_defaultOnData)); + } + + /// Subscribe to messages from [_receivePort]. + void borrow(void onData(dynamic event)?) { + if (_borrowed) { + throw StateError('ReusableIsolate has already been borrowed.'); + } + _borrowed = true; + _subscription.onData(onData); + } + + /// Unsubscribe to messages from [_receivePort]. + void release() { + if (!_borrowed) { + throw StateError('ReusableIsolate has not been borrowed.'); + } + _borrowed = false; + _subscription.onData(_defaultOnData); + } + + /// Sends [message] to the isolate. + /// + /// Throws a [StateError] if this is called while the isolate isn't borrowed, + /// or if a second message is sent before the isolate has processed the first + /// one. + void send(Uint8List message) { + if (!_borrowed) { + throw StateError('Cannot send a message before being borrowed.'); + } + var array = message.toJS; + _sendPort.postMessage(array, [array.buffer].toJS); + } + + /// Shuts down the isolate. + void kill() { + _sendPort.close(); + _worker.terminate(); + _receivePort.close(); + } +} + +/// The default handler for data events from the wrapped isolate when it's not +/// borrowed. +void _defaultOnData(dynamic _) { + throw StateError("Shouldn't receive a message before being borrowed."); +} diff --git a/lib/src/embedded/js/sync_message_port.dart b/lib/src/embedded/js/sync_message_port.dart new file mode 100644 index 000000000..2630ec959 --- /dev/null +++ b/lib/src/embedded/js/sync_message_port.dart @@ -0,0 +1,15 @@ +// Copyright 2024 Google LLC. 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 'dart:js_interop'; +import 'worker_threads.dart'; + +@JS('sync_message_port.SyncMessagePort') +extension type SyncMessagePort._(JSObject _) implements JSObject { + external static MessageChannel createChannel(); + external SyncMessagePort(MessagePort port); + external void postMessage(JSAny? value, [JSArray transferList]); + external JSAny? receiveMessage(); + external void close(); +} diff --git a/lib/src/embedded/js/sync_receive_port.dart b/lib/src/embedded/js/sync_receive_port.dart new file mode 100644 index 000000000..37b915e36 --- /dev/null +++ b/lib/src/embedded/js/sync_receive_port.dart @@ -0,0 +1,33 @@ +// Copyright 2024 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 'dart:js_interop'; +import 'dart:typed_data'; + +import '../sync_receive_port.dart'; +import 'isolate.dart'; +import 'js.dart'; +import 'sync_message_port.dart'; +import 'worker_threads.dart'; + +final class JSSyncReceivePort implements SyncReceivePort { + final SyncMessagePort _port; + + JSSyncReceivePort(MessagePort port) : _port = SyncMessagePort(port); + + Uint8List receive() { + return (_port.receiveMessage()! as JSUint8Array).toDart; + } +} + +final class JSSendPort implements SendPort { + final MessagePort _port; + + JSSendPort(this._port); + + void send(Object? message) { + var array = (message! as Uint8List).toJS; + _port.postMessage(array, [array.buffer].toJS); + } +} diff --git a/lib/src/embedded/js/worker_threads.dart b/lib/src/embedded/js/worker_threads.dart new file mode 100644 index 000000000..854e8c698 --- /dev/null +++ b/lib/src/embedded/js/worker_threads.dart @@ -0,0 +1,71 @@ +// Copyright 2024 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 'dart:js_interop'; + +@JS('worker_threads.isMainThread') +external bool get isMainThread; + +@JS('worker_threads.workerData') +external JSAny? get workerData; + +@JS('worker_threads.Worker') +extension type Worker._(JSObject _) implements JSObject { + external Worker(String filename, WorkerOptions options); + external void once(String type, JSFunction listener); + external void terminate(); +} + +@JS() +extension type WorkerOptions._(JSObject _) implements JSObject { + external WorkerOptions( + {JSArray argv, + JSObject env, + bool eval, + JSArray execArgv, + bool stdin, + bool stdout, + bool stderr, + JSAny workerData, + bool trackUnmanagedFds, + JSArray transferList, + ResourceLimits resourceLimits}); + external JSArray get argv; + external JSObject get env; + external bool get eval; + external JSArray get execArgv; + external bool get stdin; + external bool get stdout; + external bool get stderr; + external JSAny get workerData; + external bool get trackUnmanagedFds; + external JSArray get transferList; + external ResourceLimits get resourceLimits; +} + +@JS() +extension type ResourceLimits._(JSObject _) implements JSObject { + external ResourceLimits( + {int maxYoungGenerationSizeMb, + int maxOldGenerationSizeMb, + int codeRangeSizeMb, + int stackSizeMb}); + external int get maxYoungGenerationSizeMb; + external int get maxOldGenerationSizeMb; + external int get codeRangeSizeMb; + external int get stackSizeMb; +} + +@JS() +extension type MessageChannel._(JSObject _) implements JSObject { + external MessagePort get port1; + external MessagePort get port2; +} + +@JS() +extension type MessagePort._(JSObject _) implements JSObject { + external void postMessage(JSAny? value, [JSArray transferList]); + external void on(String type, JSFunction listener); + external void close(); +} diff --git a/lib/src/embedded/options.dart b/lib/src/embedded/options.dart new file mode 100644 index 000000000..e680a23dd --- /dev/null +++ b/lib/src/embedded/options.dart @@ -0,0 +1,32 @@ +// Copyright 2024 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 'dart:convert'; + +import '../io.dart'; +import 'isolate_dispatcher.dart'; + +/// Returns true if should start embedded compiler, +/// and false if should exit. +bool parseOptions(List args) { + switch (args) { + case ["--version", ...]: + var response = IsolateDispatcher.versionResponse(); + response.id = 0; + safePrint(JsonEncoder.withIndent(" ").convert(response.toProto3Json())); + return false; + + case [_, ...]: + printError( + "sass --embedded is not intended to be executed with additional " + "arguments.\n" + "See https://github.com/sass/dart-sass#embedded-dart-sass for " + "details."); + // USAGE error from https://bit.ly/2poTt90 + exitCode = 64; + return false; + } + + return true; +} diff --git a/lib/src/embedded/reusable_isolate.dart b/lib/src/embedded/reusable_isolate.dart index 4140f5a37..bf8236b1c 100644 --- a/lib/src/embedded/reusable_isolate.dart +++ b/lib/src/embedded/reusable_isolate.dart @@ -9,6 +9,8 @@ import 'dart:typed_data'; import 'package:native_synchronization/mailbox.dart'; import 'package:native_synchronization/sendable.dart'; +import 'sync_receive_port.dart'; + /// The entrypoint for a [ReusableIsolate]. /// /// This must be a static global function. It's run when the isolate is spawned, @@ -19,7 +21,7 @@ import 'package:native_synchronization/sendable.dart'; /// If the [sendPort] sends a message before [ReusableIsolate.borrow] is called, /// this will throw an unhandled [StateError]. typedef ReusableIsolateEntryPoint = FutureOr Function( - Mailbox mailbox, SendPort sink); + SyncReceivePort receivePort, SendPort sendPort); class ReusableIsolate { /// The wrapped isolate. @@ -100,5 +102,5 @@ void _defaultOnData(dynamic _) { void _isolateMain( (ReusableIsolateEntryPoint, Sendable, SendPort) message) { var (entryPoint, sendableMailbox, sendPort) = message; - entryPoint(sendableMailbox.materialize(), sendPort); + entryPoint(MailboxSyncReceivePort(sendableMailbox.materialize()), sendPort); } diff --git a/lib/src/embedded/sync_receive_port.dart b/lib/src/embedded/sync_receive_port.dart new file mode 100644 index 000000000..2bc1e0060 --- /dev/null +++ b/lib/src/embedded/sync_receive_port.dart @@ -0,0 +1,13 @@ +// Copyright 2024 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 'dart:typed_data'; +export 'vm/sync_receive_port.dart' + if (dart.library.js) 'js/sync_receive_port.dart'; + +/// A common interface that is implemented by wrapping +/// Dart Mailbox or JS SyncMessagePort. +abstract interface class SyncReceivePort { + Uint8List receive(); +} diff --git a/lib/src/embedded/unavailable.dart b/lib/src/embedded/unavailable.dart deleted file mode 100644 index bf03a52bf..000000000 --- a/lib/src/embedded/unavailable.dart +++ /dev/null @@ -1,10 +0,0 @@ -// 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 '../io.dart'; - -void main(List args) async { - printError('sass --embedded is unavailable in pure JS mode.'); - exitCode = 1; -} diff --git a/lib/src/embedded/utils.dart b/lib/src/embedded/utils.dart index ff987cf2f..f962c5549 100644 --- a/lib/src/embedded/utils.dart +++ b/lib/src/embedded/utils.dart @@ -2,7 +2,6 @@ // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. -import 'dart:io'; import 'dart:typed_data'; import 'package:protobuf/protobuf.dart'; @@ -10,6 +9,7 @@ import 'package:source_span/source_span.dart'; import 'package:stack_trace/stack_trace.dart'; import 'package:term_glyph/term_glyph.dart' as term_glyph; +import '../io.dart'; import '../syntax.dart'; import 'embedded_sass.pb.dart' as proto; import 'embedded_sass.pb.dart' hide SourceSpan, Syntax; @@ -136,15 +136,17 @@ ProtocolError handleError(Object error, StackTrace stackTrace, {int? messageId}) { if (error is ProtocolError) { error.id = messageId ?? errorId; - stderr.write("Host caused ${error.type.name.toLowerCase()} error"); - if (error.id != errorId) stderr.write(" with request ${error.id}"); - stderr.writeln(": ${error.message}"); + var buffer = StringBuffer(); + buffer.write("Host caused ${error.type.name.toLowerCase()} error"); + if (error.id != errorId) buffer.write(" with request ${error.id}"); + buffer.write(": ${error.message}"); + printError(buffer.toString()); // PROTOCOL error from https://bit.ly/2poTt90 exitCode = 76; // EX_PROTOCOL return error; } else { var errorMessage = "$error\n${Chain.forTrace(stackTrace)}"; - stderr.write("Internal compiler error: $errorMessage"); + printError("Internal compiler error: $errorMessage"); exitCode = 70; // EX_SOFTWARE return ProtocolError() ..type = ProtocolErrorType.INTERNAL diff --git a/lib/src/embedded/vm/sync_receive_port.dart b/lib/src/embedded/vm/sync_receive_port.dart new file mode 100644 index 000000000..0ad7b04d5 --- /dev/null +++ b/lib/src/embedded/vm/sync_receive_port.dart @@ -0,0 +1,19 @@ +// Copyright 2024 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 'dart:typed_data'; + +import 'package:native_synchronization/mailbox.dart'; + +import '../sync_receive_port.dart'; + +final class MailboxSyncReceivePort implements SyncReceivePort { + final Mailbox _mailbox; + + MailboxSyncReceivePort(this._mailbox); + + Uint8List receive() { + return _mailbox.take(); + } +} diff --git a/package.json b/package.json index 0eb2cbfa9..03ebbc741 100644 --- a/package.json +++ b/package.json @@ -9,6 +9,7 @@ "@parcel/watcher": "^2.4.1", "chokidar": "^4.0.0", "immutable": "^5.0.2", - "intercept-stdout": "^0.1.2" + "intercept-stdout": "^0.1.2", + "sync-message-port": "v1.1.1" } } diff --git a/package/package.json b/package/package.json index 5b096250a..4a1c0d772 100644 --- a/package/package.json +++ b/package/package.json @@ -19,7 +19,8 @@ "dependencies": { "chokidar": "^4.0.0", "immutable": "^5.0.2", - "source-map-js": ">=0.6.2 <2.0.0" + "source-map-js": ">=0.6.2 <2.0.0", + "sync-message-port": "^1.1.1" }, "optionalDependencies": { "@parcel/watcher": "^2.4.1" diff --git a/pkg/sass_api/pubspec.yaml b/pkg/sass_api/pubspec.yaml index 8ec0f5503..700160dcf 100644 --- a/pkg/sass_api/pubspec.yaml +++ b/pkg/sass_api/pubspec.yaml @@ -7,7 +7,7 @@ description: Additional APIs for Dart Sass. homepage: https://github.com/sass/dart-sass environment: - sdk: ">=3.3.0 <4.0.0" + sdk: ">=3.6.0 <4.0.0" dependencies: sass: 1.83.4 diff --git a/pubspec.yaml b/pubspec.yaml index 51fe325d2..6c9d3d99e 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -8,7 +8,7 @@ executables: sass: sass environment: - sdk: ">=3.3.0 <4.0.0" + sdk: ">=3.6.0 <4.0.0" dependencies: args: ^2.0.0 diff --git a/test/embedded/dart/file_importer_test.dart b/test/embedded/dart/file_importer_test.dart new file mode 100644 index 000000000..b9be558da --- /dev/null +++ b/test/embedded/dart/file_importer_test.dart @@ -0,0 +1,16 @@ +// Copyright 2024 Google LLC. 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('vm') +library; + +import 'package:test/test.dart'; + +import '../shared/file_importer.dart'; +import '../dart_test.dart'; + +void main() { + setUpAll(ensureSnapshotUpToDate); + sharedTests(runSassEmbedded); +} diff --git a/test/embedded/dart/function_test.dart b/test/embedded/dart/function_test.dart new file mode 100644 index 000000000..4ea5fdb8e --- /dev/null +++ b/test/embedded/dart/function_test.dart @@ -0,0 +1,16 @@ +// Copyright 2024 Google LLC. 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('vm') +library; + +import 'package:test/test.dart'; + +import '../shared/function.dart'; +import '../dart_test.dart'; + +void main() { + setUpAll(ensureSnapshotUpToDate); + sharedTests(runSassEmbedded); +} diff --git a/test/embedded/dart/importer_test.dart b/test/embedded/dart/importer_test.dart new file mode 100644 index 000000000..a3ce7a3e2 --- /dev/null +++ b/test/embedded/dart/importer_test.dart @@ -0,0 +1,16 @@ +// Copyright 2024 Google LLC. 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('vm') +library; + +import 'package:test/test.dart'; + +import '../shared/importer.dart'; +import '../dart_test.dart'; + +void main() { + setUpAll(ensureSnapshotUpToDate); + sharedTests(runSassEmbedded); +} diff --git a/test/embedded/dart/length_delimited_test.dart b/test/embedded/dart/length_delimited_test.dart new file mode 100644 index 000000000..9c7991bc7 --- /dev/null +++ b/test/embedded/dart/length_delimited_test.dart @@ -0,0 +1,16 @@ +// Copyright 2024 Google LLC. 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('vm') +library; + +import 'package:test/test.dart'; + +import '../shared/length_delimited.dart'; +import '../dart_test.dart'; + +void main() { + setUpAll(ensureSnapshotUpToDate); + sharedTests(); +} diff --git a/test/embedded/dart/protocol_test.dart b/test/embedded/dart/protocol_test.dart new file mode 100644 index 000000000..0223773ff --- /dev/null +++ b/test/embedded/dart/protocol_test.dart @@ -0,0 +1,16 @@ +// Copyright 2024 Google LLC. 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('vm') +library; + +import 'package:test/test.dart'; + +import '../shared/protocol.dart'; +import '../dart_test.dart'; + +void main() { + setUpAll(ensureSnapshotUpToDate); + sharedTests(runSassEmbedded); +} diff --git a/test/embedded/dart_test.dart b/test/embedded/dart_test.dart new file mode 100644 index 000000000..bac02cfc7 --- /dev/null +++ b/test/embedded/dart_test.dart @@ -0,0 +1,22 @@ +// Copyright 2024 Google LLC. 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('vm') +library; + +import 'package:cli_pkg/testing.dart' as pkg; +import 'package:test/test.dart'; + +import 'shared/embedded_process.dart'; + +void main() {} + +/// Ensures that the snapshot of the Dart executable used by [runSassEmbedded] is +/// up-to-date, if one has been generated. +void ensureSnapshotUpToDate() => pkg.ensureExecutableUpToDate("sass"); + +Future runSassEmbedded( + [Iterable args = const Iterable.empty()]) => + EmbeddedProcess.start(pkg.executableRunner("sass"), + [...pkg.executableArgs("sass"), "--embedded", ...args]); diff --git a/test/embedded/node/file_importer_test.dart b/test/embedded/node/file_importer_test.dart new file mode 100644 index 000000000..3723d2342 --- /dev/null +++ b/test/embedded/node/file_importer_test.dart @@ -0,0 +1,17 @@ +// Copyright 2024 Google LLC. 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('vm') +@Tags(['node']) +library; + +import 'package:test/test.dart'; + +import '../shared/file_importer.dart'; +import '../node_test.dart'; + +void main() { + setUpAll(ensureSnapshotUpToDate); + sharedTests(runSassEmbedded); +} diff --git a/test/embedded/node/function_test.dart b/test/embedded/node/function_test.dart new file mode 100644 index 000000000..9e1daf692 --- /dev/null +++ b/test/embedded/node/function_test.dart @@ -0,0 +1,17 @@ +// Copyright 2024 Google LLC. 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('vm') +@Tags(['node']) +library; + +import 'package:test/test.dart'; + +import '../shared/function.dart'; +import '../node_test.dart'; + +void main() { + setUpAll(ensureSnapshotUpToDate); + sharedTests(runSassEmbedded); +} diff --git a/test/embedded/node/importer_test.dart b/test/embedded/node/importer_test.dart new file mode 100644 index 000000000..cb6843bcc --- /dev/null +++ b/test/embedded/node/importer_test.dart @@ -0,0 +1,17 @@ +// Copyright 2024 Google LLC. 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('vm') +@Tags(['node']) +library; + +import 'package:test/test.dart'; + +import '../shared/importer.dart'; +import '../node_test.dart'; + +void main() { + setUpAll(ensureSnapshotUpToDate); + sharedTests(runSassEmbedded); +} diff --git a/test/embedded/node/length_delimited_test.dart b/test/embedded/node/length_delimited_test.dart new file mode 100644 index 000000000..b3a425e8f --- /dev/null +++ b/test/embedded/node/length_delimited_test.dart @@ -0,0 +1,17 @@ +// Copyright 2024 Google LLC. 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('vm') +@Tags(['node']) +library; + +import 'package:test/test.dart'; + +import '../shared/length_delimited.dart'; +import '../node_test.dart'; + +void main() { + setUpAll(ensureSnapshotUpToDate); + sharedTests(); +} diff --git a/test/embedded/node/protocol_test.dart b/test/embedded/node/protocol_test.dart new file mode 100644 index 000000000..f6c8c658b --- /dev/null +++ b/test/embedded/node/protocol_test.dart @@ -0,0 +1,17 @@ +// Copyright 2024 Google LLC. 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('vm') +@Tags(['node']) +library; + +import 'package:test/test.dart'; + +import '../shared/protocol.dart'; +import '../node_test.dart'; + +void main() { + setUpAll(ensureSnapshotUpToDate); + sharedTests(runSassEmbedded); +} diff --git a/test/embedded/node_test.dart b/test/embedded/node_test.dart new file mode 100644 index 000000000..5af39b2d7 --- /dev/null +++ b/test/embedded/node_test.dart @@ -0,0 +1,24 @@ +// Copyright 2024 Google LLC. 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('vm') +@Tags(['node']) +library; + +import 'package:cli_pkg/testing.dart' as pkg; +import 'package:test/test.dart'; + +import '../ensure_npm_package.dart'; +import 'shared/embedded_process.dart'; + +void main() {} + +/// Ensures that the snapshot of the npm package used by [runSassEmbedded] is +/// up-to-date, if one has been generated. +void ensureSnapshotUpToDate() => ensureNpmPackage; + +Future runSassEmbedded( + [Iterable args = const Iterable.empty()]) => + EmbeddedProcess.start(pkg.executableRunner("sass", node: true), + [...pkg.executableArgs("sass", node: true), "--embedded", ...args]); diff --git a/test/embedded/embedded_process.dart b/test/embedded/shared/embedded_process.dart similarity index 97% rename from test/embedded/embedded_process.dart rename to test/embedded/shared/embedded_process.dart index 89740b25c..872fe9da8 100644 --- a/test/embedded/embedded_process.dart +++ b/test/embedded/shared/embedded_process.dart @@ -2,15 +2,11 @@ // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. -@TestOn('vm') -library; - import 'dart:async'; import 'dart:convert'; import 'dart:io'; import 'package:async/async.dart'; -import 'package:cli_pkg/testing.dart' as pkg; import 'package:test/test.dart'; import 'package:sass/src/embedded/embedded_sass.pb.dart'; @@ -85,14 +81,13 @@ class EmbeddedProcess { /// If [forwardOutput] is `true`, the process's [outbound] messages and /// [stderr] will be printed to the console as they appear. This is only /// intended to be set temporarily to help when debugging test failures. - static Future start( + static Future start(String command, List args, {String? workingDirectory, Map? environment, bool includeParentEnvironment = true, bool runInShell = false, bool forwardOutput = false}) async { - var process = await Process.start(pkg.executableRunner("sass"), - [...pkg.executableArgs("sass"), "--embedded"], + var process = await Process.start(command, args, workingDirectory: workingDirectory, environment: environment, includeParentEnvironment: includeParentEnvironment, diff --git a/test/embedded/file_importer_test.dart b/test/embedded/shared/file_importer.dart similarity index 98% rename from test/embedded/file_importer_test.dart rename to test/embedded/shared/file_importer.dart index 61c8e1a42..1337249b3 100644 --- a/test/embedded/file_importer_test.dart +++ b/test/embedded/shared/file_importer.dart @@ -2,9 +2,6 @@ // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. -@TestOn('vm') -library; - import 'package:path/path.dart' as p; import 'package:test/test.dart'; import 'package:test_descriptor/test_descriptor.dart' as d; @@ -15,10 +12,10 @@ import 'package:sass/src/embedded/utils.dart'; import 'embedded_process.dart'; import 'utils.dart'; -void main() { +void sharedTests(Future runSassEmbedded()) { late EmbeddedProcess process; setUp(() async { - process = await EmbeddedProcess.start(); + process = await runSassEmbedded(); }); group("emits a protocol error", () { diff --git a/test/embedded/function_test.dart b/test/embedded/shared/function.dart similarity index 99% rename from test/embedded/function_test.dart rename to test/embedded/shared/function.dart index 584151e79..c8630efd7 100644 --- a/test/embedded/function_test.dart +++ b/test/embedded/shared/function.dart @@ -2,9 +2,6 @@ // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. -@TestOn('vm') -library; - import 'package:test/test.dart'; import 'package:sass/src/embedded/embedded_sass.pb.dart'; @@ -19,9 +16,9 @@ final _null = Value()..singleton = SingletonValue.NULL; late EmbeddedProcess _process; -void main() { +void sharedTests(Future runSassEmbedded()) async { setUp(() async { - _process = await EmbeddedProcess.start(); + _process = await runSassEmbedded(); }); group("emits a compile failure for a custom function with a signature", () { diff --git a/test/embedded/importer_test.dart b/test/embedded/shared/importer.dart similarity index 99% rename from test/embedded/importer_test.dart rename to test/embedded/shared/importer.dart index ee4d90c0f..87c2771ee 100644 --- a/test/embedded/importer_test.dart +++ b/test/embedded/shared/importer.dart @@ -2,9 +2,6 @@ // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. -@TestOn('vm') -library; - import 'package:source_maps/source_maps.dart' as source_maps; import 'package:test/test.dart'; import 'package:test_descriptor/test_descriptor.dart' as d; @@ -15,10 +12,10 @@ import 'package:sass/src/embedded/utils.dart'; import 'embedded_process.dart'; import 'utils.dart'; -void main() { +void sharedTests(Future runSassEmbedded()) { late EmbeddedProcess process; setUp(() async { - process = await EmbeddedProcess.start(); + process = await runSassEmbedded(); }); group("emits a protocol error", () { diff --git a/test/embedded/length_delimited_test.dart b/test/embedded/shared/length_delimited.dart similarity index 98% rename from test/embedded/length_delimited_test.dart rename to test/embedded/shared/length_delimited.dart index 1d4c6fb62..ef092c182 100644 --- a/test/embedded/length_delimited_test.dart +++ b/test/embedded/shared/length_delimited.dart @@ -2,9 +2,6 @@ // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. -@TestOn('vm') -library; - import 'dart:async'; import 'dart:typed_data'; @@ -13,7 +10,7 @@ import 'package:sass/src/embedded/util/length_delimited_transformer.dart'; import 'package:async/async.dart'; import 'package:test/test.dart'; -void main() { +void sharedTests() { group("encoder", () { late Sink> sink; late Stream> stream; diff --git a/test/embedded/protocol_test.dart b/test/embedded/shared/protocol.dart similarity index 98% rename from test/embedded/protocol_test.dart rename to test/embedded/shared/protocol.dart index c7473ac69..d09580bf3 100644 --- a/test/embedded/protocol_test.dart +++ b/test/embedded/shared/protocol.dart @@ -2,9 +2,6 @@ // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. -@TestOn('vm') -library; - import 'package:path/path.dart' as p; import 'package:pub_semver/pub_semver.dart'; import 'package:source_maps/source_maps.dart' as source_maps; @@ -17,10 +14,10 @@ import 'package:sass/src/embedded/utils.dart'; import 'embedded_process.dart'; import 'utils.dart'; -void main() { +void sharedTests(Future runSassEmbedded()) { late EmbeddedProcess process; setUp(() async { - process = await EmbeddedProcess.start(); + process = await runSassEmbedded(); }); group("exits upon protocol error", () { @@ -454,8 +451,15 @@ void main() { (InboundMessage_CompileRequest()..path = d.path("test.scss"))); var failure = await getCompileFailure(process); - expect(failure.message, startsWith("Cannot open file: ")); - expect(failure.message.replaceFirst("Cannot open file: ", "").trim(), + expect( + failure.message, + anyOf(startsWith("Cannot open file: "), + startsWith("no such file or directory: "))); + expect( + failure.message + .replaceFirst("Cannot open file: ", "") + .replaceFirst("no such file or directory: ", "") + .trim(), equalsPath(d.path('test.scss'))); expect(failure.span.text, equals('')); expect(failure.span.context, equals('')); diff --git a/test/embedded/utils.dart b/test/embedded/shared/utils.dart similarity index 100% rename from test/embedded/utils.dart rename to test/embedded/shared/utils.dart diff --git a/tool/grind.dart b/tool/grind.dart index 6f2acc9db..c20f30b2e 100644 --- a/tool/grind.dart +++ b/tool/grind.dart @@ -43,8 +43,11 @@ void main(List args) { pkg.JSRequire("fs", target: pkg.JSRequireTarget.node), pkg.JSRequire("module", target: pkg.JSRequireTarget.node, identifier: 'nodeModule'), + pkg.JSRequire("os", target: pkg.JSRequireTarget.cli), pkg.JSRequire("stream", target: pkg.JSRequireTarget.node), pkg.JSRequire("util", target: pkg.JSRequireTarget.node), + pkg.JSRequire("worker_threads", target: pkg.JSRequireTarget.cli), + pkg.JSRequire("sync-message-port", target: pkg.JSRequireTarget.cli), ]; pkg.jsModuleMainLibrary.value = "lib/src/js.dart"; pkg.npmPackageJson.fn = () =>