mirror of
https://github.com/kikootwo/ReadMeABook.git
synced 2026-06-03 04:40:09 +00:00
20c8fb0898
Introduce user-reported-issues and Goodreads shelf sync features and wire them into notifications. Adds Prisma migrations and schema changes (ReportedIssue, GoodreadsShelf, GoodreadsBookMapping), API endpoints for reporting (POST /audiobooks/[asin]/report-issue) and admin management (list, resolve/dismiss, replace), and an admin UI section to view/dismiss/replace reported issues. Adds a new notification event (issue_reported) with updates to notification schemas, docs and provider handling, plus a notification-events constants file. Refactors request creation to use createRequestForUser service, adds a Goodreads sync processor/service/hooks/UI modals, a scrape-resilience util, and related tests and minor integration updates.
149 lines
4.3 KiB
TypeScript
149 lines
4.3 KiB
TypeScript
/**
|
|
* Component: Encryption Service
|
|
* Documentation: documentation/backend/services/config.md
|
|
*/
|
|
|
|
import crypto from 'crypto';
|
|
|
|
const ALGORITHM = 'aes-256-gcm';
|
|
const IV_LENGTH = 16;
|
|
const AUTH_TAG_LENGTH = 16;
|
|
const KEY_LENGTH = 32;
|
|
|
|
export class EncryptionService {
|
|
private key: Buffer;
|
|
|
|
constructor() {
|
|
const encryptionKey = process.env.CONFIG_ENCRYPTION_KEY;
|
|
|
|
if (!encryptionKey) {
|
|
throw new Error('CONFIG_ENCRYPTION_KEY environment variable is not set');
|
|
}
|
|
|
|
// Ensure key is exactly 32 bytes
|
|
if (encryptionKey.length < KEY_LENGTH) {
|
|
// Pad with zeros if too short
|
|
this.key = Buffer.alloc(KEY_LENGTH);
|
|
Buffer.from(encryptionKey).copy(this.key);
|
|
} else if (encryptionKey.length > KEY_LENGTH) {
|
|
// Truncate if too long
|
|
this.key = Buffer.from(encryptionKey).subarray(0, KEY_LENGTH);
|
|
} else {
|
|
this.key = Buffer.from(encryptionKey);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Encrypt a plaintext string
|
|
* @param plaintext - The string to encrypt
|
|
* @returns Base64-encoded string in format: iv:authTag:encryptedData
|
|
*/
|
|
encrypt(plaintext: string): string {
|
|
try {
|
|
// Generate random IV for this encryption
|
|
const iv = crypto.randomBytes(IV_LENGTH);
|
|
|
|
// Create cipher
|
|
const cipher = crypto.createCipheriv(ALGORITHM, this.key, iv);
|
|
|
|
// Encrypt data
|
|
let encrypted = cipher.update(plaintext, 'utf8', 'base64');
|
|
encrypted += cipher.final('base64');
|
|
|
|
// Get auth tag
|
|
const authTag = cipher.getAuthTag();
|
|
|
|
// Combine IV, auth tag, and encrypted data
|
|
const result = `${iv.toString('base64')}:${authTag.toString('base64')}:${encrypted}`;
|
|
|
|
return result;
|
|
} catch (error) {
|
|
throw new Error(`Encryption failed: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Decrypt an encrypted string
|
|
* @param encryptedData - Base64-encoded string in format: iv:authTag:encryptedData
|
|
* @returns Decrypted plaintext string
|
|
*/
|
|
decrypt(encryptedData: string): string {
|
|
try {
|
|
// Split the encrypted data
|
|
const parts = encryptedData.split(':');
|
|
if (parts.length !== 3) {
|
|
throw new Error('Invalid encrypted data format');
|
|
}
|
|
|
|
const [ivBase64, authTagBase64, encrypted] = parts;
|
|
|
|
// Decode components
|
|
const iv = Buffer.from(ivBase64, 'base64');
|
|
const authTag = Buffer.from(authTagBase64, 'base64');
|
|
|
|
// Create decipher
|
|
const decipher = crypto.createDecipheriv(ALGORITHM, this.key, iv);
|
|
decipher.setAuthTag(authTag);
|
|
|
|
// Decrypt data
|
|
let decrypted = decipher.update(encrypted, 'base64', 'utf8');
|
|
decrypted += decipher.final('utf8');
|
|
|
|
return decrypted;
|
|
} catch (error) {
|
|
throw new Error(`Decryption failed: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Check if a value matches the format produced by encrypt().
|
|
* Validates: 3 colon-separated base64 parts where IV=16 bytes, authTag=16 bytes.
|
|
*/
|
|
isEncryptedFormat(value: string): boolean {
|
|
if (typeof value !== 'string') return false;
|
|
|
|
const parts = value.split(':');
|
|
if (parts.length !== 3) return false;
|
|
|
|
const [ivBase64, authTagBase64, encryptedBase64] = parts;
|
|
|
|
// All parts must be non-empty valid base64
|
|
const base64Regex = /^[A-Za-z0-9+/]+=*$/;
|
|
if (!ivBase64 || !authTagBase64 || !encryptedBase64) return false;
|
|
if (!base64Regex.test(ivBase64) || !base64Regex.test(authTagBase64) || !base64Regex.test(encryptedBase64)) {
|
|
return false;
|
|
}
|
|
|
|
try {
|
|
const iv = Buffer.from(ivBase64, 'base64');
|
|
const authTag = Buffer.from(authTagBase64, 'base64');
|
|
|
|
// IV and authTag must decode to exactly the expected byte lengths
|
|
if (iv.length !== IV_LENGTH) return false;
|
|
if (authTag.length !== AUTH_TAG_LENGTH) return false;
|
|
|
|
return true;
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Generate a random encryption key (32 bytes)
|
|
* @returns Base64-encoded random key
|
|
*/
|
|
static generateKey(): string {
|
|
return crypto.randomBytes(KEY_LENGTH).toString('base64');
|
|
}
|
|
}
|
|
|
|
// Singleton instance
|
|
let encryptionService: EncryptionService | null = null;
|
|
|
|
export function getEncryptionService(): EncryptionService {
|
|
if (!encryptionService) {
|
|
encryptionService = new EncryptionService();
|
|
}
|
|
return encryptionService;
|
|
}
|