mirror of
https://github.com/element-hq/element-desktop
synced 2025-04-04 13:13:56 +02:00
Iterate
Signed-off-by: Michael Telatynski <7t3chguy@gmail.com>
This commit is contained in:
parent
c917307170
commit
ef4b41dac5
2
src/@types/global.d.ts
vendored
2
src/@types/global.d.ts
vendored
@ -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 */
|
||||||
|
@ -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";
|
||||||
|
@ -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";
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
@ -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,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user