fix the log file space usage on path building error

the issue seems to be coming from the fact that we need at least
minSnodePool count snodes to build a path reliably.
pull/1839/head
audric 4 years ago
parent 07b6d7f6f1
commit c83776e510

@ -12,8 +12,8 @@ const minimumGuardCount = 2;
import { updateOnionPaths } from '../../state/ducks/onion';
const onionRequestHops = 3;
let onionPaths: Array<Array<Snode>> = [];
const ONION_REQUEST_HOPS = 3;
export let onionPaths: Array<Array<Snode>> = [];
/**
* Used for testing only
@ -47,7 +47,7 @@ export const clearTestOnionPath = () => {
export let pathFailureCount: Record<string, number> = {};
// tslint:disable-next-line: variable-name
export const TEST_resetPathFailureCount = () => {
export const resetPathFailureCount = () => {
pathFailureCount = {};
};
@ -57,7 +57,7 @@ const pathFailureThreshold = 3;
// This array is meant to store nodes will full info,
// so using GuardNode would not be correct (there is
// some naming issue here it seems)
let guardNodes: Array<Snode> = [];
export let guardNodes: Array<Snode> = [];
export const ed25519Str = (ed25519Key: string) => `(...${ed25519Key.substr(58)})`;
@ -121,16 +121,22 @@ export async function dropSnodeFromPath(snodeEd25519: string) {
export async function getOnionPath(toExclude?: Snode): Promise<Array<Snode>> {
let attemptNumber = 0;
while (onionPaths.length < minimumGuardCount) {
window?.log?.warn(
window?.log?.info(
`Must have at least ${minimumGuardCount} good onion paths, actual: ${onionPaths.length}, attempt #${attemptNumber} fetching more...`
);
// eslint-disable-next-line no-await-in-loop
await buildNewOnionPathsOneAtATime();
// should we add a delay? buildNewOnionPathsOneAtATime should act as one
// should we add a delay? buildNewOnionPathsOneA tATime should act as one
// reload goodPaths now
attemptNumber += 1;
if (attemptNumber >= 10) {
window?.log?.error('Failed to get an onion path after 10 attempts');
throw new Error(`Failed to build enough onion paths, current count: ${onionPaths.length}`);
}
}
if (onionPaths.length <= 0) {
@ -213,19 +219,16 @@ async function dropPathStartingWithGuardNode(guardNodeEd25519: string) {
const failingPathIndex = onionPaths.findIndex(p => p[0].pubkey_ed25519 === guardNodeEd25519);
if (failingPathIndex === -1) {
window?.log?.warn('No such path starts with this guard node ');
return;
} else {
window?.log?.info(
`Dropping path starting with guard node ${ed25519Str(
guardNodeEd25519
)}; index:${failingPathIndex}`
);
onionPaths = onionPaths.filter(p => p[0].pubkey_ed25519 !== guardNodeEd25519);
}
window?.log?.info(
`Dropping path starting with guard node ${ed25519Str(
guardNodeEd25519
)}; index:${failingPathIndex}`
);
onionPaths = onionPaths.filter(p => p[0].pubkey_ed25519 !== guardNodeEd25519);
const edKeys = guardNodes
.filter(g => g.pubkey_ed25519 !== guardNodeEd25519)
.map(n => n.pubkey_ed25519);
// make sure to drop the guard node even if the path starting with this guard node is not found
guardNodes = guardNodes.filter(g => g.pubkey_ed25519 !== guardNodeEd25519);
pathFailureCount[guardNodeEd25519] = 0;
@ -233,6 +236,12 @@ async function dropPathStartingWithGuardNode(guardNodeEd25519: string) {
// write the updates guard nodes to the db.
// the next call to getOnionPath will trigger a rebuild of the path
await internalUpdateGuardNodes(guardNodes);
}
async function internalUpdateGuardNodes(updatedGuardNodes: Array<Snode>) {
const edKeys = updatedGuardNodes.map(n => n.pubkey_ed25519);
await updateGuardNodes(edKeys);
}
@ -297,6 +306,7 @@ async function testGuardNode(snode: Snode) {
export async function selectGuardNodes(): Promise<Array<Snode>> {
// `getRandomSnodePool` is expected to refresh itself on low nodes
const nodePool = await SnodePool.getRandomSnodePool();
window.log.info('selectGuardNodes snodePool:', nodePool.length);
if (nodePool.length < desiredGuardCount) {
window?.log?.error(
'Could not select guard nodes. Not enough nodes in the pool: ',
@ -314,7 +324,7 @@ export async function selectGuardNodes(): Promise<Array<Snode>> {
// eslint-disable-next-line-no-await-in-loop
while (selectedGuardNodes.length < desiredGuardCount) {
if (shuffled.length < desiredGuardCount) {
window?.log?.error('Not enought nodes in the pool');
window?.log?.error('Not enough nodes in the pool');
break;
}
@ -336,9 +346,7 @@ export async function selectGuardNodes(): Promise<Array<Snode>> {
}
guardNodes = selectedGuardNodes;
const edKeys = guardNodes.map(n => n.pubkey_ed25519);
await updateGuardNodes(edKeys);
await internalUpdateGuardNodes(guardNodes);
return guardNodes;
}
@ -369,15 +377,19 @@ async function buildNewOnionPathsWorker() {
}
}
// If guard nodes is still empty (the old nodes are now invalid), select new ones:
if (guardNodes.length < minimumGuardCount) {
if (guardNodes.length < desiredGuardCount) {
// TODO: don't throw away potentially good guard nodes
guardNodes = await exports.selectGuardNodes();
}
// be sure to fetch again as that list might have been refreshed by selectGuardNodes
allNodes = await SnodePool.getRandomSnodePool();
window?.log?.info(
'LokiSnodeAPI::buildNewOnionPaths - after refetch, snodePool length:',
allNodes.length
);
// TODO: select one guard node and 2 other nodes randomly
let otherNodes = _.differenceBy(allNodes, guardNodes, 'pubkey_ed25519');
if (otherNodes.length < 2) {
if (otherNodes.length < SnodePool.minSnodePoolCount) {
window?.log?.warn(
'LokiSnodeAPI::buildNewOnionPaths - Too few nodes to build an onion path! Refreshing pool and retrying'
);
@ -387,17 +399,21 @@ async function buildNewOnionPathsWorker() {
// how to handle failing to rety
buildNewOnionPathsWorkerRetry = buildNewOnionPathsWorkerRetry + 1;
window.log.warn(
window?.log?.warn(
'buildNewOnionPathsWorker failed to get otherNodes. Current retry:',
buildNewOnionPathsWorkerRetry
);
if (buildNewOnionPathsWorkerRetry >= 3) {
// we failed enough. Something is wrong. Lets get out of that function and get a new fresh call.
window.log.warn(
window?.log?.warn(
`buildNewOnionPathsWorker failed to get otherNodes even after retries... Exiting after ${buildNewOnionPathsWorkerRetry} retries`
);
return;
} else {
window?.log?.info(
`buildNewOnionPathsWorker failed to get otherNodes. Next attempt: ${buildNewOnionPathsWorkerRetry}`
);
}
await buildNewOnionPathsWorker();
return;
@ -407,14 +423,12 @@ async function buildNewOnionPathsWorker() {
const guards = _.shuffle(guardNodes);
// Create path for every guard node:
const nodesNeededPerPaths = onionRequestHops - 1;
// Each path needs X (nodesNeededPerPaths) nodes in addition to the guard node:
const maxPath = Math.floor(
Math.min(
guards.length,
nodesNeededPerPaths ? otherNodes.length / nodesNeededPerPaths : otherNodes.length
)
const nodesNeededPerPaths = ONION_REQUEST_HOPS - 1;
// Each path needs nodesNeededPerPaths nodes in addition to the guard node:
const maxPath = Math.floor(Math.min(guards.length, otherNodes.length / nodesNeededPerPaths));
window?.log?.info(
`Building ${maxPath} onion paths based on guard nodes length: ${guards.length}, other nodes length ${otherNodes.length} `
);
// TODO: might want to keep some of the existing paths

@ -15,8 +15,7 @@ let snodeFailureCount: Record<string, number> = {};
import { Snode } from '../../data/data';
import { ERROR_CODE_NO_CONNECT } from './SNodeAPI';
// tslint:disable-next-line: variable-name
export const TEST_resetSnodeFailureCount = () => {
export const resetSnodeFailureCount = () => {
snodeFailureCount = {};
};

@ -8,6 +8,8 @@ import * as Data from '../../../ts/data/data';
import { allowOnlyOneAtATime } from '../utils/Promise';
import pRetry from 'p-retry';
import { ed25519Str } from '../onions/onionPath';
import { OnionPaths } from '../onions';
import { Onions, SNodeAPI } from '.';
/**
* If we get less than this snode in a swarm, we fetch new snodes for this pubkey
@ -18,7 +20,7 @@ const minSwarmSnodeCount = 3;
* If we get less than minSnodePoolCount we consider that we need to fetch the new snode pool from a seed node
* and not from those snodes.
*/
const minSnodePoolCount = 12;
export const minSnodePoolCount = 12;
/**
* If we do a request to fetch nodes from snodes and they don't return at least
@ -29,7 +31,7 @@ const minSnodePoolCount = 12;
export const requiredSnodesForAgreement = 24;
// This should be renamed to `allNodes` or something
let randomSnodePool: Array<Data.Snode> = [];
export let randomSnodePool: Array<Data.Snode> = [];
// We only store nodes' identifiers here,
const swarmCache: Map<string, Array<string>> = new Map();
@ -306,6 +308,9 @@ export async function refreshRandomPool(forceRefresh = false): Promise<void> {
}
window?.log?.info('updating snode list with snode pool length:', commonNodes.length);
randomSnodePool = commonNodes;
OnionPaths.resetPathFailureCount();
Onions.resetSnodeFailureCount();
await Data.updateSnodePoolOnDb(JSON.stringify(randomSnodePool));
},
{
@ -327,6 +332,9 @@ export async function refreshRandomPool(forceRefresh = false): Promise<void> {
// fallback to a seed node fetch of the snode pool
randomSnodePool = await exports.refreshRandomPoolDetail(seedNodes);
OnionPaths.resetPathFailureCount();
Onions.resetSnodeFailureCount();
await Data.updateSnodePoolOnDb(JSON.stringify(randomSnodePool));
}
});

@ -5,6 +5,8 @@ import { KeyPair } from '../../../libtextsecure/libsignal-protocol';
import { PubKey } from '../types';
import { fromHexToArray, toHex } from './String';
import { getConversationController } from '../conversations';
import { SnodePool } from '../snode_api';
import { OnionPaths } from '../onions';
export type HexKeyPair = {
pubKey: string;

@ -70,7 +70,7 @@ describe('OnionPathsErrors', () => {
guardPubkeys = TestUtils.generateFakePubKeys(3).map(n => n.key);
otherNodesPubkeys = TestUtils.generateFakePubKeys(9).map(n => n.key);
SNodeAPI.Onions.TEST_resetSnodeFailureCount();
SNodeAPI.Onions.resetSnodeFailureCount();
guardNodesArray = guardPubkeys.map(ed25519 => {
fakePortCurrent++;
@ -127,7 +127,7 @@ describe('OnionPathsErrors', () => {
OnionPaths.clearTestOnionPath();
OnionPaths.TEST_resetPathFailureCount();
OnionPaths.resetPathFailureCount();
await OnionPaths.getOnionPath();

@ -133,8 +133,8 @@ describe('OnionPaths', () => {
TestUtils.stubWindow('getSeedNodeList', () => ['seednode1']);
sandbox.stub(SNodeAPI.SnodePool, 'refreshRandomPoolDetail').resolves(fakeSnodePool);
SNodeAPI.Onions.TEST_resetSnodeFailureCount();
OnionPaths.TEST_resetPathFailureCount();
SNodeAPI.Onions.resetSnodeFailureCount();
OnionPaths.resetPathFailureCount();
// get a copy of what old ones look like
await OnionPaths.getOnionPath();

Loading…
Cancel
Save