build(broccoli): log the build tree after each build into tmp/build.*.log

This log can then be used to visualize the build tree and map paths in the tmp/ dir
to individual nodes in the build tree.
This commit is contained in:
Igor Minar 2015-06-08 18:56:24 -07:00
parent b5431e4cc0
commit 44b31f326b
1 changed files with 56 additions and 7 deletions

View File

@ -1,5 +1,5 @@
var broccoli = require('broccoli'); var broccoli = require('broccoli');
var fse = require('fs-extra'); var fs = require('fs');
var makeBrowserTree = require('./trees/browser_tree'); var makeBrowserTree = require('./trees/browser_tree');
var makeNodeTree = require('./trees/node_tree'); var makeNodeTree = require('./trees/node_tree');
var makeDartTree = require('./trees/dart_tree'); var makeDartTree = require('./trees/dart_tree');
@ -16,31 +16,32 @@ export class AngularBuilder {
private browserProdBuilder: BroccoliBuilder; private browserProdBuilder: BroccoliBuilder;
private dartBuilder: BroccoliBuilder; private dartBuilder: BroccoliBuilder;
private outputPath: string; private outputPath: string;
private firstResult: BuildResult;
constructor(public options: AngularBuilderOptions) { this.outputPath = options.outputPath; } constructor(public options: AngularBuilderOptions) { this.outputPath = options.outputPath; }
public rebuildBrowserDevTree(): Promise<BuildResult> { public rebuildBrowserDevTree(): Promise<BuildResult> {
this.browserDevBuilder = this.browserDevBuilder || this.makeBrowserDevBuilder(); this.browserDevBuilder = this.browserDevBuilder || this.makeBrowserDevBuilder();
return this.rebuild(this.browserDevBuilder); return this.rebuild(this.browserDevBuilder, 'js.dev');
} }
public rebuildBrowserProdTree(): Promise<BuildResult> { public rebuildBrowserProdTree(): Promise<BuildResult> {
this.browserProdBuilder = this.browserProdBuilder || this.makeBrowserProdBuilder(); this.browserProdBuilder = this.browserProdBuilder || this.makeBrowserProdBuilder();
return this.rebuild(this.browserProdBuilder); return this.rebuild(this.browserProdBuilder, 'js.prod');
} }
public rebuildNodeTree(): Promise<BuildResult> { public rebuildNodeTree(): Promise<BuildResult> {
this.nodeBuilder = this.nodeBuilder || this.makeNodeBuilder(); this.nodeBuilder = this.nodeBuilder || this.makeNodeBuilder();
return this.rebuild(this.nodeBuilder); return this.rebuild(this.nodeBuilder, 'js.cjs');
} }
public rebuildDartTree(): Promise<BuildResult> { public rebuildDartTree(): Promise<BuildResult> {
this.dartBuilder = this.dartBuilder || this.makeDartBuilder(); this.dartBuilder = this.dartBuilder || this.makeDartBuilder();
return this.rebuild(this.dartBuilder); return this.rebuild(this.dartBuilder, 'dart');
} }
@ -84,7 +85,55 @@ export class AngularBuilder {
} }
private rebuild(builder) { private rebuild(builder, name) {
return builder.build().then((result) => { printSlowTrees(result.graph); }); return builder.build().then(
(result) => {
if (!this.firstResult) {
this.firstResult = result;
}
printSlowTrees(result.graph);
writeBuildLog(result, name);
},
(error) => {
// the build tree is the same during rebuilds, only leaf properties of the nodes change
// so let's traverse it and get updated values for input/cache/output paths
if (this.firstResult) {
writeBuildLog(this.firstResult, name);
}
throw error;
});
}
}
function writeBuildLog(result: BuildResult, name: string) {
let logPath = `tmp/build.${name}.log`;
let prevLogPath = logPath + '.previous';
let formattedLogContent = JSON.stringify(broccoliNodeToBuildNode(result.graph), null, 2);
if (fs.existsSync(prevLogPath)) fs.unlinkSync(prevLogPath);
if (fs.existsSync(logPath)) fs.renameSync(logPath, prevLogPath);
fs.writeFileSync(logPath, formattedLogContent, {encoding: 'utf-8'});
}
function broccoliNodeToBuildNode(broccoliNode) {
let tree = broccoliNode.tree.newStyleTree || broccoliNode.tree;
return new BuildNode(
tree.description || tree.constructor.name,
tree.inputPath ? [tree.inputPath] : tree.inputPaths,
tree.cachePath,
tree.outputPath,
broccoliNode.subtrees.map(broccoliNodeToBuildNode)
);
}
class BuildNode {
constructor(public pluginName:string, public inputPaths: string[], public cachePath: string,
public outputPath: string, public inputNodes: BroccoliNode[]) {
} }
} }