mirror of
https://github.com/screentinker/screentinker.git
synced 2026-06-15 02:33:15 -06:00
Platform admins can now cleanly remove a customer org (account ends) or a stray workspace from the UI, instead of raw SQL that risks orphaning resources. The tenant cascade isn't pure DB CASCADE - workspace-scoped tables (devices, content, playlists, ...) are NO ACTION and must be purged before the workspace. Extracted that logic out of deleteUserCascade into shared deleteWorkspaceCascade / deleteOrgCascade helpers (one tested implementation; deleteUserCascade now reuses the purgeWorkspaces extraction). Backend (platform-admin only): GET /api/admin/orgs (list + owner + counts + workspaces), DELETE /api/admin/orgs/:id, DELETE /api/admin/workspaces/:id. UI: an Organizations section in Admin listing every org/workspace with a type-the-name confirmation before the irreversible delete. Tests: org/workspace cascade (real FKs) + endpoint gating/404. Suite 53/53.
188 lines
9.1 KiB
JavaScript
188 lines
9.1 KiB
JavaScript
'use strict';
|
|
|
|
// Issue #18: deleting a user 500'd with "FOREIGN KEY constraint failed".
|
|
//
|
|
// 23 columns reference users(id) and only 4 (the *_members join tables +
|
|
// content_folders) carry ON DELETE CASCADE, so a bare `DELETE FROM users`
|
|
// fails the moment the user is referenced anywhere - and a real user always is
|
|
// (owns an org, created a workspace, has login activity). The schema also lacks
|
|
// cascades from workspaces -> tenant resources, so we cannot rely on the DB to
|
|
// clean up; we do it explicitly here in one transaction.
|
|
//
|
|
// Policy (chosen for #18):
|
|
// - Refuse (OrgHasOtherMembersError -> 409) if the user OWNS an organization
|
|
// that has any other member: don't nuke a shared tenant; transfer first.
|
|
// - Otherwise hard-delete the organizations they solely own (and everything
|
|
// inside), and for orgs they DON'T own, preserve the resources - just unlink
|
|
// the user (SET NULL where the column is nullable, or reassign the legacy
|
|
// creator user_id to the resource's org owner where it is NOT NULL).
|
|
//
|
|
// defer_foreign_keys=ON makes intra-transaction delete ORDER forgiving (FKs are
|
|
// validated once at COMMIT); we still clear every reference so COMMIT is clean.
|
|
// A table-existence guard keeps this resilient to partial/older schemas (and
|
|
// makes it unit-testable without standing up all ~25 tables).
|
|
|
|
class OrgHasOtherMembersError extends Error {
|
|
constructor(message, sharedOrgCount) {
|
|
super(message);
|
|
this.name = 'OrgHasOtherMembersError';
|
|
this.sharedOrgCount = sharedOrgCount;
|
|
}
|
|
}
|
|
|
|
// Workspace-scoped tables whose rows must be deleted before their workspace
|
|
// (workspace_id is NO ACTION). CASCADE child tables (playlist_items, telemetry,
|
|
// assignments, layout_zones, *_devices, *_members) clean themselves up.
|
|
const WORKSPACE_SCOPED = [
|
|
'playlists', 'schedules', 'video_walls', 'device_groups', 'devices',
|
|
'content', 'layouts', 'widgets', 'content_folders', 'kiosk_pages',
|
|
'white_labels', 'alert_configs',
|
|
];
|
|
// Logs that carry a device_id but NO foreign key (so they don't block, but we
|
|
// clean them to avoid dangling rows).
|
|
const DEVICE_LOG_TABLES = ['device_status_log', 'player_debug_logs'];
|
|
// Nullable creator/inviter columns -> SET NULL (preserve the resource).
|
|
const NULLABLE_USER_REFS = [
|
|
['content', 'user_id'], ['devices', 'user_id'], ['layouts', 'user_id'], ['widgets', 'user_id'],
|
|
['workspaces', 'created_by'], ['organization_members', 'invited_by'], ['workspace_members', 'invited_by'],
|
|
['team_members', 'invited_by'], ['device_fingerprints', 'user_id'],
|
|
['activity_log', 'user_id'], ['activity_log', 'acting_user_id'],
|
|
];
|
|
// NOT NULL legacy creator columns on workspace-scoped resources -> reassign to
|
|
// the resource's org owner (fallback: the acting admin) so the row survives.
|
|
const REASSIGN_USER_TABLES = [
|
|
'playlists', 'schedules', 'video_walls', 'device_groups', 'kiosk_pages', 'white_labels', 'alert_configs',
|
|
];
|
|
|
|
const inClause = n => Array.from({ length: n }, () => '?').join(',');
|
|
function tablesPresent(db) {
|
|
return new Set(db.prepare("SELECT name FROM sqlite_master WHERE type='table'").all().map(r => r.name));
|
|
}
|
|
|
|
// Delete the given workspaces and every tenant resource inside them. The
|
|
// workspace-scoped tables are NO ACTION (won't cascade from the workspace), so
|
|
// we delete them explicitly first; their CASCADE children (playlist_items,
|
|
// telemetry, assignments, layout_zones, *_devices) and workspace_members/invites
|
|
// clean themselves up. MUST run inside a transaction with defer_foreign_keys=ON.
|
|
// `have` is the set of existing table names (tablesPresent()).
|
|
function purgeWorkspaces(db, wsIds, have) {
|
|
if (!wsIds.length) return;
|
|
const wph = inClause(wsIds.length);
|
|
if (have.has('devices')) {
|
|
const devIds = db.prepare(`SELECT id FROM devices WHERE workspace_id IN (${wph})`).all(...wsIds).map(r => r.id);
|
|
if (devIds.length) {
|
|
const dph = inClause(devIds.length);
|
|
for (const lt of DEVICE_LOG_TABLES) if (have.has(lt)) db.prepare(`DELETE FROM ${lt} WHERE device_id IN (${dph})`).run(...devIds);
|
|
}
|
|
}
|
|
for (const t of WORKSPACE_SCOPED) if (have.has(t)) db.prepare(`DELETE FROM ${t} WHERE workspace_id IN (${wph})`).run(...wsIds);
|
|
if (have.has('activity_log')) db.prepare(`UPDATE activity_log SET workspace_id = NULL WHERE workspace_id IN (${wph})`).run(...wsIds);
|
|
db.prepare(`DELETE FROM workspaces WHERE id IN (${wph})`).run(...wsIds); // cascades workspace_members/invites
|
|
}
|
|
|
|
// #36: cascade-delete a single workspace (and all its tenant resources). The
|
|
// parent org is left intact. Platform-admin action; callers gate authorization.
|
|
function deleteWorkspaceCascade(db, { workspaceId }) {
|
|
db.transaction(() => {
|
|
db.pragma('defer_foreign_keys = ON');
|
|
purgeWorkspaces(db, [workspaceId], tablesPresent(db));
|
|
})();
|
|
}
|
|
|
|
// #36: cascade-delete an organization - all its workspaces + tenant resources,
|
|
// then the org itself (cascades organization_members). Member USERS are NOT
|
|
// deleted (they may belong to other orgs); they simply lose this membership.
|
|
function deleteOrgCascade(db, { orgId }) {
|
|
db.transaction(() => {
|
|
db.pragma('defer_foreign_keys = ON');
|
|
const have = tablesPresent(db);
|
|
const wsIds = db.prepare('SELECT id FROM workspaces WHERE organization_id = ?').all(orgId).map(r => r.id);
|
|
purgeWorkspaces(db, wsIds, have);
|
|
if (have.has('activity_log')) db.prepare('UPDATE activity_log SET organization_id = NULL WHERE organization_id = ?').run(orgId);
|
|
db.prepare('DELETE FROM organizations WHERE id = ?').run(orgId); // cascades organization_members
|
|
})();
|
|
}
|
|
|
|
function listOwnedOrgsWithSharing(db, userId) {
|
|
let orgs = [];
|
|
try { orgs = db.prepare('SELECT id FROM organizations WHERE owner_user_id = ?').all(userId); }
|
|
catch { return []; } // no organizations table (legacy) -> nothing owned
|
|
return orgs.map(o => {
|
|
const otherOrgMembers = db.prepare(
|
|
'SELECT COUNT(*) AS c FROM organization_members WHERE organization_id = ? AND user_id != ?'
|
|
).get(o.id, userId).c;
|
|
const otherWsMembers = db.prepare(`
|
|
SELECT COUNT(*) AS c FROM workspace_members wm
|
|
JOIN workspaces w ON w.id = wm.workspace_id
|
|
WHERE w.organization_id = ? AND wm.user_id != ?
|
|
`).get(o.id, userId).c;
|
|
return { id: o.id, shared: (otherOrgMembers + otherWsMembers) > 0 };
|
|
});
|
|
}
|
|
|
|
// Throws OrgHasOtherMembersError if the user owns a shared org. Otherwise
|
|
// deletes the user and resolves every reference in one transaction.
|
|
function deleteUserCascade(db, { targetId, actingAdminId }) {
|
|
const owned = listOwnedOrgsWithSharing(db, targetId);
|
|
const shared = owned.filter(o => o.shared);
|
|
if (shared.length > 0) {
|
|
throw new OrgHasOtherMembersError(
|
|
'User owns an organization with other members - reassign ownership before deleting',
|
|
shared.length
|
|
);
|
|
}
|
|
const soloOrgIds = owned.map(o => o.id);
|
|
|
|
const have = tablesPresent(db);
|
|
|
|
const run = db.transaction(() => {
|
|
// FK checks deferred to COMMIT: order of our deletes no longer matters, only
|
|
// that no dangling reference remains at the end.
|
|
db.pragma('defer_foreign_keys = ON');
|
|
|
|
// 1) Hard-delete the orgs the user solely owns (and everything inside).
|
|
if (soloOrgIds.length) {
|
|
const wsIds = db.prepare(
|
|
`SELECT id FROM workspaces WHERE organization_id IN (${inClause(soloOrgIds.length)})`
|
|
).all(...soloOrgIds).map(r => r.id);
|
|
purgeWorkspaces(db, wsIds, have);
|
|
|
|
const oph = inClause(soloOrgIds.length);
|
|
if (have.has('activity_log')) db.prepare(`UPDATE activity_log SET organization_id = NULL WHERE organization_id IN (${oph})`).run(...soloOrgIds);
|
|
db.prepare(`DELETE FROM organizations WHERE id IN (${oph})`).run(...soloOrgIds); // cascades organization_members
|
|
}
|
|
|
|
// 2) Unlink the user's footprint in orgs they DON'T own (rows still present).
|
|
// 2a) nullable creator/inviter columns -> SET NULL.
|
|
for (const [t, c] of NULLABLE_USER_REFS) if (have.has(t)) db.prepare(`UPDATE ${t} SET ${c} = NULL WHERE ${c} = ?`).run(targetId);
|
|
|
|
// 2b) NOT NULL legacy creator columns -> reassign to the resource's org owner
|
|
// (fallback acting admin), preserving the resource under a valid owner.
|
|
for (const t of REASSIGN_USER_TABLES) {
|
|
if (!have.has(t)) continue;
|
|
db.prepare(`
|
|
UPDATE ${t} SET user_id = COALESCE(
|
|
(SELECT o.owner_user_id FROM workspaces w JOIN organizations o ON o.id = w.organization_id WHERE w.id = ${t}.workspace_id),
|
|
?
|
|
) WHERE user_id = ?
|
|
`).run(actingAdminId, targetId);
|
|
}
|
|
|
|
// 2c) Legacy teams + NOT NULL invite rows the user owns / sent.
|
|
if (have.has('teams')) db.prepare('DELETE FROM teams WHERE owner_id = ?').run(targetId); // cascades team_members/invites
|
|
if (have.has('team_invites')) db.prepare('DELETE FROM team_invites WHERE invited_by = ?').run(targetId);
|
|
if (have.has('workspace_invites')) db.prepare('DELETE FROM workspace_invites WHERE invited_by = ?').run(targetId);
|
|
|
|
// 3) Finally the user. Their own memberships (organization_members,
|
|
// workspace_members, team_members, content_folders) CASCADE on this delete.
|
|
db.prepare('DELETE FROM users WHERE id = ?').run(targetId);
|
|
});
|
|
|
|
run();
|
|
}
|
|
|
|
module.exports = {
|
|
deleteUserCascade, OrgHasOtherMembersError, listOwnedOrgsWithSharing,
|
|
deleteWorkspaceCascade, deleteOrgCascade,
|
|
};
|