@ethereum-sourcify/compilers
Version:
Wrapper around multiple compilers to download the right version and invoke the compilation with a common interface.
269 lines • 20.7 kB
JavaScript
;
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
var ownKeys = function(o) {
ownKeys = Object.getOwnPropertyNames || function (o) {
var ar = [];
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
return ar;
};
return ownKeys(o);
};
return function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
__setModuleDefault(result, mod);
return result;
};
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.findSolcPlatform = findSolcPlatform;
exports.useSolidityCompiler = useSolidityCompiler;
exports.getSolcExecutable = getSolcExecutable;
exports.getSolcJs = getSolcJs;
// TODO: Handle nodejs only dependencies
const path_1 = __importDefault(require("path"));
const fs_1 = __importDefault(require("fs"));
const child_process_1 = require("child_process");
const semver_1 = __importDefault(require("semver"));
const worker_threads_1 = require("worker_threads");
const logger_1 = require("../logger");
const common_1 = require("./common");
// eslint-disable-next-line @typescript-eslint/no-var-requires
const solc = require('solc');
const HOST_SOLC_REPO = 'https://binaries.soliditylang.org/';
function findSolcPlatform() {
if (process.platform === 'darwin' && process.arch === 'x64') {
return 'macosx-amd64';
}
if (process.platform === 'linux' && process.arch === 'x64') {
return 'linux-amd64';
}
if (process.platform === 'win32' && process.arch === 'x64') {
return 'windows-amd64';
}
return false;
}
/**
* Searches for a solc: first for a local executable version, then from HOST_SOLC_REPO
* and then using the getSolcJs function.
* Once the compiler is retrieved, it is used, and the stringified solc output is returned.
*
* @param version the version of solc to be used for compilation
* @param input a JSON object of the standard-json format compatible with solc
* @param log the logger
* @returns stringified solc output
*/
async function useSolidityCompiler(solcRepoPath, solJsonRepoPath, version, solcJsonInput, forceEmscripten = false) {
var _a, _b, _c;
// For nightly builds, Solidity version is saved as 0.8.17-ci.2022.8.9+commit.6b60524c instead of 0.8.17-nightly.2022.8.9+commit.6b60524c.
// Not possible to retrieve compilers with "-ci.".
if (version.includes('-ci.'))
version = version.replace('-ci.', '-nightly.');
const inputStringified = JSON.stringify(solcJsonInput);
let compiled;
const solcPlatform = findSolcPlatform();
let solcPath;
if (solcPlatform && !forceEmscripten) {
// Catch, if this fails we'll fall back to solc-js e.g. very early solc 0.1.4
try {
solcPath = await getSolcExecutable(solcRepoPath, solcPlatform, version);
}
catch (error) {
(0, logger_1.logError)('Error getting solc executable', {
error,
solcPlatform,
version,
solcRepoPath,
solJsonRepoPath,
});
}
}
let startCompilation;
if (solcPath && !forceEmscripten) {
(0, logger_1.logInfo)('Compiling with solc binary', { version, solcPath });
startCompilation = Date.now();
try {
compiled = await (0, common_1.asyncExec)(`${solcPath} --standard-json`, inputStringified, 250 * 1024 * 1024);
}
catch (error) {
if ((error === null || error === void 0 ? void 0 : error.code) === 'ENOBUFS') {
throw new Error('Compilation output size too large');
}
throw error;
}
}
else {
(0, logger_1.logInfo)('Compiling with solc-js', { version });
const solJson = await getSolcJs(solJsonRepoPath, version);
startCompilation = Date.now();
if (solJson) {
const coercedVersion = (_b = (_a = semver_1.default.coerce(new semver_1.default.SemVer(version))) === null || _a === void 0 ? void 0 : _a.version) !== null && _b !== void 0 ? _b : '';
// Run Worker for solc versions < 0.4.0 for clean compiler context. See https://github.com/argotorg/sourcify/issues/1099
if (semver_1.default.lt(coercedVersion, '0.4.0')) {
compiled = await new Promise((resolve, reject) => {
const worker = importWorker(path_1.default.resolve(__dirname, './compilerWorker.ts'), {
workerData: { solJsonRepoPath, version, inputStringified },
});
worker.once('message', (result) => {
resolve(result);
});
worker.once('error', (error) => {
reject(error);
});
});
}
else {
compiled = solJson.compile(inputStringified);
}
}
}
const endCompilation = Date.now();
(0, logger_1.logInfo)('Local compiler - Compilation done', {
compiler: 'solidity',
timeInMs: endCompilation - startCompilation,
});
if (!compiled) {
throw new Error('Compilation failed. No output from the compiler.');
}
const compiledJSON = JSON.parse(compiled);
const errorMessages = (_c = compiledJSON === null || compiledJSON === void 0 ? void 0 : compiledJSON.errors) === null || _c === void 0 ? void 0 : _c.filter((e) => e.severity === 'error');
if (errorMessages && errorMessages.length > 0) {
(0, logger_1.logError)('Compiler error', {
errorMessages,
});
throw new common_1.CompilerError('Compiler error', errorMessages);
}
return compiledJSON;
}
async function getSolcExecutable(solcRepoPath, platform, version) {
const fileName = `solc-${platform}-v${version}`;
const solcPath = path_1.default.join(solcRepoPath, fileName);
if (fs_1.default.existsSync(solcPath) && validateSolcPath(solcPath)) {
(0, logger_1.logDebug)('Found existing solc', { version, platform, solcPath });
return solcPath;
}
await fetchAndSaveSolc(platform, solcPath, version, fileName);
if (!validateSolcPath(solcPath)) {
throw new Error(`Solc not found. Maybe an incorrect version was provided. ${solcPath} - ${version} - ${platform}`);
}
return solcPath;
}
function validateSolcPath(solcPath) {
var _a;
// TODO: Handle nodejs only dependencies
const spawned = (0, child_process_1.spawnSync)(solcPath, ['--version']);
if (spawned.status === 0) {
return true;
}
const error = ((_a = spawned === null || spawned === void 0 ? void 0 : spawned.error) === null || _a === void 0 ? void 0 : _a.message) ||
spawned.stderr.toString() ||
'Error running solc, are you on the right platoform? (e.g. x64 vs arm)';
(0, logger_1.logWarn)(error);
return false;
}
/**
* Fetches a solc binary and saves it to the given path.
*
* If platform is "bin", it will download the solc-js binary.
*/
async function fetchAndSaveSolc(platform, solcPath, version, fileName) {
const encodedURIFilename = encodeURIComponent(fileName);
const githubSolcURI = `${HOST_SOLC_REPO}${platform}/${encodedURIFilename}`;
(0, logger_1.logInfo)('Fetching solc', { version, platform, githubSolcURI, solcPath });
let res = await (0, common_1.fetchWithBackoff)(githubSolcURI);
let status = res.status;
let buffer;
// handle case in which the response is a link to another version
if (status === 200) {
buffer = await res.arrayBuffer();
const responseText = Buffer.from(buffer).toString();
if (/^([\w-]+)-v(\d+\.\d+\.\d+)\+commit\.([a-fA-F0-9]+).*$/.test(responseText)) {
const githubSolcURI = `${HOST_SOLC_REPO}${platform}/${responseText}`;
res = await (0, common_1.fetchWithBackoff)(githubSolcURI);
status = res.status;
buffer = await res.arrayBuffer();
}
}
if (status === 200 && buffer) {
fs_1.default.mkdirSync(path_1.default.dirname(solcPath), { recursive: true });
try {
fs_1.default.unlinkSync(solcPath);
}
catch (_e) {
undefined;
}
fs_1.default.writeFileSync(solcPath, new DataView(buffer), { mode: 0o755 });
(0, logger_1.logInfo)('Saved solc', { version, platform, githubSolcURI, solcPath });
}
else {
(0, logger_1.logError)('Failed fetching solc', {
version,
platform,
githubSolcURI,
solcPath,
});
throw new Error(`Failed fetching solc ${version} for platform ${platform}. Please check if the version is valid.`);
}
}
/**
* Fetches the requested version of the Solidity compiler (soljson).
* First attempts to search locally; if that fails, falls back to downloading it.
*
* @param version the solc version to retrieve: the expected format is
*
* "[v]<major>.<minor>.<patch>+commit.<hash>"
*
* e.g.: "0.6.6+commit.6c089d02"
*
* defaults to "latest"
*
* @param log a logger to track the course of events
*
* @returns the requested solc instance
*/
async function getSolcJs(solJsonRepoPath, version) {
// /^\d+\.\d+\.\d+\+commit\.[a-f0-9]{8}$/
version = version.trim();
if (version !== 'latest' && !version.startsWith('v')) {
version = 'v' + version;
}
const fileName = `soljson-${version}.js`;
const solJsonPath = path_1.default.resolve(solJsonRepoPath, fileName);
if (!fs_1.default.existsSync(solJsonPath)) {
(0, logger_1.logDebug)('Solc-js not found locally, downloading', {
version,
solJsonPath,
});
await fetchAndSaveSolc('bin', solJsonPath, version, fileName);
}
const solcjsImports = await Promise.resolve(`${solJsonPath}`).then(s => __importStar(require(s)));
return solc.setupMethods(solcjsImports);
}
// https://stackoverflow.com/questions/71795469/ts-node-using-worker-thread-cause-cannot-use-import-statement-outside-a-module
function importWorker(path, options) {
const resolvedPath = require.resolve(path);
return new worker_threads_1.Worker(resolvedPath, Object.assign(Object.assign({}, options), { execArgv: /\.ts$/.test(resolvedPath)
? ['--require', 'ts-node/register']
: undefined }));
}
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic29saWRpdHlDb21waWxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9saWIvc29saWRpdHlDb21waWxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQW1CQSw0Q0FXQztBQVlELGtEQThGQztBQUVELDhDQW1CQztBQXlGRCw4QkF1QkM7QUE3UUQsd0NBQXdDO0FBQ3hDLGdEQUF3QjtBQUN4Qiw0Q0FBb0I7QUFDcEIsaURBQTBDO0FBQzFDLG9EQUE0QjtBQUU1QixtREFBd0M7QUFDeEMsc0NBQWlFO0FBQ2pFLHFDQUFzRTtBQU10RSw4REFBOEQ7QUFDOUQsTUFBTSxJQUFJLEdBQUcsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDO0FBRTdCLE1BQU0sY0FBYyxHQUFHLG9DQUFvQyxDQUFDO0FBRTVELFNBQWdCLGdCQUFnQjtJQUM5QixJQUFJLE9BQU8sQ0FBQyxRQUFRLEtBQUssUUFBUSxJQUFJLE9BQU8sQ0FBQyxJQUFJLEtBQUssS0FBSyxFQUFFLENBQUM7UUFDNUQsT0FBTyxjQUFjLENBQUM7SUFDeEIsQ0FBQztJQUNELElBQUksT0FBTyxDQUFDLFFBQVEsS0FBSyxPQUFPLElBQUksT0FBTyxDQUFDLElBQUksS0FBSyxLQUFLLEVBQUUsQ0FBQztRQUMzRCxPQUFPLGFBQWEsQ0FBQztJQUN2QixDQUFDO0lBQ0QsSUFBSSxPQUFPLENBQUMsUUFBUSxLQUFLLE9BQU8sSUFBSSxPQUFPLENBQUMsSUFBSSxLQUFLLEtBQUssRUFBRSxDQUFDO1FBQzNELE9BQU8sZUFBZSxDQUFDO0lBQ3pCLENBQUM7SUFDRCxPQUFPLEtBQUssQ0FBQztBQUNmLENBQUM7QUFDRDs7Ozs7Ozs7O0dBU0c7QUFFSSxLQUFLLFVBQVUsbUJBQW1CLENBQ3ZDLFlBQW9CLEVBQ3BCLGVBQXVCLEVBQ3ZCLE9BQWUsRUFDZixhQUFnQyxFQUNoQyxlQUFlLEdBQUcsS0FBSzs7SUFFdkIs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