-
Notifications
You must be signed in to change notification settings - Fork 213
/
Copy pathbuild_runner_build.dart
113 lines (103 loc) · 3.65 KB
/
build_runner_build.dart
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
// Copyright (c) 2025, 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 'dart:convert';
import 'dart:io';
/// Runs `dart run build_runner build` using `build` packages from pub instead
/// of the local versions. This allows the build to run even when the local
/// versions are in a broken state.
///
/// Usage: in one of the `build` repo packages, instead of running
/// `dart run build_runner build`, run `dart ../tool/build_runner_build.dart`.
void main() {
final pathSegments = Directory.current.uri.pathSegments;
if (pathSegments[pathSegments.length - 3] != 'build') {
print('Current directly should be a package inside the build repo.');
exit(1);
}
// Run `pub get` in a temp folder to get paths for published versions of
// `build` packages that won't break due to local changes.
final tempDirectory = Directory.systemTemp.createTempSync(
'build_runner_build',
);
tempDirectory.createSync(recursive: true);
File.fromUri(tempDirectory.uri.resolve('pubspec.yaml')).writeAsStringSync('''
name: none
environment:
sdk: ^3.7.0
dependencies:
build_runner: '2.4.15'
build_test: any
''');
Process.runSync('dart', ['pub', 'get'], workingDirectory: tempDirectory.path);
final pubConfig = PackageConfig(
json.decode(
File.fromUri(
tempDirectory.uri.resolve('.dart_tool/package_config.json'),
).readAsStringSync(),
)
as Map<String, Object?>,
);
// Merge `pubConfig` into the local package config.
//
// `build_runner` expects to run with the local package config because it
// creates a build script that depends on whatever generators the local
// package uses.
//
// So the merged config will have the two things needed: `build` packages
// not broken by local changes, and whatever generators are needed.
var mergedConfig = PackageConfig(
json.decode(
File.fromUri(
Directory.current.uri.resolve('../.dart_tool/package_config.json'),
).readAsStringSync(),
)
as Map<String, Object?>,
);
late String buildRunnerPath;
for (final package in [
'build',
'build_config',
'build_daemon',
'build_resolvers',
'build_runner',
'build_runner_core',
'build_test',
]) {
final packageConfig = pubConfig.packageNamed(package);
mergedConfig.packageNamed(package).rootUri = packageConfig.rootUri;
if (package == 'build_runner') {
buildRunnerPath = packageConfig.rootUri;
}
}
final mergedConfigFile = File.fromUri(
tempDirectory.uri.resolve('package_config.json'),
);
mergedConfigFile.writeAsStringSync(json.encode(mergedConfig));
final buildResult = Process.runSync('dart', [
'--packages=${mergedConfigFile.path}',
'run',
'$buildRunnerPath/bin/build_runner.dart',
'build',
'-d',
], workingDirectory: Directory.current.path);
stdout.write(buildResult.stdout);
stderr.write(buildResult.stderr);
if (buildResult.exitCode == 0) {
tempDirectory.deleteSync(recursive: true);
}
exit(buildResult.exitCode);
}
extension type PackageConfig(Map<String, Object?> node) {
List<Package> get packages =>
(node['packages'] as List<Object?>)
.map((p) => Package(p as Map<String, Object?>))
.toList();
Package packageNamed(String name) =>
packages.singleWhere((package) => package.name == name);
}
extension type Package(Map<String, Object?> node) {
String get name => node['name'] as String;
String get rootUri => node['rootUri'] as String;
set rootUri(String rootUri) => node['rootUri'] = rootUri;
}