Signed-off-by: Michael Telatynski <7t3chguy@gmail.com>
This commit is contained in:
Michael Telatynski 2025-03-28 13:46:20 +00:00
parent c917307170
commit ef4b41dac5
No known key found for this signature in database
GPG Key ID: A2B008A5F49F5D0D
5 changed files with 26 additions and 40 deletions

View File

@ -10,7 +10,7 @@ import { type BrowserWindow } from "electron";
import type Store from "electron-store"; import type Store from "electron-store";
import type AutoLaunch from "auto-launch"; import type AutoLaunch from "auto-launch";
import { type AppLocalization } from "../language-helper.js"; import { type AppLocalization } from "../language-helper.js";
import { type StoreData } from "../electron-main"; import { type StoreData } from "../electron-main.js";
// global type extensions need to use var for whatever reason // global type extensions need to use var for whatever reason
/* eslint-disable no-var */ /* eslint-disable no-var */

View File

@ -7,7 +7,6 @@ Please see LICENSE files in the repository root for full details.
import { app, autoUpdater, desktopCapturer, ipcMain, powerSaveBlocker, TouchBar, nativeImage } from "electron"; import { app, autoUpdater, desktopCapturer, ipcMain, powerSaveBlocker, TouchBar, nativeImage } from "electron";
import { relaunchApp } from "@standardnotes/electron-clear-data"; import { relaunchApp } from "@standardnotes/electron-clear-data";
import keytar from "keytar-forked";
import IpcMainEvent = Electron.IpcMainEvent; import IpcMainEvent = Electron.IpcMainEvent;
import { recordSSOSession } from "./protocol.js"; import { recordSSOSession } from "./protocol.js";

View File

@ -10,7 +10,6 @@ import { type TranslationKey as TKey } from "matrix-web-i18n";
import { dirname } from "node:path"; import { dirname } from "node:path";
import { fileURLToPath } from "node:url"; import { fileURLToPath } from "node:url";
import type Store from "electron-store";
import type EN from "./i18n/strings/en_EN.json"; import type EN from "./i18n/strings/en_EN.json";
import { loadJsonFile } from "./utils.js"; import { loadJsonFile } from "./utils.js";

View File

