-
Notifications
You must be signed in to change notification settings - Fork 33
/
index.js
193 lines (161 loc) · 6.92 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
'use strict'
const solc = require('solc')
const Bluebird = require('bluebird')
const chalk = require('chalk')
const waterfall = require('promise.waterfall')
const Web3 = require('web3')
const provider = new Web3.providers.HttpProvider('http://localhost:8545')
// TestRPC.provider() gives error "Synchronous requests are not supported."
// during newContract in latest version.
const web3 = new Web3(provider)
if (!web3.isConnected()) {
console.log('No local RPC client found to connect to.\nRun ganache-cli on port 8545 with ganache-cli.')
process.exit()
}
const newContract = require('eth-new-contract').default(provider)
const pkg = require('./package.json')
const contractName = 'ReplContainer'
const main = 'Main'
// speacial commands
const specialGlobals = {
'msg': [
{'prop': 'data', 'returnType': 'bytes'},
{'prop': 'gas', 'returnType': 'uint256'},
{'prop': 'sender', 'returnType': 'address'},
{'prop': 'sig', 'returnType': 'bytes4'},
{'prop': 'value', 'returnType': 'uint256'}
],
'block': [
{'prop': 'coinbase', 'returnType': 'address'},
{'prop': 'difficulty', 'returnType': 'uint256'},
{'prop': 'gaslimit', 'returnType': 'uint256'},
{'prop': 'number', 'returnType': 'uint256'},
{'prop': 'timestamp', 'returnType': 'uint256'}
],
'tx': [
{'prop': 'gasprice', 'returnType': 'uint'},
{'prop': 'origin', 'returnType': 'address'}
]
}
// contract template where the user commands are injected before compilation
const template = args => `pragma solidity ${pkg.dependencies.solc};
contract ${contractName} {
function ${main}() constant returns (${args.returnType}) {
${args.content}
return ${args.returnExpression}
}
}`
// helper function to get a list of desired attribute values
const pluck = (arr, key) => arr.map(item => item[key])
/** Parse the error message thrown with a naive compile in order to determine the actual return type. This is the hacky alternative to parsing an AST. */
const regexpReturnError = /Return argument type (.*) is not implicitly convertible to expected type \(type of first return variable\) bool./
const matchReturnTypeFromError = message => message.match(regexpReturnError)
// promisified getAccounts because Bluebird.promisify gives "callback is not a function"
const getAccounts = () => {
return new Bluebird((resolve, reject) => {
web3.eth.getAccounts((err, result) => {
if (err) return reject(err)
resolve(result)
})
})
}
/** Takes a list of commands and evaluates them inside a contract. Returns a promised result of the last command. Returns null if the command is not an expression. */
const evalSol = (commands, options = {}) => {
/** Returns true if the given command is an expression that can return a value. */
const isExpression = command => (!/[^=]=[^=]/.test(command)) && (!command.startsWith('delete'))
const lastCommand = commands[commands.length - 1]
const commandWithoutSemi = lastCommand.endsWith(';') ? lastCommand.substring(0, lastCommand.length - 1) : lastCommand
const content = commands.slice(0, commands.length - 1).join('\n')
let contentWithReturnExpression = commands.join('\n')
const specialCommand = specialGlobals[commandWithoutSemi]
const sourceFirstPass = template(specialCommand ? {
content,
returnType: pluck(specialGlobals[commandWithoutSemi], 'returnType').join(', '),
returnExpression: '(' + pluck(specialGlobals[commandWithoutSemi], 'prop').map(item => commandWithoutSemi + '.' + item).join(', ') + ');'
} : {
content: contentWithReturnExpression,
'returnType': 'bool',
returnExpression: isExpression(lastCommand) ? lastCommand : 'false;'
})
let source = sourceFirstPass
// Attempt to compile the first pass knowing that any return value (other than bool) will fail. Parse the error message to determine the correct return value and generate the correct source.
const compilationFirstPass = solc.compile(sourceFirstPass)
// get errors and warnings
// remove "Unused local variable" warning which will always trigger in REPL mode
if (compilationFirstPass.errors) {
const errorsFirstPass = compilationFirstPass.errors
.filter(err => !err.match(/: Warning: /))
if (options.loglevel === 'warnings') {
const warnings = compilationFirstPass.errors
.filter(err => err.match(/: Warning: /) && !err.match('Unused local variable'))
if (warnings.length) {
console.log(chalk.yellow(warnings.join('\n')))
}
}
// handle errors
if (errorsFirstPass.length > 0) {
const match = matchReturnTypeFromError(errorsFirstPass[0])
if (match) {
source = template({ content, 'returnType': match[1], returnExpression: lastCommand })
} else {
return Promise.reject(errorsFirstPass.join('\n'))
}
}
}
const compilation = solc.compile(source)
if (compilation.errors) {
const errors = compilation.errors
.filter(err => !err.match(/: Warning: /))
if (errors.length) {
console.error(chalk.red(`Error compiling Solidity:
${source.split('\n').map((line, n) => (n + 1) + ' ' + line).join('\n')}
${errors.join('\n ')}`, errors))
return Promise.reject(new Error('Error compiling Solidity'))
}
}
// if the last command is not an expression with a return value (e.g. an assignment), just return null
// this still must be done after compilation in case there is an error
if (!isExpression(lastCommand)) {
return Promise.resolve(null)
}
// if we made it this far, deploy the bytecode
const contract = compilation.contracts[`:${contractName}`]
const bytecode = contract.bytecode
const abi = JSON.parse(contract.interface)
const ReplContainer = web3.eth.contract(abi)
// execute the compiled contract and execute its Main function to get a result
return waterfall([
getAccounts,
accounts => newContract(ReplContainer, { from: accounts[0], data: bytecode, gas: 3e6 }),
contract => Bluebird.promisify(contract[main])()
])
}
/** Creates a new repl environment that can evaluate solidity commands. Returns a single function that takes a new command. */
module.exports.Repl = options => {
const commands = []
/** Takes a new command and returns the result of evaluating it in the current context. */
return rawCommand => {
const commandTrimmed = rawCommand.trim()
const command = commandTrimmed.endsWith(';')
? commandTrimmed.substring(0, commandTrimmed.length - 1)
: commandTrimmed
// ignore blank lines
if (command === '') {
return Promise.resolve(null)
}
return evalSol(commands.concat(command + ';'), options)
.then(result => {
commands.push(command + ';')
if (command in specialGlobals) {
let struct = {}
let props = specialGlobals[command]
for (let i = 0; i < props.length; i++) {
struct[props[i].prop] = result[i]
}
return JSON.stringify(struct, null, 2)
}
return result
})
}
}
module.exports.specialGlobals = specialGlobals