blob: 2ad59b29293234ec5a5c24b60ca0421851a65f1d [file] [log] [blame]
// Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'package:boolean_selector/boolean_selector.dart';
import 'package:source_span/source_span.dart';
import 'compiler.dart';
import 'operating_system.dart';
import 'runtime.dart';
import 'suite_platform.dart';
/// The set of variable names that are valid for all platform selectors.
final _universalValidVariables = {
'posix',
'dart-vm',
'browser',
'js',
'blink',
'google',
'wasm',
for (var runtime in Runtime.builtIn) runtime.identifier,
for (var compiler in Compiler.builtIn) compiler.identifier,
for (var os in OperatingSystem.all) os.identifier,
};
/// An expression for selecting certain platforms, including operating systems
/// and browsers.
///
/// This uses the [boolean selector][] syntax.
///
/// [boolean selector]: https://pub.dev/packages/boolean_selector
final class PlatformSelector {
/// A selector that declares that a test can be run on all platforms.
static const all = PlatformSelector._(BooleanSelector.all);
/// The boolean selector used to implement this selector.
final BooleanSelector _inner;
/// The source span from which this selector was parsed.
final SourceSpan? _span;
/// Parses [selector].
///
/// If [span] is passed, it indicates the location of the text for [selector]
/// in a larger document. It's used for error reporting.
PlatformSelector.parse(String selector, [SourceSpan? span])
: _inner =
_wrapFormatException(() => BooleanSelector.parse(selector), span),
_span = span;
const PlatformSelector._(this._inner) : _span = null;
/// Runs [body] and wraps any [FormatException] it throws in a
/// [SourceSpanFormatException] using [span].
///
/// If [span] is `null`, runs [body] as-is.
static T _wrapFormatException<T>(T Function() body, [SourceSpan? span]) {
if (span == null) return body();
try {
return body();
} on FormatException catch (error) {
throw SourceSpanFormatException(error.message, span);
}
}
/// Throws a [FormatException] if this selector uses any variables that don't
/// appear either in [validVariables] or in the set of variables that are
/// known to be valid for all selectors.
void validate(Set<String> validVariables) {
if (identical(this, all)) return;
_wrapFormatException(
() => _inner.validate((name) =>
_universalValidVariables.contains(name) ||
validVariables.contains(name)),
_span);
}
/// Returns whether the selector matches the given [platform].
bool evaluate(SuitePlatform platform) =>
_inner.evaluate((String variable) => switch (variable) {
_
when variable == platform.runtime.identifier ||
variable == platform.runtime.parent?.identifier ||
variable == platform.os.identifier ||
variable == platform.compiler.identifier =>
true,
'dart-vm' => platform.runtime.isDartVM,
'browser' => platform.runtime.isBrowser,
'js' => platform.runtime.isJS,
'blink' => platform.runtime.isBlink,
'posix' => platform.os.isPosix,
'google' => platform.inGoogle,
'wasm' => platform.runtime.isWasm,
_ => false,
});
/// Returns a new [PlatformSelector] that matches only platforms matched by
/// both [this] and [other].
PlatformSelector intersection(PlatformSelector other) {
if (other == PlatformSelector.all) return this;
return PlatformSelector._(_inner.intersection(other._inner));
}
@override
String toString() => _inner.toString();
@override
bool operator ==(Object other) =>
other is PlatformSelector && _inner == other._inner;
@override
int get hashCode => _inner.hashCode;
}