52 lines
1.8 KiB
TypeScript
52 lines
1.8 KiB
TypeScript
import { ipcMain } from "electron";
|
|
import { deletePassword, getPassword, setPassword } from "keytar";
|
|
|
|
import { BiometricMain } from "./biometric/biometric.main";
|
|
|
|
const AuthRequiredSuffix = "_biometric";
|
|
const AuthenticatedActions = ["getPassword"];
|
|
|
|
export class DesktopCredentialStorageListener {
|
|
constructor(private serviceName: string, private biometricService: BiometricMain) {}
|
|
|
|
init() {
|
|
ipcMain.on("keytar", async (event: any, message: any) => {
|
|
try {
|
|
let serviceName = this.serviceName;
|
|
message.keySuffix = "_" + (message.keySuffix ?? "");
|
|
if (message.keySuffix !== "_") {
|
|
serviceName += message.keySuffix;
|
|
}
|
|
|
|
const authenticationRequired =
|
|
AuthenticatedActions.includes(message.action) && AuthRequiredSuffix === message.keySuffix;
|
|
const authenticated = !authenticationRequired || (await this.authenticateBiometric());
|
|
|
|
let val: string | boolean = null;
|
|
if (authenticated && message.action && message.key) {
|
|
if (message.action === "getPassword") {
|
|
val = await getPassword(serviceName, message.key);
|
|
} else if (message.action === "hasPassword") {
|
|
const result = await getPassword(serviceName, message.key);
|
|
val = result != null;
|
|
} else if (message.action === "setPassword" && message.value) {
|
|
await setPassword(serviceName, message.key, message.value);
|
|
} else if (message.action === "deletePassword") {
|
|
await deletePassword(serviceName, message.key);
|
|
}
|
|
}
|
|
event.returnValue = val;
|
|
} catch {
|
|
event.returnValue = null;
|
|
}
|
|
});
|
|
}
|
|
|
|
private async authenticateBiometric(): Promise<boolean> {
|
|
if (this.biometricService) {
|
|
return await this.biometricService.authenticateBiometric();
|
|
}
|
|
return false;
|
|
}
|
|
}
|