pull/1624/head
Audric Ackermann 4 years ago
parent bad438eaee
commit 372119b17b
No known key found for this signature in database
GPG Key ID: 999F434D76324AD4

@ -177,7 +177,7 @@ export const sendViaOnion = async (
}); });
}, },
{ {
retries: 10, // each path can fail 3 times before being dropped, we have 3 paths at most retries: 9, // each path can fail 3 times before being dropped, we have 3 paths at most
factor: 2, factor: 2,
minTimeout: 200, minTimeout: 200,
maxTimeout: 4000, maxTimeout: 4000,

@ -380,17 +380,17 @@ export async function storeOnNode(targetNode: Snode, params: SendParams): Promis
export async function retrieveNextMessages( export async function retrieveNextMessages(
targetNode: Snode, targetNode: Snode,
lastHash: string, lastHash: string,
pubkey: string associatedWith: string
): Promise<Array<any>> { ): Promise<Array<any>> {
const params = { const params = {
pubKey: pubkey, pubKey: associatedWith,
lastHash: lastHash || '', lastHash: lastHash || '',
}; };
// let exceptions bubble up // let exceptions bubble up
try { try {
// no retry for this one as this a call we do every few seconds while polling for messages // no retry for this one as this a call we do every few seconds while polling for messages
const result = await snodeRpc('retrieve', params, targetNode, pubkey); const result = await snodeRpc('retrieve', params, targetNode, associatedWith);
if (!result) { if (!result) {
window?.log?.warn( window?.log?.warn(

@ -12,7 +12,7 @@ import { OnionPaths } from '../onions';
import { fromBase64ToArrayBuffer, toHex } from '../utils/String'; import { fromBase64ToArrayBuffer, toHex } from '../utils/String';
import pRetry from 'p-retry'; import pRetry from 'p-retry';
import { incrementBadPathCountOrDrop } from '../onions/onionPath'; import { incrementBadPathCountOrDrop } from '../onions/onionPath';
import _ from 'lodash';
// hold the ed25519 key of a snode against the time it fails. Used to remove a snode only after a few failures (snodeFailureThreshold failures) // hold the ed25519 key of a snode against the time it fails. Used to remove a snode only after a few failures (snodeFailureThreshold failures)
const snodeFailureCount: Record<string, number> = {}; const snodeFailureCount: Record<string, number> = {};
@ -205,7 +205,7 @@ async function process421Error(
if (!lsrpcEd25519Key || !associatedWith) { if (!lsrpcEd25519Key || !associatedWith) {
throw new Error('status 421 without a final destination or no associatedWith makes no sense'); throw new Error('status 421 without a final destination or no associatedWith makes no sense');
} }
window?.log?.info('Invalidating swarm'); window?.log?.info(`Invalidating swarm for ${associatedWith}`);
await handle421InvalidSwarm(lsrpcEd25519Key, body, associatedWith); await handle421InvalidSwarm(lsrpcEd25519Key, body, associatedWith);
} }
} }
@ -293,7 +293,9 @@ async function processAnyOtherErrorAtDestination(
// response.status === 503 || // response.status === 503 ||
// response.status === 504 || // response.status === 504 ||
// response.status === 404 || // response.status === 404 ||
status !== 200 // this is pretty strong. a 400 (Oxen server error) will be handled as a bad path. status !== 400 &&
status !== 406 && // handled in process406Error
status !== 421 // handled in process421Error
) { ) {
window?.log?.warn(`[path] Got status at destination: ${status}`); window?.log?.warn(`[path] Got status at destination: ${status}`);
@ -360,14 +362,21 @@ export async function decodeOnionResult(symmetricKey: ArrayBuffer, ciphertext: s
/** /**
* Only exported for testing purpose * Only exported for testing purpose
*/ */
export async function processOnionResponse( export async function processOnionResponse({
response: { text: () => Promise<string>; status: number }, response,
symmetricKey: ArrayBuffer, symmetricKey,
guardNode: Snode, guardNode,
lsrpcEd25519Key?: string, abortSignal,
abortSignal?: AbortSignal, associatedWith,
associatedWith?: string lsrpcEd25519Key,
): Promise<SnodeResponse> { }: {
response: { text: () => Promise<string>; status: number };
symmetricKey: ArrayBuffer;
guardNode: Snode;
lsrpcEd25519Key?: string;
abortSignal?: AbortSignal;
associatedWith?: string;
}): Promise<SnodeResponse> {
let ciphertext = ''; let ciphertext = '';
processAbortedRequest(abortSignal); processAbortedRequest(abortSignal);
@ -437,16 +446,14 @@ export async function processOnionResponse(
const status = jsonRes.status_code || jsonRes.status; const status = jsonRes.status_code || jsonRes.status;
await processOnionRequestErrorAtDestination({ await processOnionRequestErrorAtDestination({
statusCode: status, statusCode: status,
body: plaintext, body: jsonRes?.body, // this is really important.
destinationEd25519: lsrpcEd25519Key, destinationEd25519: lsrpcEd25519Key,
associatedWith, associatedWith,
}); });
return jsonRes as SnodeResponse; return jsonRes as SnodeResponse;
} catch (e) { } catch (e) {
window?.log?.error( window?.log?.error(`[path] lokiRpc::processingOnionResponse - Rethrowing error ${e.message}'`);
`[path] lokiRpc::processingOnionResponse - parse error outer json ${e.code} ${e.message} json: '${plaintext}'`
);
throw e; throw e;
} }
} }
@ -490,27 +497,36 @@ async function handle421InvalidSwarm(snodeEd25519: string, body: string, associa
window?.log?.warn('Got a 421 without an associatedWith publickey'); window?.log?.warn('Got a 421 without an associatedWith publickey');
return; return;
} }
const exceptionMessage = '421 handled. Retry this request with a new targetNode';
try { try {
const json = JSON.parse(body); const json = JSON.parse(body);
// The snode isn't associated with the given public key anymore // The snode isn't associated with the given public key anymore
if (json.snodes?.length) { if (json.snodes?.length) {
// the snode gave us the new swarm. Save it for the next retry // the snode gave us the new swarm. Save it for the next retry
window?.log?.warn('Wrong swarm, now looking at snodes', json.snodes); window?.log?.warn(
'Wrong swarm, now looking at snodes',
json.snodes.map((s: any) => s.pubkey_ed25519)
);
return updateSwarmFor(associatedWith, json.snodes); await updateSwarmFor(associatedWith, json.snodes);
throw new pRetry.AbortError(exceptionMessage);
} }
// remove this node from the swarm of this pubkey // remove this node from the swarm of this pubkey
return dropSnodeFromSwarmIfNeeded(associatedWith, snodeEd25519); await dropSnodeFromSwarmIfNeeded(associatedWith, snodeEd25519);
} catch (e) { } catch (e) {
console.warn('dropSnodeFromSwarmIfNeeded', snodeEd25519); if (e.message !== exceptionMessage) {
window?.log?.warn( console.warn('dropSnodeFromSwarmIfNeeded', snodeEd25519);
'Got error while parsing 421 result. Dropping this snode from the swarm of this pubkey', window?.log?.warn(
e 'Got error while parsing 421 result. Dropping this snode from the swarm of this pubkey',
); e
// could not parse result. Consider that this snode as invalid );
return dropSnodeFromSwarmIfNeeded(associatedWith, snodeEd25519); // could not parse result. Consider that this snode as invalid
await dropSnodeFromSwarmIfNeeded(associatedWith, snodeEd25519);
}
} }
// this is important we throw so another retry is made and we exit the handling of that reponse
throw new pRetry.AbortError(exceptionMessage);
} }
/** /**
@ -604,24 +620,23 @@ const sendOnionRequestHandlingSnodeEject = async ({
finalRelayOptions, finalRelayOptions,
abortSignal, abortSignal,
}); });
// this call will handle the common onion failure logic. // this call will handle the common onion failure logic.
// if an error is not retryable a AbortError is triggered, which is handled by pRetry and retries are stopped // if an error is not retryable a AbortError is triggered, which is handled by pRetry and retries are stopped
const processed = await processOnionResponse( const processed = await processOnionResponse({
response, response,
decodingSymmetricKey, symmetricKey: decodingSymmetricKey,
nodePath[0], guardNode: nodePath[0],
finalDestOptions?.destination_ed25519_hex, lsrpcEd25519Key: finalDestOptions?.destination_ed25519_hex,
abortSignal, abortSignal,
associatedWith associatedWith,
); });
return processed; return processed;
}; };
/** /**
* *
* Onion request looks like this * Onion requests looks like this
* Sender -> 1 -> 2 -> 3 -> Receiver * Sender -> 1 -> 2 -> 3 -> Receiver
* 1, 2, 3 = onion Snodes * 1, 2, 3 = onion Snodes
* *
@ -649,6 +664,9 @@ const sendOnionRequest = async ({
}) => { }) => {
// get destination pubkey in array buffer format // get destination pubkey in array buffer format
let destX25519hex = destX25519Any; let destX25519hex = destX25519Any;
// Warning be sure to do a copy otherwise the delete below creates issue with retries
const copyFinalDestOptions = _.cloneDeep(finalDestOptions);
if (typeof destX25519hex !== 'string') { if (typeof destX25519hex !== 'string') {
// convert AB to hex // convert AB to hex
window?.log?.warn('destX25519hex was not a string'); window?.log?.warn('destX25519hex was not a string');
@ -658,14 +676,14 @@ const sendOnionRequest = async ({
// safely build destination // safely build destination
let targetEd25519hex; let targetEd25519hex;
if (finalDestOptions.destination_ed25519_hex) { if (copyFinalDestOptions.destination_ed25519_hex) {
// snode destination // snode destination
targetEd25519hex = finalDestOptions.destination_ed25519_hex; targetEd25519hex = copyFinalDestOptions.destination_ed25519_hex;
// eslint-disable-next-line no-param-reassign // eslint-disable-next-line no-param-reassign
delete finalDestOptions.destination_ed25519_hex; delete copyFinalDestOptions.destination_ed25519_hex;
} }
const options = finalDestOptions; // lint const options = copyFinalDestOptions; // lint
// do we need this? // do we need this?
options.headers = options.headers || {}; options.headers = options.headers || {};
@ -791,7 +809,7 @@ export async function lokiOnionFetch(
return onionFetchRetryable(targetNode, body, associatedWith); return onionFetchRetryable(targetNode, body, associatedWith);
}, },
{ {
retries: 10, retries: 9,
factor: 1, factor: 1,
minTimeout: 200, minTimeout: 200,
maxTimeout: 2000, maxTimeout: 2000,

@ -102,13 +102,12 @@ describe('OnionPathsErrors', () => {
const abortController = new AbortController(); const abortController = new AbortController();
abortController.abort(); abortController.abort();
try { try {
await processOnionResponse( await processOnionResponse({
getFakeResponse(), response: getFakeResponse(),
new Uint8Array(), symmetricKey: new Uint8Array(),
guardSnode1, guardNode: guardSnode1,
undefined, abortSignal: abortController.signal,
abortController.signal });
);
throw new Error('Error expected'); throw new Error('Error expected');
} catch (e) { } catch (e) {
expect(e.message).to.equal('Request got aborted'); expect(e.message).to.equal('Request got aborted');
@ -119,7 +118,11 @@ describe('OnionPathsErrors', () => {
it('throws an non retryable error we get a 406 status code', async () => { it('throws an non retryable error we get a 406 status code', async () => {
try { try {
await processOnionResponse(getFakeResponse(406), new Uint8Array(), guardSnode1, undefined); await processOnionResponse({
response: getFakeResponse(406),
symmetricKey: new Uint8Array(),
guardNode: guardSnode1,
});
throw new Error('Error expected'); throw new Error('Error expected');
} catch (e) { } catch (e) {
expect(e.message).to.equal('You clock is out of sync with the network. Check your clock.'); expect(e.message).to.equal('You clock is out of sync with the network. Check your clock.');
@ -135,14 +138,14 @@ describe('OnionPathsErrors', () => {
const targetNode = otherNodesPubkeys[0]; const targetNode = otherNodesPubkeys[0];
try { try {
await processOnionResponse( await processOnionResponse({
getFakeResponse(421), response: getFakeResponse(421),
new Uint8Array(), symmetricKey: new Uint8Array(),
guardSnode1, guardNode: guardSnode1,
targetNode, lsrpcEd25519Key: targetNode,
undefined,
associatedWith associatedWith,
); });
throw new Error('Error expected'); throw new Error('Error expected');
} catch (e) { } catch (e) {
expect(e.message).to.equal('Bad Path handled. Retry this request. Status: 421'); expect(e.message).to.equal('Bad Path handled. Retry this request. Status: 421');
@ -165,14 +168,13 @@ describe('OnionPathsErrors', () => {
otherNodesArray[6], otherNodesArray[6],
]; ];
try { try {
await processOnionResponse( await processOnionResponse({
getFakeResponse(421, JSON.stringify({ snodes: resultExpected })), response: getFakeResponse(421, JSON.stringify({ snodes: resultExpected })),
new Uint8Array(), symmetricKey: new Uint8Array(),
guardSnode1, guardNode: guardSnode1,
targetNode, lsrpcEd25519Key: targetNode,
undefined, associatedWith,
associatedWith });
);
throw new Error('Error expected'); throw new Error('Error expected');
} catch (e) { } catch (e) {
expect(e.message).to.equal('Bad Path handled. Retry this request. Status: 421'); expect(e.message).to.equal('Bad Path handled. Retry this request. Status: 421');
@ -188,14 +190,14 @@ describe('OnionPathsErrors', () => {
const targetNode = otherNodesPubkeys[0]; const targetNode = otherNodesPubkeys[0];
try { try {
await processOnionResponse( await processOnionResponse({
getFakeResponse(421, 'THIS IS SOME INVALID JSON'), response: getFakeResponse(421, 'THIS IS SOME INVALID JSON'),
new Uint8Array(), symmetricKey: new Uint8Array(),
guardSnode1, guardNode: guardSnode1,
targetNode, lsrpcEd25519Key: targetNode,
undefined,
associatedWith associatedWith,
); });
throw new Error('Error expected'); throw new Error('Error expected');
} catch (e) { } catch (e) {
expect(e.message).to.equal('Bad Path handled. Retry this request. Status: 421'); expect(e.message).to.equal('Bad Path handled. Retry this request. Status: 421');
@ -223,14 +225,13 @@ describe('OnionPathsErrors', () => {
.resolves({ plaintext: json, ciphertextBuffer: new Uint8Array() }); .resolves({ plaintext: json, ciphertextBuffer: new Uint8Array() });
try { try {
await processOnionResponse( await processOnionResponse({
getFakeResponse(200, fromArrayBufferToBase64(Buffer.from(json))), response: getFakeResponse(200, fromArrayBufferToBase64(Buffer.from(json))),
new Uint8Array(), symmetricKey: new Uint8Array(),
guardSnode1, guardNode: guardSnode1,
targetNode, lsrpcEd25519Key: targetNode,
undefined, associatedWith,
associatedWith });
);
throw new Error('Error expected'); throw new Error('Error expected');
} catch (e) { } catch (e) {
expect(e.message).to.equal('Bad Path handled. Retry this request. Status: 421'); expect(e.message).to.equal('Bad Path handled. Retry this request. Status: 421');
@ -263,14 +264,13 @@ describe('OnionPathsErrors', () => {
.resolves({ plaintext: json, ciphertextBuffer: new Uint8Array() }); .resolves({ plaintext: json, ciphertextBuffer: new Uint8Array() });
try { try {
await processOnionResponse( await processOnionResponse({
getFakeResponse(200, json), response: getFakeResponse(200, json),
new Uint8Array(), symmetricKey: new Uint8Array(),
guardSnode1, guardNode: guardSnode1,
targetNode, lsrpcEd25519Key: targetNode,
undefined, associatedWith,
associatedWith });
);
throw new Error('Error expected'); throw new Error('Error expected');
} catch (e) { } catch (e) {
expect(e.message).to.equal('Bad Path handled. Retry this request. Status: 421'); expect(e.message).to.equal('Bad Path handled. Retry this request. Status: 421');

Loading…
Cancel
Save