Files
RAPPORT-HOST/src/App.jsx
T
karim 6290475ea3 Initial: RAPPORT-HOST Iteration 1 (proprietär)
Kommerzielle Hosting-/Abo-Plattform für Rapport-Instanzen.

- React-Frontend (Vite/JSX): Landing, Register, Login, Plans, Dashboard
- Node/Express-Backend: Auth (bcrypt+JWT), Stripe-Billing, Provisioning
- HOST-Postgres-Schema: accounts, subscriptions, instances
- Provisioning-Interface + Modell-A-Adapter (Studio im geteilten Stack)
- MOCK-Modus: voller End-to-End-Flow ohne Stripe/Rapport-Stack testbar
- Idempotentes Fulfillment (Upsert auf stripe_subscription_id)
- docker-compose für lokale host-db; identisch auf Hetzner deploybar

E2E lokal verifiziert: Register -> Checkout(mock) -> Instanz -> Idempotenz.

Lizenz: proprietär (kein AGPL-Code eingebunden, nur Netzwerk-API zur Familie).

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-05-30 15:37:33 +02:00

41 lines
1.5 KiB
React

import React, { useState, useEffect, useCallback } from "react";
import { auth } from "./api.js";
import Landing from "./views/Landing.jsx";
import Register from "./views/Register.jsx";
import Login from "./views/Login.jsx";
import Plans from "./views/Plans.jsx";
import Dashboard from "./views/Dashboard.jsx";
// Minimaler Pfad-Router (kein react-router nötig für 5 Seiten).
function usePath() {
const [path, setPath] = useState(window.location.pathname);
useEffect(() => {
const onPop = () => setPath(window.location.pathname);
window.addEventListener("popstate", onPop);
return () => window.removeEventListener("popstate", onPop);
}, []);
const navigate = useCallback((to) => {
window.history.pushState({}, "", to);
setPath(to);
window.scrollTo(0, 0);
}, []);
return [path, navigate];
}
export default function App() {
const [path, navigate] = usePath();
const loggedIn = auth.isLoggedIn;
// Geschützte Seiten ohne Login → Login. Login/Register mit Login → Dashboard.
useEffect(() => {
if (path === "/dashboard" && !loggedIn) navigate("/login");
if ((path === "/login" || path === "/register") && loggedIn) navigate("/dashboard");
}, [path, loggedIn, navigate]);
if (path === "/register") return <Register navigate={navigate} />;
if (path === "/login") return <Login navigate={navigate} />;
if (path === "/plans") return <Plans navigate={navigate} />;
if (path === "/dashboard") return <Dashboard navigate={navigate} />;
return <Landing navigate={navigate} />;
}