Merge remote-tracking branch 'origin/clearnet' into unstable

pull/3138/head
Audric Ackermann 8 months ago
commit 45eb547299

@ -20,7 +20,7 @@ jobs:
strategy:
fail-fast: false
matrix:
os: [windows-2022, macos-11, ubuntu-20.04]
os: [windows-2022, macos-12, ubuntu-20.04]
env:
SIGNAL_ENV: production
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}

@ -18,7 +18,7 @@ jobs:
strategy:
fail-fast: false
matrix:
os: [windows-2022, macos-11, ubuntu-20.04]
os: [windows-2022, macos-12, ubuntu-20.04]
env:
SIGNAL_ENV: production
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}

@ -12,7 +12,7 @@ jobs:
strategy:
fail-fast: false
matrix:
os: [windows-2022, macos-11, ubuntu-20.04]
os: [windows-2022, macos-12, ubuntu-20.04]
env:
SIGNAL_ENV: production
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}

@ -14,7 +14,7 @@ Please search for any [existing issues](https://github.com/oxen-io/session-deskt
## Supported platforms
Session requires Windows 10 or later, macOS Catalina (10.15) or later, or a linux distribution with glibc 2.28 or later like Debian 10 or Ubuntu 20.04.
Session requires Windows 10 or later, macOS Monterey (12) or later, or a linux distribution with glibc 2.28 or later like Debian 10 or Ubuntu 20.04.
## Build instruction

@ -2,7 +2,7 @@
"name": "session-desktop",
"productName": "Session",
"description": "Private messaging from your desktop",
"version": "1.12.4",
"version": "1.12.5",
"license": "GPL-3.0",
"author": {
"name": "Oxen Labs",

@ -47,6 +47,7 @@ export interface Snode {
port: number;
pubkey_x25519: string;
pubkey_ed25519: string;
storage_server_version: Array<number>;
}
export type SwarmNode = Snode & {

@ -17,6 +17,7 @@ import {
CLOSED_GROUP_V2_KEY_PAIRS_TABLE,
CONVERSATIONS_TABLE,
GUARD_NODE_TABLE,
ITEMS_TABLE,
LAST_HASHES_TABLE,
MESSAGES_TABLE,
NODES_FOR_PUBKEY_TABLE,
@ -26,7 +27,7 @@ import {
rebuildFtsTable,
} from '../database_utility';
import { SettingsKey } from '../../data/settings-key';
import { SettingsKey, SNODE_POOL_ITEM_ID } from '../../data/settings-key';
import { sleepFor } from '../../session/utils/Promise';
import { sqlNode } from '../sql';
import MIGRATION_HELPERS from './helpers';
@ -105,6 +106,7 @@ const LOKI_SCHEMA_VERSIONS = [
updateToSessionSchemaVersion34,
updateToSessionSchemaVersion35,
updateToSessionSchemaVersion36,
updateToSessionSchemaVersion37,
];
function updateToSessionSchemaVersion1(currentVersion: number, db: BetterSqlite3.Database) {
@ -1949,6 +1951,25 @@ function updateToSessionSchemaVersion36(currentVersion: number, db: BetterSqlite
console.log(`updateToSessionSchemaVersion${targetVersion}: success!`);
}
function updateToSessionSchemaVersion37(currentVersion: number, db: BetterSqlite3.Database) {
const targetVersion = 37;
if (currentVersion >= targetVersion) {
return;
}
console.log(`updateToSessionSchemaVersion${targetVersion}: starting...`);
db.transaction(() => {
console.info(`clearing ${SNODE_POOL_ITEM_ID} cache`);
db.prepare(`DELETE FROM ${ITEMS_TABLE} WHERE id = $snodePoolId;`).run({
snodePoolId: SNODE_POOL_ITEM_ID,
});
writeSessionSchemaVersion(targetVersion, db);
})();
console.log(`updateToSessionSchemaVersion${targetVersion}: success!`);
}
export function printTableColumns(table: string, db: BetterSqlite3.Database) {
console.info(db.pragma(`table_info('${table}');`));
}

@ -35,6 +35,7 @@ export async function fetchSnodePoolFromSeedNodeWithRetries(
port: snode.storage_port,
pubkey_x25519: snode.pubkey_x25519,
pubkey_ed25519: snode.pubkey_ed25519,
storage_server_version: snode.storage_server_version,
}));
window?.log?.info(
'SeedNodeAPI::fetchSnodePoolFromSeedNodeWithRetries - Refreshed random snode pool with',
@ -140,6 +141,7 @@ export interface SnodeFromSeed {
storage_port: number;
pubkey_x25519: string;
pubkey_ed25519: string;
storage_server_version: Array<number>;
}
const getSnodeListFromSeednodeOneAtAtime = async (seedNodes: Array<string>) =>
@ -241,6 +243,7 @@ async function getSnodesFromSeedUrl(urlObj: URL): Promise<Array<any>> {
storage_port: true,
pubkey_x25519: true,
pubkey_ed25519: true,
storage_server_version: true,
},
},
};

@ -79,6 +79,7 @@ type FetchSnodeListParams = {
storage_port: true;
pubkey_x25519: true;
pubkey_ed25519: true;
storage_server_version: true;
};
};

@ -18,6 +18,7 @@ function buildSnodeListRequests(): Array<GetServiceNodesSubRequest> {
storage_port: true,
pubkey_x25519: true,
pubkey_ed25519: true,
storage_server_version: true,
},
},
},
@ -49,14 +50,15 @@ async function getSnodePoolFromSnode(targetNode: Snode): Promise<Array<Snode>> {
}
// Filter 0.0.0.0 nodes which haven't submitted uptime proofs
const snodes = json.result.service_node_states
const snodes: Array<Snode> = json.result.service_node_states
.filter((snode: any) => snode.public_ip !== '0.0.0.0')
.map((snode: any) => ({
ip: snode.public_ip,
port: snode.storage_port,
pubkey_x25519: snode.pubkey_x25519,
pubkey_ed25519: snode.pubkey_ed25519,
})) as Array<Snode>;
storage_server_version: snode.storage_server_version,
}));
GetNetworkTime.handleTimestampOffsetFromNetwork('get_service_nodes', json.t);
// we the return list by the snode is already made of uniq snodes

