Merge pull request #3133 from Bilb/update-fs-release-endpoint

fix: update fetching desktop release endpoint
pull/3138/head
Audric Ackermann 8 months ago committed by GitHub
commit eb463a49ba
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -97,7 +97,7 @@
"fs-extra": "9.0.0", "fs-extra": "9.0.0",
"glob": "7.1.2", "glob": "7.1.2",
"image-type": "^4.1.0", "image-type": "^4.1.0",
"libsession_util_nodejs": "https://github.com/oxen-io/libsession-util-nodejs/releases/download/v0.3.20/libsession_util_nodejs-v0.3.20.tar.gz", "libsession_util_nodejs": "https://github.com/oxen-io/libsession-util-nodejs/releases/download/v0.3.21/libsession_util_nodejs-v0.3.21.tar.gz",
"libsodium-wrappers-sumo": "^0.7.9", "libsodium-wrappers-sumo": "^0.7.9",
"linkify-it": "^4.0.1", "linkify-it": "^4.0.1",
"lodash": "^4.17.21", "lodash": "^4.17.21",

@ -1,7 +1,6 @@
import { ipcRenderer } from 'electron'; import { debounce } from 'lodash';
import { useEffect, useState } from 'react'; import { useEffect, useState } from 'react';
import { debounce, isEmpty, isString } from 'lodash';
import { useDispatch, useSelector } from 'react-redux'; import { useDispatch, useSelector } from 'react-redux';
import useInterval from 'react-use/lib/useInterval'; import useInterval from 'react-use/lib/useInterval';
import useTimeoutFn from 'react-use/lib/useTimeoutFn'; import useTimeoutFn from 'react-use/lib/useTimeoutFn';
@ -37,7 +36,7 @@ import { SessionIconButton } from '../icon/SessionIconButton';
import { LeftPaneSectionContainer } from './LeftPaneSectionContainer'; import { LeftPaneSectionContainer } from './LeftPaneSectionContainer';
import { SettingsKey } from '../../data/settings-key'; import { SettingsKey } from '../../data/settings-key';
import { getLatestReleaseFromFileServer } from '../../session/apis/file_server_api/FileServerApi'; import { useFetchLatestReleaseFromFileServer } from '../../hooks/useFetchLatestReleaseFromFileServer';
import { import {
forceRefreshRandomSnodePool, forceRefreshRandomSnodePool,
getFreshSwarmFor, getFreshSwarmFor,
@ -143,11 +142,6 @@ const Section = (props: { type: SectionType }) => {
const cleanUpMediasInterval = DURATION.MINUTES * 60; const cleanUpMediasInterval = DURATION.MINUTES * 60;
// every 1 minute we fetch from the fileserver to check for a new release
// * if there is none, no request to github are made.
// * if there is a version on the fileserver more recent than our current, we fetch github to get the UpdateInfos and trigger an update as usual (asking user via dialog)
const fetchReleaseFromFileServerInterval = 1000 * 60; // try to fetch the latest release from the fileserver every minute
// Do this only if we created a new account id, or if we already received the initial configuration message // Do this only if we created a new account id, or if we already received the initial configuration message
const triggerSyncIfNeeded = async () => { const triggerSyncIfNeeded = async () => {
const us = UserUtils.getOurPubKeyStrFromCache(); const us = UserUtils.getOurPubKeyStrFromCache();
@ -207,22 +201,6 @@ const doAppStartUp = async () => {
}, 20000); }, 20000);
}; };
async function fetchReleaseFromFSAndUpdateMain() {
try {
window.log.info('[updater] about to fetchReleaseFromFSAndUpdateMain');
const latest = await getLatestReleaseFromFileServer();
window.log.info('[updater] fetched latest release from fileserver: ', latest);
if (isString(latest) && !isEmpty(latest)) {
ipcRenderer.send('set-release-from-file-server', latest);
window.readyForUpdates();
}
} catch (e) {
window.log.warn(e);
}
}
/** /**
* ActionsPanel is the far left banner (not the left pane). * ActionsPanel is the far left banner (not the left pane).
* The panel with buttons to switch between the message/contact/settings/theme views * The panel with buttons to switch between the message/contact/settings/theme views
@ -247,12 +225,7 @@ export const ActionsPanel = () => {
useInterval(cleanUpOldDecryptedMedias, startCleanUpMedia ? cleanUpMediasInterval : null); useInterval(cleanUpOldDecryptedMedias, startCleanUpMedia ? cleanUpMediasInterval : null);
useInterval(() => { useFetchLatestReleaseFromFileServer();
if (!ourPrimaryConversation) {
return;
}
void fetchReleaseFromFSAndUpdateMain();
}, fetchReleaseFromFileServerInterval);
useInterval(() => { useInterval(() => {
if (!ourPrimaryConversation) { if (!ourPrimaryConversation) {

@ -0,0 +1,20 @@
import { isEmpty } from 'lodash';
import { useSelector } from 'react-redux';
import useInterval from 'react-use/lib/useInterval';
import { fetchLatestRelease } from '../session/fetch_latest_release';
import { UserUtils } from '../session/utils';
import { getOurPrimaryConversation } from '../state/selectors/conversations';
export function useFetchLatestReleaseFromFileServer() {
const ourPrimaryConversation = useSelector(getOurPrimaryConversation);
useInterval(async () => {
if (!ourPrimaryConversation) {
return;
}
const userEd25519SecretKey = (await UserUtils.getUserED25519KeyPairBytes())?.privKeyBytes;
if (userEd25519SecretKey && !isEmpty(userEd25519SecretKey)) {
void fetchLatestRelease.fetchReleaseFromFSAndUpdateMain(userEd25519SecretKey);
}
}, fetchLatestRelease.fetchReleaseFromFileServerInterval);
}

@ -1 +0,0 @@
declare module 'ip2country';

@ -1,9 +1,12 @@
import AbortController from 'abort-controller'; import AbortController from 'abort-controller';
import { BlindingActions } from '../../../webworker/workers/browser/libsession_worker_interface';
import { OnionSending, OnionV4JSONSnodeResponse } from '../../onions/onionSend'; import { OnionSending, OnionV4JSONSnodeResponse } from '../../onions/onionSend';
import { import {
batchGlobalIsSuccess, batchGlobalIsSuccess,
parseBatchGlobalStatusCode, parseBatchGlobalStatusCode,
} from '../open_group_api/sogsv3/sogsV3BatchPoll'; } from '../open_group_api/sogsv3/sogsV3BatchPoll';
import { GetNetworkTime } from '../snode_api/getNetworkTime';
import { fromUInt8ArrayToBase64 } from '../../utils/String';
export const fileServerHost = 'filev2.getsession.org'; export const fileServerHost = 'filev2.getsession.org';
export const fileServerURL = `http://${fileServerHost}`; export const fileServerURL = `http://${fileServerHost}`;
@ -123,12 +126,27 @@ const parseStatusCodeFromOnionRequestV4 = (
* Fetch the latest desktop release available on github from the fileserver. * Fetch the latest desktop release available on github from the fileserver.
* This call is onion routed and so do not expose our ip to github nor the file server. * This call is onion routed and so do not expose our ip to github nor the file server.
*/ */
export const getLatestReleaseFromFileServer = async (): Promise<string | null> => { export const getLatestReleaseFromFileServer = async (
userEd25519SecretKey: Uint8Array
): Promise<string | null> => {
const sigTimestampSeconds = GetNetworkTime.getNowWithNetworkOffsetSeconds();
const blindedPkHex = await BlindingActions.blindVersionPubkey({
ed25519SecretKey: userEd25519SecretKey,
});
const signature = await BlindingActions.blindVersionSign({
ed25519SecretKey: userEd25519SecretKey,
sigTimestampSeconds,
});
const body = {
'X-FS-Pubkey': blindedPkHex,
'X-FS-Timestamp': `${sigTimestampSeconds}`,
'X-FS-Signature': fromUInt8ArrayToBase64(signature),
};
const result = await OnionSending.sendJsonViaOnionV4ToFileServer({ const result = await OnionSending.sendJsonViaOnionV4ToFileServer({
abortSignal: new AbortController().signal, abortSignal: new AbortController().signal,
endpoint: RELEASE_VERSION_ENDPOINT, endpoint: RELEASE_VERSION_ENDPOINT,
method: 'GET', method: 'GET',
stringifiedBody: null, stringifiedBody: JSON.stringify(body),
}); });
if (!batchGlobalIsSuccess(result) || parseStatusCodeFromOnionRequestV4(result) !== 200) { if (!batchGlobalIsSuccess(result) || parseStatusCodeFromOnionRequestV4(result) !== 200) {

@ -70,9 +70,16 @@ function getNowWithNetworkOffset() {
return Date.now() - GetNetworkTime.getLatestTimestampOffset(); return Date.now() - GetNetworkTime.getLatestTimestampOffset();
} }
function getNowWithNetworkOffsetSeconds() {
// make sure to call exports here, as we stub the exported one for testing.
return Math.floor(GetNetworkTime.getNowWithNetworkOffset() / 1000);
}
export const GetNetworkTime = { export const GetNetworkTime = {
getNetworkTime, getNetworkTime,
handleTimestampOffsetFromNetwork, handleTimestampOffsetFromNetwork,
getNowWithNetworkOffsetSeconds,
getLatestTimestampOffset, getLatestTimestampOffset,
getNowWithNetworkOffset, getNowWithNetworkOffset,
}; };

@ -93,6 +93,8 @@ export const getDecryptedMediaUrl = async (
// we consider the file is encrypted. // we consider the file is encrypted.
// if it's not, the hook caller has to fallback to setting the img src as an url to the file instead and load it // if it's not, the hook caller has to fallback to setting the img src as an url to the file instead and load it
if (urlToDecryptedBlobMap.has(url)) { if (urlToDecryptedBlobMap.has(url)) {
// typescript does not realize that the `has()` above makes sure the `get()` is not undefined
// refresh the last access timestamp so we keep the one being currently in use // refresh the last access timestamp so we keep the one being currently in use
const existing = urlToDecryptedBlobMap.get(url); const existing = urlToDecryptedBlobMap.get(url);
const existingObjUrl = existing?.decrypted as string; const existingObjUrl = existing?.decrypted as string;
@ -102,7 +104,6 @@ export const getDecryptedMediaUrl = async (
lastAccessTimestamp: Date.now(), lastAccessTimestamp: Date.now(),
forceRetain: existing?.forceRetain || false, forceRetain: existing?.forceRetain || false,
}); });
// typescript does not realize that the has above makes sure the get is not undefined
return existingObjUrl; return existingObjUrl;
} }

@ -0,0 +1,53 @@
// every 1 minute we fetch from the fileserver to check for a new release
// * if there is none, no request to github are made.
// * if there is a version on the fileserver more recent than our current, we fetch github to get the UpdateInfos and trigger an update as usual (asking user via dialog)
import { isEmpty, isString } from 'lodash';
import { ipcRenderer } from 'electron';
import { DURATION } from '../constants';
import { getLatestReleaseFromFileServer } from '../apis/file_server_api/FileServerApi';
/**
* We don't want to hit the fileserver too often. Only often on start, and then every 30 minutes
*/
const skipIfLessThan = DURATION.MINUTES * 30;
let lastFetchedTimestamp = Number.MIN_SAFE_INTEGER;
function resetForTesting() {
lastFetchedTimestamp = Number.MIN_SAFE_INTEGER;
}
async function fetchReleaseFromFSAndUpdateMain(userEd25519SecretKey: Uint8Array) {
try {
window.log.info('[updater] about to fetchReleaseFromFSAndUpdateMain');
const diff = Date.now() - lastFetchedTimestamp;
if (diff < skipIfLessThan) {
window.log.info(
`[updater] fetched release from fs ${Math.floor(diff / DURATION.MINUTES)}minutes ago, skipping until that's at least ${Math.floor(skipIfLessThan / DURATION.MINUTES)}`
);
return;
}
const justFetched = await getLatestReleaseFromFileServer(userEd25519SecretKey);
window.log.info('[updater] fetched latest release from fileserver: ', justFetched);
if (isString(justFetched) && !isEmpty(justFetched)) {
lastFetchedTimestamp = Date.now();
ipcRenderer.send('set-release-from-file-server', justFetched);
window.readyForUpdates();
}
} catch (e) {
window.log.warn(e);
}
}
export const fetchLatestRelease = {
/**
* Try to fetch the latest release from the fileserver every 1 minute.
* If we did fetch a release already in the last 30 minutes, we will skip the call.
*/
fetchReleaseFromFileServerInterval: DURATION.MINUTES * 1,
fetchReleaseFromFSAndUpdateMain,
resetForTesting,
};

@ -30,9 +30,11 @@ type UserGroupsConfigFunctions =
type ConvoInfoVolatileConfigFunctions = type ConvoInfoVolatileConfigFunctions =
| [ConvoInfoVolatileConfig, ...BaseConfigActions] | [ConvoInfoVolatileConfig, ...BaseConfigActions]
| [ConvoInfoVolatileConfig, ...ConvoInfoVolatileConfigActionsType]; | [ConvoInfoVolatileConfig, ...ConvoInfoVolatileConfigActionsType];
type BlindingFunctions = ['Blinding', ...BlindingFunctions];
export type LibSessionWorkerFunctions = export type LibSessionWorkerFunctions =
| UserConfigFunctions | UserConfigFunctions
| ContactsConfigFunctions | ContactsConfigFunctions
| UserGroupsConfigFunctions | UserGroupsConfigFunctions
| ConvoInfoVolatileConfigFunctions; | ConvoInfoVolatileConfigFunctions
| BlindingFunctions;

@ -2,6 +2,7 @@
/* eslint-disable import/no-unresolved */ /* eslint-disable import/no-unresolved */
import { import {
BaseWrapperActionsCalls, BaseWrapperActionsCalls,
BlindingActionsCalls,
ContactInfoSet, ContactInfoSet,
ContactsWrapperActionsCalls, ContactsWrapperActionsCalls,
ConvoInfoVolatileWrapperActionsCalls, ConvoInfoVolatileWrapperActionsCalls,
@ -343,6 +344,17 @@ export const ConvoInfoVolatileWrapperActions: ConvoInfoVolatileWrapperActionsCal
]) as Promise<ReturnType<ConvoInfoVolatileWrapperActionsCalls['eraseCommunityByFullUrl']>>, ]) as Promise<ReturnType<ConvoInfoVolatileWrapperActionsCalls['eraseCommunityByFullUrl']>>,
}; };
export const BlindingActions: BlindingActionsCalls = {
blindVersionPubkey: async (opts: { ed25519SecretKey: Uint8Array }) =>
callLibSessionWorker(['Blinding', 'blindVersionPubkey', opts]) as Promise<
ReturnType<BlindingActionsCalls['blindVersionPubkey']>
>,
blindVersionSign: async (opts: { ed25519SecretKey: Uint8Array; sigTimestampSeconds: number }) =>
callLibSessionWorker(['Blinding', 'blindVersionSign', opts]) as Promise<
ReturnType<BlindingActionsCalls['blindVersionSign']>
>,
};
export const callLibSessionWorker = async ( export const callLibSessionWorker = async (
callToMake: LibSessionWorkerFunctions callToMake: LibSessionWorkerFunctions
): Promise<unknown> => { ): Promise<unknown> => {

@ -2,6 +2,7 @@
/* eslint-disable no-case-declarations */ /* eslint-disable no-case-declarations */
import { import {
BaseConfigWrapperNode, BaseConfigWrapperNode,
BlindingWrapperNode,
ContactsConfigWrapperNode, ContactsConfigWrapperNode,
ConvoInfoVolatileWrapperNode, ConvoInfoVolatileWrapperNode,
UserConfigWrapperNode, UserConfigWrapperNode,
@ -55,6 +56,7 @@ function getCorrespondingWrapper(wrapperType: ConfigWrapperObjectTypes): BaseCon
throw new Error(`${wrapperType} is not init yet`); throw new Error(`${wrapperType} is not init yet`);
} }
return wrapper; return wrapper;
default: default:
assertUnreachable( assertUnreachable(
wrapperType, wrapperType,
@ -149,32 +151,53 @@ function freeUserWrapper(wrapperType: ConfigWrapperObjectTypes) {
); );
} }
} }
onmessage = async (e: {
onmessage = async (e: { data: [number, ConfigWrapperObjectTypes, string, ...any] }) => { data: [number, ConfigWrapperObjectTypes | 'Blinding', string, ...any];
}) => {
const [jobId, config, action, ...args] = e.data; const [jobId, config, action, ...args] = e.data;
try { try {
if (action === 'init') { if (action === 'init') {
initUserWrapper(args, config); if (config === 'Blinding') {
// nothing to do for the blinding wrapper, all functions are static
} else {
initUserWrapper(args, config);
}
postMessage([jobId, null, null]); postMessage([jobId, null, null]);
return; return;
} }
if (action === 'free') { if (action === 'free') {
freeUserWrapper(config); if (config !== 'Blinding') {
freeUserWrapper(config);
}
postMessage([jobId, null, null]); postMessage([jobId, null, null]);
return; return;
} }
const wrapper = getCorrespondingWrapper(config); let result: any;
const fn = (wrapper as any)[action];
if (!fn) { if (config === 'Blinding') {
throw new Error( const fn = (BlindingWrapperNode as any)[action];
`Worker: job "${jobId}" did not find function "${action}" on config "${config}"`
); if (!fn) {
throw new Error(
`Worker: job "${jobId}" did not find function "${action}" on wrapper "${config}"`
);
}
result = await (BlindingWrapperNode as any)[action](...args);
} else {
const wrapper = getCorrespondingWrapper(config);
const fn = (wrapper as any)[action];
if (!fn) {
throw new Error(
`Worker: job "${jobId}" did not find function "${action}" on config "${config}"`
);
}
result = await (wrapper as any)[action](...args);
} }
const result = await (wrapper as any)[action](...args);
postMessage([jobId, null, result]); postMessage([jobId, null, result]);
} catch (error) { } catch (error) {
const errorForDisplay = prepareErrorForPostMessage(error); const errorForDisplay = prepareErrorForPostMessage(error);

@ -4928,9 +4928,9 @@ levn@~0.3.0:
prelude-ls "~1.1.2" prelude-ls "~1.1.2"
type-check "~0.3.2" type-check "~0.3.2"
"libsession_util_nodejs@https://github.com/oxen-io/libsession-util-nodejs/releases/download/v0.3.20/libsession_util_nodejs-v0.3.20.tar.gz": "libsession_util_nodejs@https://github.com/oxen-io/libsession-util-nodejs/releases/download/v0.3.21/libsession_util_nodejs-v0.3.21.tar.gz":
version "0.3.20" version "0.3.21"
resolved "https://github.com/oxen-io/libsession-util-nodejs/releases/download/v0.3.20/libsession_util_nodejs-v0.3.20.tar.gz#f3ae2259c54bf414d5003ea4a9e0a38b3c974065" resolved "https://github.com/oxen-io/libsession-util-nodejs/releases/download/v0.3.21/libsession_util_nodejs-v0.3.21.tar.gz#64705b1f7c934ca32f929ea8127370cc82bab97a"
dependencies: dependencies:
cmake-js "^7.2.1" cmake-js "^7.2.1"
node-addon-api "^6.1.0" node-addon-api "^6.1.0"

Loading…
Cancel
Save