mirror of
https://github.com/shuaiplus/nodewarden.git
synced 2026-06-20 13:00:39 +00:00
b1c6ec50da
- Introduced new backup recommendations feature with interfaces for recommended storage providers. - Updated i18n translations for backup strategy to reflect new terminology and improved descriptions. - Enhanced types with optional private and public keys in user profiles. - Redesigned backup-related styles for better layout and responsiveness. - Updated TypeScript configuration to include shared modules. - Configured Vite to resolve shared modules and allow filesystem access. - Added cron triggers for periodic tasks in Wrangler configuration.
227 lines
6.5 KiB
TypeScript
227 lines
6.5 KiB
TypeScript
import type { Env, User } from '../types';
|
|
|
|
const RUNTIME_SALT = 'nodewarden.backup-settings.runtime.v2';
|
|
const RUNTIME_INFO = 'runtime';
|
|
const PORTABLE_ALGORITHM = 'RSA-OAEP';
|
|
const PORTABLE_HASH = 'SHA-1';
|
|
const AES_GCM_ALGORITHM = 'AES-GCM';
|
|
const AES_GCM_IV_BYTES = 12;
|
|
const PORTABLE_DEK_BYTES = 32;
|
|
|
|
export interface BackupSettingsRuntimeEnvelope {
|
|
iv: string;
|
|
ciphertext: string;
|
|
}
|
|
|
|
export interface BackupSettingsPortableWrap {
|
|
userId: string;
|
|
wrappedKey: string;
|
|
}
|
|
|
|
export interface BackupSettingsPortableEnvelope {
|
|
iv: string;
|
|
ciphertext: string;
|
|
wraps: BackupSettingsPortableWrap[];
|
|
}
|
|
|
|
export interface BackupSettingsEnvelopeV2 {
|
|
version: 2;
|
|
runtime: BackupSettingsRuntimeEnvelope;
|
|
portable: BackupSettingsPortableEnvelope;
|
|
}
|
|
|
|
function bytesToBase64(bytes: Uint8Array): string {
|
|
let text = '';
|
|
for (let index = 0; index < bytes.length; index += 1) {
|
|
text += String.fromCharCode(bytes[index]);
|
|
}
|
|
return btoa(text);
|
|
}
|
|
|
|
function base64ToBytes(value: string): Uint8Array {
|
|
const normalized = String(value || '').trim();
|
|
const binary = atob(normalized);
|
|
const bytes = new Uint8Array(binary.length);
|
|
for (let index = 0; index < binary.length; index += 1) {
|
|
bytes[index] = binary.charCodeAt(index);
|
|
}
|
|
return bytes;
|
|
}
|
|
|
|
function isPlainObject(value: unknown): value is Record<string, unknown> {
|
|
return !!value && typeof value === 'object' && !Array.isArray(value);
|
|
}
|
|
|
|
async function deriveRuntimeKey(secret: string): Promise<CryptoKey> {
|
|
const encoder = new TextEncoder();
|
|
const keyMaterial = await crypto.subtle.importKey(
|
|
'raw',
|
|
encoder.encode(secret),
|
|
'HKDF',
|
|
false,
|
|
['deriveBits']
|
|
);
|
|
const bits = await crypto.subtle.deriveBits(
|
|
{
|
|
name: 'HKDF',
|
|
hash: 'SHA-256',
|
|
salt: encoder.encode(RUNTIME_SALT),
|
|
info: encoder.encode(RUNTIME_INFO),
|
|
},
|
|
keyMaterial,
|
|
256
|
|
);
|
|
return crypto.subtle.importKey('raw', bits, { name: AES_GCM_ALGORITHM }, false, ['encrypt', 'decrypt']);
|
|
}
|
|
|
|
async function encryptAesGcm(plaintext: Uint8Array, key: CryptoKey): Promise<{ iv: Uint8Array; ciphertext: Uint8Array }> {
|
|
const iv = crypto.getRandomValues(new Uint8Array(AES_GCM_IV_BYTES));
|
|
const ciphertext = new Uint8Array(
|
|
await crypto.subtle.encrypt(
|
|
{ name: AES_GCM_ALGORITHM, iv },
|
|
key,
|
|
plaintext
|
|
)
|
|
);
|
|
return { iv, ciphertext };
|
|
}
|
|
|
|
async function decryptAesGcm(ciphertext: Uint8Array, iv: Uint8Array, key: CryptoKey): Promise<Uint8Array> {
|
|
return new Uint8Array(
|
|
await crypto.subtle.decrypt(
|
|
{ name: AES_GCM_ALGORITHM, iv },
|
|
key,
|
|
ciphertext
|
|
)
|
|
);
|
|
}
|
|
|
|
async function importPortablePublicKey(publicKeyBase64: string): Promise<CryptoKey> {
|
|
return crypto.subtle.importKey(
|
|
'spki',
|
|
base64ToBytes(publicKeyBase64),
|
|
{ name: PORTABLE_ALGORITHM, hash: PORTABLE_HASH },
|
|
false,
|
|
['encrypt']
|
|
);
|
|
}
|
|
|
|
function getEligiblePortableUsers(users: Pick<User, 'id' | 'publicKey' | 'role' | 'status'>[]): Array<Pick<User, 'id' | 'publicKey'>> {
|
|
return users
|
|
.filter(
|
|
(user) =>
|
|
user.role === 'admin' &&
|
|
user.status === 'active' &&
|
|
typeof user.publicKey === 'string' &&
|
|
user.publicKey.trim().length > 0
|
|
)
|
|
.map((user) => ({
|
|
id: user.id,
|
|
publicKey: user.publicKey!,
|
|
}));
|
|
}
|
|
|
|
export function parseBackupSettingsEnvelope(raw: string | null): BackupSettingsEnvelopeV2 | null {
|
|
if (!raw) return null;
|
|
try {
|
|
const parsed = JSON.parse(raw) as Record<string, unknown>;
|
|
if (!isPlainObject(parsed) || Number(parsed.version) !== 2) return null;
|
|
const runtime = parsed.runtime;
|
|
const portable = parsed.portable;
|
|
if (!isPlainObject(runtime) || !isPlainObject(portable)) return null;
|
|
if (!Array.isArray(portable.wraps)) return null;
|
|
if (typeof runtime.iv !== 'string' || typeof runtime.ciphertext !== 'string') return null;
|
|
if (typeof portable.iv !== 'string' || typeof portable.ciphertext !== 'string') return null;
|
|
return {
|
|
version: 2,
|
|
runtime: {
|
|
iv: runtime.iv,
|
|
ciphertext: runtime.ciphertext,
|
|
},
|
|
portable: {
|
|
iv: portable.iv,
|
|
ciphertext: portable.ciphertext,
|
|
wraps: portable.wraps
|
|
.filter((entry): entry is Record<string, unknown> => isPlainObject(entry))
|
|
.map((entry) => ({
|
|
userId: String(entry.userId || '').trim(),
|
|
wrappedKey: String(entry.wrappedKey || '').trim(),
|
|
}))
|
|
.filter((entry) => entry.userId && entry.wrappedKey),
|
|
},
|
|
};
|
|
} catch {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
export async function encryptBackupSettingsEnvelope(
|
|
plaintext: string,
|
|
env: Env,
|
|
users: Pick<User, 'id' | 'publicKey' | 'role' | 'status'>[]
|
|
): Promise<string> {
|
|
const encoder = new TextEncoder();
|
|
const eligibleUsers = getEligiblePortableUsers(users);
|
|
if (!eligibleUsers.length) {
|
|
throw new Error('No active administrator public keys are available for backup settings recovery');
|
|
}
|
|
|
|
const runtimeKey = await deriveRuntimeKey(env.JWT_SECRET);
|
|
const runtime = await encryptAesGcm(encoder.encode(plaintext), runtimeKey);
|
|
|
|
const portableDek = crypto.getRandomValues(new Uint8Array(PORTABLE_DEK_BYTES));
|
|
const portableKey = await crypto.subtle.importKey(
|
|
'raw',
|
|
portableDek,
|
|
{ name: AES_GCM_ALGORITHM },
|
|
false,
|
|
['encrypt']
|
|
);
|
|
const portableCipher = await encryptAesGcm(encoder.encode(plaintext), portableKey);
|
|
|
|
const wraps: BackupSettingsPortableWrap[] = [];
|
|
for (const user of eligibleUsers) {
|
|
const publicKey = await importPortablePublicKey(user.publicKey!);
|
|
const wrappedKey = new Uint8Array(
|
|
await crypto.subtle.encrypt(
|
|
{ name: PORTABLE_ALGORITHM },
|
|
publicKey,
|
|
portableDek
|
|
)
|
|
);
|
|
wraps.push({
|
|
userId: user.id,
|
|
wrappedKey: bytesToBase64(wrappedKey),
|
|
});
|
|
}
|
|
|
|
const envelope: BackupSettingsEnvelopeV2 = {
|
|
version: 2,
|
|
runtime: {
|
|
iv: bytesToBase64(runtime.iv),
|
|
ciphertext: bytesToBase64(runtime.ciphertext),
|
|
},
|
|
portable: {
|
|
iv: bytesToBase64(portableCipher.iv),
|
|
ciphertext: bytesToBase64(portableCipher.ciphertext),
|
|
wraps,
|
|
},
|
|
};
|
|
|
|
return JSON.stringify(envelope);
|
|
}
|
|
|
|
export async function decryptBackupSettingsRuntime(raw: string, env: Env): Promise<string> {
|
|
const envelope = parseBackupSettingsEnvelope(raw);
|
|
if (!envelope) {
|
|
throw new Error('Backup settings envelope is invalid');
|
|
}
|
|
const runtimeKey = await deriveRuntimeKey(env.JWT_SECRET);
|
|
const plaintext = await decryptAesGcm(
|
|
base64ToBytes(envelope.runtime.ciphertext),
|
|
base64ToBytes(envelope.runtime.iv),
|
|
runtimeKey
|
|
);
|
|
return new TextDecoder().decode(plaintext);
|
|
}
|