embark/lib/contracts/compiler.js

109 lines
3.4 KiB
JavaScript
Raw Normal View History

2017-01-29 06:28:01 +00:00
/*jshint esversion: 6, loopfunc: true */
2017-03-29 17:18:00 +00:00
let async = require('../utils/async_extend.js');
let SolcW = require('./solcW.js');
2017-02-17 12:14:44 +00:00
2017-03-30 11:12:39 +00:00
class Compiler {
constructor(options) {
this.plugins = options.plugins;
this.logger = options.logger;
}
2017-01-29 02:31:09 +00:00
2017-03-30 11:12:39 +00:00
compile_contracts(contractFiles, cb) {
2017-01-29 02:31:09 +00:00
2017-03-30 11:12:39 +00:00
let available_compilers = {
//".se": this.compile_serpent
".sol": this.compile_solidity.bind(this)
};
2017-01-29 02:31:09 +00:00
2017-03-30 11:12:39 +00:00
if (this.plugins) {
let compilerPlugins = this.plugins.getPluginsFor('compilers');
if (compilerPlugins.length > 0) {
compilerPlugins.forEach(function (plugin) {
plugin.compilers.forEach(function (compilerObject) {
available_compilers[compilerObject.extension] = compilerObject.cb;
});
2017-01-29 06:28:01 +00:00
});
2017-03-30 11:12:39 +00:00
}
2017-01-29 06:28:01 +00:00
}
2017-01-29 02:31:09 +00:00
2017-03-30 11:12:39 +00:00
let compiledObject = {};
2017-01-29 02:31:09 +00:00
2017-03-30 11:12:39 +00:00
async.eachObject(available_compilers,
function (extension, compiler, callback) {
// TODO: warn about files it doesn't know how to compile
let matchingFiles = contractFiles.filter(function (file) {
return (file.filename.match(/\.[0-9a-z]+$/)[0] === extension);
});
2016-08-14 12:04:34 +00:00
2017-03-30 11:12:39 +00:00
compiler.call(compiler, matchingFiles || [], function (err, compileResult) {
Object.assign(compiledObject, compileResult);
callback(err, compileResult);
});
},
function (err) {
cb(err, compiledObject);
}
2017-03-30 11:12:39 +00:00
);
}
2017-03-02 02:08:28 +00:00
2017-03-30 11:12:39 +00:00
compile_solidity(contractFiles, cb) {
let self = this;
let input = {};
let solcW;
async.waterfall([
function prepareInput(callback) {
for (let i = 0; i < contractFiles.length; i++) {
// TODO: this depends on the config
let filename = contractFiles[i].filename.replace('app/contracts/', '');
input[filename] = contractFiles[i].content.toString();
}
2017-03-02 02:08:28 +00:00
callback();
2017-03-30 11:12:39 +00:00
},
function loadCompiler(callback) {
// TODO: there ino need to load this twice
solcW = new SolcW();
if (solcW.isCompilerLoaded()) {
return callback();
}
2017-02-25 03:49:34 +00:00
2017-03-30 11:12:39 +00:00
self.logger.info("loading solc compiler..");
solcW.load_compiler(function () {
callback();
});
},
function compileContracts(callback) {
self.logger.info("compiling contracts...");
solcW.compile({sources: input}, 1, function (output) {
if (output.errors) {
return callback(new Error("Solidity errors: " + output.errors).message);
}
callback(null, output);
});
},
function createCompiledObject(output, callback) {
let json = output.contracts;
let compiled_object = {};
2017-02-25 03:49:34 +00:00
2017-03-30 11:12:39 +00:00
for (let className in json) {
let contract = json[className];
2017-02-25 03:49:34 +00:00
2017-03-30 11:12:39 +00:00
compiled_object[className] = {};
compiled_object[className].code = contract.bytecode;
compiled_object[className].runtimeBytecode = contract.runtimeBytecode;
compiled_object[className].realRuntimeBytecode = contract.runtimeBytecode.slice(0, -68);
compiled_object[className].swarmHash = contract.runtimeBytecode.slice(-68).slice(0, 64);
compiled_object[className].gasEstimates = contract.gasEstimates;
compiled_object[className].functionHashes = contract.functionHashes;
compiled_object[className].abiDefinition = JSON.parse(contract.interface);
}
callback(null, compiled_object);
2017-02-25 03:49:34 +00:00
}
2017-03-30 11:12:39 +00:00
], function (err, result) {
cb(err, result);
});
};
}
2016-08-14 12:04:34 +00:00
module.exports = Compiler;