27b1057cd4
Rapport ist jetzt dual: lokal (wie bisher) ODER Cloud auf eigenem Supabase-Server. Beide Modi haben dieselben Funktionen, Cloud zusätzlich Multi-User + Live-Sync. Storage-Architektur - src/storage/adapter.js: einheitliche Promise-API, LocalStorage- und SupabaseAdapter - src/storage/migrations.js: applyMigrations als reine Funktion, für beide Backends - Konfig-driven: VITE_SUPABASE_URL im Production-Build → automatisch Cloud-Modus Postgres-Schema (supabase/migrations/0001–0010) - 29 Tabellen, multi-tenant via studio_id + Row-Level-Security - Audit-Spalten (created_by/updated_by/at) + Trigger - Seed-Trigger pro neuem Studio (Rollen, Templates, Absenz-Typen) - Realtime-Publication für Live-Sync - RPCs: ensure_profile, create_studio_with_admin (mit Personen-Sharing), list_studios, load_persons_for_studio, attach_user_to_studio Cloud-Features (App) - BackendChoice.jsx als Erst-Screen «Lokal oder Cloud» - CloudSetup.jsx: 3-Schritt-Wizard für Erst-Einrichtung - Login.jsx: Modus-Switcher + Server-URL + Studio-Dropdown + Passwort-Vergessen - ResetPassword.jsx: empfängt Mail-Link-Klick via PASSWORD_RECOVERY-Event - Realtime: Änderungen zwischen Browsern ohne Reload sichtbar - Settings → System: Cloud-Verbindung, Studio-Switcher, weiteres Studio anlegen - Settings → Team: Mitarbeiter via Email einladen (Admin-Aktion) - Personen-Sharing: bei neuem Studio Personen aus anderen Studios übernehmen - Reload-Resume: studio_id in sessionStorage, kein erneuter Login nötig Web-Deploy - deploy/docker-compose.yml + nginx.conf: dist/ via nginx-Container, Port 8080 - .env.production.example: Build-time Cloud-URL - DEPLOY.md: Anleitung für LAN-only und extern via Nginx Proxy Manager Doku - README.md: Cloud-Variante prominent erklärt - ARCHITECTURE.md: Storage-Adapter, Migrations, neue Views in Risiko-Tabelle - DEPLOY.md: Schritt-für-Schritt für Mac Mini + NPM Version-Bump auf 0.8.0 in package.json, src-tauri/tauri.conf.json, Cargo.toml. Changelog-Entry im App.jsx-Modal (Karim sieht ihn beim ersten Start). Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
93 lines
4.1 KiB
JavaScript
93 lines
4.1 KiB
JavaScript
// State-Migrations für geladene `data`-Objekte.
|
|
// Extrahiert aus App.jsx, damit der Initial-Load über den (async) Adapter
|
|
// laufen kann und die Migrations sowohl auf Local- als auch Cloud-Daten
|
|
// dieselbe Form anwenden.
|
|
//
|
|
// Reine Funktion: nimmt geparste Rohdaten + defaultData entgegen, gibt das
|
|
// migrierte `data`-Objekt zurück. Keine Side-Effects, kein Storage-Zugriff.
|
|
|
|
import { migrateDashboardLayout } from "../utils.js";
|
|
|
|
export function applyMigrations(parsed, defaultData) {
|
|
let merged = {
|
|
...defaultData,
|
|
...parsed,
|
|
settings: { ...defaultData.settings, ...parsed.settings },
|
|
};
|
|
|
|
// Migrate: clients[] + contacts[] → persons[]
|
|
if (!merged.persons && (merged.clients?.length || merged.contacts?.length)) {
|
|
const idMap = {};
|
|
const persons = [];
|
|
const usedContactIds = new Set();
|
|
for (const c of merged.clients || []) {
|
|
const linked = (merged.contacts || []).find(ct => ct.id === c.linkedContactId);
|
|
persons.push({
|
|
...c,
|
|
isAuftraggeber: true,
|
|
isPartner: !!linked,
|
|
type: c.type || linked?.type || "",
|
|
note: c.note || linked?.note || "",
|
|
honorarOffers: c.honorarOffers || linked?.honorarOffers || [],
|
|
contacts: c.contacts?.length ? c.contacts : (linked?.contacts || []),
|
|
linkedContactId: undefined,
|
|
linkedClientId: undefined,
|
|
});
|
|
if (linked) { usedContactIds.add(linked.id); idMap[linked.id] = c.id; }
|
|
}
|
|
for (const ct of merged.contacts || []) {
|
|
if (usedContactIds.has(ct.id)) continue;
|
|
persons.push({ ...ct, isAuftraggeber: false, isPartner: true, linkedClientId: undefined });
|
|
}
|
|
const remapProjects = (merged.projects || []).map(p => ({
|
|
...p,
|
|
projectContacts: (p.projectContacts || []).map(pc => ({ ...pc, contactId: idMap[pc.contactId] || pc.contactId })),
|
|
}));
|
|
const remapProtocols = (merged.protocols || []).map(p => ({
|
|
...p,
|
|
entries: (p.entries || []).map(e => ({ ...e, assignee: e.assignee ? (idMap[e.assignee] || e.assignee) : e.assignee })),
|
|
}));
|
|
merged = { ...merged, persons, projects: remapProjects, protocols: remapProtocols, clients: undefined, contacts: undefined };
|
|
}
|
|
|
|
// Migrate: projects linked to SIA/manual quotes should be pauschal (not stundensatz)
|
|
const allQuotes = merged.quotes || [];
|
|
const projects = (merged.projects || []).map(p => {
|
|
if ((p.billingType || p.type || "stundensatz") === "stundensatz" && (p.linkedQuotes || []).length > 0) {
|
|
const linkedQs = (p.linkedQuotes || []).map(lq => allQuotes.find(q => q.id === lq.quoteId)).filter(Boolean);
|
|
if (linkedQs.some(q => q.mode === "sia" || q.mode === "manual")) {
|
|
return { ...p, billingType: "pauschal", budget: p.budget || p.budgetAmount || 0 };
|
|
}
|
|
}
|
|
return p;
|
|
});
|
|
|
|
// Migrate: add r-projektleiter if missing, seed dashboardTemplateId from defaultData
|
|
const roleDefMap = (defaultData.appRoles || []).reduce((acc, r) => { acc[r.id] = r; return acc; }, {});
|
|
const roles = (merged.appRoles || defaultData.appRoles).map(r => ({
|
|
...r,
|
|
dashboardTemplateId: r.dashboardTemplateId || roleDefMap[r.id]?.dashboardTemplateId || null,
|
|
permissions: (() => {
|
|
let perms = r.permissions;
|
|
if (perms && r.id === "r-projektleiter" && !perms.includes("mitarbeiter")) perms = [...perms, "mitarbeiter"];
|
|
if (perms && !perms.includes("settings")) perms = [...perms, "settings"];
|
|
return perms;
|
|
})(),
|
|
}));
|
|
if (!roles.find(r => r.id === "r-projektleiter") && roleDefMap["r-projektleiter"]) {
|
|
const adminIdx = roles.findIndex(r => r.id === "r-admin");
|
|
roles.splice(adminIdx + 1, 0, roleDefMap["r-projektleiter"]);
|
|
}
|
|
|
|
// Migrate user-level dashboardWidgets to Row[] format
|
|
const users = (merged.users || []).map(u => ({
|
|
...u,
|
|
dashboardWidgets: u.dashboardWidgets ? migrateDashboardLayout(u.dashboardWidgets) : undefined,
|
|
}));
|
|
|
|
// Ensure dashboardTemplates exist (old data won't have them)
|
|
const dashboardTemplates = merged.dashboardTemplates?.length ? merged.dashboardTemplates : defaultData.dashboardTemplates;
|
|
|
|
return { ...merged, projects, appRoles: roles, users, dashboardTemplates };
|
|
}
|