|
|
|
@ -1,5 +1,5 @@
|
|
|
|
|
/* eslint-disable class-methods-use-this */
|
|
|
|
|
/* global window, textsecure, ConversationController, _, log, clearTimeout, process */
|
|
|
|
|
/* global window, textsecure, ConversationController, _, log, clearTimeout, process, Buffer, StringView, dcodeIO */
|
|
|
|
|
|
|
|
|
|
const is = require('@sindresorhus/is');
|
|
|
|
|
const { lokiRpc } = require('./loki_rpc');
|
|
|
|
@ -270,6 +270,16 @@ class LokiSnodeAPI {
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async getNodesMinVersion(minVersion) {
|
|
|
|
|
const _ = window.Lodash;
|
|
|
|
|
|
|
|
|
|
return _.flatten(
|
|
|
|
|
_.entries(this.versionPools)
|
|
|
|
|
.filter(v => semver.gte(v[0], minVersion))
|
|
|
|
|
.map(v => v[1])
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// use nodes that support more than 1mb
|
|
|
|
|
async getRandomProxySnodeAddress() {
|
|
|
|
|
/* resolve random snode */
|
|
|
|
@ -624,6 +634,98 @@ class LokiSnodeAPI {
|
|
|
|
|
return newSwarmNodes;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// helper function
|
|
|
|
|
async _requestLnsMapping(node, nameHash) {
|
|
|
|
|
log.debug('[lns] lns requests to {}:{}', node.ip, node.port);
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
// Hm, in case of proxy/onion routing we
|
|
|
|
|
// are not even using ip/port...
|
|
|
|
|
return lokiRpc(
|
|
|
|
|
`https://${node.ip}`,
|
|
|
|
|
node.port,
|
|
|
|
|
'get_lns_mapping',
|
|
|
|
|
{
|
|
|
|
|
name_hash: nameHash,
|
|
|
|
|
},
|
|
|
|
|
{},
|
|
|
|
|
'/storage_rpc/v1',
|
|
|
|
|
node
|
|
|
|
|
);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
log.warn('exception caught making lns requests to a node', node, e);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async getLnsMapping(lnsName) {
|
|
|
|
|
const _ = window.Lodash;
|
|
|
|
|
|
|
|
|
|
const input = Buffer.from(lnsName);
|
|
|
|
|
|
|
|
|
|
const output = await window.blake2b(input);
|
|
|
|
|
|
|
|
|
|
const nameHash = dcodeIO.ByteBuffer.wrap(output).toString('base64');
|
|
|
|
|
|
|
|
|
|
// Get nodes capable of doing LNS
|
|
|
|
|
let lnsNodes = await this.getNodesMinVersion('2.0.3');
|
|
|
|
|
lnsNodes = _.shuffle(lnsNodes);
|
|
|
|
|
|
|
|
|
|
// Loop until 3 confirmations
|
|
|
|
|
|
|
|
|
|
// We don't trust any single node, so we accumulate
|
|
|
|
|
// answers here and select a dominating answer
|
|
|
|
|
const allResults = [];
|
|
|
|
|
let ciphertextHex = null;
|
|
|
|
|
|
|
|
|
|
while (!ciphertextHex) {
|
|
|
|
|
if (lnsNodes.length < 3) {
|
|
|
|
|
log.error('Not enough nodes for lns lookup');
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// extract 3 and make requests in parallel
|
|
|
|
|
const nodes = lnsNodes.splice(0, 3);
|
|
|
|
|
|
|
|
|
|
// eslint-disable-next-line no-await-in-loop
|
|
|
|
|
const results = await Promise.all(
|
|
|
|
|
nodes.map(node => this._requestLnsMapping(node, nameHash))
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
results.forEach(res => {
|
|
|
|
|
if (
|
|
|
|
|
res &&
|
|
|
|
|
res.result &&
|
|
|
|
|
res.result.status === 'OK' &&
|
|
|
|
|
res.result.entries &&
|
|
|
|
|
res.result.entries.length > 0
|
|
|
|
|
) {
|
|
|
|
|
allResults.push(results[0].result.entries[0].encrypted_value);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const [winner, count] = _.maxBy(
|
|
|
|
|
_.entries(_.countBy(allResults)),
|
|
|
|
|
x => x[1]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (count >= 3) {
|
|
|
|
|
// eslint-disable-next-lint prefer-destructuring
|
|
|
|
|
ciphertextHex = winner;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const ciphertext = new Uint8Array(
|
|
|
|
|
StringView.hexToArrayBuffer(ciphertextHex)
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const res = await window.decryptLnsEntry(lnsName, ciphertext);
|
|
|
|
|
|
|
|
|
|
const pubkey = StringView.arrayBufferToHex(res);
|
|
|
|
|
|
|
|
|
|
return pubkey;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async getSnodesForPubkey(snode, pubKey) {
|
|
|
|
|
try {
|
|
|
|
|
const result = await lokiRpc(
|
|
|
|
|