You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

248 lines
8.4 KiB

"use strict";
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
Object.defineProperty(exports, "__esModule", { value: true });
const res_1 = require("../utils/res");
const models_1 = require("../models");
const network = require("../network");
const constants_1 = require("../constants");
const short = require("short-uuid");
const lightning = require("../utils/lightning");
const wallet_1 = require("../utils/wallet");
const jsonUtils = require("../utils/json");
const sequelize_1 = require("sequelize");
const node_fetch_1 = require("node-fetch");
let queries = {};
const hub_pubkey = '023d70f2f76d283c6c4e58109ee3a2816eb9d8feb40b23d62469060a2b2867b77f';
function getPendingAccountings() {
return __awaiter(this, void 0, void 0, function* () {
const utxos = yield wallet_1.listUnspent(); // at least 1 confg
const accountings = yield models_1.models.Accounting.findAll({
where: {
onchain_address: {
[sequelize_1.Op.in]: utxos.map(utxo => utxo.address)
},
status: constants_1.default.statuses.pending
}
});
const ret = [];
accountings.forEach(a => {
const utxo = utxos.find(u => u.address === a.onchainAddress);
if (utxo) {
ret.push({
id: a.id,
pubkey: a.pubkey,
address: utxo.address,
amount: utxo.amount_sat,
confirmations: utxo.confirmations,
sourceApp: a.sourceApp,
date: a.sourceApp,
});
}
});
return ret;
});
}
function listUTXOs(req, res) {
return __awaiter(this, void 0, void 0, function* () {
try {
const ret = yield getPendingAccountings();
res_1.success(res, ret.map(acc => jsonUtils.accountingToJson(acc)));
}
catch (e) {
res_1.failure(res, e);
}
});
}
exports.listUTXOs = listUTXOs;
function getSuggestedSatPerByte() {
return __awaiter(this, void 0, void 0, function* () {
const MAX_AMT = 250;
try {
const r = yield node_fetch_1.default('https://mempool.space/api/v1/fees/recommended');
const j = yield r.json();
return Math.min(MAX_AMT, j.hourFee);
}
catch (e) {
return MAX_AMT;
}
});
}
// https://mempool.space/api/v1/fees/recommended
function genChannelAndConfirmAccounting(acc) {
return __awaiter(this, void 0, void 0, function* () {
const sat_per_byte = yield getSuggestedSatPerByte();
console.log("[WATCH]=> sat_per_byte", sat_per_byte);
try {
const r = yield lightning.openChannel({
node_pubkey: acc.pubkey,
local_funding_amount: acc.amount,
push_sat: 0,
sat_per_byte
});
console.log("[WATCH]=> CHANNEL OPENED!", r);
yield models_1.models.Accounting.update({
status: constants_1.default.statuses.confirmed,
fudingTxid: r.funding_txid_str
}, {
where: { id: acc.id }
});
console.log("[WATCH]=> ACCOUNTINGS UPDATED!");
}
catch (e) {
console.log('[ACCOUNTING] error creating channel', e);
}
});
}
function pollUTXOs() {
return __awaiter(this, void 0, void 0, function* () {
console.log("[WATCH]=> pollUTXOs");
const accs = yield getPendingAccountings();
if (!accs)
return;
console.log("[WATCH]=> accs", accs.length);
asyncForEach(accs, (acc) => __awaiter(this, void 0, void 0, function* () {
if (acc.confirmations < 1)
return;
yield genChannelAndConfirmAccounting(acc);
}));
});
}
function startWatchingUTXOs() {
setInterval(pollUTXOs, 600000); // every 10 minutes
}
exports.startWatchingUTXOs = startWatchingUTXOs;
function queryOnchainAddress(req, res) {
return __awaiter(this, void 0, void 0, function* () {
console.log('=> queryOnchainAddress');
const uuid = short.generate();
const owner = yield models_1.models.Contact.findOne({ where: { isOwner: true } });
const app = req.params.app;
const query = {
type: 'onchain_address',
uuid,
app
};
const opts = {
amt: constants_1.default.min_sat_amount,
dest: hub_pubkey,
data: {
type: constants_1.default.message_types.query,
message: {
content: JSON.stringify(query)
},
sender: { pub_key: owner.publicKey }
}
};
try {
yield network.signAndSend(opts);
}
catch (e) {
res_1.failure(res, e);
return;
}
let i = 0;
let interval = setInterval(() => {
if (i >= 15) {
clearInterval(interval);
delete queries[uuid];
res_1.failure(res, 'no response received');
return;
}
if (queries[uuid]) {
res_1.success(res, queries[uuid].result);
clearInterval(interval);
delete queries[uuid];
return;
}
i++;
}, 1000);
});
}
exports.queryOnchainAddress = queryOnchainAddress;
exports.receiveQuery = (payload) => __awaiter(void 0, void 0, void 0, function* () {
const dat = payload.content || payload;
const sender_pub_key = dat.sender.pub_key;
const content = dat.message.content;
const owner = yield models_1.models.Contact.findOne({ where: { isOwner: true } });
if (!sender_pub_key || !content || !owner) {
return console.log('=> wrong query format');
}
let q;
try {
q = JSON.parse(content);
}
catch (e) {
console.log("=> ERROR receiveQuery,", e);
return;
}
console.log('=> query received', q);
let result = '';
switch (q.type) {
case 'onchain_address':
const addy = yield lightning.newAddress(lightning.NESTED_PUBKEY_HASH);
const acc = {
date: new Date(),
pubkey: sender_pub_key,
onchainAddress: addy,
amount: 0,
sourceApp: q.app,
status: constants_1.default.statuses.pending,
error: '',
};
yield models_1.models.Accounting.create(acc);
result = addy;
}
const ret = {
type: q.type,
uuid: q.uuid,
app: q.app,
result,
};
const opts = {
amt: constants_1.default.min_sat_amount,
dest: sender_pub_key,
data: {
type: constants_1.default.message_types.query_response,
message: {
content: JSON.stringify(ret)
},
sender: { pub_key: owner.publicKey }
}
};
try {
yield network.signAndSend(opts);
}
catch (e) {
console.log("FAILED TO SEND QUERY_RESPONSE");
return;
}
});
exports.receiveQueryResponse = (payload) => __awaiter(void 0, void 0, void 0, function* () {
console.log('=> receiveQueryResponse');
const dat = payload.content || payload;
// const sender_pub_key = dat.sender.pub_key
const content = dat.message.content;
try {
const q = JSON.parse(content);
queries[q.uuid] = q;
}
catch (e) {
console.log("=> ERROR receiveQueryResponse,", e);
}
});
function asyncForEach(array, callback) {
return __awaiter(this, void 0, void 0, function* () {
for (let index = 0; index < array.length; index++) {
yield callback(array[index], index, array);
}
});
}
//# sourceMappingURL=queries.js.map