mirror of
https://github.com/danog/dart-sass.git
synced 2025-01-22 22:02:00 +01:00
Add a very basic protocol dispatcher and a few tests
Currently this doesn't actually invoke the compiler, it just always returns the expanded output for "a {b: c}".
This commit is contained in:
parent
ea4f9f7204
commit
7489d5796d
31
bin/dart_sass_embedded.dart
Normal file
31
bin/dart_sass_embedded.dart
Normal file
@ -0,0 +1,31 @@
|
||||
// Copyright 2019 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:io';
|
||||
|
||||
import 'package:stream_channel/stream_channel.dart';
|
||||
|
||||
import 'package:sass_embedded/src/dispatcher.dart';
|
||||
import 'package:sass_embedded/src/embedded_sass.pb.dart';
|
||||
import 'package:sass_embedded/src/util/length_delimited_transformer.dart';
|
||||
|
||||
void main(List<String> args) {
|
||||
if (args.isNotEmpty) {
|
||||
stderr.writeln(
|
||||
"This executable is not intended to be executed with arguments.\n"
|
||||
"See https://github.com/sass/embedded-protocol#readme for details.");
|
||||
// USAGE error from https://bit.ly/2poTt90
|
||||
exitCode = 64;
|
||||
return;
|
||||
}
|
||||
|
||||
var dispatcher = Dispatcher(
|
||||
StreamChannel.withGuarantees(stdin, stdout, allowSinkErrors: false)
|
||||
.transform(lengthDelimited));
|
||||
dispatcher.listen((request) {
|
||||
return OutboundMessage_CompileResponse()
|
||||
..success = (OutboundMessage_CompileResponse_CompileSuccess()
|
||||
..css = "a {\n b: c;\n}\n");
|
||||
});
|
||||
}
|
120
lib/src/dispatcher.dart
Normal file
120
lib/src/dispatcher.dart
Normal file
@ -0,0 +1,120 @@
|
||||
// Copyright 2019 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:io';
|
||||
import 'dart:typed_data';
|
||||
|
||||
import 'package:protobuf/protobuf.dart';
|
||||
import 'package:stack_trace/stack_trace.dart';
|
||||
import 'package:stream_channel/stream_channel.dart';
|
||||
|
||||
import 'embedded_sass.pb.dart';
|
||||
|
||||
/// A class that dispatches messages to and from the host.
|
||||
class Dispatcher {
|
||||
/// The channel of encoded protocol buffers, connected to the host.
|
||||
final StreamChannel<Uint8List> _channel;
|
||||
|
||||
/// Creates a [Dispatcher] that sends and receives encoded protocol buffers
|
||||
/// over [channel].
|
||||
Dispatcher(this._channel);
|
||||
|
||||
/// Listens for incoming `CompileRequests` and passes them to [callback].
|
||||
///
|
||||
/// The callback must return a `CompileResponse` which is sent to the host. It
|
||||
/// doesn't need to set [OutboundMessage_CompileResponse.id]; the [Dispatcher]
|
||||
/// will take care of that.
|
||||
///
|
||||
/// This may only be called once.
|
||||
void listen(
|
||||
FutureOr<OutboundMessage_CompileResponse> callback(
|
||||
InboundMessage_CompileRequest request)) {
|
||||
_channel.stream.listen((binaryMessage) async {
|
||||
InboundMessage message;
|
||||
try {
|
||||
try {
|
||||
message = InboundMessage.fromBuffer(binaryMessage);
|
||||
} on InvalidProtocolBufferException catch (error) {
|
||||
throw _parseError(error.message);
|
||||
}
|
||||
|
||||
switch (message.whichMessage()) {
|
||||
case InboundMessage_Message.error:
|
||||
var error = message.ensureError();
|
||||
stderr
|
||||
.write("Host reported ${error.type.name.toLowerCase()} error");
|
||||
if (error.id != -1) stderr.write(" with request ${error.id}");
|
||||
stderr.writeln(": ${error.message}");
|
||||
// SOFTWARE error from https://bit.ly/2poTt90
|
||||
exitCode = 70;
|
||||
_channel.sink.close();
|
||||
break;
|
||||
|
||||
case InboundMessage_Message.compileRequest:
|
||||
var request = message.ensureCompileRequest();
|
||||
var response = await callback(request);
|
||||
response.id = request.id;
|
||||
_send(OutboundMessage()..compileResponse = response);
|
||||
break;
|
||||
|
||||
case InboundMessage_Message.notSet:
|
||||
// PROTOCOL error from https://bit.ly/2poTt90
|
||||
exitCode = 76;
|
||||
throw _parseError("InboundMessage.message is not set.");
|
||||
|
||||
default:
|
||||
// PROTOCOL error from https://bit.ly/2poTt90
|
||||
exitCode = 76;
|
||||
throw _parseError(
|
||||
"Unknown message type: ${message.toDebugString()}");
|
||||
}
|
||||
} on ProtocolError catch (error) {
|
||||
error.id = _messageId(message) ?? -1;
|
||||
stderr.write("Host caused ${error.type.name.toLowerCase()} error");
|
||||
if (error.id != -1) stderr.write(" with request ${error.id}");
|
||||
stderr.writeln(": ${error.message}");
|
||||
_send(OutboundMessage()..error = error);
|
||||
} catch (error, stackTrace) {
|
||||
var errorMessage = "$error\n${Chain.forTrace(stackTrace)}";
|
||||
stderr.write("Internal compiler error: $errorMessage");
|
||||
_send(OutboundMessage()
|
||||
..error = (ProtocolError()
|
||||
..type = ProtocolError_ErrorType.INTERNAL
|
||||
..id = _messageId(message) ?? -1
|
||||
..message = errorMessage));
|
||||
_channel.sink.close();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
/// Sends [message] to the host.
|
||||
void _send(OutboundMessage message) =>
|
||||
_channel.sink.add(message.writeToBuffer());
|
||||
|
||||
/// Returns a [ProtocolError] with type `PARSE` and the given [message].
|
||||
ProtocolError _parseError(String message) => ProtocolError()
|
||||
..type = ProtocolError_ErrorType.PARSE
|
||||
..message = message;
|
||||
|
||||
/// Returns the id for [message] if it it's a request or response, or `null`
|
||||
/// otherwise.
|
||||
int _messageId(InboundMessage message) {
|
||||
if (message == null) return null;
|
||||
switch (message.whichMessage()) {
|
||||
case InboundMessage_Message.compileRequest:
|
||||
return message.ensureCompileRequest().id;
|
||||
case InboundMessage_Message.canonicalizeResponse:
|
||||
return message.ensureCanonicalizeResponse().id;
|
||||
case InboundMessage_Message.importResponse:
|
||||
return message.ensureImportResponse().id;
|
||||
case InboundMessage_Message.functionCallRequest:
|
||||
return message.ensureFunctionCallRequest().id;
|
||||
case InboundMessage_Message.functionCallResponse:
|
||||
return message.ensureFunctionCallResponse().id;
|
||||
default:
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
@ -7,12 +7,17 @@ homepage: https://github.com/sass/dart-sass-embedded
|
||||
environment:
|
||||
sdk: '>=2.4.0 <3.0.0'
|
||||
|
||||
executables:
|
||||
dart-sass-embedded: dart_sass_embedded
|
||||
|
||||
dependencies:
|
||||
async: ">=1.13.0 <3.0.0"
|
||||
protobuf: ^1.0.0
|
||||
stack_trace: ^1.6.0
|
||||
stream_channel: ">=1.6.0 <3.0.0"
|
||||
|
||||
dev_dependencies:
|
||||
grinder: ^0.8.0
|
||||
protoc_plugin: ^19.0.0
|
||||
path: ^1.6.0
|
||||
test: ^1.0.0
|
||||
|
183
test/embedded_process.dart
Normal file
183
test/embedded_process.dart
Normal file
@ -0,0 +1,183 @@
|
||||
// Copyright 2019 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:convert';
|
||||
import 'dart:io';
|
||||
|
||||
import 'package:async/async.dart';
|
||||
import 'package:test/test.dart';
|
||||
|
||||
import 'package:sass_embedded/src/embedded_sass.pb.dart';
|
||||
import 'package:sass_embedded/src/util/length_delimited_transformer.dart';
|
||||
|
||||
/// A wrapper for [Process] that provides a convenient API for testing the
|
||||
/// embedded Sass process.
|
||||
///
|
||||
/// If the test fails, this will automatically print out any stderr and protocol
|
||||
/// buffers from the process to aid debugging.
|
||||
///
|
||||
/// This API is based on the `test_process` package.
|
||||
class EmbeddedProcess {
|
||||
/// The underlying process.
|
||||
final Process _process;
|
||||
|
||||
/// A [StreamQueue] that emits each outbound protocol buffer from the process.
|
||||
StreamQueue<OutboundMessage> get outbound => _outbound;
|
||||
StreamQueue<OutboundMessage> _outbound;
|
||||
|
||||
/// A [StreamQueue] that emits each line of stderr from the process.
|
||||
StreamQueue<String> get stderr => _stderr;
|
||||
StreamQueue<String> _stderr;
|
||||
|
||||
/// A splitter that can emit new copies of [outbound].
|
||||
final StreamSplitter<OutboundMessage> _outboundSplitter;
|
||||
|
||||
/// A splitter that can emit new copies of [stderr].
|
||||
final StreamSplitter<String> _stderrSplitter;
|
||||
|
||||
/// A sink into which inbound messages can be passed to the process.
|
||||
final Sink<InboundMessage> inbound;
|
||||
|
||||
/// The raw standard input byte sink.
|
||||
IOSink get stdin => _process.stdin;
|
||||
|
||||
/// A log that includes lines from [stderr] and human-friendly serializations
|
||||
/// of protocol buffers from [outbound]
|
||||
final _log = <String>[];
|
||||
|
||||
/// Whether [_log] has been passed to [printOnFailure] yet.
|
||||
bool _loggedOutput = false;
|
||||
|
||||
/// Returns a [Future] which completes to the exit code of the process, once
|
||||
/// it completes.
|
||||
Future<int> get exitCode => _process.exitCode;
|
||||
|
||||
/// The process ID of the process.
|
||||
int get pid => _process.pid;
|
||||
|
||||
/// Completes to [_process]'s exit code if it's exited, otherwise completes to
|
||||
/// `null` immediately.
|
||||
Future<int> get _exitCodeOrNull async =>
|
||||
await exitCode.timeout(Duration.zero, onTimeout: () => null);
|
||||
|
||||
/// Starts a process.
|
||||
///
|
||||
/// [executable], [workingDirectory], [environment],
|
||||
/// [includeParentEnvironment], and [runInShell] have the same meaning as for
|
||||
/// [Process.start].
|
||||
///
|
||||
/// 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<EmbeddedProcess> start(
|
||||
{String workingDirectory,
|
||||
Map<String, String> environment,
|
||||
bool includeParentEnvironment = true,
|
||||
bool runInShell = false,
|
||||
bool forwardOutput = false}) async {
|
||||
// TODO(nweiz): Support running from a native executable.
|
||||
var process = await Process.start(
|
||||
Platform.executable, ["bin/dart_sass_embedded.dart"],
|
||||
workingDirectory: workingDirectory,
|
||||
environment: environment,
|
||||
includeParentEnvironment: includeParentEnvironment,
|
||||
runInShell: runInShell);
|
||||
|
||||
return EmbeddedProcess._(process, forwardOutput: forwardOutput);
|
||||
}
|
||||
|
||||
/// Creates a [EmbeddedProcess] for [process].
|
||||
///
|
||||
/// The [forwardOutput] argument is the same as that to [start].
|
||||
EmbeddedProcess._(Process process, {bool forwardOutput = false})
|
||||
: _process = process,
|
||||
_outboundSplitter = StreamSplitter(process.stdout
|
||||
.transform(lengthDelimitedDecoder)
|
||||
.map((message) => OutboundMessage.fromBuffer(message))),
|
||||
_stderrSplitter = StreamSplitter(process.stderr
|
||||
.transform(utf8.decoder)
|
||||
.transform(const LineSplitter())),
|
||||
inbound = StreamSinkTransformer<InboundMessage, List<int>>.fromHandlers(
|
||||
handleData: (message, sink) =>
|
||||
sink.add(message.writeToBuffer())).bind(
|
||||
StreamSinkTransformer.fromStreamTransformer(lengthDelimitedEncoder)
|
||||
.bind(process.stdin)) {
|
||||
addTearDown(_tearDown);
|
||||
expect(_process.exitCode.then((_) => _logOutput()), completes,
|
||||
reason: "Process `dart_sass_embedded` never exited.");
|
||||
|
||||
_outbound = StreamQueue(_outboundSplitter.split());
|
||||
_stderr = StreamQueue(_stderrSplitter.split());
|
||||
|
||||
_outboundSplitter.split().listen((message) {
|
||||
for (var line in message.toDebugString().split("\n")) {
|
||||
if (forwardOutput) print(line);
|
||||
_log.add(" $line");
|
||||
}
|
||||
});
|
||||
|
||||
_stderrSplitter.split().listen((line) {
|
||||
if (forwardOutput) print(line);
|
||||
_log.add("[e] $line");
|
||||
});
|
||||
}
|
||||
|
||||
/// A callback that's run when the test completes.
|
||||
Future _tearDown() async {
|
||||
// If the process is already dead, do nothing.
|
||||
if (await _exitCodeOrNull != null) return;
|
||||
|
||||
_process.kill(ProcessSignal.sigkill);
|
||||
|
||||
// Log output now rather than waiting for the exitCode callback so that
|
||||
// it's visible even if we time out waiting for the process to die.
|
||||
await _logOutput();
|
||||
}
|
||||
|
||||
/// Formats the contents of [_log] and passes them to [printOnFailure].
|
||||
Future _logOutput() async {
|
||||
if (_loggedOutput) return;
|
||||
_loggedOutput = true;
|
||||
|
||||
var exitCodeOrNull = await _exitCodeOrNull;
|
||||
|
||||
// Wait a timer tick to ensure that all available lines have been flushed to
|
||||
// [_log].
|
||||
await Future.delayed(Duration.zero);
|
||||
|
||||
var buffer = StringBuffer();
|
||||
buffer.write("Process `dart_sass_embedded` ");
|
||||
if (exitCodeOrNull == null) {
|
||||
buffer.write("was killed with SIGKILL in a tear-down.");
|
||||
} else {
|
||||
buffer.write("exited with exitCode $exitCodeOrNull.");
|
||||
}
|
||||
buffer.writeln(" Output:");
|
||||
buffer.writeln(_log.join("\n"));
|
||||
|
||||
printOnFailure(buffer.toString());
|
||||
}
|
||||
|
||||
/// Kills the process (with SIGKILL on POSIX operating systems), and returns a
|
||||
/// future that completes once it's dead.
|
||||
///
|
||||
/// If this is called after the process is already dead, it does nothing.
|
||||
Future kill() async {
|
||||
_process.kill(ProcessSignal.sigkill);
|
||||
await exitCode;
|
||||
}
|
||||
|
||||
/// Waits for the process to exit, and verifies that the exit code matches
|
||||
/// [expectedExitCode] (if given).
|
||||
///
|
||||
/// If this is called after the process is already dead, it verifies its
|
||||
/// existing exit code.
|
||||
Future shouldExit([expectedExitCode]) async {
|
||||
var exitCode = await this.exitCode;
|
||||
if (expectedExitCode == null) return;
|
||||
expect(exitCode, expectedExitCode,
|
||||
reason: "Process `dart_sass_embedded` had an unexpected exit code.");
|
||||
}
|
||||
}
|
47
test/protocol_test.dart
Normal file
47
test/protocol_test.dart
Normal file
@ -0,0 +1,47 @@
|
||||
// Copyright 2019 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:test/test.dart';
|
||||
|
||||
import 'package:sass_embedded/src/embedded_sass.pb.dart';
|
||||
|
||||
import 'embedded_process.dart';
|
||||
import 'utils.dart';
|
||||
|
||||
void main() {
|
||||
EmbeddedProcess process;
|
||||
setUp(() async {
|
||||
process = await EmbeddedProcess.start();
|
||||
});
|
||||
|
||||
group("gracefully handles a protocol error", () {
|
||||
test("caused by an empty message", () async {
|
||||
process.inbound.add(InboundMessage());
|
||||
await expectParseError(process, "InboundMessage.message is not set.");
|
||||
await process.kill();
|
||||
});
|
||||
|
||||
test("caused by an invalid message", () async {
|
||||
process.stdin.add([1, 0, 0, 0, 0]);
|
||||
await expectParseError(
|
||||
process, "Protocol message contained an invalid tag (zero).");
|
||||
await process.kill();
|
||||
});
|
||||
|
||||
test("without shutting down the compiler", () async {
|
||||
process.inbound.add(InboundMessage());
|
||||
await expectParseError(process, "InboundMessage.message is not set.");
|
||||
|
||||
process.inbound.add(compileString("a {b: c}"));
|
||||
await expectLater(process.outbound, emits(isSuccess("a { b: c; }")));
|
||||
await process.kill();
|
||||
});
|
||||
});
|
||||
|
||||
test("compiles a CSS from a string", () async {
|
||||
process.inbound.add(compileString("a {b: c}"));
|
||||
await expectLater(process.outbound, emits(isSuccess("a { b: c; }")));
|
||||
await process.kill();
|
||||
});
|
||||
}
|
61
test/utils.dart
Normal file
61
test/utils.dart
Normal file
@ -0,0 +1,61 @@
|
||||
// Copyright 2019 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:test/test.dart';
|
||||
|
||||
import 'package:sass_embedded/src/embedded_sass.pb.dart';
|
||||
|
||||
import 'embedded_process.dart';
|
||||
|
||||
/// Returns a [InboundMessage] that compiles the given plain CSS
|
||||
/// string.
|
||||
InboundMessage compileString(String css) => InboundMessage()
|
||||
..compileRequest = (InboundMessage_CompileRequest()
|
||||
..string = (InboundMessage_CompileRequest_StringInput()..source = css));
|
||||
|
||||
/// Asserts that [process] emits a [ProtocolError] parse error with the given
|
||||
/// [message] on its protobuf stream and prints a notice on stderr.
|
||||
Future<void> expectParseError(EmbeddedProcess process, message) async {
|
||||
await expectLater(process.outbound,
|
||||
emits(isProtocolError(-1, ProtocolError_ErrorType.PARSE, message)));
|
||||
await expectLater(process.stderr, emits("Host caused parse error: $message"));
|
||||
}
|
||||
|
||||
/// Asserts that an [OutboundMessage] is a [ProtocolError] with the given [id],
|
||||
/// [type], and optionally [message].
|
||||
Matcher isProtocolError(int id, ProtocolError_ErrorType type, [message]) =>
|
||||
predicate((value) {
|
||||
expect(value, isA<OutboundMessage>());
|
||||
var outboundMessage = value as OutboundMessage;
|
||||
expect(outboundMessage.hasError(), isTrue,
|
||||
reason: "Expected $message to be a ProtocolError");
|
||||
expect(outboundMessage.error.id, equals(id));
|
||||
expect(outboundMessage.error.type, equals(type));
|
||||
if (message != null) expect(outboundMessage.error.message, message);
|
||||
return true;
|
||||
});
|
||||
|
||||
/// Asserts that [message] is an [OutboundMessage] with a [CompileResponse] and
|
||||
/// returns it.
|
||||
OutboundMessage_CompileResponse getCompileResponse(value) {
|
||||
expect(value, isA<OutboundMessage>());
|
||||
var message = value as OutboundMessage;
|
||||
expect(message.hasCompileResponse(), isTrue,
|
||||
reason: "Expected $message to have a CompileResponse");
|
||||
return message.compileResponse;
|
||||
}
|
||||
|
||||
/// Asserts that an [OutboundMessage] is a [CompileResponse] with CSS that
|
||||
/// matches [css].
|
||||
///
|
||||
/// If [css] is a [String], this automatically wraps it in
|
||||
/// [equalsIgnoringWhitespace].
|
||||
Matcher isSuccess(css) => predicate((value) {
|
||||
var response = getCompileResponse(value);
|
||||
expect(response.hasSuccess(), isTrue,
|
||||
reason: "Expected $response to be successful");
|
||||
expect(response.success.css,
|
||||
css is String ? equalsIgnoringWhitespace(css) : css);
|
||||
return true;
|
||||
});
|
Loading…
x
Reference in New Issue
Block a user