mirror of
https://github.com/kmc7468/arkvault.git
synced 2025-12-14 22:08:45 +00:00
로그인 및 암호 키 등록 이후 클라이언트 승인 대기 페이지로의 자동 리다이렉션 구현
This commit is contained in:
@@ -1,10 +1,14 @@
|
|||||||
import { callAPI } from "$lib/hooks";
|
import { callAPI } from "$lib/hooks";
|
||||||
|
import { storeMasterKeys } from "$lib/indexedDB";
|
||||||
import {
|
import {
|
||||||
encodeToBase64,
|
encodeToBase64,
|
||||||
decodeFromBase64,
|
decodeFromBase64,
|
||||||
decryptRSACiphertext,
|
decryptRSACiphertext,
|
||||||
signRSAMessage,
|
signRSAMessage,
|
||||||
|
makeAESKeyNonextractable,
|
||||||
|
unwrapAESKeyUsingRSA,
|
||||||
} from "$lib/modules/crypto";
|
} from "$lib/modules/crypto";
|
||||||
|
import { masterKeyStore } from "$lib/stores";
|
||||||
|
|
||||||
export const requestClientRegistration = async (
|
export const requestClientRegistration = async (
|
||||||
encryptKeyBase64: string,
|
encryptKeyBase64: string,
|
||||||
@@ -40,3 +44,35 @@ export const requestClientRegistration = async (
|
|||||||
});
|
});
|
||||||
return res.ok;
|
return res.ok;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const requestMasterKeyDownload = async (decryptKey: CryptoKey) => {
|
||||||
|
const res = await callAPI("/api/mek/list", { method: "GET" });
|
||||||
|
if (!res.ok) return false;
|
||||||
|
|
||||||
|
const data = await res.json();
|
||||||
|
const { meks: masterKeysWrapped } = data as {
|
||||||
|
meks: {
|
||||||
|
version: number;
|
||||||
|
state: "active" | "retired";
|
||||||
|
mek: string;
|
||||||
|
}[];
|
||||||
|
};
|
||||||
|
const masterKeys = await Promise.all(
|
||||||
|
masterKeysWrapped.map(async ({ version, state, mek: masterKeyWrapped }) => ({
|
||||||
|
version,
|
||||||
|
state,
|
||||||
|
masterKey: await makeAESKeyNonextractable(
|
||||||
|
await unwrapAESKeyUsingRSA(decodeFromBase64(masterKeyWrapped), decryptKey),
|
||||||
|
),
|
||||||
|
})),
|
||||||
|
);
|
||||||
|
|
||||||
|
await storeMasterKeys(
|
||||||
|
masterKeys.map(({ version, state, masterKey }) => ({ version, state, key: masterKey })),
|
||||||
|
);
|
||||||
|
masterKeyStore.set(
|
||||||
|
new Map(masterKeys.map(({ version, state, masterKey }) => [version, { state, masterKey }])),
|
||||||
|
);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|||||||
@@ -5,28 +5,35 @@
|
|||||||
import { TitleDiv, BottomDiv } from "$lib/components/divs";
|
import { TitleDiv, BottomDiv } from "$lib/components/divs";
|
||||||
import { TextInput } from "$lib/components/inputs";
|
import { TextInput } from "$lib/components/inputs";
|
||||||
import { refreshToken } from "$lib/hooks/callAPI";
|
import { refreshToken } from "$lib/hooks/callAPI";
|
||||||
import { clientKeyStore } from "$lib/stores";
|
import { clientKeyStore, masterKeyStore } from "$lib/stores";
|
||||||
import { requestLogin, requestTokenUpgrade } from "./service";
|
import { requestLogin, requestTokenUpgrade, requestMasterKeyDownload } from "./service";
|
||||||
|
|
||||||
let { data } = $props();
|
let { data } = $props();
|
||||||
|
|
||||||
let email = $state("");
|
let email = $state("");
|
||||||
let password = $state("");
|
let password = $state("");
|
||||||
|
|
||||||
|
const redirect = async (url: string) => {
|
||||||
|
return await goto(`${url}?redirect=${encodeURIComponent(data.redirectPath)}`);
|
||||||
|
};
|
||||||
|
|
||||||
const login = async () => {
|
const login = async () => {
|
||||||
// TODO: Validation
|
// TODO: Validation
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (!(await requestLogin(email, password))) throw new Error("Failed to login");
|
if (!(await requestLogin(email, password))) throw new Error("Failed to login");
|
||||||
|
|
||||||
if ($clientKeyStore && !(await requestTokenUpgrade($clientKeyStore)))
|
if (!$clientKeyStore) return await redirect("/key/generate");
|
||||||
throw new Error("Failed to upgrade token");
|
|
||||||
|
|
||||||
await goto(
|
if (!(await requestTokenUpgrade($clientKeyStore))) throw new Error("Failed to upgrade token");
|
||||||
$clientKeyStore
|
|
||||||
? data.redirectPath
|
// TODO: Multi-user support
|
||||||
: "/key/generate?redirect=" + encodeURIComponent(data.redirectPath),
|
|
||||||
);
|
if ($masterKeyStore || (await requestMasterKeyDownload($clientKeyStore.decryptKey))) {
|
||||||
|
await goto(data.redirectPath);
|
||||||
|
} else {
|
||||||
|
await redirect("/client/pending");
|
||||||
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
// TODO: Alert
|
// TODO: Alert
|
||||||
throw e;
|
throw e;
|
||||||
|
|||||||
@@ -3,6 +3,8 @@ import { requestTokenUpgrade as requestTokenUpgradeInternal } from "$lib/service
|
|||||||
import { requestClientRegistration } from "$lib/services/key";
|
import { requestClientRegistration } from "$lib/services/key";
|
||||||
import type { ClientKeys } from "$lib/stores";
|
import type { ClientKeys } from "$lib/stores";
|
||||||
|
|
||||||
|
export { requestMasterKeyDownload } from "$lib/services/key";
|
||||||
|
|
||||||
export const requestLogin = async (email: string, password: string) => {
|
export const requestLogin = async (email: string, password: string) => {
|
||||||
const res = await fetch("/api/auth/login", {
|
const res = await fetch("/api/auth/login", {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
|
|||||||
@@ -1,13 +1,6 @@
|
|||||||
import { callAPI } from "$lib/hooks";
|
import { exportRSAKey, digestSHA256 } from "$lib/modules/crypto";
|
||||||
import { storeMasterKeys } from "$lib/indexedDB";
|
|
||||||
import {
|
export { requestMasterKeyDownload } from "$lib/services/key";
|
||||||
decodeFromBase64,
|
|
||||||
exportRSAKey,
|
|
||||||
makeAESKeyNonextractable,
|
|
||||||
unwrapAESKeyUsingRSA,
|
|
||||||
digestSHA256,
|
|
||||||
} from "$lib/modules/crypto";
|
|
||||||
import { masterKeyStore } from "$lib/stores";
|
|
||||||
|
|
||||||
export const generateEncryptKeyFingerprint = async (encryptKey: CryptoKey) => {
|
export const generateEncryptKeyFingerprint = async (encryptKey: CryptoKey) => {
|
||||||
const { key } = await exportRSAKey(encryptKey);
|
const { key } = await exportRSAKey(encryptKey);
|
||||||
@@ -19,35 +12,3 @@ export const generateEncryptKeyFingerprint = async (encryptKey: CryptoKey) => {
|
|||||||
.match(/.{1,4}/g)!
|
.match(/.{1,4}/g)!
|
||||||
.join(" ");
|
.join(" ");
|
||||||
};
|
};
|
||||||
|
|
||||||
export const requestMasterKeyDownload = async (decryptKey: CryptoKey) => {
|
|
||||||
const res = await callAPI("/api/mek/list", { method: "GET" });
|
|
||||||
if (!res.ok) return false;
|
|
||||||
|
|
||||||
const data = await res.json();
|
|
||||||
const { meks: masterKeysWrapped } = data as {
|
|
||||||
meks: {
|
|
||||||
version: number;
|
|
||||||
state: "active" | "retired";
|
|
||||||
mek: string;
|
|
||||||
}[];
|
|
||||||
};
|
|
||||||
const masterKeys = await Promise.all(
|
|
||||||
masterKeysWrapped.map(async ({ version, state, mek: masterKeyWrapped }) => ({
|
|
||||||
version,
|
|
||||||
state,
|
|
||||||
masterKey: await makeAESKeyNonextractable(
|
|
||||||
await unwrapAESKeyUsingRSA(decodeFromBase64(masterKeyWrapped), decryptKey),
|
|
||||||
),
|
|
||||||
})),
|
|
||||||
);
|
|
||||||
|
|
||||||
await storeMasterKeys(
|
|
||||||
masterKeys.map(({ version, state, masterKey }) => ({ version, state, key: masterKey })),
|
|
||||||
);
|
|
||||||
masterKeyStore.set(
|
|
||||||
new Map(masterKeys.map(({ version, state, masterKey }) => [version, { state, masterKey }])),
|
|
||||||
);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|||||||
@@ -76,7 +76,7 @@
|
|||||||
)
|
)
|
||||||
throw new Error("Failed to register initial MEK");
|
throw new Error("Failed to register initial MEK");
|
||||||
|
|
||||||
await goto(data.redirectPath);
|
await goto("/client/pending?redirect=" + encodeURIComponent(data.redirectPath));
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
// TODO: Error handling
|
// TODO: Error handling
|
||||||
throw e;
|
throw e;
|
||||||
|
|||||||
Reference in New Issue
Block a user