BillTracker/services/userDataService.js

60 lines
2.9 KiB
JavaScript

'use strict';
const { ensureUserDefaultCategories } = require('../db/database');
// Independent user-owned tables (each has a user_id) wiped wholesale.
const USER_TABLES = [
'bill_merchant_rules', 'match_suggestion_rejections', 'autopay_suggestion_dismissals',
'declined_subscription_hints', 'subscription_recommendation_feedback', 'user_catalog_descriptors',
'spending_category_rules', 'spending_budgets', 'bill_templates', 'snowball_plans',
'monthly_starting_amounts', 'monthly_income', 'calendar_tokens', 'import_sessions', 'import_history',
];
/**
* Permanently erase a user's financial + imported data, resetting them to a clean
* slate — WITHOUT touching the account, sessions, 2FA/WebAuthn, login history, or
* preferences. Runs in one transaction (child → parent order so it holds with
* foreign_keys ON), then re-seeds default categories so the app stays usable and
* writes an audit row to import_history.
*
* @returns {{ erased: number, counts: Record<string, number> }}
*/
function eraseUserData(db, userId) {
const counts = {};
const del = (label, sql) => { counts[label] = db.prepare(sql).run(userId).changes; };
const run = db.transaction(() => {
// Bill-children (no user_id of their own) — remove before bills.
del('payments', 'DELETE FROM payments WHERE bill_id IN (SELECT id FROM bills WHERE user_id = ?)');
del('monthly_bill_state', 'DELETE FROM monthly_bill_state WHERE bill_id IN (SELECT id FROM bills WHERE user_id = ?)');
del('bill_history_ranges', 'DELETE FROM bill_history_ranges WHERE bill_id IN (SELECT id FROM bills WHERE user_id = ?)');
// Bank-data chain: transactions → accounts → sources.
del('transactions', 'DELETE FROM transactions WHERE user_id = ?');
del('financial_accounts', 'DELETE FROM financial_accounts WHERE user_id = ?');
del('data_sources', 'DELETE FROM data_sources WHERE user_id = ?');
for (const t of USER_TABLES) del(t, `DELETE FROM ${t} WHERE user_id = ?`);
// bills → categories → category_groups (bills.category_id, categories.group_id FKs).
del('bills', 'DELETE FROM bills WHERE user_id = ?');
del('categories', 'DELETE FROM categories WHERE user_id = ?');
del('category_groups', 'DELETE FROM category_groups WHERE user_id = ?');
});
run();
// Re-seed default categories so the app is immediately usable again.
ensureUserDefaultCategories(userId);
const erased = Object.values(counts).reduce((sum, n) => sum + n, 0);
db.prepare(`
INSERT INTO import_history (user_id, imported_at, source_filename, file_type,
rows_parsed, rows_created, rows_updated, rows_skipped, rows_ambiguous, rows_errored, options_json, summary_json)
VALUES (?, ?, 'erase-my-data', 'erase-data', ?, 0, 0, 0, 0, 0, ?, ?)
`).run(userId, new Date().toISOString(), erased, JSON.stringify({ action: 'erase-my-data' }), JSON.stringify(counts));
return { erased, counts };
}
module.exports = { eraseUserData };