blob: 21ae6311efc350f4be26d946bbc4c6c2952d59c7 [file] [log] [blame]
// Copyright (c) 2019, 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.
/// Model for a modular test.
/// A modular test declares the structure of the test code: what files are
/// grouped as a module and how modules depend on one another.
class ModularTest {
/// Modules that will be compiled by for modular test
final List<Module> modules;
/// The module containing the main entry method.
final Module mainModule;
/// Flags provided to tools that compile and execute the test.
final List<String> flags;
ModularTest(this.modules, this.mainModule, this.flags) {
if (modules.isEmpty) throw ArgumentError("modules cannot be empty");
for (var module in modules) {
module._validate();
}
}
String debugString() => modules.map((m) => m.debugString()).join('\n');
}
/// A single module in a modular test.
class Module {
/// A short name to identify this module.
final String name;
/// Other modules that need to be compiled first and whose result may be
/// necessary in order to compile this module.
final List<Module> dependencies;
/// Root under which all sources in the module can be found.
final Uri rootUri;
/// Source files that are part of this module only. Stored as a relative [Uri]
/// from [rootUri].
final List<Uri> sources;
/// The file containing the main entry method, if any. Stored as a relative
/// [Uri] from [rootUri].
final Uri? mainSource;
/// Whether this module is also available as a package import, where the
/// package name matches the module name.
bool isPackage;
/// Whether this module represents part of the sdk.
bool isSdk;
/// When [isPackage], the base where all package URIs are resolved against.
/// Stored as a relative [Uri] from [rootUri].
final Uri? packageBase;
/// Whether this is the main entry module of a test.
bool isMain;
/// Whether this module is test specific or shared across tests. Usually this
/// will be true only for the SDK and shared packages like `package:expect`.
bool isShared;
Module(this.name, this.dependencies, this.rootUri, this.sources,
{this.mainSource,
this.isPackage = false,
this.isMain = false,
this.packageBase,
this.isShared = false,
this.isSdk = false}) {
if (!_validModuleName.hasMatch(name)) {
throw ArgumentError("invalid module name: $name");
}
}
void _validate() {
if (!isPackage && !isShared && !isSdk) return;
// Note: we validate this now and not in the constructor because loader.dart
// may update `isPackage` after the module is created.
if (isSdk && isPackage) {
throw InvalidModularTestError("invalid module: $name is an sdk "
"module but was also marked as a package module.");
}
for (var dependency in dependencies) {
if (isPackage && !dependency.isPackage && !dependency.isSdk) {
throw InvalidModularTestError("invalid dependency: $name is a package "
"but it depends on ${dependency.name}, which is not.");
}
if (isShared && !dependency.isShared) {
throw InvalidModularTestError(
"invalid dependency: $name is a shared module "
"but it depends on ${dependency.name}, which is not.");
}
if (isSdk) {
// TODO(sigmund): we should allow to split sdk modules in smaller
// pieces. This requires a bit of work:
// - allow to compile subsets of the sdk (see #30957 regarding
// extraRequiredLibraries in CFE)
// - add logic to specify sdk dependencies.
throw InvalidModularTestError(
"invalid dependency: $name is an sdk module that depends on "
"${dependency.name}, but sdk modules are not expected to "
"have dependencies.");
}
}
}
@override
String toString() => '[module $name]';
String debugString() {
var buffer = StringBuffer();
buffer.write(' ');
buffer.write(name);
buffer.write(': ');
buffer.write(isPackage ? 'package' : '(not package)');
buffer.write(', deps: {${dependencies.map((d) => d.name).join(", ")}}');
if (isSdk) {
buffer.write(', sources: {...omitted ${sources.length} sources...}');
} else {
buffer.write(', sources: {${sources.map((u) => "$u").join(', ')}}');
}
return '$buffer';
}
}
final RegExp _validModuleName = RegExp(r'^[a-zA-Z_][a-zA-Z0-9_]*$');
/// Helper to compute transitive dependencies from [module].
Set<Module> computeTransitiveDependencies(Module module) {
Set<Module> deps = {};
helper(Module m) {
if (deps.add(m)) m.dependencies.forEach(helper);
}
module.dependencies.forEach(helper);
return deps;
}
/// A registry that can map a test configuration to a simple id.
///
/// This is used to help determine whether two tests are run with the same set
/// of flags (the same configuration), and thus pipelines could reuse the
/// results of shared modules from the first test when running the second test.
class ConfigurationRegistry {
final Map<String, int> _configurationId = {};
/// Compute an id to identify the configuration of a modular test.
///
/// A configuration is defined in terms of the set of flags provided to a
/// test. If two test provided to this registry share the same set of flags,
/// the resulting ids are the same. Similarly, if the flags are different,
/// their ids will be different as well.
int computeConfigurationId(ModularTest test) {
return _configurationId[test.flags.join(' ')] ??= _configurationId.length;
}
}
class InvalidModularTestError extends Error {
final String message;
InvalidModularTestError(this.message);
@override
String toString() => "Invalid modular test: $message";
}