Unverified Commit bf6c4c4a authored by Jonah Williams's avatar Jonah Williams Committed by GitHub

[flutter_tools]web]Add support for --csp mode to build web (#48319)

parent 965ba387
...@@ -150,7 +150,14 @@ class WebFs { ...@@ -150,7 +150,14 @@ class WebFs {
/// Recompile the web application and return whether this was successful. /// Recompile the web application and return whether this was successful.
Future<bool> recompile() async { Future<bool> recompile() async {
if (!_useBuildRunner) { if (!_useBuildRunner) {
await buildWeb(_flutterProject, _target, _buildInfo, _initializePlatform, _dartDefines); await buildWeb(
_flutterProject,
_target,
_buildInfo,
_initializePlatform,
_dartDefines,
false,
);
return true; return true;
} }
_client.startBuild(); _client.startBuild();
...@@ -309,7 +316,14 @@ class WebFs { ...@@ -309,7 +316,14 @@ class WebFs {
handler = pipeline.addHandler(proxyHandler('http://localhost:$daemonAssetPort/web/')); handler = pipeline.addHandler(proxyHandler('http://localhost:$daemonAssetPort/web/'));
} }
} else { } else {
await buildWeb(flutterProject, target, buildInfo, initializePlatform, dartDefines); await buildWeb(
flutterProject,
target,
buildInfo,
initializePlatform,
dartDefines,
false,
);
firstBuildCompleter.complete(true); firstBuildCompleter.complete(true);
} }
......
...@@ -26,6 +26,9 @@ const String kHasWebPlugins = 'HasWebPlugins'; ...@@ -26,6 +26,9 @@ const String kHasWebPlugins = 'HasWebPlugins';
/// Valid values are O1 (lowest, profile default) to O4 (highest, release default). /// Valid values are O1 (lowest, profile default) to O4 (highest, release default).
const String kDart2jsOptimization = 'Dart2jsOptimization'; const String kDart2jsOptimization = 'Dart2jsOptimization';
/// Whether to disable dynamic generation code to satisfy csp policies.
const String kCspMode = 'cspMode';
/// Generates an entry point for a web target. /// Generates an entry point for a web target.
class WebEntrypointTarget extends Target { class WebEntrypointTarget extends Target {
const WebEntrypointTarget(); const WebEntrypointTarget();
...@@ -146,6 +149,7 @@ class Dart2JSTarget extends Target { ...@@ -146,6 +149,7 @@ class Dart2JSTarget extends Target {
@override @override
Future<void> build(Environment environment) async { Future<void> build(Environment environment) async {
final String dart2jsOptimization = environment.defines[kDart2jsOptimization]; final String dart2jsOptimization = environment.defines[kDart2jsOptimization];
final bool csp = environment.defines[kCspMode] == 'true';
final BuildMode buildMode = getBuildModeForName(environment.defines[kBuildMode]); final BuildMode buildMode = getBuildModeForName(environment.defines[kBuildMode]);
final String specPath = globals.fs.path.join(globals.artifacts.getArtifactPath(Artifact.flutterWebSdk), 'libraries.json'); final String specPath = globals.fs.path.join(globals.artifacts.getArtifactPath(Artifact.flutterWebSdk), 'libraries.json');
final String packageFile = FlutterProject.fromDirectory(environment.projectDir).hasBuilders final String packageFile = FlutterProject.fromDirectory(environment.projectDir).hasBuilders
...@@ -170,6 +174,8 @@ class Dart2JSTarget extends Target { ...@@ -170,6 +174,8 @@ class Dart2JSTarget extends Target {
'-Ddart.vm.profile=true' '-Ddart.vm.profile=true'
else else
'-Ddart.vm.product=true', '-Ddart.vm.product=true',
if (csp)
'--csp',
for (final String dartDefine in parseDartDefines(environment)) for (final String dartDefine in parseDartDefines(environment))
'-D$dartDefine', '-D$dartDefine',
environment.buildDir.childFile('main.dart').path, environment.buildDir.childFile('main.dart').path,
......
...@@ -25,6 +25,12 @@ class BuildWebCommand extends BuildSubCommand { ...@@ -25,6 +25,12 @@ class BuildWebCommand extends BuildSubCommand {
hide: true, hide: true,
help: 'Whether to automatically invoke webOnlyInitializePlatform.', help: 'Whether to automatically invoke webOnlyInitializePlatform.',
); );
argParser.addFlag('csp',
defaultsTo: false,
negatable: false,
help: 'Disable dynamic generation of code in the generated output.'
'This is necessary to satisfy CSP restrictions (see http://www.w3.org/TR/CSP/).'
);
} }
@override @override
...@@ -59,6 +65,7 @@ class BuildWebCommand extends BuildSubCommand { ...@@ -59,6 +65,7 @@ class BuildWebCommand extends BuildSubCommand {
buildInfo, buildInfo,
boolArg('web-initialize-platform'), boolArg('web-initialize-platform'),
dartDefines, dartDefines,
boolArg('csp')
); );
return null; return null;
} }
......
...@@ -28,6 +28,7 @@ Future<void> buildWeb( ...@@ -28,6 +28,7 @@ Future<void> buildWeb(
BuildInfo buildInfo, BuildInfo buildInfo,
bool initializePlatform, bool initializePlatform,
List<String> dartDefines, List<String> dartDefines,
bool csp,
) async { ) async {
if (!flutterProject.web.existsSync()) { if (!flutterProject.web.existsSync()) {
throwToolExit('Missing index.html.'); throwToolExit('Missing index.html.');
...@@ -50,6 +51,7 @@ Future<void> buildWeb( ...@@ -50,6 +51,7 @@ Future<void> buildWeb(
kInitializePlatform: initializePlatform.toString(), kInitializePlatform: initializePlatform.toString(),
kHasWebPlugins: hasWebPlugins.toString(), kHasWebPlugins: hasWebPlugins.toString(),
kDartDefines: jsonEncode(dartDefines), kDartDefines: jsonEncode(dartDefines),
kCspMode: csp.toString(),
}, },
)); ));
if (!result.success) { if (!result.success) {
......
...@@ -59,6 +59,7 @@ void main() { ...@@ -59,6 +59,7 @@ void main() {
BuildInfo.debug, BuildInfo.debug,
false, false,
const <String>[], const <String>[],
false,
), throwsA(isInstanceOf<ToolExit>())); ), throwsA(isInstanceOf<ToolExit>()));
})); }));
......
...@@ -165,6 +165,33 @@ void main() { ...@@ -165,6 +165,33 @@ void main() {
expect(generated, contains('entrypoint.main();')); expect(generated, contains('entrypoint.main();'));
})); }));
test('Dart2JSTarget calls dart2js with expected args with csp', () => testbed.run(() async {
environment.defines[kBuildMode] = 'profile';
environment.defines[kCspMode] = 'true';
when(globals.processManager.run(any)).thenAnswer((Invocation invocation) async {
return FakeProcessResult(exitCode: 0);
});
await const Dart2JSTarget().build(environment);
final List<String> expected = <String>[
globals.fs.path.join('bin', 'cache', 'dart-sdk', 'bin', 'dart'),
globals.fs.path.join('bin', 'cache', 'dart-sdk', 'bin', 'snapshots', 'dart2js.dart.snapshot'),
'--libraries-spec=' + globals.fs.path.join('bin', 'cache', 'flutter_web_sdk', 'libraries.json'),
'-O4', // highest optimizations
'--no-minify', // but uses unminified names for debugging
'-o',
environment.buildDir.childFile('main.dart.js').absolute.path,
'--packages=${globals.fs.path.join('foo', '.packages')}',
'-Ddart.vm.profile=true',
'--csp',
environment.buildDir.childFile('main.dart').absolute.path,
];
verify(globals.processManager.run(expected)).called(1);
}, overrides: <Type, Generator>{
ProcessManager: () => MockProcessManager(),
}));
test('Dart2JSTarget calls dart2js with expected args in profile mode', () => testbed.run(() async { test('Dart2JSTarget calls dart2js with expected args in profile mode', () => testbed.run(() async {
environment.defines[kBuildMode] = 'profile'; environment.defines[kBuildMode] = 'profile';
when(globals.processManager.run(any)).thenAnswer((Invocation invocation) async { when(globals.processManager.run(any)).thenAnswer((Invocation invocation) async {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment