2022-06-03 18:54:05 +02:00
|
|
|
import assert from 'assert';
|
2024-05-20 11:54:13 +02:00
|
|
|
|
2019-06-08 14:57:44 +02:00
|
|
|
import { LightningCustodianWallet } from '../../class';
|
2018-07-15 21:56:28 +02:00
|
|
|
|
2022-06-03 18:54:05 +02:00
|
|
|
jest.setTimeout(200 * 1000);
|
2021-09-21 13:46:24 +02:00
|
|
|
const baseUri = 'https://lndhub-staging.herokuapp.com';
|
|
|
|
|
2023-07-25 15:50:04 +02:00
|
|
|
// eslint-disable-next-line jest/no-disabled-tests
|
2021-07-13 16:55:05 +02:00
|
|
|
describe.skip('LightningCustodianWallet', () => {
|
2020-06-01 14:54:23 +02:00
|
|
|
const l1 = new LightningCustodianWallet();
|
2021-09-21 13:46:24 +02:00
|
|
|
l1.setBaseURI(baseUri);
|
2021-08-26 16:16:30 +02:00
|
|
|
l1.init();
|
2018-07-15 21:56:28 +02:00
|
|
|
|
2023-07-25 15:50:04 +02:00
|
|
|
// eslint-disable-next-line jest/no-disabled-tests
|
2019-06-16 00:36:00 +02:00
|
|
|
it.skip('issue credentials', async () => {
|
|
|
|
assert.ok(l1.refill_addressess.length === 0);
|
|
|
|
assert.ok(l1._refresh_token_created_ts === 0);
|
|
|
|
assert.ok(l1._access_token_created_ts === 0);
|
|
|
|
l1.balance = 'FAKE';
|
|
|
|
|
|
|
|
await l1.createAccount(false);
|
|
|
|
await l1.authorize();
|
|
|
|
|
|
|
|
assert.ok(l1.access_token);
|
|
|
|
assert.ok(l1.refresh_token);
|
|
|
|
assert.ok(l1._refresh_token_created_ts > 0);
|
|
|
|
assert.ok(l1._access_token_created_ts > 0);
|
|
|
|
console.log(l1.getSecret());
|
|
|
|
});
|
|
|
|
|
2018-09-01 01:28:19 +02:00
|
|
|
it('can create, auth and getbtc', async () => {
|
|
|
|
assert.ok(l1.refill_addressess.length === 0);
|
|
|
|
assert.ok(l1._refresh_token_created_ts === 0);
|
|
|
|
assert.ok(l1._access_token_created_ts === 0);
|
|
|
|
l1.balance = 'FAKE';
|
|
|
|
|
2018-09-04 00:44:45 +02:00
|
|
|
await l1.createAccount(true);
|
2018-09-01 01:28:19 +02:00
|
|
|
await l1.authorize();
|
|
|
|
await l1.fetchBtcAddress();
|
|
|
|
await l1.fetchBalance();
|
|
|
|
await l1.fetchInfo();
|
|
|
|
await l1.fetchTransactions();
|
|
|
|
await l1.fetchPendingTransactions();
|
|
|
|
|
|
|
|
assert.ok(l1.access_token);
|
|
|
|
assert.ok(l1.refresh_token);
|
|
|
|
assert.ok(l1._refresh_token_created_ts > 0);
|
|
|
|
assert.ok(l1._access_token_created_ts > 0);
|
|
|
|
assert.ok(l1.refill_addressess.length > 0);
|
|
|
|
assert.ok(l1.balance === 0);
|
|
|
|
assert.ok(l1.info_raw);
|
|
|
|
assert.ok(l1.pending_transactions_raw.length === 0);
|
|
|
|
assert.ok(l1.transactions_raw.length === 0);
|
|
|
|
assert.ok(l1.transactions_raw.length === l1.getTransactions().length);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can refresh token', async () => {
|
2020-06-01 14:54:23 +02:00
|
|
|
const oldRefreshToken = l1.refresh_token;
|
|
|
|
const oldAccessToken = l1.access_token;
|
2018-09-01 01:28:19 +02:00
|
|
|
await l1.refreshAcessToken();
|
|
|
|
assert.ok(oldRefreshToken !== l1.refresh_token);
|
|
|
|
assert.ok(oldAccessToken !== l1.access_token);
|
|
|
|
assert.ok(l1.access_token);
|
|
|
|
assert.ok(l1.refresh_token);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can use existing login/pass', async () => {
|
|
|
|
if (!process.env.BLITZHUB) {
|
|
|
|
console.error('process.env.BLITZHUB not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
2020-06-01 14:54:23 +02:00
|
|
|
const l2 = new LightningCustodianWallet();
|
2018-09-01 01:28:19 +02:00
|
|
|
l2.setSecret(process.env.BLITZHUB);
|
2021-09-21 13:46:24 +02:00
|
|
|
l2.setBaseURI(baseUri);
|
|
|
|
l2.init();
|
2018-09-01 01:28:19 +02:00
|
|
|
await l2.authorize();
|
|
|
|
await l2.fetchPendingTransactions();
|
|
|
|
await l2.fetchTransactions();
|
2018-09-22 13:40:47 +02:00
|
|
|
|
2018-09-01 01:28:19 +02:00
|
|
|
assert.ok(l2.pending_transactions_raw.length === 0);
|
|
|
|
assert.ok(l2.transactions_raw.length > 0);
|
|
|
|
assert.ok(l2.transactions_raw.length === l2.getTransactions().length);
|
2020-06-01 14:54:23 +02:00
|
|
|
for (const tx of l2.getTransactions()) {
|
2018-09-22 13:40:47 +02:00
|
|
|
assert.ok(typeof tx.fee !== 'undefined');
|
|
|
|
assert.ok(tx.value);
|
2018-12-27 01:47:46 +01:00
|
|
|
assert.ok(tx.timestamp);
|
|
|
|
assert.ok(tx.description || tx.memo, JSON.stringify(tx));
|
2018-09-22 13:40:47 +02:00
|
|
|
assert.ok(!isNaN(tx.value));
|
2018-09-25 23:23:56 +02:00
|
|
|
assert.ok(tx.type === 'bitcoind_tx' || tx.type === 'paid_invoice', 'unexpected tx type ' + tx.type);
|
2018-09-22 13:40:47 +02:00
|
|
|
}
|
2018-09-01 01:28:19 +02:00
|
|
|
await l2.fetchBalance();
|
|
|
|
assert.ok(l2.getBalance() > 0);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can decode & check invoice', async () => {
|
|
|
|
if (!process.env.BLITZHUB) {
|
|
|
|
console.error('process.env.BLITZHUB not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
2020-06-01 14:54:23 +02:00
|
|
|
const l2 = new LightningCustodianWallet();
|
2018-09-01 01:28:19 +02:00
|
|
|
l2.setSecret(process.env.BLITZHUB);
|
2021-09-21 13:46:24 +02:00
|
|
|
l2.setBaseURI(baseUri);
|
|
|
|
l2.init();
|
2018-09-01 01:28:19 +02:00
|
|
|
await l2.authorize();
|
|
|
|
|
|
|
|
let invoice =
|
|
|
|
'lnbc1u1pdcqpt3pp5ltuevvq2g69kdrzcegrs9gfqjer45rwjc0w736qjl92yvwtxhn6qdp8dp6kuerjv4j9xct5daeks6tnyp3xc6t50f582cscqp2zrkghzl535xjav52ns0rpskcn20takzdr2e02wn4xqretlgdemg596acq5qtfqhjk4jpr7jk8qfuuka2k0lfwjsk9mchwhxcgxzj3tsp09gfpy';
|
2020-06-01 14:54:23 +02:00
|
|
|
const decoded = l2.decodeInvoice(invoice);
|
2018-09-01 01:28:19 +02:00
|
|
|
|
|
|
|
assert.ok(decoded.payment_hash);
|
|
|
|
assert.ok(decoded.description);
|
|
|
|
assert.ok(decoded.num_satoshis);
|
2023-07-25 15:50:04 +02:00
|
|
|
assert.strictEqual(parseInt(decoded.num_satoshis, 10) * 1000, parseInt(decoded.num_millisatoshis, 10));
|
2018-09-01 01:28:19 +02:00
|
|
|
|
|
|
|
// checking that bad invoice cant be decoded
|
|
|
|
invoice = 'gsom';
|
|
|
|
let error = false;
|
|
|
|
try {
|
2019-12-26 20:34:35 +01:00
|
|
|
l2.decodeInvoice(invoice);
|
2018-09-01 01:28:19 +02:00
|
|
|
} catch (Err) {
|
|
|
|
error = true;
|
|
|
|
}
|
|
|
|
assert.ok(error);
|
|
|
|
});
|
|
|
|
|
2020-01-19 21:55:10 +01:00
|
|
|
it('decode can handle zero sats but present msats', async () => {
|
2020-06-01 14:54:23 +02:00
|
|
|
const l = new LightningCustodianWallet();
|
|
|
|
const decoded = l.decodeInvoice(
|
2020-01-19 21:55:10 +01:00
|
|
|
'lnbc89n1p0zptvhpp5j3h5e80vdlzn32df8y80nl2t7hssn74lzdr96ve0u4kpaupflx2sdphgfkx7cmtwd68yetpd5s9xct5v4kxc6t5v5s9gunpdeek66tnwd5k7mscqp2sp57m89zv0lrgc9zzaxy5p3d5rr2cap2pm6zm4n0ew9vyp2d5zf2mfqrzjqfxj8p6qjf5l8du7yuytkwdcjhylfd4gxgs48t65awjg04ye80mq7z990yqq9jsqqqqqqqqqqqqq05qqrc9qy9qsq9mynpa9ucxg53hwnvw323r55xdd3l6lcadzs584zvm4wdw5pv3eksdlcek425pxaqrn9u5gpw0dtpyl9jw2pynjtqexxgh50akwszjgq4ht4dh',
|
|
|
|
);
|
|
|
|
assert.strictEqual(decoded.num_satoshis, '8.9');
|
|
|
|
});
|
|
|
|
|
2020-01-12 01:00:33 +01:00
|
|
|
it('can decode invoice locally & remotely', async () => {
|
2020-01-19 21:55:10 +01:00
|
|
|
if (!process.env.BLITZHUB) {
|
|
|
|
console.error('process.env.BLITZHUB not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
2020-06-01 14:54:23 +02:00
|
|
|
const l2 = new LightningCustodianWallet();
|
2020-01-12 01:00:33 +01:00
|
|
|
l2.setSecret(process.env.BLITZHUB);
|
2021-09-21 13:46:24 +02:00
|
|
|
l2.setBaseURI(baseUri);
|
|
|
|
l2.init();
|
2020-01-12 01:00:33 +01:00
|
|
|
await l2.authorize();
|
2020-06-01 14:54:23 +02:00
|
|
|
const invoice =
|
2020-01-12 01:00:33 +01:00
|
|
|
'lnbc1u1pdcqpt3pp5ltuevvq2g69kdrzcegrs9gfqjer45rwjc0w736qjl92yvwtxhn6qdp8dp6kuerjv4j9xct5daeks6tnyp3xc6t50f582cscqp2zrkghzl535xjav52ns0rpskcn20takzdr2e02wn4xqretlgdemg596acq5qtfqhjk4jpr7jk8qfuuka2k0lfwjsk9mchwhxcgxzj3tsp09gfpy';
|
2020-06-01 14:54:23 +02:00
|
|
|
const decodedLocally = l2.decodeInvoice(invoice);
|
|
|
|
const decodedRemotely = await l2.decodeInvoiceRemote(invoice);
|
2020-01-12 01:00:33 +01:00
|
|
|
assert.strictEqual(decodedLocally.destination, decodedRemotely.destination);
|
|
|
|
assert.strictEqual(decodedLocally.num_satoshis, decodedRemotely.num_satoshis);
|
|
|
|
assert.strictEqual(decodedLocally.timestamp, decodedRemotely.timestamp);
|
2020-01-12 19:08:56 +01:00
|
|
|
assert.strictEqual(decodedLocally.expiry, decodedRemotely.expiry);
|
2020-01-12 01:00:33 +01:00
|
|
|
assert.strictEqual(decodedLocally.payment_hash, decodedRemotely.payment_hash);
|
|
|
|
assert.strictEqual(decodedLocally.description, decodedRemotely.description);
|
|
|
|
assert.strictEqual(decodedLocally.cltv_expiry, decodedRemotely.cltv_expiry);
|
|
|
|
});
|
|
|
|
|
2020-05-08 12:21:06 +02:00
|
|
|
it('can pay invoice from opennode', async () => {
|
|
|
|
if (!process.env.BLITZHUB) {
|
|
|
|
console.error('process.env.BLITZHUB not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!process.env.OPENNODE) {
|
|
|
|
console.error('process.env.OPENNODE not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const response = await fetch('https://api.opennode.co/v1/charges', {
|
|
|
|
method: 'POST',
|
2020-05-08 12:21:06 +02:00
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
Authorization: process.env.OPENNODE,
|
|
|
|
},
|
2024-08-15 17:46:32 +02:00
|
|
|
body: JSON.stringify({
|
|
|
|
amount: '0.01',
|
|
|
|
currency: 'USD',
|
|
|
|
}),
|
2020-05-08 12:21:06 +02:00
|
|
|
});
|
2024-08-15 17:46:32 +02:00
|
|
|
|
|
|
|
const res = await response.json();
|
|
|
|
if (!res.data || !res.data.lightning_invoice || !res.data.lightning_invoice.payreq) {
|
2020-05-08 12:21:06 +02:00
|
|
|
throw new Error('Opennode problem');
|
|
|
|
}
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const invoice = res.data.lightning_invoice.payreq;
|
2020-05-08 12:21:06 +02:00
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
const l2 = new LightningCustodianWallet();
|
2020-05-08 12:21:06 +02:00
|
|
|
l2.setSecret(process.env.BLITZHUB);
|
2021-09-21 13:46:24 +02:00
|
|
|
l2.setBaseURI(baseUri);
|
|
|
|
l2.init();
|
2020-05-08 12:21:06 +02:00
|
|
|
await l2.authorize();
|
|
|
|
await l2.fetchTransactions();
|
2020-06-01 14:54:23 +02:00
|
|
|
const txLen = l2.transactions_raw.length;
|
2020-05-08 12:21:06 +02:00
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
const start = +new Date();
|
2020-05-08 12:21:06 +02:00
|
|
|
await l2.payInvoice(invoice);
|
2020-06-01 14:54:23 +02:00
|
|
|
const end = +new Date();
|
2020-05-08 12:21:06 +02:00
|
|
|
if ((end - start) / 1000 > 9) {
|
|
|
|
console.warn('payInvoice took', (end - start) / 1000, 'sec');
|
|
|
|
}
|
|
|
|
|
|
|
|
await l2.fetchTransactions();
|
|
|
|
assert.strictEqual(l2.transactions_raw.length, txLen + 1);
|
2020-06-01 14:54:23 +02:00
|
|
|
const lastTx = l2.transactions_raw[l2.transactions_raw.length - 1];
|
2020-05-08 12:21:06 +02:00
|
|
|
assert.strictEqual(typeof lastTx.payment_preimage, 'string', 'preimage is present and is a string');
|
|
|
|
assert.strictEqual(lastTx.payment_preimage.length, 64, 'preimage is present and is a string of 32 hex-encoded bytes');
|
|
|
|
// transactions became more after paying an invoice
|
|
|
|
});
|
|
|
|
|
2021-02-23 20:31:15 +01:00
|
|
|
// turned off because acinq strike is shutting down
|
2023-07-25 15:50:04 +02:00
|
|
|
// eslint-disable-next-line jest/no-disabled-tests
|
2021-02-23 20:31:15 +01:00
|
|
|
it.skip('can pay invoice (acinq)', async () => {
|
2018-09-01 01:28:19 +02:00
|
|
|
if (!process.env.BLITZHUB) {
|
|
|
|
console.error('process.env.BLITZHUB not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!process.env.STRIKE) {
|
|
|
|
console.error('process.env.STRIKE not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const response = await fetch('https://api.strike.acinq.co/api/v1/charges', {
|
|
|
|
method: 'POST',
|
2018-09-01 01:28:19 +02:00
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
2024-08-15 17:46:32 +02:00
|
|
|
Authorization: `Basic ${btoa(process.env.STRIKE + ':')}`,
|
2018-09-01 01:28:19 +02:00
|
|
|
},
|
|
|
|
body: 'amount=1¤cy=btc&description=acceptance+test',
|
|
|
|
});
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const res = await response.json();
|
|
|
|
|
|
|
|
if (!res.payment_request) {
|
2018-09-01 01:28:19 +02:00
|
|
|
throw new Error('Strike problem: ' + JSON.stringify(res));
|
|
|
|
}
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const invoice = res.payment_request;
|
2018-09-01 01:28:19 +02:00
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
const l2 = new LightningCustodianWallet();
|
2018-09-01 01:28:19 +02:00
|
|
|
l2.setSecret(process.env.BLITZHUB);
|
2021-09-21 13:46:24 +02:00
|
|
|
l2.setBaseURI(baseUri);
|
|
|
|
l2.init();
|
2018-09-01 01:28:19 +02:00
|
|
|
await l2.authorize();
|
2018-12-27 01:47:46 +01:00
|
|
|
await l2.fetchTransactions();
|
2020-06-01 14:54:23 +02:00
|
|
|
const txLen = l2.transactions_raw.length;
|
2018-09-01 01:28:19 +02:00
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
const decoded = l2.decodeInvoice(invoice);
|
2018-09-01 01:28:19 +02:00
|
|
|
assert.ok(decoded.payment_hash);
|
|
|
|
assert.ok(decoded.description);
|
|
|
|
|
|
|
|
let start = +new Date();
|
|
|
|
await l2.payInvoice(invoice);
|
|
|
|
let end = +new Date();
|
|
|
|
if ((end - start) / 1000 > 9) {
|
|
|
|
console.warn('payInvoice took', (end - start) / 1000, 'sec');
|
|
|
|
}
|
2018-09-04 00:44:45 +02:00
|
|
|
|
2018-12-27 01:47:46 +01:00
|
|
|
await l2.fetchTransactions();
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(l2.transactions_raw.length, txLen + 1);
|
2020-06-01 14:54:23 +02:00
|
|
|
const lastTx = l2.transactions_raw[l2.transactions_raw.length - 1];
|
2019-05-04 00:26:00 +02:00
|
|
|
assert.strictEqual(typeof lastTx.payment_preimage, 'string', 'preimage is present and is a string');
|
|
|
|
assert.strictEqual(lastTx.payment_preimage.length, 64, 'preimage is present and is a string of 32 hex-encoded bytes');
|
2018-12-27 01:47:46 +01:00
|
|
|
// transactions became more after paying an invoice
|
|
|
|
|
2018-09-04 00:44:45 +02:00
|
|
|
// now, trying to pay duplicate invoice
|
|
|
|
start = +new Date();
|
|
|
|
let caughtError = false;
|
|
|
|
try {
|
|
|
|
await l2.payInvoice(invoice);
|
|
|
|
} catch (Err) {
|
|
|
|
caughtError = true;
|
|
|
|
}
|
|
|
|
assert.ok(caughtError);
|
2018-12-27 01:47:46 +01:00
|
|
|
await l2.fetchTransactions();
|
2021-02-23 20:31:15 +01:00
|
|
|
assert.strictEqual(l2.transactions_raw.length, txLen + 1);
|
|
|
|
// havent changed since last time
|
|
|
|
end = +new Date();
|
|
|
|
if ((end - start) / 1000 > 9) {
|
|
|
|
console.warn('duplicate payInvoice took', (end - start) / 1000, 'sec');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can pay invoice (bitrefill)', async () => {
|
|
|
|
if (!process.env.BLITZHUB) {
|
|
|
|
console.error('process.env.BLITZHUB not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!process.env.BITREFILL) {
|
|
|
|
console.error('process.env.BITREFILL not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const response = await fetch(`https://api-bitrefill.com/v1/lnurl_pay/${process.env.BITREFILL}/callback?amount=1000`, {
|
|
|
|
method: 'GET',
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
},
|
2021-02-23 20:31:15 +01:00
|
|
|
});
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const res = await response.json();
|
2021-02-23 20:31:15 +01:00
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
if (!res.pr) {
|
2021-02-23 20:31:15 +01:00
|
|
|
throw new Error('Bitrefill problem: ' + JSON.stringify(res));
|
|
|
|
}
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const invoice = res.pr;
|
2021-02-23 20:31:15 +01:00
|
|
|
|
|
|
|
const l2 = new LightningCustodianWallet();
|
|
|
|
l2.setSecret(process.env.BLITZHUB);
|
2021-09-21 13:46:24 +02:00
|
|
|
l2.setBaseURI(baseUri);
|
|
|
|
l2.init();
|
2021-02-23 20:31:15 +01:00
|
|
|
await l2.authorize();
|
|
|
|
await l2.fetchTransactions();
|
|
|
|
const txLen = l2.transactions_raw.length;
|
|
|
|
|
|
|
|
const decoded = l2.decodeInvoice(invoice);
|
|
|
|
assert.ok(decoded.payment_hash);
|
|
|
|
|
|
|
|
let start = +new Date();
|
|
|
|
await l2.payInvoice(invoice);
|
|
|
|
let end = +new Date();
|
|
|
|
if ((end - start) / 1000 > 9) {
|
|
|
|
console.warn('payInvoice took', (end - start) / 1000, 'sec');
|
|
|
|
}
|
|
|
|
|
|
|
|
await l2.fetchTransactions();
|
|
|
|
assert.strictEqual(l2.transactions_raw.length, txLen + 1);
|
|
|
|
const lastTx = l2.transactions_raw[l2.transactions_raw.length - 1];
|
|
|
|
assert.strictEqual(typeof lastTx.payment_preimage, 'string', 'preimage is present and is a string');
|
|
|
|
assert.strictEqual(lastTx.payment_preimage.length, 64, 'preimage is present and is a string of 32 hex-encoded bytes');
|
|
|
|
// transactions became more after paying an invoice
|
|
|
|
|
|
|
|
// now, trying to pay duplicate invoice
|
|
|
|
start = +new Date();
|
|
|
|
let caughtError = false;
|
|
|
|
try {
|
|
|
|
await l2.payInvoice(invoice);
|
|
|
|
} catch (Err) {
|
|
|
|
caughtError = true;
|
|
|
|
}
|
|
|
|
assert.ok(caughtError);
|
|
|
|
await l2.fetchTransactions();
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(l2.transactions_raw.length, txLen + 1);
|
2018-12-27 01:47:46 +01:00
|
|
|
// havent changed since last time
|
2018-09-04 00:44:45 +02:00
|
|
|
end = +new Date();
|
|
|
|
if ((end - start) / 1000 > 9) {
|
|
|
|
console.warn('duplicate payInvoice took', (end - start) / 1000, 'sec');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-12-11 23:52:46 +01:00
|
|
|
it('can create invoice and pay other blitzhub invoice', async () => {
|
2018-09-04 00:44:45 +02:00
|
|
|
if (!process.env.BLITZHUB) {
|
|
|
|
console.error('process.env.BLITZHUB not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
const lOld = new LightningCustodianWallet();
|
2018-09-25 23:23:56 +02:00
|
|
|
lOld.setSecret(process.env.BLITZHUB);
|
2021-09-21 13:46:24 +02:00
|
|
|
lOld.setBaseURI(baseUri);
|
|
|
|
lOld.init();
|
2018-09-25 23:23:56 +02:00
|
|
|
await lOld.authorize();
|
2018-12-27 01:47:46 +01:00
|
|
|
await lOld.fetchTransactions();
|
|
|
|
let txLen = lOld.transactions_raw.length;
|
2018-09-25 23:23:56 +02:00
|
|
|
|
|
|
|
// creating LND wallet
|
2020-06-01 14:54:23 +02:00
|
|
|
const lNew = new LightningCustodianWallet();
|
2021-09-21 13:46:24 +02:00
|
|
|
lNew.setBaseURI(baseUri);
|
|
|
|
lNew.init();
|
2018-09-25 23:23:56 +02:00
|
|
|
await lNew.createAccount(true);
|
|
|
|
await lNew.authorize();
|
|
|
|
await lNew.fetchBalance();
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(lNew.balance, 0);
|
2018-09-04 00:44:45 +02:00
|
|
|
|
2018-09-25 23:23:56 +02:00
|
|
|
let invoices = await lNew.getUserInvoices();
|
2022-04-14 20:20:59 +02:00
|
|
|
let invoice = await lNew.addInvoice(2, 'test memo');
|
2020-08-10 16:17:50 +02:00
|
|
|
const decoded = lNew.decodeInvoice(invoice);
|
2018-09-25 23:23:56 +02:00
|
|
|
let invoices2 = await lNew.getUserInvoices();
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(invoices2.length, invoices.length + 1);
|
2018-09-25 23:23:56 +02:00
|
|
|
assert.ok(invoices2[0].ispaid === false);
|
|
|
|
assert.ok(invoices2[0].description);
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(invoices2[0].description, 'test memo');
|
2018-09-25 23:23:56 +02:00
|
|
|
assert.ok(invoices2[0].payment_request);
|
2018-12-25 19:42:13 +01:00
|
|
|
assert.ok(invoices2[0].timestamp);
|
|
|
|
assert.ok(invoices2[0].expire_time);
|
2022-04-14 20:20:59 +02:00
|
|
|
assert.strictEqual(invoices2[0].amt, 2);
|
2020-06-01 14:54:23 +02:00
|
|
|
for (const inv of invoices2) {
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(inv.type, 'user_invoice');
|
2018-12-25 19:42:13 +01:00
|
|
|
}
|
2018-09-04 00:44:45 +02:00
|
|
|
|
2018-09-25 23:23:56 +02:00
|
|
|
await lOld.fetchBalance();
|
|
|
|
let oldBalance = lOld.balance;
|
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
const start = +new Date();
|
2018-09-25 23:23:56 +02:00
|
|
|
await lOld.payInvoice(invoice);
|
2020-06-01 14:54:23 +02:00
|
|
|
const end = +new Date();
|
2018-09-04 00:44:45 +02:00
|
|
|
if ((end - start) / 1000 > 9) {
|
|
|
|
console.warn('payInvoice took', (end - start) / 1000, 'sec');
|
|
|
|
}
|
2018-09-25 23:23:56 +02:00
|
|
|
|
|
|
|
invoices2 = await lNew.getUserInvoices();
|
|
|
|
assert.ok(invoices2[0].ispaid);
|
|
|
|
|
2020-08-10 16:17:50 +02:00
|
|
|
assert.ok(lNew.weOwnTransaction(decoded.payment_hash));
|
|
|
|
assert.ok(!lNew.weOwnTransaction('d45818ae11a584357f7b74da26012d2becf4ef064db015a45bdfcd9cb438929d'));
|
|
|
|
|
2018-09-25 23:23:56 +02:00
|
|
|
await lOld.fetchBalance();
|
|
|
|
await lNew.fetchBalance();
|
2022-04-14 20:20:59 +02:00
|
|
|
assert.strictEqual(oldBalance - lOld.balance, 2);
|
|
|
|
assert.strictEqual(lNew.balance, 2);
|
2018-09-25 23:23:56 +02:00
|
|
|
|
2018-12-27 01:47:46 +01:00
|
|
|
await lOld.fetchTransactions();
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(lOld.transactions_raw.length, txLen + 1, 'internal invoice should also produce record in payer`s tx list');
|
2020-06-01 14:54:23 +02:00
|
|
|
const newTx = lOld.transactions_raw.slice().pop();
|
2018-12-27 01:47:46 +01:00
|
|
|
assert.ok(typeof newTx.fee !== 'undefined');
|
|
|
|
assert.ok(newTx.value);
|
|
|
|
assert.ok(newTx.description || newTx.memo, JSON.stringify(newTx));
|
|
|
|
assert.ok(newTx.timestamp);
|
|
|
|
assert.ok(!isNaN(newTx.value));
|
|
|
|
assert.ok(newTx.type === 'paid_invoice', 'unexpected tx type ' + newTx.type);
|
|
|
|
|
2018-09-25 23:23:56 +02:00
|
|
|
// now, paying back that amount
|
2018-12-11 23:52:46 +01:00
|
|
|
oldBalance = lOld.balance;
|
2018-09-25 23:23:56 +02:00
|
|
|
invoice = await lOld.addInvoice(1, 'test memo');
|
|
|
|
await lNew.payInvoice(invoice);
|
|
|
|
await lOld.fetchBalance();
|
|
|
|
await lNew.fetchBalance();
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(lOld.balance - oldBalance, 1);
|
2022-04-14 20:20:59 +02:00
|
|
|
assert.strictEqual(lNew.balance, 1); // ok, forfeit this 1, unrecoverable
|
2019-01-09 02:54:59 +01:00
|
|
|
|
|
|
|
// now, paying same internal invoice. should fail:
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
let caughtError = false;
|
2019-01-09 02:54:59 +01:00
|
|
|
await lOld.fetchTransactions();
|
|
|
|
txLen = lOld.transactions_raw.length;
|
2020-06-01 14:54:23 +02:00
|
|
|
const invLen = (await lNew.getUserInvoices()).length;
|
2019-01-09 02:54:59 +01:00
|
|
|
try {
|
|
|
|
await lOld.payInvoice(invoice);
|
|
|
|
} catch (Err) {
|
2024-08-15 17:46:32 +02:00
|
|
|
caughtError = true;
|
2019-01-09 02:54:59 +01:00
|
|
|
}
|
2024-08-15 17:46:32 +02:00
|
|
|
assert.ok(caughtError);
|
2019-01-09 02:54:59 +01:00
|
|
|
|
|
|
|
await lOld.fetchTransactions();
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(txLen, lOld.transactions_raw.length, 'tx count should not be changed');
|
|
|
|
assert.strictEqual(invLen, (await lNew.getUserInvoices()).length, 'invoices count should not be changed');
|
2019-01-10 18:50:33 +01:00
|
|
|
|
|
|
|
// testing how limiting works:
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(lNew.user_invoices_raw.length, 1);
|
2019-01-10 18:50:33 +01:00
|
|
|
await lNew.addInvoice(666, 'test memo 2');
|
|
|
|
invoices = await lNew.getUserInvoices(1);
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(invoices.length, 2);
|
2022-04-14 20:20:59 +02:00
|
|
|
assert.strictEqual(invoices[0].amt, 2);
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(invoices[1].amt, 666);
|
2018-09-01 01:28:19 +02:00
|
|
|
});
|
2019-01-05 17:29:13 +01:00
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
it('can pay invoice with free amount (tippin.me)', async function () {
|
2019-01-05 17:29:13 +01:00
|
|
|
if (!process.env.BLITZHUB) {
|
|
|
|
console.error('process.env.BLITZHUB not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
// fetching invoice from tippin.me :
|
2019-01-05 17:29:13 +01:00
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const response = await fetch('https://tippin.me/lndreq/newinvoice.php', {
|
|
|
|
method: 'POST',
|
2019-01-05 17:29:13 +01:00
|
|
|
headers: {
|
|
|
|
Origin: 'https://tippin.me',
|
|
|
|
'Accept-Encoding': 'gzip, deflate, br',
|
|
|
|
'Accept-Language': 'en-GB,en-US;q=0.9,en;q=0.8',
|
|
|
|
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
|
|
|
|
Accept: 'application/json, text/javascript, */*; q=0.01',
|
|
|
|
},
|
|
|
|
body: 'userid=1188&username=overtorment&istaco=0&customAmnt=0&customMemo=',
|
|
|
|
});
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const res = await response.json();
|
2024-08-15 17:51:10 +02:00
|
|
|
if (!res || !res.message) {
|
2019-01-05 17:29:13 +01:00
|
|
|
throw new Error('tippin.me problem: ' + JSON.stringify(res));
|
|
|
|
}
|
2024-08-15 17:51:10 +02:00
|
|
|
const invoice = res.message;
|
2019-01-05 17:29:13 +01:00
|
|
|
|
|
|
|
// --> use to pay specific invoice
|
|
|
|
// invoice =
|
|
|
|
// 'lnbc1pwrp35spp5z62nvj8yw6luq7ns4a8utpwn2qkkdwdt0ludwm54wjeazk2xv5wsdpu235hqurfdcsx7an9wf6x7undv4h8ggpgw35hqurfdchx6eff9p6nzvfc8q5scqzysxqyz5vqj8xq6wz6dezmunw6qxleuw67ensjnt3fldltrmmkvzurge0dczpn94fkwwh7hkh5wqrhsvfegtvhswn252hn6uw5kx99dyumz4v5n9sp337py2';
|
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
const l2 = new LightningCustodianWallet();
|
2019-01-05 17:29:13 +01:00
|
|
|
l2.setSecret(process.env.BLITZHUB);
|
2021-09-21 13:46:24 +02:00
|
|
|
l2.setBaseURI(baseUri);
|
|
|
|
l2.init();
|
2019-01-05 17:29:13 +01:00
|
|
|
await l2.authorize();
|
|
|
|
await l2.fetchTransactions();
|
|
|
|
await l2.fetchBalance();
|
2020-04-29 13:31:03 +02:00
|
|
|
const oldBalance = +l2.balance;
|
2020-06-01 14:54:23 +02:00
|
|
|
const txLen = l2.transactions_raw.length;
|
2019-01-05 17:29:13 +01:00
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
const decoded = l2.decodeInvoice(invoice);
|
2019-01-05 17:29:13 +01:00
|
|
|
assert.ok(decoded.payment_hash);
|
|
|
|
assert.ok(decoded.description);
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(+decoded.num_satoshis, 0);
|
2019-01-05 17:29:13 +01:00
|
|
|
|
2019-01-06 15:40:15 +01:00
|
|
|
// first, tip invoice without amount should not work:
|
|
|
|
let gotError = false;
|
|
|
|
try {
|
|
|
|
await l2.payInvoice(invoice);
|
2019-01-07 21:12:37 +01:00
|
|
|
} catch (_) {
|
2019-01-06 15:40:15 +01:00
|
|
|
gotError = true;
|
|
|
|
}
|
|
|
|
assert.ok(gotError);
|
|
|
|
|
|
|
|
// then, pay:
|
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
const start = +new Date();
|
2019-01-05 17:29:13 +01:00
|
|
|
await l2.payInvoice(invoice, 3);
|
2020-06-01 14:54:23 +02:00
|
|
|
const end = +new Date();
|
2019-01-05 17:29:13 +01:00
|
|
|
if ((end - start) / 1000 > 9) {
|
|
|
|
console.warn('payInvoice took', (end - start) / 1000, 'sec');
|
|
|
|
}
|
|
|
|
|
|
|
|
await l2.fetchTransactions();
|
2019-01-21 14:55:39 +01:00
|
|
|
assert.strictEqual(l2.transactions_raw.length, txLen + 1);
|
2019-01-05 17:29:13 +01:00
|
|
|
// transactions became more after paying an invoice
|
|
|
|
|
|
|
|
await l2.fetchBalance();
|
2020-04-29 13:31:03 +02:00
|
|
|
assert.ok(oldBalance - l2.balance >= 3);
|
|
|
|
assert.ok(oldBalance - l2.balance < 10); // sanity check
|
2019-01-05 17:29:13 +01:00
|
|
|
});
|
2019-02-01 20:16:23 +01:00
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
it('cant create zero amt invoices yet', async () => {
|
2023-07-25 15:50:04 +02:00
|
|
|
const l = new LightningCustodianWallet();
|
|
|
|
l.setBaseURI(baseUri);
|
|
|
|
l.init();
|
|
|
|
assert.ok(l.refill_addressess.length === 0);
|
|
|
|
assert.ok(l._refresh_token_created_ts === 0);
|
|
|
|
assert.ok(l._access_token_created_ts === 0);
|
|
|
|
l.balance = 'FAKE';
|
|
|
|
|
|
|
|
await l.createAccount(true);
|
|
|
|
await l.authorize();
|
|
|
|
await l.fetchBalance();
|
|
|
|
|
|
|
|
assert.ok(l.access_token);
|
|
|
|
assert.ok(l.refresh_token);
|
|
|
|
assert.ok(l._refresh_token_created_ts > 0);
|
|
|
|
assert.ok(l._access_token_created_ts > 0);
|
|
|
|
assert.ok(l.balance === 0);
|
2019-02-01 20:16:23 +01:00
|
|
|
|
2019-04-04 14:08:01 +02:00
|
|
|
let err = false;
|
|
|
|
try {
|
2023-07-25 15:50:04 +02:00
|
|
|
await l.addInvoice(0, 'zero amt inv');
|
2019-04-04 14:08:01 +02:00
|
|
|
} catch (_) {
|
|
|
|
err = true;
|
|
|
|
}
|
|
|
|
assert.ok(err);
|
2019-04-14 21:08:16 +02:00
|
|
|
|
|
|
|
err = false;
|
|
|
|
try {
|
2023-07-25 15:50:04 +02:00
|
|
|
await l.addInvoice(NaN, 'zero amt inv');
|
2019-04-14 21:08:16 +02:00
|
|
|
} catch (_) {
|
|
|
|
err = true;
|
|
|
|
}
|
|
|
|
assert.ok(err);
|
2019-04-04 14:08:01 +02:00
|
|
|
});
|
|
|
|
|
2019-04-04 16:45:13 +02:00
|
|
|
it('cant pay negative free amount', async () => {
|
2019-04-04 14:08:01 +02:00
|
|
|
if (!process.env.BLITZHUB) {
|
|
|
|
console.error('process.env.BLITZHUB not set, skipped');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
// fetching invoice from tippin.me :
|
2019-04-04 14:08:01 +02:00
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const response = await fetch('https://tippin.me/lndreq/newinvoice.php', {
|
|
|
|
method: 'POST',
|
2019-04-04 14:08:01 +02:00
|
|
|
headers: {
|
|
|
|
Origin: 'https://tippin.me',
|
|
|
|
'Accept-Encoding': 'gzip, deflate, br',
|
|
|
|
'Accept-Language': 'en-GB,en-US;q=0.9,en;q=0.8',
|
|
|
|
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
|
|
|
|
Accept: 'application/json, text/javascript, */*; q=0.01',
|
|
|
|
},
|
|
|
|
body: 'userid=1188&username=overtorment&istaco=0&customAmnt=0&customMemo=',
|
|
|
|
});
|
|
|
|
|
2024-08-15 17:46:32 +02:00
|
|
|
const res = await response.json();
|
2024-08-15 17:51:10 +02:00
|
|
|
if (!res || !res.message) {
|
2019-04-04 14:08:01 +02:00
|
|
|
throw new Error('tippin.me problem: ' + JSON.stringify(res));
|
|
|
|
}
|
2024-08-15 17:51:10 +02:00
|
|
|
const invoice = res.message;
|
2020-06-01 14:54:23 +02:00
|
|
|
const l2 = new LightningCustodianWallet();
|
2019-04-04 14:08:01 +02:00
|
|
|
l2.setSecret(process.env.BLITZHUB);
|
2021-09-21 13:46:24 +02:00
|
|
|
l2.setBaseURI(baseUri);
|
|
|
|
l2.init();
|
2019-04-04 14:08:01 +02:00
|
|
|
await l2.authorize();
|
|
|
|
await l2.fetchTransactions();
|
|
|
|
await l2.fetchBalance();
|
2020-06-01 14:54:23 +02:00
|
|
|
const oldBalance = +l2.balance;
|
|
|
|
const txLen = l2.transactions_raw.length;
|
2019-04-04 14:08:01 +02:00
|
|
|
|
2020-06-01 14:54:23 +02:00
|
|
|
const decoded = l2.decodeInvoice(invoice);
|
2019-04-04 14:08:01 +02:00
|
|
|
assert.ok(decoded.payment_hash);
|
|
|
|
assert.ok(decoded.description);
|
|
|
|
assert.strictEqual(+decoded.num_satoshis, 0);
|
|
|
|
|
2019-02-01 20:16:23 +01:00
|
|
|
let error = false;
|
|
|
|
try {
|
2019-04-04 14:08:01 +02:00
|
|
|
await l2.payInvoice(invoice, -1);
|
2019-02-01 20:16:23 +01:00
|
|
|
} catch (Err) {
|
|
|
|
error = true;
|
|
|
|
}
|
|
|
|
assert.ok(error);
|
2019-04-04 14:08:01 +02:00
|
|
|
await l2.fetchBalance();
|
|
|
|
assert.strictEqual(l2.balance, oldBalance);
|
|
|
|
await l2.fetchTransactions();
|
|
|
|
assert.strictEqual(l2.transactions_raw.length, txLen);
|
2019-02-01 20:16:23 +01:00
|
|
|
});
|
2018-07-15 21:56:28 +02:00
|
|
|
});
|