@ -1,10 +1,11 @@
/* eslint-disable import/no-mutable-exports */
/* eslint-disable no-await-in-loop */
import _, { compact } from 'lodash';
import _, { compact, isFinite, isNumber, sample } from 'lodash';
import pRetry from 'p-retry';
// eslint-disable-next-line import/no-named-default
import { default as insecureNodeFetch } from 'node-fetch';
import semver from 'semver';
import { Data, Snode } from '../../data/data';
import * as SnodePool from '../apis/snode_api/snodePool';
import { UserUtils } from '../utils';
@ -15,11 +16,16 @@ import { ERROR_CODE_NO_CONNECT } from '../apis/snode_api/SNodeAPI';
import { OnionPaths } from '.';
import { APPLICATION_JSON } from '../../types/MIME';
import { ed25519Str } from '../utils/String';
import { DURATION } from '../constants';
const desiredGuardCount = 3;
const minimumGuardCount = 2;
const ONION_REQUEST_HOPS = 3;
export function getOnionPathMinTimeout() {
return DURATION.SECONDS;
}
export let onionPaths: Array<Array<Snode>> = [];
/**
@ -498,16 +504,27 @@ async function buildNewOnionPathsWorker() {
for (let i = 0; i < maxPath; i += 1) {
const path = [guards[i]];
for (let j = 0; j < nodesNeededPerPaths; j += 1) {
const randomWinner = _.sample(otherNodes);
if (!randomWinner) {
throw new Error('randomWinner unset during path building task');
do {
// selection of the last snode (edge snode) needs at least v2.8.0
if (path.length === nodesNeededPerPaths) {
const randomEdgeSnode = getRandomEdgeSnode(otherNodes);
otherNodes = otherNodes.filter(n => {
return n.pubkey_ed25519 !== randomEdgeSnode?.pubkey_ed25519;
});
path.push(randomEdgeSnode);
} else {
const snode = sample(otherNodes);
if (!snode) {
throw new Error('no more snode found for path building');
}
otherNodes = otherNodes.filter(n => {
return n.pubkey_ed25519 !== snode?.pubkey_ed25519;
});
path.push(snode);
}
otherNodes = otherNodes.filter(n => {
return n.pubkey_ed25519 !== randomWinner?.pubkey_ed25519;
});
path.push(randomWinner);
}
} while (path.length <= nodesNeededPerPaths);
onionPaths.push(path);
}
@ -516,7 +533,7 @@ async function buildNewOnionPathsWorker() {
{
retries: 3, // 4 total
factor: 1,
minTimeout: 1000,
minTimeout: OnionPaths.getOnionPathMinTimeout(),
onFailedAttempt: e => {
window?.log?.warn(
`buildNewOnionPathsWorker attempt #${e.attemptNumber} failed. ${e.retriesLeft} retries left... Error: ${e.message}`
@ -525,3 +542,33 @@ async function buildNewOnionPathsWorker() {
}
);
}
export function getRandomEdgeSnode(snodes: Array<Snode>) {
const allSnodesWithv280 = snodes.filter(snode => {
const snodeStorageVersion = snode.storage_server_version;
if (
!snodeStorageVersion ||
!Array.isArray(snodeStorageVersion) ||
snodeStorageVersion.length !== 3 ||
snodeStorageVersion.some(m => !isNumber(m) || !isFinite(m))
) {
return false;
}
const storageVersionAsString = `${snodeStorageVersion[0]}.${snodeStorageVersion[1]}.${snodeStorageVersion[2]}`;
const verifiedStorageVersion = semver.valid(storageVersionAsString);
if (!verifiedStorageVersion) {
return false;
}
if (semver.lt(verifiedStorageVersion, '2.8.0')) {
return false;
}
return true;
});
const randomEdgeSnode = sample(allSnodesWithv280);
if (!randomEdgeSnode) {
throw new Error('did not find a single snode which can be the edge');
}
return randomEdgeSnode;
}

@ -16,6 +16,7 @@ import {
} from '../../../test-utils/utils';
import { SeedNodeAPI } from '../../../../session/apis/seed_node_api';
import { ServiceNodesList } from '../../../../session/apis/snode_api/getServiceNodesList';
import { TEST_resetState } from '../../../../session/apis/snode_api/snodePool';
chai.use(chaiAsPromised as any);
chai.should();
@ -110,4 +111,137 @@ describe('OnionPaths', () => {
});
});
});
describe('getRandomEdgeSnode', () => {
it('find single valid snode in poll of many non valid snodes', () => {
const originalSnodePool = generateFakeSnodes(20);
const firstValidSnodePool = originalSnodePool.map((m, i) => {
if (i > 0) {
return {
...m,
storage_server_version: [2, 7, 0],
};
}
return m;
});
expect(OnionPaths.getRandomEdgeSnode(firstValidSnodePool)).to.be.deep.eq(
originalSnodePool[0]
);
const lastValidSnodePool = originalSnodePool.map((m, i) => {
if (i !== originalSnodePool.length - 1) {
return {
...m,
storage_server_version: [2, 7, 0],
};
}
return m;
});
expect(OnionPaths.getRandomEdgeSnode(lastValidSnodePool)).to.be.deep.eq(
originalSnodePool[originalSnodePool.length - 1]
);
});
it('random if multiple matches', () => {
const originalSnodePool = generateFakeSnodes(5);
const multipleMatchesSnodePool = originalSnodePool.map((m, i) => {
if (i % 5 === 0) {
return {
...m,
storage_server_version: [2, 7, 0],
};
}
return m;
});
const filtered = originalSnodePool.filter((_m, i) => i % 5 !== 0);
const winner = OnionPaths.getRandomEdgeSnode(multipleMatchesSnodePool);
expect(filtered).to.deep.include(winner);
});
it('throws if we run out of snodes with valid version', () => {
const originalSnodePool = generateFakeSnodes(5);
const multipleMatchesSnodePool = originalSnodePool.map(m => {
return {
...m,
storage_server_version: [2, 7, 0],
};
});
expect(() => {
OnionPaths.getRandomEdgeSnode(multipleMatchesSnodePool);
}).to.throw();
});
});
describe('pick edge snode with at least storage server v2.8.0', () => {
let fetchSnodePoolFromSeedNodeWithRetries: Sinon.SinonStub;
beforeEach(async () => {
// Utils Stubs
Sinon.stub(OnionPaths, 'selectGuardNodes').resolves(fakeGuardNodes);
Sinon.stub(ServiceNodesList, 'getSnodePoolFromSnode').resolves(fakeGuardNodes);
// we can consider that nothing is in the DB for those tests
stubData('getSnodePoolFromDb').resolves([]);
TestUtils.stubData('getGuardNodes').resolves(fakeGuardNodesFromDB);
TestUtils.stubData('createOrUpdateItem').resolves();
TestUtils.stubWindow('getSeedNodeList', () => ['seednode1']);
TestUtils.stubWindowLog();
TEST_resetState();
fetchSnodePoolFromSeedNodeWithRetries = Sinon.stub(
SeedNodeAPI,
'fetchSnodePoolFromSeedNodeWithRetries'
);
SNodeAPI.Onions.resetSnodeFailureCount();
OnionPaths.resetPathFailureCount();
OnionPaths.clearTestOnionPath();
Sinon.stub(OnionPaths, 'getOnionPathMinTimeout').returns(10);
});
afterEach(() => {
Sinon.restore();
});
it('builds a path correctly if no issues with input', async () => {
fetchSnodePoolFromSeedNodeWithRetries.resolves(generateFakeSnodes(20));
const newOnionPath = await OnionPaths.getOnionPath({});
expect(newOnionPath.length).to.eq(3);
});
it('throws if we cannot find a valid edge snode', async () => {
const badPool = generateFakeSnodes(20).map(m => {
return { ...m, storage_server_version: [2, 1, 1] };
});
fetchSnodePoolFromSeedNodeWithRetries.reset();
fetchSnodePoolFromSeedNodeWithRetries.resolves(badPool);
if (OnionPaths.TEST_getTestOnionPath().length) {
throw new Error('expected this to be empty');
}
try {
await OnionPaths.getOnionPath({});
throw new Error('fake error');
} catch (e) {
expect(e.message).to.not.be.eq('fake error');
}
});
it('rebuild a bunch of paths and check that last snode is always >=2.8.0', async () => {
for (let index = 0; index < 1000; index++) {
// build 20 times a path and make sure that the edge snode is always with at least version 2.8.0, when half of the snodes are not upgraded
const pool = generateFakeSnodes(20).map((m, i) => {
return i % 2 === 0 ? { ...m, storage_server_version: [2, 1, 1] } : m;
});
fetchSnodePoolFromSeedNodeWithRetries.resolves(pool);
const newOnionPath = await OnionPaths.getOnionPath({});
expect(newOnionPath.length).to.eq(3);
expect(newOnionPath[2].storage_server_version).to.deep.eq([2, 8, 0]);
}
});
});
});

@ -35,6 +35,7 @@ const fakeSnodePoolFromSeedNode: Array<SnodeFromSeed> = fakeSnodePool.map(m => {
storage_port: m.port,
pubkey_x25519: m.pubkey_x25519,
pubkey_ed25519: m.pubkey_ed25519,
storage_server_version: m.storage_server_version,
};
});

@ -45,19 +45,25 @@ export function generateFakePubKeys(amount: number): Array<PubKey> {
export function generateFakeSnode(): Snode {
return {
ip: `136.243.${Math.random() * 255}.${Math.random() * 255}`,
ip: `${ipv4Section()}.${ipv4Section()}.${ipv4Section()}.${ipv4Section()}`,
port: 22116,
pubkey_x25519: generateFakePubKeyStr(),
pubkey_ed25519: generateFakePubKeyStr(),
storage_server_version: [2, 8, 0],
};
}
function ipv4Section() {
return Math.floor(Math.random() * 255);
}
export function generateFakeSnodeWithEdKey(ed25519Pubkey: string): Snode {
return {
ip: `136.243.${Math.random() * 255}.${Math.random() * 255}`,
ip: `${ipv4Section()}.${ipv4Section()}.${ipv4Section()}.${ipv4Section()}`,
port: 22116,
pubkey_x25519: generateFakePubKeyStr(),
pubkey_ed25519: ed25519Pubkey,
storage_server_version: [2, 8, 0],
};
}

Loading…
Cancel
Save