Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1253)

Unified Diff: observatory_pub_packages/polymer/src/build/runner.dart

Issue 816693004: Add observatory_pub_packages snapshot to third_party (Closed) Base URL: http://dart.googlecode.com/svn/third_party/
Patch Set: Created 6 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: observatory_pub_packages/polymer/src/build/runner.dart
===================================================================
--- observatory_pub_packages/polymer/src/build/runner.dart (revision 0)
+++ observatory_pub_packages/polymer/src/build/runner.dart (working copy)
@@ -0,0 +1,378 @@
+// Copyright (c) 2013, 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.
+
+/// Definitions used to run the polymer linter and deploy tools without using
+/// pub serve or pub deploy.
+library polymer.src.build.runner;
+
+import 'dart:async';
+import 'dart:convert';
+import 'dart:io';
+
+import 'package:barback/barback.dart';
+import 'package:path/path.dart' as path;
+import 'package:stack_trace/stack_trace.dart';
+import 'package:yaml/yaml.dart';
+
+
+/// Collects different parameters needed to configure and run barback.
+class BarbackOptions {
+ /// Phases of transformers to run for the current package.
+ /// Use packagePhases to specify phases for other packages.
+ final List<List<Transformer>> phases;
+
+ /// Package to treat as the current package in barback.
+ final String currentPackage;
+
+ /// Directory root for the current package.
+ final String packageHome;
+
+ /// Mapping between package names and the path in the file system where
+ /// to find the sources of such package.
+ final Map<String, String> packageDirs;
+
+ /// Whether to run transformers on the test folder.
+ final bool transformTests;
+
+ /// Directory where to generate code, if any.
+ final String outDir;
+
+ /// Disregard files that match these filters when copying in non
+ /// transformed files
+ List<String> fileFilter;
+
+ /// Whether to print error messages using a json-format that tools, such as
+ /// the Dart Editor, can process.
+ final bool machineFormat;
+
+ /// Whether to follow symlinks when listing directories. By default this is
+ /// false because directories have symlinks for the packages directory created
+ /// by pub, but it can be turned on for custom uses of this library.
+ final bool followLinks;
+
+ /// Phases of transformers to apply to packages other than the current
+ /// package, keyed by the package name.
+ final Map<String, List<List<Transformer>>> packagePhases;
+
+ BarbackOptions(this.phases, this.outDir, {currentPackage, String packageHome,
+ packageDirs, this.transformTests: false, this.machineFormat: false,
+ this.followLinks: false,
+ this.packagePhases: const {},
+ this.fileFilter: const []})
+ : currentPackage = (currentPackage != null
+ ? currentPackage : readCurrentPackageFromPubspec()),
+ packageHome = packageHome,
+ packageDirs = (packageDirs != null
+ ? packageDirs : readPackageDirsFromPub(packageHome, currentPackage));
+
+}
+
+/// Creates a barback system as specified by [options] and runs it. Returns a
+/// future that contains the list of assets generated after barback runs to
+/// completion.
+Future<AssetSet> runBarback(BarbackOptions options) {
+ var barback = new Barback(new _PackageProvider(options.packageDirs));
+ _initBarback(barback, options);
+ _attachListeners(barback, options);
+ if (options.outDir == null) return barback.getAllAssets();
+ return _emitAllFiles(barback, options);
+}
+
+/// Extract the current package from the pubspec.yaml file.
+String readCurrentPackageFromPubspec([String dir]) {
+ var pubspec = new File(
+ dir == null ? 'pubspec.yaml' : path.join(dir, 'pubspec.yaml'));
+ if (!pubspec.existsSync()) {
+ print('error: pubspec.yaml file not found, please run this script from '
+ 'your package root directory.');
+ return null;
+ }
+ return loadYaml(pubspec.readAsStringSync())['name'];
+}
+
+/// Extract a mapping between package names and the path in the file system
+/// which has the source of the package. This map will contain an entry for the
+/// current package and everything it depends on (extracted via `pub
+/// list-package-dirs`).
+Map<String, String> readPackageDirsFromPub(
+ [String packageHome, String currentPackage]) {
+ var cachedDir = Directory.current;
+ if (packageHome != null) {
+ Directory.current = new Directory(packageHome);
+ } else {
+ packageHome = cachedDir.path;
+ }
+
+ var dartExec = Platform.executable;
+ // If dartExec == dart, then dart and pub are in standard PATH.
+ var sdkDir = dartExec == 'dart' ? '' : path.dirname(dartExec);
+ var pub = path.join(sdkDir, Platform.isWindows ? 'pub.bat' : 'pub');
+ var result = Process.runSync(pub, ['list-package-dirs']);
+ if (result.exitCode != 0) {
+ print("unexpected error invoking 'pub':");
+ print(result.stdout);
+ print(result.stderr);
+ exit(result.exitCode);
+ }
+ var map = JSON.decode(result.stdout)["packages"];
+ map.forEach((k, v) { map[k] = path.absolute(packageHome, path.dirname(v)); });
+
+ if (currentPackage == null) {
+ currentPackage = readCurrentPackageFromPubspec(packageHome);
+ }
+ map[currentPackage] = packageHome;
+
+ Directory.current = cachedDir;
+ return map;
+}
+
+bool shouldSkip(List<String> filters, String path) {
+ return filters.any((filter) => path.contains(filter));
+}
+
+/// Return the relative path of each file under [subDir] in [package].
+Iterable<String> _listPackageDir(String package, String subDir,
+ BarbackOptions options) {
+ var packageDir = options.packageDirs[package];
+ if (packageDir == null) return const [];
+ var dir = new Directory(path.join(packageDir, subDir));
+ if (!dir.existsSync()) return const [];
+ return dir.listSync(recursive: true, followLinks: options.followLinks)
+ .where((f) => f is File)
+ .where((f) => !shouldSkip(options.fileFilter, f.path))
+ .map((f) => path.relative(f.path, from: packageDir));
+}
+
+/// A simple provider that reads files directly from the pub cache.
+class _PackageProvider implements PackageProvider {
+ Map<String, String> packageDirs;
+ Iterable<String> get packages => packageDirs.keys;
+
+ _PackageProvider(this.packageDirs);
+
+ Future<Asset> getAsset(AssetId id) => new Future.value(
+ new Asset.fromPath(id, path.join(packageDirs[id.package],
+ _toSystemPath(id.path))));
+}
+
+/// Convert asset paths to system paths (Assets always use the posix style).
+String _toSystemPath(String assetPath) {
+ if (path.Style.platform != path.Style.windows) return assetPath;
+ return path.joinAll(path.posix.split(assetPath));
+}
+
+/// Tell barback which transformers to use and which assets to process.
+void _initBarback(Barback barback, BarbackOptions options) {
+ var assets = [];
+ void addAssets(String package, String subDir) {
+ for (var filepath in _listPackageDir(package, subDir, options)) {
+ assets.add(new AssetId(package, filepath));
+ }
+ }
+
+ for (var package in options.packageDirs.keys) {
+ // Notify barback to process anything under 'lib' and 'asset'.
+ addAssets(package, 'lib');
+ addAssets(package, 'asset');
+
+ if (options.packagePhases.containsKey(package)) {
+ barback.updateTransformers(package, options.packagePhases[package]);
+ }
+ }
+ barback.updateTransformers(options.currentPackage, options.phases);
+
+ // In case of the current package, include also 'web'.
+ addAssets(options.currentPackage, 'web');
+ if (options.transformTests) addAssets(options.currentPackage, 'test');
+
+ // Add the sources after the transformers so all transformers are present
+ // when barback starts processing the assets.
+ barback.updateSources(assets);
+}
+
+/// Attach error listeners on [barback] so we can report errors.
+void _attachListeners(Barback barback, BarbackOptions options) {
+ // Listen for errors and results
+ barback.errors.listen((e) {
+ var trace = null;
+ if (e is Error) trace = e.stackTrace;
+ if (trace != null) {
+ print(Trace.format(trace));
+ }
+ print('error running barback: $e');
+ exit(1);
+ });
+
+ barback.results.listen((result) {
+ if (!result.succeeded) {
+ print("build failed with errors: ${result.errors}");
+ exit(1);
+ }
+ });
+
+ barback.log.listen((entry) {
+ if (options.machineFormat) {
+ print(_jsonFormatter(entry));
+ } else {
+ print(_consoleFormatter(entry));
+ }
+ });
+}
+
+/// Emits all outputs of [barback] and copies files that we didn't process (like
+/// dependent package's libraries).
+Future _emitAllFiles(Barback barback, BarbackOptions options) {
+ return barback.getAllAssets().then((assets) {
+ // Delete existing output folder before we generate anything
+ var dir = new Directory(options.outDir);
+ if (dir.existsSync()) dir.deleteSync(recursive: true);
+ return _emitPackagesDir(options)
+ .then((_) => _emitTransformedFiles(assets, options))
+ .then((_) => _addPackagesSymlinks(assets, options))
+ .then((_) => assets);
+ });
+}
+
+Future _emitTransformedFiles(AssetSet assets, BarbackOptions options) {
+ // Copy all the assets we transformed
+ var futures = [];
+ var currentPackage = options.currentPackage;
+ var transformTests = options.transformTests;
+ var outPackages = path.join(options.outDir, 'packages');
+
+ return Future.forEach(assets, (asset) {
+ var id = asset.id;
+ var dir = _firstDir(id.path);
+ if (dir == null) return null;
+
+ var filepath;
+ if (dir == 'lib') {
+ // Put lib files directly under the packages folder (e.g. 'lib/foo.dart'
+ // will be emitted at out/packages/package_name/foo.dart).
+ filepath = path.join(outPackages, id.package,
+ _toSystemPath(id.path.substring(4)));
+ } else if (id.package == currentPackage &&
+ (dir == 'web' || (transformTests && dir == 'test'))) {
+ filepath = path.join(options.outDir, _toSystemPath(id.path));
+ } else {
+ // TODO(sigmund): do something about other assets?
+ return null;
+ }
+
+ return _writeAsset(filepath, asset);
+ });
+}
+
+/// Adds a package symlink from each directory under `out/web/foo/` to
+/// `out/packages`.
+void _addPackagesSymlinks(AssetSet assets, BarbackOptions options) {
+ var outPackages = path.join(options.outDir, 'packages');
+ var currentPackage = options.currentPackage;
+ for (var asset in assets) {
+ var id = asset.id;
+ if (id.package != currentPackage) continue;
+ var firstDir = _firstDir(id.path);
+ if (firstDir == null) continue;
+
+ if (firstDir == 'web' || (options.transformTests && firstDir == 'test')) {
+ var dir = path.join(options.outDir, path.dirname(_toSystemPath(id.path)));
+ var linkPath = path.join(dir, 'packages');
+ var link = new Link(linkPath);
+ if (!link.existsSync()) {
+ var targetPath = Platform.operatingSystem == 'windows'
+ ? path.normalize(path.absolute(outPackages))
+ : path.normalize(path.relative(outPackages, from: dir));
+ link.createSync(targetPath);
+ }
+ }
+ }
+}
+
+/// Emits a 'packages' directory directly under `out/packages` with the contents
+/// of every file that was not transformed by barback.
+Future _emitPackagesDir(BarbackOptions options) {
+ var outPackages = path.join(options.outDir, 'packages');
+ _ensureDir(outPackages);
+
+ // Copy all the files we didn't process
+ var dirs = options.packageDirs;
+ return Future.forEach(dirs.keys, (package) {
+ return Future.forEach(_listPackageDir(package, 'lib', options), (relpath) {
+ var inpath = path.join(dirs[package], relpath);
+ var outpath = path.join(outPackages, package, relpath.substring(4));
+ return _copyFile(inpath, outpath);
+ });
+ });
+}
+
+/// Ensure [dirpath] exists.
+void _ensureDir(String dirpath) {
+ new Directory(dirpath).createSync(recursive: true);
+}
+
+/// Returns the first directory name on a url-style path, or null if there are
+/// no slashes.
+String _firstDir(String url) {
+ var firstSlash = url.indexOf('/');
+ if (firstSlash == -1) return null;
+ return url.substring(0, firstSlash);
+}
+
+/// Copy a file from [inpath] to [outpath].
+Future _copyFile(String inpath, String outpath) {
+ _ensureDir(path.dirname(outpath));
+ return new File(inpath).openRead().pipe(new File(outpath).openWrite());
+}
+
+/// Write contents of an [asset] into a file at [filepath].
+Future _writeAsset(String filepath, Asset asset) {
+ _ensureDir(path.dirname(filepath));
+ return asset.read().pipe(new File(filepath).openWrite());
+}
+
+String _kindFromEntry(LogEntry entry) {
+ var level = entry.level;
+ return level == LogLevel.ERROR ? 'error'
+ : (level == LogLevel.WARNING ? 'warning' : 'info');
+}
+
+/// Formatter that generates messages using a format that can be parsed
+/// by tools, such as the Dart Editor, for reporting error messages.
+String _jsonFormatter(LogEntry entry) {
+ var kind = _kindFromEntry(entry);
+ var span = entry.span;
+ return JSON.encode((span == null)
+ ? [{'method': kind, 'params': {'message': entry.message}}]
+ : [{'method': kind,
+ 'params': {
+ 'file': span.sourceUrl.toString(),
+ 'message': entry.message,
+ 'line': span.start.line + 1,
+ 'charStart': span.start.offset,
+ 'charEnd': span.end.offset,
+ }}]);
+}
+
+/// Formatter that generates messages that are easy to read on the console (used
+/// by default).
+String _consoleFormatter(LogEntry entry) {
+ var kind = _kindFromEntry(entry);
+ var useColors = stdioType(stdout) == StdioType.TERMINAL;
+ var levelColor = (kind == 'error') ? _RED_COLOR : _MAGENTA_COLOR;
+ var output = new StringBuffer();
+ if (useColors) output.write(levelColor);
+ output..write(kind)..write(' ');
+ if (useColors) output.write(_NO_COLOR);
+ if (entry.span == null) {
+ output.write(entry.message);
+ } else {
+ output.write(entry.span.message(entry.message,
+ color: useColors ? levelColor : null));
+ }
+ return output.toString();
+}
+
+const String _RED_COLOR = '\u001b[31m';
+const String _MAGENTA_COLOR = '\u001b[35m';
+const String _NO_COLOR = '\u001b[0m';

Powered by Google App Engine
This is Rietveld 408576698