From 0c70aa628bdeeda952a5dbcea97b463d56a49bc2 Mon Sep 17 00:00:00 2001 From: larabr Date: Fri, 12 Apr 2024 10:13:36 +0200 Subject: [PATCH] Add `Argon2S2K.reloadWasmModule()` for manually triggering memory deallocation (#14) Also, make `ARGON2_WASM_MEMORY_THRESHOLD_RELOAD` a static class property, to be able to change its value. --- .eslintrc.js | 2 +- openpgp.d.ts | 2 ++ src/type/s2k/argon2.js | 25 ++++++++++++++++++++----- test/benchmarks/memory_usage.js | 14 ++++++++++++++ 4 files changed, 37 insertions(+), 6 deletions(-) diff --git a/.eslintrc.js b/.eslintrc.js index 6a58bb27a..7e0698b6d 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -1,7 +1,7 @@ module.exports = { 'extends': 'airbnb-base', 'parserOptions': { - 'ecmaVersion': 11, + 'ecmaVersion': 2022, 'sourceType': 'module' }, diff --git a/openpgp.d.ts b/openpgp.d.ts index b85a652e1..fc4a83cec 100644 --- a/openpgp.d.ts +++ b/openpgp.d.ts @@ -925,6 +925,8 @@ export namespace enums { } export declare class Argon2S2K { + static reloadWasmModule(): void; + static ARGON2_WASM_MEMORY_THRESHOLD_RELOAD: number; constructor(config: Config); salt: Uint8Array; /** @throws Argon2OutOfMemoryError */ diff --git a/src/type/s2k/argon2.js b/src/type/s2k/argon2.js index f79e3ef6f..bdda332d4 100644 --- a/src/type/s2k/argon2.js +++ b/src/type/s2k/argon2.js @@ -23,9 +23,26 @@ export class Argon2OutOfMemoryError extends Error { let loadArgonWasmModule; let argon2Promise; // reload wasm module above this treshold, to deallocated used memory -const ARGON2_WASM_MEMORY_THRESHOLD_RELOAD = 2 << 19; +// (cannot be declared as a simple `static` field as its not supported by Safari 14) +let ARGON2_WASM_MEMORY_THRESHOLD_RELOAD = 2 << 19; class Argon2S2K { + static get ARGON2_WASM_MEMORY_THRESHOLD_RELOAD() { + return ARGON2_WASM_MEMORY_THRESHOLD_RELOAD; + } + + static set ARGON2_WASM_MEMORY_THRESHOLD_RELOAD(memoryThreshold) { + ARGON2_WASM_MEMORY_THRESHOLD_RELOAD = memoryThreshold; + } + + static reloadWasmModule() { + if (!loadArgonWasmModule) return; + + // it will be awaited if needed at the next `produceKey` invocation + argon2Promise = loadArgonWasmModule(); + argon2Promise.catch(() => {}); + } + /** * @param {Object} [config] - Full configuration, defaults to openpgp.config */ @@ -113,10 +130,8 @@ class Argon2S2K { }); // a lot of memory was used, reload to deallocate - if (decodedM > ARGON2_WASM_MEMORY_THRESHOLD_RELOAD) { - // it will be awaited if needed at the next `produceKey` invocation - argon2Promise = loadArgonWasmModule(); - argon2Promise.catch(() => {}); + if (decodedM > Argon2S2K.ARGON2_WASM_MEMORY_THRESHOLD_RELOAD) { + Argon2S2K.reloadWasmModule(); } return hash; } catch (e) { diff --git a/test/benchmarks/memory_usage.js b/test/benchmarks/memory_usage.js index 58166105e..0a2ace390 100644 --- a/test/benchmarks/memory_usage.js +++ b/test/benchmarks/memory_usage.js @@ -336,6 +336,20 @@ class MemoryBenchamrkSuite { }); }); + suite.add('openpgp.encrypt/decryptSessionKeys (argon2)', async () => { + const config = { s2kType: openpgp.enums.s2k.argon2 }; + const passwords = 'password'; + const sessionKey = { + algorithm: 'aes128', + data: require('crypto').getRandomValues(new Uint8Array(16)) + }; + const encrypted = await openpgp.encryptSessionKey({ ...sessionKey, passwords, config, format: 'object' }); + assert(encrypted.packets.length === 1); + const skesk = encrypted.packets[0]; + assert(skesk.s2k.type === 'argon2'); + await openpgp.decryptSessionKeys({ message: encrypted, passwords }); + }); + const stats = await suite.run(); // Print JSON stats to stdout console.log(JSON.stringify(stats, null, 4));