-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathcode.js
371 lines (312 loc) · 11.8 KB
/
code.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
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
const {
Amount,
Coin,
KeyRing,
MTX,
Network,
Outpoint,
Script,
ScriptNum,
Stack
} = require('bcoin');
const fs = require('fs');
const assert = require('assert');
const { WalletClient, NodeClient } = require('bclient');
const network = Network.get('regtest');
const apiKey = fs.readFileSync('./secrets.env');
const clientOptions = {
network: network.type,
apiKey: apiKey.toString()
}
const walletClient = new WalletClient({...clientOptions, port: network.walletPort});
const nodeClient = new NodeClient({ ...clientOptions, port: network.rpcPort });
/**
* @param {String} locktime - Time that the script can not
* be redeemed before
* @param {Buffer} public key hash
* @returns {Script}
**/
function createScript(locktime='100', publicKeyHash) {
let pkh;
if (typeof publicKeyHash === 'string')
pkh = Buffer.from(publicKeyHash);
else pkh = publicKeyHash;
assert(Buffer.isBuffer(pkh), 'publicKey must be a Buffer');
const script = new Script();
// lock the transactions until
// the locktime has been reached
script.pushNum(ScriptNum.fromString(locktime, 10));
// check the locktime
script.pushSym('CHECKLOCKTIMEVERIFY');
// if verifies, drop time from the stack
script.pushSym('drop');
// duplicate item on the top of the stack
// which should be.the public key
script.pushSym('dup');
// hash the top item from the stack (the public key)
script.pushSym('hash160')
// push the hash to the top of the stack
script.pushData(pkh);
// confirm they match
script.pushSym('equalverify');
// confirm the signature matches
script.pushSym('checksig');
// Compile the script to its binary representation
// (you must do this if you change something!).
script.compile();
return script;
}
/**
* @param {Script} script to get corresponding address for
* @param {Network} to determine encoding based on network
* @returns {Address} - p2wsh segwit address for specified network
**/
function getAddress(script, network) {
// get the hash of the script
// and derive address from that
const p2wsh = script.forWitness();
const segwitAddress = p2wsh.getAddress().toBech32(network);
return segwitAddress;
}
/*
* Create a coin/UTXO that locks some funds w/ our script
* @param {Number} lockingValue - value in satoshis, defaults to 50,000
* @param {Address} lockingAddr - "destination" for the funds
* @returns {Coin}
*/
function createLockingCoin(lockingValue = 50000, lockingAddr) {
const cb = new MTX();
cb.addInput({
prevout: new Outpoint(),
script: new Script(),
sequence: 0xffffffff
});
// Send 50,000 satoshis to our locking address.
// this will lock up the funds to whoever can solve
// the CLTV script
cb.addOutput(lockingAddr, lockingValue);
// Convert the coinbase output to a Coin object
// In reality you might get these coins from a wallet.
return Coin.fromTX(cb, 0, -1);
}
/* script the inputs w/ our custom script for an mtx
* This is modeled after the scriptInput method on
* the `MTX` class
* @param {MTX} mtx with unscripted input
* @param {Number} index - index of input to script
* @param {Coin} coin- UTXO we are spending
* @param {KeyRing} ring - keyring we are signing with
* @returns {MTX}
*/
function scriptInput(mtx, index, coin, ring) {
const input = mtx.inputs[index];
const prev = coin.script;
const wsh = prev.getWitnessScripthash();
assert(ring instanceof KeyRing, 'Must pass a KeyRing to scriptInput');
wredeem = ring.getRedeem(wsh);
assert(wredeem, 'keyring has no redeem script');
const vector = new Stack();
// first add empty space in stack for signature and public key
vector.pushInt(0);
vector.pushInt(0);
// add the raw redeem script to the stack
vector.push(wredeem.toRaw());
input.witness.fromStack(vector);
mtx.inputs[index] = input;
return mtx;
}
function signInput(mtx, index, coin, ring) {
const input = mtx.inputs[index];
let witness, version;
const redeem = input.witness.getRedeem();
assert(
redeem,
'Witness has not been templated'
);
witness = input.witness;
version = 1;
const stack = witness.toStack();
// let's get the signature and replace the placeholder
// in the stack. We can use the MTX `signature` method
const sig =
mtx.signature(
index,
wredeem,
coin.value,
ring.privateKey,
null,
version
);
stack.setData(0, sig);
stack.setData(1, ring.getPublicKey());
witness.fromStack(stack);
return mtx;
}
/****
* THE SOLUTION
****/
// Setup some constants
const flags = Script.flags.STANDARD_VERIFY_FLAGS;
const amountToFund = Amount.fromBTC('.5');
async function mockSolution() {
const keyring = KeyRing.generate(true);
const keyring2 = KeyRing.generate(true);
const locktime = '100';
keyring.witness = true;
keyring2.witness = true;
// Step 1: make the script and save it to keychain
// Note: when making P2SH or P2WSH scripts,
// need to keep track of the script in order to redeem it later
const pkh = keyring.getKeyHash();
const script = createScript(locktime, pkh);
script.compile();
keyring.script = script;
// Step 2: create address to receive (and lock) the funds to
const lockingAddr = getAddress(script, network);
// Step 3: Create our fake tx that sends an output
// to our locking address
const coin = createLockingCoin(amountToFund.toValue(), lockingAddr);
// Now let's unlock the funds from our CLTV locking script
// Step 4: Setup our redeeming tx with the locked coin
// as our input, spending to another address, and the correct locktime
let mtx = new MTX();
mtx.addCoin(coin);
const receiveAddr = keyring2.getAddress('string', network);
// value of the input minus arbitrary amount for fee
// normally we could do this by querying our node to estimate rate
// or use the `fund` method if we had other coins to spend with
const receiveValue = coin.value - 1000;
mtx.addOutput(receiveAddr, receiveValue);
// now set the locktime
// in a live blockchain environment this will be checked against
// the current state of the chain in the mempool
// You can test if the CLTV script is working or not
// by changing this to a value less than what our script requires
// which will cause the `mtx.verify` call to fail below
mtx.setLocktime(parseInt(locktime));
// So now we have an mtx with the right input and output
// but our input still hasn't been signed
console.log('mtx:', mtx);
// Step 5: Sign and verify the input
// To do this with the bcoin API and a "live" wallet
// most of this is the same. bcoin though only checks
// for multisig smart contracts behind P2SH addresses
// so we need to manually handle the redeem script and
// signing of the transactions.
// You could make this more complex by putting a multisig
// script behind the locktime instead of a normal p2pkh
mtx = scriptInput(mtx, 0, coin, keyring);
mtx = signInput(mtx, 0, coin, keyring);
// if you console log the input being signed,
// you'll notice it now has a witness stack and redeem script
// before script, witness, and redeem were empty
console.log('signed mtx:', mtx);
assert(mtx.verify(flags), 'mtx did not verify');
// make tx immutable
const tx = mtx.toTX();
// it should still verify (need mtx's coin view to verify tx)
assert(tx.verify(mtx.view), 'tx did not verify');
console.log('Transaction verified!');
}
// Doing this with real money and the bcoin
// wallet system isn't too much different
// Key differences are:
// 1) We need to use real UTXOs/Coins
// 2) We need to manually keep track of the redeem script
// 3) We will interact with the REST API for signing
// of our transaction
// 4) We will need to check against the real height of
// a blockchain in order to redeem
async function lockAndRedeemCLTV(walletId) {
try {
const txInfoPath = './tx-info.json';
const wallet = walletClient.wallet(walletId);
// live solution
let redeemScript, lockingAddr, locktime;
let txInfo = fs.existsSync(txInfoPath) ? fs.readFileSync(txInfoPath) : '';
if (!txInfo.length) {
// Step 1: Setup wallet client and confirm balance
const { balance } = await wallet.getInfo();
assert(balance.confirmed > amountToFund.toValue(), 'Not enough funds!');
// Step 2: Setup keyring w/ pkh and create locking address
// that can be redeemed by our real wallet
const { publicKey, address } = await wallet.createAddress('default');
// create the keyring from the public key
// and get pkh for the locking script
const keyring = KeyRing.fromKey(Buffer.from(publicKey, 'hex'), true);
keyring.witness = true;
const pkh = keyring.getKeyHash();
// Get current height and set locktime to 10 blocks from now
const { chain: { height }} = await nodeClient.getInfo();
locktime = height + 10;
redeemScript = createScript(locktime.toString(), pkh);
lockingAddr = getAddress(redeemScript, network);
// Step 3: use the wallet client to send funds to the locking address
const output = {
value: amountToFund.toValue(),
address: lockingAddr
};
const lockedTx = await wallet.send({ outputs: [output], rate: 7000 });
console.log('transaction sent to mempool');
txInfo = { lockedTx, lockingAddr, redeemScript, locktime, redeemAddress: address };
fs.writeFileSync(txInfoPath, JSON.stringify(txInfo, null, 2));
// mine one block to get tx on chain
const minedBlock = await nodeClient.execute('generate', [1]);
console.log('Block mined: ', minedBlock);
} else {
const {
lockedTx,
lockingAddr,
redeemScript,
locktime,
redeemAddress
} = JSON.parse(txInfo);
const { chain: { height }} = await nodeClient.getInfo();
// in reality this could be block height or Unix epoch time
assert(locktime <= height, `Too soon to redeem the UTXO. Wait until block ${locktime}`);
// Prepare redeeming tx
// get index of utxo
const index = lockedTx.outputs.findIndex(
output => output.address === lockingAddr
);
const coinJSON = await nodeClient.getCoin(lockedTx.hash, index);
// create a new coin that references the UTXO we want to spend
const coin = Coin.fromJSON(coinJSON);
let mtx = new MTX();
mtx.addCoin(coin);
const { address } = await walletClient.createAddress('secondary', 'default');
// send to ourselves a value minus the fee
mtx.addOutput(address, coin.value - 1500);
// set nLocktime field on transaction
// mempool and chain will check against this
// to verify finality for each input
mtx.setLocktime(height);
const script = Script.fromRaw(redeemScript, 'hex');
const { privateKey } = await wallet.getWIF(redeemAddress);
const ring = KeyRing.fromSecret(privateKey, network);
ring.witness = true;
ring.script= script;
mtx = scriptInput(mtx, index, coin, ring);
mtx = signInput(mtx, index, coin, ring);
// note that the verification won't check against current height
// of the blockchain and node won't reject the tx and will
// still try and broadcast (check node for mempool verification errors)
assert(mtx.verify(), 'MTX did not verify');
const tx = mtx.toTX();
assert(tx.verify(mtx.view), 'TX did not verify');
const raw = tx.toRaw().toString('hex');
const result = await nodeClient.broadcast(raw);
assert(result.success, 'There was a problem broadcasting the tx');
// confirm the tx is in the mempool
const txFromHash = await nodeClient.getTX(tx.rhash());
assert(txFromHash, 'The tx does not appear to be in the mempool or chain');
console.log('Success!');
console.log('Tx: ', tx);
fs.writeFileSync(txInfoPath, '');
}
} catch(e) {
console.error('There was an error with live solution:', e);
}
};
// mockSolution(); return;
lockAndRedeemCLTV('witness');