| // Copyright 2014 The Flutter Authors. 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:process/process.dart'; |
| import 'package:unified_analytics/unified_analytics.dart'; |
| |
| import '../artifacts.dart'; |
| import '../base/common.dart'; |
| import '../base/file_system.dart'; |
| import '../base/logger.dart'; |
| import '../base/project_migrator.dart'; |
| import '../base/terminal.dart'; |
| import '../build_info.dart'; |
| import '../build_system/build_system.dart'; |
| import '../cache.dart'; |
| import '../flutter_plugins.dart'; |
| import '../globals.dart' as globals; |
| import '../platform_plugins.dart'; |
| import '../plugins.dart'; |
| import '../project.dart'; |
| import '../version.dart'; |
| import 'compiler_config.dart'; |
| import 'file_generators/flutter_service_worker_js.dart'; |
| import 'migrations/scrub_generated_plugin_registrant.dart'; |
| |
| export 'compiler_config.dart'; |
| |
| /// Whether the application has web plugins. |
| const kHasWebPlugins = 'HasWebPlugins'; |
| |
| /// Base href to set in index.html in flutter build command |
| const kBaseHref = 'baseHref'; |
| |
| /// Static assets url to set in index.html in flutter build command |
| const kStaticAssetsUrl = 'staticAssetsUrl'; |
| |
| /// The caching strategy to use for service worker generation. |
| const kServiceWorkerStrategy = 'ServiceWorkerStrategy'; |
| |
| class WebBuilder { |
| WebBuilder({ |
| required Logger logger, |
| required ProcessManager processManager, |
| required BuildSystem buildSystem, |
| required Analytics analytics, |
| required FlutterVersion flutterVersion, |
| required FileSystem fileSystem, |
| }) : _logger = logger, |
| _processManager = processManager, |
| _buildSystem = buildSystem, |
| _analytics = analytics, |
| _flutterVersion = flutterVersion, |
| _fileSystem = fileSystem; |
| |
| final Logger _logger; |
| final ProcessManager _processManager; |
| final BuildSystem _buildSystem; |
| final Analytics _analytics; |
| final FlutterVersion _flutterVersion; |
| final FileSystem _fileSystem; |
| |
| Future<void> buildWeb( |
| FlutterProject flutterProject, |
| String target, |
| BuildInfo buildInfo, |
| ServiceWorkerStrategy serviceWorkerStrategy, { |
| required List<WebCompilerConfig> compilerConfigs, |
| String? baseHref, |
| String? staticAssetsUrl, |
| String? outputDirectoryPath, |
| }) async { |
| final bool hasWebPlugins = (await findPlugins( |
| flutterProject, |
| )).any((Plugin p) => p.platforms.containsKey(WebPlugin.kConfigKey)); |
| final Directory outputDirectory = outputDirectoryPath == null |
| ? _fileSystem.directory( |
| _fileSystem.path.join(flutterProject.directory.path, getWebBuildDirectory()), |
| ) |
| : _fileSystem.directory(outputDirectoryPath); |
| outputDirectory.createSync(recursive: true); |
| |
| // The migrators to apply to a Web project. |
| final migrators = <ProjectMigrator>[ |
| ScrubGeneratedPluginRegistrant(flutterProject.web, _logger), |
| ]; |
| |
| final migration = ProjectMigration(migrators); |
| await migration.run(); |
| |
| final Status status = _logger.startProgress('Compiling $target for the Web...'); |
| final sw = Stopwatch()..start(); |
| try { |
| final BuildResult result = await _buildSystem.build( |
| globals.buildTargets.webServiceWorker(_fileSystem, compilerConfigs, _analytics), |
| Environment( |
| projectDir: flutterProject.directory, |
| outputDir: outputDirectory, |
| buildDir: flutterProject.directory |
| .childDirectory('.dart_tool') |
| .childDirectory('flutter_build'), |
| defines: <String, String>{ |
| kTargetFile: target, |
| kHasWebPlugins: hasWebPlugins.toString(), |
| kBaseHref: ?baseHref, |
| kStaticAssetsUrl: ?staticAssetsUrl, |
| kServiceWorkerStrategy: serviceWorkerStrategy.cliName, |
| ...buildInfo.toBuildSystemEnvironment(), |
| }, |
| packageConfigPath: buildInfo.packageConfigPath, |
| artifacts: globals.artifacts!, |
| fileSystem: _fileSystem, |
| logger: _logger, |
| processManager: _processManager, |
| platform: globals.platform, |
| analytics: _analytics, |
| cacheDir: globals.cache.getRoot(), |
| engineVersion: globals.artifacts!.usesLocalArtifacts |
| ? null |
| : _flutterVersion.engineRevision, |
| flutterRootDir: _fileSystem.directory(Cache.flutterRoot), |
| // Web uses a different Dart plugin registry. |
| // https://github.com/flutter/flutter/issues/80406 |
| generateDartPluginRegistry: false, |
| ), |
| ); |
| if (!result.success) { |
| for (final ExceptionMeasurement measurement in result.exceptions.values) { |
| _logger.printError( |
| 'Target ${measurement.target} failed: ${measurement.exception}', |
| stackTrace: measurement.fatal ? measurement.stackTrace : null, |
| ); |
| } |
| throwToolExit('Failed to compile application for the Web.'); |
| } |
| } on Exception catch (err) { |
| throwToolExit(err.toString()); |
| } finally { |
| status.stop(); |
| } |
| |
| // We don't print a size because the output directory can contain |
| // optional files not needed by the user. |
| globals.printStatus( |
| '${globals.terminal.successMark} ' |
| 'Built ${globals.fs.path.relative(outputDirectory.path)}', |
| color: TerminalColor.green, |
| ); |
| |
| final String buildSettingsString = _buildEventAnalyticsSettings(configs: compilerConfigs); |
| |
| _analytics.send( |
| Event.flutterBuildInfo(label: 'web-compile', buildType: 'web', settings: buildSettingsString), |
| ); |
| |
| final Duration elapsedDuration = sw.elapsed; |
| final variableName = compilerConfigs.length > 1 ? 'dual-compile' : 'dart2js'; |
| _analytics.send( |
| Event.timing( |
| workflow: 'build', |
| variableName: variableName, |
| elapsedMilliseconds: elapsedDuration.inMilliseconds, |
| ), |
| ); |
| } |
| } |
| |
| /// Web rendering backend mode. |
| enum WebRendererMode { |
| /// Always uses canvaskit. |
| canvaskit, |
| |
| /// Always use skwasm. |
| skwasm; |
| |
| factory WebRendererMode.fromDartDefines(Iterable<String> defines, {required bool useWasm}) { |
| if (defines.contains('FLUTTER_WEB_USE_SKIA=false') && |
| defines.contains('FLUTTER_WEB_USE_SKWASM=true')) { |
| return skwasm; |
| } else if (defines.contains('FLUTTER_WEB_USE_SKIA=true') && |
| defines.contains('FLUTTER_WEB_USE_SKWASM=false')) { |
| return canvaskit; |
| } |
| return getDefault(useWasm: useWasm); |
| } |
| |
| static WebRendererMode getDefault({required bool useWasm}) { |
| return useWasm ? defaultForWasm : defaultForJs; |
| } |
| |
| static const WebRendererMode defaultForJs = WebRendererMode.canvaskit; |
| static const WebRendererMode defaultForWasm = WebRendererMode.skwasm; |
| |
| /// Returns [dartDefines] in a way usable from the CLI. |
| /// |
| /// This is used to start integration tests. |
| Iterable<String> get toCliDartDefines => |
| dartDefines.map((String define) => '--dart-define=$define'); |
| |
| Iterable<String> get dartDefines => switch (this) { |
| canvaskit => const <String>{'FLUTTER_WEB_USE_SKIA=true', 'FLUTTER_WEB_USE_SKWASM=false'}, |
| skwasm => const <String>{'FLUTTER_WEB_USE_SKIA=false', 'FLUTTER_WEB_USE_SKWASM=true'}, |
| }; |
| |
| /// Sets the dart defines for the currently selected WebRendererMode |
| List<String> updateDartDefines(List<String> inputDefines) { |
| final Set<String> dartDefinesSet = inputDefines.toSet(); |
| |
| dartDefinesSet |
| ..removeWhere((String d) { |
| return d.startsWith('FLUTTER_WEB_USE_SKIA=') || d.startsWith('FLUTTER_WEB_USE_SKWASM='); |
| }) |
| ..addAll(dartDefines); |
| |
| return dartDefinesSet.toList(); |
| } |
| } |
| |
| /// The correct precompiled artifact to use for each build and render mode for DDC with AMD modules. |
| // TODO(markzipan): delete this when DDC's AMD module system is deprecated, https://github.com/flutter/flutter/issues/142060. |
| const kAmdDartSdkJsArtifactMap = <WebRendererMode, HostArtifact>{ |
| WebRendererMode.canvaskit: HostArtifact.webPrecompiledAmdCanvaskitSdk, |
| }; |
| |
| /// The correct source map artifact to use for each build and render mode for DDC with AMD modules. |
| // TODO(markzipan): delete this when DDC's AMD module system is deprecated, https://github.com/flutter/flutter/issues/142060. |
| const kAmdDartSdkJsMapArtifactMap = <WebRendererMode, HostArtifact>{ |
| WebRendererMode.canvaskit: HostArtifact.webPrecompiledAmdCanvaskitSdkSourcemaps, |
| }; |
| |
| /// The correct precompiled artifact to use for each build and render mode for |
| /// DDC with DDC library bundle module format. |
| const kDdcLibraryBundleDartSdkJsArtifactMap = <WebRendererMode, HostArtifact>{ |
| WebRendererMode.canvaskit: HostArtifact.webPrecompiledDdcLibraryBundleCanvaskitSdk, |
| }; |
| |
| /// The correct source map artifact to use for each build and render mode for |
| /// DDC with DDC library bundle module format. |
| const kDdcLibraryBundleDartSdkJsMapArtifactMap = <WebRendererMode, HostArtifact>{ |
| WebRendererMode.canvaskit: HostArtifact.webPrecompiledDdcLibraryBundleCanvaskitSdkSourcemaps, |
| }; |
| |
| String _buildEventAnalyticsSettings({required List<WebCompilerConfig> configs}) { |
| final values = <String, Object>{}; |
| final renderers = <String>[]; |
| final targets = <String>[]; |
| for (final config in configs) { |
| values.addAll(config.buildEventAnalyticsValues); |
| renderers.add(config.renderer.name); |
| targets.add(config.compileTarget.name); |
| } |
| values['web-renderer'] = renderers.join(','); |
| values['web-target'] = targets.join(','); |
| |
| final List<String> sortedList = |
| values.entries.map((MapEntry<String, Object> e) => '${e.key}: ${e.value};').toList()..sort(); |
| |
| return sortedList.join(' '); |
| } |