2020-02-01 12:37:20 +03:00
|
|
|
#!/usr/bin/env node
|
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
import fs from "fs";
|
|
|
|
import _module from "module";
|
|
|
|
import { resolve } from "path";
|
|
|
|
|
2020-02-06 05:55:26 +03:00
|
|
|
import { assemble, disassemble, formatBytecode, parse, SemanticError, SemanticErrorSeverity } from "@ethersproject/asm";
|
2020-02-01 12:37:20 +03:00
|
|
|
|
|
|
|
import { ArgParser, CLI, Help, Plugin } from "../cli";
|
|
|
|
|
|
|
|
function repeat(text: string, length: number): string {
|
|
|
|
if (text.length === 0) { throw new Error("boo"); }
|
|
|
|
let result = text;
|
|
|
|
while (result.length < length) {
|
|
|
|
result += result;
|
|
|
|
}
|
|
|
|
return result.substring(0, length);
|
|
|
|
}
|
|
|
|
|
|
|
|
let cli = new CLI(null, {
|
|
|
|
account: false,
|
|
|
|
provider: false,
|
|
|
|
transaction: false
|
|
|
|
});
|
|
|
|
|
|
|
|
class AssemblePlugin extends Plugin {
|
|
|
|
|
|
|
|
filename: string;
|
|
|
|
content: string;
|
2020-02-06 05:55:26 +03:00
|
|
|
|
2020-02-01 12:37:20 +03:00
|
|
|
disassemble: boolean;
|
2020-02-06 05:55:26 +03:00
|
|
|
ignoreWarnings: boolean;
|
|
|
|
target: string;
|
|
|
|
|
2020-02-01 12:37:20 +03:00
|
|
|
defines: { [ key: string ]: string | boolean }
|
|
|
|
|
|
|
|
static getHelp(): Help {
|
|
|
|
return {
|
|
|
|
name: "[ FILENAME ]",
|
|
|
|
help: "Process the file (or stdin)"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
static getOptionHelp(): Array<Help> {
|
|
|
|
return [
|
2020-02-06 05:55:26 +03:00
|
|
|
{
|
|
|
|
name: "--define KEY=VALUE",
|
|
|
|
help: "provide assembler defines"
|
|
|
|
},
|
2020-02-01 12:37:20 +03:00
|
|
|
{
|
|
|
|
name: "--disassemble",
|
|
|
|
help: "Disassemble input bytecode"
|
|
|
|
},
|
|
|
|
{
|
2020-02-06 05:55:26 +03:00
|
|
|
name: "--ignore-warnings",
|
|
|
|
help: "Ignore warnings"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "--target LABEL",
|
|
|
|
help: "output LABEL bytecode (default: _)"
|
|
|
|
},
|
2020-02-01 12:37:20 +03:00
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
async prepareOptions(argParser: ArgParser): Promise<void> {
|
|
|
|
await super.prepareOptions(argParser);
|
|
|
|
|
|
|
|
// Get all the --define key=value pairs
|
|
|
|
this.defines = { };
|
|
|
|
argParser.consumeOptions("define").forEach((pair) => {
|
|
|
|
const match = pair.match(/([a-z][a-z0-9_]+)(=.*)?/i);
|
|
|
|
if (!match) {
|
|
|
|
this.throwError(`invalid define: ${ pair }`);
|
|
|
|
}
|
|
|
|
this.defines[match[1]] = (match[2] ? match[2].substring(1): true);
|
|
|
|
});
|
|
|
|
|
|
|
|
// We are disassembling...
|
|
|
|
this.disassemble = argParser.consumeFlag("disassemble");
|
2020-02-06 05:55:26 +03:00
|
|
|
|
|
|
|
this.ignoreWarnings = argParser.consumeFlag("ignore-warnings");
|
|
|
|
this.target = argParser.consumeOption("target");
|
2020-02-01 12:37:20 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
async prepareArgs(args: Array<string>): Promise<void> {
|
|
|
|
await super.prepareArgs(args);
|
|
|
|
|
|
|
|
if (args.length > 1) {
|
|
|
|
this.throwError("assembler requires at most one FILENAME");
|
|
|
|
|
|
|
|
} else if (args.length === 1) {
|
|
|
|
this.filename = resolve(args[0]);
|
|
|
|
this.content = fs.readFileSync(this.filename).toString();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
this.content = await (new Promise((resolve, reject) => {
|
|
|
|
let data = "";
|
|
|
|
|
|
|
|
process.stdin.setEncoding('utf8');
|
|
|
|
|
|
|
|
process.stdin.on("data", (chunk) => {
|
|
|
|
data += chunk;
|
|
|
|
});
|
|
|
|
|
|
|
|
process.stdin.on("end", () => {
|
|
|
|
resolve(data);
|
|
|
|
});
|
|
|
|
|
|
|
|
process.stdin.on("error", (error) => {
|
|
|
|
reject(error);
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.disassemble) {
|
|
|
|
const bytecodes: Array<string> = [ ];
|
|
|
|
const leftovers = this.content.replace(/(?:(?:0x)?((:?[0-9a-f][0-9a-f])*))/gi, (all, bytecode) => {
|
|
|
|
bytecodes.push(bytecode);
|
|
|
|
return repeat(" ", all.length);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (leftovers.trim()) {
|
|
|
|
|
|
|
|
// Process the leftovers
|
|
|
|
const chunks = leftovers.split(/(\s+)/);
|
|
|
|
const offset = (chunks[0] ? 0: chunks[1].length);
|
|
|
|
const column = (chunks[0] ? 0: chunks[1].split(/[\r\n]/).pop().length);
|
|
|
|
|
|
|
|
|
|
|
|
// Count the lines in the prefix
|
|
|
|
const prefix = this.content.substring(0, offset);
|
|
|
|
const lineNo = prefix.length - prefix.replace(/\r|\n|\r\n/g, '').length;
|
|
|
|
|
|
|
|
// Get teh actual error line
|
|
|
|
const line = this.content.substring(offset - column).split(/\n/)[0];
|
|
|
|
|
|
|
|
// Formatted output line
|
|
|
|
let output = `Invalid Bytecode Character found in line ${ lineNo + 1 }, column ${ column + 1 }\n`;
|
|
|
|
output += line + "\n";
|
|
|
|
output += repeat("-", column) + "^";
|
|
|
|
|
|
|
|
this.throwError(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.content = "0x" + bytecodes.join("");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async run(): Promise<void> {
|
|
|
|
if (this.disassemble) {
|
|
|
|
console.log(formatBytecode(disassemble(this.content)));
|
|
|
|
} else {
|
2020-02-06 05:55:26 +03:00
|
|
|
try {
|
|
|
|
const ast = parse(this.content, {
|
|
|
|
ignoreWarnings: !!this.ignoreWarnings
|
|
|
|
});
|
|
|
|
console.log(await assemble(ast, {
|
|
|
|
defines: this.defines,
|
|
|
|
filename: this.filename,
|
|
|
|
target: (this.target || "_")
|
|
|
|
}));
|
|
|
|
} catch (error) {
|
|
|
|
if ((<any>error).errors) {
|
|
|
|
((<any>error).errors).forEach((error: SemanticError) => {
|
|
|
|
if (error.severity === SemanticErrorSeverity.error) {
|
|
|
|
console.log(`Error: ${ error.message } (line: ${ error.node.location.line + 1 })`);
|
|
|
|
} else if (error.severity === SemanticErrorSeverity.warning) {
|
|
|
|
console.log(`Warning: ${ error.message } (line: ${ error.node.location.line + 1 })`);
|
|
|
|
} else {
|
|
|
|
console.log(error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2020-02-01 12:37:20 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cli.setPlugin(AssemblePlugin);
|
|
|
|
|
|
|
|
cli.run(process.argv.slice(2))
|
|
|
|
|