-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathcrowdfund-live.js
168 lines (136 loc) · 5.59 KB
/
crowdfund-live.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
'use strict';
const assert = require('assert');
const bcoin = require('bcoin');
const MTX = bcoin.mtx;
const httpWallet = bcoin.http.Wallet;
const policy = bcoin.protocol.policy;
const Utils = require('./utils.js');
const network = 'simnet';
const composeWalletCrowdfund = async function composeWalletCrowdfund() {
const fundingTarget = 100000000; // 1 BTC
let amountToFund = 50000000; // .5 BTC
const rate = 10000; // satoshis per kb
const maxInputs = 5; // this will be used in calculating fee
const client = await new bcoin.http.Client({ network });
// Step 1: Setup our wallets and funding targets
const fundeeWallet = await new httpWallet({ id: 'fundee', network });
const fundeeAddress = await fundeeWallet.createAddress('default');
const funders = {
'funder-1': await new httpWallet({ id: 'funder-1', network }),
'funder-2': await new httpWallet({ id: 'funder-2', network })
};
// Step 2: Prepare coins for funding
// Because ALL | ANYONECANPAY inputs txs must keep a fixed number of outputs
// There can be no change outputs, which means that inputs must come from exact
// change outpoints/coins. So what we need to do is fund a tx for each wallet
// that gets sent back to the same wallet and has 1 output equal to the value
// we want to fund our tx with.
const fundingCoins = {};
// go through each funding wallet to prepare coins
for(let id in funders) {
const funder = funders[id];
const coins = await funder.getCoins();
const funderInfo = await funder.getInfo();
// First we need to see what the fee would be for each funding wallet
// Since different inputs can be using different transaction types
// of different sizes (p2sh, multisig, etc.) we will add the estimated fee on each input
// and use that to split the coins.
const funderKey = await funder.getWIF(coins[0].address);
const funderKeyring = new bcoin.keyring.fromSecret(funderKey.privateKey);
const feeForInput = Utils.getFeeForInput(coins[0], fundeeAddress.address, funderKeyring, rate);
amountToFund += feeForInput;
// go through available coins to find a coin equal to or greater than value to fund
let fundingCoin = {};
for(let coin of coins) {
if (coin.value === amountToFund) {
// if we already have a coin of the right value we can use that
fundingCoin = coin;
break;
}
}
if (!Object.keys(fundingCoin).length) {
// if we don't have a coin of the right amount to fund with
// we need to create one by sending the funder wallet
// a tx that includes an output of the right amount
const receiveAddress = await funder.createAddress('default') // send it back to the funder
const tx = await funder.send({
rate,
outputs: [{
value: amountToFund,
address: receiveAddress.address
}]
});
// get index of ouput for fundingCoin
let coinIndex;
for (let i=0; i < tx.outputs.length; i++) {
if (tx.outputs[i].value === amountToFund) {
coinIndex = i;
break;
}
}
assert(tx.outputs[coinIndex].value === amountToFund, 'value of output at index not correct');
// first argument is for the account
// default is being used for all examples
fundingCoin = await funder.getCoin('default', tx.hash, coinIndex);
}
fundingCoins[funder.id] = fundingCoin;
}
/**
fundingCoins should be object with wallet id and corresponding coin to be used for funding
```javascript
{ 'funder-1':
{ version: 1,
height: -1,
value: 50000000,
script: '76a914127cb1a40212169c49fe22d13307b18af1fa07ad88ac',
address: 'SNykaBMuTyeUQkK8exZyymWNrnYX5vVPuY',
coinbase: false,
hash: '163068016a39e2d9c869bcdb8646dbca93e07824db39217b5c444e7c61d1a82c',
index: 0 },
'funder-2':
{ version: 1,
height: -1,
value: 50000000,
script: '76a9146e08c73b355e690ba0b1198d578c4c6c52b3813688ac',
address: 'SXKossD65D7h62fhzGrntERBrPeXUfiC92',
coinbase: false,
hash: '11f3180c5069f2692f1ee1463257b21dc217441e792493c8f5ee230c35d97d96',
index: 0 }
}
```
**/
// Step 3: Create and template the mtx with output for funding target
const fundMe = new MTX();
fundMe.addOutput({value: fundingTarget, address: fundeeAddress.address });
// Step 4: Add inputs from the funder wallets
let inputCounter = 0;
for(let funder in fundingCoins) {
const wallet = funders[funder];
const coinOptions = fundingCoins[funder];
const key = await wallet.getWIF(coinOptions.address);
const keyring = new bcoin.keyring.fromSecret(key.privateKey);
Utils.addInput(coinOptions, inputCounter, fundMe, keyring);
inputCounter++;
assert(fundMe.isSigned(), 'Input has not been signed correctly');
}
// confirm that the transaction has been properly templated and signed
assert(
fundMe.inputs.length === Object.keys(funders).length,
'Number of inputs in MTX is incorrect'
);
assert(fundMe.verify(), 'MTX is malformed');
const tx = fundMe.toTX();
assert(tx.verify(fundMe.view), 'TX is malformed. Fix before broadcasting');
console.log('Total value of inputs: ', fundMe.getInputValue() );
console.log('Fee to go to miners: ', fundMe.getInputValue() - fundingTarget);
// Step 6: broadcast tx
try {
await client.broadcast(tx);
return tx;
} catch (e) {
console.log('There was a problem: ', e);
}
}
composeWalletCrowdfund()
.then(myCrowdfundTx => console.log('Transaction broadcast: ', myCrowdfundTx))
.catch(e => console.log('There was a problem: ', e));