embark/lib/pipeline/pipeline.js

384 lines
12 KiB
JavaScript
Raw Normal View History

const fs = require('../core/fs.js');
const async = require('async');
const utils = require('../utils/utils.js');
2017-12-12 17:20:57 +00:00
const webpack = require("webpack");
const constants = require('../constants');
2018-05-07 20:33:30 +00:00
const File = require('../core/file');
2016-08-21 16:02:02 +00:00
2018-01-17 00:17:52 +00:00
require("babel-preset-react");
require("babel-preset-es2015");
require("babel-preset-es2016");
require("babel-preset-es2017");
let pipeline;
2018-05-07 20:33:30 +00:00
// Override process.chdir so that we have a partial-implementation PWD for Windows
const realChdir = process.chdir;
process.chdir = (...args) => {
if (!process.env.PWD) {
process.env.PWD = process.cwd();
}
realChdir(...args);
};
2017-03-30 11:12:39 +00:00
class Pipeline {
2017-03-30 11:12:39 +00:00
constructor(options) {
this.buildDir = options.buildDir;
this.contractsFiles = options.contractsFiles;
this.assetFiles = options.assetFiles;
this.pipelinePlugins = options.pipelinePlugins;
this.pluginImports = options.pluginImports;
this.web3Location = options.web3Location;
this.providerCode = options.providerCode;
this.interceptLogs();
}
interceptLogs() {
const context = {};
context.console = console;
2018-05-08 15:31:47 +00:00
context.console.log = this.log.bind(this, 'log');
context.console.warn = this.log.bind(this, 'warn');
context.console.info = this.log.bind(this, 'info');
context.console.debug = this.log.bind(this, 'debug');
context.console.trace = this.log.bind(this, 'trace');
context.console.dir = this.log.bind(this, 'dir');
}
2018-05-08 15:31:47 +00:00
log(type, ...messages) {
process.send({result: constants.pipeline.log, message: messages, type});
2017-03-30 11:12:39 +00:00
}
build(abi, contractsJSON, path, callback) {
2017-03-30 11:12:39 +00:00
let self = this;
2018-05-08 13:36:50 +00:00
const importsList = {};
2018-05-08 13:36:50 +00:00
async.waterfall([
function buildTheContracts(next) {
self.buildContracts(contractsJSON, next);
},
function buildWeb3(next) {
self.buildWeb3JS(next);
},
function createImportList(next) {
importsList["Embark/EmbarkJS"] = fs.dappPath(".embark", 'embark.js');
importsList["Embark/web3"] = fs.dappPath(".embark", 'web3_instance.js');
2018-05-08 13:36:50 +00:00
self.pluginImports.forEach(function (importObject) {
let [importName, importLocation] = importObject;
importsList[importName] = importLocation;
});
2017-12-12 21:10:12 +00:00
2018-05-08 13:36:50 +00:00
next();
},
function writeContracts(next) {
async.each(Object.keys(contractsJSON), (contractName, eachCb) => {
2018-05-08 13:41:19 +00:00
self.buildContractJS(contractName, (err, contractCode) => {
let filePath = fs.dappPath(".embark", contractName + '.js');
importsList["Embark/contracts/" + contractName] = filePath;
fs.writeFile(filePath, contractCode, eachCb);
});
2018-05-08 13:36:50 +00:00
}, next);
},
function assetFileWrite(next) {
// limit:1 due to issues when downloading required files such as web3.js
async.eachOfLimit(self.assetFiles, 1, function (files, targetFile, cb) {
// limit:1 due to issues when downloading required files such as web3.js
async.mapLimit(files, 1,
function (file, fileCb) {
file = new File(file); // Re-instantiate a File as through the process, we lose its prototype
2018-05-08 15:31:47 +00:00
console.trace("reading " + file.filename);
2018-05-08 13:36:50 +00:00
if (file.filename.indexOf('.js') < 0) {
return file.content(function (fileContent) {
self.runPlugins(file, fileContent, fileCb);
});
}
2018-05-08 13:36:50 +00:00
// JS files
let realCwd;
2018-05-08 13:36:50 +00:00
async.waterfall([
2018-05-08 13:36:50 +00:00
function findImports(next) {
self.webpackRun(file.filename, {}, false, importsList, false, next);
},
2018-05-08 13:04:53 +00:00
2018-05-08 13:36:50 +00:00
function changeCwd(next) {
realCwd = utils.pwd();
process.chdir(fs.embarkPath(''));
next();
},
2018-05-08 13:25:37 +00:00
2018-05-08 13:36:50 +00:00
function runWebpack(next) {
self.webpackRun(file.filename, {}, true, importsList, true, next);
},
2018-05-08 13:25:37 +00:00
2018-05-08 13:36:50 +00:00
function changeCwdBack(next) {
process.chdir(realCwd);
next();
},
2018-05-08 13:25:37 +00:00
2018-05-08 13:36:50 +00:00
function checkFile(next) {
fs.access('./.embark/' + file.filename, (err) => {
if (err) {
2018-05-08 15:31:47 +00:00
console.error("couldn't find file: " + file.filename);
2018-05-08 13:36:50 +00:00
return next("couldn't find file: " + file.filename);
}
2018-05-08 13:25:37 +00:00
next();
2018-05-08 13:36:50 +00:00
});
},
function readFile(next) {
fs.readFile('./.embark/' + file.filename, (err, data) => {
if (err) {
return next(err);
}
next(null, data.toString());
});
},
2018-05-08 13:25:37 +00:00
2018-05-08 13:36:50 +00:00
function runPluginsOnContent(fileContent, next) {
self.runPlugins(file, fileContent, next);
}
2018-05-08 13:25:37 +00:00
2018-05-08 13:36:50 +00:00
], function (err, contentFile) {
2018-05-08 13:04:53 +00:00
if (err) {
2018-05-08 13:36:50 +00:00
process.chdir(realCwd);
2018-05-08 15:31:47 +00:00
console.error(err);
2018-05-08 13:36:50 +00:00
return fileCb(err);
2018-05-08 13:04:53 +00:00
}
2018-05-08 13:36:50 +00:00
fileCb(null, contentFile);
});
},
function (err, contentFiles) {
if (err) {
2018-05-08 15:31:47 +00:00
console.error('errors found while generating ' + targetFile);
2018-05-08 13:36:50 +00:00
}
let dir = targetFile.split('/').slice(0, -1).join('/');
2018-05-08 15:31:47 +00:00
console.trace("creating dir " + self.buildDir + dir);
2018-05-08 13:36:50 +00:00
fs.mkdirpSync(self.buildDir + dir);
2018-05-08 13:02:46 +00:00
2018-05-08 13:36:50 +00:00
// if it's a directory
if (targetFile.slice(-1) === '/' || targetFile.indexOf('.') === -1) {
let targetDir = targetFile;
2018-05-08 13:02:46 +00:00
2018-05-08 13:36:50 +00:00
if (targetDir.slice(-1) !== '/') {
targetDir = targetDir + '/';
2018-05-08 13:25:37 +00:00
}
2018-05-08 13:04:53 +00:00
2018-05-08 13:36:50 +00:00
async.each(contentFiles, function (file, mapCb) {
let filename = file.filename.replace(file.basedir + '/', '');
2018-05-08 15:31:47 +00:00
console.info("writing file " + (self.buildDir + targetDir + filename).bold.dim);
2018-05-08 13:25:37 +00:00
2018-05-08 13:36:50 +00:00
fs.copy(file.path, self.buildDir + targetDir + filename, {overwrite: true}, mapCb);
}, cb);
return;
2018-05-08 13:25:37 +00:00
}
2018-05-08 13:36:50 +00:00
let content = contentFiles.map(function (file) {
if (file === undefined) {
return "";
}
return file.content;
}).join("\n");
2018-05-08 15:31:47 +00:00
console.info("writing file " + (self.buildDir + targetFile).bold.dim);
2018-05-08 13:36:50 +00:00
fs.writeFile(self.buildDir + targetFile, content, cb);
}
);
},
next);
}
], callback);
2017-12-12 17:20:57 +00:00
}
2018-02-28 23:09:10 +00:00
runPlugins(file, fileContent, fileCb) {
const self = this;
if (self.pipelinePlugins.length <= 0) {
2018-05-08 13:04:53 +00:00
return fileCb(null, {
content: fileContent,
filename: file.filename,
path: file.path,
basedir: file.basedir,
modified: true
});
2018-02-28 23:09:10 +00:00
}
async.eachSeries(self.pipelinePlugins,
2018-05-08 13:04:53 +00:00
function (plugin, pluginCB) {
2018-02-28 23:09:10 +00:00
if (file.options && file.options.skipPipeline) {
return pluginCB();
}
fileContent = plugin.runPipeline({targetFile: file.filename, source: fileContent});
file.modified = true;
pluginCB();
},
function (err) {
if (err) {
2018-05-08 15:31:47 +00:00
console.error(err.message);
2018-02-28 23:09:10 +00:00
}
2018-05-08 13:04:53 +00:00
return fileCb(null, {
content: fileContent,
filename: file.filename,
path: file.path,
basedir: file.basedir,
modified: true
});
2018-02-28 23:09:10 +00:00
}
);
}
2018-02-23 19:04:08 +00:00
webpackRun(filename, options, includeModules, importsList, detectErrors, callback) {
let defaultOptions = {
2018-04-02 19:30:16 +00:00
entry: fs.dappPath(filename),
output: {
libraryTarget: 'umd',
2018-04-02 19:30:16 +00:00
path: fs.dappPath('.embark'),
filename: filename
},
resolve: {
alias: importsList,
modules: [
fs.embarkPath('node_modules'),
2018-04-02 19:30:16 +00:00
fs.dappPath('node_modules')
]
},
2018-05-08 13:04:53 +00:00
externals: function (context, request, callback) {
callback();
}
};
2018-05-08 13:04:53 +00:00
let webpackOptions = utils.recursiveMerge(defaultOptions, options);
if (includeModules) {
webpackOptions.module = {
rules: [
{
test: /\.css$/,
use: [{loader: "style-loader"}, {loader: "css-loader"}]
},
{
test: /\.scss$/,
use: [{loader: "style-loader"}, {loader: "css-loader"}]
},
{
test: /\.(png|woff|woff2|eot|ttf|svg)$/,
loader: 'url-loader?limit=100000'
},
{
test: /\.js$/,
loader: "babel-loader",
exclude: /(node_modules|bower_components)/,
options: {
2018-02-23 23:24:46 +00:00
presets: ['babel-preset-es2016', 'babel-preset-es2017', 'babel-preset-react'].map(require.resolve),
plugins: ["babel-plugin-webpack-aliases"].map(require.resolve),
compact: false
}
}
]
};
}
2018-05-08 15:31:47 +00:00
webpack(webpackOptions).run((err, stats) => {
if (err) {
console.error(err);
}
2018-02-23 19:04:08 +00:00
if (!detectErrors) {
return callback();
}
2018-05-08 15:31:47 +00:00
if (stats.hasErrors()) {
return callback(stats.toJson().errors.join("\n"));
2018-02-23 19:04:08 +00:00
}
callback();
});
}
2018-05-08 13:36:50 +00:00
buildContracts(contractsJSON, callback) {
fs.mkdirp(fs.dappPath(this.buildDir, 'contracts'), (err) => {
if (err) {
return callback(err);
}
async.each(Object.keys(contractsJSON), (className, eachCb) => {
let contract = contractsJSON[className];
fs.writeJson(fs.dappPath(this.buildDir, 'contracts', className + ".json"), contract, {spaces: 2}, eachCb);
}, callback);
});
2016-08-21 16:02:02 +00:00
}
2018-05-08 13:41:19 +00:00
buildContractJS(contractName, callback) {
fs.readFile(fs.dappPath(this.buildDir, 'contracts', contractName + '.json'), (err, contractJSON) => {
if (err) {
return callback(err);
}
contractJSON = contractJSON.toString();
2017-12-12 17:20:57 +00:00
2018-05-08 13:41:19 +00:00
let contractCode = "";
contractCode += "import web3 from 'Embark/web3';\n";
contractCode += "import EmbarkJS from 'Embark/EmbarkJS';\n";
contractCode += "let " + contractName + "JSONConfig = " + contractJSON + ";\n";
contractCode += "let " + contractName + " = new EmbarkJS.Contract(" + contractName + "JSONConfig);\n";
2017-12-12 17:20:57 +00:00
2018-05-08 13:41:19 +00:00
contractCode += "\n__embarkContext.execWhenReady(function() {\n";
contractCode += "\n" + contractName + ".setProvider(web3.currentProvider);\n";
contractCode += "\n});\n";
2018-05-08 13:41:19 +00:00
contractCode += "export default " + contractName + ";\n";
callback(null, contractCode);
});
}
2017-12-12 19:45:20 +00:00
buildWeb3JS(cb) {
2018-01-10 15:43:25 +00:00
const self = this;
let code = "";
async.waterfall([
function getImports(next) {
self.web3Location = self.web3Location.replace(/\\/g, '/'); // Import paths must always have forward slashes
code += "\nimport Web3 from '" + self.web3Location + "';\n";
2018-01-10 15:43:25 +00:00
code += "\n if (typeof web3 !== 'undefined') {";
code += "\n } else {";
code += "\n var web3 = new Web3();\n";
code += "\n }";
code += self.providerCode;
code += "\nglobal.__embarkContext = __mainContext.__loadManagerInstance;\n";
code += "\nwindow.web3 = web3;\n";
code += "\nexport default web3;\n";
next();
},
function makeDirectory(next) {
fs.mkdirp(fs.dappPath(".embark"), (err, _result) => {
next(err);
2018-01-10 15:43:25 +00:00
});
},
function writeFile(next) {
fs.writeFile(fs.dappPath(".embark", 'web3_instance.js'), code, next);
2018-01-10 15:43:25 +00:00
}
], cb);
2017-12-12 19:45:20 +00:00
}
2017-03-30 11:12:39 +00:00
}
2016-08-21 16:02:02 +00:00
process.on('message', (msg) => {
if (msg.action === constants.pipeline.init) {
pipeline = new Pipeline(msg.options);
return process.send({result: constants.pipeline.initiated});
}
if (msg.action === constants.pipeline.build) {
2018-05-08 13:52:06 +00:00
return pipeline.build(msg.abi, msg.contractsJSON, msg.path, (err) => {
process.send({result: constants.pipeline.built, error: err});
});
}
});
process.on('exit', () => {
process.exit(0);
});