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

Unified Diff: pkg/polymer/lib/src/build/import_inliner.dart

Issue 180933002: combine script extractor and import inliner (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: fix multiple linked scripts Created 6 years, 10 months 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
« no previous file with comments | « pkg/polymer/lib/src/build/common.dart ('k') | pkg/polymer/lib/src/build/script_compactor.dart » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: pkg/polymer/lib/src/build/import_inliner.dart
diff --git a/pkg/polymer/lib/src/build/import_inliner.dart b/pkg/polymer/lib/src/build/import_inliner.dart
index 40d5ca58d197a51ebea3106ccf5fb457baad3470..8b019dec4190a34ddaf01d0121945d996a304032 100644
--- a/pkg/polymer/lib/src/build/import_inliner.dart
+++ b/pkg/polymer/lib/src/build/import_inliner.dart
@@ -8,14 +8,15 @@ library polymer.src.build.import_inliner;
import 'dart:async';
import 'dart:convert';
+import 'package:analyzer/src/generated/ast.dart';
import 'package:barback/barback.dart';
import 'package:path/path.dart' as path;
import 'package:html5lib/dom.dart' show
Document, DocumentFragment, Element, Node;
import 'package:html5lib/dom_parsing.dart' show TreeVisitor;
+import 'package:source_maps/refactor.dart' show TextEditTransaction;
import 'package:source_maps/span.dart';
-import 'code_extractor.dart'; // import just for documentation.
import 'common.dart';
class _HtmlInliner extends PolymerTransformer {
@@ -26,8 +27,9 @@ class _HtmlInliner extends PolymerTransformer {
final seen = new Set<AssetId>();
final scriptIds = <AssetId>[];
- static const TYPE_DART = 'application/dart';
- static const TYPE_JS = 'text/javascript';
+ /// The number of extracted inline Dart scripts. Used as a counter to give
+ /// unique-ish filenames.
+ int inlineScriptCounter = 0;
_HtmlInliner(this.options, Transform transform)
: transform = transform,
@@ -38,21 +40,26 @@ class _HtmlInliner extends PolymerTransformer {
seen.add(docId);
Document document;
+ bool changed;
- return readPrimaryAsHtml(transform).then((document) =>
- _visitImports(document, docId).then((importsFound) {
+ return readPrimaryAsHtml(transform).then((doc) {
+ document = doc;
+ // Add the main script's ID, or null if none is present.
+ // This will be used by ScriptCompactor.
+ changed = _extractScripts(document, docId);
+ return _visitImports(document);
+ }).then((importsFound) {
+ changed = changed || importsFound;
var output = transform.primaryInput;
- if (importsFound) {
- output = new Asset.fromString(docId, document.outerHtml);
- }
+ if (changed) output = new Asset.fromString(docId, document.outerHtml);
transform.addOutput(output);
// We produce a secondary asset with extra information for later phases.
transform.addOutput(new Asset.fromString(
docId.addExtension('.scriptUrls'),
JSON.encode(scriptIds, toEncodable: (id) => id.serialize())));
- }));
+ });
}
/// Visits imports in [document] and add the imported documents to documents.
@@ -61,7 +68,7 @@ class _HtmlInliner extends PolymerTransformer {
///
/// Returns `true` if and only if the document was changed and should be
/// written out.
- Future<bool> _visitImports(Document document, AssetId sourceId) {
+ Future<bool> _visitImports(Document document) {
bool changed = false;
_moveHeadToBody(document);
@@ -71,8 +78,9 @@ class _HtmlInliner extends PolymerTransformer {
var rel = tag.attributes['rel'];
if (rel != 'import' && rel != 'stylesheet') return null;
+ // Note: URL has already been normalized so use docId.
var href = tag.attributes['href'];
- var id = resolve(sourceId, href, transform.logger, tag.sourceSpan,
+ var id = resolve(docId, href, transform.logger, tag.sourceSpan,
allowAbsolute: rel == 'stylesheet');
if (rel == 'import') {
@@ -107,7 +115,7 @@ class _HtmlInliner extends PolymerTransformer {
var insertionPoint = doc.body.firstChild;
for (var node in doc.head.nodes.toList(growable: false)) {
if (node is! Element) continue;
- var tag = node.tagName;
+ var tag = node.localName;
var type = node.attributes['type'];
var rel = node.attributes['rel'];
if (tag == 'style' || tag == 'script' &&
@@ -119,63 +127,115 @@ class _HtmlInliner extends PolymerTransformer {
}
}
- // Loads an asset identified by [id], visits its imports and collects its
- // html imports. Then inlines it into the main document.
- Future _inlineImport(AssetId id, Element link) =>
- readAsHtml(id, transform).then((doc) => _visitImports(doc, id).then((_) {
-
- new _UrlNormalizer(transform, id).visit(doc);
- _extractScripts(doc);
-
- // TODO(jmesserly): figure out how this is working in vulcanizer.
- // Do they produce a <body> tag with a <head> and <body> inside?
- var imported = new DocumentFragment();
- imported.nodes..addAll(doc.head.nodes)..addAll(doc.body.nodes);
- link.replaceWith(imported);
- }));
+ /// Loads an asset identified by [id], visits its imports and collects its
+ /// html imports. Then inlines it into the main document.
+ Future _inlineImport(AssetId id, Element link) {
+ return readAsHtml(id, transform).then((doc) {
+ new _UrlNormalizer(transform, id).visit(doc);
+ return _visitImports(doc).then((_) {
+ _extractScripts(doc, id);
+ _removeScripts(doc);
+
+ // TODO(jmesserly): figure out how this is working in vulcanizer.
+ // Do they produce a <body> tag with a <head> and <body> inside?
+ var imported = new DocumentFragment();
+ imported.nodes..addAll(doc.head.nodes)..addAll(doc.body.nodes);
+ link.replaceWith(imported);
+ });
+ });
+ }
Future _inlineStylesheet(AssetId id, Element link) {
return transform.readInputAsString(id).then((css) {
- var url = spanUrlFor(id, transform);
- css = new _UrlNormalizer(transform, id).visitCss(css, url);
+ css = new _UrlNormalizer(transform, id).visitCss(css);
link.replaceWith(new Element.tag('style')..text = css);
});
}
- /// Split Dart script tags from all the other elements. Now that Dartium
- /// only allows a single script tag per page, we can't inline script
- /// tags. Instead, we collect the urls of each script tag so we import
- /// them directly from the Dart bootstrap code.
- void _extractScripts(Document document) {
- bool first = true;
- for (var script in document.querySelectorAll('script')) {
+ /// Remove scripts from HTML imports, and remember their [AssetId]s for later
+ /// use.
+ ///
+ /// Dartium only allows a single script tag per page, so we can't inline
+ /// the script tags. Instead we remove them entirely.
+ void _removeScripts(Document doc) {
+ for (var script in doc.querySelectorAll('script')) {
if (script.attributes['type'] == TYPE_DART) {
script.remove();
+ var src = script.attributes['src'];
+ scriptIds.add(resolve(docId, src, logger, script.sourceSpan));
- // only one Dart script per document is supported in Dartium.
- if (first) {
- first = false;
-
- var src = script.attributes['src'];
- if (src == null) {
- logger.warning('unexpected script without a src url. The '
- 'ImportInliner transformer should run after running the '
- 'InlineCodeExtractor', span: script.sourceSpan);
- continue;
- }
- scriptIds.add(resolve(docId, src, logger, script.sourceSpan));
-
- } else {
- // TODO(jmesserly): remove this when we are running linter.
- logger.warning('more than one Dart script per HTML '
- 'document is not supported. Script will be ignored.',
- span: script.sourceSpan);
- }
+ // only the first script needs to be added.
+ // others are already removed by _extractScripts
+ return;
+ }
+ }
+ }
+
+ /// Split inline scripts into their own files. We need to do this for dart2js
+ /// to be able to compile them.
+ ///
+ /// This also validates that there weren't any duplicate scripts.
+ bool _extractScripts(Document doc, AssetId sourceId) {
+ bool changed = false;
+ bool first = true;
+ for (var script in doc.querySelectorAll('script')) {
+ if (script.attributes['type'] != TYPE_DART) continue;
+
+ // only one Dart script per document is supported in Dartium.
+ if (!first) {
+ // Remove the script. It's invalid to have more than one in Dartium.
+ script.remove();
+ changed = true;
+
+ // TODO(jmesserly): remove this when we are running linter.
+ logger.warning('more than one Dart script per HTML '
+ 'document is not supported. Script will be ignored.',
+ span: script.sourceSpan);
+ continue;
}
+
+ first = false;
+
+ var src = script.attributes['src'];
+ if (src != null) continue;
+
+ final filename = path.url.basename(docId.path);
+ final count = inlineScriptCounter++;
+ var code = script.text;
+ // TODO(sigmund): ensure this path is unique (dartbug.com/12618).
+ script.attributes['src'] = src = '$filename.$count.dart';
+ script.text = '';
+ changed = true;
+
+ var newId = docId.addExtension('.$count.dart');
+ // TODO(jmesserly): consolidate this check with our other parsing of the
+ // Dart code, so we only parse it once.
+ if (!_hasLibraryDirective(code)) {
+ // Inject a library tag with an appropriate library name.
+
+ // Transform AssetId into a package name. For example:
+ // myPkgName|lib/foo/bar.html -> myPkgName.foo.bar_html
+ // myPkgName|web/foo/bar.html -> myPkgName.web.foo.bar_html
+ // This should roughly match the recommended library name conventions.
+ var libName = '${path.withoutExtension(sourceId.path)}_'
+ '${path.extension(sourceId.path).substring(1)}';
+ if (libName.startsWith('lib/')) libName = libName.substring(4);
+ libName = libName.replaceAll('/', '.').replaceAll('-', '_');
+ libName = '${sourceId.package}.$libName';
+
+ code = "library $libName;\n$code";
+ }
+ transform.addOutput(new Asset.fromString(newId, code));
}
+ return changed;
}
}
+/// Parse [code] and determine whether it has a library directive.
+bool _hasLibraryDirective(String code) =>
+ parseCompilationUnit(code).directives.any((d) => d is LibraryDirective);
+
+
/// Recursively inlines the contents of HTML imports. Produces as output a
/// single HTML file that inlines the polymer-element definitions, and a text
/// file that contains, in order, the URIs to each library that sourced in a
@@ -197,6 +257,8 @@ class ImportInliner extends Transformer {
new _HtmlInliner(options, transform).apply();
}
+const TYPE_DART = 'application/dart';
+const TYPE_JS = 'text/javascript';
/// Internally adjusts urls in the html that we are about to inline.
class _UrlNormalizer extends TreeVisitor {
@@ -208,13 +270,20 @@ class _UrlNormalizer extends TreeVisitor {
_UrlNormalizer(this.transform, this.sourceId);
visitElement(Element node) {
- for (var key in node.attributes.keys) {
- if (_urlAttributes.contains(key)) {
- var url = node.attributes[key];
- if (url != null && url != '' && !url.startsWith('{{')) {
- node.attributes[key] = _newUrl(url, node.sourceSpan);
+ node.attributes.forEach((name, value) {
+ if (_urlAttributes.contains(name)) {
+ if (value != '' && !value.trim().startsWith('{{')) {
+ node.attributes[name] = _newUrl(value, node.sourceSpan);
}
}
+ });
+ if (node.localName == 'style') {
+ node.text = visitCss(node.text);
+ } else if (node.localName == 'script' &&
+ node.attributes['type'] == TYPE_DART) {
+ // TODO(jmesserly): we might need to visit JS too to handle ES Harmony
+ // modules.
+ node.text = visitInlineDart(node.text);
}
super.visitElement(node);
}
@@ -227,7 +296,8 @@ class _UrlNormalizer extends TreeVisitor {
// https://github.com/Polymer/vulcanize/blob/c14f63696797cda18dc3d372b78aa3378acc691f/lib/vulcan.js#L149
// TODO(jmesserly): use csslib here instead? Parsing with RegEx is sadness.
// Maybe it's reliable enough for finding URLs in CSS? I'm not sure.
- String visitCss(String cssText, String url) {
+ String visitCss(String cssText) {
+ var url = spanUrlFor(sourceId, transform);
var src = new SourceFile.text(url, cssText);
return cssText.replaceAllMapped(_URL, (match) {
// Extract the URL, without any surrounding quotes.
@@ -238,7 +308,35 @@ class _UrlNormalizer extends TreeVisitor {
});
}
- _newUrl(String href, Span span) {
+ String visitInlineDart(String code) {
+ var unit = parseCompilationUnit(code);
+ var file = new SourceFile.text(spanUrlFor(sourceId, transform), code);
+ var output = new TextEditTransaction(code, file);
+
+ for (Directive directive in unit.directives) {
+ if (directive is UriBasedDirective) {
+ var uri = directive.uri.stringValue;
+ var span = _getSpan(file, directive.uri);
+
+ var id = resolve(sourceId, uri, transform.logger, span);
+ if (id == null) continue;
+
+ var primaryId = transform.primaryInput.id;
+ var newUri = assetUrlFor(id, primaryId, transform.logger);
+ if (newUri != uri) {
+ output.edit(span.start.offset, span.end.offset, "'$newUri'");
+ }
+ }
+ }
+
+ if (!output.hasEdits) return code;
+
+ // TODO(sigmund): emit source maps when barback supports it (see
+ // dartbug.com/12340)
+ return (output.commit()..build(file.url)).text;
+ }
+
+ String _newUrl(String href, Span span) {
var uri = Uri.parse(href);
if (uri.isAbsolute) return href;
if (!uri.scheme.isEmpty) return href;
@@ -289,3 +387,5 @@ const _urlAttributes = const [
'src', // in audio, embed, iframe, img, input, script, source, track,
// video
];
+
+_getSpan(SourceFile file, ASTNode node) => file.span(node.offset, node.end);
« no previous file with comments | « pkg/polymer/lib/src/build/common.dart ('k') | pkg/polymer/lib/src/build/script_compactor.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698