@ -1,5 +1,5 @@
/* /*
Copyright 2022 New Vector Ltd Copyright 2022-2025 New Vector Ltd
Licensed under the Apache License, Version 2.0 (the "License"); Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License. you may not use this file except in compliance with the License.
@ -15,43 +15,35 @@ limitations under the License.
*/ */
import { safeStorage } from "electron"; import { safeStorage } from "electron";
import * as keytar from "keytar-forked";
import type * as Keytar from "keytar";
const KEYTAR_SERVICE = "element.io"; const KEYTAR_SERVICE = "element.io";
const LEGACY_KEYTAR_SERVICE = "riot.im"; const LEGACY_KEYTAR_SERVICE = "riot.im";
let keytar: typeof Keytar | undefined; const getStorageKey = (key: string) => `safeStorage.${key}` as const;
try {
// eslint-disable-next-line @typescript-eslint/no-var-requires
keytar = require("keytar");
} catch (e) {
if ((<NodeJS.ErrnoException>e).code === "MODULE_NOT_FOUND") {
console.log("Keytar isn't installed; secure key storage is disabled.");
} else {
console.warn("Keytar unexpected error:", e);
}
}
/**
* Migrates keytar data to safeStorage,
* deletes data from legacy keytar but keeps it in the new keytar for downgrade compatibility.
*/
export async function migrate(): Promise<void> { export async function migrate(): Promise<void> {
if (global.store.get("migratedToSafeStorage")) return; // already done if (global.store.get("migratedToSafeStorage")) return; // already done
if (keytar) { const credentials = [
const credentials = [ ...(await keytar.findCredentials(LEGACY_KEYTAR_SERVICE)),
...(await keytar.findCredentials(LEGACY_KEYTAR_SERVICE)), ...(await keytar.findCredentials(KEYTAR_SERVICE)),
...(await keytar.findCredentials(KEYTAR_SERVICE)), ];
]; credentials.forEach((cred) => {
credentials.forEach((cred) => { deletePassword(cred.account); // delete from keytar & keytar legacy
deletePassword(cred.account); setPassword(cred.account, cred.password); // write to safeStorage & keytar for downgrade compatibility
setPassword(cred.account, cred.password); });
});
}
global.store.set("migratedToSafeStorage", true); global.store.set("migratedToSafeStorage", true);
} }
/** /**
* Get the stored password for the key. * Get the stored password for the key.
* We read from safeStorage first, then keytar & keytar legacy.
* *
* @param key The string key name. * @param key The string key name.
* *
@ -59,21 +51,17 @@ export async function migrate(): Promise<void> {
*/ */
export async function getPassword(key: string): Promise<string | null> { export async function getPassword(key: string): Promise<string | null> {
if (safeStorage.isEncryptionAvailable()) { if (safeStorage.isEncryptionAvailable()) {
const encryptedValue = global.store.get(`safeStorage.${key}`); const encryptedValue = global.store.get(getStorageKey(key));
if (typeof encryptedValue === "string") { if (typeof encryptedValue === "string") {
return safeStorage.decryptString(Buffer.from(encryptedValue)); return safeStorage.decryptString(Buffer.from(encryptedValue));
} }
} }
if (keytar) { return (await keytar.getPassword(KEYTAR_SERVICE, key)) ?? (await keytar.getPassword(LEGACY_KEYTAR_SERVICE, key));
return (
(await keytar.getPassword(KEYTAR_SERVICE, key)) ?? (await keytar.getPassword(LEGACY_KEYTAR_SERVICE, key))
);
}
return null;
} }
/** /**
* Add the password for the key to the keychain. * Add the password for the key to the keychain.
* We write to both safeStorage & keytar to support downgrading the application.
* *
* @param key The string key name. * @param key The string key name.
* @param password The string password. * @param password The string password.
@ -83,13 +71,14 @@ export async function getPassword(key: string): Promise<string | null> {
export async function setPassword(key: string, password: string): Promise<void> { export async function setPassword(key: string, password: string): Promise<void> {
if (safeStorage.isEncryptionAvailable()) { if (safeStorage.isEncryptionAvailable()) {
const encryptedValue = safeStorage.encryptString(password); const encryptedValue = safeStorage.encryptString(password);
global.store.set(`safeStorage.${key}`, encryptedValue.toString()); global.store.set(getStorageKey(key), encryptedValue.toString());
} }
await keytar?.setPassword(KEYTAR_SERVICE, key, password); await keytar.setPassword(KEYTAR_SERVICE, key, password);
} }
/** /**
* Delete the stored password for the key. * Delete the stored password for the key.
* Removes from safeStorage, keytar & keytar legacy.
* *
* @param key The string key name. * @param key The string key name.
* *
@ -97,9 +86,9 @@ export async function setPassword(key: string, password: string): Promise<void>
*/ */
export async function deletePassword(key: string): Promise<boolean> { export async function deletePassword(key: string): Promise<boolean> {
if (safeStorage.isEncryptionAvailable()) { if (safeStorage.isEncryptionAvailable()) {
global.store.delete(`safeStorage.${key}`); global.store.delete(getStorageKey(key));
await keytar?.deletePassword(LEGACY_KEYTAR_SERVICE, key); await keytar.deletePassword(LEGACY_KEYTAR_SERVICE, key);
await keytar?.deletePassword(KEYTAR_SERVICE, key); await keytar.deletePassword(KEYTAR_SERVICE, key);
return true; return true;
} }
return false; return false;

View File

@ -8,7 +8,6 @@ Please see LICENSE files in the repository root for full details.
import { app, ipcMain } from "electron"; import { app, ipcMain } from "electron";
import { promises as afs } from "node:fs"; import { promises as afs } from "node:fs";
import path from "node:path"; import path from "node:path";
import keytar from "keytar-forked";
import type { import type {
Seshat as SeshatType, Seshat as SeshatType,