mirror of
https://github.com/shuaiplus/nodewarden.git
synced 2026-06-20 13:00:39 +00:00
336 lines
13 KiB
TypeScript
336 lines
13 KiB
TypeScript
import { base64ToBytes, bytesToBase64, decryptBw, decryptBwFileData, decryptStr, encryptBw, encryptBwFileData, hkdf, pbkdf2 } from '../crypto';
|
|
import type { Send, SendDraft, SessionState } from '../types';
|
|
import { chunkArray, createApiError, parseErrorMessage, parseJson, uploadDirectEncryptedPayload, type AuthedFetch } from './shared';
|
|
|
|
function toIsoDateFromDays(value: string, required: boolean): string | null {
|
|
const raw = String(value || '').trim();
|
|
if (!raw) {
|
|
if (required) throw new Error('Deletion days is required');
|
|
return null;
|
|
}
|
|
const n = Number(raw);
|
|
if (!Number.isFinite(n) || n < 0) {
|
|
if (required) throw new Error('Invalid deletion days');
|
|
throw new Error('Invalid expiration days');
|
|
}
|
|
if (!required && n === 0) return null;
|
|
const date = new Date(Date.now() + Math.floor(n) * 24 * 60 * 60 * 1000);
|
|
return date.toISOString();
|
|
}
|
|
|
|
function bytesToBase64Url(bytes: Uint8Array): string {
|
|
return bytesToBase64(bytes).replace(/\+/g, '-').replace(/\//g, '_').replace(/=+$/g, '');
|
|
}
|
|
|
|
function base64UrlToBytes(value: string): Uint8Array {
|
|
const raw = value.replace(/-/g, '+').replace(/_/g, '/');
|
|
const padded = raw + '='.repeat((4 - (raw.length % 4)) % 4);
|
|
return base64ToBytes(padded);
|
|
}
|
|
|
|
const SEND_KEY_SALT = 'bitwarden-send';
|
|
const SEND_KEY_PURPOSE = 'send';
|
|
const SEND_KEY_SEED_BYTES = 16;
|
|
const SEND_PASSWORD_ITERATIONS = 100000;
|
|
|
|
async function encryptTextValue(value: string, enc: Uint8Array, mac: Uint8Array): Promise<string | null> {
|
|
const s = String(value || '');
|
|
if (!s.trim()) return null;
|
|
return encryptBw(new TextEncoder().encode(s), enc, mac);
|
|
}
|
|
|
|
async function toSendKeyParts(sendKeyMaterial: Uint8Array): Promise<{ enc: Uint8Array; mac: Uint8Array }> {
|
|
if (sendKeyMaterial.length >= 64) {
|
|
return { enc: sendKeyMaterial.slice(0, 32), mac: sendKeyMaterial.slice(32, 64) };
|
|
}
|
|
const derived = await hkdf(sendKeyMaterial, SEND_KEY_SALT, SEND_KEY_PURPOSE, 64);
|
|
return { enc: derived.slice(0, 32), mac: derived.slice(32, 64) };
|
|
}
|
|
|
|
async function hashSendPasswordB64(password: string, sendKeyMaterial: Uint8Array): Promise<string> {
|
|
const hash = await pbkdf2(password, sendKeyMaterial, SEND_PASSWORD_ITERATIONS, 32);
|
|
return bytesToBase64(hash);
|
|
}
|
|
|
|
function parseMaxAccessCountRaw(value: string): number | null {
|
|
const raw = String(value || '').trim();
|
|
if (!raw) return null;
|
|
const n = Number(raw);
|
|
if (!Number.isFinite(n) || n < 0) throw new Error('Invalid max access count');
|
|
return Math.floor(n);
|
|
}
|
|
|
|
export async function getSends(authedFetch: AuthedFetch): Promise<Send[]> {
|
|
const resp = await authedFetch('/api/sends');
|
|
if (!resp.ok) throw new Error('Failed to load sends');
|
|
const body = await parseJson<{ data?: Send[] }>(resp);
|
|
return body?.data || [];
|
|
}
|
|
|
|
export async function createSend(
|
|
authedFetch: AuthedFetch,
|
|
session: SessionState,
|
|
draft: SendDraft,
|
|
onProgress?: (percent: number | null) => void
|
|
): Promise<Send> {
|
|
if (!session.symEncKey || !session.symMacKey) throw new Error('Vault key unavailable');
|
|
const userEnc = base64ToBytes(session.symEncKey);
|
|
const userMac = base64ToBytes(session.symMacKey);
|
|
const sendKeyMaterial = crypto.getRandomValues(new Uint8Array(SEND_KEY_SEED_BYTES));
|
|
const sendKeyForUser = await encryptBw(sendKeyMaterial, userEnc, userMac);
|
|
const sendKey = await toSendKeyParts(sendKeyMaterial);
|
|
const nameCipher = await encryptTextValue(draft.name || '', sendKey.enc, sendKey.mac);
|
|
const notesCipher = await encryptTextValue(draft.notes || '', sendKey.enc, sendKey.mac);
|
|
|
|
const deletionIso = toIsoDateFromDays(draft.deletionDays, true)!;
|
|
const expirationIso = toIsoDateFromDays(draft.expirationDays, false);
|
|
const maxAccessCount = parseMaxAccessCountRaw(draft.maxAccessCount);
|
|
const password = String(draft.password || '');
|
|
const passwordHash = password ? await hashSendPasswordB64(password, sendKeyMaterial) : null;
|
|
|
|
if (draft.type === 'text') {
|
|
const text = String(draft.text || '').trim();
|
|
if (!text) throw new Error('Send text is required');
|
|
const textCipher = await encryptTextValue(text, sendKey.enc, sendKey.mac);
|
|
|
|
const payload = {
|
|
type: 0,
|
|
name: nameCipher,
|
|
notes: notesCipher,
|
|
key: sendKeyForUser,
|
|
text: {
|
|
text: textCipher,
|
|
hidden: false,
|
|
},
|
|
maxAccessCount,
|
|
password: passwordHash,
|
|
hideEmail: false,
|
|
disabled: !!draft.disabled,
|
|
deletionDate: deletionIso,
|
|
expirationDate: expirationIso,
|
|
};
|
|
|
|
const resp = await authedFetch('/api/sends', {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify(payload),
|
|
});
|
|
if (!resp.ok) throw new Error(await parseErrorMessage(resp, 'Create send failed'));
|
|
const body = await parseJson<Send>(resp);
|
|
if (!body?.id) throw new Error('Create send failed');
|
|
return body;
|
|
}
|
|
|
|
if (!draft.file) throw new Error('File is required');
|
|
const fileNameCipher = await encryptTextValue(draft.file.name, sendKey.enc, sendKey.mac);
|
|
if (!fileNameCipher) throw new Error('Invalid file name');
|
|
const plainFileBytes = new Uint8Array(await draft.file.arrayBuffer());
|
|
const encryptedFileBytes = await encryptBwFileData(plainFileBytes, sendKey.enc, sendKey.mac);
|
|
|
|
const fileResp = await authedFetch('/api/sends/file/v2', {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify({
|
|
type: 1,
|
|
name: nameCipher,
|
|
notes: notesCipher,
|
|
key: sendKeyForUser,
|
|
file: {
|
|
fileName: fileNameCipher,
|
|
},
|
|
fileLength: encryptedFileBytes.byteLength,
|
|
maxAccessCount,
|
|
password: passwordHash,
|
|
hideEmail: false,
|
|
disabled: !!draft.disabled,
|
|
deletionDate: deletionIso,
|
|
expirationDate: expirationIso,
|
|
}),
|
|
});
|
|
if (!fileResp.ok) throw new Error(await parseErrorMessage(fileResp, 'Create file send failed'));
|
|
|
|
const uploadInfo = await parseJson<{ url?: string; sendResponse?: Send; fileUploadType?: number }>(fileResp);
|
|
const uploadUrl = uploadInfo?.url;
|
|
if (!uploadUrl) throw new Error('Create file send failed: missing upload URL');
|
|
if (!session.accessToken) throw new Error('Unauthorized');
|
|
const payload = new ArrayBuffer(encryptedFileBytes.byteLength);
|
|
new Uint8Array(payload).set(encryptedFileBytes);
|
|
const uploadResp = await uploadDirectEncryptedPayload({
|
|
accessToken: session.accessToken,
|
|
uploadUrl,
|
|
payload,
|
|
fileUploadType: uploadInfo?.fileUploadType,
|
|
unsupportedMessage: 'Unsupported send upload type',
|
|
onProgress,
|
|
});
|
|
if (!uploadResp.ok) throw new Error(await parseErrorMessage(uploadResp, 'Upload send file failed'));
|
|
if (!uploadInfo?.sendResponse?.id) throw new Error('Create file send failed');
|
|
return uploadInfo.sendResponse;
|
|
}
|
|
|
|
export async function updateSend(
|
|
authedFetch: AuthedFetch,
|
|
session: SessionState,
|
|
send: Send,
|
|
draft: SendDraft
|
|
): Promise<Send> {
|
|
if (!session.symEncKey || !session.symMacKey) throw new Error('Vault key unavailable');
|
|
if (!send.key) throw new Error('Send key unavailable');
|
|
const userEnc = base64ToBytes(session.symEncKey);
|
|
const userMac = base64ToBytes(session.symMacKey);
|
|
const sendKeyMaterial = await decryptBw(send.key, userEnc, userMac);
|
|
const sendKey = await toSendKeyParts(sendKeyMaterial);
|
|
const nameCipher = await encryptTextValue(draft.name || '', sendKey.enc, sendKey.mac);
|
|
const notesCipher = await encryptTextValue(draft.notes || '', sendKey.enc, sendKey.mac);
|
|
|
|
const deletionIso = toIsoDateFromDays(draft.deletionDays, true)!;
|
|
const expirationIso = toIsoDateFromDays(draft.expirationDays, false);
|
|
const maxAccessCount = parseMaxAccessCountRaw(draft.maxAccessCount);
|
|
|
|
if (draft.type === 'file' && draft.file) {
|
|
throw new Error('Updating file content is not supported yet');
|
|
}
|
|
|
|
const textCipher = await encryptTextValue(String(draft.text || ''), sendKey.enc, sendKey.mac);
|
|
|
|
const passwordRaw = String(draft.password || '');
|
|
const passwordHash = passwordRaw ? await hashSendPasswordB64(passwordRaw, sendKeyMaterial) : null;
|
|
|
|
const payload = {
|
|
id: send.id,
|
|
type: draft.type === 'file' ? 1 : 0,
|
|
name: nameCipher,
|
|
notes: notesCipher,
|
|
key: send.key,
|
|
text: {
|
|
text: textCipher,
|
|
hidden: false,
|
|
},
|
|
maxAccessCount,
|
|
password: passwordHash,
|
|
hideEmail: false,
|
|
disabled: !!draft.disabled,
|
|
deletionDate: deletionIso,
|
|
expirationDate: expirationIso,
|
|
};
|
|
|
|
const resp = await authedFetch(`/api/sends/${encodeURIComponent(send.id)}`, {
|
|
method: 'PUT',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify(payload),
|
|
});
|
|
if (!resp.ok) throw new Error(await parseErrorMessage(resp, 'Update send failed'));
|
|
const body = await parseJson<Send>(resp);
|
|
if (!body?.id) throw new Error('Update send failed');
|
|
return body;
|
|
}
|
|
|
|
export async function deleteSend(authedFetch: AuthedFetch, sendId: string): Promise<void> {
|
|
const resp = await authedFetch(`/api/sends/${encodeURIComponent(sendId)}`, { method: 'DELETE' });
|
|
if (!resp.ok) throw new Error(await parseErrorMessage(resp, 'Delete send failed'));
|
|
}
|
|
|
|
export async function bulkDeleteSends(authedFetch: AuthedFetch, ids: string[]): Promise<void> {
|
|
const uniqueIds = Array.from(new Set(ids.map((id) => String(id || '').trim()).filter(Boolean)));
|
|
for (const chunk of chunkArray(uniqueIds, 200)) {
|
|
const resp = await authedFetch('/api/sends/delete', {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify({ ids: chunk }),
|
|
});
|
|
if (!resp.ok) throw new Error('Bulk delete sends failed');
|
|
}
|
|
}
|
|
|
|
async function buildPublicSendAccessPayload(password?: string, keyPart?: string | null): Promise<Record<string, unknown>> {
|
|
const payload: Record<string, unknown> = {};
|
|
const plainPassword = String(password || '').trim();
|
|
if (!plainPassword) return payload;
|
|
|
|
if (keyPart) {
|
|
try {
|
|
const sendKeyMaterial = base64UrlToBytes(keyPart);
|
|
const passwordHashB64 = await hashSendPasswordB64(plainPassword, sendKeyMaterial);
|
|
payload.passwordHash = passwordHashB64;
|
|
payload.password_hash_b64 = passwordHashB64;
|
|
payload.passwordHashB64 = passwordHashB64;
|
|
} catch {
|
|
// Key material invalid; server will reject as unauthorized.
|
|
}
|
|
}
|
|
return payload;
|
|
}
|
|
|
|
export async function accessPublicSend(
|
|
accessId: string,
|
|
keyPart?: string | null,
|
|
password?: string,
|
|
options?: { signal?: AbortSignal }
|
|
): Promise<unknown> {
|
|
const payload = await buildPublicSendAccessPayload(password, keyPart);
|
|
const resp = await fetch(`/api/sends/access/${encodeURIComponent(accessId)}`, {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify(payload),
|
|
signal: options?.signal,
|
|
});
|
|
if (!resp.ok) {
|
|
const message = await parseErrorMessage(resp, 'Failed to access send');
|
|
throw createApiError(message, resp.status);
|
|
}
|
|
return (await parseJson<unknown>(resp)) || null;
|
|
}
|
|
|
|
export async function accessPublicSendFile(sendId: string, fileId: string, keyPart?: string | null, password?: string): Promise<string> {
|
|
const payload = await buildPublicSendAccessPayload(password, keyPart);
|
|
const resp = await fetch(`/api/sends/${encodeURIComponent(sendId)}/access/file/${encodeURIComponent(fileId)}`, {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify(payload),
|
|
});
|
|
if (!resp.ok) {
|
|
const message = await parseErrorMessage(resp, 'Failed to access send file');
|
|
throw createApiError(message, resp.status);
|
|
}
|
|
const body = await parseJson<{ url?: string }>(resp);
|
|
if (!body?.url) throw new Error('Missing file URL');
|
|
return body.url;
|
|
}
|
|
|
|
export async function decryptPublicSend(accessData: unknown, urlSafeKey: string): Promise<unknown> {
|
|
const sendKeyMaterial = base64UrlToBytes(urlSafeKey);
|
|
const sendKey = await toSendKeyParts(sendKeyMaterial);
|
|
const source = accessData && typeof accessData === 'object' ? accessData as Record<string, unknown> : {};
|
|
const text = source.text && typeof source.text === 'object' ? source.text as Record<string, unknown> : null;
|
|
const file = source.file && typeof source.file === 'object' ? source.file as Record<string, unknown> : null;
|
|
const out: Record<string, unknown> = { ...source };
|
|
out.decName = await decryptStr(String(source.name || ''), sendKey.enc, sendKey.mac);
|
|
if (text?.text) {
|
|
out.decText = await decryptStr(String(text.text), sendKey.enc, sendKey.mac);
|
|
}
|
|
if (file?.fileName) {
|
|
try {
|
|
out.decFileName = await decryptStr(String(file.fileName), sendKey.enc, sendKey.mac);
|
|
} catch {
|
|
out.decFileName = String(file.fileName);
|
|
}
|
|
}
|
|
return out;
|
|
}
|
|
|
|
export async function decryptPublicSendFileBytes(
|
|
encryptedBytes: ArrayBuffer | Uint8Array,
|
|
urlSafeKey: string
|
|
): Promise<Uint8Array> {
|
|
const sendKeyMaterial = base64UrlToBytes(urlSafeKey);
|
|
const sendKey = await toSendKeyParts(sendKeyMaterial);
|
|
const encrypted = encryptedBytes instanceof Uint8Array ? encryptedBytes : new Uint8Array(encryptedBytes);
|
|
return decryptBwFileData(encrypted, sendKey.enc, sendKey.mac);
|
|
}
|
|
|
|
export function buildSendShareKey(sendKeyEncrypted: string, userEncB64: string, userMacB64: string): Promise<string> {
|
|
const userEnc = base64ToBytes(userEncB64);
|
|
const userMac = base64ToBytes(userMacB64);
|
|
return decryptBw(sendKeyEncrypted, userEnc, userMac).then((keyMaterial) => bytesToBase64Url(keyMaterial));
|
|
}
|