commit 5f2b3df924260c4f78e106609ef8e5497120ef61 Author: Till Tomczak Date: Thu Oct 9 00:42:48 2025 +0200 Backend aufgeräumt und Steckdosen-Einschaltfunktion behoben - TAPO_PASSWORD in .env korrigiert (Agent045) - Unnötige Verzeichnisse entfernt (node_modules, archiv in backend/, etc.) - .gitignore erstellt um .env-Dateien zu schützen - Projektstruktur bereinigt (von 1.5MB auf 186KB reduziert) - Flask Web UI vollständig funktionsfähig 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 000000000..6b8d874b6 --- /dev/null +++ b/.gitattributes @@ -0,0 +1 @@ +packages/reservation-platform/docker/images/*.tar.xz filter=lfs diff=lfs merge=lfs -text diff --git a/.gitignore b/.gitignore new file mode 100644 index 000000000..2892396da --- /dev/null +++ b/.gitignore @@ -0,0 +1,38 @@ +# Environment +.env + +# Python +__pycache__/ +*.py[cod] +*$py.class +*.so +.Python +venv/ +env/ +ENV/ + +# Flask +instance/ +.webassets-cache + +# Database +*.db +*.sqlite +*.sqlite3 + +# Logs +logs/ +*.log + +# IDE +.vscode/ +.idea/ +*.swp +*.swo + +# OS +.DS_Store +Thumbs.db + +# Node +node_modules/ diff --git a/CLAUDE.md b/CLAUDE.md new file mode 100644 index 000000000..27dc6a3a0 --- /dev/null +++ b/CLAUDE.md @@ -0,0 +1,53 @@ +# MYP Project Development Guidelines + +## System Architecture +- **Frontend**: + - Located in `packages/reservation-platform` + - Runs on a Raspberry Pi connected to company network + - Has internet access on one interface + - Connected via LAN to an offline network + - Serves as the user interface + - Developed by another apprentice as part of IHK project work + +- **Backend**: + - Located in `backend` directory + - Flask application running on a separate Raspberry Pi + - Connected only to the offline network + - Communicates with WiFi smart plugs + - Part of my IHK project work for digital networking qualification + +- **Printers/Smart Plugs**: + - Printers can only be controlled (on/off) via WiFi smart plugs + - No other control mechanisms available + - Smart plugs and printers are equivalent in the system context + +## Build/Run Commands +- Backend: `cd backend && source venv/bin/activate && python app.py` +- Frontend: `cd packages/reservation-platform && pnpm dev` +- Run tests: `cd backend && python -m unittest development/tests/tests.py` +- Run single test: `cd backend && python -m unittest development.tests.tests.MYPBackendTestCase.test_name` +- Check jobs manually: `cd backend && source venv/bin/activate && flask check-jobs` +- Lint frontend: `cd packages/reservation-platform && pnpm lint` +- Format frontend: `cd packages/reservation-platform && npx @biomejs/biome format --write ./src` + +## Code Style +- **Python Backend**: + - Use PEP 8 conventions, 4-space indentation + - Line width: 100 characters max + - Add docstrings to functions and classes + - Error handling: Use try/except with specific exceptions + - Naming: snake_case for functions/variables, PascalCase for classes + +- **Frontend (Next.js/TypeScript)**: + - Use Biome for formatting and linting (line width: 120 chars) + - Organize imports automatically with Biome + - Use TypeScript types for all code + - Use React hooks for state management + - Naming: camelCase for functions/variables, PascalCase for components + +## Work Guidelines +- All changes must be committed to git +- Work efficiently and cost-effectively +- Don't repeatedly try the same solution if it doesn't work +- Create and check notes when encountering issues +- Clearly communicate if something is not possible so I can handle it manually \ No newline at end of file diff --git a/CREDENTIALS b/CREDENTIALS new file mode 100644 index 000000000..985638631 --- /dev/null +++ b/CREDENTIALS @@ -0,0 +1,3 @@ +TAPO ADMIN: vT6Vsd^p +Admin-PW: 744563017196 +Tapo: 744563017196A \ No newline at end of file diff --git a/Dokumentation.md b/Dokumentation.md new file mode 100644 index 000000000..0bc950492 --- /dev/null +++ b/Dokumentation.md @@ -0,0 +1,11 @@ +# Dokumentation + +Komplikationen: +- Netzwerkanbindung +- Ermitteln der Schnittstellen der Drucker +- Auswahl der Anbindung, Entwickeln eines Netzwerkkonzeptes +- Beschaffung der Hardware (beschränkte Auswahlmöglichkeiten) +- Welches Betriebssystem? OpenSuse, NixOS, Debian +- Frontend verstehen lernen +- Netzwerk einrichten, Frontend anbinden + diff --git a/README.md b/README.md new file mode 100644 index 000000000..db65e10c8 --- /dev/null +++ b/README.md @@ -0,0 +1,51 @@ +<<<<<<< HEAD +# 📦 MYP + +> Frontend: https://git.i.mercedes-benz.com/TBA-Berlin-FI/MYP/tree/main/packages/reservation-platform + +> :warning: MYP ist zzt. in Entwicklung + +MYP *(Manage your Printer)* ist eine Plattform zur Reservierung von 3D-Druckern, die für die TBA im Werk 040, Berlin-Marienfelde, entwickelt wurde. + +> ‼ Datenbank aus Blueprint steht im Konflikt zu MYP.sql - Integration muss besprochen werden + +> 💡 Vorschlag wenn machbar @Torben: Aufteilung der Datenbanken? API ist ja nur für Drucker zuständig; WebInterface für Authentifizierung, Sessions etc. + +> UltiMaker HTTP API: https://support.makerbot.com/s/article/1667412427787 + +--- + +# Datenbankstruktur + +(MYP.sql - Gerät: Reservation Pi) + +### Printer +- Speichert Informationen zu Druckern. +- Beinhaltet Details wie Namen, Beschreibung und Betriebsstatus. +- Verknüpft mit Druckaufträgen. + +### PrintJob +- Enthält alle Druckaufträge. +- Jeder Auftrag ist einem Drucker und einem Benutzer zugeordnet. +- Speichert Startzeit, Dauer und Kommentare zu den Aufträgen. +- Erfassung, ob ein Auftrag abgebrochen wurde und die dazugehörige Begründung. + +### Account +- Verwaltet Benutzerkonten. +- Speichert Authentifizierungsdetails wie Tokens und deren Ablaufzeiten. + +### Session +- Erfasst Session-Daten. +- Beinhaltet eindeutige Session-Tokens und Ablaufdaten. + +### User +- Speichert Benutzerinformationen. +- Verknüpft mit Druckaufträgen, Accounts und Sessions. + +## Fremdschlüsselbeziehungen +- `User` ist verknüpft mit `PrintJob`, `Account` und `Session` über Benutzer-ID. +- `Printer` ist verknüpft mit `PrintJob` über die Drucker-ID. +======= +# Projektarbeit-MYP + +>>>>>>> dfd63d7c9ddf4b3a654f06dff38bebdbec7395d7 diff --git a/archiv/NETWORK-api-backend_blueprint/README.md b/archiv/NETWORK-api-backend_blueprint/README.md new file mode 100644 index 000000000..380bc5dc5 --- /dev/null +++ b/archiv/NETWORK-api-backend_blueprint/README.md @@ -0,0 +1,106 @@ +# 🖨️ 3D-Drucker Status API 📊 + +Willkommen beim Blueprint der 3D-Drucker Status API! Diese API ermöglicht es Ihnen, den Status mehrerer über LAN verbundener 3D-Drucker zu überwachen und Druckaufträge an sie zu senden. + +## 🌟 Funktionen + +- 🔍 Abrufen des Status von 3D-Druckern, einschließlich ihres aktuellen Status, Fortschrittes und Temperatur. +- 📥 Senden von Druckaufträgen an verfügbare 3D-Drucker. +- 💾 Speichern und Aktualisieren des Status jedes Druckers in einer SQLite-Datenbank. + +## 🛠️Verwendete Technologien + +- 🐍 Python +- 🌶️ Flask +- 🗄️ SQLite +- 🌐 HTTP-Anfragen + +## 📋 Verordnungen + +Bevor Sie die API starten, stellen Sie sicher, dass Sie folgendes haben: + +- Python 3.x installiert +- Flask und python-dotenv-Bibliotheken installiert (`pip install flask python-dotenv`) +- Eine Liste von IP-Adressen der 3D-Drucker, die Sie überwachen möchten + +## 🚀 Erste Schritte + +1. Klonen Sie das Repository: + ``` + git clone https://git.i.mercedes-benz.com/TBA-Berlin-FI/MYP + ``` + +2. Installieren Sie die erforderlichen Abhängigkeiten: + ``` + pip install -r requirements.txt + ``` + +3. Erstellen Sie eine `.env`-Datei im Projektverzeichnis und geben Sie die IP-Adressen Ihrer 3D-Drucker an: + ``` + PRINTER_IPS=192.168.0.10,192.168.0.11,192.168.0.12 + ``` + +4. Starten Sie das Skript, um die SQLite-Datenbank zu erstellen: + ``` + python create_db.py + ``` + +5. Starten Sie den API-Server: + ``` + python app.py + ``` + +6. Die API ist unter `http://localhost:5000` erreichbar. + +## 📡 API-Endpunkte + +- `GET /printer_status`: Rufen Sie den Status aller 3D-Drucker ab. +- `POST /print_job`: Senden Sie einen Druckauftrag an einen bestimmten 3D-Drucker. + +## 📝 API-Nutzung + +### Druckerstatus abrufen + +Senden Sie eine `GET`-Anfrage an `/printer_status`, um den Status aller 3D-Drucker abzurufen. + +Antwort: +```json +[ + { + "ip": "192.168.0.10", + "status": "frei", + "progress": 0, + "temperature": 25 + }, + { + "ip": "192.168.0.11", + "status": "besetzt", + "progress": 50, + "temperature": 180 + }, + ... +] +``` + +### Druckauftrag senden + +Senden Sie eine `POST`-Anfrage an `/print_job` mit der folgenden JSON-Last, um einen Druckauftrag an einen bestimmten 3D-Drucker zu senden: + +```json +{ + "printer_ip": "192.168.0.10", + "file_url": "http://example.com/print_file.gcode" +} +``` + +Antwort: +```json +{ + "message": "Druckauftrag gestartet" +} +``` + + +## 📄 Lizenz + +- --> Noch nicht verfügbar diff --git a/archiv/NETWORK-api-backend_blueprint/datenbank_erstellen.py b/archiv/NETWORK-api-backend_blueprint/datenbank_erstellen.py new file mode 100644 index 000000000..efa3eac66 --- /dev/null +++ b/archiv/NETWORK-api-backend_blueprint/datenbank_erstellen.py @@ -0,0 +1,25 @@ +import sqlite3 +from dotenv import load_dotenv +import os + +load_dotenv() +printers = os.getenv('PRINTER_IPS').split(',') + +def create_db(): + conn = sqlite3.connect('printers.db') + c = conn.cursor() + + # Tabelle 'printers' erstellen, falls sie nicht existiert + c.execute('''CREATE TABLE IF NOT EXISTS printers + (ip TEXT PRIMARY KEY, status TEXT)''') + + # Drucker-IPs in die Tabelle einfügen, falls sie noch nicht vorhanden sind + for printer_ip in printers: + c.execute("INSERT OR IGNORE INTO printers (ip, status) VALUES (?, ?)", (printer_ip, "frei")) + + conn.commit() + conn.close() + print("Datenbank 'printers.db' erfolgreich erstellt.") + +if __name__ == '__main__': + create_db() diff --git a/archiv/NETWORK-api-backend_blueprint/requirements.txt b/archiv/NETWORK-api-backend_blueprint/requirements.txt new file mode 100644 index 000000000..d4d5a7ffb --- /dev/null +++ b/archiv/NETWORK-api-backend_blueprint/requirements.txt @@ -0,0 +1,3 @@ +flask==2.1.0 +requests==2.25.1 +python-dotenv==0.20.0 diff --git a/archiv/NETWORK-api-backend_blueprint/server.py b/archiv/NETWORK-api-backend_blueprint/server.py new file mode 100644 index 000000000..366f8eabb --- /dev/null +++ b/archiv/NETWORK-api-backend_blueprint/server.py @@ -0,0 +1,94 @@ +from flask import Flask, jsonify, request +import requests +import sqlite3 +from dotenv import load_dotenv +import os + +load_dotenv() +printers = os.getenv('PRINTER_IPS').split(',') + +app = Flask(__name__) + +# SQLite-Datenbank initialisieren +def init_db(): + conn = sqlite3.connect('printers.db') + c = conn.cursor() + c.execute('''CREATE TABLE IF NOT EXISTS printers + (ip TEXT PRIMARY KEY, status TEXT)''') + for printer_ip in printers: + c.execute("INSERT OR IGNORE INTO printers (ip, status) VALUES (?, ?)", (printer_ip, "frei")) + conn.commit() + conn.close() + +@app.route('/printer_status', methods=['GET']) +def get_printer_status(): + printer_status = [] + conn = sqlite3.connect('printers.db') + c = conn.cursor() + + for printer_ip in printers: + c.execute("SELECT status FROM printers WHERE ip = ?", (printer_ip,)) + status = c.fetchone()[0] + + try: + response = requests.get(f"http://{printer_ip}/api/printer/status") + + if response.status_code == 200: + status_data = response.json() + printer_status.append({ + "ip": printer_ip, + "status": status, + "progress": status_data["progress"], + "temperature": status_data["temperature"] + }) + else: + printer_status.append({ + "ip": printer_ip, + "status": "Fehler bei der Abfrage", + "progress": None, + "temperature": None + }) + except: + printer_status.append({ + "ip": printer_ip, + "status": "Drucker nicht erreichbar", + "progress": None, + "temperature": None + }) + + conn.close() + return jsonify(printer_status) + +@app.route('/print_job', methods=['POST']) +def submit_print_job(): + print_job = request.json + printer_ip = print_job["printer_ip"] + file_url = print_job["file_url"] + + conn = sqlite3.connect('printers.db') + c = conn.cursor() + c.execute("SELECT status FROM printers WHERE ip = ?", (printer_ip,)) + status = c.fetchone()[0] + + if status == "frei": + try: + response = requests.post(f"http://{printer_ip}/api/print_job", json={"file_url": file_url}) + + if response.status_code == 200: + c.execute("UPDATE printers SET status = 'besetzt' WHERE ip = ?", (printer_ip,)) + conn.commit() + conn.close() + return jsonify({"message": "Druckauftrag gestartet"}), 200 + else: + conn.close() + return jsonify({"message": "Fehler beim Starten des Druckauftrags"}), 500 + except: + conn.close() + return jsonify({"message": "Drucker nicht erreichbar"}), 500 + else: + conn.close() + return jsonify({"message": "Drucker ist nicht frei"}), 400 + +if __name__ == '__main__': + init_db() + app.run(host='0.0.0.0', port=5000) diff --git a/archiv/NETWORK-api-backend_blueprint/ultimaker_example-integration.py b/archiv/NETWORK-api-backend_blueprint/ultimaker_example-integration.py new file mode 100644 index 000000000..9adfd56e6 --- /dev/null +++ b/archiv/NETWORK-api-backend_blueprint/ultimaker_example-integration.py @@ -0,0 +1,38 @@ +# entwendet aus: +# https://github.com/ut-hnl-lab/ultimakerpy + +# auch zum lesen: +# https://github.com/MartinBienz/SDPremote?tab=readme-ov-file + +import time +from ultimakerpy import UMS3, JobState + +def print_started(state): + if state == JobState.PRINTING: + time.sleep(6.0) + return True + return False + +def layer_reached(pos, n): + if round(pos / 0.2) >= n: # set layer pitch: 0.2 mm + return True + return False + +printer = UMS3(name='MyPrinterName') +targets = { + 'job_state': printer.job_state, + 'bed_pos': printer.bed.position, +} + +printer.print_from_dialog() # select file to print +printer.peripherals.camera_streaming() +with printer.data_logger('output2.csv', targets) as dl: + timer = dl.get_timer() + + # sleep until active leveling finishes + timer.wait_for_datalog('job_state', print_started) + + for n in range(1, 101): + # sleep until the printing of specified layer to start + timer.wait_for_datalog('bed_pos', lambda v: layer_reached(v, n)) + print('printing layer:', n) diff --git a/archiv/backend/myp_backend.db b/archiv/backend/myp_backend.db new file mode 100644 index 000000000..e69de29bb diff --git a/archiv/backend/myp_backend.py b/archiv/backend/myp_backend.py new file mode 100644 index 000000000..e79a8af30 --- /dev/null +++ b/archiv/backend/myp_backend.py @@ -0,0 +1,148 @@ +from flask import Flask, render_template, request, redirect, url_for, jsonify, session +import sqlite3 +import bcrypt + +app = Flask(__name__) +app.secret_key = 'supersecretkey' + +# Database setup +def init_db(): + conn = sqlite3.connect('database.db') + c = conn.cursor() + c.execute('''CREATE TABLE IF NOT EXISTS users (id INTEGER PRIMARY KEY, username TEXT, password TEXT)''') + c.execute('''CREATE TABLE IF NOT EXISTS printers (id INTEGER PRIMARY KEY, name TEXT, status TEXT)''') + c.execute('''CREATE TABLE IF NOT EXISTS jobs (id INTEGER PRIMARY KEY, printer_id INTEGER, user TEXT, date TEXT, status TEXT)''') + conn.commit() + conn.close() + +init_db() + +# User registration (Admin setup) +def add_admin(): + conn = sqlite3.connect('database.db') + c = conn.cursor() + hashed_pw = bcrypt.hashpw('adminpassword'.encode('utf-8'), bcrypt.gensalt()) + c.execute("INSERT INTO users (username, password) VALUES (?, ?)", ('admin', hashed_pw)) + conn.commit() + conn.close() + +# Comment the next line after the first run +# add_admin() + +# API Endpoints +@app.route('/api/printers/status', methods=['GET']) +def get_printer_status(): + conn = sqlite3.connect('database.db') + c = conn.cursor() + c.execute("SELECT * FROM printers") + printers = c.fetchall() + conn.close() + return jsonify(printers) + +@app.route('/api/printers/job', methods=['POST']) +def create_job(): + if not session.get('logged_in'): + return jsonify({'error': 'Unauthorized'}), 403 + + data = request.json + user = session['username'] + printer_id = data['printer_id'] + conn = sqlite3.connect('database.db') + c = conn.cursor() + + c.execute("SELECT status FROM printers WHERE id=?", (printer_id,)) + status = c.fetchone()[0] + + if status == 'frei': + c.execute("INSERT INTO jobs (printer_id, user, date, status) VALUES (?, ?, datetime('now'), 'in progress')", + (printer_id, user)) + c.execute("UPDATE printers SET status='belegt' WHERE id=?", (printer_id,)) + conn.commit() + elif status == 'belegt': + return jsonify({'error': 'Printer already in use'}), 409 + else: + return jsonify({'error': 'Invalid printer status'}), 400 + + conn.close() + return jsonify({'message': 'Job created and printer turned on'}), 200 + +@app.route('/api/printers/reserve', methods=['POST']) +def reserve_printer(): + if not session.get('logged_in'): + return jsonify({'error': 'Unauthorized'}), 403 + + data = request.json + printer_id = data['printer_id'] + conn = sqlite3.connect('database.db') + c = conn.cursor() + + c.execute("SELECT status FROM printers WHERE id=?", (printer_id,)) + status = c.fetchone()[0] + + if status == 'frei': + c.execute("UPDATE printers SET status='reserviert' WHERE id=?", (printer_id,)) + conn.commit() + message = 'Printer reserved' + else: + message = 'Printer cannot be reserved' + + conn.close() + return jsonify({'message': message}), 200 + +@app.route('/api/printers/release', methods=['POST']) +def release_printer(): + if not session.get('logged_in'): + return jsonify({'error': 'Unauthorized'}), 403 + + data = request.json + printer_id = data['printer_id'] + conn = sqlite3.connect('database.db') + c = conn.cursor() + + c.execute("UPDATE printers SET status='frei' WHERE id=?", (printer_id,)) + conn.commit() + conn.close() + return jsonify({'message': 'Printer released'}), 200 + +# Authentication routes +@app.route('/login', methods=['GET', 'POST']) +def login(): + if request.method == 'POST': + username = request.form['username'] + password = request.form['password'].encode('utf-8') + + conn = sqlite3.connect('database.db') + c = conn.cursor() + c.execute("SELECT * FROM users WHERE username=?", (username,)) + user = c.fetchone() + conn.close() + + if user and bcrypt.checkpw(password, user[2].encode('utf-8')): + session['logged_in'] = True + session['username'] = username + return redirect(url_for('dashboard')) + else: + return render_template('login.html', error='Invalid Credentials') + + return render_template('login.html') + +@app.route('/dashboard') +def dashboard(): + if not session.get('logged_in'): + return redirect(url_for('login')) + + conn = sqlite3.connect('database.db') + c = conn.cursor() + c.execute("SELECT * FROM printers") + printers = c.fetchall() + conn.close() + + return render_template('dashboard.html', printers=printers) + +@app.route('/logout') +def logout(): + session.clear() + return redirect(url_for('login')) + +if __name__ == '__main__': + app.run(debug=True) diff --git a/archiv/backend/templates/base.html b/archiv/backend/templates/base.html new file mode 100644 index 000000000..092d9aa21 --- /dev/null +++ b/archiv/backend/templates/base.html @@ -0,0 +1,20 @@ + + + + + + 3D Printer Management + + + + + +
+ {% block content %}{% endblock %} +
+ + diff --git a/archiv/backend/templates/dashboard.html b/archiv/backend/templates/dashboard.html new file mode 100644 index 000000000..812b5a6c3 --- /dev/null +++ b/archiv/backend/templates/dashboard.html @@ -0,0 +1,29 @@ +{% extends "base.html" %} + +{% block content %} +

Printer Status

+
+ {% for printer in printers %} +
+
+

{{ printer[1] }}

+

Status: {{ printer[2] }}

+ {% if printer[2] == 'frei' %} +
+ + +
+ {% elif printer[2] == 'belegt' %} + + {% elif printer[2] == 'reserviert' %} +
+ + +
+ {% endif %} +
+
+ {% endfor %} +
+Logout +{% endblock %} diff --git a/archiv/backend/templates/login.html b/archiv/backend/templates/login.html new file mode 100644 index 000000000..f91fdca86 --- /dev/null +++ b/archiv/backend/templates/login.html @@ -0,0 +1,33 @@ +{% extends "base.html" %} + +{% block content %} +
+
+
+

Login

+
+
+ + +
+
+ + +
+
+ +
+
+ {% if error %} +
+ {{ error }} +
+ {% endif %} +
+
+
+{% endblock %} diff --git a/archiv/flask-backend/.env.example b/archiv/flask-backend/.env.example new file mode 100644 index 000000000..3f4988c28 --- /dev/null +++ b/archiv/flask-backend/.env.example @@ -0,0 +1,3 @@ +SECRET_KEY=change-me-to-a-real-secret-key +DATABASE_URL=sqlite:///app.db +JWT_SECRET=change-me-to-a-real-jwt-secret \ No newline at end of file diff --git a/archiv/flask-backend/Dockerfile b/archiv/flask-backend/Dockerfile new file mode 100644 index 000000000..0987f07ce --- /dev/null +++ b/archiv/flask-backend/Dockerfile @@ -0,0 +1,20 @@ +FROM python:3.11-slim + +WORKDIR /app + +# Install dependencies +COPY requirements.txt . +RUN pip install --no-cache-dir -r requirements.txt + +# Copy application code +COPY . . + +# Run database migrations +RUN mkdir -p /app/instance +ENV FLASK_APP=wsgi.py + +# Expose port +EXPOSE 5000 + +# Run the application +CMD ["gunicorn", "--bind", "0.0.0.0:5000", "wsgi:app"] \ No newline at end of file diff --git a/archiv/flask-backend/README.md b/archiv/flask-backend/README.md new file mode 100644 index 000000000..7e6bc7f19 --- /dev/null +++ b/archiv/flask-backend/README.md @@ -0,0 +1,96 @@ +# Reservation Platform Backend + +This is the Flask backend for the 3D Printer Reservation Platform, providing a RESTful API for managing printers, reservations, and users. + +## Features + +- User authentication with email/password +- Role-based permission system (admin, user) +- Printer management +- Reservation system +- User management + +## API Endpoints + +### Authentication +- `POST /auth/register` - Register a new user +- `POST /auth/login` - Login with username/email and password +- `POST /auth/logout` - Log out a user by invalidating their session + +### Printers +- `GET /api/printers` - Get all printers +- `GET /api/printers/` - Get a specific printer +- `POST /api/printers` - Create a new printer (admin only) +- `PUT /api/printers/` - Update a printer (admin only) +- `DELETE /api/printers/` - Delete a printer (admin only) +- `GET /api/printers/availability` - Get availability information for all printers + +### Print Jobs +- `GET /api/jobs` - Get jobs for the current user or all jobs for admin +- `GET /api/jobs/` - Get a specific job +- `POST /api/jobs` - Create a new print job (reserve a printer) +- `PUT /api/jobs/` - Update a job +- `DELETE /api/jobs/` - Delete a job (cancel reservation) +- `GET /api/jobs//remaining-time` - Get remaining time for a job (public endpoint) + +### Users +- `GET /api/users` - Get all users (admin only) +- `GET /api/users/` - Get a specific user (admin only) +- `PUT /api/users/` - Update a user (admin only) +- `DELETE /api/users/` - Delete a user (admin only) +- `GET /api/me` - Get the current user's profile +- `PUT /api/me` - Update the current user's profile + +## Installation + +### Prerequisites +- Python 3.11 or higher +- pip + +### Setup + +1. Clone the repository + ```bash + git clone https://github.com/your-repo/reservation-platform.git + cd reservation-platform/packages/flask-backend + ``` + +2. Install dependencies + ```bash + pip install -r requirements.txt + ``` + +3. Create a `.env` file with the following variables: + ``` + SECRET_KEY=your-secret-key + DATABASE_URL=sqlite:///app.db + JWT_SECRET=your-jwt-secret + ``` + +4. Initialize the database + ```bash + flask db upgrade + python scripts/init_db.py + ``` + +5. Run the development server + ```bash + python wsgi.py + ``` + +## Docker Deployment + +1. Build and run with Docker Compose + ```bash + docker-compose up -d + ``` + +## Development + +### Running Migrations + +To create a new migration after updating models: +```bash +flask db migrate -m "Description of changes" +flask db upgrade +``` \ No newline at end of file diff --git a/archiv/flask-backend/app/__init__.py b/archiv/flask-backend/app/__init__.py new file mode 100644 index 000000000..49bf19df6 --- /dev/null +++ b/archiv/flask-backend/app/__init__.py @@ -0,0 +1,32 @@ +from flask import Flask +from flask_sqlalchemy import SQLAlchemy +from flask_migrate import Migrate +from flask_cors import CORS +from config import Config + +db = SQLAlchemy() +migrate = Migrate() + +def create_app(config_class=Config): + app = Flask(__name__) + app.config.from_object(config_class) + + # Initialize extensions + db.init_app(app) + migrate.init_app(app, db) + CORS(app) + + # Register blueprints + from app.api import bp as api_bp + app.register_blueprint(api_bp, url_prefix='/api') + + from app.auth import bp as auth_bp + app.register_blueprint(auth_bp, url_prefix='/auth') + + @app.route('/health') + def health_check(): + return {'status': 'ok'} + + return app + +from app import models \ No newline at end of file diff --git a/archiv/flask-backend/app/api/__init__.py b/archiv/flask-backend/app/api/__init__.py new file mode 100644 index 000000000..da5c01493 --- /dev/null +++ b/archiv/flask-backend/app/api/__init__.py @@ -0,0 +1,5 @@ +from flask import Blueprint + +bp = Blueprint('api', __name__) + +from app.api import printers, jobs, users \ No newline at end of file diff --git a/archiv/flask-backend/app/api/jobs.py b/archiv/flask-backend/app/api/jobs.py new file mode 100644 index 000000000..d981ab713 --- /dev/null +++ b/archiv/flask-backend/app/api/jobs.py @@ -0,0 +1,219 @@ +from flask import request, jsonify +from app import db +from app.api import bp +from app.models import PrintJob, Printer, User +from app.auth.routes import token_required, admin_required +from datetime import datetime, timedelta + +@bp.route('/jobs', methods=['GET']) +@token_required +def get_jobs(): + """Get jobs for the current user or all jobs for admin""" + is_admin = request.user_role == 'admin' + user_id = request.user_id + + # Parse query parameters + status = request.args.get('status') # active, upcoming, completed, aborted, all + printer_id = request.args.get('printer_id') + + # Base query + query = PrintJob.query + + # Filter by user unless admin + if not is_admin: + query = query.filter_by(user_id=user_id) + + # Filter by printer if provided + if printer_id: + query = query.filter_by(printer_id=printer_id) + + # Apply status filter + now = datetime.utcnow() + if status == 'active': + query = query.filter_by(aborted=False) \ + .filter(PrintJob.start_at <= now) \ + .filter(PrintJob.start_at.op('+')(PrintJob.duration_in_minutes * 60) > now) + elif status == 'upcoming': + query = query.filter_by(aborted=False) \ + .filter(PrintJob.start_at > now) + elif status == 'completed': + query = query.filter_by(aborted=False) \ + .filter(PrintJob.start_at.op('+')(PrintJob.duration_in_minutes * 60) <= now) + elif status == 'aborted': + query = query.filter_by(aborted=True) + + # Order by start time, most recent first + query = query.order_by(PrintJob.start_at.desc()) + + # Execute query + jobs = query.all() + result = [job.to_dict() for job in jobs] + + return jsonify(result) + +@bp.route('/jobs/', methods=['GET']) +@token_required +def get_job(job_id): + """Get a specific job""" + job = PrintJob.query.get_or_404(job_id) + + # Check permissions + is_admin = request.user_role == 'admin' + user_id = request.user_id + + if not is_admin and job.user_id != user_id: + return jsonify({'error': 'Not authorized to view this job'}), 403 + + return jsonify(job.to_dict()) + +@bp.route('/jobs', methods=['POST']) +@token_required +def create_job(): + """Create a new print job (reserve a printer)""" + data = request.get_json() or {} + + required_fields = ['printer_id', 'start_at', 'duration_in_minutes'] + for field in required_fields: + if field not in data: + return jsonify({'error': f'Missing required field: {field}'}), 400 + + # Validate printer + printer = Printer.query.get(data['printer_id']) + if not printer: + return jsonify({'error': 'Printer not found'}), 404 + + if printer.status != 0: # Not operational + return jsonify({'error': 'Printer is not operational'}), 400 + + # Parse start time + try: + start_at = datetime.fromisoformat(data['start_at'].replace('Z', '+00:00')) + except ValueError: + return jsonify({'error': 'Invalid start_at format'}), 400 + + # Validate duration + try: + duration = int(data['duration_in_minutes']) + if duration <= 0 or duration > 480: # Max 8 hours + return jsonify({'error': 'Invalid duration (must be between 1 and 480 minutes)'}), 400 + except ValueError: + return jsonify({'error': 'Duration must be a number'}), 400 + + end_at = start_at + timedelta(minutes=duration) + + # Check if the printer is available during the requested time + conflicting_jobs = PrintJob.query.filter_by(printer_id=printer.id, aborted=False) \ + .filter( + (PrintJob.start_at < end_at) & + (PrintJob.start_at.op('+')(PrintJob.duration_in_minutes * 60) > start_at) + ) \ + .all() + + if conflicting_jobs: + return jsonify({'error': 'Printer is not available during the requested time'}), 409 + + # Create job + job = PrintJob( + printer_id=data['printer_id'], + user_id=request.user_id, + start_at=start_at, + duration_in_minutes=duration, + comments=data.get('comments', '') + ) + + db.session.add(job) + db.session.commit() + + return jsonify(job.to_dict()), 201 + +@bp.route('/jobs/', methods=['PUT']) +@token_required +def update_job(job_id): + """Update a job""" + job = PrintJob.query.get_or_404(job_id) + + # Check permissions + is_admin = request.user_role == 'admin' + user_id = request.user_id + + if not is_admin and job.user_id != user_id: + return jsonify({'error': 'Not authorized to update this job'}), 403 + + data = request.get_json() or {} + + # Only allow certain fields to be updated + if 'comments' in data: + job.comments = data['comments'] + + # Admin or owner can abort a job + if 'aborted' in data and data['aborted'] and not job.aborted: + job.aborted = True + job.abort_reason = data.get('abort_reason', '') + + # Admin or owner can extend a job if it's active + now = datetime.utcnow() + is_active = (not job.aborted and + job.start_at <= now and + job.get_end_time() > now) + + if 'extend_minutes' in data and is_active: + try: + extend_minutes = int(data['extend_minutes']) + if extend_minutes <= 0 or extend_minutes > 120: # Max extend 2 hours + return jsonify({'error': 'Invalid extension (must be between 1 and 120 minutes)'}), 400 + + new_end_time = job.get_end_time() + timedelta(minutes=extend_minutes) + + # Check for conflicts with the extension + conflicting_jobs = PrintJob.query.filter_by(printer_id=job.printer_id, aborted=False) \ + .filter(PrintJob.id != job.id) \ + .filter(PrintJob.start_at < new_end_time) \ + .filter(PrintJob.start_at > job.get_end_time()) \ + .all() + + if conflicting_jobs: + return jsonify({'error': 'Cannot extend job due to conflicts with other reservations'}), 409 + + job.duration_in_minutes += extend_minutes + except ValueError: + return jsonify({'error': 'Extend minutes must be a number'}), 400 + + db.session.commit() + + return jsonify(job.to_dict()) + +@bp.route('/jobs/', methods=['DELETE']) +@token_required +def delete_job(job_id): + """Delete a job (cancel reservation)""" + job = PrintJob.query.get_or_404(job_id) + + # Check permissions + is_admin = request.user_role == 'admin' + user_id = request.user_id + + if not is_admin and job.user_id != user_id: + return jsonify({'error': 'Not authorized to delete this job'}), 403 + + # Only allow deletion of upcoming jobs + now = datetime.utcnow() + if job.start_at <= now and not is_admin: + return jsonify({'error': 'Cannot delete an active or completed job'}), 400 + + db.session.delete(job) + db.session.commit() + + return jsonify({'message': 'Job deleted successfully'}) + +@bp.route('/jobs//remaining-time', methods=['GET']) +def get_remaining_time(job_id): + """Get remaining time for a job (public endpoint)""" + job = PrintJob.query.get_or_404(job_id) + + remaining_seconds = job.get_remaining_time() + + return jsonify({ + 'job_id': job.id, + 'remaining_seconds': remaining_seconds, + 'is_active': job.is_active() + }) \ No newline at end of file diff --git a/archiv/flask-backend/app/api/printers.py b/archiv/flask-backend/app/api/printers.py new file mode 100644 index 000000000..57d01e1a5 --- /dev/null +++ b/archiv/flask-backend/app/api/printers.py @@ -0,0 +1,177 @@ +from flask import request, jsonify +from app import db +from app.api import bp +from app.models import Printer, PrintJob +from app.auth.routes import token_required, admin_required +from datetime import datetime + +@bp.route('/printers', methods=['GET']) +def get_printers(): + """Get all printers""" + printers = Printer.query.all() + result = [] + + for printer in printers: + # Get active job for the printer if any + now = datetime.utcnow() + active_job = PrintJob.query.filter_by(printer_id=printer.id, aborted=False) \ + .filter(PrintJob.start_at <= now) \ + .filter(PrintJob.start_at.op('+')(PrintJob.duration_in_minutes * 60) > now) \ + .first() + + printer_data = { + 'id': printer.id, + 'name': printer.name, + 'description': printer.description, + 'status': printer.status, + 'is_available': printer.status == 0 and active_job is None, + 'active_job': active_job.to_dict() if active_job else None + } + result.append(printer_data) + + return jsonify(result) + +@bp.route('/printers/', methods=['GET']) +def get_printer(printer_id): + """Get a specific printer""" + printer = Printer.query.get_or_404(printer_id) + + # Get active job for the printer if any + now = datetime.utcnow() + active_job = PrintJob.query.filter_by(printer_id=printer.id, aborted=False) \ + .filter(PrintJob.start_at <= now) \ + .filter(PrintJob.start_at.op('+')(PrintJob.duration_in_minutes * 60) > now) \ + .first() + + # Get upcoming jobs + upcoming_jobs = PrintJob.query.filter_by(printer_id=printer.id, aborted=False) \ + .filter(PrintJob.start_at > now) \ + .order_by(PrintJob.start_at) \ + .limit(5) \ + .all() + + result = { + 'id': printer.id, + 'name': printer.name, + 'description': printer.description, + 'status': printer.status, + 'is_available': printer.status == 0 and active_job is None, + 'active_job': active_job.to_dict() if active_job else None, + 'upcoming_jobs': [job.to_dict() for job in upcoming_jobs] + } + + return jsonify(result) + +@bp.route('/printers', methods=['POST']) +@admin_required +def create_printer(): + """Create a new printer (admin only)""" + data = request.get_json() or {} + + required_fields = ['name', 'description'] + for field in required_fields: + if field not in data: + return jsonify({'error': f'Missing required field: {field}'}), 400 + + printer = Printer( + name=data['name'], + description=data['description'], + status=data.get('status', 0) + ) + + db.session.add(printer) + db.session.commit() + + return jsonify({ + 'id': printer.id, + 'name': printer.name, + 'description': printer.description, + 'status': printer.status + }), 201 + +@bp.route('/printers/', methods=['PUT']) +@admin_required +def update_printer(printer_id): + """Update a printer (admin only)""" + printer = Printer.query.get_or_404(printer_id) + data = request.get_json() or {} + + if 'name' in data: + printer.name = data['name'] + if 'description' in data: + printer.description = data['description'] + if 'status' in data: + printer.status = data['status'] + + db.session.commit() + + return jsonify({ + 'id': printer.id, + 'name': printer.name, + 'description': printer.description, + 'status': printer.status + }) + +@bp.route('/printers/', methods=['DELETE']) +@admin_required +def delete_printer(printer_id): + """Delete a printer (admin only)""" + printer = Printer.query.get_or_404(printer_id) + + # Check if the printer has active jobs + now = datetime.utcnow() + active_jobs = PrintJob.query.filter_by(printer_id=printer.id, aborted=False) \ + .filter(PrintJob.start_at <= now) \ + .filter(PrintJob.start_at.op('+')(PrintJob.duration_in_minutes * 60) > now) \ + .all() + + if active_jobs: + return jsonify({'error': 'Cannot delete printer with active jobs'}), 400 + + db.session.delete(printer) + db.session.commit() + + return jsonify({'message': 'Printer deleted successfully'}) + +@bp.route('/printers/availability', methods=['GET']) +def get_availability(): + """Get availability information for all printers""" + start_date = request.args.get('start_date') + end_date = request.args.get('end_date') + + if not start_date or not end_date: + return jsonify({'error': 'start_date and end_date are required'}), 400 + + try: + start = datetime.fromisoformat(start_date.replace('Z', '+00:00')) + end = datetime.fromisoformat(end_date.replace('Z', '+00:00')) + except ValueError: + return jsonify({'error': 'Invalid date format'}), 400 + + if start >= end: + return jsonify({'error': 'start_date must be before end_date'}), 400 + + printers = Printer.query.all() + result = [] + + for printer in printers: + # Get all jobs for this printer in the date range + jobs = PrintJob.query.filter_by(printer_id=printer.id, aborted=False) \ + .filter( + (PrintJob.start_at <= end) & + (PrintJob.start_at.op('+')(PrintJob.duration_in_minutes * 60) >= start) + ) \ + .order_by(PrintJob.start_at) \ + .all() + + # Convert to availability slots + availability = { + 'printer_id': printer.id, + 'printer_name': printer.name, + 'status': printer.status, + 'jobs': [job.to_dict() for job in jobs] + } + + result.append(availability) + + return jsonify(result) \ No newline at end of file diff --git a/archiv/flask-backend/app/api/users.py b/archiv/flask-backend/app/api/users.py new file mode 100644 index 000000000..3c348fc70 --- /dev/null +++ b/archiv/flask-backend/app/api/users.py @@ -0,0 +1,139 @@ +from flask import request, jsonify +from app import db +from app.api import bp +from app.models import User, PrintJob +from app.auth.routes import admin_required, token_required + +@bp.route('/users', methods=['GET']) +@admin_required +def get_users(): + """Get all users (admin only)""" + users = User.query.all() + result = [] + + for user in users: + # Count jobs + total_jobs = PrintJob.query.filter_by(user_id=user.id).count() + active_jobs = PrintJob.query.filter_by(user_id=user.id, aborted=False).count() + + user_data = { + 'id': user.id, + 'github_id': user.github_id, + 'username': user.username, + 'display_name': user.display_name, + 'email': user.email, + 'role': user.role, + 'job_count': total_jobs, + 'active_job_count': active_jobs + } + result.append(user_data) + + return jsonify(result) + +@bp.route('/users/', methods=['GET']) +@admin_required +def get_user(user_id): + """Get a specific user (admin only)""" + user = User.query.get_or_404(user_id) + + # Count jobs + total_jobs = PrintJob.query.filter_by(user_id=user.id).count() + active_jobs = PrintJob.query.filter_by(user_id=user.id, aborted=False).count() + + result = { + 'id': user.id, + 'github_id': user.github_id, + 'username': user.username, + 'display_name': user.display_name, + 'email': user.email, + 'role': user.role, + 'job_count': total_jobs, + 'active_job_count': active_jobs + } + + return jsonify(result) + +@bp.route('/users/', methods=['PUT']) +@admin_required +def update_user(user_id): + """Update a user (admin only)""" + user = User.query.get_or_404(user_id) + data = request.get_json() or {} + + if 'role' in data and data['role'] in ['admin', 'user', 'guest']: + user.role = data['role'] + + if 'display_name' in data: + user.display_name = data['display_name'] + + db.session.commit() + + return jsonify({ + 'id': user.id, + 'github_id': user.github_id, + 'username': user.username, + 'display_name': user.display_name, + 'email': user.email, + 'role': user.role + }) + +@bp.route('/users/', methods=['DELETE']) +@admin_required +def delete_user(user_id): + """Delete a user (admin only)""" + user = User.query.get_or_404(user_id) + + # Check if user has active jobs + active_jobs = PrintJob.query.filter_by(user_id=user.id, aborted=False).first() + if active_jobs: + return jsonify({'error': 'Cannot delete user with active jobs'}), 400 + + db.session.delete(user) + db.session.commit() + + return jsonify({'message': 'User deleted successfully'}) + +@bp.route('/me', methods=['GET']) +@token_required +def get_current_user(): + """Get the current user's profile""" + user = User.query.get(request.user_id) + if not user: + return jsonify({'error': 'User not found'}), 404 + + result = { + 'id': user.id, + 'github_id': user.github_id, + 'username': user.username, + 'display_name': user.display_name, + 'email': user.email, + 'role': user.role + } + + return jsonify(result) + +@bp.route('/me', methods=['PUT']) +@token_required +def update_current_user(): + """Update the current user's profile""" + user = User.query.get(request.user_id) + if not user: + return jsonify({'error': 'User not found'}), 404 + + data = request.get_json() or {} + + if 'display_name' in data: + user.display_name = data['display_name'] + + db.session.commit() + + result = { + 'id': user.id, + 'github_id': user.github_id, + 'username': user.username, + 'display_name': user.display_name, + 'email': user.email, + 'role': user.role + } + + return jsonify(result) \ No newline at end of file diff --git a/archiv/flask-backend/app/auth/__init__.py b/archiv/flask-backend/app/auth/__init__.py new file mode 100644 index 000000000..2834ca1ef --- /dev/null +++ b/archiv/flask-backend/app/auth/__init__.py @@ -0,0 +1,5 @@ +from flask import Blueprint + +bp = Blueprint('auth', __name__) + +from app.auth import routes \ No newline at end of file diff --git a/archiv/flask-backend/app/auth/routes.py b/archiv/flask-backend/app/auth/routes.py new file mode 100644 index 000000000..69380a1a8 --- /dev/null +++ b/archiv/flask-backend/app/auth/routes.py @@ -0,0 +1,156 @@ +from flask import request, jsonify, current_app +from app import db +from app.auth import bp +from app.models import User, Session +from datetime import datetime, timedelta +import time +import functools +import re + +@bp.route('/register', methods=['POST']) +def register(): + """Register a new user""" + data = request.get_json() or {} + + # Validate required fields + required_fields = ['username', 'email', 'password'] + for field in required_fields: + if field not in data: + return jsonify({'error': f'Missing required field: {field}'}), 400 + + # Validate email format + email_regex = r'^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$' + if not re.match(email_regex, data['email']): + return jsonify({'error': 'Invalid email format'}), 400 + + # Validate password strength (at least 8 characters) + if len(data['password']) < 8: + return jsonify({'error': 'Password must be at least 8 characters long'}), 400 + + # Check if username already exists + if User.query.filter_by(username=data['username']).first(): + return jsonify({'error': 'Username already exists'}), 400 + + # Check if email already exists + if User.query.filter_by(email=data['email']).first(): + return jsonify({'error': 'Email already exists'}), 400 + + # Create new user + user = User( + username=data['username'], + email=data['email'], + display_name=data.get('display_name', data['username']), + role='user' # Default role + ) + user.set_password(data['password']) + + db.session.add(user) + db.session.commit() + + return jsonify({ + 'id': user.id, + 'username': user.username, + 'email': user.email, + 'display_name': user.display_name, + 'role': user.role + }), 201 + +@bp.route('/login', methods=['POST']) +def login(): + """Login a user with username/email and password""" + data = request.get_json() or {} + + # Validate required fields + if 'password' not in data: + return jsonify({'error': 'Password is required'}), 400 + + if 'username' not in data and 'email' not in data: + return jsonify({'error': 'Username or email is required'}), 400 + + # Find user by username or email + user = None + if 'username' in data: + user = User.query.filter_by(username=data['username']).first() + else: + user = User.query.filter_by(email=data['email']).first() + + # Check if user exists and verify password + if not user or not user.check_password(data['password']): + return jsonify({'error': 'Invalid credentials'}), 401 + + # Create a session for the user + expires_at = int((datetime.utcnow() + timedelta(days=7)).timestamp()) + session = Session( + user_id=user.id, + expires_at=expires_at + ) + db.session.add(session) + db.session.commit() + + # Generate JWT token + token = user.generate_token() + + return jsonify({ + 'token': token, + 'user': { + 'id': user.id, + 'username': user.username, + 'email': user.email, + 'display_name': user.display_name, + 'role': user.role + } + }) + +@bp.route('/logout', methods=['POST']) +def logout(): + """Log out a user by invalidating their session""" + auth_header = request.headers.get('Authorization') + if not auth_header or not auth_header.startswith('Bearer '): + return jsonify({'error': 'Authorization header required'}), 401 + + token = auth_header.split(' ')[1] + payload = User.verify_token(token) + if not payload: + return jsonify({'error': 'Invalid token'}), 401 + + # Delete all sessions for this user + Session.query.filter_by(user_id=payload['user_id']).delete() + db.session.commit() + + return jsonify({'message': 'Successfully logged out'}) + +def token_required(f): + @functools.wraps(f) + def decorated(*args, **kwargs): + auth_header = request.headers.get('Authorization') + if not auth_header or not auth_header.startswith('Bearer '): + return jsonify({'error': 'Authorization header required'}), 401 + + token = auth_header.split(' ')[1] + payload = User.verify_token(token) + if not payload: + return jsonify({'error': 'Invalid token'}), 401 + + # Check if user has an active session + user_id = payload['user_id'] + current_time = int(time.time()) + session = Session.query.filter_by(user_id=user_id).filter(Session.expires_at > current_time).first() + if not session: + return jsonify({'error': 'No active session found'}), 401 + + # Add user to request context + request.user_id = user_id + request.user_role = payload['role'] + + return f(*args, **kwargs) + return decorated + +def admin_required(f): + @functools.wraps(f) + @token_required + def decorated(*args, **kwargs): + if request.user_role != 'admin': + return jsonify({'error': 'Admin privileges required'}), 403 + + return f(*args, **kwargs) + return decorated \ No newline at end of file diff --git a/archiv/flask-backend/app/models.py b/archiv/flask-backend/app/models.py new file mode 100644 index 000000000..a3b8e8043 --- /dev/null +++ b/archiv/flask-backend/app/models.py @@ -0,0 +1,124 @@ +from app import db +import uuid +from datetime import datetime, timedelta +import jwt +from config import Config +import bcrypt + +class User(db.Model): + __tablename__ = 'user' + + id = db.Column(db.String(36), primary_key=True, default=lambda: str(uuid.uuid4())) + username = db.Column(db.String(64), index=True, unique=True, nullable=False) + display_name = db.Column(db.String(120)) + email = db.Column(db.String(120), index=True, unique=True, nullable=False) + password_hash = db.Column(db.String(128), nullable=False) + role = db.Column(db.String(20), default='user') + + print_jobs = db.relationship('PrintJob', backref='user', lazy='dynamic', cascade='all, delete-orphan') + sessions = db.relationship('Session', backref='user', lazy='dynamic', cascade='all, delete-orphan') + + def set_password(self, password): + """Hash and set the user's password""" + password_bytes = password.encode('utf-8') + salt = bcrypt.gensalt() + self.password_hash = bcrypt.hashpw(password_bytes, salt).decode('utf-8') + + def check_password(self, password): + """Check if the provided password matches the stored hash""" + password_bytes = password.encode('utf-8') + stored_hash = self.password_hash.encode('utf-8') + return bcrypt.checkpw(password_bytes, stored_hash) + + def generate_token(self): + """Generate a JWT token for this user""" + payload = { + 'user_id': self.id, + 'username': self.username, + 'email': self.email, + 'role': self.role, + 'exp': datetime.utcnow() + timedelta(seconds=Config.JWT_ACCESS_TOKEN_EXPIRES) + } + return jwt.encode(payload, Config.JWT_SECRET, algorithm='HS256') + + @staticmethod + def verify_token(token): + """Verify and decode a JWT token""" + try: + payload = jwt.decode(token, Config.JWT_SECRET, algorithms=['HS256']) + return payload + except: + return None + + +class Session(db.Model): + __tablename__ = 'session' + + id = db.Column(db.String(36), primary_key=True, default=lambda: str(uuid.uuid4())) + user_id = db.Column(db.String(36), db.ForeignKey('user.id', ondelete='CASCADE'), nullable=False) + expires_at = db.Column(db.Integer, nullable=False) + + +class Printer(db.Model): + __tablename__ = 'printer' + + id = db.Column(db.String(36), primary_key=True, default=lambda: str(uuid.uuid4())) + name = db.Column(db.String(120), nullable=False) + description = db.Column(db.Text, nullable=False) + status = db.Column(db.Integer, nullable=False, default=0) # 0: OPERATIONAL, 1: OUT_OF_ORDER + + print_jobs = db.relationship('PrintJob', backref='printer', lazy='dynamic', cascade='all, delete-orphan') + + +class PrintJob(db.Model): + __tablename__ = 'printJob' + + id = db.Column(db.String(36), primary_key=True, default=lambda: str(uuid.uuid4())) + printer_id = db.Column(db.String(36), db.ForeignKey('printer.id', ondelete='CASCADE'), nullable=False) + user_id = db.Column(db.String(36), db.ForeignKey('user.id', ondelete='CASCADE'), nullable=False) + start_at = db.Column(db.DateTime, nullable=False, default=datetime.utcnow) + duration_in_minutes = db.Column(db.Integer, nullable=False) + comments = db.Column(db.Text) + aborted = db.Column(db.Boolean, nullable=False, default=False) + abort_reason = db.Column(db.Text) + + def get_end_time(self): + return self.start_at + timedelta(minutes=self.duration_in_minutes) + + def is_active(self): + now = datetime.utcnow() + return (not self.aborted and + self.start_at <= now and + now < self.get_end_time()) + + def get_remaining_time(self): + if self.aborted: + return 0 + + now = datetime.utcnow() + if now < self.start_at: + # Job hasn't started yet + return self.duration_in_minutes * 60 + + end_time = self.get_end_time() + if now >= end_time: + # Job has ended + return 0 + + # Job is ongoing + remaining_seconds = (end_time - now).total_seconds() + return int(remaining_seconds) + + def to_dict(self): + return { + 'id': self.id, + 'printer_id': self.printer_id, + 'user_id': self.user_id, + 'start_at': self.start_at.isoformat(), + 'duration_in_minutes': self.duration_in_minutes, + 'comments': self.comments, + 'aborted': self.aborted, + 'abort_reason': self.abort_reason, + 'remaining_time': self.get_remaining_time(), + 'is_active': self.is_active() + } \ No newline at end of file diff --git a/archiv/flask-backend/config.py b/archiv/flask-backend/config.py new file mode 100644 index 000000000..b2766c55c --- /dev/null +++ b/archiv/flask-backend/config.py @@ -0,0 +1,13 @@ +import os +from dotenv import load_dotenv + +basedir = os.path.abspath(os.path.dirname(__file__)) +load_dotenv(os.path.join(basedir, '.env')) + +class Config: + SECRET_KEY = os.environ.get('SECRET_KEY') or 'you-will-never-guess' + SQLALCHEMY_DATABASE_URI = os.environ.get('DATABASE_URL') or \ + 'sqlite:///' + os.path.join(basedir, 'app.db') + SQLALCHEMY_TRACK_MODIFICATIONS = False + JWT_SECRET = os.environ.get('JWT_SECRET') or 'jwt-secret-key' + JWT_ACCESS_TOKEN_EXPIRES = 3600 # 1 hour in seconds \ No newline at end of file diff --git a/archiv/flask-backend/docker-compose.yml b/archiv/flask-backend/docker-compose.yml new file mode 100644 index 000000000..8747837c9 --- /dev/null +++ b/archiv/flask-backend/docker-compose.yml @@ -0,0 +1,20 @@ +version: '3.8' + +services: + flask-backend: + build: + context: . + dockerfile: Dockerfile + restart: always + ports: + - "5000:5000" + environment: + - SECRET_KEY=your-secret-key + - DATABASE_URL=sqlite:///app.db + - JWT_SECRET=your-jwt-secret + volumes: + - ./instance:/app/instance + command: > + bash -c "python -m flask db upgrade && + python scripts/init_db.py && + gunicorn --bind 0.0.0.0:5000 wsgi:app" \ No newline at end of file diff --git a/archiv/flask-backend/migrations/alembic.ini b/archiv/flask-backend/migrations/alembic.ini new file mode 100644 index 000000000..187d090d4 --- /dev/null +++ b/archiv/flask-backend/migrations/alembic.ini @@ -0,0 +1,89 @@ +# A generic, single database configuration. + +[alembic] +# path to migration scripts +script_location = migrations + +# template used to generate migration files +file_template = %%(year)d%%(month).2d%%(day).2d_%%(hour).2d%%(minute).2d%%(second).2d_%%(slug)s + +# sys.path path, will be prepended to sys.path if present. +# defaults to the current working directory. +prepend_sys_path = . + +# timezone to use when rendering the date +# within the migration file as well as the filename. +# string value is passed to dateutil.tz.gettz() +# leave blank for localtime +# timezone = + +# max length of characters to apply to the +# "slug" field +truncate_slug_length = 40 + +# set to 'true' to run the environment during +# the 'revision' command, regardless of autogenerate +# revision_environment = false + +# set to 'true' to allow .pyc and .pyo files without +# a source .py file to be detected as revisions in the +# versions/ directory +# sourceless = false + +# version location specification; this defaults +# to migrations/versions. When using multiple version +# directories, initial revisions must be specified with --version-path +# version_locations = %(here)s/bar %(here)s/bat migrations/versions + +# the output encoding used when revision files +# are written from script.py.mako +# output_encoding = utf-8 + +sqlalchemy.url = driver://user:pass@localhost/dbname + + +[post_write_hooks] +# post_write_hooks defines scripts or Python functions that are run +# on newly generated revision scripts. See the documentation for further +# detail and examples + +# format using "black" - use the console_scripts runner, against the "black" entrypoint +# hooks = black +# black.type = console_scripts +# black.entrypoint = black +# black.options = -l 79 REVISION_SCRIPT_FILENAME + +# Logging configuration +[loggers] +keys = root,sqlalchemy,alembic + +[handlers] +keys = console + +[formatters] +keys = generic + +[logger_root] +level = WARN +handlers = console +qualname = + +[logger_sqlalchemy] +level = WARN +handlers = +qualname = sqlalchemy.engine + +[logger_alembic] +level = INFO +handlers = +qualname = alembic + +[handler_console] +class = StreamHandler +args = (sys.stderr,) +level = NOTSET +formatter = generic + +[formatter_generic] +format = %(levelname)-5.5s [%(name)s] %(message)s +datefmt = %H:%M:%S \ No newline at end of file diff --git a/archiv/flask-backend/migrations/env.py b/archiv/flask-backend/migrations/env.py new file mode 100644 index 000000000..29192f010 --- /dev/null +++ b/archiv/flask-backend/migrations/env.py @@ -0,0 +1,91 @@ +from __future__ import with_statement + +import logging +from logging.config import fileConfig + +from flask import current_app + +from alembic import context + +# this is the Alembic Config object, which provides +# access to the values within the .ini file in use. +config = context.config + +# Interpret the config file for Python logging. +# This line sets up loggers basically. +fileConfig(config.config_file_name) +logger = logging.getLogger('alembic.env') + +# add your model's MetaData object here +# for 'autogenerate' support +# from myapp import mymodel +# target_metadata = mymodel.Base.metadata +config.set_main_option( + 'sqlalchemy.url', + str(current_app.extensions['migrate'].db.get_engine().url).replace( + '%', '%%')) +target_metadata = current_app.extensions['migrate'].db.metadata + +# other values from the config, defined by the needs of env.py, +# can be acquired: +# my_important_option = config.get_main_option("my_important_option") +# ... etc. + + +def run_migrations_offline(): + """Run migrations in 'offline' mode. + + This configures the context with just a URL + and not an Engine, though an Engine is acceptable + here as well. By skipping the Engine creation + we don't even need a DBAPI to be available. + + Calls to context.execute() here emit the given string to the + script output. + + """ + url = config.get_main_option("sqlalchemy.url") + context.configure( + url=url, target_metadata=target_metadata, literal_binds=True + ) + + with context.begin_transaction(): + context.run_migrations() + + +def run_migrations_online(): + """Run migrations in 'online' mode. + + In this scenario we need to create an Engine + and associate a connection with the context. + + """ + + # this callback is used to prevent an auto-migration from being generated + # when there are no changes to the schema + # reference: http://alembic.zzzcomputing.com/en/latest/cookbook.html + def process_revision_directives(context, revision, directives): + if getattr(config.cmd_opts, 'autogenerate', False): + script = directives[0] + if script.upgrade_ops.is_empty(): + directives[:] = [] + logger.info('No changes in schema detected.') + + connectable = current_app.extensions['migrate'].db.get_engine() + + with connectable.connect() as connection: + context.configure( + connection=connection, + target_metadata=target_metadata, + process_revision_directives=process_revision_directives, + **current_app.extensions['migrate'].configure_args + ) + + with context.begin_transaction(): + context.run_migrations() + + +if context.is_offline_mode(): + run_migrations_offline() +else: + run_migrations_online() \ No newline at end of file diff --git a/archiv/flask-backend/migrations/script.py.mako b/archiv/flask-backend/migrations/script.py.mako new file mode 100644 index 000000000..1e4564e5e --- /dev/null +++ b/archiv/flask-backend/migrations/script.py.mako @@ -0,0 +1,24 @@ +"""${message} + +Revision ID: ${up_revision} +Revises: ${down_revision | comma,n} +Create Date: ${create_date} + +""" +from alembic import op +import sqlalchemy as sa +${imports if imports else ""} + +# revision identifiers, used by Alembic. +revision = ${repr(up_revision)} +down_revision = ${repr(down_revision)} +branch_labels = ${repr(branch_labels)} +depends_on = ${repr(depends_on)} + + +def upgrade(): + ${upgrades if upgrades else "pass"} + + +def downgrade(): + ${downgrades if downgrades else "pass"} \ No newline at end of file diff --git a/archiv/flask-backend/migrations/versions/initial_migration.py b/archiv/flask-backend/migrations/versions/initial_migration.py new file mode 100644 index 000000000..9883ff42a --- /dev/null +++ b/archiv/flask-backend/migrations/versions/initial_migration.py @@ -0,0 +1,75 @@ +"""Initial migration + +Revision ID: initial_migration +Revises: +Create Date: 2025-03-06 12:00:00.000000 + +""" +from alembic import op +import sqlalchemy as sa + + +# revision identifiers, used by Alembic. +revision = 'initial_migration' +down_revision = None +branch_labels = None +depends_on = None + + +def upgrade(): + # Create user table + op.create_table('user', + sa.Column('id', sa.String(length=36), nullable=False), + sa.Column('username', sa.String(length=64), nullable=False), + sa.Column('display_name', sa.String(length=120), nullable=True), + sa.Column('email', sa.String(length=120), nullable=False), + sa.Column('password_hash', sa.String(length=128), nullable=False), + sa.Column('role', sa.String(length=20), nullable=True), + sa.PrimaryKeyConstraint('id'), + sa.UniqueConstraint('email'), + sa.UniqueConstraint('username') + ) + op.create_index(op.f('ix_user_email'), 'user', ['email'], unique=True) + op.create_index(op.f('ix_user_username'), 'user', ['username'], unique=True) + + # Create session table + op.create_table('session', + sa.Column('id', sa.String(length=36), nullable=False), + sa.Column('user_id', sa.String(length=36), nullable=False), + sa.Column('expires_at', sa.Integer(), nullable=False), + sa.ForeignKeyConstraint(['user_id'], ['user.id'], ondelete='CASCADE'), + sa.PrimaryKeyConstraint('id') + ) + + # Create printer table + op.create_table('printer', + sa.Column('id', sa.String(length=36), nullable=False), + sa.Column('name', sa.String(length=120), nullable=False), + sa.Column('description', sa.Text(), nullable=False), + sa.Column('status', sa.Integer(), nullable=False), + sa.PrimaryKeyConstraint('id') + ) + + # Create printJob table + op.create_table('printJob', + sa.Column('id', sa.String(length=36), nullable=False), + sa.Column('printer_id', sa.String(length=36), nullable=False), + sa.Column('user_id', sa.String(length=36), nullable=False), + sa.Column('start_at', sa.DateTime(), nullable=False), + sa.Column('duration_in_minutes', sa.Integer(), nullable=False), + sa.Column('comments', sa.Text(), nullable=True), + sa.Column('aborted', sa.Boolean(), nullable=False), + sa.Column('abort_reason', sa.Text(), nullable=True), + sa.ForeignKeyConstraint(['printer_id'], ['printer.id'], ondelete='CASCADE'), + sa.ForeignKeyConstraint(['user_id'], ['user.id'], ondelete='CASCADE'), + sa.PrimaryKeyConstraint('id') + ) + + +def downgrade(): + op.drop_table('printJob') + op.drop_table('printer') + op.drop_table('session') + op.drop_index(op.f('ix_user_username'), table_name='user') + op.drop_index(op.f('ix_user_email'), table_name='user') + op.drop_table('user') \ No newline at end of file diff --git a/archiv/flask-backend/requirements.txt b/archiv/flask-backend/requirements.txt new file mode 100644 index 000000000..06d4a3e23 --- /dev/null +++ b/archiv/flask-backend/requirements.txt @@ -0,0 +1,9 @@ +Flask==2.3.3 +Flask-SQLAlchemy==3.1.1 +Flask-Migrate==4.0.5 +Flask-CORS==4.0.0 +python-dotenv==1.0.0 +SQLAlchemy==2.0.25 +pyjwt==2.8.0 +bcrypt==4.1.2 +gunicorn==21.2.0 \ No newline at end of file diff --git a/archiv/flask-backend/run.sh b/archiv/flask-backend/run.sh new file mode 100644 index 000000000..301fe3573 --- /dev/null +++ b/archiv/flask-backend/run.sh @@ -0,0 +1,23 @@ +#!/bin/bash + +# Initialize virtual environment if it doesn't exist +if [ ! -d "venv" ]; then + echo "Creating virtual environment..." + python3 -m venv venv +fi + +# Activate virtual environment +source venv/bin/activate + +# Install dependencies +echo "Installing dependencies..." +pip install -r requirements.txt + +# Initialize database +echo "Initializing database..." +flask db upgrade +python scripts/init_db.py + +# Run the application +echo "Starting Flask application..." +python wsgi.py \ No newline at end of file diff --git a/archiv/flask-backend/scripts/init_db.py b/archiv/flask-backend/scripts/init_db.py new file mode 100644 index 000000000..67ee41bf1 --- /dev/null +++ b/archiv/flask-backend/scripts/init_db.py @@ -0,0 +1,55 @@ +#!/usr/bin/env python +from app import create_app, db +from app.models import User, Printer +import uuid + +def init_db(): + app = create_app() + with app.app_context(): + # Create tables + db.create_all() + + # Check if we already have an admin user + admin = User.query.filter_by(role='admin').first() + if not admin: + # Create admin user + admin = User( + id=str(uuid.uuid4()), + username='admin', + display_name='Administrator', + email='admin@example.com', + role='admin' + ) + admin.set_password('admin123') # Default password, change in production! + db.session.add(admin) + print("Created admin user with username 'admin' and password 'admin123'") + + # Check if we have any printers + printer_count = Printer.query.count() + if printer_count == 0: + # Create sample printers + printers = [ + Printer( + name='Printer 1', + description='3D Printer for general use', + status=0 # OPERATIONAL + ), + Printer( + name='Printer 2', + description='High resolution printer for detailed work', + status=0 # OPERATIONAL + ), + Printer( + name='Printer 3', + description='Large format printer for big projects', + status=0 # OPERATIONAL + ) + ] + db.session.add_all(printers) + print("Created sample printers") + + db.session.commit() + print("Database initialized successfully!") + +if __name__ == '__main__': + init_db() \ No newline at end of file diff --git a/archiv/flask-backend/wsgi.py b/archiv/flask-backend/wsgi.py new file mode 100644 index 000000000..7425f4cc2 --- /dev/null +++ b/archiv/flask-backend/wsgi.py @@ -0,0 +1,6 @@ +from app import create_app + +app = create_app() + +if __name__ == '__main__': + app.run(host='0.0.0.0', port=5000, debug=True) \ No newline at end of file diff --git a/backend/.gitignore b/backend/.gitignore new file mode 100644 index 000000000..c7beae632 --- /dev/null +++ b/backend/.gitignore @@ -0,0 +1,48 @@ +# Python +__pycache__/ +*.py[cod] +*$py.class +*.so +.Python +env/ +build/ +develop-eggs/ +dist/ +downloads/ +eggs/ +.eggs/ +lib/ +lib64/ +parts/ +sdist/ +var/ +wheels/ +*.egg-info/ +.installed.cfg +*.egg + +# Flask +instance/ +.webassets-cache + +# Logs +logs/ +*.log + +# SQLite Datenbank-Dateien +*.db +*.db-journal + +# Virtuelle Umgebungen +venv/ +ENV/ + +# IDE +.idea/ +.vscode/ +*.swp +*.swo + +# Betriebssystem +.DS_Store +Thumbs.db \ No newline at end of file diff --git a/backend/Dockerfile b/backend/Dockerfile new file mode 100644 index 000000000..ccffe7097 --- /dev/null +++ b/backend/Dockerfile @@ -0,0 +1,18 @@ +FROM python:3.11-slim + +WORKDIR /app + +COPY requirements.txt . + +RUN pip install --no-cache-dir -r requirements.txt + +COPY . . + +RUN mkdir -p logs + +ENV FLASK_APP=app.py +ENV PYTHONUNBUFFERED=1 + +EXPOSE 5000 + +CMD ["gunicorn", "--bind", "0.0.0.0:5000", "app:app"] \ No newline at end of file diff --git a/backend/app.py b/backend/app.py new file mode 100644 index 000000000..7d2e356a3 --- /dev/null +++ b/backend/app.py @@ -0,0 +1,1719 @@ +from flask import Flask, request, jsonify, g, redirect, url_for, session as flask_session, render_template, flash +from flask_cors import CORS +from werkzeug.security import generate_password_hash, check_password_hash +import secrets # Für bessere Salt-Generierung +from functools import wraps +import jwt +import datetime +import os +import json +import logging +import uuid +import sqlite3 +import threading +import time +from logging.handlers import RotatingFileHandler +from datetime import timedelta +from PyP100 import PyP100 +from dotenv import load_dotenv + +# Lade Umgebungsvariablen +load_dotenv() + +# Initialisierung +app = Flask(__name__) +CORS(app, supports_credentials=True) + +# Konfiguration +app.config['SECRET_KEY'] = os.environ.get('SECRET_KEY', 'dev_secret_key') +app.config['DATABASE'] = os.environ.get('DATABASE_PATH', 'instance/myp.db') +app.config['SESSION_COOKIE_HTTPONLY'] = True +app.config['SESSION_COOKIE_SECURE'] = os.environ.get('FLASK_ENV') == 'production' +app.config['SESSION_COOKIE_SAMESITE'] = 'Lax' +app.config['PERMANENT_SESSION_LIFETIME'] = timedelta(days=7) +app.config['JOB_CHECK_INTERVAL'] = int(os.environ.get('JOB_CHECK_INTERVAL', '60')) # Sekunden + +# Steckdosen-Konfiguration +TAPO_USERNAME = os.environ.get('TAPO_USERNAME') +TAPO_PASSWORD = os.environ.get('TAPO_PASSWORD') + +# Logging +if not os.path.exists('logs'): + os.mkdir('logs') +file_handler = RotatingFileHandler('logs/myp.log', maxBytes=10240, backupCount=10) +file_handler.setFormatter(logging.Formatter( + '%(asctime)s %(levelname)s: %(message)s [in %(pathname)s:%(lineno)d]' +)) +file_handler.setLevel(logging.INFO) +app.logger.addHandler(file_handler) +app.logger.setLevel(logging.INFO) +app.logger.info('MYP Backend starting up') + +# Database functions +def get_db(): + if 'db' not in g: + # Stelle sicher, dass das instance-Verzeichnis existiert + os.makedirs(os.path.dirname(app.config['DATABASE']), exist_ok=True) + g.db = sqlite3.connect(app.config['DATABASE']) + g.db.row_factory = sqlite3.Row + return g.db + +def close_db(e=None): + db = g.pop('db', None) + if db is not None: + db.close() + +def init_db(): + """Initialisiere die Datenbank, falls sie noch nicht existiert.""" + db = get_db() + db.execute('PRAGMA foreign_keys = ON') # SQLite-Fremdschlüsselunterstützung aktivieren + + # Tabellen erstellen + db.executescript(''' + CREATE TABLE IF NOT EXISTS user ( + id TEXT PRIMARY KEY, + username TEXT UNIQUE NOT NULL, + password_hash TEXT NOT NULL, + display_name TEXT, + email TEXT UNIQUE, + role TEXT DEFAULT 'user' + ); + + CREATE TABLE IF NOT EXISTS session ( + id TEXT PRIMARY KEY, + user_id TEXT NOT NULL, + expires_at TIMESTAMP NOT NULL, + FOREIGN KEY (user_id) REFERENCES user (id) ON DELETE CASCADE + ); + + CREATE TABLE IF NOT EXISTS socket ( + id TEXT PRIMARY KEY, + name TEXT NOT NULL, + description TEXT NOT NULL, + status INTEGER DEFAULT 0, + ip_address TEXT, + last_seen TIMESTAMP, + connection_status TEXT DEFAULT 'unknown' + ); + + CREATE TABLE IF NOT EXISTS job ( + id TEXT PRIMARY KEY, + socket_id TEXT NOT NULL, + user_id TEXT NOT NULL, + start_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, + duration_in_minutes INTEGER NOT NULL, + comments TEXT, + aborted INTEGER DEFAULT 0, + abort_reason TEXT, + waiting_approval INTEGER DEFAULT 0, + FOREIGN KEY (socket_id) REFERENCES socket (id) ON DELETE CASCADE, + FOREIGN KEY (user_id) REFERENCES user (id) ON DELETE CASCADE + ); + + CREATE TABLE IF NOT EXISTS socket_uptime ( + id TEXT PRIMARY KEY, + socket_id TEXT NOT NULL, + timestamp TIMESTAMP DEFAULT CURRENT_TIMESTAMP, + status TEXT NOT NULL, + duration_seconds INTEGER, + FOREIGN KEY (socket_id) REFERENCES socket (id) ON DELETE CASCADE + ); + ''') + + # Überprüfe, ob die fehlenden Spalten bereits existieren, und füge sie hinzu, falls nicht + try: + # Prüfe, ob die connection_status-Spalte existiert + db.execute('SELECT connection_status FROM socket LIMIT 1') + except sqlite3.OperationalError: + # Spalte existiert nicht, füge sie hinzu + app.logger.info("Füge connection_status-Spalte zur socket-Tabelle hinzu") + db.execute('ALTER TABLE socket ADD COLUMN connection_status TEXT DEFAULT "unknown"') + + try: + # Prüfe, ob die last_seen-Spalte existiert + db.execute('SELECT last_seen FROM socket LIMIT 1') + except sqlite3.OperationalError: + # Spalte existiert nicht, füge sie hinzu + app.logger.info("Füge last_seen-Spalte zur socket-Tabelle hinzu") + db.execute('ALTER TABLE socket ADD COLUMN last_seen TIMESTAMP') + + db.commit() + +PRINTERS = json.loads(os.environ.get('PRINTERS', '{}')) + +def init_printers(): + app.logger.info("Initialisiere Drucker aus Umgebungsvariablen") + db = get_db() + + # Existierende IP-Adressen aus der Datenbank abrufen + existing_ips = {row['ip_address']: row['id'] for row in db.execute('SELECT id, ip_address FROM socket').fetchall() if row['ip_address']} + + for printer_name, config in PRINTERS.items(): + ip_address = config.get('ip') + if not ip_address: + continue # Überspringe Einträge ohne IP + description = f"Drucker mit IP: {ip_address}" + + if ip_address in existing_ips: + app.logger.info(f"Drucker mit IP {ip_address} existiert bereits in der Datenbank") + # Setze den Status des existierenden Druckers auf 0 (verfügbar) + socket_id = existing_ips[ip_address] + update_socket(socket_id, status=0) + # Stelle sicher, dass die Steckdose wirklich ausgeschaltet ist + turn_off_socket(ip_address) + app.logger.info(f"Steckdose mit IP {ip_address} wurde beim Start ausgeschaltet") + else: + # Neuen Drucker eintragen + new_socket = create_socket(name=printer_name, description=description, ip_address=ip_address, status=0) + app.logger.info(f"Neuer Drucker angelegt: {printer_name} mit IP {ip_address}") + # Stelle sicher, dass die Steckdose wirklich ausgeschaltet ist + turn_off_socket(ip_address) + app.logger.info(f"Neue Steckdose mit IP {ip_address} wurde beim Start ausgeschaltet") + +# Benutzerverwaltung +def get_user_by_id(user_id): + db = get_db() + row = db.execute('SELECT * FROM user WHERE id = ?', (user_id,)).fetchone() + if not row: + return None + return dict(row) + +def get_user_by_username(username): + db = get_db() + row = db.execute('SELECT * FROM user WHERE username = ?', (username,)).fetchone() + if not row: + return None + return dict(row) + +def get_all_users(): + db = get_db() + rows = db.execute('SELECT * FROM user').fetchall() + return [dict(row) for row in rows] + +def create_user(username, password, display_name=None, email=None, role='user'): + user_id = str(uuid.uuid4()) + + # Verwende einen sicheren Hash-Algorithmus (pbkdf2:sha256) mit mehr Iterationen (150000) + # und automatischem Salting durch Werkzeug + password_hash = generate_password_hash( + password, + method='pbkdf2:sha256', + salt_length=16 # Standardwert ist 8, aber wir erhöhen auf 16 für mehr Sicherheit + ) + + display_name = display_name or username + + db = get_db() + db.execute( + 'INSERT INTO user (id, username, password_hash, display_name, email, role) VALUES (?, ?, ?, ?, ?, ?)', + (user_id, username, password_hash, display_name, email, role) + ) + db.commit() + + app.logger.info(f"Benutzer {username} erstellt mit sicherem Password-Hash (pbkdf2:sha256, salt_length=16)") + return get_user_by_id(user_id) + +def update_user(user_id, username=None, password=None, display_name=None, email=None, role=None): + user = get_user_by_id(user_id) + if not user: + return None + + values = [] + params = [] + + if username: + values.append('username = ?') + params.append(username) + + if password: + values.append('password_hash = ?') + # Verwende den gleichen verbesserten Hashing-Mechanismus wie bei create_user + params.append(generate_password_hash( + password, + method='pbkdf2:sha256', + salt_length=16 + )) + + if display_name: + values.append('display_name = ?') + params.append(display_name) + + if email: + values.append('email = ?') + params.append(email) + + if role: + values.append('role = ?') + params.append(role) + + if not values: + return user + + query = f'UPDATE user SET {", ".join(values)} WHERE id = ?' + params.append(user_id) + + db = get_db() + db.execute(query, params) + db.commit() + + return get_user_by_id(user_id) + +def delete_user(user_id): + db = get_db() + db.execute('DELETE FROM user WHERE id = ?', (user_id,)) + db.commit() + return True + +def check_password(user_dict, password): + # Überprüfe das Passwort mit dem gespeicherten Hash + is_valid = check_password_hash(user_dict['password_hash'], password) + + # Wenn das Passwort gültig ist, überprüfe, ob der Hash aktualisiert werden muss + if is_valid: + # Überprüfe, ob der aktuelle Hash nicht das empfohlene Format verwendet + if not user_dict['password_hash'].startswith('pbkdf2:sha256:'): + # Hash muss aktualisiert werden, da er nicht den neuesten Sicherheitsstandards entspricht + app.logger.info(f"Migriere unsicheren Passwort-Hash für Benutzer {user_dict['username']} zu pbkdf2:sha256") + + # Erstelle neuen Hash mit dem bestätigten Passwort + new_hash = generate_password_hash( + password, + method='pbkdf2:sha256', + salt_length=16 + ) + + # Aktualisiere in der Datenbank + db = get_db() + db.execute('UPDATE user SET password_hash = ? WHERE id = ?', + (new_hash, user_dict['id'])) + db.commit() + + return is_valid + +def user_to_dict(user): + if not user: + return None + return { + 'id': user['id'], + 'username': user['username'], + 'displayName': user['display_name'], + 'email': user['email'], + 'role': user['role'] + } + +# Session-Verwaltung +def get_session_by_id(session_id): + db = get_db() + row = db.execute('SELECT * FROM session WHERE id = ?', (session_id,)).fetchone() + if not row: + return None + return dict(row) + +def delete_sessions_by_user(user_id): + db = get_db() + db.execute('DELETE FROM session WHERE user_id = ?', (user_id,)) + db.commit() + +def create_session(user_id): + session_id = str(uuid.uuid4()) + expires_at = datetime.datetime.utcnow() + timedelta(days=7) + + db = get_db() + db.execute( + 'INSERT INTO session (id, user_id, expires_at) VALUES (?, ?, ?)', + (session_id, user_id, expires_at.isoformat()) + ) + db.commit() + + flask_session['session_id'] = session_id + flask_session.permanent = True + + return session_id + +def delete_session(session_id): + db = get_db() + db.execute('DELETE FROM session WHERE id = ?', (session_id,)) + db.commit() + +# Steckdosen-Verwaltung +def get_socket_by_id(socket_id): + db = get_db() + row = db.execute('SELECT * FROM socket WHERE id = ?', (socket_id,)).fetchone() + if not row: + return None + return dict(row) + +def get_all_sockets(): + db = get_db() + rows = db.execute('SELECT * FROM socket').fetchall() + return [dict(row) for row in rows] + +def create_socket(name, description, ip_address=None, status=0): + socket_id = str(uuid.uuid4()) + + db = get_db() + db.execute( + 'INSERT INTO socket (id, name, description, status, ip_address) VALUES (?, ?, ?, ?, ?)', + (socket_id, name, description, status, ip_address) + ) + db.commit() + + return get_socket_by_id(socket_id) + +def update_socket(socket_id, name=None, description=None, status=None, ip_address=None): + socket = get_socket_by_id(socket_id) + if not socket: + return None + + values = [] + params = [] + + if name: + values.append('name = ?') + params.append(name) + + if description: + values.append('description = ?') + params.append(description) + + if status is not None: + values.append('status = ?') + params.append(status) + + if ip_address: + values.append('ip_address = ?') + params.append(ip_address) + + if not values: + return socket + + query = f'UPDATE socket SET {", ".join(values)} WHERE id = ?' + params.append(socket_id) + + db = get_db() + db.execute(query, params) + db.commit() + + return get_socket_by_id(socket_id) + +def delete_socket(socket_id): + db = get_db() + db.execute('DELETE FROM socket WHERE id = ?', (socket_id,)) + db.commit() + return True + +def get_latest_job_for_socket(socket_id): + db = get_db() + row = db.execute(''' + SELECT * FROM job + WHERE socket_id = ? + ORDER BY start_at DESC + LIMIT 1 + ''', (socket_id,)).fetchone() + + if not row: + return None + return dict(row) + +def socket_to_dict(socket): + if not socket: + return None + + latest_job = get_latest_job_for_socket(socket['id']) + waiting_jobs = get_waiting_jobs_for_socket(socket['id']) + + # Verbindungsstatus-Informationen + connection_status = socket.get('connection_status', 'unknown') + last_seen = socket.get('last_seen') + uptime_info = None + + if last_seen and connection_status == 'offline': + # Berechne wie lange die Steckdose offline ist + try: + last_seen_dt = datetime.datetime.fromisoformat(last_seen) + now = datetime.datetime.utcnow() + offline_duration = int((now - last_seen_dt).total_seconds()) + + # Formatiere die Offline-Zeit benutzerfreundlich + hours, remainder = divmod(offline_duration, 3600) + minutes, seconds = divmod(remainder, 60) + + uptime_info = { + 'offline_since': last_seen, + 'offline_duration': offline_duration, + 'offline_duration_formatted': f"{hours}h {minutes}m {seconds}s" + } + except (ValueError, TypeError): + # Wenn das Datum nicht geparst werden kann + uptime_info = { + 'offline_since': last_seen, + 'offline_duration': None, + 'offline_duration_formatted': "Unbekannt" + } + + return { + 'id': socket['id'], + 'name': socket['name'], + 'description': socket['description'], + 'status': socket['status'], + 'ipAddress': socket.get('ip_address'), + 'connectionStatus': connection_status, + 'lastSeen': last_seen, + 'uptimeInfo': uptime_info, + 'latestJob': job_to_dict(latest_job) if latest_job else None, + 'waitingJobs': [job_to_dict(job) for job in waiting_jobs] if waiting_jobs else [] + } + +# Job-Verwaltung +def get_job_by_id(job_id): + db = get_db() + row = db.execute('SELECT * FROM job WHERE id = ?', (job_id,)).fetchone() + if not row: + return None + return dict(row) + +def get_jobs_by_user(user_id): + db = get_db() + rows = db.execute('SELECT * FROM job WHERE user_id = ?', (user_id,)).fetchall() + return [dict(row) for row in rows] + +def get_all_jobs(): + db = get_db() + rows = db.execute('SELECT * FROM job').fetchall() + return [dict(row) for row in rows] + +def get_expired_jobs(): + db = get_db() + now = datetime.datetime.utcnow().isoformat() + rows = db.execute(''' + SELECT * FROM job + WHERE aborted = 0 + AND waiting_approval = 0 + AND datetime(start_at, '+' || duration_in_minutes || ' minutes') <= datetime(?) + ''', (now,)).fetchall() + return [dict(row) for row in rows] + +def get_waiting_jobs_for_socket(socket_id): + """Findet alle Jobs, die auf Freischaltung für eine bestimmte Steckdose warten.""" + db = get_db() + rows = db.execute(''' + SELECT * FROM job + WHERE socket_id = ? + AND aborted = 0 + AND waiting_approval = 1 + ORDER BY start_at ASC + ''', (socket_id,)).fetchall() + return [dict(row) for row in rows] + +def create_job(socket_id, user_id, duration_in_minutes, comments=None, waiting_approval=0): + job_id = str(uuid.uuid4()) + start_at = datetime.datetime.utcnow() + + db = get_db() + db.execute( + '''INSERT INTO job + (id, socket_id, user_id, start_at, duration_in_minutes, comments, aborted, abort_reason, waiting_approval) + VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)''', + (job_id, socket_id, user_id, start_at.isoformat(), duration_in_minutes, comments, 0, None, waiting_approval) + ) + db.commit() + + return get_job_by_id(job_id) + +def update_job(job_id, socket_id=None, user_id=None, duration_in_minutes=None, + comments=None, aborted=None, abort_reason=None, waiting_approval=None): + job = get_job_by_id(job_id) + if not job: + return None + + values = [] + params = [] + + if socket_id: + values.append('socket_id = ?') + params.append(socket_id) + + if user_id: + values.append('user_id = ?') + params.append(user_id) + + if duration_in_minutes: + values.append('duration_in_minutes = ?') + params.append(duration_in_minutes) + + if comments is not None: + values.append('comments = ?') + params.append(comments) + + if aborted is not None: + values.append('aborted = ?') + params.append(1 if aborted else 0) + + if abort_reason is not None: + values.append('abort_reason = ?') + params.append(abort_reason) + + if waiting_approval is not None: + values.append('waiting_approval = ?') + params.append(1 if waiting_approval else 0) + + if not values: + return job + + query = f'UPDATE job SET {", ".join(values)} WHERE id = ?' + params.append(job_id) + + db = get_db() + db.execute(query, params) + db.commit() + + return get_job_by_id(job_id) + +def delete_job(job_id): + db = get_db() + db.execute('DELETE FROM job WHERE id = ?', (job_id,)) + db.commit() + return True + +def calculate_remaining_time(job): + if job['aborted']: + return 0 + + start_at = datetime.datetime.fromisoformat(job['start_at']) + end_at = start_at + timedelta(minutes=job['duration_in_minutes']) + + now = datetime.datetime.utcnow() + if now > end_at: + return 0 + + diff = end_at - now + return int(diff.total_seconds() / 60) + +def job_to_dict(job): + if not job: + return None + + # Bei älteren Jobs könnte waiting_approval fehlen, deshalb mit get abrufen und Default setzen + waiting_approval = job.get('waiting_approval', 0) if isinstance(job, dict) else getattr(job, 'waiting_approval', 0) + + return { + 'id': job['id'], + 'socketId': job['socket_id'], + 'userId': job['user_id'], + 'startAt': job['start_at'], + 'durationInMinutes': job['duration_in_minutes'], + 'comments': job['comments'], + 'aborted': bool(job['aborted']), + 'abortReason': job['abort_reason'], + 'waitingApproval': bool(waiting_approval), + 'remainingMinutes': calculate_remaining_time(job) + } + +# Socket Uptime-Überwachung +def log_socket_connection_event(socket_id, status, duration_seconds=None): + """Speichert ein Ereignis zum Verbindungsstatus einer Steckdose""" + event_id = str(uuid.uuid4()) + timestamp = datetime.datetime.utcnow().isoformat() + + db = get_db() + db.execute( + 'INSERT INTO socket_uptime (id, socket_id, timestamp, status, duration_seconds) VALUES (?, ?, ?, ?, ?)', + (event_id, socket_id, timestamp, status, duration_seconds) + ) + db.commit() + app.logger.info(f"Verbindungsstatus für Steckdose {socket_id} geändert: {status}") + + # Aktualisiere auch den Verbindungsstatus in der socket-Tabelle + db.execute( + 'UPDATE socket SET connection_status = ?, last_seen = ? WHERE id = ?', + (status, timestamp if status == 'online' else None, socket_id) + ) + db.commit() + + return event_id + +def get_socket_uptime_events(socket_id=None, limit=100): + """Ruft Verbindungsereignisse für eine oder alle Steckdosen ab""" + db = get_db() + + if socket_id: + rows = db.execute(''' + SELECT su.*, s.name, s.ip_address FROM socket_uptime su + JOIN socket s ON su.socket_id = s.id + WHERE su.socket_id = ? + ORDER BY su.timestamp DESC + LIMIT ? + ''', (socket_id, limit)).fetchall() + else: + rows = db.execute(''' + SELECT su.*, s.name, s.ip_address FROM socket_uptime su + JOIN socket s ON su.socket_id = s.id + ORDER BY su.timestamp DESC + LIMIT ? + ''', (limit,)).fetchall() + + return [dict(row) for row in rows] + +def check_socket_connection(socket_id, timeout=8): + """ + Überprüft die Verbindung zu einer Steckdose und aktualisiert den Status. + + Args: + socket_id: ID der Steckdose + timeout: Timeout in Sekunden, nach dem die Verbindung als fehlgeschlagen gilt + + Returns: + True wenn die Steckdose online ist, sonst False + """ + socket = get_socket_by_id(socket_id) + if not socket or not socket['ip_address']: + return False + + previous_status = socket.get('connection_status', 'unknown') + last_seen = socket.get('last_seen') + + try: + # Verwende den Timeout-Parameter für die Geräteverbindung + device = get_socket_device(socket['ip_address'], timeout=timeout) + if device: + # Verbindung erfolgreich + if previous_status != 'online': + # Status hat sich von offline/unknown auf online geändert + duration = None + if previous_status == 'offline' and last_seen: + # Berechne die Dauer des Ausfalls + try: + offline_since = datetime.datetime.fromisoformat(last_seen) + now = datetime.datetime.utcnow() + duration = int((now - offline_since).total_seconds()) + except (ValueError, TypeError): + # Wenn das Datum nicht geparst werden kann + duration = None + + log_socket_connection_event(socket_id, 'online', duration) + return True + else: + # Keine Verbindung möglich oder Timeout + if previous_status != 'offline': + # Status hat sich von online/unknown auf offline geändert + log_socket_connection_event(socket_id, 'offline') + return False + except Exception as e: + app.logger.error(f"Fehler bei der Überprüfung der Steckdose {socket['ip_address']}: {e}") + if previous_status != 'offline': + log_socket_connection_event(socket_id, 'offline') + return False + +# Steckdosen-Steuerung mit PyP100 +def get_socket_device(ip_address, timeout=8): + """ + Stellt eine Verbindung zu einer Tapo P100-Steckdose her, mit einem konfigurierbaren Timeout. + + Args: + ip_address: IP-Adresse der Steckdose + timeout: Timeout in Sekunden, nach dem die Verbindung als fehlgeschlagen gilt + + Returns: + Das PyP100-Geräteobjekt bei erfolgreicher Verbindung, sonst None + """ + try: + # Nutze Threading mit Timeout für die Verbindung + import threading + import queue + + result_queue = queue.Queue() + + def connect_with_timeout(): + try: + device = PyP100.P100(ip_address, TAPO_USERNAME, TAPO_PASSWORD) + device.handshake() # Erstellt die erforderlichen Cookies + device.login() # Sendet Anmeldedaten und erstellt AES-Schlüssel + result_queue.put(device) + except Exception as e: + app.logger.error(f"Fehler bei der Anmeldung an P100-Gerät {ip_address}: {e}") + result_queue.put(None) + + # Starte den Verbindungsversuch in einem Thread + connect_thread = threading.Thread(target=connect_with_timeout) + connect_thread.daemon = True + connect_thread.start() + + # Warte mit Timeout auf das Ergebnis + try: + device = result_queue.get(timeout=timeout) + if device: + app.logger.info(f"PyP100 Verbindung zu {ip_address} hergestellt") + return device + except queue.Empty: + app.logger.error(f"Timeout bei der Verbindung zu {ip_address} nach {timeout} Sekunden") + return None + + except Exception as e: + app.logger.error(f"Unerwarteter Fehler bei der Anmeldung an P100-Gerät {ip_address}: {e}") + return None + +def turn_on_socket(ip_address, timeout=8): + """ + Schaltet eine Steckdose ein mit konfiguriertem Timeout. + + Args: + ip_address: IP-Adresse der Steckdose + timeout: Timeout in Sekunden für die Verbindung + + Returns: + True bei Erfolg, False bei Fehlern oder Timeout + """ + try: + device = get_socket_device(ip_address, timeout=timeout) + if device: + device.turnOn() + app.logger.info(f"P100-Steckdose {ip_address} eingeschaltet") + return True + return False + except Exception as e: + app.logger.error(f"Fehler beim Einschalten der P100-Steckdose {ip_address}: {e}") + return False + +def turn_off_socket(ip_address, timeout=8): + """ + Schaltet eine Steckdose aus mit konfiguriertem Timeout. + + Args: + ip_address: IP-Adresse der Steckdose + timeout: Timeout in Sekunden für die Verbindung + + Returns: + True bei Erfolg, False bei Fehlern oder Timeout + """ + try: + device = get_socket_device(ip_address, timeout=timeout) + if device: + device.turnOff() + app.logger.info(f"P100-Steckdose {ip_address} ausgeschaltet") + return True + return False + except Exception as e: + app.logger.error(f"Fehler beim Ausschalten der P100-Steckdose {ip_address}: {e}") + return False + +# Authentifizierung und Autorisierung +def get_current_user(): + session_id = flask_session.get('session_id') + if not session_id: + return None + + session = get_session_by_id(session_id) + if not session or datetime.datetime.fromisoformat(session['expires_at']) < datetime.datetime.utcnow(): + if session: + delete_session(session['id']) + flask_session.pop('session_id', None) + return None + + return get_user_by_id(session['user_id']) + +def login_required(f): + @wraps(f) + def decorated(*args, **kwargs): + user = get_current_user() + if not user: + return jsonify({'message': 'Authentifizierung erforderlich!'}), 401 + + g.current_user = user + return f(*args, **kwargs) + + return decorated + +def admin_required(f): + @wraps(f) + def decorated(*args, **kwargs): + if not g.get('current_user') or g.current_user['role'] != 'admin': + return jsonify({'message': 'Admin-Rechte erforderlich!'}), 403 + return f(*args, **kwargs) + + return decorated + +# Authentifizierungs-Routen +@app.route('/auth/register', methods=['POST']) +def register(): + data = request.get_json() + + if not data or not data.get('username') or not data.get('password'): + return jsonify({'message': 'Benutzername und Passwort sind erforderlich!'}), 400 + + username = data.get('username') + password = data.get('password') + display_name = data.get('displayName', username) + email = data.get('email', '') + + if get_user_by_username(username): + return jsonify({'message': 'Benutzername bereits vergeben!'}), 400 + + # Prüfen, ob es bereits einen Admin gibt + db = get_db() + admin_exists = db.execute("SELECT 1 FROM user WHERE role = 'admin' LIMIT 1").fetchone() is not None + + # Falls kein Admin existiert, wird der erste Benutzer zum Admin + role = 'admin' if not admin_exists else 'user' + + user = create_user(username, password, display_name, email, role) + app.logger.info(f'Neuer Benutzer registriert: {username} (Rolle: {role})') + + # Session erstellen + create_session(user['id']) + + return jsonify({ + 'message': 'Registrierung erfolgreich!', + 'user': user_to_dict(user) + }), 201 + +@app.route('/auth/login', methods=['POST']) +def login(): + data = request.get_json() + + if not data or not data.get('username') or not data.get('password'): + return jsonify({'message': 'Benutzername und Passwort sind erforderlich!'}), 400 + + username = data.get('username') + password = data.get('password') + + user = get_user_by_username(username) + + if not user or not check_password(user, password): + return jsonify({'message': 'Ungültiger Benutzername oder Passwort!'}), 401 + + # Session erstellen + create_session(user['id']) + + return jsonify({ + 'message': 'Anmeldung erfolgreich!', + 'user': user_to_dict(user) + }) + +@app.route('/auth/logout', methods=['POST']) +def logout(): + session_id = flask_session.get('session_id') + if session_id: + delete_session(session_id) + flask_session.pop('session_id', None) + + return jsonify({'message': 'Erfolgreich abgemeldet!'}), 200 + +# API-Routen +@app.route('/api/me', methods=['GET']) +def get_me(): + user = get_current_user() + if not user: + return jsonify({'authenticated': False}), 401 + + return jsonify({ + 'authenticated': True, + 'user': user_to_dict(user) + }) + +@app.route('/api/printers', methods=['GET']) +def get_printers(): + sockets = get_all_sockets() + return jsonify([socket_to_dict(socket) for socket in sockets]) + +@app.route('/api/printers', methods=['POST']) +@login_required +@admin_required +def create_printer(): + data = request.get_json() + + if not data or not data.get('name') or not data.get('description'): + return jsonify({'message': 'Name und Beschreibung sind erforderlich!'}), 400 + + socket = create_socket( + name=data.get('name'), + description=data.get('description'), + status=data.get('status', 0), + ip_address=data.get('ipAddress') + ) + + return jsonify(socket_to_dict(socket)), 201 + +@app.route('/api/printers/', methods=['GET']) +def get_printer(printer_id): + socket = get_socket_by_id(printer_id) + if not socket: + return jsonify({'message': 'Steckdose nicht gefunden!'}), 404 + return jsonify(socket_to_dict(socket)) + +@app.route('/api/printers/', methods=['PUT']) +@login_required +@admin_required +def update_printer(printer_id): + socket = get_socket_by_id(printer_id) + if not socket: + return jsonify({'message': 'Steckdose nicht gefunden!'}), 404 + + data = request.get_json() + + updated_socket = update_socket( + printer_id, + name=data.get('name'), + description=data.get('description'), + status=data.get('status') if 'status' in data else None, + ip_address=data.get('ipAddress') + ) + + return jsonify(socket_to_dict(updated_socket)) + +@app.route('/api/printers/', methods=['DELETE']) +@login_required +@admin_required +def delete_printer(printer_id): + socket = get_socket_by_id(printer_id) + if not socket: + return jsonify({'message': 'Steckdose nicht gefunden!'}), 404 + + delete_socket(printer_id) + return jsonify({'message': 'Steckdose gelöscht!'}) + +@app.route('/api/jobs', methods=['GET']) +@login_required +def get_jobs(): + # Admins sehen alle Jobs, normale Benutzer nur ihre eigenen + if g.current_user['role'] == 'admin': + jobs = get_all_jobs() + else: + jobs = get_jobs_by_user(g.current_user['id']) + + return jsonify([job_to_dict(job) for job in jobs]) + +@app.route('/api/jobs', methods=['POST']) +@login_required +def create_job_endpoint(): + data = request.get_json() + + if not data or not data.get('printerId') or not data.get('durationInMinutes'): + return jsonify({'message': 'Steckdosen-ID und Dauer sind erforderlich!'}), 400 + + socket = get_socket_by_id(data['printerId']) + if not socket: + return jsonify({'message': 'Steckdose nicht gefunden!'}), 404 + + duration = int(data['durationInMinutes']) + allow_queued_jobs = data.get('allowQueuedJobs', False) + + # Prüfen, ob der Drucker bereits belegt ist + if socket['status'] != 0: # 0 = available + if allow_queued_jobs: + # Erstelle einen Job, der auf Freischaltung wartet + job = create_job( + socket_id=socket['id'], + user_id=g.current_user['id'], + duration_in_minutes=duration, + comments=data.get('comments', ''), + waiting_approval=1 # Job wartet auf Freischaltung + ) + app.logger.info(f"Wartender Job {job['id']} für belegten Drucker {socket['id']} erstellt.") + return jsonify(job_to_dict(job)), 201 + else: + return jsonify({'message': 'Steckdose ist nicht verfügbar!'}), 400 + + # Normaler Job für verfügbaren Drucker + job = create_job( + socket_id=socket['id'], + user_id=g.current_user['id'], + duration_in_minutes=duration, + comments=data.get('comments', ''), + waiting_approval=0 # Job ist sofort aktiv + ) + + # Steckdose als belegt markieren + update_socket(socket['id'], status=1) # 1 = busy + + # Steckdose einschalten, falls IP-Adresse hinterlegt ist + if socket['ip_address']: + try: + success = turn_on_socket(socket['ip_address']) + if success: + app.logger.info(f"Steckdose {socket['ip_address']} für Job {job['id']} eingeschaltet.") + else: + app.logger.warning(f"Konnte Steckdose {socket['ip_address']} für Job {job['id']} nicht einschalten.") + except Exception as e: + app.logger.error(f"Fehler beim Einschalten der Steckdose {socket['ip_address']}: {e}") + + return jsonify(job_to_dict(job)), 201 + +@app.route('/api/jobs/', methods=['GET']) +@login_required +def get_job_endpoint(job_id): + # Admins können alle Jobs sehen, Benutzer nur ihre eigenen + job = get_job_by_id(job_id) + if not job: + return jsonify({'message': 'Job nicht gefunden!'}), 404 + + if g.current_user['role'] != 'admin' and job['user_id'] != g.current_user['id']: + return jsonify({'message': 'Keine Berechtigung für diesen Job!'}), 403 + + return jsonify(job_to_dict(job)) + +@app.route('/api/jobs//abort', methods=['POST']) +@login_required +def abort_job(job_id): + # Admins können alle Jobs abbrechen, Benutzer nur ihre eigenen + job = get_job_by_id(job_id) + if not job: + return jsonify({'message': 'Job nicht gefunden!'}), 404 + + if g.current_user['role'] != 'admin' and job['user_id'] != g.current_user['id']: + return jsonify({'message': 'Keine Berechtigung für diesen Job!'}), 403 + + data = request.get_json() + + updated_job = update_job(job_id, aborted=True, abort_reason=data.get('reason', '')) + + # Steckdose wieder verfügbar machen + socket = get_socket_by_id(job['socket_id']) + if socket: + update_socket(socket['id'], status=0) # 0 = available + + # Steckdose ausschalten, falls IP-Adresse hinterlegt ist + if socket['ip_address']: + # Mehrmals versuchen, die Steckdose auszuschalten, um sicherzustellen, dass sie wirklich aus ist + max_attempts = 3 + for attempt in range(1, max_attempts + 1): + try: + success = turn_off_socket(socket['ip_address']) + if success: + app.logger.info(f"Steckdose {socket['ip_address']} für abgebrochenen Job {job['id']} ausgeschaltet (Versuch {attempt}).") + break + app.logger.warning(f"Konnte Steckdose {socket['ip_address']} nicht ausschalten (Versuch {attempt}/{max_attempts}).") + except Exception as e: + app.logger.error(f"Fehler beim Ausschalten der Steckdose {socket['ip_address']}: {e} (Versuch {attempt}/{max_attempts})") + + # Nur wenn es nicht der letzte Versuch war, kurz warten und neu versuchen + if attempt < max_attempts: + import time + time.sleep(1) + + return jsonify(job_to_dict(updated_job)) + +@app.route('/api/jobs//finish', methods=['POST']) +@login_required +def finish_job(job_id): + # Admins können alle Jobs beenden, Benutzer nur ihre eigenen + job = get_job_by_id(job_id) + if not job: + return jsonify({'message': 'Job nicht gefunden!'}), 404 + + if g.current_user['role'] != 'admin' and job['user_id'] != g.current_user['id']: + return jsonify({'message': 'Keine Berechtigung für diesen Job!'}), 403 + + # Aktuelle Zeit als Ende setzen + now = datetime.datetime.utcnow() + start_at = datetime.datetime.fromisoformat(job['start_at']) + actual_duration = int((now - start_at).total_seconds() / 60) + + updated_job = update_job(job_id, duration_in_minutes=actual_duration) + + # Steckdose wieder verfügbar machen + socket = get_socket_by_id(job['socket_id']) + if socket: + update_socket(socket['id'], status=0) # 0 = available + + # Steckdose ausschalten, falls IP-Adresse hinterlegt ist + if socket['ip_address']: + # Mehrmals versuchen, die Steckdose auszuschalten, um sicherzustellen, dass sie wirklich aus ist + max_attempts = 3 + for attempt in range(1, max_attempts + 1): + try: + success = turn_off_socket(socket['ip_address']) + if success: + app.logger.info(f"Steckdose {socket['ip_address']} für beendeten Job {job['id']} ausgeschaltet (Versuch {attempt}).") + break + app.logger.warning(f"Konnte Steckdose {socket['ip_address']} nicht ausschalten (Versuch {attempt}/{max_attempts}).") + except Exception as e: + app.logger.error(f"Fehler beim Ausschalten der Steckdose {socket['ip_address']}: {e} (Versuch {attempt}/{max_attempts})") + + # Nur wenn es nicht der letzte Versuch war, kurz warten und neu versuchen + if attempt < max_attempts: + import time + time.sleep(1) + + return jsonify(job_to_dict(updated_job)) + +@app.route('/api/jobs//extend', methods=['POST']) +@login_required +def extend_job(job_id): + # Admins können alle Jobs verlängern, Benutzer nur ihre eigenen + job = get_job_by_id(job_id) + if not job: + return jsonify({'message': 'Job nicht gefunden!'}), 404 + + if g.current_user['role'] != 'admin' and job['user_id'] != g.current_user['id']: + return jsonify({'message': 'Keine Berechtigung für diesen Job!'}), 403 + + data = request.get_json() + minutes = int(data.get('minutes', 0)) + hours = int(data.get('hours', 0)) + + additional_minutes = minutes + (hours * 60) + if additional_minutes <= 0: + return jsonify({'message': 'Ungültige Verlängerungszeit!'}), 400 + + new_duration = job['duration_in_minutes'] + additional_minutes + updated_job = update_job(job_id, duration_in_minutes=new_duration) + + return jsonify(job_to_dict(updated_job)) + +@app.route('/api/jobs//approve', methods=['POST']) +@login_required +def approve_job(job_id): + """Aktiviert einen wartenden Job und schaltet die Steckdose ein.""" + # Nur Admins oder der Job-Ersteller können Jobs freischalten + job = get_job_by_id(job_id) + if not job: + return jsonify({'message': 'Job nicht gefunden!'}), 404 + + if g.current_user['role'] != 'admin' and job['user_id'] != g.current_user['id']: + return jsonify({'message': 'Keine Berechtigung für diesen Job!'}), 403 + + # Prüfen, ob Job auf Freischaltung wartet + waiting_approval = job.get('waiting_approval', 0) + if not waiting_approval: + return jsonify({'message': 'Dieser Job wartet nicht auf Freischaltung!'}), 400 + + # Drucker abrufen + socket = get_socket_by_id(job['socket_id']) + if not socket: + return jsonify({'message': 'Drucker nicht gefunden!'}), 404 + + # Prüfen, ob der Drucker verfügbar ist + if socket['status'] != 0: # 0 = available + return jsonify({'message': 'Drucker ist noch belegt! Bitte warten, bis der laufende Job beendet ist.'}), 400 + + # Job aktualisieren + updated_job = update_job(job_id, waiting_approval=0) + + # Steckdose als belegt markieren + update_socket(socket['id'], status=1) # 1 = busy + + # Steckdose einschalten, falls IP-Adresse hinterlegt ist + if socket['ip_address']: + try: + success = turn_on_socket(socket['ip_address']) + if success: + app.logger.info(f"Steckdose {socket['ip_address']} für freigeschalteten Job {job['id']} eingeschaltet.") + else: + app.logger.warning(f"Konnte Steckdose {socket['ip_address']} für freigeschalteten Job {job['id']} nicht einschalten.") + except Exception as e: + app.logger.error(f"Fehler beim Einschalten der Steckdose {socket['ip_address']}: {e}") + + return jsonify(job_to_dict(updated_job)) + +@app.route('/api/jobs//comments', methods=['PUT']) +@login_required +def update_job_comments(job_id): + # Admins können alle Jobs bearbeiten, Benutzer nur ihre eigenen + job = get_job_by_id(job_id) + if not job: + return jsonify({'message': 'Job nicht gefunden!'}), 404 + + if g.current_user['role'] != 'admin' and job['user_id'] != g.current_user['id']: + return jsonify({'message': 'Keine Berechtigung für diesen Job!'}), 403 + + data = request.get_json() + updated_job = update_job(job_id, comments=data.get('comments', '')) + + return jsonify(job_to_dict(updated_job)) + +@app.route('/api/job//remaining-time', methods=['GET']) +def job_remaining_time(job_id): + job = get_job_by_id(job_id) + if not job: + return jsonify({'message': 'Job nicht gefunden!'}), 404 + + remaining = calculate_remaining_time(job) + + # Wenn die verbleibende Zeit 0 ist und der Job nicht manuell abgebrochen wurde, + # automatisch die Steckdose ausschalten und Status aktualisieren + if remaining == 0 and not job['aborted']: + socket = get_socket_by_id(job['socket_id']) + if socket and socket['status'] == 1: # busy + update_socket(socket['id'], status=0) # available + app.logger.info(f"Job {job['id']} abgelaufen. Steckdose {socket['id']} auf verfügbar gesetzt.") + + # Steckdose ausschalten, falls IP-Adresse hinterlegt ist + if socket['ip_address']: + # Mehrmals versuchen, die Steckdose auszuschalten, um sicherzustellen, dass sie wirklich aus ist + max_attempts = 3 + for attempt in range(1, max_attempts + 1): + try: + success = turn_off_socket(socket['ip_address']) + if success: + app.logger.info(f"Steckdose {socket['ip_address']} für abgelaufenen Job {job['id']} automatisch ausgeschaltet (Versuch {attempt}).") + break + app.logger.warning(f"Konnte Steckdose {socket['ip_address']} nicht ausschalten (Versuch {attempt}/{max_attempts}).") + except Exception as e: + app.logger.error(f"Fehler beim Ausschalten der Steckdose {socket['ip_address']}: {e} (Versuch {attempt}/{max_attempts})") + + # Nur wenn es nicht der letzte Versuch war, kurz warten und neu versuchen + if attempt < max_attempts: + import time + time.sleep(1) + + return jsonify({ + 'remaining_minutes': remaining, + 'job_status': 'completed' if remaining == 0 else 'active', + 'socket_status': 'available' if remaining == 0 else 'busy' + }) + +@app.route('/api/users', methods=['GET']) +@login_required +@admin_required +def get_users(): + users = get_all_users() + return jsonify([user_to_dict(user) for user in users]) + +@app.route('/api/users/', methods=['GET']) +@login_required +@admin_required +def get_user(user_id): + user = get_user_by_id(user_id) + if not user: + return jsonify({'message': 'Benutzer nicht gefunden!'}), 404 + return jsonify(user_to_dict(user)) + +@app.route('/api/users/', methods=['PUT']) +@login_required +@admin_required +def update_user_endpoint(user_id): + user = get_user_by_id(user_id) + if not user: + return jsonify({'message': 'Benutzer nicht gefunden!'}), 404 + + data = request.get_json() + updated_user = update_user( + user_id, + username=data.get('username'), + password=data.get('password'), + display_name=data.get('displayName'), + email=data.get('email'), + role=data.get('role') + ) + + return jsonify(user_to_dict(updated_user)) + +@app.route('/api/users/', methods=['DELETE']) +@login_required +@admin_required +def delete_user_endpoint(user_id): + user = get_user_by_id(user_id) + if not user: + return jsonify({'message': 'Benutzer nicht gefunden!'}), 404 + + # Löschen aller Sessions des Benutzers + delete_sessions_by_user(user_id) + + delete_user(user_id) + return jsonify({'message': 'Benutzer gelöscht!'}) + +@app.route('/api/stats', methods=['GET']) +@login_required +@admin_required +def stats(): + db = get_db() + + # Steckdosen-Nutzungsstatistiken + total_sockets = db.execute('SELECT COUNT(*) as count FROM socket').fetchone()['count'] + available_sockets = db.execute('SELECT COUNT(*) as count FROM socket WHERE status = 0').fetchone()['count'] + + # Verbindungsstatistiken + online_sockets = db.execute("SELECT COUNT(*) as count FROM socket WHERE connection_status = 'online'").fetchone()['count'] + offline_sockets = db.execute("SELECT COUNT(*) as count FROM socket WHERE connection_status = 'offline'").fetchone()['count'] + unknown_sockets = db.execute("SELECT COUNT(*) as count FROM socket WHERE connection_status = 'unknown' OR connection_status IS NULL").fetchone()['count'] + + # Job-Statistiken + total_jobs = db.execute('SELECT COUNT(*) as count FROM job').fetchone()['count'] + + now = datetime.datetime.utcnow().isoformat() + active_jobs = db.execute(''' + SELECT COUNT(*) as count FROM job + WHERE aborted = 0 + AND datetime(start_at, '+' || duration_in_minutes || ' minutes') > datetime(?) + ''', (now,)).fetchone()['count'] + + completed_jobs = db.execute(''' + SELECT COUNT(*) as count FROM job + WHERE aborted = 0 + AND datetime(start_at, '+' || duration_in_minutes || ' minutes') <= datetime(?) + ''', (now,)).fetchone()['count'] + + # Benutzerstatistiken + total_users = db.execute('SELECT COUNT(*) as count FROM user').fetchone()['count'] + + # Durchschnittliche Druckdauer + avg_duration_result = db.execute('SELECT AVG(duration_in_minutes) as avg FROM job').fetchone() + avg_duration = int(avg_duration_result['avg']) if avg_duration_result['avg'] else 0 + + # Steckdosen-Fehlerstatistiken (letzten 7 Tage) + seven_days_ago = (datetime.datetime.utcnow() - timedelta(days=7)).isoformat() + outages = db.execute(''' + SELECT COUNT(*) as count FROM socket_uptime + WHERE status = 'offline' + AND timestamp > ? + ''', (seven_days_ago,)).fetchone()['count'] + + # Steckdosen mit aktuellen Problemen + problem_sockets = db.execute(''' + SELECT s.name, s.connection_status, s.last_seen + FROM socket s + WHERE s.connection_status = 'offline' + ''').fetchall() + + return jsonify({ + 'printers': { + 'total': total_sockets, + 'available': available_sockets, + 'utilization_rate': (total_sockets - available_sockets) / total_sockets if total_sockets > 0 else 0, + 'online': online_sockets, + 'offline': offline_sockets, + 'unknown': unknown_sockets, + 'connectivity_rate': online_sockets / total_sockets if total_sockets > 0 else 0 + }, + 'jobs': { + 'total': total_jobs, + 'active': active_jobs, + 'completed': completed_jobs, + 'avg_duration': avg_duration + }, + 'users': { + 'total': total_users + }, + 'uptime': { + 'outages_last_7_days': outages, + 'problem_printers': [{'name': row['name'], 'status': row['connection_status'], 'last_seen': row['last_seen']} for row in problem_sockets] + } + }) + +@app.route('/api/uptime', methods=['GET']) +@login_required +@admin_required +def uptime_stats(): + """Liefert detaillierte Uptime-Statistiken für das Dashboard.""" + socket_id = request.args.get('socket_id') + limit = int(request.args.get('limit', 100)) + + # Rufe die letzten Uptime-Ereignisse ab + events = get_socket_uptime_events(socket_id, limit) + + # Gruppiere Ereignisse nach Steckdose + sockets = {} + for event in events: + socket_id = event['socket_id'] + if socket_id not in sockets: + sockets[socket_id] = { + 'id': socket_id, + 'name': event['name'], + 'ip_address': event['ip_address'], + 'events': [] + } + + # Füge Ereignis zur Steckdosenliste hinzu + sockets[socket_id]['events'].append({ + 'id': event['id'], + 'timestamp': event['timestamp'], + 'status': event['status'], + 'duration_seconds': event['duration_seconds'] + }) + + # Hole den aktuellen Status aller Steckdosen + all_sockets = get_all_sockets() + current_status = {} + for socket in all_sockets: + current_status[socket['id']] = { + 'connection_status': socket.get('connection_status', 'unknown'), + 'last_seen': socket.get('last_seen') + } + + # Füge den aktuellen Status zu den Socket-Informationen hinzu + for socket_id, socket_data in sockets.items(): + if socket_id in current_status: + socket_data['current_status'] = current_status[socket_id] + + return jsonify({ + 'sockets': list(sockets.values()) + }) + +# Regelmäßige Überprüfung der Jobs und automatische Abschaltung der Steckdosen +def check_jobs(): + """Überprüft abgelaufene Jobs und schaltet Steckdosen automatisch aus.""" + with app.app_context(): + expired_jobs = get_expired_jobs() + handled_jobs = 0 + + for job in expired_jobs: + socket = get_socket_by_id(job['socket_id']) + + if socket and socket['status'] == 1: # busy + update_socket(socket['id'], status=0) # available + app.logger.info(f"Job {job['id']} abgelaufen. Steckdose {socket['id']} auf verfügbar gesetzt.") + handled_jobs += 1 + + # Steckdose ausschalten, falls IP-Adresse hinterlegt ist + if socket['ip_address']: + # Mehrmals versuchen, die Steckdose auszuschalten, um sicherzustellen, dass sie wirklich aus ist + max_attempts = 3 + for attempt in range(1, max_attempts + 1): + try: + success = turn_off_socket(socket['ip_address']) + if success: + app.logger.info(f"Steckdose {socket['ip_address']} für abgelaufenen Job {job['id']} automatisch ausgeschaltet (Versuch {attempt}).") + break + app.logger.warning(f"Konnte Steckdose {socket['ip_address']} nicht ausschalten (Versuch {attempt}/{max_attempts}).") + except Exception as e: + app.logger.error(f"Fehler beim Ausschalten der Steckdose {socket['ip_address']}: {e} (Versuch {attempt}/{max_attempts})") + + # Nur wenn es nicht der letzte Versuch war, kurz warten und neu versuchen + if attempt < max_attempts: + time.sleep(1) + + app.logger.info(f"{len(expired_jobs)} abgelaufene Jobs überprüft, {handled_jobs} Steckdosen aktualisiert.") + +def check_socket_connections(): + """Überprüft periodisch die Verbindung zu allen Steckdosen mit 8-Sekunden-Timeout.""" + with app.app_context(): + sockets = get_all_sockets() + app.logger.info(f"Überprüfe Verbindungsstatus von {len(sockets)} Steckdosen") + + online_count = 0 + offline_count = 0 + skipped_count = 0 + + for socket in sockets: + if not socket['ip_address']: + skipped_count += 1 + continue # Überspringe Steckdosen ohne IP-Adresse + + is_online = check_socket_connection(socket['id']) + if is_online: + online_count += 1 + else: + offline_count += 1 + app.logger.warning(f"Steckdose {socket['name']} ({socket['ip_address']}) ist nicht erreichbar") + + app.logger.info(f"Verbindungsüberprüfung abgeschlossen: {online_count} online, {offline_count} offline, {skipped_count} übersprungen") + +# Hintergrund-Thread für das Job-Polling und Steckdosen-Monitoring +def background_job_checker(): + """Hintergrund-Thread, der regelmäßig abgelaufene Jobs und Steckdosenverbindungen überprüft.""" + app.logger.info("Starte Hintergrund-Thread für Job-Überprüfung und Steckdosen-Monitoring") + + # Standardintervall für Socket-Überprüfungen (2 Minuten) + socket_check_interval = int(os.environ.get('SOCKET_CHECK_INTERVAL', '120')) + last_socket_check = 0 + + while True: + try: + # Überprüfe Jobs bei jedem Durchlauf + check_jobs() + + # Überprüfe Steckdosen in regelmäßigen Intervallen + current_time = time.time() + if current_time - last_socket_check >= socket_check_interval: + # Socket-Überprüfung mit 8-Sekunden-Timeout pro Gerät + check_socket_connections() + last_socket_check = current_time + app.logger.info(f"Nächste Socket-Überprüfung in {socket_check_interval} Sekunden") + + except Exception as e: + app.logger.error(f"Fehler im Hintergrund-Thread: {e}") + + # Pause zwischen den Überprüfungen + time.sleep(app.config['JOB_CHECK_INTERVAL']) + +# CLI-Befehle für manuelle Ausführung +@app.cli.command("check-jobs") +def cli_check_jobs(): + """CLI-Befehl zur manuellen Überprüfung abgelaufener Jobs.""" + check_jobs() + +@app.cli.command("check-sockets") +def cli_check_sockets(): + """CLI-Befehl zur manuellen Überprüfung aller Steckdosenverbindungen.""" + check_socket_connections() + +@app.route('/api/job//status', methods=['GET']) +def job_status(job_id): + """Endpunkt zum Überprüfen des Status eines Jobs für Frontend-Polling.""" + job = get_job_by_id(job_id) + if not job: + return jsonify({'message': 'Job nicht gefunden!'}), 404 + + remaining = calculate_remaining_time(job) + socket = get_socket_by_id(job['socket_id']) + socket_status = socket['status'] if socket else None + + # Wenn die verbleibende Zeit 0 ist und der Job nicht manuell abgebrochen wurde, + # automatisch die Steckdose ausschalten und Status aktualisieren + if remaining == 0 and not job['aborted'] and socket and socket['status'] == 1: + # Update socket status to available + update_socket(socket['id'], status=0) + socket_status = 0 + app.logger.info(f"Job {job['id']} abgelaufen. Steckdose {socket['id']} auf verfügbar gesetzt.") + + # Steckdose ausschalten, falls IP-Adresse hinterlegt ist + if socket['ip_address']: + # Mehrmals versuchen, die Steckdose auszuschalten, um sicherzustellen, dass sie wirklich aus ist + max_attempts = 3 + for attempt in range(1, max_attempts + 1): + try: + success = turn_off_socket(socket['ip_address']) + if success: + app.logger.info(f"Steckdose {socket['ip_address']} für abgelaufenen Job {job['id']} automatisch ausgeschaltet (Versuch {attempt}).") + break + app.logger.warning(f"Konnte Steckdose {socket['ip_address']} nicht ausschalten (Versuch {attempt}/{max_attempts}).") + except Exception as e: + app.logger.error(f"Fehler beim Ausschalten der Steckdose {socket['ip_address']}: {e} (Versuch {attempt}/{max_attempts})") + + # Nur wenn es nicht der letzte Versuch war, kurz warten und neu versuchen + if attempt < max_attempts: + import time + time.sleep(1) + + job_status = 'aborted' if job['aborted'] else ('completed' if remaining == 0 else 'active') + + return jsonify({ + 'job': job_to_dict(job), + 'status': job_status, + 'socketStatus': 'available' if socket_status == 0 else 'busy', + 'remainingMinutes': remaining + }) + +@app.route('/api/test', methods=['GET']) +def test(): + return jsonify({'message': 'MYP Backend API funktioniert!'}) + +@app.route('/api/create-initial-admin', methods=['POST']) +def create_initial_admin(): + db = get_db() + admin_exists = db.execute("SELECT 1 FROM user WHERE role = 'admin' LIMIT 1").fetchone() is not None + + if admin_exists: + return jsonify({'message': 'Es existiert bereits ein Administrator!'}), 400 + + data = request.get_json() + + if not data or not data.get('username') or not data.get('password'): + return jsonify({'message': 'Benutzername und Passwort sind erforderlich!'}), 400 + + username = data.get('username') + password = data.get('password') + display_name = data.get('displayName', username) + email = data.get('email', '') + + user = create_user(username, password, display_name, email, 'admin') + app.logger.info(f'Initialer Admin-Benutzer erstellt: {username}') + + return jsonify({ + 'message': 'Administrator wurde erfolgreich erstellt!', + 'user': user_to_dict(user) + }), 201 + +# Error Handler +@app.errorhandler(404) +def not_found(error): + return jsonify({'message': 'Nicht gefunden!'}), 404 + +@app.errorhandler(500) +def server_error(error): + app.logger.error(f'Serverfehler: {error}') + return jsonify({'message': 'Interner Serverfehler!'}), 500 + +# Web UI Routen +@app.route('/') +def index(): + current_user = get_current_user() + if current_user: + return render_template('dashboard.html', current_user=current_user, active_page='home') + return redirect(url_for('login_page')) + +@app.route('/login') +def login_page(): + return render_template('login.html', active_page='login') + +@app.route('/register') +def register_page(): + return render_template('register.html', active_page='register') + +@app.route('/logout') +def logout_page(): + session_id = flask_session.get('session_id') + if session_id: + delete_session(session_id) + flask_session.pop('session_id', None) + + flash('Sie wurden erfolgreich abgemeldet.', 'success') + return redirect(url_for('login_page')) + +@app.route('/admin/printers') +def printers_page(): + current_user = get_current_user() + if not current_user: + return redirect(url_for('login_page')) + return render_template('printers.html', current_user=current_user, active_page='printers') + +@app.route('/admin/jobs') +def jobs_page(): + current_user = get_current_user() + if not current_user: + return redirect(url_for('login_page')) + return render_template('jobs.html', current_user=current_user, active_page='jobs') + +@app.route('/admin/users') +def users_page(): + current_user = get_current_user() + if not current_user or current_user['role'] != 'admin': + flash('Sie haben keine Berechtigung, diese Seite zu besuchen.', 'danger') + return redirect(url_for('index')) + return render_template('users.html', current_user=current_user, active_page='users') + +@app.route('/admin/stats') +def stats_page(): + current_user = get_current_user() + if not current_user or current_user['role'] != 'admin': + flash('Sie haben keine Berechtigung, diese Seite zu besuchen.', 'danger') + return redirect(url_for('index')) + return render_template('stats.html', current_user=current_user, active_page='stats') + +# Initialisierung und Start des Hintergrund-Threads beim ersten Request +with app.app_context(): + # Diese Funktion wird nach dem App-Start aber vor dem ersten Request ausgeführt + @app.before_request + def initialize_background_tasks(): + """Startet den Hintergrund-Thread für Job-Überprüfung beim ersten Request.""" + # Überprüfung, ob dieser Handler bereits ausgeführt wurde + if getattr(app, '_job_thread_initialized', False): + return + + # Starte den Hintergrund-Thread nur, wenn er noch nicht läuft + for thread in threading.enumerate(): + if thread.name == 'job_checker_thread': + app.logger.info("Hintergrund-Thread für Job-Überprüfung läuft bereits") + app._job_thread_initialized = True + return + + # Thread starten + job_thread = threading.Thread(target=background_job_checker, daemon=True, name='job_checker_thread') + job_thread.start() + app.logger.info("Hintergrund-Thread für Job-Überprüfung beim ersten Request gestartet") + app._job_thread_initialized = True + +# Server starten +if __name__ == '__main__': + with app.app_context(): + init_db() + if PRINTERS: + # Initialisiere Drucker und schalte alle Steckdosen beim Start aus + init_printers() + + # Starte den Hintergrund-Thread für die Job-Überprüfung + job_thread = threading.Thread(target=background_job_checker, daemon=True, name='job_checker_thread') + job_thread.start() + app.logger.info("Hintergrund-Thread für Job-Überprüfung gestartet") + + app.run(debug=True, host='0.0.0.0') \ No newline at end of file diff --git a/backend/development/crontab-example b/backend/development/crontab-example new file mode 100644 index 000000000..e133188b5 --- /dev/null +++ b/backend/development/crontab-example @@ -0,0 +1,8 @@ +# MYP Backend Cron-Jobs +# Installiere mit: crontab crontab-example + +# Prüfe alle 5 Minuten auf abgelaufene Reservierungen und schalte Steckdosen aus +*/5 * * * * cd /pfad/zum/projektarbeit-myp/backend && /pfad/zur/venv/bin/flask check-jobs >> /pfad/zum/projektarbeit-myp/backend/logs/cron.log 2>&1 + +# Tägliche Sicherung der Datenbank um 3:00 Uhr +0 3 * * * cd /pfad/zum/projektarbeit-myp/backend && cp instance/myp.db instance/backups/myp-$(date +\%Y\%m\%d).db \ No newline at end of file diff --git a/backend/development/initialize_myp_database.sh b/backend/development/initialize_myp_database.sh new file mode 100644 index 000000000..51e00fc14 --- /dev/null +++ b/backend/development/initialize_myp_database.sh @@ -0,0 +1,84 @@ +#!/bin/bash + +# MYP Datenbank Initialisierungs-Skript +# Dieses Skript erstellt die erforderlichen Datenbanktabellen für das MYP Backend + +echo "=== MYP Datenbank Initialisierung ===" +echo "" + +# Prüfe, ob sqlite3 installiert ist +if ! command -v sqlite3 &> /dev/null; then + echo "FEHLER: sqlite3 ist nicht installiert." + echo "Bitte installiere sqlite3 mit deinem Paketmanager, z.B. 'apt install sqlite3'" + exit 1 +fi + +# Erstelle Instance-Ordner, falls nicht vorhanden +echo "Erstelle instance-Ordner, falls nicht vorhanden..." +mkdir -p instance/backups + +# Prüfen, ob die Datenbank bereits existiert +if [ -f "instance/myp.db" ]; then + echo "Datenbank existiert bereits." + echo "Erstelle Backup in instance/backups..." + cp instance/myp.db "instance/backups/myp_$(date '+%Y%m%d_%H%M%S').db" +fi + +# Erstelle die Datenbank und ihre Tabellen +echo "Erstelle neue Datenbank..." +sqlite3 instance/myp.db <&1 | awk '{print $2}') +echo "Python-Version: $python_version" + +# Prüfe, ob die Python-Version mindestens 3.8 ist +required_version="3.8.0" +if [[ "$(printf '%s\n' "$required_version" "$python_version" | sort -V | head -n1)" != "$required_version" ]]; then + echo "FEHLER: Python $required_version oder höher wird benötigt" + exit 1 +fi + +# Prüfe, ob sqlite3 installiert ist +if ! command -v sqlite3 &> /dev/null; then + echo "FEHLER: sqlite3 ist nicht installiert." + echo "Bitte installiere sqlite3 mit deinem Paketmanager, z.B. 'apt install sqlite3'" + exit 1 +fi + +# Erstelle virtuelle Umgebung +echo "" +echo "Erstelle virtuelle Python-Umgebung..." +python3 -m venv venv +source venv/bin/activate + +# Installiere Abhängigkeiten +echo "" +echo "Installiere Abhängigkeiten..." +pip install --upgrade pip +pip install -r requirements.txt + +# Erstelle .env-Datei +echo "" +echo "Erstelle .env-Datei..." +if [ ! -f .env ]; then + cp .env.example .env + echo "Die .env-Datei wurde aus der Beispieldatei erstellt." + echo "Bitte passe die Konfiguration an, falls nötig." +else + echo ".env-Datei existiert bereits." +fi + +# Erstelle Logs-Ordner +echo "" +echo "Erstelle logs-Ordner..." +mkdir -p logs + +# Initialisiere die Datenbank +echo "" +echo "Initialisiere die Datenbank..." +bash initialize_myp_database.sh + +echo "" +echo "=== Installation abgeschlossen ===" +echo "" +echo "Wichtige Schritte vor dem Start:" +echo "1. Passe die Konfigurationen in der .env-Datei an" +echo "2. Konfiguriere die Tapo-Steckdosen-Zugangsdaten in der .env-Datei (optional)" +echo "3. Passe die crontab-example an und installiere den Cron-Job (optional)" +echo "" +echo "Starte den Server mit:" +echo "source venv/bin/activate" +echo "python app.py" +echo "" +echo "Oder mit Gunicorn für Produktion:" +echo "gunicorn --bind 0.0.0.0:5000 app:app" +echo "" \ No newline at end of file diff --git a/backend/development/tests/api-test.drucker.py b/backend/development/tests/api-test.drucker.py new file mode 100644 index 000000000..e39339047 --- /dev/null +++ b/backend/development/tests/api-test.drucker.py @@ -0,0 +1,95 @@ +import requests +import json + +# Basis-URL inkl. Token +url = "http://192.168.0.102:80/app?token=9DFAC92C53CEC92E67A9CB2E00B3CB2F" + +# HTTP-Header wie in der Originalanfrage +headers = { + "Referer": "http://192.168.0.102:80", + "Accept": "application/json", + "requestByApp": "true", + "Content-Type": "application/json; charset=UTF-8", + "Host": "192.168.0.102", + "Connection": "Keep-Alive", + "Accept-Encoding": "gzip", + "User-Agent": "okhttp/3.14.9" +} + +# Liste der Payloads (als Python-Dictionaries) +payloads = [ + { + "method": "securePassthrough", + "params": { + "request": ( + "ZC4CHp6bbfBO1rtmuH6I+TStBIiFRfQpayYPwet5NBmL35dib5xXHeEeLM7c0OSQSyxO6fnbXrC1\n" + "gXdfowwwq4Fum9ispgt8yT7cgbDcqnoVrhxEtHIDfuwLh8YAGmDSfTMo/JlsGspWPYMKd1EWXtb5\n" + "gP9FA9LHnV2kxKsNSPQ=\n" + ) + } + }, + { + "method": "securePassthrough", + "params": { + "request": ( + "k111EbfCcfVzAouNbu1vyos9Ltsg+a97n4xUUQMviQVJfhqxvKOhv1SrvEk2LvpD0LwNVUNPZdwU\n" + "6pH5E/NOwdc1WzTPeqHiY760GpUuqn0tToHEHEyO2HaSKdrAYnw2gN410bvHb0pM3gYWS43eOA==\n" + ) + } + }, + { + "method": "securePassthrough", + "params": { + "request": ( + "7/uYVDwyNfFhg9y7rHyp+4AGKBYQPyaBN6cFMl9j4ER/JpJTcGBdaUteSmx8P8Fkz+b2kkNLjYa2\n" + "wQr2gA3m6vEq9jpnAF2V3fv9c4Yg9gja9MlTIZqM6EdMi7YbfbhLme34Bh8kMcohDR3u1F4DwFDz\n" + "hNZPckf/CegbY9KGFeGwT4rWyX3BTk9+FE7ldtJn\n" + ) + } + }, + { + "method": "securePassthrough", + "params": { + "request": ( + "EjWZb+YYS9tihgLdX4x+Wwx7q+e5X/ZHicr4jOnYmpFToDANzpm5ZpzD49BITcTCdQMOHlJBis85\n" + "9GX6Hv8j66OITyH0XmfG9dQo2tgIykyagCZIofr/BpAWYX4aRaOkU4z14mVa2XpDtHJQjc+pXYkh\n" + "JuWvLE+h01U5RoyPtvE=\n" + ) + } + }, + { + "method": "securePassthrough", + "params": { + "request": ( + "OwyTsm5HdB/ReJMhVRrkjnV0NLTanw6iXOxVrDDexT456edWuwKiBOsZUyBHmUyJKgiPQzOXqyWWi220bX8IjLX4q8YNgPwRlj+7nRbfzpC/I57wBZBTWIt626pSdIH0vpiuPq84KMfPD5BB2p78/LjsqlzyeLGYzkSsGRBMT8TnLMDFzZE864nfDUZ9muH2kk8NRMN9l6xoCXBJqGA9q8XxIWRTpsl0kTx52kUszY69hYlfFSrrCDIls1ykul14/T1NtOVF8KOgiwaSGOZf7L4QlbhYvRj9kkVVkrxhlwt8jtMqfJKEqq+CIPh3Mp4440WYMLRo6VNIEJ3pWjplkJmc+htnYC4FwVgT7mHZ8eeGGKBvsJz+78gTaHnGBnwZ26I8UdFparyp6QXpOhK9zFmGVh0yapiTHo6jOOI+4Q3Ru+aPnidX/ZASPmR7CZO70CUpvv9zIKJnrAaoTMmH7A6+kcmCRLgLFaTaM+4DFmiz6JGP+4W7MmVPJxxvn0IFlo1P/xwNDuL3T6GLUIEVNk89JG5roBm7AdchUZJO38dGZ0eFiiTK/NhKPvjj+fk9A4FGh7EDshXZhL2u50cdLcdUtcP/CAMDjgWlMm4Kk3vxMQO+UGE+jsB7NkaulmTW1jcl+PSnAE5P71oqVVQ0ng==\n" + ) + } + }, + { + "method": "securePassthrough", + "params": { + "request": ( + "7/uYVDwyNfFhg9y7rHyp+4AGKBYQPyaBN6cFMl9j4ER/JpJTcGBdaUteSmx8P8FkURmv/LWV1FpO\n" + "M3RWvsiC5UAsei2G+vwTVuQpOPjKKAx+qwftr9Qs2mSkPNjNLpWHK68EZkIw+h04TQkt0Q99Dirg\n" + "0BcrPgHTVKjiK8mdZ6w6gcld/h/FOKYMqJrP0Z+2\n" + ) + } + }, + { + "method": "securePassthrough", + "params": { + "request": ( + "ZE/+XlUmTA9D3DFfp4x3xhS3vdsQ+60tz4TOodtZDby/4DPoqk9EBvJZ1JtUCr5c0AHuv/sfwcvN\n" + "Vx1zJP9RkltrAKVTWoaESAeewLozpXt/x0s/jkYC1rh7eTrxm+nYTZ5LJgNtcQq8yJxhEPez1w==\n" + ) + } + } +] + +# Sende die Payloads sequenziell per POST-Anfrage +for idx, payload in enumerate(payloads, start=1): + response = requests.post(url, headers=headers, data=json.dumps(payload)) + print(f"Anfrage {idx}:") + print("Status Code:", response.status_code) + print("Response Text:", response.text) + print("-" * 60) diff --git a/backend/development/tests/capture.pcap b/backend/development/tests/capture.pcap new file mode 100644 index 000000000..1ec8bc2c4 Binary files /dev/null and b/backend/development/tests/capture.pcap differ diff --git a/backend/development/tests/handshake.py b/backend/development/tests/handshake.py new file mode 100644 index 000000000..a5e354c17 --- /dev/null +++ b/backend/development/tests/handshake.py @@ -0,0 +1,128 @@ +import requests +import json + +# Constants from the Wireshark capture +PUBLIC_KEY = """-----BEGIN PUBLIC KEY----- +MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQCMl89OZsjqE8yZ9TQhUb9h539WTX3U8Y5YCNdp +OhuXvLFYcAT5mvC074VFROmD0xhvw5hrwESOisqpPPU9r78JpLuYUKd+/aidvykqBT8OW5rDLb6d +O9FO6Gc+bV8L8ttHVlBFoX69EqiRhcreGPG6FQz4JqGJF4T1nFi0EvALXwIDAQAB +-----END PUBLIC KEY-----""" + +# Vorbereitete verschlüsselte Befehle (aus Wireshark extrahiert) +COMMAND_ON = """ps0Puxc37EK4PhfcevceL3lyyDrjwLT1+443DDXNbcNRsltlgCQ6+oXgsrE2Pl5OhV73ZI/oM5Nj +37cWEaHpXPiHdr1W0cD3aJ5qJ55TfTRkHP9xcMNQJHCn6aWPEHpR7xvvXW9WbJWfShnE2Xdvmw== +""" + +COMMAND_OFF = """FlO5i3DRcrUmu2ZwIIv8b68EisGu8VCuqfGOydaR+xCA0n3f2W/EcqVj8MurRBFXYTrZ/uwa1W26 +ftCfvhdXNebBRwHr9Rj3id4bVfltJ8eT5/R3xY8kputklW2mrw9UfdISzAJqOPp9KZcU4K9p8g== +""" + +class TapoP115Controller: + def __init__(self, device_ip): + self.device_ip = device_ip + self.session_id = None + self.token = None + + def perform_handshake(self): + """Führt den ersten Handshake durch und speichert die Session-ID""" + handshake_data = { + "method": "handshake", + "params": { + "key": PUBLIC_KEY + }, + "requestTimeMils": 0 + } + + headers = { + "Referer": f"http://{self.device_ip}:80", + "Accept": "application/json", + "requestByApp": "true", + "Content-Type": "application/json; charset=UTF-8" + } + + response = requests.post( + f"http://{self.device_ip}/app", + json=handshake_data, + headers=headers + ) + + if response.status_code == 200: + data = response.json() + if data["error_code"] == 0: + # Session-ID aus dem Cookie extrahieren + self.session_id = response.cookies.get("TP_SESSIONID") + print(f"Handshake erfolgreich, Session-ID: {self.session_id}") + + # In einem echten Szenario würden wir hier den verschlüsselten Schlüssel entschlüsseln + # Da wir keinen privaten Schlüssel haben, speichern wir nur die Antwort + encrypted_key = data["result"]["key"] + print(f"Verschlüsselter Schlüssel: {encrypted_key}") + return True + + print("Handshake fehlgeschlagen") + return False + + def send_command(self, encrypted_command): + """Sendet einen vorbereiteten verschlüsselten Befehl""" + if not self.session_id: + print("Keine Session-ID. Bitte zuerst Handshake durchführen.") + return None + + # Token aus der Wireshark-Aufnahme (könnte sich ändern, oder vom Gerät abhängen) + token = "9DFAC92C53CEC92E67A9CB2E00B3CB2F" + + secure_data = { + "method": "securePassthrough", + "params": { + "request": encrypted_command + } + } + + headers = { + "Referer": f"http://{self.device_ip}:80", + "Accept": "application/json", + "requestByApp": "true", + "Content-Type": "application/json; charset=UTF-8", + "Cookie": f"TP_SESSIONID={self.session_id}" + } + + response = requests.post( + f"http://{self.device_ip}/app?token={token}", + json=secure_data, + headers=headers + ) + + if response.status_code == 200: + data = response.json() + if data["error_code"] == 0: + # In einem echten Szenario würden wir die Antwort entschlüsseln + encrypted_response = data["result"]["response"] + print("Befehl erfolgreich gesendet") + return encrypted_response + + print("Fehler beim Senden des Befehls") + return None + + def turn_on(self): + """Schaltet die Steckdose ein""" + return self.send_command(COMMAND_ON) + + def turn_off(self): + """Schaltet die Steckdose aus""" + return self.send_command(COMMAND_OFF) + +# Verwendungsbeispiel +if __name__ == "__main__": + controller = TapoP115Controller("192.168.0.102") + + # Handshake durchführen + if controller.perform_handshake(): + # Steckdose einschalten + controller.turn_on() + + # Kurze Pause (im echten Code mit time.sleep) + print("Steckdose ist eingeschaltet") + + # Steckdose ausschalten + controller.turn_off() + print("Steckdose ist ausgeschaltet") \ No newline at end of file diff --git a/backend/development/tests/tapo.py b/backend/development/tests/tapo.py new file mode 100644 index 000000000..2ab740e01 --- /dev/null +++ b/backend/development/tests/tapo.py @@ -0,0 +1,9 @@ +from PyP100 import PyP100 + +p100 = PyP100.P100("192.168.0.102", "till.tomczak@mercedes-benz.com", "Agent045") #Creates a P100 plug object + +p100.handshake() #Creates the cookies required for further methods +p100.login() #Sends credentials to the plug and creates AES Key and IV for further methods + +p100.turnOn() #Turns the connected plug on +p100.turnOff() #Turns the connected plug off \ No newline at end of file diff --git a/backend/development/tests/tests.py b/backend/development/tests/tests.py new file mode 100644 index 000000000..75ae6fdcb --- /dev/null +++ b/backend/development/tests/tests.py @@ -0,0 +1,253 @@ +import unittest +import json +import os +import tempfile +from datetime import datetime, timedelta + +from app import app, db, User, Printer, PrintJob + +class MYPBackendTestCase(unittest.TestCase): + def setUp(self): + # Temporäre Datenbank für Tests + self.db_fd, app.config['DATABASE'] = tempfile.mkstemp() + app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///' + app.config['DATABASE'] + app.config['TESTING'] = True + self.app = app.test_client() + + # Datenbank-Tabellen erstellen und Test-Daten einfügen + with app.app_context(): + db.create_all() + + # Admin-Benutzer erstellen + admin = User(username='admin_test', email='admin@test.com', role='admin') + admin.set_password('admin') + db.session.add(admin) + + # Normaler Benutzer erstellen + user = User(username='user_test', email='user@test.com', role='user') + user.set_password('user') + db.session.add(user) + + # Drucker erstellen + printer1 = Printer(name='Printer 1', location='Room A', type='3D', + status='available', description='Test printer 1') + printer2 = Printer(name='Printer 2', location='Room B', type='3D', + status='busy', description='Test printer 2') + db.session.add(printer1) + db.session.add(printer2) + + # Job erstellen + start_time = datetime.utcnow() + end_time = start_time + timedelta(minutes=60) + job = PrintJob(title='Test Job', start_time=start_time, end_time=end_time, + duration=60, status='active', comments='Test job', + user_id=2, printer_id=2) + db.session.add(job) + + db.session.commit() + + def tearDown(self): + # Aufräumen nach dem Test + os.close(self.db_fd) + os.unlink(app.config['DATABASE']) + + def get_token(self, username, password): + response = self.app.post('/api/auth/login', + data=json.dumps({'username': username, 'password': password}), + content_type='application/json') + data = json.loads(response.data) + return data.get('token') + + def test_login(self): + # Test: Erfolgreicher Login + response = self.app.post('/api/auth/login', + data=json.dumps({'username': 'admin_test', 'password': 'admin'}), + content_type='application/json') + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertIn('token', data) + self.assertIn('user', data) + + # Test: Fehlgeschlagener Login (falsches Passwort) + response = self.app.post('/api/auth/login', + data=json.dumps({'username': 'admin_test', 'password': 'wrong'}), + content_type='application/json') + self.assertEqual(response.status_code, 401) + + def test_register(self): + # Test: Erfolgreiche Registrierung + response = self.app.post('/api/auth/register', + data=json.dumps({ + 'username': 'new_user', + 'email': 'new@test.com', + 'password': 'password' + }), + content_type='application/json') + self.assertEqual(response.status_code, 201) + + # Test: Doppelte Registrierung + response = self.app.post('/api/auth/register', + data=json.dumps({ + 'username': 'new_user', + 'email': 'another@test.com', + 'password': 'password' + }), + content_type='application/json') + self.assertEqual(response.status_code, 400) + + def test_get_printers(self): + # Test: Drucker abrufen + response = self.app.get('/api/printers') + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertEqual(len(data), 2) + + def test_get_single_printer(self): + # Test: Einzelnen Drucker abrufen + response = self.app.get('/api/printers/1') + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertEqual(data['name'], 'Printer 1') + + def test_create_printer(self): + # Als Admin einen Drucker erstellen + token = self.get_token('admin_test', 'admin') + response = self.app.post('/api/printers', + headers={'Authorization': f'Bearer {token}'}, + data=json.dumps({ + 'name': 'New Printer', + 'location': 'Room C', + 'type': '3D', + 'description': 'New test printer' + }), + content_type='application/json') + self.assertEqual(response.status_code, 201) + data = json.loads(response.data) + self.assertEqual(data['name'], 'New Printer') + + def test_update_printer(self): + # Als Admin einen Drucker aktualisieren + token = self.get_token('admin_test', 'admin') + response = self.app.put('/api/printers/1', + headers={'Authorization': f'Bearer {token}'}, + data=json.dumps({ + 'name': 'Updated Printer', + 'location': 'Room D' + }), + content_type='application/json') + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertEqual(data['name'], 'Updated Printer') + self.assertEqual(data['location'], 'Room D') + + def test_delete_printer(self): + # Als Admin einen Drucker löschen + token = self.get_token('admin_test', 'admin') + response = self.app.delete('/api/printers/1', + headers={'Authorization': f'Bearer {token}'}) + self.assertEqual(response.status_code, 200) + + # Überprüfen, ob der Drucker wirklich gelöscht wurde + response = self.app.get('/api/printers/1') + self.assertEqual(response.status_code, 404) + + def test_get_jobs_as_admin(self): + # Als Admin alle Jobs abrufen + token = self.get_token('admin_test', 'admin') + response = self.app.get('/api/jobs', + headers={'Authorization': f'Bearer {token}'}) + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertEqual(len(data), 1) + + def test_get_jobs_as_user(self): + # Als normaler Benutzer nur eigene Jobs abrufen + token = self.get_token('user_test', 'user') + response = self.app.get('/api/jobs', + headers={'Authorization': f'Bearer {token}'}) + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertEqual(len(data), 1) # Der Benutzer hat einen Job + + def test_create_job(self): + # Als Benutzer einen Job erstellen + token = self.get_token('user_test', 'user') + response = self.app.post('/api/jobs', + headers={'Authorization': f'Bearer {token}'}, + data=json.dumps({ + 'title': 'New Job', + 'printer_id': 1, + 'duration': 30, + 'comments': 'Test job creation' + }), + content_type='application/json') + self.assertEqual(response.status_code, 201) + data = json.loads(response.data) + self.assertEqual(data['title'], 'New Job') + self.assertEqual(data['duration'], 30) + + def test_update_job(self): + # Als Benutzer den eigenen Job aktualisieren + token = self.get_token('user_test', 'user') + response = self.app.put('/api/jobs/1', + headers={'Authorization': f'Bearer {token}'}, + data=json.dumps({ + 'comments': 'Updated comments', + 'duration': 15 # Verlängerung + }), + content_type='application/json') + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertEqual(data['comments'], 'Updated comments') + self.assertEqual(data['duration'], 75) # 60 + 15 + + def test_complete_job(self): + # Als Benutzer einen Job als abgeschlossen markieren + token = self.get_token('user_test', 'user') + response = self.app.put('/api/jobs/1', + headers={'Authorization': f'Bearer {token}'}, + data=json.dumps({ + 'status': 'completed' + }), + content_type='application/json') + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertEqual(data['status'], 'completed') + + # Überprüfen, ob der Drucker wieder verfügbar ist + response = self.app.get('/api/printers/2') + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertEqual(data['status'], 'available') + + def test_get_remaining_time(self): + # Test: Verbleibende Zeit für einen aktiven Job abrufen + response = self.app.get('/api/job/1/remaining-time') + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertIn('remaining_minutes', data) + # Der genaue Wert kann nicht überprüft werden, da er von der Zeit abhängt + + def test_stats(self): + # Als Admin Statistiken abrufen + token = self.get_token('admin_test', 'admin') + response = self.app.get('/api/stats', + headers={'Authorization': f'Bearer {token}'}) + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertIn('printers', data) + self.assertIn('jobs', data) + self.assertIn('users', data) + self.assertEqual(data['printers']['total'], 2) + self.assertEqual(data['jobs']['total'], 1) + self.assertEqual(data['users']['total'], 2) + + def test_test_endpoint(self): + # Test: API-Test-Endpunkt + response = self.app.get('/api/test') + self.assertEqual(response.status_code, 200) + data = json.loads(response.data) + self.assertEqual(data['message'], 'MYP Backend API funktioniert!') + +if __name__ == '__main__': + unittest.main() \ No newline at end of file diff --git a/backend/docker-compose.yml b/backend/docker-compose.yml new file mode 100644 index 000000000..6297b924e --- /dev/null +++ b/backend/docker-compose.yml @@ -0,0 +1,15 @@ +version: '3' + +services: + backend: + build: . + container_name: myp-backend + ports: + - "5000:5000" + environment: + - SECRET_KEY=change_me_in_production + - DATABASE_URL=sqlite:///myp.db + volumes: + - ./logs:/app/logs + - ./instance:/app/instance + restart: unless-stopped \ No newline at end of file diff --git a/backend/docs/API_DOCS.md b/backend/docs/API_DOCS.md new file mode 100644 index 000000000..0e43753df --- /dev/null +++ b/backend/docs/API_DOCS.md @@ -0,0 +1,647 @@ +# MYP Backend API-Dokumentation + +Dieses Dokument beschreibt detailliert die API-Endpunkte des MYP (Manage Your Printer) Backend-Systems. + +## Basis-URL + +Die Basis-URL für alle API-Anfragen ist: `http://localhost:5000` (Entwicklungsumgebung) oder die URL, unter der die Anwendung gehostet wird. + +## Authentifizierung + +Die meisten Endpunkte erfordern eine Authentifizierung. Diese erfolgt über Cookies/Sessions, die bei der Anmeldung erstellt werden. Die Session wird für 7 Tage gespeichert. + +### Benutzerregistrierung + +**Endpunkt:** `POST /auth/register` + +**Beschreibung:** Registriert einen neuen Benutzer im System. + +**Request-Body:** +```json +{ + "username": "string", // Erforderlich + "password": "string", // Erforderlich + "displayName": "string", // Optional (Standard: username) + "email": "string" // Optional +} +``` + +**Erfolgsantwort:** +```json +{ + "message": "Registrierung erfolgreich!", + "user": { + "id": "string", + "username": "string", + "displayName": "string", + "email": "string", + "role": "string" + } +} +``` + +**Fehlerantwort:** +```json +{ + "message": "Benutzername bereits vergeben!" +} +``` + +### Benutzeranmeldung + +**Endpunkt:** `POST /auth/login` + +**Beschreibung:** Meldet einen Benutzer an und erstellt eine Session. + +**Request-Body:** +```json +{ + "username": "string", // Erforderlich + "password": "string" // Erforderlich +} +``` + +**Erfolgsantwort:** +```json +{ + "message": "Anmeldung erfolgreich!", + "user": { + "id": "string", + "username": "string", + "displayName": "string", + "email": "string", + "role": "string" + } +} +``` + +**Fehlerantwort:** +```json +{ + "message": "Ungültiger Benutzername oder Passwort!" +} +``` + +### Initialer Administrator + +**Endpunkt:** `POST /api/create-initial-admin` + +**Beschreibung:** Erstellt einen initialen Admin-Benutzer, falls noch keiner existiert. + +**Request-Body:** +```json +{ + "username": "string", // Erforderlich + "password": "string", // Erforderlich + "displayName": "string", // Optional (Standard: username) + "email": "string" // Optional +} +``` + +**Erfolgsantwort:** +```json +{ + "message": "Administrator wurde erfolgreich erstellt!", + "user": { + "id": "string", + "username": "string", + "displayName": "string", + "email": "string", + "role": "string" + } +} +``` + +**Fehlerantwort:** +```json +{ + "message": "Es existiert bereits ein Administrator!" +} +``` + +## Benutzer-Endpunkte + +### Alle Benutzer abrufen (Admin) + +**Endpunkt:** `GET /api/users` + +**Beschreibung:** Gibt eine Liste aller Benutzer zurück. + +**Erforderliche Rechte:** Admin + +**Erfolgsantwort:** +```json +[ + { + "id": 1, + "username": "string", + "email": "string", + "role": "string" + } +] +``` + +### Benutzer abrufen (Admin) + +**Endpunkt:** `GET /api/users/{userId}` + +**Beschreibung:** Gibt die Details eines bestimmten Benutzers zurück. + +**Erforderliche Rechte:** Admin + +**Erfolgsantwort:** +```json +{ + "id": 1, + "username": "string", + "email": "string", + "role": "string" +} +``` + +**Fehlerantwort:** +```json +{ + "message": "Nicht gefunden!" +} +``` + +### Benutzer aktualisieren (Admin) + +**Endpunkt:** `PUT /api/users/{userId}` + +**Beschreibung:** Aktualisiert die Daten eines Benutzers. + +**Erforderliche Rechte:** Admin + +**Request-Body:** +```json +{ + "username": "string", + "email": "string", + "password": "string", + "role": "string" +} +``` + +**Erfolgsantwort:** +```json +{ + "id": 1, + "username": "string", + "email": "string", + "role": "string" +} +``` + +**Fehlerantwort:** +```json +{ + "message": "Benutzername bereits vergeben!" +} +``` + +### Benutzer löschen (Admin) + +**Endpunkt:** `DELETE /api/users/{userId}` + +**Beschreibung:** Löscht einen Benutzer. + +**Erforderliche Rechte:** Admin + +**Erfolgsantwort:** +```json +{ + "message": "Benutzer gelöscht!" +} +``` + +## Drucker-Endpunkte + +### Alle Drucker abrufen + +**Endpunkt:** `GET /api/printers` + +**Beschreibung:** Gibt eine Liste aller Drucker (Steckdosen) zurück. + +**Erfolgsantwort:** +```json +[ + { + "id": "uuid-string", + "name": "string", + "description": "string", + "status": 0, // 0 = available, 1 = busy + "latestJob": { + // Job-Objekt oder null, wenn kein aktiver Job + } + } +] +``` + +### Drucker hinzufügen (Admin) + +**Endpunkt:** `POST /api/printers` + +**Beschreibung:** Fügt einen neuen Drucker hinzu. + +**Erforderliche Rechte:** Admin + +**Request-Body:** +```json +{ + "name": "string", + "description": "string", + "ipAddress": "string" // IP-Adresse der Tapo-Steckdose +} +``` + +**Erfolgsantwort:** +```json +{ + "id": "uuid-string", + "name": "string", + "description": "string", + "status": 0, // 0 = available, 1 = busy + "latestJob": null +} +``` + +### Drucker abrufen + +**Endpunkt:** `GET /api/printers/{printerId}` + +**Beschreibung:** Gibt die Details eines bestimmten Druckers zurück. + +**Erfolgsantwort:** +```json +{ + "id": "uuid-string", + "name": "string", + "description": "string", + "status": 0, // 0 = available, 1 = busy + "latestJob": { + // Job-Objekt oder null, wenn kein aktiver Job + } +} +``` + +**Fehlerantwort:** +```json +{ + "message": "Nicht gefunden!" +} +``` + +### Drucker aktualisieren (Admin) + +**Endpunkt:** `PUT /api/printers/{printerId}` + +**Beschreibung:** Aktualisiert die Daten eines Druckers. + +**Erforderliche Rechte:** Admin + +**Request-Body:** +```json +{ + "name": "string", + "description": "string", + "ipAddress": "string", // IP-Adresse der Tapo-Steckdose + "status": 0 // 0 = available, 1 = busy +} +``` + +**Erfolgsantwort:** +```json +{ + "id": "uuid-string", + "name": "string", + "description": "string", + "status": 0, // 0 = available, 1 = busy + "latestJob": { + // Job-Objekt oder null, wenn kein aktiver Job + } +} +``` + +### Drucker löschen (Admin) + +**Endpunkt:** `DELETE /api/printers/{printerId}` + +**Beschreibung:** Löscht einen Drucker. + +**Erforderliche Rechte:** Admin + +**Erfolgsantwort:** +```json +{ + "message": "Drucker gelöscht!" +} +``` + +## Druckauftrags-Endpunkte + +### Alle Druckaufträge abrufen + +**Endpunkt:** `GET /api/jobs` + +**Beschreibung:** Gibt eine Liste aller Druckaufträge zurück (für Admins) oder der eigenen Druckaufträge (für Benutzer). + +**Erfolgsantwort:** +```json +[ + { + "id": "uuid-string", + "socketId": "uuid-string", + "userId": "uuid-string", + "startAt": "string (ISO 8601)", + "durationInMinutes": 60, + "comments": "string", + "aborted": false, + "abortReason": null, + "remainingMinutes": 30 + } +] +``` + +### Druckauftrag erstellen + +**Endpunkt:** `POST /api/jobs` + +**Beschreibung:** Erstellt einen neuen Druckauftrag. + +**Request-Body:** +```json +{ + "printerId": "uuid-string", + "durationInMinutes": 60, + "comments": "string" +} +``` + +**Erfolgsantwort:** +```json +{ + "id": "uuid-string", + "socketId": "uuid-string", + "userId": "uuid-string", + "startAt": "string (ISO 8601)", + "durationInMinutes": 60, + "comments": "string", + "aborted": false, + "abortReason": null, + "remainingMinutes": 60 +} +``` + +**Fehlerantwort:** +```json +{ + "message": "Drucker ist nicht verfügbar!" +} +``` + +### Druckauftrag abrufen + +**Endpunkt:** `GET /api/jobs/{jobId}` + +**Beschreibung:** Gibt die Details eines bestimmten Druckauftrags zurück. + +**Erfolgsantwort:** +```json +{ + "id": "uuid-string", + "socketId": "uuid-string", + "userId": "uuid-string", + "startAt": "string (ISO 8601)", + "durationInMinutes": 60, + "comments": "string", + "aborted": false, + "abortReason": null, + "remainingMinutes": 30 +} +``` + +**Fehlerantwort:** +```json +{ + "message": "Nicht gefunden!" +} +``` + +### Druckauftrag Kommentare aktualisieren + +**Endpunkt:** `PUT /api/jobs/{jobId}/comments` + +**Beschreibung:** Aktualisiert die Kommentare eines Druckauftrags. + +**Request-Body:** +```json +{ + "comments": "string" +} +``` + +**Erfolgsantwort:** +```json +{ + "id": "uuid-string", + "socketId": "uuid-string", + "userId": "uuid-string", + "startAt": "string (ISO 8601)", + "durationInMinutes": 60, + "comments": "string", + "aborted": false, + "abortReason": null, + "remainingMinutes": 30 +} +``` + +### Druckauftrag abbrechen + +**Endpunkt:** `POST /api/jobs/{jobId}/abort` + +**Beschreibung:** Bricht einen laufenden Druckauftrag ab. + +**Request-Body:** +```json +{ + "reason": "string" // Optional +} +``` + +**Erfolgsantwort:** +```json +{ + "id": "uuid-string", + "socketId": "uuid-string", + "userId": "uuid-string", + "startAt": "string (ISO 8601)", + "durationInMinutes": 60, + "comments": "string", + "aborted": true, + "abortReason": "string", + "remainingMinutes": 0 +} +``` + +### Druckauftrag vorzeitig beenden + +**Endpunkt:** `POST /api/jobs/{jobId}/finish` + +**Beschreibung:** Beendet einen laufenden Druckauftrag vorzeitig. + +**Erfolgsantwort:** +```json +{ + "id": "uuid-string", + "socketId": "uuid-string", + "userId": "uuid-string", + "startAt": "string (ISO 8601)", + "durationInMinutes": 45, // Tatsächliche Dauer bis zum Beenden + "comments": "string", + "aborted": false, + "abortReason": null, + "remainingMinutes": 0 +} +``` + +### Druckauftrag verlängern + +**Endpunkt:** `POST /api/jobs/{jobId}/extend` + +**Beschreibung:** Verlängert die Laufzeit eines Druckauftrags. + +**Request-Body:** +```json +{ + "minutes": 30, // Zusätzliche Minuten + "hours": 0 // Zusätzliche Stunden (optional) +} +``` + +**Erfolgsantwort:** +```json +{ + "id": "uuid-string", + "socketId": "uuid-string", + "userId": "uuid-string", + "startAt": "string (ISO 8601)", + "durationInMinutes": 90, // Aktualisierte Gesamtdauer + "comments": "string", + "aborted": false, + "abortReason": null, + "remainingMinutes": 60 +} +``` + +### Druckauftrag löschen + +**Endpunkt:** `DELETE /api/jobs/{jobId}` + +**Beschreibung:** Löscht einen Druckauftrag. + +**Erfolgsantwort:** +```json +{ + "message": "Druckauftrag gelöscht!" +} +``` + +### Verbleibende Zeit eines Druckauftrags abrufen + +**Endpunkt:** `GET /api/job/{jobId}/remaining-time` + +**Beschreibung:** Gibt die verbleibende Zeit eines aktiven Druckauftrags in Minuten zurück. + +**Erfolgsantwort:** +```json +{ + "remaining_minutes": 30, + "job_status": "active", // active, completed + "socket_status": "busy" // busy, available +} +``` + +### Status eines Druckauftrags abrufen + +**Endpunkt:** `GET /api/job/{jobId}/status` + +**Beschreibung:** Gibt detaillierte Statusinformationen zu einem Druckauftrag zurück. + +**Erfolgsantwort:** +```json +{ + "job": { + "id": "uuid-string", + "socketId": "uuid-string", + "userId": "uuid-string", + "startAt": "string (ISO 8601)", + "durationInMinutes": 60, + "comments": "string", + "aborted": false, + "abortReason": null, + "remainingMinutes": 30 + }, + "status": "active", // active, completed, aborted + "socketStatus": "busy", // busy, available + "remainingMinutes": 30 +} +``` + +## Statistik-Endpunkte + +### Systemstatistiken abrufen (Admin) + +**Endpunkt:** `GET /api/stats` + +**Beschreibung:** Gibt Statistiken zu Druckern, Aufträgen und Benutzern zurück. + +**Erforderliche Rechte:** Admin + +**Erfolgsantwort:** +```json +{ + "printers": { + "total": 10, + "available": 5, + "utilization_rate": 0.5 + }, + "jobs": { + "total": 100, + "active": 5, + "completed": 90, + "avg_duration": 120 + }, + "users": { + "total": 50 + } +} +``` + +## Test-Endpunkt + +### API-Test + +**Endpunkt:** `GET /api/test` + +**Beschreibung:** Testet, ob die API funktioniert. + +**Erfolgsantwort:** +```json +{ + "message": "MYP Backend API funktioniert!" +} +``` + +## Fehlercodes + +| Statuscode | Beschreibung | +|------------|-----------------------------| +| 200 | OK - Anfrage erfolgreich | +| 201 | Created - Ressource erstellt | +| 400 | Bad Request - Ungültige Anfrage | +| 401 | Unauthorized - Authentifizierung erforderlich | +| 403 | Forbidden - Unzureichende Rechte | +| 404 | Not Found - Ressource nicht gefunden | +| 500 | Internal Server Error - Serverfehler | \ No newline at end of file diff --git a/backend/docs/PROJEKTDOKUMENTATION.md b/backend/docs/PROJEKTDOKUMENTATION.md new file mode 100644 index 000000000..822af80ea --- /dev/null +++ b/backend/docs/PROJEKTDOKUMENTATION.md @@ -0,0 +1,213 @@ +# MYP - Projektdokumentation für das IHK-Abschlussprojekt + +## Projektübersicht + +**Projektname:** MYP (Manage Your Printer) +**Projekttyp:** IHK-Abschlussprojekt für Fachinformatiker für digitale Vernetzung +**Zeitraum:** [Dein Projektzeitraum] +**Team:** 2 Personen (Frontend- und Backend-Entwicklung) + +## Projektziel + +Das Ziel des Projektes ist die Entwicklung einer Reservierungs- und Steuerungsplattform für 3D-Drucker, die es Benutzern ermöglicht, Drucker zu reservieren und deren Stromversorgung automatisch über WLAN-Steckdosen (Tapo P115) zu steuern. Die Plattform soll eine einfache Verwaltung der Drucker und ihrer Auslastung bieten sowie den Stromverbrauch optimieren, indem Drucker nur während aktiver Reservierungen mit Strom versorgt werden. + +## Aufgabenbeschreibung + +Als Fachinformatiker für digitale Vernetzung besteht meine Aufgabe in der Entwicklung des Backend-Systems, das folgende Funktionen bereitstellt: + +1. **API-Backend für das Frontend**: Entwicklung einer RESTful API, die mit dem Frontend kommuniziert und alle notwendigen Daten bereitstellt. + +2. **Authentifizierungssystem**: Integration einer OAuth-Authentifizierung über GitHub, um Benutzer zu identifizieren und Zugriffskontrolle zu gewährleisten. + +3. **Datenbankverwaltung**: Erstellung und Verwaltung der Datenbankmodelle für Benutzer, Drucker und Reservierungen. + +4. **Steckdosensteuerung**: Implementierung einer Schnittstelle zu Tapo P115 WLAN-Steckdosen, um die Stromversorgung der Drucker basierend auf Reservierungen zu steuern. + +5. **Automatisierung**: Entwicklung von Mechanismen zur automatischen Überwachung von Reservierungen und Steuerung der Steckdosen. + +6. **Sicherheit**: Implementierung von Sicherheitsmaßnahmen zum Schutz der Anwendung und der Daten. + +7. **Dokumentation**: Erstellung einer umfassenden Dokumentation für Entwicklung, Installation und Nutzung des Systems. + +## Technische Umsetzung + +### Backend (Mein Verantwortungsbereich) + +#### Verwendete Technologien + +- **Programmiersprache**: Python 3.11 +- **Web-Framework**: Flask 2.3.3 +- **Datenbank-ORM**: SQLAlchemy 3.1.1 +- **Datenbank**: SQLite (für Entwicklung), erweiterbar auf PostgreSQL für Produktion +- **Authentifizierung**: Authlib für GitHub OAuth +- **Steckdosen-Steuerung**: Tapo Python Library +- **Container-Technologie**: Docker und Docker Compose + +#### Architektur + +Die Backend-Anwendung folgt einer klassischen dreischichtigen Architektur: + +1. **Datenmodell-Schicht**: SQLAlchemy ORM-Modelle für Benutzer, Sessions, Drucker und Druckaufträge +2. **Business-Logic-Schicht**: Implementierung der Geschäftslogik für Reservierungsverwaltung und Steckdosensteuerung +3. **API-Schicht**: RESTful API-Endpunkte, die vom Frontend konsumiert werden + +Zusätzlich wurden folgende Features implementiert: + +- **OAuth-Authentifizierung**: Implementierung einer sicheren Authentifizierung über GitHub +- **Session-Management**: Server-seitige Session-Verwaltung für Benutzerauthentifizierung +- **Steckdosensteuerung**: Asynchrone Steuerung der Tapo P115 WLAN-Steckdosen +- **CLI-Befehle**: Flask CLI-Befehle für automatisierte Aufgaben wie die Überprüfung abgelaufener Reservierungen + +#### Datenmodell + +Das Datenmodell besteht aus vier Hauptentitäten: + +1. **User**: Benutzer mit GitHub-Authentifizierung und Rollenverwaltung +2. **Session**: Sitzungsdaten für die Authentifizierung +3. **Printer**: Drucker mit Status und IP-Adresse der zugehörigen Steckdose +4. **PrintJob**: Reservierungen mit Start- und Endzeit, Dauer und Status + +#### API-Endpunkte + +Die API wurde speziell entwickelt, um nahtlos mit dem bestehenden Frontend zusammenzuarbeiten. Sie bietet Endpunkte für: + +- Authentifizierung und Benutzerverwaltung +- Druckerverwaltung +- Reservierungsverwaltung (Erstellen, Abbrechen, Verlängern) +- Statusinformationen wie verbleibende Zeit + +#### Steckdosensteuerung + +Die Steuerung der Tapo P115 WLAN-Steckdosen erfolgt über die Tapo Python Library. Das System: + +- Schaltet Steckdosen bei Erstellung einer Reservierung ein +- Schaltet Steckdosen bei Abbruch oder Beendigung einer Reservierung aus +- Überprüft regelmäßig abgelaufene Reservierungen und schaltet die entsprechenden Steckdosen aus + +#### Automatisierung + +Das System implementiert mehrere Automatisierungsmechanismen: + +- **Automatische Steckdosensteuerung**: Ein- und Ausschalten der Steckdosen basierend auf Reservierungsstatus +- **Job-Überprüfung**: CLI-Befehl `flask check-jobs` zur regelmäßigen Überprüfung abgelaufener Reservierungen +- **Logging**: Automatische Protokollierung aller Aktionen zur Fehlerdiagnose + +### Frontend (Verantwortungsbereich des Teampartners) + +Das Frontend wurde von meinem Teampartner entwickelt und besteht aus: + +- Next.js-Anwendung mit React-Komponenten +- Tailwind CSS für das Styling +- Serverless Functions für API-Integrationen +- Responsive Design für Desktop- und Mobile-Nutzung + +## Projektergebnisse + +Das Projekt hat erfolgreich eine funktionsfähige Reservierungs- und Steuerungsplattform für 3D-Drucker geschaffen, die es Benutzern ermöglicht: + +1. Sich über GitHub zu authentifizieren +2. Verfügbare Drucker zu sehen und zu reservieren +3. Ihre Reservierungen zu verwalten (verlängern, abbrechen, kommentieren) +4. Als Administrator Drucker und Benutzer zu verwalten + +Technische Errungenschaften: + +1. Nahtlose Integration mit dem Frontend +2. Erfolgreiche Implementierung der Steckdosensteuerung +3. Sichere Authentifizierung über GitHub OAuth +4. Optimierte Stromnutzung durch automatische Steckdosensteuerung + +## Herausforderungen und Lösungen + +### Herausforderung 1: GitHub OAuth-Integration + +Die Integration der GitHub-Authentifizierung, insbesondere mit GitHub Enterprise, erforderte eine sorgfältige Konfiguration der OAuth-Einstellungen und URL-Anpassungen. + +**Lösung:** Implementierung mit Authlib und anpassbaren Konfigurationsoptionen für verschiedene GitHub-Instanzen. + +### Herausforderung 2: Tapo P115 Steuerung + +Die Kommunikation mit den Tapo P115 WLAN-Steckdosen erforderte eine zuverlässige und asynchrone Implementierung. + +**Lösung:** Verwendung der Tapo Python Library mit asynchronem Handling und robusten Fehlerbehandlungsmechanismen. + +### Herausforderung 3: Kompatibilität mit bestehendem Frontend + +Das Backend musste mit dem bereits entwickelten Frontend kompatibel sein, was eine genaue Anpassung der API-Endpunkte und Datenstrukturen erforderte. + +**Lösung:** Sorgfältige Analyse des Frontend-Codes, um die erwarteten API-Strukturen zu verstehen und das Backend entsprechend zu implementieren. + +### Herausforderung 4: Automatische Steckdosensteuerung + +Die zuverlässige Steuerung der Steckdosen bei abgelaufenen Reservierungen war eine Herausforderung. + +**Lösung:** Implementierung eines CLI-Befehls, der regelmäßig durch Cron-Jobs ausgeführt werden kann, um abgelaufene Reservierungen zu überprüfen. + +## Fachliche Reflexion + +Das Projekt erforderte ein breites Spektrum an Fähigkeiten aus dem Bereich der digitalen Vernetzung: + +1. **Netzwerkkommunikation**: Implementierung der Kommunikation zwischen Backend, Frontend und WLAN-Steckdosen über verschiedene Protokolle. + +2. **Systemintegration**: Integration verschiedener Systeme (GitHub OAuth, Datenbank, Tapo-Steckdosen) zu einer kohärenten Anwendung. + +3. **API-Design**: Entwicklung einer RESTful API, die den Anforderungen des Frontends entspricht und zukunftssicher ist. + +4. **Datenbankentwurf**: Erstellung eines optimierten Datenbankschemas für die Anwendung. + +5. **Sicherheitskonzepte**: Implementierung von Sicherheitsmaßnahmen wie OAuth, Session-Management und Zugriffskontrollen. + +6. **Automatisierung**: Entwicklung von Automatisierungsprozessen für die Steckdosensteuerung und Job-Überwachung. + +Diese Aspekte entsprechen direkt den Kernkompetenzen des Berufsbildes "Fachinformatiker für digitale Vernetzung" und zeigen die praktische Anwendung dieser Fähigkeiten in einem realen Projekt. + +## Ausblick und Weiterentwicklung + +Das System bietet verschiedene Möglichkeiten zur Weiterentwicklung: + +1. **Erweiterung der Steckdosenunterstützung**: Integration weiterer Smart-Home-Geräte neben Tapo P115. + +2. **Benachrichtigungssystem**: Implementierung von E-Mail- oder Push-Benachrichtigungen für Reservierungserinnerungen. + +3. **Erweiterte Statistiken**: Detailliertere Nutzungsstatistiken und Visualisierungen für Administratoren. + +4. **Mobile App**: Entwicklung einer nativen mobilen App für iOS und Android. + +5. **Verbesserte Automatisierung**: Integration mit weiteren Systemen wie 3D-Drucker-APIs für direktes Monitoring des Druckstatus. + +## Fazit + +Das MYP-Projekt zeigt erfolgreich, wie moderne Webtechnologien und IoT-Geräte kombiniert werden können, um eine praktische Lösung für die Verwaltung von 3D-Druckern zu schaffen. + +Als angehender Fachinformatiker für digitale Vernetzung konnte ich meine Fähigkeiten in den Bereichen Programmierung, Systemintegration, Netzwerkkommunikation und Automatisierung anwenden und erweitern. + +Die Zusammenarbeit im Team mit klarer Aufgabenteilung (Frontend/Backend) hat zu einem erfolgreichen Projektergebnis geführt, das die gestellten Anforderungen erfüllt und einen praktischen Nutzen bietet. + +--- + +## Anhang + +### Installation und Einrichtung + +Detaillierte Anweisungen zur Installation und Einrichtung des Backend-Systems finden sich in der README.md-Datei. + +### Wichtige Konfigurationsparameter + +Die folgenden Umgebungsvariablen müssen konfiguriert werden: + +- `SECRET_KEY`: Geheimer Schlüssel für die Session-Verschlüsselung +- `DATABASE_URL`: URL zur Datenbank +- `OAUTH_CLIENT_ID`: GitHub OAuth Client ID +- `OAUTH_CLIENT_SECRET`: GitHub OAuth Client Secret +- `GITHUB_API_BASE_URL`, `GITHUB_AUTHORIZE_URL`, `GITHUB_TOKEN_URL`: URLs für GitHub OAuth +- `TAPO_USERNAME`: Benutzername für die Tapo-Steckdosen +- `TAPO_PASSWORD`: Passwort für die Tapo-Steckdosen +- `TAPO_DEVICES`: JSON-Objekt mit der Zuordnung von Drucker-IDs zu IP-Adressen + +### Cron-Job-Einrichtung + +Für die automatische Überprüfung abgelaufener Jobs kann folgender Cron-Job eingerichtet werden: + +``` +*/5 * * * * cd /pfad/zum/projekt && /pfad/zur/venv/bin/flask check-jobs >> /pfad/zum/projekt/logs/cron.log 2>&1 +``` \ No newline at end of file diff --git a/backend/docs/README.md b/backend/docs/README.md new file mode 100644 index 000000000..0da735c53 --- /dev/null +++ b/backend/docs/README.md @@ -0,0 +1,185 @@ +# MYP Backend-Steuerungsplattform + +Dies ist das Backend für das MYP (Manage Your Printer) Projekt, ein IHK-Abschlussprojekt für Fachinformatiker für digitale Vernetzung. Die Plattform ist mit Python und Flask implementiert und stellt eine RESTful API zur Verfügung, die es ermöglicht, 3D-Drucker zu verwalten, zu reservieren und über WLAN-Steckdosen (Tapo P115) zu steuern. + +## Funktionen + +- Lokales Authentifizierungssystem (Offline-fähig) +- Rollen-basierte Zugriffskontrolle (Admin/User/Guest) +- Druckerverwaltung (Hinzufügen, Bearbeiten, Löschen) +- Reservierungsverwaltung (Erstellen, Abbrechen, Verlängern) +- Fernsteuerung von WLAN-Steckdosen (Tapo P115) +- Statistikerfassung und -anzeige +- RESTful API für die Kommunikation mit dem Frontend + +## Technologie-Stack + +- **Python**: Programmiersprache +- **Flask**: Web-Framework +- **SQLite**: Integrierte Datenbank (kann für Produktion durch PostgreSQL ersetzt werden) +- **PyP100**: Python-Bibliothek zur Steuerung der Tapo P115 WLAN-Steckdosen +- **Gunicorn**: WSGI HTTP Server für die Produktionsumgebung +- **Docker**: Containerisierung der Anwendung + +## Projekt-Struktur + +- `app.py`: Hauptanwendungsdatei mit allen Routen und Modellen +- `requirements.txt`: Liste aller Python-Abhängigkeiten +- `Dockerfile`: Docker-Konfiguration +- `docker-compose.yml`: Docker Compose Konfiguration für einfaches Deployment +- `.env.example`: Beispiel für die Umgebungsvariablen +- `logs/`: Logdateien (automatisch erstellt) +- `instance/`: SQLite-Datenbank (automatisch erstellt) + +## Installation und Ausführung + +### Lokal (Entwicklung) + +1. Python 3.8 oder höher installieren +2. Repository klonen +3. Ins Projektverzeichnis wechseln +4. Virtuelle Umgebung erstellen (optional, aber empfohlen) + ``` + python -m venv venv + source venv/bin/activate # Unter Windows: venv\Scripts\activate + ``` +5. Abhängigkeiten installieren + ``` + pip install -r requirements.txt + ``` +6. `.env.example` nach `.env` kopieren und anpassen + ``` + cp .env.example .env + ``` +7. Anwendung starten + ``` + python app.py + ``` + +Die Anwendung ist dann unter http://localhost:5000 erreichbar. + +### Mit Docker + +1. Docker und Docker Compose installieren +2. Ins Projektverzeichnis wechseln +3. `.env.example` nach `.env` kopieren und anpassen + ``` + cp .env.example .env + ``` +4. Anwendung starten + ``` + docker-compose up -d + ``` + +Die Anwendung ist dann unter http://localhost:5000 erreichbar. + +## API-Endpunkte + +### Authentifizierung + +- `POST /auth/register`: Neuen Benutzer registrieren +- `POST /auth/login`: Benutzer anmelden +- `POST /auth/logout`: Abmelden und Session beenden +- `POST /api/create-initial-admin`: Initialen Administrator erstellen +- `GET /api/me`: Aktuelle Benutzerinformationen abrufen + +### Benutzer + +- `GET /api/users`: Liste aller Benutzer (Admin) +- `GET /api/users/`: Details zu einem Benutzer (Admin) +- `PUT /api/users/`: Benutzer aktualisieren (Admin) +- `DELETE /api/users/`: Benutzer löschen (Admin) + +### Drucker + +- `GET /api/printers`: Liste aller Drucker +- `POST /api/printers`: Drucker hinzufügen (Admin) +- `GET /api/printers/`: Details zu einem Drucker +- `PUT /api/printers/`: Drucker aktualisieren (Admin) +- `DELETE /api/printers/`: Drucker löschen (Admin) + +### Druckaufträge + +- `GET /api/jobs`: Liste aller Druckaufträge (Admin) oder eigener Druckaufträge (Benutzer) +- `POST /api/jobs`: Druckauftrag erstellen +- `GET /api/jobs/`: Details zu einem Druckauftrag +- `POST /api/jobs//abort`: Druckauftrag abbrechen +- `POST /api/jobs//finish`: Druckauftrag vorzeitig beenden +- `POST /api/jobs//extend`: Druckauftrag verlängern +- `PUT /api/jobs//comments`: Kommentare aktualisieren +- `GET /api/job//remaining-time`: Verbleibende Zeit für einen aktiven Druckauftrag + +### Statistiken + +- `GET /api/stats`: Statistiken zu Druckern, Aufträgen und Benutzern (Admin) + +## Datenmodell + +### Benutzer (User) +- id (String UUID, Primary Key) +- username (String, Unique) +- password_hash (String) +- display_name (String) +- email (String, Unique) +- role (String, 'admin', 'user' oder 'guest') + +### Session +- id (String UUID, Primary Key) +- user_id (String UUID, Foreign Key zu User) +- expires_at (DateTime) + +### Drucker (Printer) +- id (String UUID, Primary Key) +- name (String) +- description (Text) +- status (Integer, 0=available, 1=busy, 2=maintenance) +- ip_address (String, IP-Adresse der Tapo-Steckdose) + +### Druckauftrag (PrintJob) +- id (String UUID, Primary Key) +- printer_id (String UUID, Foreign Key zu Printer) +- user_id (String UUID, Foreign Key zu User) +- start_at (DateTime) +- duration_in_minutes (Integer) +- comments (Text) +- aborted (Boolean) +- abort_reason (Text) + +## Steckdosensteuerung + +Die Anwendung steuert Tapo P115 WLAN-Steckdosen, um die Drucker basierend auf Reservierungen ein- und auszuschalten: + +- Bei Erstellung eines Druckauftrags wird die Steckdose des zugehörigen Druckers automatisch eingeschaltet +- Bei Abbruch oder vorzeitiger Beendigung eines Druckauftrags wird die Steckdose ausgeschaltet +- Nach Ablauf der Reservierungszeit wird die Steckdose automatisch ausgeschaltet +- Ein CLI-Befehl `flask check-jobs` überprüft regelmäßig abgelaufene Jobs und schaltet Steckdosen aus + +## Sicherheit + +- Die Anwendung verwendet ein lokales Authentifizierungssystem mit Passwort-Hashing +- Sitzungsdaten werden in Server-Side-Sessions gespeichert +- Zugriffskontrollen sind implementiert, um sicherzustellen, dass Benutzer nur auf ihre eigenen Daten zugreifen können +- Admin-Benutzer haben Zugriff auf alle Daten und können Systemkonfigurationen ändern +- Der erste registrierte Benutzer wird automatisch zum Administrator + +## Logging + +Die Anwendung protokolliert Aktivitäten in rotierenden Logdateien in einem `logs` Verzeichnis. Dies hilft bei der Fehlersuche und Überwachung der Anwendung im Betrieb. + +## Umgebungsvariablen + +Die folgenden Umgebungsvariablen müssen konfiguriert werden: + +- `SECRET_KEY`: Geheimer Schlüssel für die Session-Verschlüsselung +- `DATABASE_PATH`: Pfad zur Datenbank (Standard: SQLite-Datenbank im Instance-Verzeichnis) +- `TAPO_USERNAME`: Benutzername für die Tapo-Steckdosen +- `TAPO_PASSWORD`: Passwort für die Tapo-Steckdosen +- `PRINTERS`: JSON-Objekt mit der Zuordnung von Drucker-Namen zu IP-Adressen der Steckdosen im Format: `{"Printer 1": {"ip": "192.168.1.100"}, "Printer 2": {"ip": "192.168.1.101"}, ...}` + +## Automatisierung + +Die Anwendung beinhaltet einen CLI-Befehl `flask check-jobs`, der regelmäßig ausgeführt werden sollte (z.B. als Cron-Job), um abgelaufene Druckaufträge zu überprüfen und die zugehörigen Steckdosen auszuschalten. + +## Kompatibilität mit dem Frontend + +Das Backend wurde speziell für die Kompatibilität mit dem bestehenden Frontend entwickelt, welches in `/packages/reservation-platform` zu finden ist. Die API-Endpunkte und Datenstrukturen sind so gestaltet, dass sie nahtlos mit dem Frontend zusammenarbeiten. \ No newline at end of file diff --git a/backend/migrations/README b/backend/migrations/README new file mode 100644 index 000000000..0e0484415 --- /dev/null +++ b/backend/migrations/README @@ -0,0 +1 @@ +Single-database configuration for Flask. diff --git a/backend/migrations/alembic.ini b/backend/migrations/alembic.ini new file mode 100644 index 000000000..ec9d45c26 --- /dev/null +++ b/backend/migrations/alembic.ini @@ -0,0 +1,50 @@ +# A generic, single database configuration. + +[alembic] +# template used to generate migration files +# file_template = %%(rev)s_%%(slug)s + +# set to 'true' to run the environment during +# the 'revision' command, regardless of autogenerate +# revision_environment = false + + +# Logging configuration +[loggers] +keys = root,sqlalchemy,alembic,flask_migrate + +[handlers] +keys = console + +[formatters] +keys = generic + +[logger_root] +level = WARN +handlers = console +qualname = + +[logger_sqlalchemy] +level = WARN +handlers = +qualname = sqlalchemy.engine + +[logger_alembic] +level = INFO +handlers = +qualname = alembic + +[logger_flask_migrate] +level = INFO +handlers = +qualname = flask_migrate + +[handler_console] +class = StreamHandler +args = (sys.stderr,) +level = NOTSET +formatter = generic + +[formatter_generic] +format = %(levelname)-5.5s [%(name)s] %(message)s +datefmt = %H:%M:%S diff --git a/backend/migrations/env.py b/backend/migrations/env.py new file mode 100644 index 000000000..4c9709271 --- /dev/null +++ b/backend/migrations/env.py @@ -0,0 +1,113 @@ +import logging +from logging.config import fileConfig + +from flask import current_app + +from alembic import context + +# this is the Alembic Config object, which provides +# access to the values within the .ini file in use. +config = context.config + +# Interpret the config file for Python logging. +# This line sets up loggers basically. +fileConfig(config.config_file_name) +logger = logging.getLogger('alembic.env') + + +def get_engine(): + try: + # this works with Flask-SQLAlchemy<3 and Alchemical + return current_app.extensions['migrate'].db.get_engine() + except (TypeError, AttributeError): + # this works with Flask-SQLAlchemy>=3 + return current_app.extensions['migrate'].db.engine + + +def get_engine_url(): + try: + return get_engine().url.render_as_string(hide_password=False).replace( + '%', '%%') + except AttributeError: + return str(get_engine().url).replace('%', '%%') + + +# add your model's MetaData object here +# for 'autogenerate' support +# from myapp import mymodel +# target_metadata = mymodel.Base.metadata +config.set_main_option('sqlalchemy.url', get_engine_url()) +target_db = current_app.extensions['migrate'].db + +# other values from the config, defined by the needs of env.py, +# can be acquired: +# my_important_option = config.get_main_option("my_important_option") +# ... etc. + + +def get_metadata(): + if hasattr(target_db, 'metadatas'): + return target_db.metadatas[None] + return target_db.metadata + + +def run_migrations_offline(): + """Run migrations in 'offline' mode. + + This configures the context with just a URL + and not an Engine, though an Engine is acceptable + here as well. By skipping the Engine creation + we don't even need a DBAPI to be available. + + Calls to context.execute() here emit the given string to the + script output. + + """ + url = config.get_main_option("sqlalchemy.url") + context.configure( + url=url, target_metadata=get_metadata(), literal_binds=True + ) + + with context.begin_transaction(): + context.run_migrations() + + +def run_migrations_online(): + """Run migrations in 'online' mode. + + In this scenario we need to create an Engine + and associate a connection with the context. + + """ + + # this callback is used to prevent an auto-migration from being generated + # when there are no changes to the schema + # reference: http://alembic.zzzcomputing.com/en/latest/cookbook.html + def process_revision_directives(context, revision, directives): + if getattr(config.cmd_opts, 'autogenerate', False): + script = directives[0] + if script.upgrade_ops.is_empty(): + directives[:] = [] + logger.info('No changes in schema detected.') + + conf_args = current_app.extensions['migrate'].configure_args + if conf_args.get("process_revision_directives") is None: + conf_args["process_revision_directives"] = process_revision_directives + + connectable = get_engine() + + with connectable.connect() as connection: + context.configure( + connection=connection, + target_metadata=get_metadata(), + **conf_args + ) + + with context.begin_transaction(): + context.run_migrations() + + +if context.is_offline_mode(): + run_migrations_offline() +else: + run_migrations_online() diff --git a/backend/migrations/script.py.mako b/backend/migrations/script.py.mako new file mode 100644 index 000000000..2c0156303 --- /dev/null +++ b/backend/migrations/script.py.mako @@ -0,0 +1,24 @@ +"""${message} + +Revision ID: ${up_revision} +Revises: ${down_revision | comma,n} +Create Date: ${create_date} + +""" +from alembic import op +import sqlalchemy as sa +${imports if imports else ""} + +# revision identifiers, used by Alembic. +revision = ${repr(up_revision)} +down_revision = ${repr(down_revision)} +branch_labels = ${repr(branch_labels)} +depends_on = ${repr(depends_on)} + + +def upgrade(): + ${upgrades if upgrades else "pass"} + + +def downgrade(): + ${downgrades if downgrades else "pass"} diff --git a/backend/migrations/versions/add_waiting_approval.py b/backend/migrations/versions/add_waiting_approval.py new file mode 100644 index 000000000..523c8a595 --- /dev/null +++ b/backend/migrations/versions/add_waiting_approval.py @@ -0,0 +1,42 @@ +"""Add waiting_approval column to job table + +Revision ID: add_waiting_approval +Revises: af3faaa3844c +Create Date: 2025-03-12 14:00:00.000000 + +""" +from alembic import op +import sqlalchemy as sa + + +# revision identifiers, used by Alembic. +revision = 'add_waiting_approval' +down_revision = 'af3faaa3844c' +branch_labels = None +depends_on = None + + +def upgrade(): + # Füge die neue Spalte waiting_approval zur job-Tabelle hinzu + with op.batch_alter_table('job', schema=None) as batch_op: + batch_op.add_column(sa.Column('waiting_approval', sa.Integer(), server_default='0', nullable=False)) + + # SQLite-kompatible Migration für die print_job-Tabelle, falls diese existiert + try: + with op.batch_alter_table('print_job', schema=None) as batch_op: + batch_op.add_column(sa.Column('waiting_approval', sa.Boolean(), server_default='0', nullable=False)) + except Exception as e: + print(f"Migration für print_job-Tabelle übersprungen: {e}") + + +def downgrade(): + # Entferne die waiting_approval-Spalte aus der job-Tabelle + with op.batch_alter_table('job', schema=None) as batch_op: + batch_op.drop_column('waiting_approval') + + # SQLite-kompatible Migration für die print_job-Tabelle, falls diese existiert + try: + with op.batch_alter_table('print_job', schema=None) as batch_op: + batch_op.drop_column('waiting_approval') + except Exception as e: + print(f"Downgrade für print_job-Tabelle übersprungen: {e}") \ No newline at end of file diff --git a/backend/migrations/versions/af3faaa3844c_.py b/backend/migrations/versions/af3faaa3844c_.py new file mode 100644 index 000000000..f9f2c6d54 --- /dev/null +++ b/backend/migrations/versions/af3faaa3844c_.py @@ -0,0 +1,81 @@ +"""empty message + +Revision ID: af3faaa3844c +Revises: +Create Date: 2025-03-11 11:16:04.961964 + +""" +from alembic import op +import sqlalchemy as sa + + +# revision identifiers, used by Alembic. +revision = 'af3faaa3844c' +down_revision = None +branch_labels = None +depends_on = None + + +def upgrade(): + # ### commands auto generated by Alembic - please adjust! ### + op.create_table('printer', + sa.Column('id', sa.String(length=36), nullable=False), + sa.Column('name', sa.String(length=64), nullable=False), + sa.Column('description', sa.Text(), nullable=False), + sa.Column('status', sa.Integer(), nullable=True), + sa.Column('ip_address', sa.String(length=15), nullable=True), + sa.PrimaryKeyConstraint('id') + ) + with op.batch_alter_table('printer', schema=None) as batch_op: + batch_op.create_index(batch_op.f('ix_printer_name'), ['name'], unique=False) + + op.create_table('user', + sa.Column('id', sa.String(length=36), nullable=False), + sa.Column('username', sa.String(length=64), nullable=True), + sa.Column('password_hash', sa.String(length=128), nullable=True), + sa.Column('display_name', sa.String(length=100), nullable=True), + sa.Column('email', sa.String(length=120), nullable=True), + sa.Column('role', sa.String(length=20), nullable=True), + sa.PrimaryKeyConstraint('id') + ) + with op.batch_alter_table('user', schema=None) as batch_op: + batch_op.create_index(batch_op.f('ix_user_email'), ['email'], unique=True) + batch_op.create_index(batch_op.f('ix_user_username'), ['username'], unique=True) + + op.create_table('print_job', + sa.Column('id', sa.String(length=36), nullable=False), + sa.Column('printer_id', sa.String(length=36), nullable=False), + sa.Column('user_id', sa.String(length=36), nullable=False), + sa.Column('start_at', sa.DateTime(), nullable=True), + sa.Column('duration_in_minutes', sa.Integer(), nullable=False), + sa.Column('comments', sa.Text(), nullable=True), + sa.Column('aborted', sa.Boolean(), nullable=True), + sa.Column('abort_reason', sa.Text(), nullable=True), + sa.ForeignKeyConstraint(['printer_id'], ['printer.id'], ), + sa.ForeignKeyConstraint(['user_id'], ['user.id'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_table('session', + sa.Column('id', sa.String(length=36), nullable=False), + sa.Column('user_id', sa.String(length=36), nullable=False), + sa.Column('expires_at', sa.DateTime(), nullable=False), + sa.ForeignKeyConstraint(['user_id'], ['user.id'], ), + sa.PrimaryKeyConstraint('id') + ) + # ### end Alembic commands ### + + +def downgrade(): + # ### commands auto generated by Alembic - please adjust! ### + op.drop_table('session') + op.drop_table('print_job') + with op.batch_alter_table('user', schema=None) as batch_op: + batch_op.drop_index(batch_op.f('ix_user_username')) + batch_op.drop_index(batch_op.f('ix_user_email')) + + op.drop_table('user') + with op.batch_alter_table('printer', schema=None) as batch_op: + batch_op.drop_index(batch_op.f('ix_printer_name')) + + op.drop_table('printer') + # ### end Alembic commands ### diff --git a/backend/requirements.txt b/backend/requirements.txt new file mode 100644 index 000000000..79a3a167e --- /dev/null +++ b/backend/requirements.txt @@ -0,0 +1,7 @@ +flask==2.3.3 +flask-cors==4.0.0 +pyjwt==2.8.0 +python-dotenv==1.0.0 +werkzeug==2.3.7 +gunicorn==21.2.0 +PyP100==0.0.19 \ No newline at end of file diff --git a/backend/static/fontawesome/webfonts/fa-brands-400.ttf b/backend/static/fontawesome/webfonts/fa-brands-400.ttf new file mode 100644 index 000000000..0f82a8360 Binary files /dev/null and b/backend/static/fontawesome/webfonts/fa-brands-400.ttf differ diff --git a/backend/static/fontawesome/webfonts/fa-brands-400.woff2 b/backend/static/fontawesome/webfonts/fa-brands-400.woff2 new file mode 100644 index 000000000..3c5cf97ec Binary files /dev/null and b/backend/static/fontawesome/webfonts/fa-brands-400.woff2 differ diff --git a/backend/static/fontawesome/webfonts/fa-regular-400.woff2 b/backend/static/fontawesome/webfonts/fa-regular-400.woff2 new file mode 100644 index 000000000..57d917965 Binary files /dev/null and b/backend/static/fontawesome/webfonts/fa-regular-400.woff2 differ diff --git a/backend/static/fontawesome/webfonts/fa-solid-900.ttf b/backend/static/fontawesome/webfonts/fa-solid-900.ttf new file mode 100644 index 000000000..1c10972ec Binary files /dev/null and b/backend/static/fontawesome/webfonts/fa-solid-900.ttf differ diff --git a/backend/static/fontawesome/webfonts/fa-solid-900.woff2 b/backend/static/fontawesome/webfonts/fa-solid-900.woff2 new file mode 100644 index 000000000..16721020f Binary files /dev/null and b/backend/static/fontawesome/webfonts/fa-solid-900.woff2 differ diff --git a/backend/templates/base.html b/backend/templates/base.html new file mode 100644 index 000000000..23588dd96 --- /dev/null +++ b/backend/templates/base.html @@ -0,0 +1,169 @@ + + + + + + {% block title %}MYP API Tester{% endblock %} + + + + + + +
+ {% with messages = get_flashed_messages(with_categories=true) %} + {% if messages %} + {% for category, message in messages %} + + {% endfor %} + {% endif %} + {% endwith %} + + {% block content %}{% endblock %} +
+ + + + {% block scripts %}{% endblock %} + + \ No newline at end of file diff --git a/backend/templates/dashboard.html b/backend/templates/dashboard.html new file mode 100644 index 000000000..6f5f3af0d --- /dev/null +++ b/backend/templates/dashboard.html @@ -0,0 +1,304 @@ +{% extends "base.html" %} + +{% block title %}Dashboard - MYP API Tester{% endblock %} + +{% block content %} +
+
+
+
+

Willkommen, {{ current_user.display_name }}

+
+
+

Benutzerdetails:

+
    +
  • ID: {{ current_user.id }}
  • +
  • Benutzername: {{ current_user.username }}
  • +
  • E-Mail: {{ current_user.email or "Nicht angegeben" }}
  • +
  • Rolle: {{ current_user.role }}
  • +
+
+ Drucker verwalten + Druckaufträge verwalten + {% if current_user.role == 'admin' %} + Benutzer verwalten + Statistiken + {% endif %} +
+
+
+
+
+ +
+
+
+
+
Aktive Druckaufträge
+
+
+
+ +
+ +
+
Lade Druckaufträge...
+
+ +
+
API-Antwort:
+

+                
+
+
+
+ +
+
+
+
Verfügbare Drucker
+
+
+
+ +
+ +
+
Lade Drucker...
+
+ +
+
API-Antwort:
+

+                
+
+
+
+
+ + + +{% endblock %} + +{% block scripts %} + +{% endblock %} \ No newline at end of file diff --git a/backend/templates/jobs.html b/backend/templates/jobs.html new file mode 100644 index 000000000..326e6611a --- /dev/null +++ b/backend/templates/jobs.html @@ -0,0 +1,443 @@ +{% extends "base.html" %} + +{% block title %}Druckaufträge - MYP API Tester{% endblock %} + +{% block content %} +
+
+
+
+

Druckaufträge verwalten

+ +
+
+
+
+
+ + +
+
+ + +
+
+ + +
+
+ + +
+ +
+
+
Antwort:
+

+                    
+
+
+
+
+ +
+ +
+ + + + + + + + + + + + + + + + + +
IDDruckerBenutzerStartDauer (Min)Verbleibend (Min)StatusKommentareAktionen
+
+ +
+
API-Antwort:
+

+                
+
+
+
+
+ + + + + + + + + + + + + + + +{% endblock %} + +{% block scripts %} + +{% endblock %} \ No newline at end of file diff --git a/backend/templates/login.html b/backend/templates/login.html new file mode 100644 index 000000000..c8a5cd3ab --- /dev/null +++ b/backend/templates/login.html @@ -0,0 +1,37 @@ +{% extends "base.html" %} + +{% block title %}Anmelden - MYP API Tester{% endblock %} + +{% block content %} +
+
+
+
+

Anmelden

+
+
+
+
+ + +
+
+ + +
+ +
+ +
+

Noch kein Konto? Registrieren

+
+ +
+
Antwort:
+

+                
+
+
+
+
+{% endblock %} \ No newline at end of file diff --git a/backend/templates/printers.html b/backend/templates/printers.html new file mode 100644 index 000000000..1657f77f8 --- /dev/null +++ b/backend/templates/printers.html @@ -0,0 +1,280 @@ +{% extends "base.html" %} + +{% block title %}Drucker - MYP API Tester{% endblock %} + +{% block content %} +
+
+
+
+

Drucker verwalten

+ +
+
+
+
+
+ + +
+
+ + +
+
+ + +
+
+ + +
+ +
+
+
Antwort:
+

+                    
+
+
+
+
+ +
+ +
+ + + + + + + + + + + + + + + + +
IDNameBeschreibungStatusIP-AdresseAktueller JobWartende JobsAktionen
+
+ +
+
API-Antwort:
+

+                
+
+
+
+
+ + + + + + +{% endblock %} + +{% block scripts %} + +{% endblock %} \ No newline at end of file diff --git a/backend/templates/register.html b/backend/templates/register.html new file mode 100644 index 000000000..1c5c16869 --- /dev/null +++ b/backend/templates/register.html @@ -0,0 +1,45 @@ +{% extends "base.html" %} + +{% block title %}Registrieren - MYP API Tester{% endblock %} + +{% block content %} +
+
+
+
+

Registrieren

+
+
+
+
+ + +
+
+ + +
+
+ + +
+
+ + +
+ +
+ +
+

Bereits registriert? Anmelden

+
+ +
+
Antwort:
+

+                
+
+
+
+
+{% endblock %} \ No newline at end of file diff --git a/backend/templates/stats.html b/backend/templates/stats.html new file mode 100644 index 000000000..c711a715d --- /dev/null +++ b/backend/templates/stats.html @@ -0,0 +1,395 @@ +{% extends "base.html" %} + +{% block title %}Statistiken - MYP API Tester{% endblock %} + +{% block content %} +
+
+
+
+

Systemstatistiken

+
+
+
+ +
+ +
+ +
+ + +
+
+
+
+
Drucker mit Verbindungsproblemen
+
+
+
Keine Verbindungsprobleme festgestellt.
+
+
+
+
+ + +
+
+
+
+
Steckdosen-Verfügbarkeit
+
+
+
+ +
+ +
+
+
+
+ + +
+
+
Stats API-Antwort:
+

+                    
+
+
Uptime API-Antwort:
+

+                    
+
+
+
+
+
+{% endblock %} + +{% block scripts %} + + + +{% endblock %} \ No newline at end of file diff --git a/backend/templates/users.html b/backend/templates/users.html new file mode 100644 index 000000000..7c68e4f8f --- /dev/null +++ b/backend/templates/users.html @@ -0,0 +1,238 @@ +{% extends "base.html" %} + +{% block title %}Benutzer - MYP API Tester{% endblock %} + +{% block content %} +
+
+
+
+

Benutzer verwalten

+ +
+
+
+
+
+ + +
+
+ + +
+
+ + +
+
+ + +
+ +
+
+
Antwort:
+

+                    
+
+
+
+
+ +
+ +
+ + + + + + + + + + + + + + +
IDBenutzernameAnzeigenameE-MailRolleAktionen
+
+ +
+
API-Antwort:
+

+                
+
+
+
+
+ + + + + + +{% endblock %} + +{% block scripts %} + +{% endblock %} \ No newline at end of file diff --git a/docs/.gitkeep b/docs/.gitkeep new file mode 100644 index 000000000..8b1378917 --- /dev/null +++ b/docs/.gitkeep @@ -0,0 +1 @@ + diff --git a/docs/Aktueller Stand.md b/docs/Aktueller Stand.md new file mode 100644 index 000000000..2fe85be5c --- /dev/null +++ b/docs/Aktueller Stand.md @@ -0,0 +1,53 @@ +- HTTP Broadcast funktioniert nicht -> können IP nicht finden +- Backend im Unternehmen, Frontend auf GitHub funktioniert nicht -> Frontend hat HTTPS und Backend nur HTTP, self-signed certificate würde Fehler werfen + +--- + +[Knowledge Base - [PG Netzwerk] DHCP/DNS-Services: Anfragen und Änderungen von IP-Adressen/DNS (mercedes-benz.com)](https://servicenow.i.mercedes-benz.com/esc?id=kb_article&table=kb_knowledge&sysparm_article=KB0426678) + +Statische IP beantragen +Alias Zuordnung (druckerapp.xyz.corpintra.net) + +!!! SSL Zertifikat??? +-> [Server-Zertifikat - Employee Center (mercedes-benz.com)](https://servicenow.i.mercedes-benz.com/esc?id=sc_cat_item&table=sc_cat_item&sys_id=7ef47a1c1b0d5450c4f43113dd4bcbf5) +8€ (wahrscheinlich einmalig) +benötigt Anwenderservice +[Application Service (Erstellen) - Employee Center (mercedes-benz.com)](https://servicenow.i.mercedes-benz.com/esc?id=sc_cat_item&table=sc_cat_item&sys_id=ab095def1bf68810c4f43113dd4bcb06) + + +-> [ServiceNow-Berechtigungen - Employee Center (mercedes-benz.com)](https://servicenow.i.mercedes-benz.com/esc?id=sc_cat_item&table=sc_cat_item&sys_id=8d645a30db3dc4501754ccd40596192c) + +Sollten Sie für Ihre Applikation / Ihr Endgerät eine statisch zugewiesene IP Adresse (S-DHCP), einen manuellen IP-Pool (M-DHCP) oder eine entsprechende neue oder angepasste DNS-Zuordnung benötigen, nutzen Sie bitte die nachfolgenden SNOW-Templates (Hinweis zur Nutzung der Ticket-Templates benötigen Sie die **Service Now - Rechte eines "Agent Fulfilers"**. Die Beantragung für diese Rechte ist über das SNOW Employee Service Portal im Social Intranet möglich: [ServiceNow Permissions - Employee Service Center (mercedes-benz.com))](https://servicenow.i.mercedes-benz.com/esc?id=sc_cat_item&table=sc_cat_item&sys_id=8d645a30db3dc4501754ccd40596192c). +-> Martin diesbezüglich Fragen, ob er Sie für sich bestellt? + +--- + +- Martin Rechte SNOW +- Martin Termin Volker Otto + +[Examples · Diagrams (mingrammer.com)](https://diagrams.mingrammer.com/docs/getting-started/examples) + +Hier ist eine Antwort, die ich mit Microsoft Copilot erhalten habe, der weltweit ersten KI-gestützten Antwort-Engine. Wählen Sie diese Option aus, um die vollständige Antwort anzuzeigen, oder probieren Sie sie selbst aus. https://sl.bing.net/iv24ZLcukjk + +- use devcontainer !!! + +https://x.com/tremorlabs/status/1779873516533547064?s=46 + +use mix of shadcn ui and tremor.so blocks + + +Bestellliste: +- 1x Switch +- 1x wlan access point +- 2x Raspberry Pi (4B / 5) +(- Xx LAN-Kabel) +- Adapter für 3D-Drucker oder Schaltbare Steckdosen + +-> Datensicherung?!?!?! +make it a user thing +download db copy +import/export +OR Zugang zu CORP O über Anwendungsservice?? + + +Move TRILLUM to GITHUB WIKI diff --git a/docs/Dokumentation_IHK.md b/docs/Dokumentation_IHK.md new file mode 100644 index 000000000..208752646 --- /dev/null +++ b/docs/Dokumentation_IHK.md @@ -0,0 +1,24 @@ + +Notizen: +- Wollten zuerst OpenSUSE, haben uns dagegen entschieden, weil NixOS einfacher zu konfigurieren ist und besser geeignet für diesen Einsatzzweck +- Mussten eine IP-Adresse und eine Domain organisieren von unserer IT-Abteilung +- haben ein Netzwerkplan gemacht +- haben uns die akutellen Prozesse und Konventionen bei der Organisation einer internen Domain angeguckt +- haben uns für Raspberrys "entschieden", stand aber mehr oder weniger schon fest weil diese einfach perfekt für den Einsatzzweck sind +- Da Till digitale Vernetzung hat macht er Backend, weil die Schnittstelle der Vernetzung zum cyberphysischen System dort lag +- für die Dokumentation: Daten (Datums) müssen stimmen! + +python schnittstelle funktionierte nicht +nach etlichem rumprobieren festgestellt: geht nicht so einfach +wireshark mitschnitt gemacht → auffällig: immer die selben responses bei verschlüsselter verbindung +ohne erfolg beim simulieren einzelner anfragen +dann: geistesblitz: anfragensequenz muss es sein! +hat funktioniert → es hat klick gemacht!! . +verbindung verschlüsselt und mit temporärem cookie +→ proprietäre Verschlüsselung +wie wird die verbindung ausgehandelt? + +------ + +11.09 : Teile bestellt im internen Technikshop +12.09 : DNS Alias festlegen / beantragen diff --git a/docs/Infrastruktur.tldr b/docs/Infrastruktur.tldr new file mode 100644 index 000000000..58ed62a02 --- /dev/null +++ b/docs/Infrastruktur.tldr @@ -0,0 +1 @@ +{"tldrawFileFormatVersion":1,"schema":{"schemaVersion":1,"storeVersion":4,"recordVersions":{"asset":{"version":1,"subTypeKey":"type","subTypeVersions":{"image":3,"video":3,"bookmark":1}},"camera":{"version":1},"document":{"version":2},"instance":{"version":24},"instance_page_state":{"version":5},"page":{"version":1},"shape":{"version":4,"subTypeKey":"type","subTypeVersions":{"group":0,"text":1,"bookmark":2,"draw":1,"geo":8,"note":5,"line":4,"frame":0,"arrow":3,"highlight":0,"embed":4,"image":3,"video":2}},"instance_presence":{"version":5},"pointer":{"version":1}}},"records":[{"gridSize":10,"name":"","meta":{},"id":"document:document","typeName":"document"},{"meta":{},"id":"page:page","name":"Page 1","index":"a1","typeName":"page"},{"followingUserId":null,"opacityForNextShape":1,"stylesForNextShape":{"tldraw:geo":"rectangle","tldraw:color":"black","tldraw:size":"m","tldraw:fill":"pattern","tldraw:arrowheadEnd":"none","tldraw:dash":"solid","tldraw:horizontalAlign":"start"},"brush":null,"scribbles":[],"cursor":{"type":"default","rotation":0},"isFocusMode":false,"exportBackground":true,"isDebugMode":false,"isToolLocked":false,"screenBounds":{"x":0,"y":0,"w":1912,"h":976},"insets":[false,false,false,false],"zoomBrush":null,"isGridMode":false,"isPenMode":false,"chatMessage":"","isChatting":false,"highlightedUserIds":[],"canMoveCamera":true,"isFocused":true,"devicePixelRatio":1,"isCoarsePointer":false,"isHoveringCanvas":false,"openMenus":["main menu","main-menu-sub.file"],"isChangingStyle":false,"isReadonly":false,"meta":{},"duplicateProps":null,"id":"instance:instance","currentPageId":"page:page","typeName":"instance"},{"x":2704.1858512271715,"y":1128.2874355067452,"z":0.2875934622186635,"meta":{},"id":"camera:page:page","typeName":"camera"},{"editingShapeId":null,"croppingShapeId":null,"selectedShapeIds":[],"hoveredShapeId":null,"erasingShapeIds":[],"hintingShapeIds":[],"focusedGroupId":null,"meta":{},"id":"instance_page_state:page:page","pageId":"page:page","typeName":"instance_page_state"},{"id":"pointer:pointer","typeName":"pointer","x":-2603.3490666345137,"y":-982.2479543725512,"lastActivityTimestamp":1713247133460,"meta":{}},{"x":-28.720245737112407,"y":-35.26654302941165,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:bCslUaG8j4a7h9UZNM9m2","type":"geo","props":{"w":220.43422559716703,"h":220.43422559716703,"geo":"rectangle","color":"black","labelColor":"black","fill":"none","dash":"draw","size":"m","font":"draw","text":"","align":"middle","verticalAlign":"middle","growY":0,"url":""},"parentId":"shape:Xo9xDtOawOXKSXdAPypLP","index":"a1","typeName":"shape"},{"x":-31.458276574997797,"y":189.65514255405427,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:fEzDJcv5UmYXI5qFifyuu","type":"text","props":{"color":"black","size":"m","w":179.859375,"text":"Reservation Pi","font":"draw","align":"middle","autoSize":true,"scale":0.7800803420404604},"parentId":"shape:Xo9xDtOawOXKSXdAPypLP","index":"a2","typeName":"shape"},{"x":31.458276574997797,"y":35.26654302941165,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:Xo9xDtOawOXKSXdAPypLP","type":"group","parentId":"shape:TWiEjzVllgCVTqRD8L4r1","index":"a1","props":{},"typeName":"shape"},{"x":51.31916599376575,"y":10.06346443994454,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:DIVmNP4GHPavX1csuaL8Y","type":"image","props":{"w":134.3792274104758,"h":80.45956241202238,"assetId":"asset:-1084722902","playing":true,"url":"","crop":null},"parentId":"shape:TWiEjzVllgCVTqRD8L4r1","index":"a2","typeName":"shape"},{"x":35.37513705494166,"y":129.05612439679783,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:ha5jFTEcpXfCrXrwI6K1a","type":"image","props":{"w":156.2300822193515,"h":74.07909731900918,"assetId":"asset:809752036","playing":true,"url":"","crop":null},"parentId":"shape:TWiEjzVllgCVTqRD8L4r1","index":"a3","typeName":"shape"},{"x":-98.80133633652491,"y":294.0758708757765,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:TWiEjzVllgCVTqRD8L4r1","type":"group","parentId":"shape:ws367qTacKGtRALROEq_7","index":"a1","props":{},"typeName":"shape"},{"x":112.57631629328608,"y":71.67239421888075,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:ZGF-b7KUnndVaCUOXb9wi","type":"arrow","parentId":"shape:TWiEjzVllgCVTqRD8L4r1","index":"a4","props":{"dash":"draw","size":"m","fill":"none","color":"black","labelColor":"black","bend":0,"start":{"type":"binding","boundShapeId":"shape:DIVmNP4GHPavX1csuaL8Y","normalizedAnchor":{"x":0.45585282398152055,"y":0.7657129610455162},"isPrecise":false,"isExact":false},"end":{"type":"binding","boundShapeId":"shape:ha5jFTEcpXfCrXrwI6K1a","normalizedAnchor":{"x":0.526113352183538,"y":0.19316412532300226},"isPrecise":true,"isExact":false},"arrowheadStart":"none","arrowheadEnd":"arrow","text":"","labelPosition":0.5,"font":"draw"},"typeName":"shape"},{"x":-49.78199466282621,"y":387.30287307218475,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:M1DcIJ5c_8r0plPUVOrkU","type":"text","props":{"color":"black","size":"s","w":170.265625,"text":"speichert Daten in","font":"draw","align":"middle","autoSize":true,"scale":0.7800803420404604},"parentId":"shape:ws367qTacKGtRALROEq_7","index":"a2","typeName":"shape"},{"x":-262.36918685748145,"y":-749.7434502182293,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:ws367qTacKGtRALROEq_7","type":"group","parentId":"page:page","index":"a6","props":{},"typeName":"shape"},{"x":-582.5753183088976,"y":-489.924720919305,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:G-BnFiLbUeSt7-dKGbTMW","type":"geo","props":{"w":2460.5970879378106,"h":1567.8497415361312,"geo":"rectangle","color":"black","labelColor":"black","fill":"none","dash":"draw","size":"l","font":"draw","text":"","align":"middle","verticalAlign":"middle","growY":0,"url":""},"parentId":"page:page","index":"a7","typeName":"shape"},{"x":-580.5600872786465,"y":-177.56391123023076,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:ihndiQ47_feGlWuj54X0u","type":"geo","props":{"w":356.69589235462104,"h":1162.7883044554599,"geo":"rectangle","color":"black","labelColor":"black","fill":"pattern","dash":"draw","size":"l","font":"draw","text":"(Drucker)","align":"middle","verticalAlign":"middle","growY":0,"url":""},"parentId":"page:page","index":"a8","typeName":"shape"},{"x":19.97875973647865,"y":-423.422096920986,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:Nsf14yAXFvu_ko0ipZIdp","type":"geo","props":{"w":705.3308605882339,"h":187.41648581344498,"geo":"rectangle","color":"black","labelColor":"black","fill":"pattern","dash":"draw","size":"l","font":"draw","text":"Arbeitsplatz 1","align":"middle","verticalAlign":"middle","growY":0,"url":""},"parentId":"page:page","index":"a9","typeName":"shape"},{"x":924.81749231967,"y":-423.422096920986,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:klBsSdviP9o0v09ntX3Sq","type":"geo","props":{"w":705.3308605882339,"h":187.41648581344498,"geo":"rectangle","color":"black","labelColor":"black","fill":"pattern","dash":"draw","size":"l","font":"draw","text":"Arbeitsplatz 2","align":"middle","verticalAlign":"middle","growY":0,"url":""},"parentId":"page:page","index":"aA","typeName":"shape"},{"x":26.0244528272342,"y":-506.0465691613222,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:8gQ0fIb31PgupkyQb5mZC","type":"geo","props":{"w":78.59401017983191,"h":36.274158544537784,"geo":"rectangle","color":"black","labelColor":"black","fill":"pattern","dash":"draw","size":"l","font":"draw","text":"","align":"middle","verticalAlign":"middle","growY":0,"url":""},"parentId":"page:page","index":"aB","typeName":"shape"},{"x":-17.944364067723654,"y":-568.6328337361119,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:rfK_uioFcbY_V-YoxPZe8","type":"text","props":{"color":"black","size":"l","w":168.546875,"text":"LAN-Port","font":"draw","align":"middle","autoSize":true,"scale":1},"parentId":"page:page","index":"aC","typeName":"shape"},{"x":-115.53732759275044,"y":-427.80968855962954,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:Tsvmdvj4PVVHi6QcYEHa6","type":"arrow","parentId":"page:page","index":"aBV","props":{"dash":"draw","size":"l","fill":"pattern","color":"black","labelColor":"black","bend":0,"start":{"type":"binding","boundShapeId":"shape:bCslUaG8j4a7h9UZNM9m2","normalizedAnchor":{"x":0.9643521609419151,"y":0.13401855552588923},"isPrecise":false,"isExact":false},"end":{"type":"binding","boundShapeId":"shape:8gQ0fIb31PgupkyQb5mZC","normalizedAnchor":{"x":0.3663003680526707,"y":0.6553144449157916},"isPrecise":true,"isExact":false},"arrowheadStart":"none","arrowheadEnd":"none","text":"","labelPosition":0.5,"font":"draw"},"typeName":"shape"},{"x":6.2836674683221645,"y":0,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:aS6vClkZogBZmo2r0GHzI","type":"geo","props":{"w":261.0797866274253,"h":261.0797866274253,"geo":"rectangle","color":"black","labelColor":"black","fill":"none","dash":"draw","size":"l","font":"draw","text":"","align":"middle","verticalAlign":"middle","growY":0,"url":""},"parentId":"shape:HhD4sxU9jNE1uEGplvYIq","index":"a1","typeName":"shape"},{"x":0,"y":267.10185459060995,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:vAvfAwgZ9iAQmJMQXjmU5","type":"text","props":{"color":"black","size":"l","w":234.90625,"text":"Controller Pi","font":"draw","align":"middle","autoSize":true,"scale":1},"parentId":"shape:HhD4sxU9jNE1uEGplvYIq","index":"a2","typeName":"shape"},{"x":0,"y":0,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:HhD4sxU9jNE1uEGplvYIq","type":"group","parentId":"shape:2w2fWR9yInbg5bFdyOm-f","index":"a1","props":{},"typeName":"shape"},{"x":211.86010618759735,"y":-223.70037364693843,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:WK3g_IbJV4V5wuzyZ-289","type":"text","props":{"color":"black","size":"l","w":263.296875,"text":"192.168.X.X/24","font":"draw","align":"middle","autoSize":true,"scale":1},"parentId":"page:page","index":"aF","typeName":"shape"},{"x":1136.3573493851131,"y":-220.45082270424064,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:FCpHTIRw40mltKy6Jie4p","type":"text","props":{"color":"black","size":"l","w":263.296875,"text":"192.168.X.X/24","font":"draw","align":"middle","autoSize":true,"scale":1},"parentId":"page:page","index":"aG","typeName":"shape"},{"x":-364.93518614125696,"y":-499.91220377624893,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:zFgZ-hoGkhCHFv8y-TPqp","type":"text","props":{"color":"black","size":"l","w":263.296875,"text":"192.168.X.X/24","font":"draw","align":"middle","autoSize":true,"scale":1},"parentId":"page:page","index":"aH","typeName":"shape"},{"x":478.42080447309456,"y":-565.4273337161576,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:z4sEG1E5VA88OVB9lYMMM","type":"text","props":{"color":"black","size":"xl","w":321.59375,"text":"192.168.X.X/24","font":"draw","align":"middle","autoSize":true,"scale":1},"parentId":"page:page","index":"aI","typeName":"shape"},{"x":36.52080283073042,"y":40.441753340071614,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:jmGEkrZD99EdA_jRkCQcH","type":"image","props":{"w":187.46337448302393,"h":186.7310956764496,"assetId":"asset:670164008","playing":true,"url":"","crop":null},"parentId":"shape:2w2fWR9yInbg5bFdyOm-f","index":"a2","typeName":"shape"},{"x":-1179.3998115610402,"y":263.9918538585895,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:2w2fWR9yInbg5bFdyOm-f","type":"group","parentId":"page:page","index":"aJ","props":{},"typeName":"shape"},{"x":-1316.9078787313986,"y":-667.6505882592148,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:u9F5OPoRDpK3tG90jQiA-","type":"geo","props":{"w":1257.4034743674365,"h":1819.7750685408998,"geo":"rectangle","color":"black","labelColor":"black","fill":"none","dash":"dotted","size":"xl","font":"draw","text":"","align":"middle","verticalAlign":"middle","growY":0,"url":""},"parentId":"page:page","index":"aK","typeName":"shape"},{"x":-1314.8068504313453,"y":-801.651316485265,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:-vmQTQy_vOvrWB9Q_Sgsv","type":"text","props":{"color":"black","size":"xl","w":717.328125,"text":"Pi-Netzwerk (AdHoc Netzwerk)\n10.0.0.X/24","font":"draw","align":"start","autoSize":true,"scale":1},"parentId":"page:page","index":"aL","typeName":"shape"},{"x":-734.6603262151287,"y":302.084411089334,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:yLt-hqzNkrGUz7g-m6X5c","type":"geo","props":{"w":42.194747461994666,"h":156.1205656093798,"geo":"rectangle","color":"black","labelColor":"black","fill":"pattern","dash":"draw","size":"m","font":"draw","text":"","align":"start","verticalAlign":"middle","growY":0,"url":""},"parentId":"page:page","index":"aM","typeName":"shape"},{"x":-756.9936993943913,"y":464.14978545135045,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:jmaRpJB1v03bLjyQqLOvx","type":"text","props":{"color":"black","size":"m","w":82.34375,"text":"Switch","font":"draw","align":"start","autoSize":true,"scale":1},"parentId":"page:page","index":"aN","typeName":"shape"},{"x":-903.9031454447263,"y":375.89857750832164,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:cxeIphcPwRtos8NqOE_e9","type":"arrow","parentId":"page:page","index":"aMV","props":{"dash":"draw","size":"m","fill":"pattern","color":"black","labelColor":"black","bend":0,"start":{"type":"binding","boundShapeId":"shape:aS6vClkZogBZmo2r0GHzI","normalizedAnchor":{"x":0.9847699302394234,"y":0.4460237172052744},"isPrecise":false,"isExact":false},"end":{"type":"binding","boundShapeId":"shape:yLt-hqzNkrGUz7g-m6X5c","normalizedAnchor":{"x":0.16093572303884734,"y":0.5479722045573353},"isPrecise":true,"isExact":false},"arrowheadStart":"none","arrowheadEnd":"none","text":"","labelPosition":0.5,"font":"draw"},"typeName":"shape"},{"x":-693.8365481491963,"y":372.377908279905,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:hTdZXK1FfB9zej_rX-lMy","type":"arrow","parentId":"page:page","index":"aMG","props":{"dash":"draw","size":"m","fill":"pattern","color":"black","labelColor":"black","bend":0,"start":{"type":"binding","boundShapeId":"shape:yLt-hqzNkrGUz7g-m6X5c","normalizedAnchor":{"x":0.9675085294135931,"y":0.45025136128732873},"isPrecise":false,"isExact":false},"end":{"type":"binding","boundShapeId":"shape:ihndiQ47_feGlWuj54X0u","normalizedAnchor":{"x":0.11342835630439173,"y":0.18733019139842674},"isPrecise":true,"isExact":false},"arrowheadStart":"none","arrowheadEnd":"none","text":"","labelPosition":0.5,"font":"draw"},"typeName":"shape"},{"x":-699.7043301965575,"y":371.2043518704328,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:E1Ch1lftgW4sQLimsiuq0","type":"arrow","parentId":"page:page","index":"aM8","props":{"dash":"draw","size":"m","fill":"pattern","color":"black","labelColor":"black","bend":0,"start":{"type":"binding","boundShapeId":"shape:yLt-hqzNkrGUz7g-m6X5c","normalizedAnchor":{"x":0.8284442524524297,"y":0.44273437334348215},"isPrecise":false,"isExact":false},"end":{"type":"binding","boundShapeId":"shape:ihndiQ47_feGlWuj54X0u","normalizedAnchor":{"x":0.24503137552901513,"y":0.45983055052948174},"isPrecise":false,"isExact":false},"arrowheadStart":"none","arrowheadEnd":"none","text":"","labelPosition":0.5,"font":"draw"},"typeName":"shape"},{"x":-699.7043301965575,"y":415.79949543037765,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:xJuUtx7CD1NeIjWBroGS-","type":"arrow","parentId":"page:page","index":"aM4","props":{"dash":"draw","size":"m","fill":"pattern","color":"black","labelColor":"black","bend":0,"start":{"type":"binding","boundShapeId":"shape:yLt-hqzNkrGUz7g-m6X5c","normalizedAnchor":{"x":0.8284442524524297,"y":0.7283799152096562},"isPrecise":false,"isExact":false},"end":{"type":"binding","boundShapeId":"shape:ihndiQ47_feGlWuj54X0u","normalizedAnchor":{"x":0.0969779789013138,"y":0.7343494308392852},"isPrecise":true,"isExact":false},"arrowheadStart":"none","arrowheadEnd":"none","text":"","labelPosition":0.5,"font":"draw"},"typeName":"shape"},{"x":0,"y":-0.5154255211780878,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:gPSCaZ64vZzwpMjhevnpr","type":"geo","props":{"w":188.17658873949267,"h":188.17658873949267,"geo":"ellipse","color":"black","labelColor":"black","fill":"solid","dash":"solid","size":"m","font":"draw","text":"","align":"start","verticalAlign":"middle","growY":0,"url":""},"parentId":"shape:WBNyIQc5bEj3Q17u6WyZP","index":"a1","typeName":"shape"},{"x":35.70934414545172,"y":44.73141315720261,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:UadvsAFxGTQRLv_5b35Mm","type":"geo","props":{"w":29.070139614653044,"h":29.070139614653044,"geo":"ellipse","color":"black","labelColor":"black","fill":"solid","dash":"solid","size":"m","font":"draw","text":"","align":"start","verticalAlign":"middle","growY":0,"url":""},"parentId":"shape:WBNyIQc5bEj3Q17u6WyZP","index":"a2","typeName":"shape"},{"x":108.38469318208445,"y":45.8494954500739,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:Oq3vmzFm03AIGjZH95Usq","type":"geo","props":{"w":29.070139614653044,"h":29.070139614653044,"geo":"ellipse","color":"black","labelColor":"black","fill":"solid","dash":"solid","size":"m","font":"draw","text":"","align":"start","verticalAlign":"middle","growY":0,"url":""},"parentId":"shape:WBNyIQc5bEj3Q17u6WyZP","index":"a3","typeName":"shape"},{"x":35.70934414545172,"y":111.81635072947887,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:Ki1iip6LgRYPOOHMEOkqG","type":"draw","props":{"segments":[{"type":"free","points":[{"x":0,"y":0,"z":0.5},{"x":0,"y":1.12,"z":0.5},{"x":1.12,"y":1.12,"z":0.5},{"x":2.24,"y":1.12,"z":0.5},{"x":2.24,"y":2.24,"z":0.5},{"x":3.35,"y":2.24,"z":0.5},{"x":4.47,"y":3.35,"z":0.5},{"x":5.59,"y":4.47,"z":0.5},{"x":6.71,"y":5.59,"z":0.5},{"x":7.83,"y":6.71,"z":0.5},{"x":8.94,"y":6.71,"z":0.5},{"x":10.06,"y":7.83,"z":0.5},{"x":11.18,"y":10.06,"z":0.5},{"x":12.3,"y":10.06,"z":0.5},{"x":12.3,"y":11.18,"z":0.5},{"x":13.42,"y":12.3,"z":0.5},{"x":14.54,"y":12.3,"z":0.5},{"x":15.65,"y":14.54,"z":0.5},{"x":16.77,"y":14.54,"z":0.5},{"x":17.89,"y":15.65,"z":0.5},{"x":19.01,"y":15.65,"z":0.5},{"x":20.13,"y":16.77,"z":0.5},{"x":21.24,"y":16.77,"z":0.5},{"x":21.24,"y":17.89,"z":0.5},{"x":22.36,"y":19.01,"z":0.5},{"x":23.48,"y":19.01,"z":0.5},{"x":24.6,"y":20.13,"z":0.5},{"x":25.72,"y":21.24,"z":0.5},{"x":26.83,"y":21.24,"z":0.5},{"x":27.95,"y":23.48,"z":0.5},{"x":30.19,"y":23.48,"z":0.5},{"x":31.31,"y":24.6,"z":0.5},{"x":33.54,"y":25.72,"z":0.5},{"x":34.66,"y":26.83,"z":0.5},{"x":34.66,"y":27.95,"z":0.5},{"x":35.78,"y":27.95,"z":0.5},{"x":36.9,"y":29.07,"z":0.5},{"x":38.01,"y":29.07,"z":0.5},{"x":39.13,"y":30.19,"z":0.5},{"x":40.25,"y":30.19,"z":0.5},{"x":41.37,"y":31.31,"z":0.5},{"x":42.49,"y":31.31,"z":0.5},{"x":43.61,"y":32.42,"z":0.5},{"x":44.72,"y":33.54,"z":0.5},{"x":46.96,"y":34.66,"z":0.5},{"x":49.2,"y":34.66,"z":0.5},{"x":51.43,"y":35.78,"z":0.5},{"x":53.67,"y":36.9,"z":0.5},{"x":55.9,"y":36.9,"z":0.5},{"x":58.14,"y":38.01,"z":0.5},{"x":59.26,"y":38.01,"z":0.5},{"x":61.49,"y":38.01,"z":0.5},{"x":62.61,"y":38.01,"z":0.5},{"x":63.73,"y":38.01,"z":0.5},{"x":64.85,"y":38.01,"z":0.5},{"x":65.97,"y":38.01,"z":0.5},{"x":67.08,"y":38.01,"z":0.5},{"x":68.2,"y":36.9,"z":0.5},{"x":70.44,"y":35.78,"z":0.5},{"x":71.56,"y":35.78,"z":0.5},{"x":72.68,"y":34.66,"z":0.5},{"x":73.79,"y":34.66,"z":0.5},{"x":74.91,"y":33.54,"z":0.5},{"x":76.03,"y":33.54,"z":0.5},{"x":77.15,"y":32.42,"z":0.5},{"x":78.27,"y":31.31,"z":0.5},{"x":80.5,"y":30.19,"z":0.5},{"x":81.62,"y":29.07,"z":0.5},{"x":83.86,"y":27.95,"z":0.5},{"x":84.97,"y":26.83,"z":0.5},{"x":86.09,"y":25.72,"z":0.5},{"x":87.21,"y":24.6,"z":0.5},{"x":88.33,"y":23.48,"z":0.5},{"x":90.56,"y":21.24,"z":0.5},{"x":91.68,"y":20.13,"z":0.5},{"x":91.68,"y":19.01,"z":0.5},{"x":93.92,"y":17.89,"z":0.5},{"x":95.04,"y":16.77,"z":0.5},{"x":96.16,"y":14.54,"z":0.5},{"x":98.39,"y":14.54,"z":0.5},{"x":98.39,"y":12.3,"z":0.5},{"x":99.51,"y":11.18,"z":0.5},{"x":100.63,"y":11.18,"z":0.5},{"x":100.63,"y":10.06,"z":0.5},{"x":101.75,"y":10.06,"z":0.5},{"x":102.86,"y":8.94,"z":0.5},{"x":102.86,"y":7.83,"z":0.5}]}],"color":"black","fill":"solid","dash":"solid","size":"m","isComplete":true,"isClosed":false,"isPen":false},"parentId":"shape:WBNyIQc5bEj3Q17u6WyZP","index":"a4","typeName":"shape"},{"x":503.6476640403421,"y":-440.10380417378974,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:WBNyIQc5bEj3Q17u6WyZP","type":"group","parentId":"page:page","index":"aR","props":{},"typeName":"shape"},{"x":644.6459205484521,"y":504.0695894976558,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:dAs-42W6Zu9ePKwOO_2QC","type":"arrow","parentId":"page:page","index":"aS","props":{"dash":"dotted","size":"m","fill":"solid","color":"black","labelColor":"black","bend":0,"start":{"type":"binding","boundShapeId":"shape:gPSCaZ64vZzwpMjhevnpr","normalizedAnchor":{"x":0.1661237794856864,"y":0.33112564654950893},"isPrecise":true,"isExact":false},"end":{"type":"binding","boundShapeId":"shape:bCslUaG8j4a7h9UZNM9m2","normalizedAnchor":{"x":0.9876012248295638,"y":0.4493390926137407},"isPrecise":true,"isExact":false},"arrowheadStart":"none","arrowheadEnd":"none","text":"Greift über\nFirmennetz auf Server zu","labelPosition":0.5,"font":"draw"},"typeName":"shape"},{"x":-1046.6049264225544,"y":258.77366512781555,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:l7JAZCXlhd9XSndFxD5f-","type":"arrow","parentId":"page:page","index":"aKV","props":{"dash":"dotted","size":"m","fill":"solid","color":"black","labelColor":"black","bend":-202.62279854247708,"start":{"type":"binding","boundShapeId":"shape:u9F5OPoRDpK3tG90jQiA-","normalizedAnchor":{"x":0.21496914699144268,"y":0.5090872324840892},"isPrecise":false,"isExact":false},"end":{"type":"binding","boundShapeId":"shape:bCslUaG8j4a7h9UZNM9m2","normalizedAnchor":{"x":0.05051902544138064,"y":0.3126812718696317},"isPrecise":true,"isExact":false},"arrowheadStart":"none","arrowheadEnd":"none","text":"Fragt alle X Minuten\nden aktuellen Stand ab","labelPosition":0.5,"font":"draw"},"typeName":"shape"},{"x":-1187.3901880962412,"y":217.97251602925576,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:IXh66eySuDcc3l78ljZxO","type":"text","props":{"color":"black","size":"m","w":130.65625,"text":"10.0.0.X/24","font":"draw","align":"start","autoSize":true,"scale":1},"parentId":"page:page","index":"aT","typeName":"shape"},{"x":-516.6353056610758,"y":-367.56350839160575,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:zDl5vNPUgOGNkwIXwWM6V","type":"text","props":{"color":"black","size":"m","w":130.65625,"text":"10.0.0.X/24","font":"draw","align":"start","autoSize":true,"scale":1},"parentId":"page:page","index":"aU","typeName":"shape"},{"x":-1081.7625288303404,"y":-135.41617135616832,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:uik3RsxktAnWhoPUI_im4","type":"text","props":{"color":"black","size":"s","w":354.28125,"text":"(benötigt API Key \noder irgendein anderes Verfahren für\nAuthentifizierung)","font":"draw","align":"start","autoSize":true,"scale":1},"parentId":"page:page","index":"aV","typeName":"shape"},{"x":-612.088779816152,"y":1291.5018737613298,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:EBBGSJu0ROVCIfxENiMX8","type":"geo","props":{"w":700.8358878477673,"h":304.2505690224648,"geo":"rectangle","color":"black","labelColor":"black","fill":"pattern","dash":"dotted","size":"l","font":"draw","text":"(Anzeige bezüglich Belegungen)","align":"start","verticalAlign":"middle","growY":0,"url":""},"parentId":"page:page","index":"aW","typeName":"shape"},{"x":-910.2846111466276,"y":510.4407114947036,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:fHzu6PHpnyGryWZgUG9mB","type":"arrow","parentId":"page:page","index":"aX","props":{"dash":"solid","size":"l","fill":"pattern","color":"black","labelColor":"black","bend":-107.18361815914412,"start":{"type":"binding","boundShapeId":"shape:aS6vClkZogBZmo2r0GHzI","normalizedAnchor":{"x":0.9603273420776365,"y":0.9613532864697871},"isPrecise":false,"isExact":false},"end":{"type":"binding","boundShapeId":"shape:EBBGSJu0ROVCIfxENiMX8","normalizedAnchor":{"x":0.04751619870410365,"y":0.2089552238805961},"isPrecise":true,"isExact":false},"arrowheadStart":"none","arrowheadEnd":"none","text":"HDMI Kabel","labelPosition":0.5,"font":"draw"},"typeName":"shape"},{"x":-1040.893049409412,"y":981.737519931889,"rotation":0,"isLocked":false,"opacity":1,"meta":{},"id":"shape:QJiAFNLEVp8_ZfNAmpb29","type":"text","props":{"color":"black","size":"m","w":397.75,"text":"startet GUI\nmit Chrome Kiosk Mode\nund Zeigt 10.0.0.X/anzeige (o. Ä.)\nan","font":"draw","align":"start","autoSize":true,"scale":1},"parentId":"page:page","index":"aY","typeName":"shape"},{"meta":{},"id":"asset:-1084722902","type":"image","typeName":"asset","props":{"name":"tldrawFile","src":"","w":800,"h":479,"mimeType":"image/png","isAnimated":false}},{"meta":{},"id":"asset:809752036","type":"image","typeName":"asset","props":{"name":"tldrawFile","src":"","w":1000,"h":474.1666666666667,"mimeType":"image/png","isAnimated":false}},{"meta":{},"id":"asset:670164008","type":"image","typeName":"asset","props":{"name":"tldrawFile","src":"","w":1000,"h":996.09375,"mimeType":"image/png","isAnimated":false}}]} \ No newline at end of file diff --git a/docs/LICENSE.md b/docs/LICENSE.md new file mode 100644 index 000000000..1adecc903 --- /dev/null +++ b/docs/LICENSE.md @@ -0,0 +1,360 @@ +# Mercedes-Benz Inner Source License 1.0 ("ISL") + +Copyright © 2022 Mercedes-Benz Group AG + +SPDX-License-Identifier: LicenseRef-MB-ISL-1.0 + +## 0. Preamble + +0.0 This Mercedes-Benz Inner Source License 1.0 succeeds the +Daimler Inner Source License 1.0 as a later version and is similar in spirit. + +0.1 Mercedes-Benz Group AG ("Mercedes-Benz") provides a platform for collaborative +development ("SCR") and use of source code and associated data (e.g. parameters, +documentation) for use by and for purposes of companies of the Mercedes-Benz +Group (§§ 15 et seq. Aktiengesetz, German Stock Companies Act; all such +companies together "Mercedes-Benz Companies"). This also includes the use by +third parties only as far as these are commissioned directly by a +Mercedes-Benz Company with performances for purposes of Mercedes-Benz Companies +("Commissioned Third Parties"). + +0.2 The SCR and cooperation model is named "Mercedes-Benz +Inner Source Platform". Inner Source +follows the pattern of Open Source software development, is however restricted to +use by and for purposes of Mercedes-Benz Companies due to technical, security, +relevance and majority reasons. + +0.3 Source code uploads to repositories ("Projects") on the SCR come +with the expectation that other parties in this cooperation model will +improve such Projects through comments, suggestions, source code to +provide benefits of the collaboration for all involved parties. + +## 1. Scope of ISL + +1.1 The ISL is an agreement by a Mercedes-Benz Company or a Commissioned Third +Party, which contributes to or uses content on the SCR ("Participant"), +with Mercedes-Benz or another Mercedes-Benz Company also for the direct benefit of +other Mercedes-Benz Companies and Commissioned Third Parties. The direct +benefit of Commissioned Third Parties is limited to grant of rights +according to Section 4.4. + +1.2 The ISL applies to Projects uploaded to or available on the SCR, +especially source code and data, that contain this `LICENSE.md` file +incorporating the unmodified ISL or that expressly refer to the ISL e.g. +by stating "Licensed under Mercedes-Benz Inner Source License 1.0" (all +together "Content"). + +1.3 The ISL applies to any and all handling and use of Content on the +SCR, including copying, integration, making and/or making available +modifications as well as any possible use of Content in foreseen or +unforeseen manner by any means. + +## 2. Consent to and Prevalence of ISL + +2.1 The ISL is accepted by a Mercedes-Benz Company by accessing or making use +of Content on the SCR through its employees or vicarious agents. + +2.2 The ISL is accepted by a Commissioned Third Party by accessing or +making use of Content on the SCR through its employees or vicarious +agents. Subsidiarily, a user acting for a Commissioned Third Party +accepts the ISL by such action for himself/herself and on behalf of such +Commissioned Third Party. + +2.3 Subsidiarily to Sections 2.1 and 2.2, the regulations in this ISL +for rights to Content in Sections 3 and 4 are accepted by and apply +analogously to any party, which contributes Content to the SCR, in +respect of the offer of concession of rights according to Section 3.1 +and the scope of rights granted according to Section 4. Such party is +not able to accept an offer in respect of Content (see Section 3.2). + +2.4 The ISL is prevalent for any Content on the SCR. Any exception from +the ISL is only valid if and applicable as far as (i) the ISL itself +expressly provides for other stipulations or such is expressly agreed +upon in written form for each individual case and (ii) in any case the +Content to which an exception applies is to be transparently marked as +underlying different stipulations. + +2.5 The ISL itself can only be terminated extraordinarily for good cause +by notice in written form by one affected party to the other affected +parties in respect of specific affected Content. Any obligations which +arose before or are caused by such termination remain unaffected. The +stipulations on termination of rights according Section 6 of this ISL remain unaffected. + +## 3. Offer of and Consent to Concession of Rights to Content available on SCR + +3.1 The Mercedes-Benz Company or Commissioned Third Party which makes Content +available on the SCR thereby offers to any other Mercedes-Benz Company and +Commissioned Third Party the rights to Content described in Section 4. +This applies irrespective of the means of making available Content on +the SCR (e.g. upload, pull request, etc.). Contributions, e.g. through +pull request, to Content and/or Projects under ISL shall state, +"Licensed under Mercedes-Benz Inner Source License 1.0 and any later version +similar in spirit". Such offer is valid as long as such Content or a +modification of such Content is available on the SCR. + +3.2 Any Mercedes-Benz Company can accept such offer in respect of Content +available on the SCR by accessing or using such Content in any way by +itself or through Commissioned Third Parties as agreed between the +Mercedes-Benz Company and the Commissioned Third Party in the course of +commissioning. A Commissioned Third Party can accept such offer only +within the scope of its duties commissioned by a Mercedes-Benz Company and +subject to Section 4.4 by accessing or using such Content. Prerequisite +for an acceptance is a valid agreement of the ISL. Any restriction or +reservation for acceptance of the ISL makes acceptance of the offer null +and void. + +## 4. Rights granted to Content + +4.1 Upon acceptance of an offer to Content, +the rights to such Content are granted directly by the Participant to +the accepting entity non-exclusively. + +4.2 The rights to Content encompass any and all possible handling and +use of such Content for any purposes of Mercedes-Benz Companies in any +possible foreseen and unforeseen manner by any means and in any form. +This especially encompasses any reproduction, translation, adaptation, +arrangement and other modifications as well as the reproduction of the +results thereof, any form of distribution of Content or modified Content +including rental. This also encompasses any use without attribution to +any authors of Content as far as legally possible. The rights do not +entitle to remove any copyright information or legal notice unless +otherwise expressly entitled under this ISL and especially Section 5.5. +Any communication of Content to the public is subject to the additional +provisions of Section 5. + +4.3 The rights to Content are granted to a Mercedes-Benz Company for the +duration of its membership to the Mercedes-Benz Companies. + +4.4 The rights to Content granted to Commissioned Third Parties are +limited to the necessary use of Content for the benefit of performances +for purposes of Mercedes-Benz Companies as far as and as long as such +Commissioned Third Party is commissioned with such performances. For the +avoidance of doubt, rights to Content granted to Commissioned Third +Parties do also terminate with the end of the membership of its +commissioning Mercedes-Benz Company to the Mercedes-Benz Companies. + +4.5 Any grant of rights to Content beyond the scope of this Section 4 by +a holder of rights by separate means remains +unaffected. As far as a Commissioned Third Party has granted exclusive +rights to specific content to a Mercedes-Benz Company, this Mercedes-Benz Company +consents to the grant of rights according to this Section 4 by ordering +or allowing the Commissioned Third Party to make such content available +on the SCR. Any grant of rights according to this ISL does not restrict +rights of a Mercedes-Benz Company to Content except as expressly stated in +this Section 4. + +4.6 Any rights granted to specific Content according to this Section 4 +remain unaffected if the availability of the respective Content on the +SCR ends. + +4.7 Any license terms and conditions pertaining to Free and Open Source +Software (FOSS) as parts of Content or parts of Content from third +parties shall prevail over this Section 4. Any obligations of a +Participant in connection with FOSS remain unaffected, especially any +obligations of a Commissioned Third Parties in the context of its +commission. + +4.8 Use of licensed Content under a particular version of the ISL, may +continue under the terms of that version of the ISL notwithstanding the +application of a later ISL version to a repository. Any party +contributing to a repository accepts and grants rights to Content +including such later application of a later ISL version in accordance +with Section 3.1. + +4.9 Any Participant who makes Content available on a repository in the +SCR entitles the employee responsible for a Project ("Repository Owner") +of such repository to decide on external use of such Content according +to Section 5 in his/her own discretion as far as entitled by its respective +Mercedes-Benz Company. + +## 5. External Use of Content + +5.1 Any use of Content beyond the internal purposes of Mercedes-Benz Companies +requires the prior express consent of the Repository Owner as far as +entitled by its respective Mercedes-Benz Company in each individual case. Use +beyond internal purposes of Mercedes-Benz Companies in particular applies to +communication of Content to the public, including making available +Content or modified Content to the public ("External Use"), whether on a +restricted basis or by way of open source software. Consent to External +Use is deemed to be given with upload of Content to the SCR, subject to +the following provisions. This prerequisite of consent must not be +circumvented in any way (e.g. by forking a repository). + +5.2 The rights to Content granted to Commissioned Third Parties are +limited according to this ISL and do not +include the rights to External Use. + +5.3 The External Use in form of publication under an open source license +is limited to the Repository Owner, as far as entitled by the employing +Mercedes-Benz Company. + +5.4 For any Content intended for External Use, all current standards, +especially for source code, of Mercedes-Benz Companies apply irrespective of +the ISL. The Repository Owner has to decide on any External Use +including its respective scope in each individual case, taking into +account the foreseen benefit for Mercedes-Benz Companies of External +Use as well as the inevitable loss of confidentiality of externally +disclosed Content. The Repository Owner has to withdraw +consent to External Use if its prerequisites lapse. Any Repository Owner +shall adequately document any decision in respect of External Use of +Content in the repository including the decision and date of decision. + +5.5 As long as the necessary consent of the Repository Owner is not +withdrawn, the consent includes the External Use of any later +contributions to Content of such repository, irrespective of the means +of a contribution (e.g. upload, pull request, etc.). Any rights already +granted for External Use of specific Content remain unaffected by a +later withdrawal of necessary consent to External Use. + +5.6 Any External Use of Content requires prior relicensing of the +Content, e.g. using a Mercedes-Benz proprietary license. For the avoidance of +doubt, this ISL shall not be used in externally disclosed Content. + +5.7 Any Participant has to instruct and entitle its Repository Owners +according to this Section 5. + +## 6. Termination of Rights granted to Content + +6.1 Any rights to any Content granted to a recipient according to +Section 4 terminate automatically and completely if the recipient of +such rights + +(i) declares a termination of the ISL, except within the scope and +under the prerequisites of Section 2.5, + +(ii) violates the scope of rights granted by this ISL by acts or +omissions by itself or through third parties, in case of a Mercedes-Benz +Company only after a reasonable cure period expressly set has lapsed +without effect, or + +(iii) is a legal entity whose membership to the Mercedes-Benz Companies +ends. In case of termination of membership of a Mercedes-Benz Company to the +Mercedes-Benz Companies "Former Member", the Former Member requires a +separate agreement with the Content providing Mercedes-Benz Companies for +the further use of Content. + +For the avoidance of doubt, any rights granted to Content provided by +a Former Member do not terminate by this Section 6. + +6.2 Any rights to specific Content granted to a Commissioned Third Party +terminate completely and automatically if the commissioning with the +performances for Mercedes-Benz Companies, which require the use of the +affected Content, ends for whatsoever reason. + +6.3 If the ISL is terminated according to Section 2.5, any and all +rights to affected Content granted by the terminating party to the +affected parties terminate completely and automatically. + +6.4 If rights to Content are terminated according to Section 6.1(ii) the +recipient may request a license to Content from the offering Mercedes-Benz +Company at a later point in time. It is solely up to the discretion of +the offering Mercedes-Benz Company to grant rights to such party under the ISL +again. + +## 7. Prerequisites for Use of Content available on SCR + +7.1 It lies completely in the sole responsibility of any recipient of +any Content to assess and verify if and to which extent such Content is +suitable and technically qualified to be used for a particular purpose +and a particular application intended by such recipient. Any contractual +obligations and duties of Commissioned Third Parties and commissioned +Mercedes-Benz Companies (a Mercedes-Benz Company that is commissioned by another +Mercedes-Benz Company), especially based on any contract with a Mercedes-Benz +Company, remain unaffected and are not restricted. + +7.2 The user is solely and completely responsible for any use of Content +by itself and on its behalf. + +7.3 The user is solely and completely responsible for the fulfillment of +any obligations in connection with Content, especially arising out of +Free and Open Source licenses or licenses for content from third +parties. + +## 8. Remuneration, Warranty and Liability + +8.1 Content is made available and rights to Content are granted without +remuneration but with respect to Section 0.3. + +8.2 Any Content is made available on the SCR "as is" without warranty. +There is no warranty provided by any Participant beyond mandatory +statutory obligations and subject to Section 8.4. + +8.3 No Participant assumes any liability exceeding mandatory statutory +obligations (intentional damage; damage to life and/or body; statutory +product liability). This especially applies to any obligations of care +or indemnification in connection with Content. + +8.4 Any warranty and liability of Commissioned Third Parties and +commissioned Mercedes-Benz Companies, especially based on any contract with a +Mercedes-Benz Company, remain unaffected and are not restricted by Sections +8.2 and 8.3. + +## 9. Confidentiality of Content + +9.1 As any Content is made available for purposes of Mercedes-Benz Companies +only, any user of the SCR is obliged to take adequate precautions +against unintended or unauthorized access or disclosure of Content to +third parties. + +9.2 Any Content not expressly marked or declared as "public", according +to Mercedes-Benz Information Classification, (by and within Mercedes-Benz Companies) +is to be treated as confidential and business secret of Mercedes-Benz +Companies by any Mercedes-Benz Company and any Commissioned Third Party. Any +such party has to take reasonable precautions, at least as for its own +business secrets, to keep such Content secret and protect it against +unauthorized disclosure to third parties, especially if a third party is +not bound to adequate secrecy. Mercedes-Benz Companies may be entitled to use +such Content for External Use in adherence with Section 5. For the +avoidance of doubt, the SCR function "public" is by no means to be +misunderstood to be such a declaration or marking of "public". + +## 10. Final Provisions + +10.1 Any party, which accesses, uses or downloads Content, is solely +responsible for adherence to all applicable regulations and requirements +including but not limited to the regulations and requirements in +connection with export control. + +10.2 Any later or new version of the ISL must be approved by the FOSS +Center of Competence or any comparable successor thereof. + +10.3 Except where otherwise expressly stated, this ISL does not +encompass grant of rights to trade marks, patents or similar +intellectual property rights. + +10.4 This ISL constitutes the complete agreement on its subject matter. +Any modifications shall be agreed upon in written form. If any +stipulation of this ISL should be held to be invalid or unenforceable, +this does not affect the remaining stipulations. The affected parties +will substitute an invalid or unenforceable stipulation with a provision +that comes as close as possible to the intent and effect of the affected +stipulation. + +10.5 Irrespective the regulations on dispute resolution in intra-group +agreements, the following applies: In case of conflict over +applicability and/or interpretation of the ISL between more than one +Mercedes-Benz Company, the FOSS Center of Competence, or any comparable +successor thereof, shall be addressed by one or the parties jointly to +resolve the conflict through an ultimate decision. In the event that +such ultimate decision cannot be made, the parties adhere to the +regulations on dispute resolution in intra-group agreements if +concluded. + +10.6 The ISL is governed by applicable German Law for domestic business +under exclusion of the UN-purchase rules (CISG). Sole legal venue for +any conflict arising in connection with the ISL is the appropriate civil +courts in Stuttgart (Mitte), Germany; mandatory statutory jurisdiction +remains unaffected. + +10.7 Any and all Participants on the SCR are obliged to desist from all +practices which may lead to penal liability due to fraud or +embezzlement, insolvency crimes, crimes in violation of competition, +guaranteeing advantages, bribery, acceptance of bribes or other +corruption crimes on the part of persons employed by the respective +Participant or other third parties. In the event of violation of the +above, Mercedes-Benz or the right granting Mercedes-Benz Company has the right to +immediately withdraw from or terminate all legal transactions existing +with the respective Participant and the right to cancel all +negotiations. The above notwithstanding, any Participant is obliged to +adhere to all laws and regulations applicable to both itself and the +commercial relationship with Mercedes-Benz Companies. diff --git a/docs/MYP.dbml b/docs/MYP.dbml new file mode 100644 index 000000000..7d90590d6 --- /dev/null +++ b/docs/MYP.dbml @@ -0,0 +1,60 @@ + +Table "Printer" { + "id" int [unique, pk] + "name" text + "description" text + "status" int [note: '0: OPERATIONAL\n1: OUT_OF_ORDER'] + "created_at" timestamp + "updated_at" timestamp +} + +Table "PrintJob" { + "id" int [unique, pk] + "printer_id" int + "user_id" int + "start_at" timestamp + "duration_in_minutes" int + "comments" text + "aborted" bool [default: false] + "abort_reason" text [note: 'Error code displayed on printer'] + "created_at" timestamp + "updated_at" timestamp +} + + +Table "Account" { + "id" text [unique, pk] + "userId" text + "type" text + "provider" text + "providerAccountId" text + "refresh_token" text + "access_token" text + "expires_at" int + "token_type" text + "scope" text + "id_token" text + "session_state" text +} + +Table "Session" { + "id" text [unique, pk] + "sessionToken" text [unique] + "userId" text + "expires" datetime +} + +Table "User" { + "id" text [pk] + "name" text + "email" text [unique] + "role" text [note: 'ADMIN,USER,GUEST'] +} + +Ref:"PrintJob"."user_id" < "User"."id" + +Ref:"Account"."userId" < "User"."id" + +Ref:"Session"."userId" < "User"."id" + +Ref:"PrintJob"."printer_id" < "Printer"."id" diff --git a/docs/MYP.png b/docs/MYP.png new file mode 100644 index 000000000..5d7a21886 Binary files /dev/null and b/docs/MYP.png differ diff --git a/docs/MYP.sql b/docs/MYP.sql new file mode 100644 index 000000000..9c0e3fbde --- /dev/null +++ b/docs/MYP.sql @@ -0,0 +1,57 @@ +CREATE TABLE `Printer` ( + `id` int UNIQUE PRIMARY KEY, + `name` text, + `description` text, + `status` int COMMENT '0: OPERATIONAL +1: OUT_OF_ORDER', + `created_at` timestamp, + `updated_at` timestamp +); + +CREATE TABLE `PrintJob` ( + `id` int UNIQUE PRIMARY KEY, + `printer_id` int, + `user_id` int, + `start_at` timestamp, + `duration_in_minutes` int, + `comments` text, + `aborted` boolean DEFAULT false, + `abort_reason` text COMMENT 'Error code displayed on printer' +); + +CREATE TABLE `Account` ( + `id` text UNIQUE PRIMARY KEY, + `userId` text, + `type` text, + `provider` text, + `providerAccountId` text, + `refresh_token` text, + `access_token` text, + `expires_at` int, + `token_type` text, + `scope` text, + `id_token` text, + `session_state` text +); + +CREATE TABLE `Session` ( + `id` text UNIQUE PRIMARY KEY, + `sessionToken` text UNIQUE, + `userId` text, + `expires` datetime +); + +CREATE TABLE `User` ( + `id` text PRIMARY KEY, + `name` text, + `email` text UNIQUE, + `role` text COMMENT 'ADMIN,USER,GUEST' +); + +ALTER TABLE `User` ADD FOREIGN KEY (`id`) REFERENCES `PrintJob` (`user_id`); + +ALTER TABLE `User` ADD FOREIGN KEY (`id`) REFERENCES `Account` (`userId`); + +ALTER TABLE `User` ADD FOREIGN KEY (`id`) REFERENCES `Session` (`userId`); + +ALTER TABLE `Printer` ADD FOREIGN KEY (`id`) REFERENCES `PrintJob` (`printer_id`); diff --git a/frontend-aenderungen.md b/frontend-aenderungen.md new file mode 100644 index 000000000..97b5bba8e --- /dev/null +++ b/frontend-aenderungen.md @@ -0,0 +1,22 @@ +# Notwendige Frontend-Änderungen + +1. Frontend-Authentifizierung anpassen: + - GitHub OAuth durch lokale Authentifizierung ersetzen + - Login-Komponenten für Benutzername/Passwort erstellen + - Registrierungs-Formular implementieren + - API-Routen für Login- und Registrierungsprozess anpassen + +2. Datenbankschema: + - Users-Tabelle anpassen um Passwort-Hash zu unterstützen + - GitHub-ID entfernen oder optional machen + +3. Auth-System: + - Lucia.js: Anpassung von OAuth auf Formular-basierte Authentifizierung + - Session-Management beibehalten + +4. API-Endpunktanpassungen: + - Neue Login und Register-Endpunkte erstellen + - Route für initialen Admin-Setup + +Die Änderungen im Frontend sind umfangreicher, da das aktuelle System stark auf GitHub OAuth ausgerichtet ist und komplett umgestellt werden muss. + diff --git a/packages/reservation-platform/.dockerignore b/packages/reservation-platform/.dockerignore new file mode 100644 index 000000000..036c1c8d0 --- /dev/null +++ b/packages/reservation-platform/.dockerignore @@ -0,0 +1,27 @@ +# Build and utility assets +docker/ +scripts/ + +# Ignore node_modules as they will be installed in the container +node_modules + +# Ignore build artifacts +.next + +# Ignore runtime data +db/ + +# Ignore local configuration files +.env +.env.example + +# Ignore version control files +.git +.gitignore + +# Ignore IDE/editor specific files +*.log +*.tmp +*.DS_Store +.vscode/ +.idea/ \ No newline at end of file diff --git a/packages/reservation-platform/.env.example b/packages/reservation-platform/.env.example new file mode 100644 index 000000000..4ab3091ee --- /dev/null +++ b/packages/reservation-platform/.env.example @@ -0,0 +1,3 @@ +# OAuth Configuration +OAUTH_CLIENT_ID=client_id +OAUTH_CLIENT_SECRET=client_secret \ No newline at end of file diff --git a/packages/reservation-platform/.gitignore b/packages/reservation-platform/.gitignore new file mode 100644 index 000000000..70020991f --- /dev/null +++ b/packages/reservation-platform/.gitignore @@ -0,0 +1,43 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# db folder +db/ + +# Env file +.env + + +# dependencies +/node_modules +/.pnp +.pnp.js +.yarn/install-state.gz + +# testing +/coverage + +# next.js +/.next/ +/out/ + +# production +/build + +# misc +.DS_Store +*.pem + +# debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# local env files +.env*.local + +# vercel +.vercel + +# typescript +*.tsbuildinfo +next-env.d.ts diff --git a/packages/reservation-platform/Dockerfile b/packages/reservation-platform/Dockerfile new file mode 100644 index 000000000..11ed8adac --- /dev/null +++ b/packages/reservation-platform/Dockerfile @@ -0,0 +1,34 @@ +FROM node:20-bookworm-slim + +# Create application directory +RUN mkdir -p /usr/src/app + +# Set environment variables +ENV PORT=3000 +ENV NEXT_TELEMETRY_DISABLED=1 + +WORKDIR /usr/src/app + +# Copy package.json and pnpm-lock.yaml +COPY package.json /usr/src/app +COPY pnpm-lock.yaml /usr/src/app + +# Install pnpm +RUN corepack enable pnpm + +# Install dependencies +RUN pnpm install + +# Copy the rest of the application code +COPY . /usr/src/app + +# Initialize Database, if it not already exists +RUN pnpm run db + +# Build the application +RUN pnpm run build + +EXPOSE 3000 + +# Start the application +CMD ["/bin/sh", "-c", "if [ ! -f ./db/sqlite.db ]; then pnpm db; fi && pnpm start"] diff --git a/packages/reservation-platform/README.md b/packages/reservation-platform/README.md new file mode 100644 index 000000000..8faebf075 --- /dev/null +++ b/packages/reservation-platform/README.md @@ -0,0 +1,32 @@ +# MYP - Manage Your Printer + +MYP (Manage Your Printer) ist eine Webanwendung zur Reservierung von 3D-Druckern. +Sie wurde im Rahmen des Abschlussprojektes der Fachinformatiker Ausbildung für Daten- und Prozessanalyse für die Technische Berufsausbildung des Mercedes-Benz Werkes Berlin-Marienfelde entwickelt. + +## Deployment + +### Voraussetzungen + +- Netzwerk auf Raspberry Pi ist eingerichtet +- Docker ist installiert + +### Schritte + +1. Docker-Container bauen (docker/build.sh) +2. Docker-Container speichern (docker/save.sh caddy:2.8 myp-rp:latest) +3. Docker-Container auf Raspberry Pi bereitstellen (docker/deploy.sh) + +## Entwicklerinformationen + +### Raspberry Pi Einstellungen + +Auf dem Raspberry Pi wurde Raspbian Lite installiert. +Unter /srv/* sind die Projektdateien zu finden. + +### Anmeldedaten + +``` +Benutzer: myp +Passwort: (persönlich bekannt) +``` + diff --git a/packages/reservation-platform/biome.json b/packages/reservation-platform/biome.json new file mode 100644 index 000000000..4a0b3f426 --- /dev/null +++ b/packages/reservation-platform/biome.json @@ -0,0 +1,19 @@ +{ + "$schema": "https://biomejs.dev/schemas/1.7.0/schema.json", + "organizeImports": { + "enabled": true + }, + "formatter": { + "enabled": true, + "lineWidth": 120 + }, + "linter": { + "enabled": true, + "rules": { + "recommended": true, + "correctness": { + "noUnusedImports": "error" + } + } + } +} \ No newline at end of file diff --git a/packages/reservation-platform/components.json b/packages/reservation-platform/components.json new file mode 100644 index 000000000..a158e943c --- /dev/null +++ b/packages/reservation-platform/components.json @@ -0,0 +1,17 @@ +{ + "$schema": "https://ui.shadcn.com/schema.json", + "style": "new-york", + "rsc": true, + "tsx": true, + "tailwind": { + "config": "tailwind.config.ts", + "css": "src/app/globals.css", + "baseColor": "neutral", + "cssVariables": true, + "prefix": "" + }, + "aliases": { + "components": "@/components", + "utils": "@/utils/styles" + } +} \ No newline at end of file diff --git a/packages/reservation-platform/docker/build.sh b/packages/reservation-platform/docker/build.sh new file mode 100644 index 000000000..8b04b01b3 --- /dev/null +++ b/packages/reservation-platform/docker/build.sh @@ -0,0 +1,31 @@ +#!/bin/bash + +# Define image name +MYP_RP_IMAGE_NAME="myp-rp" + +# Function to build Docker image +build_image() { + local image_name=$1 + local dockerfile=$2 + local platform=$3 + + echo "Building $image_name Docker image for $platform..." + + docker buildx build --platform $platform -t ${image_name}:latest -f $dockerfile --load . + if [ $? -eq 0 ]; then + echo "$image_name Docker image built successfully" + else + echo "Error occurred while building $image_name Docker image" + exit 1 + fi +} + +# Create and use a builder instance (if not already created) +BUILDER_NAME="myp-rp-arm64-builder" +docker buildx create --name $BUILDER_NAME --use || docker buildx use $BUILDER_NAME + +# Build myp-rp image +build_image "$MYP_RP_IMAGE_NAME" "$PWD/Dockerfile" "linux/arm64" + +# Remove the builder instance +docker buildx rm $BUILDER_NAME diff --git a/packages/reservation-platform/docker/caddy/Caddyfile b/packages/reservation-platform/docker/caddy/Caddyfile new file mode 100644 index 000000000..beacec7e0 --- /dev/null +++ b/packages/reservation-platform/docker/caddy/Caddyfile @@ -0,0 +1,8 @@ +{ + debug +} + +m040tbaraspi001.de040.corpintra.net, m040tbaraspi001.de040.corpinter.net { + reverse_proxy myp-rp:3000 + tls internal +} \ No newline at end of file diff --git a/packages/reservation-platform/docker/compose.yml b/packages/reservation-platform/docker/compose.yml new file mode 100644 index 000000000..538564f22 --- /dev/null +++ b/packages/reservation-platform/docker/compose.yml @@ -0,0 +1,19 @@ +services: + caddy: + image: caddy:2.8 + container_name: caddy + restart: unless-stopped + ports: + - 80:80 + - 443:443 + volumes: + - ./caddy/data:/data + - ./caddy/config:/config + - ./caddy/Caddyfile:/etc/caddy/Caddyfile:ro + myp-rp: + image: myp-rp:latest + container_name: myp-rp + env_file: "/srv/myp-env/github.env" + volumes: + - /srv/MYP-DB:/usr/src/app/db + restart: unless-stopped diff --git a/packages/reservation-platform/docker/deploy.sh b/packages/reservation-platform/docker/deploy.sh new file mode 100644 index 000000000..09b73d6cf --- /dev/null +++ b/packages/reservation-platform/docker/deploy.sh @@ -0,0 +1,36 @@ +#!/bin/bash + +# Directory containing the Docker images +IMAGE_DIR="docker/images" + +# Load all Docker images from the tar.xz files in the IMAGE_DIR +echo "Loading Docker images from $IMAGE_DIR..." + +for image_file in "$IMAGE_DIR"/*.tar.xz; do + if [ -f "$image_file" ]; then + echo "Loading Docker image from $image_file..." + docker load -i "$image_file" + + # Check if the image loading was successful + if [ $? -ne 0 ]; then + echo "Error occurred while loading Docker image from $image_file" + exit 1 + fi + else + echo "No Docker image tar.xz files found in $IMAGE_DIR." + fi +done + +# Execute docker compose +echo "Running docker compose..." +docker compose -f "docker/compose.yml" up -d + +# Check if the operation was successful +if [ $? -eq 0 ]; then + echo "Docker compose executed successfully" +else + echo "Error occurred while executing docker compose" + exit 1 +fi + +echo "Deployment completed successfully" diff --git a/packages/reservation-platform/docker/images/.gitattributes b/packages/reservation-platform/docker/images/.gitattributes new file mode 100644 index 000000000..8b3e666a5 --- /dev/null +++ b/packages/reservation-platform/docker/images/.gitattributes @@ -0,0 +1,2 @@ +caddy_2.8.tar.xz filter=lfs diff=lfs merge=lfs -text +myp-rp_latest.tar.xz filter=lfs diff=lfs merge=lfs -text diff --git a/packages/reservation-platform/docker/images/caddy_2.8.tar.xz b/packages/reservation-platform/docker/images/caddy_2.8.tar.xz new file mode 100644 index 000000000..1b4070e3f --- /dev/null +++ b/packages/reservation-platform/docker/images/caddy_2.8.tar.xz @@ -0,0 +1,3 @@ +version https://git-lfs.github.com/spec/v1 +oid sha256:0e04636560b45cf05774bb4fb734329951c1693a3febd56358ba4d58b10fee82 +size 12365924 diff --git a/packages/reservation-platform/docker/images/myp-rp_latest.tar.xz b/packages/reservation-platform/docker/images/myp-rp_latest.tar.xz new file mode 100644 index 000000000..f617dcd2b --- /dev/null +++ b/packages/reservation-platform/docker/images/myp-rp_latest.tar.xz @@ -0,0 +1,3 @@ +version https://git-lfs.github.com/spec/v1 +oid sha256:1e6b377b9a7a36b32f229567ad766eb06b1e959c7572b7e92dc81c6a3d03e13b +size 191478008 diff --git a/packages/reservation-platform/docker/save.sh b/packages/reservation-platform/docker/save.sh new file mode 100644 index 000000000..2ce2712e6 --- /dev/null +++ b/packages/reservation-platform/docker/save.sh @@ -0,0 +1,68 @@ +#!/bin/bash + +# Get image name as argument +IMAGE_NAME=$1 +PLATFORM="linux/arm64" + +# Define paths +IMAGE_DIR="docker/images" +IMAGE_FILE="${IMAGE_DIR}/${IMAGE_NAME//[:\/]/_}.tar" +COMPRESSED_FILE="${IMAGE_FILE}.xz" + +# Function to pull the image +pull_image() { + local image=$1 + if [[ $image == arm64v8/* ]]; then + echo "Pulling image $image without platform specification..." + docker pull $image + else + echo "Pulling image $image for platform $PLATFORM..." + docker pull --platform $PLATFORM $image + fi + return $? +} + +# Pull the image if it is not available locally +if ! docker image inspect ${IMAGE_NAME} &>/dev/null; then + if pull_image ${IMAGE_NAME}; then + echo "Image $IMAGE_NAME pulled successfully." + else + echo "Error occurred while pulling $IMAGE_NAME for platform $PLATFORM" + echo "Trying to pull $IMAGE_NAME without platform specification..." + + # Attempt to pull again without platform + if pull_image ${IMAGE_NAME}; then + echo "Image $IMAGE_NAME pulled successfully without platform." + else + echo "Error occurred while pulling $IMAGE_NAME without platform." + echo "Trying to pull arm64v8/${IMAGE_NAME} instead..." + + # Construct new image name + NEW_IMAGE_NAME="arm64v8/${IMAGE_NAME}" + if pull_image ${NEW_IMAGE_NAME}; then + echo "Image $NEW_IMAGE_NAME pulled successfully." + IMAGE_NAME=${NEW_IMAGE_NAME} # Update IMAGE_NAME to use the new one + else + echo "Error occurred while pulling $NEW_IMAGE_NAME" + exit 1 + fi + fi + fi +else + echo "Image $IMAGE_NAME found locally. Skipping pull." +fi + +# Save the Docker image +echo "Saving $IMAGE_NAME Docker image..." +docker save ${IMAGE_NAME} > $IMAGE_FILE + +# Compress the Docker image (overwriting if file exists) +echo "Compressing $IMAGE_FILE..." +xz -z --force $IMAGE_FILE + +if [ $? -eq 0 ]; then + echo "$IMAGE_NAME Docker image saved and compressed successfully as $COMPRESSED_FILE" +else + echo "Error occurred while compressing $IMAGE_NAME Docker image" + exit 1 +fi diff --git a/packages/reservation-platform/docs/Admin-Dashboard.md b/packages/reservation-platform/docs/Admin-Dashboard.md new file mode 100644 index 000000000..e192aa696 --- /dev/null +++ b/packages/reservation-platform/docs/Admin-Dashboard.md @@ -0,0 +1,116 @@ +# **Detaillierte Dokumentation des Admin-Dashboards** + +In diesem Abschnitt werde ich die Funktionen und Nutzung des Admin-Dashboards genauer beschreiben, einschließlich der verschiedenen Module, Diagramme und deren Zweck. + +--- + +## **1. Überblick über das Admin-Dashboard** + +Das Admin-Dashboard ist der zentrale Verwaltungsbereich für Administratoren. Es bietet Funktionen wie die Verwaltung von Druckern, Benutzern und Druckaufträgen sowie detaillierte Statistiken und Analysen. + +### **1.1. Navigation** +Das Dashboard enthält ein Sidebar-Menü mit den folgenden Hauptbereichen: +1. **Dashboard:** Übersicht der wichtigsten Statistiken. +2. **Benutzer:** Verwaltung von Benutzerkonten. +3. **Drucker:** Hinzufügen, Bearbeiten und Verwalten von Druckern. +4. **Druckaufträge:** Einsicht in alle Druckaufträge und deren Status. +5. **Einstellungen:** Konfiguration der Anwendung. +6. **Über MYP:** Informationen über das Projekt und den Entwickler. + +Die Sidebar wird in der Datei `src/app/admin/admin-sidebar.tsx` definiert und dynamisch basierend auf der aktuellen Seite hervorgehoben. + +--- + +## **2. Funktionen des Admin-Dashboards** + +### **2.1. Benutzerverwaltung** +- **Datei:** `src/app/admin/users/page.tsx` +- **Beschreibung:** Ermöglicht das Anzeigen, Bearbeiten und Löschen von Benutzerkonten. +- **Funktionen:** + - Anzeige einer Liste aller registrierten Benutzer. + - Bearbeiten von Benutzerrollen (z. B. „admin“ oder „user“). + - Deaktivieren oder Löschen von Benutzerkonten. + +--- + +### **2.2. Druckerverwaltung** +- **Datei:** `src/app/admin/printers/page.tsx` +- **Beschreibung:** Verwaltung der Drucker, einschließlich Hinzufügen, Bearbeiten und Deaktivieren. +- **Funktionen:** + - Statusanzeige der Drucker (aktiv/inaktiv). + - Hinzufügen neuer Drucker mit Namen und Beschreibung. + - Löschen oder Bearbeiten bestehender Drucker. + +--- + +### **2.3. Druckaufträge** +- **Datei:** `src/app/admin/jobs/page.tsx` +- **Beschreibung:** Übersicht aller Druckaufträge, einschließlich Details wie Startzeit, Dauer und Status. +- **Funktionen:** + - Filtern nach Benutzern, Druckern oder Status (abgeschlossen, abgebrochen). + - Anzeigen von Abbruchgründen und Fehlermeldungen. + - Sortieren nach Zeit oder Benutzer. + +--- + +### **2.4. Einstellungen** +- **Datei:** `src/app/admin/settings/page.tsx` +- **Beschreibung:** Konfigurationsseite für die Anwendung. +- **Funktionen:** + - Ändern von globalen Einstellungen wie Standardzeiten oder Fehlerrichtlinien. + - Download von Daten (z. B. Export der Druckhistorie). + +--- + +## **3. Statistiken und Diagramme** + +Das Admin-Dashboard enthält interaktive Diagramme, die wichtige Statistiken visualisieren. Hier einige der zentralen Diagramme: + +### **3.1. Abbruchgründe** +- **Datei:** `src/app/admin/charts/printer-error-chart.tsx` +- **Beschreibung:** Zeigt die Häufigkeit der Abbruchgründe für Druckaufträge in einem Balkendiagramm. +- **Nutzen:** Identifiziert häufige Probleme wie Materialmangel oder Düsenverstopfungen. + +--- + +### **3.2. Fehlerraten** +- **Datei:** `src/app/admin/charts/printer-error-rate.tsx` +- **Beschreibung:** Zeigt die prozentuale Fehlerrate für jeden Drucker in einem Balkendiagramm. +- **Nutzen:** Ermöglicht die Überwachung und Identifizierung von problematischen Druckern. + +--- + +### **3.3. Druckvolumen** +- **Datei:** `src/app/admin/charts/printer-volume.tsx` +- **Beschreibung:** Zeigt das Druckvolumen für heute, diese Woche und diesen Monat. +- **Nutzen:** Vergleich des Druckeroutputs über verschiedene Zeiträume. + +--- + +### **3.4. Prognostizierte Nutzung** +- **Datei:** `src/app/admin/charts/printer-forecast.tsx` +- **Beschreibung:** Ein Bereichsdiagramm zeigt die erwartete Druckernutzung pro Wochentag. +- **Nutzen:** Hilft bei der Planung von Wartungsarbeiten oder Ressourcenzuweisungen. + +--- + +### **3.5. Druckerauslastung** +- **Datei:** `src/app/admin/charts/printer-utilization.tsx` +- **Beschreibung:** Zeigt die aktuelle Nutzung eines Druckers in Prozent in einem Kreisdiagramm. +- **Nutzen:** Überwacht die Auslastung und identifiziert ungenutzte Ressourcen. + +--- + +## **4. Rollenbasierte Zugriffssteuerung** + +Das Admin-Dashboard ist nur für Benutzer mit der Rolle „admin“ zugänglich. Nicht berechtigte Benutzer werden auf die Startseite umgeleitet. Die Zugriffssteuerung erfolgt durch folgende Logik: +- **Datei:** `src/app/admin/layout.tsx` +- **Funktion:** `validateRequest` prüft die Rolle des aktuellen Benutzers. +- **Umleitung:** Falls die Rolle unzureichend ist, wird der Benutzer automatisch umgeleitet: + ```typescript + if (guard(user, IS_NOT, UserRole.ADMIN)) { + redirect("/"); + } + ``` + +Nächster Schritt: [=> API-Endpunkte und deren Nutzung](./API.md) \ No newline at end of file diff --git a/packages/reservation-platform/docs/Architektur.md b/packages/reservation-platform/docs/Architektur.md new file mode 100644 index 000000000..f64fc96b9 --- /dev/null +++ b/packages/reservation-platform/docs/Architektur.md @@ -0,0 +1,79 @@ +# **Technische Architektur und Codeaufbau** + +In diesem Abschnitt erläutere ich die Architektur und Struktur des MYP-Projekts sowie die Funktionalitäten der zentralen Komponenten. + +--- + +## **1. Technische Architektur** + +### **1.1. Architekturübersicht** +MYP basiert auf einer modernen Webanwendungsarchitektur: +- **Frontend:** Entwickelt mit React und Next.js. Stellt die Benutzeroberfläche bereit. +- **Backend:** Nutzt Node.js und Drizzle ORM für die Datenbankinteraktion und Geschäftslogik. +- **Datenbank:** SQLite zur Speicherung von Nutzerdaten, Druckaufträgen und Druckerkonfigurationen. +- **Containerisierung:** Docker wird verwendet, um die Anwendung in isolierten Containern bereitzustellen. +- **Webserver:** Caddy dient als Reverse Proxy mit HTTPS-Unterstützung. + +### **1.2. Modulübersicht** +- **Datenfluss:** Die Anwendung ist stark datengetrieben. API-Routen werden genutzt, um Daten zwischen Frontend und Backend auszutauschen. +- **Rollenbasierter Zugriff:** Über ein Berechtigungssystem können Administratoren und Benutzer unterschiedliche Funktionen nutzen. + +--- + +## **2. Codeaufbau** + +### **2.1. Ordnerstruktur** +Die Datei `repomix-output.txt` zeigt eine strukturierte Übersicht des Projekts. Nachfolgend einige wichtige Verzeichnisse: + +| **Verzeichnis** | **Inhalt** | +|--------------------------|---------------------------------------------------------------------------| +| `src/app` | Next.js-Seiten und Komponenten für Benutzer und Admins. | +| `src/components` | Wiederverwendbare UI-Komponenten wie Karten, Diagramme, Buttons etc. | +| `src/server` | Backend-Logik, Authentifizierung und Datenbankinteraktionen. | +| `src/utils` | Hilfsfunktionen für Analysen, Validierungen und Datenbankzugriffe. | +| `drizzle` | Datenbank-Migrationsdateien und Metadaten. | +| `docker` | Docker-Konfigurations- und Bereitstellungsskripte. | + +--- + +### **2.2. Hauptdateien** +#### **Frontend** +- **`src/app/page.tsx`:** Startseite der Anwendung. +- **`src/app/admin/`:** Admin-spezifische Seiten, z. B. Druckerverwaltung oder Fehlerstatistiken. +- **`src/components/ui/`:** UI-Komponenten wie Dialoge, Formulare und Tabellen. + +#### **Backend** +- **`src/server/auth/`:** Authentifizierung und Benutzerrollenmanagement. +- **`src/server/actions/`:** Funktionen zur Interaktion mit Druckaufträgen und Druckern. +- **`src/utils/`:** Analyse und Verarbeitung von Druckdaten (z. B. Fehlerquoten und Auslastung). + +#### **Datenbank** +- **`drizzle/0000_overjoyed_strong_guy.sql`:** SQLite-Datenbankschema mit Tabellen für Drucker, Benutzer und Druckaufträge. +- **`drizzle.meta/`:** Metadaten zur Datenbankmigration. + +--- + +### **2.3. Datenbankschema** +Das Schema enthält vier Haupttabellen: +1. **`user`:** Speichert Benutzerinformationen, einschließlich Rollen und E-Mail-Adressen. +2. **`printer`:** Beschreibt die Drucker, ihren Status und ihre Eigenschaften. +3. **`printJob`:** Zeichnet Druckaufträge auf, einschließlich Startzeit, Dauer und Abbruchgrund. +4. **`session`:** Verwaltert Benutzer-Sitzungen und Ablaufzeiten. + +--- + +## **3. Wichtige Funktionen** + +### **3.1. Authentifizierung** +Das System nutzt OAuth zur Anmeldung. Benutzerrollen werden in der Tabelle `user` gespeichert und im Backend überprüft. + +### **3.2. Statistiken** +- **Fehlerrate:** Berechnet die Häufigkeit von Abbrüchen für jeden Drucker. +- **Auslastung:** Prozentuale Nutzung der Drucker, basierend auf geplanten und abgeschlossenen Druckaufträgen. +- **Prognosen:** Verwenden historische Daten, um zukünftige Drucknutzungen vorherzusagen. + +### **3.3. API-Endpunkte** +- **`src/app/api/printers/`:** Zugriff auf Druckerkonfigurationsdaten. +- **`src/app/api/job/[jobId]/`:** Verwaltung einzelner Druckaufträge. + +Nächster Schritt: [=> Datenbank und Analytik-Funktionen](./Datenbank.md) \ No newline at end of file diff --git a/packages/reservation-platform/docs/Bereitstellungsdetails .md b/packages/reservation-platform/docs/Bereitstellungsdetails .md new file mode 100644 index 000000000..78a5b5a32 --- /dev/null +++ b/packages/reservation-platform/docs/Bereitstellungsdetails .md @@ -0,0 +1,150 @@ +# **Bereitstellungsdetails und Best Practices** + +In diesem Abschnitt erläutere ich, wie das MYP-Projekt auf einem Server bereitgestellt wird, sowie empfohlene Praktiken zur Verwaltung und Optimierung des Systems. + +--- + +## **1. Bereitstellungsschritte** + +### **1.1. Voraussetzungen** +- **Server:** Raspberry Pi mit installiertem Raspbian Lite. +- **Docker:** Docker und Docker Compose müssen vorab installiert sein. +- **Netzwerk:** Der Server muss über eine statische IP-Adresse oder einen DNS-Namen erreichbar sein. + +### **1.2. Vorbereitung** +#### **1.2.1. Docker-Images erstellen und speichern** +Führen Sie die folgenden Schritte auf dem Entwicklungssystem aus: +1. **Images erstellen:** + ```bash + bash docker/build.sh + ``` +2. **Images exportieren und komprimieren:** + ```bash + bash docker/save.sh + ``` + Dies speichert die Docker-Images im Verzeichnis `docker/images/`. + +#### **1.2.2. Übertragung auf den Server** +Kopieren Sie die erzeugten `.tar.xz`-Dateien auf den Raspberry Pi: +```bash +scp docker/images/*.tar.xz @:/path/to/destination/ +``` + +--- + +### **1.3. Images auf dem Server laden** +Loggen Sie sich auf dem Server ein und laden Sie die Docker-Images: +```bash +docker load -i /path/to/destination/.tar.xz +``` + +--- + +### **1.4. Starten der Anwendung** +Führen Sie das Bereitstellungsskript aus: +```bash +bash docker/deploy.sh +``` +Dieses Skript: +- Startet die Docker-Container mithilfe von `docker compose`. +- Verbindet den Reverse Proxy (Caddy) mit der Anwendung. + +Die Anwendung sollte unter `http://` oder der konfigurierten Domain erreichbar sein. + +--- + +## **2. Best Practices** + +### **2.1. Sicherheit** +1. **Umgebungsvariablen schützen:** + - Stellen Sie sicher, dass die Datei `.env` nicht versehentlich in ein öffentliches Repository hochgeladen wird. + - Verwenden Sie geeignete Zugriffsrechte: + ```bash + chmod 600 .env + ``` +2. **HTTPS aktivieren:** + - Der Caddy-Webserver unterstützt automatisch HTTPS. Stellen Sie sicher, dass eine gültige Domain konfiguriert ist. + +3. **Zugriffsrechte beschränken:** + - Verwenden Sie Benutzerrollen („admin“, „guest“), um den Zugriff auf kritische Funktionen zu steuern. + +--- + +### **2.2. Performance** +1. **Docker-Container optimieren:** + - Reduzieren Sie die Größe der Docker-Images, indem Sie unnötige Dateien in `.dockerignore` ausschließen. + +2. **Datenbankwartung:** + - Führen Sie regelmäßige Backups der SQLite-Datenbank durch: + ```bash + cp db/sqlite.db /path/to/backup/location/ + ``` + - Optimieren Sie die Datenbank regelmäßig: + ```sql + VACUUM; + ``` + +3. **Skalierung:** + - Bei hoher Last kann die Anwendung mit Kubernetes oder einer Cloud-Lösung (z. B. AWS oder Azure) skaliert werden. + +--- + +### **2.3. Fehlerbehebung** +1. **Logs überprüfen:** + - Docker-Logs können wichtige Debug-Informationen liefern: + ```bash + docker logs + ``` + +2. **Health Checks:** + - Integrieren Sie Health Checks in die Docker Compose-Datei, um sicherzustellen, dass die Dienste korrekt laufen. + +3. **Fehlerhafte Drucker deaktivieren:** + - Deaktivieren Sie Drucker mit einer hohen Fehlerrate über das Admin-Dashboard, um die Benutzererfahrung zu verbessern. + +--- + +### **2.4. Updates** +1. **Neue Funktionen hinzufügen:** + - Aktualisieren Sie die Anwendung und erstellen Sie neue Docker-Images: + ```bash + git pull origin main + bash docker/build.sh + ``` + - Stellen Sie die aktualisierten Images bereit: + ```bash + bash docker/deploy.sh + ``` + +2. **Datenbankmigrationen:** + - Führen Sie neue Migrationsskripte mit folgendem Befehl aus: + ```bash + pnpm run db:migrate + ``` + +--- + +## **3. Backup und Wiederherstellung** + +### **3.1. Backups erstellen** +Sichern Sie wichtige Dateien und Datenbanken regelmäßig: +- **SQLite-Datenbank:** + ```bash + cp db/sqlite.db /backup/location/sqlite-$(date +%F).db + ``` +- **Docker-Images:** + ```bash + docker save myp-rp:latest | gzip > /backup/location/myp-rp-$(date +%F).tar.gz + ``` + +### **3.2. Wiederherstellung** +- **Datenbank wiederherstellen:** + ```bash + cp /backup/location/sqlite-.db db/sqlite.db + ``` +- **Docker-Images importieren:** + ```bash + docker load < /backup/location/myp-rp-.tar.gz + ``` + +Nächster Schritt: [=> Admin-Dashboard](./Admin-Dashboard.md) \ No newline at end of file diff --git a/packages/reservation-platform/docs/Datenbank.md b/packages/reservation-platform/docs/Datenbank.md new file mode 100644 index 000000000..253a16eaf --- /dev/null +++ b/packages/reservation-platform/docs/Datenbank.md @@ -0,0 +1,153 @@ +# **Datenbank und Analytik-Funktionen** + +Dieser Abschnitt konzentriert sich auf die Struktur der Datenbank sowie die Analyse- und Prognosefunktionen, die im Projekt verwendet werden. + +--- + +## **1. Datenbankstruktur** + +Das Datenbankschema wurde mit **Drizzle ORM** definiert und basiert auf SQLite. Die wichtigsten Tabellen und ihre Zwecke sind: + +### **1.1. Tabellenübersicht** + +#### **`user`** +- Speichert Benutzerinformationen. +- Enthält Rollen wie „admin“ oder „guest“ zur Verwaltung von Berechtigungen. + +| **Feld** | **Typ** | **Beschreibung** | +|-------------------|------------|-------------------------------------------| +| `id` | `text` | Eindeutige ID des Benutzers. | +| `github_id` | `integer` | ID des Benutzers aus dem OAuth-Dienst. | +| `name` | `text` | Benutzername. | +| `displayName` | `text` | Angezeigter Name. | +| `email` | `text` | E-Mail-Adresse. | +| `role` | `text` | Benutzerrolle, Standardwert: „guest“. | + +--- + +#### **`printer`** +- Beschreibt verfügbare Drucker und deren Status. + +| **Feld** | **Typ** | **Beschreibung** | +|-------------------|------------|-------------------------------------------| +| `id` | `text` | Eindeutige Drucker-ID. | +| `name` | `text` | Name des Druckers. | +| `description` | `text` | Beschreibung oder Spezifikationen. | +| `status` | `integer` | Betriebsstatus (0 = inaktiv, 1 = aktiv). | + +--- + +#### **`printJob`** +- Speichert Informationen zu Druckaufträgen. + +| **Feld** | **Typ** | **Beschreibung** | +|-----------------------|---------------|-------------------------------------------------------| +| `id` | `text` | Eindeutige Auftrags-ID. | +| `printerId` | `text` | Verweis auf die ID des Druckers. | +| `userId` | `text` | Verweis auf die ID des Benutzers. | +| `startAt` | `integer` | Startzeit des Druckauftrags (Unix-Timestamp). | +| `durationInMinutes` | `integer` | Dauer des Druckauftrags in Minuten. | +| `comments` | `text` | Zusätzliche Kommentare. | +| `aborted` | `integer` | 1 = Abgebrochen, 0 = Erfolgreich abgeschlossen. | +| `abortReason` | `text` | Grund für den Abbruch (falls zutreffend). | + +--- + +#### **`session`** +- Verwaltert Benutzer-Sitzungen und Ablaufzeiten. + +| **Feld** | **Typ** | **Beschreibung** | +|-------------------|------------|-------------------------------------------| +| `id` | `text` | Eindeutige Sitzungs-ID. | +| `user_id` | `text` | Verweis auf die ID des Benutzers. | +| `expires_at` | `integer` | Zeitpunkt, wann die Sitzung abläuft. | + +--- + +### **1.2. Relationen** +- `printer` → `printJob`: Druckaufträge sind an spezifische Drucker gebunden. +- `user` → `printJob`: Druckaufträge werden Benutzern zugewiesen. +- `user` → `session`: Sitzungen verknüpfen Benutzer mit Login-Details. + +--- + +## **2. Analytik-Funktionen** + +Das Projekt bietet verschiedene Analytik- und Prognosetools, um die Druckernutzung und Fehler zu überwachen. + +### **2.1. Fehlerratenanalyse** +- Funktion: `calculatePrinterErrorRate` (in `src/utils/analytics/error-rate.ts`). +- Berechnet die prozentuale Fehlerrate für jeden Drucker basierend auf abgebrochenen Aufträgen. + +Beispielausgabe: +```json +[ + { "name": "Drucker 1", "errorRate": 5.2 }, + { "name": "Drucker 2", "errorRate": 3.7 } +] +``` + +--- + +### **2.2. Abbruchgründe** +- Funktion: `calculateAbortReasonsCount` (in `src/utils/analytics/errors.ts`). +- Zählt die Häufigkeit der Abbruchgründe aus der Tabelle `printJob`. + +Beispielausgabe: +```json +[ + { "abortReason": "Materialmangel", "count": 10 }, + { "abortReason": "Düsenverstopfung", "count": 7 } +] +``` + +--- + +### **2.3. Nutzung und Prognosen** +#### Nutzung: +- Funktion: `calculatePrinterUtilization` (in `src/utils/analytics/utilization.ts`). +- Berechnet die Nutzung der Drucker in Prozent. + +Beispielausgabe: +```json +{ "printerId": "1", "utilizationPercentage": 85 } +``` + +#### Prognosen: +- Funktion: `forecastPrinterUsage` (in `src/utils/analytics/forecast.ts`). +- Nutzt historische Daten, um die erwartete Druckernutzung für kommende Tage/Wochen zu schätzen. + +Beispielausgabe: +```json +[ + { "day": 1, "usageMinutes": 300 }, + { "day": 2, "usageMinutes": 200 } +] +``` + +--- + +### **2.4. Druckvolumen** +- Funktion: `calculatePrintVolumes` (in `src/utils/analytics/volume.ts`). +- Vergleicht die Anzahl der abgeschlossenen Druckaufträge für heute, diese Woche und diesen Monat. + +Beispielausgabe: +```json +{ + "today": 15, + "thisWeek": 90, + "thisMonth": 300 +} +``` + +--- + +## **3. Datenbankinitialisierung** +Die Datenbank wird über Skripte in der `package.json` initialisiert: +```bash +pnpm run db:clean # Datenbank und Migrationsordner löschen +pnpm run db:generate # Neues Schema generieren +pnpm run db:migrate # Migrationsskripte ausführen +``` + +Nächster Schritt: [=> Bereitstellungsdetails und Best Practices](./Bereitstellungsdetails.md) \ No newline at end of file diff --git a/packages/reservation-platform/docs/Installation.md b/packages/reservation-platform/docs/Installation.md new file mode 100644 index 000000000..c1d866da9 --- /dev/null +++ b/packages/reservation-platform/docs/Installation.md @@ -0,0 +1,93 @@ +# **Installation und Einrichtung** + +In diesem Abschnitt wird beschrieben, wie die MYP-Anwendung installiert und eingerichtet wird. Diese Schritte umfassen die Vorbereitung der Umgebung, das Konfigurieren der notwendigen Dienste und die Bereitstellung des Projekts. + +--- + +## **Voraussetzungen** +### **Hardware und Software** +- **Raspberry Pi:** Die Anwendung ist für den Einsatz auf einem Raspberry Pi optimiert, auf dem Raspbian Lite installiert sein sollte. +- **Docker:** Docker und Docker Compose müssen installiert sein. +- **Netzwerkzugriff:** Der Raspberry Pi muss im Netzwerk erreichbar sein. + +### **Abhängigkeiten** +- Node.js (mindestens Version 20) +- PNPM (Paketmanager) +- SQLite (für lokale Datenbankverwaltung) + +--- + +## **Schritte zur Einrichtung** + +### **1. Repository klonen** +Klonen Sie das Repository auf Ihr System: +```bash +git clone +cd +``` + +### **2. Konfiguration der Umgebungsvariablen** +Passen Sie die Datei `.env.example` an und benennen Sie sie in `.env` um: +```bash +cp .env.example .env +``` +Erforderliche Variablen: +- `OAUTH_CLIENT_ID`: Client-ID für die OAuth-Authentifizierung +- `OAUTH_CLIENT_SECRET`: Geheimnis für die OAuth-Authentifizierung + +### **3. Docker-Container erstellen** +Führen Sie das Skript `build.sh` aus, um Docker-Images zu erstellen: +```bash +bash docker/build.sh +``` +Dies erstellt die notwendigen Docker-Images, einschließlich der Anwendung und eines Caddy-Webservers. + +### **4. Docker-Images speichern** +Speichern Sie die Images in komprimierter Form, um sie auf anderen Geräten bereitzustellen: +```bash +bash docker/save.sh +``` + +### **5. Bereitstellung** +Kopieren Sie die Docker-Images auf den Zielserver (z. B. Raspberry Pi) und führen Sie `deploy.sh` aus: +```bash +scp docker/images/*.tar.xz :/path/to/deployment/ +bash docker/deploy.sh +``` +Das Skript führt die Docker Compose-Konfiguration aus und startet die Anwendung. + +### **(Optional: 6. Admin-User anlegen)** + +Um einen Admin-User anzulegen, muss zuerst das Container-Image gestartet werden. Anschließend meldet man sich mittels +der GitHub-Authentifizierung bei der Anwendung an. + +Der nun in der Datenbank angelegte User hat die Rolle `guest`. Über das CLI muss man nun in die SQLite-Datenbank (die Datenbank sollte außerhalb des Container-Images liegen) wechseln und +den User updaten. + + +#### SQL-Befehl, um den User zu updaten: +```bash +sqlite3 db.sqlite3 +UPDATE users SET role = 'admin' WHERE id = ; +``` + +--- + +## **Start der Anwendung** +Sobald die Docker-Container laufen, ist die Anwendung unter der angegebenen Domain oder IP-Adresse erreichbar. Standardmäßig verwendet der Caddy-Webserver Port 80 (HTTP) und 443 (HTTPS). + +--- + +## **Optional: Entwicklungsmodus** +Für lokale Tests können Sie die Anwendung ohne Docker starten: +1. Installieren Sie Abhängigkeiten: + ```bash + pnpm install + ``` +2. Starten Sie den Entwicklungsserver: + ```bash + pnpm dev + ``` + Die Anwendung ist dann unter `http://localhost:3000` verfügbar. + +Nächster Schritt: [=> Nutzung](./Nutzung.md) \ No newline at end of file diff --git a/packages/reservation-platform/docs/Nutzung.md b/packages/reservation-platform/docs/Nutzung.md new file mode 100644 index 000000000..2080933b2 --- /dev/null +++ b/packages/reservation-platform/docs/Nutzung.md @@ -0,0 +1,75 @@ +# **Features und Nutzung der Anwendung** + +In diesem Abschnitt beschreibe ich die Hauptfunktionen von MYP (Manage Your Printer) und gebe Anweisungen zur Nutzung der verschiedenen Module. + +--- + +## **1. Hauptfunktionen** + +### **1.1. Druckerreservierung** +- Nutzer können Drucker für einen definierten Zeitraum reservieren. +- Konflikte bei Reservierungen werden durch ein Echtzeit-Überprüfungssystem verhindert. + +### **1.2. Fehler- und Auslastungsanalyse** +- Darstellung von Druckfehlern nach Kategorien und Häufigkeiten. +- Übersicht der aktuellen und historischen Druckernutzung. +- Diagramme zur Fehlerrate, Nutzung und Druckvolumen. + +### **1.3. Admin-Dashboard** +- Verwaltung von Druckern, Nutzern und Druckaufträgen. +- Überblick über alle Abbruchgründe und Druckfehler. +- Zugriff auf erweiterte Statistiken und Prognosen. + +--- + +## **2. Nutzung der Anwendung** + +### **2.1. Login und Authentifizierung** +- Die Anwendung unterstützt OAuth-basierte Authentifizierung. +- Nutzer müssen sich mit einem gültigen Konto anmelden, um Zugriff auf die Funktionen zu erhalten. + +### **2.2. Dashboard** +- Nach dem Login gelangen die Nutzer auf das Dashboard, das einen Überblick über die aktuelle Druckernutzung bietet. +- Administratoren haben Zugriff auf zusätzliche Menüpunkte, wie z. B. Benutzerverwaltung. + +--- + +## **3. Admin-Funktionen** + +### **3.1. Druckerverwaltung** +- Administratoren können Drucker hinzufügen, bearbeiten oder löschen. +- Status eines Druckers (z. B. „in Betrieb“, „außer Betrieb“) kann angepasst werden. + +### **3.2. Nutzerverwaltung** +- Verwalten von Benutzerkonten, einschließlich Rollen (z. B. „Admin“ oder „User“). +- Benutzer können aktiviert oder deaktiviert werden. + +### **3.3. Statistiken und Berichte** +- Diagramme wie: + - **Abbruchgründe:** Zeigt häufige Fehlerursachen. + - **Fehlerrate:** Prozentuale Fehlerquote der Drucker. + - **Nutzung:** Prognosen für die Druckernutzung pro Wochentag. + +--- + +## **4. Diagramme und Visualisierungen** + +### **4.1. Abbruchgründe** +- Ein Säulendiagramm zeigt die Häufigkeiten der Fehlerursachen. +- Nutzt Echtzeit-Daten aus der Druckhistorie. + +### **4.2. Prognostizierte Nutzung** +- Ein Liniendiagramm zeigt die erwartete Druckernutzung pro Tag. +- Hilft bei der Planung von Wartungszeiten. + +### **4.3. Druckvolumen** +- Balkendiagramme vergleichen Druckaufträge heute, diese Woche und diesen Monat. + +--- + +## **5. Interaktive Komponenten** +- **Benachrichtigungen:** Informieren über Druckaufträge, Fehler oder Systemereignisse. +- **Filter und Suchfunktionen:** Erleichtern das Auffinden von Druckern oder Druckaufträgen. +- **Rollenbasierter Zugriff:** Funktionen sind je nach Benutzerrolle eingeschränkt. + +Nächster Schritt: [=> Technische Architektur und Codeaufbau](./Architektur.md) \ No newline at end of file diff --git a/packages/reservation-platform/docs/README.md b/packages/reservation-platform/docs/README.md new file mode 100644 index 000000000..741f73b22 --- /dev/null +++ b/packages/reservation-platform/docs/README.md @@ -0,0 +1,37 @@ +# **Einleitung** + +> Information: Die Dokumenation wurde mit generativer AI erstellt und kann fehlerhaft sein. Im Zweifel bitte die Quellcode-Dateien anschauen oder die Entwickler kontaktieren. + +## **Projektbeschreibung** +MYP (Manage Your Printer) ist eine Webanwendung zur Verwaltung und Reservierung von 3D-Druckern. Das Projekt wurde als Abschlussarbeit im Rahmen der Fachinformatiker-Ausbildung mit Schwerpunkt Daten- und Prozessanalyse entwickelt und dient als Plattform zur einfachen Koordination und Überwachung von Druckressourcen. Es wurde speziell für die Technische Berufsausbildung des Mercedes-Benz Werkes in Berlin-Marienfelde erstellt. + +--- + +## **Hauptmerkmale** +- **Druckerreservierungen:** Nutzer können 3D-Drucker in definierten Zeitfenstern reservieren. +- **Fehleranalyse:** Statistiken über Druckfehler und Abbruchgründe werden visuell dargestellt. +- **Druckauslastung:** Echtzeit-Daten über die Nutzung der Drucker. +- **Admin-Dashboard:** Übersichtliche Verwaltung und Konfiguration von Druckern, Benutzern und Druckaufträgen. +- **Datenbankintegration:** Alle Daten werden in einer SQLite-Datenbank gespeichert und verwaltet. + +--- + +## **Technologien** +- **Frontend:** React, Next.js, TailwindCSS +- **Backend:** Node.js, Drizzle ORM +- **Datenbank:** SQLite +- **Deployment:** Docker und Raspberry Pi +- **Zusätzliche Bibliotheken:** recharts für Diagramme, Faker.js für Testdaten, sowie diverse Radix-UI-Komponenten. + +--- + +## **Dateistruktur** +Die Repository-Dateien sind in logische Abschnitte unterteilt: +1. **Docker-Konfigurationen** (`docker/`) - Skripte und Konfigurationsdateien für die Bereitstellung. +2. **Frontend-Komponenten** (`src/app/`) - Weboberfläche und deren Funktionalitäten. +3. **Backend-Funktionen** (`src/server/`) - Datenbankinteraktionen und Authentifizierungslogik. +4. **Utils und Helferfunktionen** (`src/utils/`) - Wiederverwendbare Dienste und Hilfsmethoden. +5. **Datenbank-Skripte** (`drizzle/`) - Datenbankschemas und Migrationsdateien. + + +Nächster Schritt: [=> Installation](./Installation.md) diff --git a/packages/reservation-platform/drizzle.config.ts b/packages/reservation-platform/drizzle.config.ts new file mode 100644 index 000000000..965ecba70 --- /dev/null +++ b/packages/reservation-platform/drizzle.config.ts @@ -0,0 +1,12 @@ +import { defineConfig } from "drizzle-kit"; + +//@ts-ignore - better-sqlite driver throws an error even though its an valid value +export default defineConfig({ + dialect: "sqlite", + schema: "./src/server/db/schema.ts", + out: "./drizzle", + driver: "libsql", + dbCredentials: { + url: "file:./db/sqlite.db", + }, +}); diff --git a/packages/reservation-platform/drizzle/0000_overjoyed_strong_guy.sql b/packages/reservation-platform/drizzle/0000_overjoyed_strong_guy.sql new file mode 100644 index 000000000..13501e107 --- /dev/null +++ b/packages/reservation-platform/drizzle/0000_overjoyed_strong_guy.sql @@ -0,0 +1,35 @@ +CREATE TABLE `printJob` ( + `id` text PRIMARY KEY NOT NULL, + `printerId` text NOT NULL, + `userId` text NOT NULL, + `startAt` integer NOT NULL, + `durationInMinutes` integer NOT NULL, + `comments` text, + `aborted` integer DEFAULT false NOT NULL, + `abortReason` text, + FOREIGN KEY (`printerId`) REFERENCES `printer`(`id`) ON UPDATE no action ON DELETE cascade, + FOREIGN KEY (`userId`) REFERENCES `user`(`id`) ON UPDATE no action ON DELETE cascade +); +--> statement-breakpoint +CREATE TABLE `printer` ( + `id` text PRIMARY KEY NOT NULL, + `name` text NOT NULL, + `description` text NOT NULL, + `status` integer DEFAULT 0 NOT NULL +); +--> statement-breakpoint +CREATE TABLE `session` ( + `id` text PRIMARY KEY NOT NULL, + `user_id` text NOT NULL, + `expires_at` integer NOT NULL, + FOREIGN KEY (`user_id`) REFERENCES `user`(`id`) ON UPDATE no action ON DELETE cascade +); +--> statement-breakpoint +CREATE TABLE `user` ( + `id` text PRIMARY KEY NOT NULL, + `github_id` integer NOT NULL, + `name` text, + `displayName` text, + `email` text NOT NULL, + `role` text DEFAULT 'guest' +); diff --git a/packages/reservation-platform/drizzle/meta/0000_snapshot.json b/packages/reservation-platform/drizzle/meta/0000_snapshot.json new file mode 100644 index 000000000..6a12d0d99 --- /dev/null +++ b/packages/reservation-platform/drizzle/meta/0000_snapshot.json @@ -0,0 +1,241 @@ +{ + "version": "6", + "dialect": "sqlite", + "id": "791dc197-5254-4432-bd9f-1368d1a5aa6a", + "prevId": "00000000-0000-0000-0000-000000000000", + "tables": { + "printJob": { + "name": "printJob", + "columns": { + "id": { + "name": "id", + "type": "text", + "primaryKey": true, + "notNull": true, + "autoincrement": false + }, + "printerId": { + "name": "printerId", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "userId": { + "name": "userId", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "startAt": { + "name": "startAt", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "durationInMinutes": { + "name": "durationInMinutes", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "comments": { + "name": "comments", + "type": "text", + "primaryKey": false, + "notNull": false, + "autoincrement": false + }, + "aborted": { + "name": "aborted", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false, + "default": false + }, + "abortReason": { + "name": "abortReason", + "type": "text", + "primaryKey": false, + "notNull": false, + "autoincrement": false + } + }, + "indexes": {}, + "foreignKeys": { + "printJob_printerId_printer_id_fk": { + "name": "printJob_printerId_printer_id_fk", + "tableFrom": "printJob", + "tableTo": "printer", + "columnsFrom": [ + "printerId" + ], + "columnsTo": [ + "id" + ], + "onDelete": "cascade", + "onUpdate": "no action" + }, + "printJob_userId_user_id_fk": { + "name": "printJob_userId_user_id_fk", + "tableFrom": "printJob", + "tableTo": "user", + "columnsFrom": [ + "userId" + ], + "columnsTo": [ + "id" + ], + "onDelete": "cascade", + "onUpdate": "no action" + } + }, + "compositePrimaryKeys": {}, + "uniqueConstraints": {} + }, + "printer": { + "name": "printer", + "columns": { + "id": { + "name": "id", + "type": "text", + "primaryKey": true, + "notNull": true, + "autoincrement": false + }, + "name": { + "name": "name", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "description": { + "name": "description", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "status": { + "name": "status", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false, + "default": 0 + } + }, + "indexes": {}, + "foreignKeys": {}, + "compositePrimaryKeys": {}, + "uniqueConstraints": {} + }, + "session": { + "name": "session", + "columns": { + "id": { + "name": "id", + "type": "text", + "primaryKey": true, + "notNull": true, + "autoincrement": false + }, + "user_id": { + "name": "user_id", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "expires_at": { + "name": "expires_at", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false + } + }, + "indexes": {}, + "foreignKeys": { + "session_user_id_user_id_fk": { + "name": "session_user_id_user_id_fk", + "tableFrom": "session", + "tableTo": "user", + "columnsFrom": [ + "user_id" + ], + "columnsTo": [ + "id" + ], + "onDelete": "cascade", + "onUpdate": "no action" + } + }, + "compositePrimaryKeys": {}, + "uniqueConstraints": {} + }, + "user": { + "name": "user", + "columns": { + "id": { + "name": "id", + "type": "text", + "primaryKey": true, + "notNull": true, + "autoincrement": false + }, + "github_id": { + "name": "github_id", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "name": { + "name": "name", + "type": "text", + "primaryKey": false, + "notNull": false, + "autoincrement": false + }, + "displayName": { + "name": "displayName", + "type": "text", + "primaryKey": false, + "notNull": false, + "autoincrement": false + }, + "email": { + "name": "email", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "role": { + "name": "role", + "type": "text", + "primaryKey": false, + "notNull": false, + "autoincrement": false, + "default": "'guest'" + } + }, + "indexes": {}, + "foreignKeys": {}, + "compositePrimaryKeys": {}, + "uniqueConstraints": {} + } + }, + "enums": {}, + "_meta": { + "schemas": {}, + "tables": {}, + "columns": {} + } +} \ No newline at end of file diff --git a/packages/reservation-platform/drizzle/meta/_journal.json b/packages/reservation-platform/drizzle/meta/_journal.json new file mode 100644 index 000000000..73e233cad --- /dev/null +++ b/packages/reservation-platform/drizzle/meta/_journal.json @@ -0,0 +1,13 @@ +{ + "version": "6", + "dialect": "sqlite", + "entries": [ + { + "idx": 0, + "version": "6", + "when": 1715416514336, + "tag": "0000_overjoyed_strong_guy", + "breakpoints": true + } + ] +} \ No newline at end of file diff --git a/packages/reservation-platform/next.config.mjs b/packages/reservation-platform/next.config.mjs new file mode 100644 index 000000000..887b2d216 --- /dev/null +++ b/packages/reservation-platform/next.config.mjs @@ -0,0 +1,26 @@ +/** @type {import('next').NextConfig} */ +const nextConfig = { + async headers() { + return [ + { + source: "/:path*", + headers: [ + { + key: "Access-Control-Allow-Origin", + value: "m040tbaraspi001.de040.corpintra.net", + }, + { + key: "Access-Control-Allow-Methods", + value: "GET, POST, PUT, DELETE, OPTIONS", + }, + { + key: "Access-Control-Allow-Headers", + value: "Content-Type, Authorization", + }, + ], + }, + ]; + }, +}; + +export default nextConfig; diff --git a/packages/reservation-platform/package.json b/packages/reservation-platform/package.json new file mode 100644 index 000000000..d3cc00604 --- /dev/null +++ b/packages/reservation-platform/package.json @@ -0,0 +1,83 @@ +{ + "name": "myp-rp", + "version": "1.0.0", + "private": true, + "packageManager": "pnpm@9.12.1", + "scripts": { + "dev": "next dev", + "build": "next build", + "start": "next start", + "lint": "next lint", + "db:create-default": "mkdir -p db/", + "db:generate-sqlite": "pnpm drizzle-kit generate", + "db:clean": "rm -rf db/ drizzle/", + "db:migrate": "pnpm drizzle-kit migrate", + "db": "pnpm db:create-default && pnpm db:generate-sqlite && pnpm db:migrate", + "db:reset": "pnpm db:clean && pnpm db" + }, + "dependencies": { + "@faker-js/faker": "^9.2.0", + "@headlessui/react": "^2.1.10", + "@headlessui/tailwindcss": "^0.2.1", + "@hookform/resolvers": "^3.9.0", + "@libsql/client": "^0.14.0", + "@lucia-auth/adapter-drizzle": "^1.1.0", + "@radix-ui/react-alert-dialog": "^1.1.2", + "@radix-ui/react-avatar": "^1.1.1", + "@radix-ui/react-dialog": "^1.1.2", + "@radix-ui/react-dropdown-menu": "^2.1.2", + "@radix-ui/react-hover-card": "^1.1.2", + "@radix-ui/react-icons": "^1.3.0", + "@radix-ui/react-label": "^2.1.0", + "@radix-ui/react-scroll-area": "^1.2.0", + "@radix-ui/react-select": "^2.1.2", + "@radix-ui/react-slot": "^1.1.0", + "@radix-ui/react-tabs": "^1.1.1", + "@radix-ui/react-toast": "^1.2.2", + "@remixicon/react": "^4.3.0", + "@tanstack/react-table": "^8.20.5", + "@tremor/react": "^3.18.3", + "arctic": "^1.9.2", + "class-variance-authority": "^0.7.0", + "clsx": "^2.1.1", + "date-fns": "^4.1.0", + "drizzle-orm": "^0.30.10", + "lodash": "^4.17.21", + "lucia": "^3.2.1", + "lucide-react": "^0.378.0", + "luxon": "^3.5.0", + "next": "14.2.3", + "next-themes": "^0.3.0", + "oslo": "^1.2.1", + "react": "^18.3.1", + "react-dom": "^18.3.1", + "react-hook-form": "^7.53.0", + "react-if": "^4.1.5", + "react-timer-hook": "^3.0.7", + "recharts": "^2.13.3", + "regression": "^2.0.1", + "sonner": "^1.5.0", + "sqlite": "^5.1.1", + "sqlite3": "^5.1.7", + "swr": "^2.2.5", + "tailwind-merge": "^2.5.3", + "tailwindcss-animate": "^1.0.7", + "use-debounce": "^10.0.3", + "uuid": "^11.0.2", + "zod": "^3.23.8" + }, + "devDependencies": { + "@biomejs/biome": "^1.9.3", + "@tailwindcss/forms": "^0.5.9", + "@types/lodash": "^4.17.13", + "@types/luxon": "^3.4.2", + "@types/node": "^20.16.11", + "@types/react": "^18.3.11", + "@types/react-dom": "^18.3.1", + "drizzle-kit": "^0.21.4", + "postcss": "^8.4.47", + "tailwindcss": "^3.4.13", + "ts-node": "^10.9.2", + "typescript": "^5.6.3" + } +} diff --git a/packages/reservation-platform/pnpm-lock.yaml b/packages/reservation-platform/pnpm-lock.yaml new file mode 100644 index 000000000..cef9b75aa --- /dev/null +++ b/packages/reservation-platform/pnpm-lock.yaml @@ -0,0 +1,5707 @@ +lockfileVersion: '9.0' + +settings: + autoInstallPeers: true + excludeLinksFromLockfile: false + +importers: + + .: + dependencies: + '@faker-js/faker': + specifier: ^9.2.0 + version: 9.2.0 + '@headlessui/react': + specifier: ^2.1.10 + version: 2.1.10(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@headlessui/tailwindcss': + specifier: ^0.2.1 + version: 0.2.1(tailwindcss@3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3))) + '@hookform/resolvers': + specifier: ^3.9.0 + version: 3.9.0(react-hook-form@7.53.0(react@18.3.1)) + '@libsql/client': + specifier: ^0.14.0 + version: 0.14.0 + '@lucia-auth/adapter-drizzle': + specifier: ^1.1.0 + version: 1.1.0(drizzle-orm@0.30.10(@libsql/client@0.14.0)(@types/better-sqlite3@7.6.11)(@types/react@18.3.11)(better-sqlite3@9.6.0)(react@18.3.1)(sqlite3@5.1.7))(lucia@3.2.1) + '@radix-ui/react-alert-dialog': + specifier: ^1.1.2 + version: 1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-avatar': + specifier: ^1.1.1 + version: 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-dialog': + specifier: ^1.1.2 + version: 1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-dropdown-menu': + specifier: ^2.1.2 + version: 2.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-hover-card': + specifier: ^1.1.2 + version: 1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-icons': + specifier: ^1.3.0 + version: 1.3.0(react@18.3.1) + '@radix-ui/react-label': + specifier: ^2.1.0 + version: 2.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-scroll-area': + specifier: ^1.2.0 + version: 1.2.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-select': + specifier: ^2.1.2 + version: 2.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-slot': + specifier: ^1.1.0 + version: 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-tabs': + specifier: ^1.1.1 + version: 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-toast': + specifier: ^1.2.2 + version: 1.2.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@remixicon/react': + specifier: ^4.3.0 + version: 4.3.0(react@18.3.1) + '@tanstack/react-table': + specifier: ^8.20.5 + version: 8.20.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@tremor/react': + specifier: ^3.18.3 + version: 3.18.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(tailwindcss@3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3))) + arctic: + specifier: ^1.9.2 + version: 1.9.2 + class-variance-authority: + specifier: ^0.7.0 + version: 0.7.0 + clsx: + specifier: ^2.1.1 + version: 2.1.1 + date-fns: + specifier: ^4.1.0 + version: 4.1.0 + drizzle-orm: + specifier: ^0.30.10 + version: 0.30.10(@libsql/client@0.14.0)(@types/better-sqlite3@7.6.11)(@types/react@18.3.11)(better-sqlite3@9.6.0)(react@18.3.1)(sqlite3@5.1.7) + lodash: + specifier: ^4.17.21 + version: 4.17.21 + lucia: + specifier: ^3.2.1 + version: 3.2.1 + lucide-react: + specifier: ^0.378.0 + version: 0.378.0(react@18.3.1) + luxon: + specifier: ^3.5.0 + version: 3.5.0 + next: + specifier: 14.2.3 + version: 14.2.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + next-themes: + specifier: ^0.3.0 + version: 0.3.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + oslo: + specifier: ^1.2.1 + version: 1.2.1 + react: + specifier: ^18.3.1 + version: 18.3.1 + react-dom: + specifier: ^18.3.1 + version: 18.3.1(react@18.3.1) + react-hook-form: + specifier: ^7.53.0 + version: 7.53.0(react@18.3.1) + react-if: + specifier: ^4.1.5 + version: 4.1.5(react@18.3.1) + react-timer-hook: + specifier: ^3.0.7 + version: 3.0.7(react@18.3.1) + recharts: + specifier: ^2.13.3 + version: 2.13.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + regression: + specifier: ^2.0.1 + version: 2.0.1 + sonner: + specifier: ^1.5.0 + version: 1.5.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + sqlite: + specifier: ^5.1.1 + version: 5.1.1 + sqlite3: + specifier: ^5.1.7 + version: 5.1.7 + swr: + specifier: ^2.2.5 + version: 2.2.5(react@18.3.1) + tailwind-merge: + specifier: ^2.5.3 + version: 2.5.3 + tailwindcss-animate: + specifier: ^1.0.7 + version: 1.0.7(tailwindcss@3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3))) + use-debounce: + specifier: ^10.0.3 + version: 10.0.3(react@18.3.1) + uuid: + specifier: ^11.0.2 + version: 11.0.2 + zod: + specifier: ^3.23.8 + version: 3.23.8 + devDependencies: + '@biomejs/biome': + specifier: ^1.9.3 + version: 1.9.3 + '@tailwindcss/forms': + specifier: ^0.5.9 + version: 0.5.9(tailwindcss@3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3))) + '@types/lodash': + specifier: ^4.17.13 + version: 4.17.13 + '@types/luxon': + specifier: ^3.4.2 + version: 3.4.2 + '@types/node': + specifier: ^20.16.11 + version: 20.16.11 + '@types/react': + specifier: ^18.3.11 + version: 18.3.11 + '@types/react-dom': + specifier: ^18.3.1 + version: 18.3.1 + drizzle-kit: + specifier: ^0.21.4 + version: 0.21.4 + postcss: + specifier: ^8.4.47 + version: 8.4.47 + tailwindcss: + specifier: ^3.4.13 + version: 3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3)) + ts-node: + specifier: ^10.9.2 + version: 10.9.2(@types/node@20.16.11)(typescript@5.6.3) + typescript: + specifier: ^5.6.3 + version: 5.6.3 + +packages: + + '@alloc/quick-lru@5.2.0': + resolution: {integrity: sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==} + engines: {node: '>=10'} + + '@babel/runtime@7.24.5': + resolution: {integrity: sha512-Nms86NXrsaeU9vbBJKni6gXiEXZ4CVpYVzEjDH9Sb8vmZ3UljyA1GSOJl/6LGPO8EHLuSF9H+IxNXHPX8QHJ4g==} + engines: {node: '>=6.9.0'} + + '@biomejs/biome@1.9.3': + resolution: {integrity: sha512-POjAPz0APAmX33WOQFGQrwLvlu7WLV4CFJMlB12b6ZSg+2q6fYu9kZwLCOA+x83zXfcPd1RpuWOKJW0GbBwLIQ==} + engines: {node: '>=14.21.3'} + hasBin: true + + '@biomejs/cli-darwin-arm64@1.9.3': + resolution: {integrity: sha512-QZzD2XrjJDUyIZK+aR2i5DDxCJfdwiYbUKu9GzkCUJpL78uSelAHAPy7m0GuPMVtF/Uo+OKv97W3P9nuWZangQ==} + engines: {node: '>=14.21.3'} + cpu: [arm64] + os: [darwin] + + '@biomejs/cli-darwin-x64@1.9.3': + resolution: {integrity: sha512-vSCoIBJE0BN3SWDFuAY/tRavpUtNoqiceJ5PrU3xDfsLcm/U6N93JSM0M9OAiC/X7mPPfejtr6Yc9vSgWlEgVw==} + engines: {node: '>=14.21.3'} + cpu: [x64] + os: [darwin] + + '@biomejs/cli-linux-arm64-musl@1.9.3': + resolution: {integrity: sha512-VBzyhaqqqwP3bAkkBrhVq50i3Uj9+RWuj+pYmXrMDgjS5+SKYGE56BwNw4l8hR3SmYbLSbEo15GcV043CDSk+Q==} + engines: {node: '>=14.21.3'} + cpu: [arm64] + os: [linux] + + '@biomejs/cli-linux-arm64@1.9.3': + resolution: {integrity: sha512-vJkAimD2+sVviNTbaWOGqEBy31cW0ZB52KtpVIbkuma7PlfII3tsLhFa+cwbRAcRBkobBBhqZ06hXoZAN8NODQ==} + engines: {node: '>=14.21.3'} + cpu: [arm64] + os: [linux] + + '@biomejs/cli-linux-x64-musl@1.9.3': + resolution: {integrity: sha512-TJmnOG2+NOGM72mlczEsNki9UT+XAsMFAOo8J0me/N47EJ/vkLXxf481evfHLlxMejTY6IN8SdRSiPVLv6AHlA==} + engines: {node: '>=14.21.3'} + cpu: [x64] + os: [linux] + + '@biomejs/cli-linux-x64@1.9.3': + resolution: {integrity: sha512-x220V4c+romd26Mu1ptU+EudMXVS4xmzKxPVb9mgnfYlN4Yx9vD5NZraSx/onJnd3Gh/y8iPUdU5CDZJKg9COA==} + engines: {node: '>=14.21.3'} + cpu: [x64] + os: [linux] + + '@biomejs/cli-win32-arm64@1.9.3': + resolution: {integrity: sha512-lg/yZis2HdQGsycUvHWSzo9kOvnGgvtrYRgoCEwPBwwAL8/6crOp3+f47tPwI/LI1dZrhSji7PNsGKGHbwyAhw==} + engines: {node: '>=14.21.3'} + cpu: [arm64] + os: [win32] + + '@biomejs/cli-win32-x64@1.9.3': + resolution: {integrity: sha512-cQMy2zanBkVLpmmxXdK6YePzmZx0s5Z7KEnwmrW54rcXK3myCNbQa09SwGZ8i/8sLw0H9F3X7K4rxVNGU8/D4Q==} + engines: {node: '>=14.21.3'} + cpu: [x64] + os: [win32] + + '@cspotcode/source-map-support@0.8.1': + resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==} + engines: {node: '>=12'} + + '@emnapi/core@0.45.0': + resolution: {integrity: sha512-DPWjcUDQkCeEM4VnljEOEcXdAD7pp8zSZsgOujk/LGIwCXWbXJngin+MO4zbH429lzeC3WbYLGjE2MaUOwzpyw==} + + '@emnapi/runtime@0.45.0': + resolution: {integrity: sha512-Txumi3td7J4A/xTTwlssKieHKTGl3j4A1tglBx72auZ49YK7ePY6XZricgIg9mnZT4xPfA+UPCUdnhRuEFDL+w==} + + '@esbuild-kit/core-utils@3.3.2': + resolution: {integrity: sha512-sPRAnw9CdSsRmEtnsl2WXWdyquogVpB3yZ3dgwJfe8zrOzTsV7cJvmwrKVa+0ma5BoiGJ+BoqkMvawbayKUsqQ==} + deprecated: 'Merged into tsx: https://tsx.is' + + '@esbuild-kit/esm-loader@2.6.5': + resolution: {integrity: sha512-FxEMIkJKnodyA1OaCUoEvbYRkoZlLZ4d/eXFu9Fh8CbBBgP5EmZxrfTRyN0qpXZ4vOvqnE5YdRdcrmUUXuU+dA==} + deprecated: 'Merged into tsx: https://tsx.is' + + '@esbuild/aix-ppc64@0.19.12': + resolution: {integrity: sha512-bmoCYyWdEL3wDQIVbcyzRyeKLgk2WtWLTWz1ZIAZF/EGbNOwSA6ew3PftJ1PqMiOOGu0OyFMzG53L0zqIpPeNA==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [aix] + + '@esbuild/android-arm64@0.18.20': + resolution: {integrity: sha512-Nz4rJcchGDtENV0eMKUNa6L12zz2zBDXuhj/Vjh18zGqB44Bi7MBMSXjgunJgjRhCmKOjnPuZp4Mb6OKqtMHLQ==} + engines: {node: '>=12'} + cpu: [arm64] + os: [android] + + '@esbuild/android-arm64@0.19.12': + resolution: {integrity: sha512-P0UVNGIienjZv3f5zq0DP3Nt2IE/3plFzuaS96vihvD0Hd6H/q4WXUGpCxD/E8YrSXfNyRPbpTq+T8ZQioSuPA==} + engines: {node: '>=12'} + cpu: [arm64] + os: [android] + + '@esbuild/android-arm@0.18.20': + resolution: {integrity: sha512-fyi7TDI/ijKKNZTUJAQqiG5T7YjJXgnzkURqmGj13C6dCqckZBLdl4h7bkhHt/t0WP+zO9/zwroDvANaOqO5Sw==} + engines: {node: '>=12'} + cpu: [arm] + os: [android] + + '@esbuild/android-arm@0.19.12': + resolution: {integrity: sha512-qg/Lj1mu3CdQlDEEiWrlC4eaPZ1KztwGJ9B6J+/6G+/4ewxJg7gqj8eVYWvao1bXrqGiW2rsBZFSX3q2lcW05w==} + engines: {node: '>=12'} + cpu: [arm] + os: [android] + + '@esbuild/android-x64@0.18.20': + resolution: {integrity: sha512-8GDdlePJA8D6zlZYJV/jnrRAi6rOiNaCC/JclcXpB+KIuvfBN4owLtgzY2bsxnx666XjJx2kDPUmnTtR8qKQUg==} + engines: {node: '>=12'} + cpu: [x64] + os: [android] + + '@esbuild/android-x64@0.19.12': + resolution: {integrity: sha512-3k7ZoUW6Q6YqhdhIaq/WZ7HwBpnFBlW905Fa4s4qWJyiNOgT1dOqDiVAQFwBH7gBRZr17gLrlFCRzF6jFh7Kew==} + engines: {node: '>=12'} + cpu: [x64] + os: [android] + + '@esbuild/darwin-arm64@0.18.20': + resolution: {integrity: sha512-bxRHW5kHU38zS2lPTPOyuyTm+S+eobPUnTNkdJEfAddYgEcll4xkT8DB9d2008DtTbl7uJag2HuE5NZAZgnNEA==} + engines: {node: '>=12'} + cpu: [arm64] + os: [darwin] + + '@esbuild/darwin-arm64@0.19.12': + resolution: {integrity: sha512-B6IeSgZgtEzGC42jsI+YYu9Z3HKRxp8ZT3cqhvliEHovq8HSX2YX8lNocDn79gCKJXOSaEot9MVYky7AKjCs8g==} + engines: {node: '>=12'} + cpu: [arm64] + os: [darwin] + + '@esbuild/darwin-x64@0.18.20': + resolution: {integrity: sha512-pc5gxlMDxzm513qPGbCbDukOdsGtKhfxD1zJKXjCCcU7ju50O7MeAZ8c4krSJcOIJGFR+qx21yMMVYwiQvyTyQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [darwin] + + '@esbuild/darwin-x64@0.19.12': + resolution: {integrity: sha512-hKoVkKzFiToTgn+41qGhsUJXFlIjxI/jSYeZf3ugemDYZldIXIxhvwN6erJGlX4t5h417iFuheZ7l+YVn05N3A==} + engines: {node: '>=12'} + cpu: [x64] + os: [darwin] + + '@esbuild/freebsd-arm64@0.18.20': + resolution: {integrity: sha512-yqDQHy4QHevpMAaxhhIwYPMv1NECwOvIpGCZkECn8w2WFHXjEwrBn3CeNIYsibZ/iZEUemj++M26W3cNR5h+Tw==} + engines: {node: '>=12'} + cpu: [arm64] + os: [freebsd] + + '@esbuild/freebsd-arm64@0.19.12': + resolution: {integrity: sha512-4aRvFIXmwAcDBw9AueDQ2YnGmz5L6obe5kmPT8Vd+/+x/JMVKCgdcRwH6APrbpNXsPz+K653Qg8HB/oXvXVukA==} + engines: {node: '>=12'} + cpu: [arm64] + os: [freebsd] + + '@esbuild/freebsd-x64@0.18.20': + resolution: {integrity: sha512-tgWRPPuQsd3RmBZwarGVHZQvtzfEBOreNuxEMKFcd5DaDn2PbBxfwLcj4+aenoh7ctXcbXmOQIn8HI6mCSw5MQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [freebsd] + + '@esbuild/freebsd-x64@0.19.12': + resolution: {integrity: sha512-EYoXZ4d8xtBoVN7CEwWY2IN4ho76xjYXqSXMNccFSx2lgqOG/1TBPW0yPx1bJZk94qu3tX0fycJeeQsKovA8gg==} + engines: {node: '>=12'} + cpu: [x64] + os: [freebsd] + + '@esbuild/linux-arm64@0.18.20': + resolution: {integrity: sha512-2YbscF+UL7SQAVIpnWvYwM+3LskyDmPhe31pE7/aoTMFKKzIc9lLbyGUpmmb8a8AixOL61sQ/mFh3jEjHYFvdA==} + engines: {node: '>=12'} + cpu: [arm64] + os: [linux] + + '@esbuild/linux-arm64@0.19.12': + resolution: {integrity: sha512-EoTjyYyLuVPfdPLsGVVVC8a0p1BFFvtpQDB/YLEhaXyf/5bczaGeN15QkR+O4S5LeJ92Tqotve7i1jn35qwvdA==} + engines: {node: '>=12'} + cpu: [arm64] + os: [linux] + + '@esbuild/linux-arm@0.18.20': + resolution: {integrity: sha512-/5bHkMWnq1EgKr1V+Ybz3s1hWXok7mDFUMQ4cG10AfW3wL02PSZi5kFpYKrptDsgb2WAJIvRcDm+qIvXf/apvg==} + engines: {node: '>=12'} + cpu: [arm] + os: [linux] + + '@esbuild/linux-arm@0.19.12': + resolution: {integrity: sha512-J5jPms//KhSNv+LO1S1TX1UWp1ucM6N6XuL6ITdKWElCu8wXP72l9MM0zDTzzeikVyqFE6U8YAV9/tFyj0ti+w==} + engines: {node: '>=12'} + cpu: [arm] + os: [linux] + + '@esbuild/linux-ia32@0.18.20': + resolution: {integrity: sha512-P4etWwq6IsReT0E1KHU40bOnzMHoH73aXp96Fs8TIT6z9Hu8G6+0SHSw9i2isWrD2nbx2qo5yUqACgdfVGx7TA==} + engines: {node: '>=12'} + cpu: [ia32] + os: [linux] + + '@esbuild/linux-ia32@0.19.12': + resolution: {integrity: sha512-Thsa42rrP1+UIGaWz47uydHSBOgTUnwBwNq59khgIwktK6x60Hivfbux9iNR0eHCHzOLjLMLfUMLCypBkZXMHA==} + engines: {node: '>=12'} + cpu: [ia32] + os: [linux] + + '@esbuild/linux-loong64@0.18.20': + resolution: {integrity: sha512-nXW8nqBTrOpDLPgPY9uV+/1DjxoQ7DoB2N8eocyq8I9XuqJ7BiAMDMf9n1xZM9TgW0J8zrquIb/A7s3BJv7rjg==} + engines: {node: '>=12'} + cpu: [loong64] + os: [linux] + + '@esbuild/linux-loong64@0.19.12': + resolution: {integrity: sha512-LiXdXA0s3IqRRjm6rV6XaWATScKAXjI4R4LoDlvO7+yQqFdlr1Bax62sRwkVvRIrwXxvtYEHHI4dm50jAXkuAA==} + engines: {node: '>=12'} + cpu: [loong64] + os: [linux] + + '@esbuild/linux-mips64el@0.18.20': + resolution: {integrity: sha512-d5NeaXZcHp8PzYy5VnXV3VSd2D328Zb+9dEq5HE6bw6+N86JVPExrA6O68OPwobntbNJ0pzCpUFZTo3w0GyetQ==} + engines: {node: '>=12'} + cpu: [mips64el] + os: [linux] + + '@esbuild/linux-mips64el@0.19.12': + resolution: {integrity: sha512-fEnAuj5VGTanfJ07ff0gOA6IPsvrVHLVb6Lyd1g2/ed67oU1eFzL0r9WL7ZzscD+/N6i3dWumGE1Un4f7Amf+w==} + engines: {node: '>=12'} + cpu: [mips64el] + os: [linux] + + '@esbuild/linux-ppc64@0.18.20': + resolution: {integrity: sha512-WHPyeScRNcmANnLQkq6AfyXRFr5D6N2sKgkFo2FqguP44Nw2eyDlbTdZwd9GYk98DZG9QItIiTlFLHJHjxP3FA==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [linux] + + '@esbuild/linux-ppc64@0.19.12': + resolution: {integrity: sha512-nYJA2/QPimDQOh1rKWedNOe3Gfc8PabU7HT3iXWtNUbRzXS9+vgB0Fjaqr//XNbd82mCxHzik2qotuI89cfixg==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [linux] + + '@esbuild/linux-riscv64@0.18.20': + resolution: {integrity: sha512-WSxo6h5ecI5XH34KC7w5veNnKkju3zBRLEQNY7mv5mtBmrP/MjNBCAlsM2u5hDBlS3NGcTQpoBvRzqBcRtpq1A==} + engines: {node: '>=12'} + cpu: [riscv64] + os: [linux] + + '@esbuild/linux-riscv64@0.19.12': + resolution: {integrity: sha512-2MueBrlPQCw5dVJJpQdUYgeqIzDQgw3QtiAHUC4RBz9FXPrskyyU3VI1hw7C0BSKB9OduwSJ79FTCqtGMWqJHg==} + engines: {node: '>=12'} + cpu: [riscv64] + os: [linux] + + '@esbuild/linux-s390x@0.18.20': + resolution: {integrity: sha512-+8231GMs3mAEth6Ja1iK0a1sQ3ohfcpzpRLH8uuc5/KVDFneH6jtAJLFGafpzpMRO6DzJ6AvXKze9LfFMrIHVQ==} + engines: {node: '>=12'} + cpu: [s390x] + os: [linux] + + '@esbuild/linux-s390x@0.19.12': + resolution: {integrity: sha512-+Pil1Nv3Umes4m3AZKqA2anfhJiVmNCYkPchwFJNEJN5QxmTs1uzyy4TvmDrCRNT2ApwSari7ZIgrPeUx4UZDg==} + engines: {node: '>=12'} + cpu: [s390x] + os: [linux] + + '@esbuild/linux-x64@0.18.20': + resolution: {integrity: sha512-UYqiqemphJcNsFEskc73jQ7B9jgwjWrSayxawS6UVFZGWrAAtkzjxSqnoclCXxWtfwLdzU+vTpcNYhpn43uP1w==} + engines: {node: '>=12'} + cpu: [x64] + os: [linux] + + '@esbuild/linux-x64@0.19.12': + resolution: {integrity: sha512-B71g1QpxfwBvNrfyJdVDexenDIt1CiDN1TIXLbhOw0KhJzE78KIFGX6OJ9MrtC0oOqMWf+0xop4qEU8JrJTwCg==} + engines: {node: '>=12'} + cpu: [x64] + os: [linux] + + '@esbuild/netbsd-x64@0.18.20': + resolution: {integrity: sha512-iO1c++VP6xUBUmltHZoMtCUdPlnPGdBom6IrO4gyKPFFVBKioIImVooR5I83nTew5UOYrk3gIJhbZh8X44y06A==} + engines: {node: '>=12'} + cpu: [x64] + os: [netbsd] + + '@esbuild/netbsd-x64@0.19.12': + resolution: {integrity: sha512-3ltjQ7n1owJgFbuC61Oj++XhtzmymoCihNFgT84UAmJnxJfm4sYCiSLTXZtE00VWYpPMYc+ZQmB6xbSdVh0JWA==} + engines: {node: '>=12'} + cpu: [x64] + os: [netbsd] + + '@esbuild/openbsd-x64@0.18.20': + resolution: {integrity: sha512-e5e4YSsuQfX4cxcygw/UCPIEP6wbIL+se3sxPdCiMbFLBWu0eiZOJ7WoD+ptCLrmjZBK1Wk7I6D/I3NglUGOxg==} + engines: {node: '>=12'} + cpu: [x64] + os: [openbsd] + + '@esbuild/openbsd-x64@0.19.12': + resolution: {integrity: sha512-RbrfTB9SWsr0kWmb9srfF+L933uMDdu9BIzdA7os2t0TXhCRjrQyCeOt6wVxr79CKD4c+p+YhCj31HBkYcXebw==} + engines: {node: '>=12'} + cpu: [x64] + os: [openbsd] + + '@esbuild/sunos-x64@0.18.20': + resolution: {integrity: sha512-kDbFRFp0YpTQVVrqUd5FTYmWo45zGaXe0X8E1G/LKFC0v8x0vWrhOWSLITcCn63lmZIxfOMXtCfti/RxN/0wnQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [sunos] + + '@esbuild/sunos-x64@0.19.12': + resolution: {integrity: sha512-HKjJwRrW8uWtCQnQOz9qcU3mUZhTUQvi56Q8DPTLLB+DawoiQdjsYq+j+D3s9I8VFtDr+F9CjgXKKC4ss89IeA==} + engines: {node: '>=12'} + cpu: [x64] + os: [sunos] + + '@esbuild/win32-arm64@0.18.20': + resolution: {integrity: sha512-ddYFR6ItYgoaq4v4JmQQaAI5s7npztfV4Ag6NrhiaW0RrnOXqBkgwZLofVTlq1daVTQNhtI5oieTvkRPfZrePg==} + engines: {node: '>=12'} + cpu: [arm64] + os: [win32] + + '@esbuild/win32-arm64@0.19.12': + resolution: {integrity: sha512-URgtR1dJnmGvX864pn1B2YUYNzjmXkuJOIqG2HdU62MVS4EHpU2946OZoTMnRUHklGtJdJZ33QfzdjGACXhn1A==} + engines: {node: '>=12'} + cpu: [arm64] + os: [win32] + + '@esbuild/win32-ia32@0.18.20': + resolution: {integrity: sha512-Wv7QBi3ID/rROT08SABTS7eV4hX26sVduqDOTe1MvGMjNd3EjOz4b7zeexIR62GTIEKrfJXKL9LFxTYgkyeu7g==} + engines: {node: '>=12'} + cpu: [ia32] + os: [win32] + + '@esbuild/win32-ia32@0.19.12': + resolution: {integrity: sha512-+ZOE6pUkMOJfmxmBZElNOx72NKpIa/HFOMGzu8fqzQJ5kgf6aTGrcJaFsNiVMH4JKpMipyK+7k0n2UXN7a8YKQ==} + engines: {node: '>=12'} + cpu: [ia32] + os: [win32] + + '@esbuild/win32-x64@0.18.20': + resolution: {integrity: sha512-kTdfRcSiDfQca/y9QIkng02avJ+NCaQvrMejlsB3RRv5sE9rRoeBPISaZpKxHELzRxZyLvNts1P27W3wV+8geQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [win32] + + '@esbuild/win32-x64@0.19.12': + resolution: {integrity: sha512-T1QyPSDCyMXaO3pzBkF96E8xMkiRYbUEZADd29SyPGabqxMViNoii+NcK7eWJAEoU6RZyEm5lVSIjTmcdoB9HA==} + engines: {node: '>=12'} + cpu: [x64] + os: [win32] + + '@faker-js/faker@9.2.0': + resolution: {integrity: sha512-ulqQu4KMr1/sTFIYvqSdegHT8NIkt66tFAkugGnHA+1WAfEn6hMzNR+svjXGFRVLnapxvej67Z/LwchFrnLBUg==} + engines: {node: '>=18.0.0', npm: '>=9.0.0'} + + '@floating-ui/core@1.6.1': + resolution: {integrity: sha512-42UH54oPZHPdRHdw6BgoBD6cg/eVTmVrFcgeRDM3jbO7uxSoipVcmcIGFcA5jmOHO5apcyvBhkSKES3fQJnu7A==} + + '@floating-ui/dom@1.6.5': + resolution: {integrity: sha512-Nsdud2X65Dz+1RHjAIP0t8z5e2ff/IRbei6BqFrl1urT8sDVzM1HMQ+R0XcU5ceRfyO3I6ayeqIfh+6Wb8LGTw==} + + '@floating-ui/react-dom@1.3.0': + resolution: {integrity: sha512-htwHm67Ji5E/pROEAr7f8IKFShuiCKHwUC/UY4vC3I5jiSvGFAYnSYiZO5MlGmads+QqvUkR9ANHEguGrDv72g==} + peerDependencies: + react: '>=16.8.0' + react-dom: '>=16.8.0' + + '@floating-ui/react-dom@2.0.9': + resolution: {integrity: sha512-q0umO0+LQK4+p6aGyvzASqKbKOJcAHJ7ycE9CuUvfx3s9zTHWmGJTPOIlM/hmSBfUfg/XfY5YhLBLR/LHwShQQ==} + peerDependencies: + react: '>=16.8.0' + react-dom: '>=16.8.0' + + '@floating-ui/react-dom@2.1.2': + resolution: {integrity: sha512-06okr5cgPzMNBy+Ycse2A6udMi4bqwW/zgBF/rwjcNqWkyr82Mcg8b0vjX8OJpZFy/FKjJmw6wV7t44kK6kW7A==} + peerDependencies: + react: '>=16.8.0' + react-dom: '>=16.8.0' + + '@floating-ui/react@0.19.2': + resolution: {integrity: sha512-JyNk4A0Ezirq8FlXECvRtQOX/iBe5Ize0W/pLkrZjfHW9GUV7Xnq6zm6fyZuQzaHHqEnVizmvlA96e1/CkZv+w==} + peerDependencies: + react: '>=16.8.0' + react-dom: '>=16.8.0' + + '@floating-ui/react@0.26.24': + resolution: {integrity: sha512-2ly0pCkZIGEQUq5H8bBK0XJmc1xIK/RM3tvVzY3GBER7IOD1UgmC2Y2tjj4AuS+TC+vTE1KJv2053290jua0Sw==} + peerDependencies: + react: '>=16.8.0' + react-dom: '>=16.8.0' + + '@floating-ui/utils@0.2.8': + resolution: {integrity: sha512-kym7SodPp8/wloecOpcmSnWJsK7M0E5Wg8UcFA+uO4B9s5d0ywXOEro/8HM9x0rW+TljRzul/14UYz3TleT3ig==} + + '@gar/promisify@1.1.3': + resolution: {integrity: sha512-k2Ty1JcVojjJFwrg/ThKi2ujJ7XNLYaFGNB/bWT9wGR+oSMJHMa5w+CUq6p/pVrKeNNgA7pCqEcjSnHVoqJQFw==} + + '@headlessui/react@1.7.19': + resolution: {integrity: sha512-Ll+8q3OlMJfJbAKM/+/Y2q6PPYbryqNTXDbryx7SXLIDamkF6iQFbriYHga0dY44PvDhvvBWCx1Xj4U5+G4hOw==} + engines: {node: '>=10'} + peerDependencies: + react: ^16 || ^17 || ^18 + react-dom: ^16 || ^17 || ^18 + + '@headlessui/react@2.1.10': + resolution: {integrity: sha512-6mLa2fjMDAFQi+/R10B+zU3edsUk/MDtENB2zHho0lqKU1uzhAfJLUduWds4nCo8wbl3vULtC5rJfZAQ1yqIng==} + engines: {node: '>=10'} + peerDependencies: + react: ^18 + react-dom: ^18 + + '@headlessui/tailwindcss@0.2.1': + resolution: {integrity: sha512-2+5+NZ+RzMyrVeCZOxdbvkUSssSxGvcUxphkIfSVLpRiKsj+/63T2TOL9dBYMXVfj/CGr6hMxSRInzXv6YY7sA==} + engines: {node: '>=10'} + peerDependencies: + tailwindcss: ^3.0 + + '@hookform/resolvers@3.9.0': + resolution: {integrity: sha512-bU0Gr4EepJ/EQsH/IwEzYLsT/PEj5C0ynLQ4m+GSHS+xKH4TfSelhluTgOaoc4kA5s7eCsQbM4wvZLzELmWzUg==} + peerDependencies: + react-hook-form: ^7.0.0 + + '@isaacs/cliui@8.0.2': + resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} + engines: {node: '>=12'} + + '@jridgewell/gen-mapping@0.3.5': + resolution: {integrity: sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==} + engines: {node: '>=6.0.0'} + + '@jridgewell/resolve-uri@3.1.2': + resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} + engines: {node: '>=6.0.0'} + + '@jridgewell/set-array@1.2.1': + resolution: {integrity: sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==} + engines: {node: '>=6.0.0'} + + '@jridgewell/sourcemap-codec@1.4.15': + resolution: {integrity: sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==} + + '@jridgewell/trace-mapping@0.3.25': + resolution: {integrity: sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==} + + '@jridgewell/trace-mapping@0.3.9': + resolution: {integrity: sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==} + + '@libsql/client@0.14.0': + resolution: {integrity: sha512-/9HEKfn6fwXB5aTEEoMeFh4CtG0ZzbncBb1e++OCdVpgKZ/xyMsIVYXm0w7Pv4RUel803vE6LwniB3PqD72R0Q==} + + '@libsql/core@0.14.0': + resolution: {integrity: sha512-nhbuXf7GP3PSZgdCY2Ecj8vz187ptHlZQ0VRc751oB2C1W8jQUXKKklvt7t1LJiUTQBVJuadF628eUk+3cRi4Q==} + + '@libsql/darwin-arm64@0.4.6': + resolution: {integrity: sha512-45i604CJ2Lubbg7NqtDodjarF6VgST8rS5R8xB++MoRqixtDns9PZ6tocT9pRJDWuTWEiy2sjthPOFWMKwYAsg==} + cpu: [arm64] + os: [darwin] + + '@libsql/darwin-x64@0.4.6': + resolution: {integrity: sha512-dRKliflhfr5zOPSNgNJ6C2nZDd4YA8bTXF3MUNqNkcxQ8BffaH9uUwL9kMq89LkFIZQHcyP75bBgZctxfJ/H5Q==} + cpu: [x64] + os: [darwin] + + '@libsql/hrana-client@0.7.0': + resolution: {integrity: sha512-OF8fFQSkbL7vJY9rfuegK1R7sPgQ6kFMkDamiEccNUvieQ+3urzfDFI616oPl8V7T9zRmnTkSjMOImYCAVRVuw==} + + '@libsql/isomorphic-fetch@0.3.1': + resolution: {integrity: sha512-6kK3SUK5Uu56zPq/Las620n5aS9xJq+jMBcNSOmjhNf/MUvdyji4vrMTqD7ptY7/4/CAVEAYDeotUz60LNQHtw==} + engines: {node: '>=18.0.0'} + + '@libsql/isomorphic-ws@0.1.5': + resolution: {integrity: sha512-DtLWIH29onUYR00i0GlQ3UdcTRC6EP4u9w/h9LxpUZJWRMARk6dQwZ6Jkd+QdwVpuAOrdxt18v0K2uIYR3fwFg==} + + '@libsql/linux-arm64-gnu@0.4.6': + resolution: {integrity: sha512-DMPavVyY6vYPAYcQR1iOotHszg+5xSjHSg6F9kNecPX0KKdGq84zuPJmORfKOPtaWvzPewNFdML/e+s1fu09XQ==} + cpu: [arm64] + os: [linux] + + '@libsql/linux-arm64-musl@0.4.6': + resolution: {integrity: sha512-whuHSYAZyclGjM3L0mKGXyWqdAy7qYvPPn+J1ve7FtGkFlM0DiIPjA5K30aWSGJSRh72sD9DBZfnu8CpfSjT6w==} + cpu: [arm64] + os: [linux] + + '@libsql/linux-x64-gnu@0.4.6': + resolution: {integrity: sha512-0ggx+5RwEbYabIlDBBAvavdfIJCZ757u6nDZtBeQIhzW99EKbWG3lvkXHM3qudFb/pDWSUY4RFBm6vVtF1cJGA==} + cpu: [x64] + os: [linux] + + '@libsql/linux-x64-musl@0.4.6': + resolution: {integrity: sha512-SWNrv7Hz72QWlbM/ZsbL35MPopZavqCUmQz2HNDZ55t0F+kt8pXuP+bbI2KvmaQ7wdsoqAA4qBmjol0+bh4ndw==} + cpu: [x64] + os: [linux] + + '@libsql/win32-x64-msvc@0.4.6': + resolution: {integrity: sha512-Q0axn110zDNELfkEog3Nl8p9BU4eI/UvgaHevGyOiSDN7s0KPfj0j6jwVHk4oz3o/d/Gg3DRIxomZ4ftfTOy/g==} + cpu: [x64] + os: [win32] + + '@lucia-auth/adapter-drizzle@1.1.0': + resolution: {integrity: sha512-iCTnZWvfI5lLZOdUHZYiXA1jaspIFEeo2extLxQ3DjP3uOVys7IPwBi7zezLIRu9dhro4H4Kji+7gSYyjcef2A==} + peerDependencies: + drizzle-orm: '>= 0.29 <1' + lucia: 3.x + + '@neon-rs/load@0.0.4': + resolution: {integrity: sha512-kTPhdZyTQxB+2wpiRcFWrDcejc4JI6tkPuS7UZCG4l6Zvc5kU/gGQ/ozvHTh1XR5tS+UlfAfGuPajjzQjCiHCw==} + + '@next/env@14.2.3': + resolution: {integrity: sha512-W7fd7IbkfmeeY2gXrzJYDx8D2lWKbVoTIj1o1ScPHNzvp30s1AuoEFSdr39bC5sjxJaxTtq3OTCZboNp0lNWHA==} + + '@next/swc-darwin-arm64@14.2.3': + resolution: {integrity: sha512-3pEYo/RaGqPP0YzwnlmPN2puaF2WMLM3apt5jLW2fFdXD9+pqcoTzRk+iZsf8ta7+quAe4Q6Ms0nR0SFGFdS1A==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [darwin] + + '@next/swc-darwin-x64@14.2.3': + resolution: {integrity: sha512-6adp7waE6P1TYFSXpY366xwsOnEXM+y1kgRpjSRVI2CBDOcbRjsJ67Z6EgKIqWIue52d2q/Mx8g9MszARj8IEA==} + engines: {node: '>= 10'} + cpu: [x64] + os: [darwin] + + '@next/swc-linux-arm64-gnu@14.2.3': + resolution: {integrity: sha512-cuzCE/1G0ZSnTAHJPUT1rPgQx1w5tzSX7POXSLaS7w2nIUJUD+e25QoXD/hMfxbsT9rslEXugWypJMILBj/QsA==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [linux] + + '@next/swc-linux-arm64-musl@14.2.3': + resolution: {integrity: sha512-0D4/oMM2Y9Ta3nGuCcQN8jjJjmDPYpHX9OJzqk42NZGJocU2MqhBq5tWkJrUQOQY9N+In9xOdymzapM09GeiZw==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [linux] + + '@next/swc-linux-x64-gnu@14.2.3': + resolution: {integrity: sha512-ENPiNnBNDInBLyUU5ii8PMQh+4XLr4pG51tOp6aJ9xqFQ2iRI6IH0Ds2yJkAzNV1CfyagcyzPfROMViS2wOZ9w==} + engines: {node: '>= 10'} + cpu: [x64] + os: [linux] + + '@next/swc-linux-x64-musl@14.2.3': + resolution: {integrity: sha512-BTAbq0LnCbF5MtoM7I/9UeUu/8ZBY0i8SFjUMCbPDOLv+un67e2JgyN4pmgfXBwy/I+RHu8q+k+MCkDN6P9ViQ==} + engines: {node: '>= 10'} + cpu: [x64] + os: [linux] + + '@next/swc-win32-arm64-msvc@14.2.3': + resolution: {integrity: sha512-AEHIw/dhAMLNFJFJIJIyOFDzrzI5bAjI9J26gbO5xhAKHYTZ9Or04BesFPXiAYXDNdrwTP2dQceYA4dL1geu8A==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [win32] + + '@next/swc-win32-ia32-msvc@14.2.3': + resolution: {integrity: sha512-vga40n1q6aYb0CLrM+eEmisfKCR45ixQYXuBXxOOmmoV8sYST9k7E3US32FsY+CkkF7NtzdcebiFT4CHuMSyZw==} + engines: {node: '>= 10'} + cpu: [ia32] + os: [win32] + + '@next/swc-win32-x64-msvc@14.2.3': + resolution: {integrity: sha512-Q1/zm43RWynxrO7lW4ehciQVj+5ePBhOK+/K2P7pLFX3JaJ/IZVC69SHidrmZSOkqz7ECIOhhy7XhAFG4JYyHA==} + engines: {node: '>= 10'} + cpu: [x64] + os: [win32] + + '@node-rs/argon2-android-arm-eabi@1.7.0': + resolution: {integrity: sha512-udDqkr5P9E+wYX1SZwAVPdyfYvaF4ry9Tm+R9LkfSHbzWH0uhU6zjIwNRp7m+n4gx691rk+lqqDAIP8RLKwbhg==} + engines: {node: '>= 10'} + cpu: [arm] + os: [android] + + '@node-rs/argon2-android-arm64@1.7.0': + resolution: {integrity: sha512-s9j/G30xKUx8WU50WIhF0fIl1EdhBGq0RQ06lEhZ0Gi0ap8lhqbE2Bn5h3/G2D1k0Dx+yjeVVNmt/xOQIRG38A==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [android] + + '@node-rs/argon2-darwin-arm64@1.7.0': + resolution: {integrity: sha512-ZIz4L6HGOB9U1kW23g+m7anGNuTZ0RuTw0vNp3o+2DWpb8u8rODq6A8tH4JRL79S+Co/Nq608m9uackN2pe0Rw==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [darwin] + + '@node-rs/argon2-darwin-x64@1.7.0': + resolution: {integrity: sha512-5oi/pxqVhODW/pj1+3zElMTn/YukQeywPHHYDbcAW3KsojFjKySfhcJMd1DjKTc+CHQI+4lOxZzSUzK7mI14Hw==} + engines: {node: '>= 10'} + cpu: [x64] + os: [darwin] + + '@node-rs/argon2-freebsd-x64@1.7.0': + resolution: {integrity: sha512-Ify08683hA4QVXYoIm5SUWOY5DPIT/CMB0CQT+IdxQAg/F+qp342+lUkeAtD5bvStQuCx/dFO3bnnzoe2clMhA==} + engines: {node: '>= 10'} + cpu: [x64] + os: [freebsd] + + '@node-rs/argon2-linux-arm-gnueabihf@1.7.0': + resolution: {integrity: sha512-7DjDZ1h5AUHAtRNjD19RnQatbhL+uuxBASuuXIBu4/w6Dx8n7YPxwTP4MXfsvuRgKuMWiOb/Ub/HJ3kXVCXRkg==} + engines: {node: '>= 10'} + cpu: [arm] + os: [linux] + + '@node-rs/argon2-linux-arm64-gnu@1.7.0': + resolution: {integrity: sha512-nJDoMP4Y3YcqGswE4DvP080w6O24RmnFEDnL0emdI8Nou17kNYBzP2546Nasx9GCyLzRcYQwZOUjrtUuQ+od2g==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [linux] + + '@node-rs/argon2-linux-arm64-musl@1.7.0': + resolution: {integrity: sha512-BKWS8iVconhE3jrb9mj6t1J9vwUqQPpzCbUKxfTGJfc+kNL58F1SXHBoe2cDYGnHrFEHTY0YochzXoAfm4Dm/A==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [linux] + + '@node-rs/argon2-linux-x64-gnu@1.7.0': + resolution: {integrity: sha512-EmgqZOlf4Jurk/szW1iTsVISx25bKksVC5uttJDUloTgsAgIGReCpUUO1R24pBhu9ESJa47iv8NSf3yAfGv6jQ==} + engines: {node: '>= 10'} + cpu: [x64] + os: [linux] + + '@node-rs/argon2-linux-x64-musl@1.7.0': + resolution: {integrity: sha512-/o1efYCYIxjfuoRYyBTi2Iy+1iFfhqHCvvVsnjNSgO1xWiWrX0Rrt/xXW5Zsl7vS2Y+yu8PL8KFWRzZhaVxfKA==} + engines: {node: '>= 10'} + cpu: [x64] + os: [linux] + + '@node-rs/argon2-wasm32-wasi@1.7.0': + resolution: {integrity: sha512-Evmk9VcxqnuwQftfAfYEr6YZYSPLzmKUsbFIMep5nTt9PT4XYRFAERj7wNYp+rOcBenF3X4xoB+LhwcOMTNE5w==} + engines: {node: '>=14.0.0'} + cpu: [wasm32] + + '@node-rs/argon2-win32-arm64-msvc@1.7.0': + resolution: {integrity: sha512-qgsU7T004COWWpSA0tppDqDxbPLgg8FaU09krIJ7FBl71Sz8SFO40h7fDIjfbTT5w7u6mcaINMQ5bSHu75PCaA==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [win32] + + '@node-rs/argon2-win32-ia32-msvc@1.7.0': + resolution: {integrity: sha512-JGafwWYQ/HpZ3XSwP4adQ6W41pRvhcdXvpzIWtKvX+17+xEXAe2nmGWM6s27pVkg1iV2ZtoYLRDkOUoGqZkCcg==} + engines: {node: '>= 10'} + cpu: [ia32] + os: [win32] + + '@node-rs/argon2-win32-x64-msvc@1.7.0': + resolution: {integrity: sha512-9oq4ShyFakw8AG3mRls0AoCpxBFcimYx7+jvXeAf2OqKNO+mSA6eZ9z7KQeVCi0+SOEUYxMGf5UiGiDb9R6+9Q==} + engines: {node: '>= 10'} + cpu: [x64] + os: [win32] + + '@node-rs/argon2@1.7.0': + resolution: {integrity: sha512-zfULc+/tmcWcxn+nHkbyY8vP3+MpEqKORbszt4UkpqZgBgDAAIYvuDN/zukfTgdmo6tmJKKVfzigZOPk4LlIog==} + engines: {node: '>= 10'} + + '@node-rs/bcrypt-android-arm-eabi@1.9.0': + resolution: {integrity: sha512-nOCFISGtnodGHNiLrG0WYLWr81qQzZKYfmwHc7muUeq+KY0sQXyHOwZk9OuNQAWv/lnntmtbwkwT0QNEmOyLvA==} + engines: {node: '>= 10'} + cpu: [arm] + os: [android] + + '@node-rs/bcrypt-android-arm64@1.9.0': + resolution: {integrity: sha512-+ZrIAtigVmjYkqZQTThHVlz0+TG6D+GDHWhVKvR2DifjtqJ0i+mb9gjo++hN+fWEQdWNGxKCiBBjwgT4EcXd6A==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [android] + + '@node-rs/bcrypt-darwin-arm64@1.9.0': + resolution: {integrity: sha512-CQiS+F9Pa0XozvkXR1g7uXE9QvBOPOplDg0iCCPRYTN9PqA5qYxhwe48G3o+v2UeQceNRrbnEtWuANm7JRqIhw==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [darwin] + + '@node-rs/bcrypt-darwin-x64@1.9.0': + resolution: {integrity: sha512-4pTKGawYd7sNEjdJ7R/R67uwQH1VvwPZ0SSUMmeNHbxD5QlwAPXdDH11q22uzVXsvNFZ6nGQBg8No5OUGpx6Ug==} + engines: {node: '>= 10'} + cpu: [x64] + os: [darwin] + + '@node-rs/bcrypt-freebsd-x64@1.9.0': + resolution: {integrity: sha512-UmWzySX4BJhT/B8xmTru6iFif3h0Rpx3TqxRLCcbgmH43r7k5/9QuhpiyzpvKGpKHJCFNm4F3rC2wghvw5FCIg==} + engines: {node: '>= 10'} + cpu: [x64] + os: [freebsd] + + '@node-rs/bcrypt-linux-arm-gnueabihf@1.9.0': + resolution: {integrity: sha512-8qoX4PgBND2cVwsbajoAWo3NwdfJPEXgpCsZQZURz42oMjbGyhhSYbovBCskGU3EBLoC8RA2B1jFWooeYVn5BA==} + engines: {node: '>= 10'} + cpu: [arm] + os: [linux] + + '@node-rs/bcrypt-linux-arm64-gnu@1.9.0': + resolution: {integrity: sha512-TuAC6kx0SbcIA4mSEWPi+OCcDjTQUMl213v5gMNlttF+D4ieIZx6pPDGTaMO6M2PDHTeCG0CBzZl0Lu+9b0c7Q==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [linux] + + '@node-rs/bcrypt-linux-arm64-musl@1.9.0': + resolution: {integrity: sha512-/sIvKDABOI8QOEnLD7hIj02BVaNOuCIWBKvxcJOt8+TuwJ6zmY1UI5kSv9d99WbiHjTp97wtAUbZQwauU4b9ew==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [linux] + + '@node-rs/bcrypt-linux-x64-gnu@1.9.0': + resolution: {integrity: sha512-DyyhDHDsLBsCKz1tZ1hLvUZSc1DK0FU0v52jK6IBQxrj24WscSU9zZe7ie/V9kdmA4Ep57BfpWX8Dsa2JxGdgQ==} + engines: {node: '>= 10'} + cpu: [x64] + os: [linux] + + '@node-rs/bcrypt-linux-x64-musl@1.9.0': + resolution: {integrity: sha512-duIiuqQ+Lew8ASSAYm6ZRqcmfBGWwsi81XLUwz86a2HR7Qv6V4yc3ZAUQovAikhjCsIqe8C11JlAZSK6+PlXYg==} + engines: {node: '>= 10'} + cpu: [x64] + os: [linux] + + '@node-rs/bcrypt-wasm32-wasi@1.9.0': + resolution: {integrity: sha512-ylaGmn9Wjwv/D5lxtawttx3H6Uu2WTTR7lWlRHGT6Ga/MB1Vj4OjSGUW8G8zIVnKuXpGbZ92pgHlt4HUpSLctw==} + engines: {node: '>=14.0.0'} + cpu: [wasm32] + + '@node-rs/bcrypt-win32-arm64-msvc@1.9.0': + resolution: {integrity: sha512-2h86gF7QFyEzODuDFml/Dp1MSJoZjxJ4yyT2Erf4NkwsiA5MqowUhUsorRwZhX6+2CtlGa7orbwi13AKMsYndw==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [win32] + + '@node-rs/bcrypt-win32-ia32-msvc@1.9.0': + resolution: {integrity: sha512-kqxalCvhs4FkN0+gWWfa4Bdy2NQAkfiqq/CEf6mNXC13RSV673Ev9V8sRlQyNpCHCNkeXfOT9pgoBdJmMs9muA==} + engines: {node: '>= 10'} + cpu: [ia32] + os: [win32] + + '@node-rs/bcrypt-win32-x64-msvc@1.9.0': + resolution: {integrity: sha512-2y0Tuo6ZAT2Cz8V7DHulSlv1Bip3zbzeXyeur+uR25IRNYXKvI/P99Zl85Fbuu/zzYAZRLLlGTRe6/9IHofe/w==} + engines: {node: '>= 10'} + cpu: [x64] + os: [win32] + + '@node-rs/bcrypt@1.9.0': + resolution: {integrity: sha512-u2OlIxW264bFUfvbFqDz9HZKFjwe8FHFtn7T/U8mYjPZ7DWYpbUB+/dkW/QgYfMSfR0ejkyuWaBBe0coW7/7ig==} + engines: {node: '>= 10'} + + '@nodelib/fs.scandir@2.1.5': + resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} + engines: {node: '>= 8'} + + '@nodelib/fs.stat@2.0.5': + resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==} + engines: {node: '>= 8'} + + '@nodelib/fs.walk@1.2.8': + resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} + engines: {node: '>= 8'} + + '@npmcli/fs@1.1.1': + resolution: {integrity: sha512-8KG5RD0GVP4ydEzRn/I4BNDuxDtqVbOdm8675T49OIG/NGhaK0pjPX7ZcDlvKYbA+ulvVK3ztfcF4uBdOxuJbQ==} + + '@npmcli/move-file@1.1.2': + resolution: {integrity: sha512-1SUf/Cg2GzGDyaf15aR9St9TWlb+XvbZXWpDx8YKs7MLzMH/BCeopv+y9vzrzgkfykCGuWOlSu3mZhj2+FQcrg==} + engines: {node: '>=10'} + deprecated: This functionality has been moved to @npmcli/fs + + '@oslojs/asn1@1.0.0': + resolution: {integrity: sha512-zw/wn0sj0j0QKbIXfIlnEcTviaCzYOY3V5rAyjR6YtOByFtJiT574+8p9Wlach0lZH9fddD4yb9laEAIl4vXQA==} + + '@oslojs/binary@1.0.0': + resolution: {integrity: sha512-9RCU6OwXU6p67H4NODbuxv2S3eenuQ4/WFLrsq+K/k682xrznH5EVWA7N4VFk9VYVcbFtKqur5YQQZc0ySGhsQ==} + + '@oslojs/crypto@1.0.1': + resolution: {integrity: sha512-7n08G8nWjAr/Yu3vu9zzrd0L9XnrJfpMioQcvCMxBIiF5orECHe5/3J0jmXRVvgfqMm/+4oxlQ+Sq39COYLcNQ==} + + '@oslojs/encoding@1.1.0': + resolution: {integrity: sha512-70wQhgYmndg4GCPxPPxPGevRKqTIJ2Nh4OkiMWmDAVYsTQ+Ta7Sq+rPevXyXGdzr30/qZBnyOalCszoMxlyldQ==} + + '@pkgjs/parseargs@0.11.0': + resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==} + engines: {node: '>=14'} + + '@radix-ui/number@1.1.0': + resolution: {integrity: sha512-V3gRzhVNU1ldS5XhAPTom1fOIo4ccrjjJgmE+LI2h/WaFpHmx0MQApT+KZHnx8abG6Avtfcz4WoEciMnpFT3HQ==} + + '@radix-ui/primitive@1.1.0': + resolution: {integrity: sha512-4Z8dn6Upk0qk4P74xBhZ6Hd/w0mPEzOOLxy4xiPXOXqjF7jZS0VAKk7/x/H6FyY2zCkYJqePf1G5KmkmNJ4RBA==} + + '@radix-ui/react-alert-dialog@1.1.2': + resolution: {integrity: sha512-eGSlLzPhKO+TErxkiGcCZGuvbVMnLA1MTnyBksGOeGRGkxHiiJUujsjmNTdWTm4iHVSRaUao9/4Ur671auMghQ==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-arrow@1.1.0': + resolution: {integrity: sha512-FmlW1rCg7hBpEBwFbjHwCW6AmWLQM6g/v0Sn8XbP9NvmSZ2San1FpQeyPtufzOMSIx7Y4dzjlHoifhp+7NkZhw==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-avatar@1.1.1': + resolution: {integrity: sha512-eoOtThOmxeoizxpX6RiEsQZ2wj5r4+zoeqAwO0cBaFQGjJwIH3dIX0OCxNrCyrrdxG+vBweMETh3VziQG7c1kw==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-collection@1.1.0': + resolution: {integrity: sha512-GZsZslMJEyo1VKm5L1ZJY8tGDxZNPAoUeQUIbKeJfoi7Q4kmig5AsgLMYYuyYbfjd8fBmFORAIwYAkXMnXZgZw==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-compose-refs@1.1.0': + resolution: {integrity: sha512-b4inOtiaOnYf9KWyO3jAeeCG6FeyfY6ldiEPanbUjWd+xIk5wZeHa8yVwmrJ2vderhu/BQvzCrJI0lHd+wIiqw==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-context@1.1.0': + resolution: {integrity: sha512-OKrckBy+sMEgYM/sMmqmErVn0kZqrHPJze+Ql3DzYsDDp0hl0L62nx/2122/Bvps1qz645jlcu2tD9lrRSdf8A==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-context@1.1.1': + resolution: {integrity: sha512-UASk9zi+crv9WteK/NU4PLvOoL3OuE6BWVKNF6hPRBtYBDXQ2u5iu3O59zUlJiTVvkyuycnqrztsHVJwcK9K+Q==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-dialog@1.1.2': + resolution: {integrity: sha512-Yj4dZtqa2o+kG61fzB0H2qUvmwBA2oyQroGLyNtBj1beo1khoQ3q1a2AO8rrQYjd8256CO9+N8L9tvsS+bnIyA==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-direction@1.1.0': + resolution: {integrity: sha512-BUuBvgThEiAXh2DWu93XsT+a3aWrGqolGlqqw5VU1kG7p/ZH2cuDlM1sRLNnY3QcBS69UIz2mcKhMxDsdewhjg==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-dismissable-layer@1.1.1': + resolution: {integrity: sha512-QSxg29lfr/xcev6kSz7MAlmDnzbP1eI/Dwn3Tp1ip0KT5CUELsxkekFEMVBEoykI3oV39hKT4TKZzBNMbcTZYQ==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-dropdown-menu@2.1.2': + resolution: {integrity: sha512-GVZMR+eqK8/Kes0a36Qrv+i20bAPXSn8rCBTHx30w+3ECnR5o3xixAlqcVaYvLeyKUsm0aqyhWfmUcqufM8nYA==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-focus-guards@1.1.1': + resolution: {integrity: sha512-pSIwfrT1a6sIoDASCSpFwOasEwKTZWDw/iBdtnqKO7v6FeOzYJ7U53cPzYFVR3geGGXgVHaH+CdngrrAzqUGxg==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-focus-scope@1.1.0': + resolution: {integrity: sha512-200UD8zylvEyL8Bx+z76RJnASR2gRMuxlgFCPAe/Q/679a/r0eK3MBVYMb7vZODZcffZBdob1EGnky78xmVvcA==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-hover-card@1.1.2': + resolution: {integrity: sha512-Y5w0qGhysvmqsIy6nQxaPa6mXNKznfoGjOfBgzOjocLxr2XlSjqBMYQQL+FfyogsMuX+m8cZyQGYhJxvxUzO4w==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-icons@1.3.0': + resolution: {integrity: sha512-jQxj/0LKgp+j9BiTXz3O3sgs26RNet2iLWmsPyRz2SIcR4q/4SbazXfnYwbAr+vLYKSfc7qxzyGQA1HLlYiuNw==} + peerDependencies: + react: ^16.x || ^17.x || ^18.x + + '@radix-ui/react-id@1.1.0': + resolution: {integrity: sha512-EJUrI8yYh7WOjNOqpoJaf1jlFIH2LvtgAl+YcFqNCa+4hj64ZXmPkAKOFs/ukjz3byN6bdb/AVUqHkI8/uWWMA==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-label@2.1.0': + resolution: {integrity: sha512-peLblDlFw/ngk3UWq0VnYaOLy6agTZZ+MUO/WhVfm14vJGML+xH4FAl2XQGLqdefjNb7ApRg6Yn7U42ZhmYXdw==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-menu@2.1.2': + resolution: {integrity: sha512-lZ0R4qR2Al6fZ4yCCZzu/ReTFrylHFxIqy7OezIpWF4bL0o9biKo0pFIvkaew3TyZ9Fy5gYVrR5zCGZBVbO1zg==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-popper@1.2.0': + resolution: {integrity: sha512-ZnRMshKF43aBxVWPWvbj21+7TQCvhuULWJ4gNIKYpRlQt5xGRhLx66tMp8pya2UkGHTSlhpXwmjqltDYHhw7Vg==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-portal@1.1.2': + resolution: {integrity: sha512-WeDYLGPxJb/5EGBoedyJbT0MpoULmwnIPMJMSldkuiMsBAv7N1cRdsTWZWht9vpPOiN3qyiGAtbK2is47/uMFg==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-presence@1.1.1': + resolution: {integrity: sha512-IeFXVi4YS1K0wVZzXNrbaaUvIJ3qdY+/Ih4eHFhWA9SwGR9UDX7Ck8abvL57C4cv3wwMvUE0OG69Qc3NCcTe/A==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-primitive@2.0.0': + resolution: {integrity: sha512-ZSpFm0/uHa8zTvKBDjLFWLo8dkr4MBsiDLz0g3gMUwqgLHz9rTaRRGYDgvZPtBJgYCBKXkS9fzmoySgr8CO6Cw==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-roving-focus@1.1.0': + resolution: {integrity: sha512-EA6AMGeq9AEeQDeSH0aZgG198qkfHSbvWTf1HvoDmOB5bBG/qTxjYMWUKMnYiV6J/iP/J8MEFSuB2zRU2n7ODA==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-scroll-area@1.2.0': + resolution: {integrity: sha512-q2jMBdsJ9zB7QG6ngQNzNwlvxLQqONyL58QbEGwuyRZZb/ARQwk3uQVbCF7GvQVOtV6EU/pDxAw3zRzJZI3rpQ==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-select@2.1.2': + resolution: {integrity: sha512-rZJtWmorC7dFRi0owDmoijm6nSJH1tVw64QGiNIZ9PNLyBDtG+iAq+XGsya052At4BfarzY/Dhv9wrrUr6IMZA==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-slot@1.1.0': + resolution: {integrity: sha512-FUCf5XMfmW4dtYl69pdS4DbxKy8nj4M7SafBgPllysxmdachynNflAdp/gCsnYWNDnge6tI9onzMp5ARYc1KNw==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-tabs@1.1.1': + resolution: {integrity: sha512-3GBUDmP2DvzmtYLMsHmpA1GtR46ZDZ+OreXM/N+kkQJOPIgytFWWTfDQmBQKBvaFS0Vno0FktdbVzN28KGrMdw==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-toast@1.2.2': + resolution: {integrity: sha512-Z6pqSzmAP/bFJoqMAston4eSNa+ud44NSZTiZUmUen+IOZ5nBY8kzuU5WDBVyFXPtcW6yUalOHsxM/BP6Sv8ww==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-use-callback-ref@1.1.0': + resolution: {integrity: sha512-CasTfvsy+frcFkbXtSJ2Zu9JHpN8TYKxkgJGWbjiZhFivxaeW7rMeZt7QELGVLaYVfFMsKHjb7Ak0nMEe+2Vfw==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-controllable-state@1.1.0': + resolution: {integrity: sha512-MtfMVJiSr2NjzS0Aa90NPTnvTSg6C/JLCV7ma0W6+OMV78vd8OyRpID+Ng9LxzsPbLeuBnWBA1Nq30AtBIDChw==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-escape-keydown@1.1.0': + resolution: {integrity: sha512-L7vwWlR1kTTQ3oh7g1O0CBF3YCyyTj8NmhLR+phShpyA50HCfBFKVJTpshm9PzLiKmehsrQzTYTpX9HvmC9rhw==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-layout-effect@1.1.0': + resolution: {integrity: sha512-+FPE0rOdziWSrH9athwI1R0HDVbWlEhd+FR+aSDk4uWGmSJ9Z54sdZVDQPZAinJhJXwfT+qnj969mCsT2gfm5w==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-previous@1.1.0': + resolution: {integrity: sha512-Z/e78qg2YFnnXcW88A4JmTtm4ADckLno6F7OXotmkQfeuCVaKuYzqAATPhVzl3delXE7CxIV8shofPn3jPc5Og==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-rect@1.1.0': + resolution: {integrity: sha512-0Fmkebhr6PiseyZlYAOtLS+nb7jLmpqTrJyv61Pe68MKYW6OWdRE2kI70TaYY27u7H0lajqM3hSMMLFq18Z7nQ==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-size@1.1.0': + resolution: {integrity: sha512-XW3/vWuIXHa+2Uwcc2ABSfcCledmXhhQPlGbfcRXbiUQI5Icjcg19BGCZVKKInYbvUCut/ufbbLLPFC5cbb1hw==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-visually-hidden@1.1.0': + resolution: {integrity: sha512-N8MDZqtgCgG5S3aV60INAB475osJousYpZ4cTJ2cFbMpdHS5Y6loLTH8LPtkj2QN0x93J30HT/M3qJXM0+lyeQ==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/rect@1.1.0': + resolution: {integrity: sha512-A9+lCBZoaMJlVKcRBz2YByCG+Cp2t6nAnMnNba+XiWxnj6r4JUFqfsgwocMBZU9LPtdxC6wB56ySYpc7LQIoJg==} + + '@react-aria/focus@3.18.3': + resolution: {integrity: sha512-WKUElg+5zS0D3xlVn8MntNnkzJql2J6MuzAMP8Sv5WTgFDse/XGR842dsxPTIyKKdrWVCRegCuwa4m3n/GzgJw==} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0 + + '@react-aria/interactions@3.22.3': + resolution: {integrity: sha512-RRUb/aG+P0IKTIWikY/SylB6bIbLZeztnZY2vbe7RAG5MgVaCgn5HQ45SI15GlTmhsFG8CnF6slJsUFJiNHpbQ==} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0 + + '@react-aria/ssr@3.9.6': + resolution: {integrity: sha512-iLo82l82ilMiVGy342SELjshuWottlb5+VefO3jOQqQRNYnJBFpUSadswDPbRimSgJUZuFwIEYs6AabkP038fA==} + engines: {node: '>= 12'} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0 + + '@react-aria/utils@3.25.3': + resolution: {integrity: sha512-PR5H/2vaD8fSq0H/UB9inNbc8KDcVmW6fYAfSWkkn+OAdhTTMVKqXXrZuZBWyFfSD5Ze7VN6acr4hrOQm2bmrA==} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0 + + '@react-stately/utils@3.10.4': + resolution: {integrity: sha512-gBEQEIMRh5f60KCm7QKQ2WfvhB2gLUr9b72sqUdIZ2EG+xuPgaIlCBeSicvjmjBvYZwOjoOEnmIkcx2GHp/HWw==} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0 + + '@react-types/shared@3.25.0': + resolution: {integrity: sha512-OZSyhzU6vTdW3eV/mz5i6hQwQUhkRs7xwY2d1aqPvTdMe0+2cY7Fwp45PAiwYLEj73i9ro2FxF9qC4DvHGSCgQ==} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0 + + '@remixicon/react@4.3.0': + resolution: {integrity: sha512-mAVDn8pAa9dURltGwiYrf7bPIqjG4ZAnCUHfjpgz3g+HLSDNXOaJ67Z5wmjVB5KMGpp9JbbTN5vsp2z+ajVLWg==} + peerDependencies: + react: '>=18.2.0' + + '@swc/counter@0.1.3': + resolution: {integrity: sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==} + + '@swc/helpers@0.5.5': + resolution: {integrity: sha512-KGYxvIOXcceOAbEk4bi/dVLEK9z8sZ0uBB3Il5b1rhfClSpcX0yfRO0KmTkqR2cnQDymwLB+25ZyMzICg/cm/A==} + + '@tailwindcss/forms@0.5.9': + resolution: {integrity: sha512-tM4XVr2+UVTxXJzey9Twx48c1gcxFStqn1pQz0tRsX8o3DvxhN5oY5pvyAbUx7VTaZxpej4Zzvc6h+1RJBzpIg==} + peerDependencies: + tailwindcss: '>=3.0.0 || >= 3.0.0-alpha.1 || >= 4.0.0-alpha.20' + + '@tanstack/react-table@8.20.5': + resolution: {integrity: sha512-WEHopKw3znbUZ61s9i0+i9g8drmDo6asTWbrQh8Us63DAk/M0FkmIqERew6P71HI75ksZ2Pxyuf4vvKh9rAkiA==} + engines: {node: '>=12'} + peerDependencies: + react: '>=16.8' + react-dom: '>=16.8' + + '@tanstack/react-virtual@3.10.8': + resolution: {integrity: sha512-VbzbVGSsZlQktyLrP5nxE+vE1ZR+U0NFAWPbJLoG2+DKPwd2D7dVICTVIIaYlJqX1ZCEnYDbaOpmMwbsyhBoIA==} + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 + + '@tanstack/react-virtual@3.5.0': + resolution: {integrity: sha512-rtvo7KwuIvqK9zb0VZ5IL7fiJAEnG+0EiFZz8FUOs+2mhGqdGmjKIaT1XU7Zq0eFqL0jonLlhbayJI/J2SA/Bw==} + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 + + '@tanstack/table-core@8.20.5': + resolution: {integrity: sha512-P9dF7XbibHph2PFRz8gfBKEXEY/HJPOhym8CHmjF8y3q5mWpKx9xtZapXQUWCgkqvsK0R46Azuz+VaxD4Xl+Tg==} + engines: {node: '>=12'} + + '@tanstack/virtual-core@3.10.8': + resolution: {integrity: sha512-PBu00mtt95jbKFi6Llk9aik8bnR3tR/oQP1o3TSi+iG//+Q2RTIzCEgKkHG8BB86kxMNW6O8wku+Lmi+QFR6jA==} + + '@tanstack/virtual-core@3.5.0': + resolution: {integrity: sha512-KnPRCkQTyqhanNC0K63GBG3wA8I+D1fQuVnAvcBF8f13akOKeQp1gSbu6f77zCxhEk727iV5oQnbHLYzHrECLg==} + + '@tootallnate/once@1.1.2': + resolution: {integrity: sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==} + engines: {node: '>= 6'} + + '@tremor/react@3.18.3': + resolution: {integrity: sha512-7QyGE2W9f2FpwH24TKy3/mqBgLl4sHZeQcXP3rxXZ8W2AUq7AVaG1+vIT3xXxISrkh7zknjWlZsuhoF8NWNVDw==} + peerDependencies: + react: ^18.0.0 + react-dom: '>=16.6.0' + + '@tsconfig/node10@1.0.11': + resolution: {integrity: sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==} + + '@tsconfig/node12@1.0.11': + resolution: {integrity: sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==} + + '@tsconfig/node14@1.0.3': + resolution: {integrity: sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==} + + '@tsconfig/node16@1.0.4': + resolution: {integrity: sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==} + + '@tybys/wasm-util@0.8.3': + resolution: {integrity: sha512-Z96T/L6dUFFxgFJ+pQtkPpne9q7i6kIPYCFnQBHSgSPV9idTsKfIhCss0h5iM9irweZCatkrdeP8yi5uM1eX6Q==} + + '@types/better-sqlite3@7.6.11': + resolution: {integrity: sha512-i8KcD3PgGtGBLl3+mMYA8PdKkButvPyARxA7IQAd6qeslht13qxb1zzO8dRCtE7U3IoJS782zDBAeoKiM695kg==} + + '@types/d3-array@3.2.1': + resolution: {integrity: sha512-Y2Jn2idRrLzUfAKV2LyRImR+y4oa2AntrgID95SHJxuMUrkNXmanDSed71sRNZysveJVt1hLLemQZIady0FpEg==} + + '@types/d3-color@3.1.3': + resolution: {integrity: sha512-iO90scth9WAbmgv7ogoq57O9YpKmFBbmoEoCHDB2xMBY0+/KVrqAaCDyCE16dUspeOvIxFFRI+0sEtqDqy2b4A==} + + '@types/d3-ease@3.0.2': + resolution: {integrity: sha512-NcV1JjO5oDzoK26oMzbILE6HW7uVXOHLQvHshBUW4UMdZGfiY6v5BeQwh9a9tCzv+CeefZQHJt5SRgK154RtiA==} + + '@types/d3-interpolate@3.0.4': + resolution: {integrity: sha512-mgLPETlrpVV1YRJIglr4Ez47g7Yxjl1lj7YKsiMCb27VJH9W8NVM6Bb9d8kkpG/uAQS5AmbA48q2IAolKKo1MA==} + + '@types/d3-path@3.1.0': + resolution: {integrity: sha512-P2dlU/q51fkOc/Gfl3Ul9kicV7l+ra934qBFXCFhrZMOL6du1TM0pm1ThYvENukyOn5h9v+yMJ9Fn5JK4QozrQ==} + + '@types/d3-scale@4.0.8': + resolution: {integrity: sha512-gkK1VVTr5iNiYJ7vWDI+yUFFlszhNMtVeneJ6lUTKPjprsvLLI9/tgEGiXJOnlINJA8FyA88gfnQsHbybVZrYQ==} + + '@types/d3-shape@3.1.6': + resolution: {integrity: sha512-5KKk5aKGu2I+O6SONMYSNflgiP0WfZIQvVUMan50wHsLG1G94JlxEVnCpQARfTtzytuY0p/9PXXZb3I7giofIA==} + + '@types/d3-time@3.0.3': + resolution: {integrity: sha512-2p6olUZ4w3s+07q3Tm2dbiMZy5pCDfYwtLXXHUnVzXgQlZ/OyPtUz6OL382BkOuGlLXqfT+wqv8Fw2v8/0geBw==} + + '@types/d3-timer@3.0.2': + resolution: {integrity: sha512-Ps3T8E8dZDam6fUyNiMkekK3XUsaUEik+idO9/YjPtfj2qruF8tFBXS7XhtE4iIXBLxhmLjP3SXpLhVf21I9Lw==} + + '@types/lodash@4.17.13': + resolution: {integrity: sha512-lfx+dftrEZcdBPczf9d0Qv0x+j/rfNCMuC6OcfXmO8gkfeNAY88PgKUbvG56whcN23gc27yenwF6oJZXGFpYxg==} + + '@types/luxon@3.4.2': + resolution: {integrity: sha512-TifLZlFudklWlMBfhubvgqTXRzLDI5pCbGa4P8a3wPyUQSW+1xQ5eDsreP9DWHX3tjq1ke96uYG/nwundroWcA==} + + '@types/node@20.16.11': + resolution: {integrity: sha512-y+cTCACu92FyA5fgQSAI8A1H429g7aSK2HsO7K4XYUWc4dY5IUz55JSDIYT6/VsOLfGy8vmvQYC2hfb0iF16Uw==} + + '@types/prop-types@15.7.12': + resolution: {integrity: sha512-5zvhXYtRNRluoE/jAp4GVsSduVUzNWKkOZrCDBWYtE7biZywwdC2AcEzg+cSMLFRfVgeAFqpfNabiPjxFddV1Q==} + + '@types/react-dom@18.3.1': + resolution: {integrity: sha512-qW1Mfv8taImTthu4KoXgDfLuk4bydU6Q/TkADnDWWHwi4NX4BR+LWfTp2sVmTqRrsHvyDDTelgelxJ+SsejKKQ==} + + '@types/react@18.3.11': + resolution: {integrity: sha512-r6QZ069rFTjrEYgFdOck1gK7FLVsgJE7tTz0pQBczlBNUhBNk0MQH4UbnFSwjpQLMkLzgqvBBa+qGpLje16eTQ==} + + '@types/ws@8.5.12': + resolution: {integrity: sha512-3tPRkv1EtkDpzlgyKyI8pGsGZAGPEaXeu0DOj5DI25Ja91bdAYddYHbADRYVrZMRbfW+1l5YwXVDKohDJNQxkQ==} + + abbrev@1.1.1: + resolution: {integrity: sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==} + + acorn-walk@8.3.4: + resolution: {integrity: sha512-ueEepnujpqee2o5aIYnvHU6C0A42MNdsIDeqy5BydrkuC5R1ZuUFnm27EeFJGoEHJQgn3uleRvmTXaJgfXbt4g==} + engines: {node: '>=0.4.0'} + + acorn@8.14.0: + resolution: {integrity: sha512-cl669nCJTZBsL97OF4kUQm5g5hC2uihk0NxY3WENAC0TYdILVkAyHymAntgxGkl7K+t0cXIrH5siy5S4XkFycA==} + engines: {node: '>=0.4.0'} + hasBin: true + + agent-base@6.0.2: + resolution: {integrity: sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==} + engines: {node: '>= 6.0.0'} + + agentkeepalive@4.5.0: + resolution: {integrity: sha512-5GG/5IbQQpC9FpkRGsSvZI5QYeSCzlJHdpBQntCsuTOxhKD8lqKhrleg2Yi7yvMIf82Ycmmqln9U8V9qwEiJew==} + engines: {node: '>= 8.0.0'} + + aggregate-error@3.1.0: + resolution: {integrity: sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==} + engines: {node: '>=8'} + + ansi-regex@5.0.1: + resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} + engines: {node: '>=8'} + + ansi-regex@6.0.1: + resolution: {integrity: sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==} + engines: {node: '>=12'} + + ansi-styles@4.3.0: + resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} + engines: {node: '>=8'} + + ansi-styles@6.2.1: + resolution: {integrity: sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==} + engines: {node: '>=12'} + + any-promise@1.3.0: + resolution: {integrity: sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==} + + anymatch@3.1.3: + resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} + engines: {node: '>= 8'} + + aproba@2.0.0: + resolution: {integrity: sha512-lYe4Gx7QT+MKGbDsA+Z+he/Wtef0BiwDOlK/XkBrdfsh9J/jPPXbX0tE9x9cl27Tmu5gg3QUbUrQYa/y+KOHPQ==} + + arctic@1.9.2: + resolution: {integrity: sha512-VTnGpYx+ypboJdNrWnK17WeD7zN/xSCHnpecd5QYsBfVZde/5i+7DJ1wrf/ioSDMiEjagXmyNWAE3V2C9f1hNg==} + + are-we-there-yet@3.0.1: + resolution: {integrity: sha512-QZW4EDmGwlYur0Yyf/b2uGucHQMa8aFUP7eu9ddR73vvhFyt4V0Vl3QHPcTNJ8l6qYOBdxgXdnBXQrHilfRQBg==} + engines: {node: ^12.13.0 || ^14.15.0 || >=16.0.0} + deprecated: This package is no longer supported. + + arg@4.1.3: + resolution: {integrity: sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==} + + arg@5.0.2: + resolution: {integrity: sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==} + + aria-hidden@1.2.4: + resolution: {integrity: sha512-y+CcFFwelSXpLZk/7fMB2mUbGtX9lKycf1MWJ7CaTIERyitVlyQx6C+sxcROU2BAJ24OiZyK+8wj2i8AlBoS3A==} + engines: {node: '>=10'} + + balanced-match@1.0.2: + resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} + + base64-js@1.5.1: + resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} + + better-sqlite3@9.6.0: + resolution: {integrity: sha512-yR5HATnqeYNVnkaUTf4bOP2dJSnyhP4puJN/QPRyx4YkBEEUxib422n2XzPqDEHjQQqazoYoADdAm5vE15+dAQ==} + + binary-extensions@2.3.0: + resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} + engines: {node: '>=8'} + + bindings@1.5.0: + resolution: {integrity: sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==} + + bl@4.1.0: + resolution: {integrity: sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==} + + brace-expansion@1.1.11: + resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==} + + brace-expansion@2.0.1: + resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} + + braces@3.0.2: + resolution: {integrity: sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==} + engines: {node: '>=8'} + + buffer-from@1.1.2: + resolution: {integrity: sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==} + + buffer@5.7.1: + resolution: {integrity: sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==} + + busboy@1.6.0: + resolution: {integrity: sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==} + engines: {node: '>=10.16.0'} + + cacache@15.3.0: + resolution: {integrity: sha512-VVdYzXEn+cnbXpFgWs5hTT7OScegHVmLhJIR8Ufqk3iFD6A6j5iSX1KuBTfNEv4tdJWE2PzA6IVFtcLC7fN9wQ==} + engines: {node: '>= 10'} + + camelcase-css@2.0.1: + resolution: {integrity: sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA==} + engines: {node: '>= 6'} + + caniuse-lite@1.0.30001617: + resolution: {integrity: sha512-mLyjzNI9I+Pix8zwcrpxEbGlfqOkF9kM3ptzmKNw5tizSyYwMe+nGLTqMK9cO+0E+Bh6TsBxNAaHWEM8xwSsmA==} + + chokidar@3.6.0: + resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} + engines: {node: '>= 8.10.0'} + + chownr@1.1.4: + resolution: {integrity: sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==} + + chownr@2.0.0: + resolution: {integrity: sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==} + engines: {node: '>=10'} + + class-variance-authority@0.7.0: + resolution: {integrity: sha512-jFI8IQw4hczaL4ALINxqLEXQbWcNjoSkloa4IaufXCJr6QawJyw7tuRysRsrE8w2p/4gGaxKIt/hX3qz/IbD1A==} + + clean-stack@2.2.0: + resolution: {integrity: sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==} + engines: {node: '>=6'} + + cli-color@2.0.4: + resolution: {integrity: sha512-zlnpg0jNcibNrO7GG9IeHH7maWFeCz+Ja1wx/7tZNU5ASSSSZ+/qZciM0/LHCYxSdqv5h2sdbQ/PXYdOuetXvA==} + engines: {node: '>=0.10'} + + client-only@0.0.1: + resolution: {integrity: sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA==} + + clsx@2.0.0: + resolution: {integrity: sha512-rQ1+kcj+ttHG0MKVGBUXwayCCF1oh39BF5COIpRzuCEv8Mwjv0XucrI2ExNTOn9IlLifGClWQcU9BrZORvtw6Q==} + engines: {node: '>=6'} + + clsx@2.1.1: + resolution: {integrity: sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==} + engines: {node: '>=6'} + + color-convert@2.0.1: + resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} + engines: {node: '>=7.0.0'} + + color-name@1.1.4: + resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} + + color-support@1.1.3: + resolution: {integrity: sha512-qiBjkpbMLO/HL68y+lh4q0/O1MZFj2RX6X/KmMa3+gJD3z+WwI1ZzDHysvqHGS3mP6mznPckpXmw1nI9cJjyRg==} + hasBin: true + + commander@4.1.1: + resolution: {integrity: sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==} + engines: {node: '>= 6'} + + commander@9.5.0: + resolution: {integrity: sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==} + engines: {node: ^12.20.0 || >=14} + + concat-map@0.0.1: + resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} + + console-control-strings@1.1.0: + resolution: {integrity: sha512-ty/fTekppD2fIwRvnZAVdeOiGd1c7YXEixbgJTNzqcxJWKQnjJ/V1bNEEE6hygpM3WjwHFUVK6HTjWSzV4a8sQ==} + + create-require@1.1.1: + resolution: {integrity: sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==} + + cross-spawn@7.0.3: + resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} + engines: {node: '>= 8'} + + cssesc@3.0.0: + resolution: {integrity: sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==} + engines: {node: '>=4'} + hasBin: true + + csstype@3.1.3: + resolution: {integrity: sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==} + + d3-array@3.2.4: + resolution: {integrity: sha512-tdQAmyA18i4J7wprpYq8ClcxZy3SC31QMeByyCFyRt7BVHdREQZ5lpzoe5mFEYZUWe+oq8HBvk9JjpibyEV4Jg==} + engines: {node: '>=12'} + + d3-color@3.1.0: + resolution: {integrity: sha512-zg/chbXyeBtMQ1LbD/WSoW2DpC3I0mpmPdW+ynRTj/x2DAWYrIY7qeZIHidozwV24m4iavr15lNwIwLxRmOxhA==} + engines: {node: '>=12'} + + d3-ease@3.0.1: + resolution: {integrity: sha512-wR/XK3D3XcLIZwpbvQwQ5fK+8Ykds1ip7A2Txe0yxncXSdq1L9skcG7blcedkOX+ZcgxGAmLX1FrRGbADwzi0w==} + engines: {node: '>=12'} + + d3-format@3.1.0: + resolution: {integrity: sha512-YyUI6AEuY/Wpt8KWLgZHsIU86atmikuoOmCfommt0LYHiQSPjvX2AcFc38PX0CBpr2RCyZhjex+NS/LPOv6YqA==} + engines: {node: '>=12'} + + d3-interpolate@3.0.1: + resolution: {integrity: sha512-3bYs1rOD33uo8aqJfKP3JWPAibgw8Zm2+L9vBKEHJ2Rg+viTR7o5Mmv5mZcieN+FRYaAOWX5SJATX6k1PWz72g==} + engines: {node: '>=12'} + + d3-path@3.1.0: + resolution: {integrity: sha512-p3KP5HCf/bvjBSSKuXid6Zqijx7wIfNW+J/maPs+iwR35at5JCbLUT0LzF1cnjbCHWhqzQTIN2Jpe8pRebIEFQ==} + engines: {node: '>=12'} + + d3-scale@4.0.2: + resolution: {integrity: sha512-GZW464g1SH7ag3Y7hXjf8RoUuAFIqklOAq3MRl4OaWabTFJY9PN/E1YklhXLh+OQ3fM9yS2nOkCoS+WLZ6kvxQ==} + engines: {node: '>=12'} + + d3-shape@3.2.0: + resolution: {integrity: sha512-SaLBuwGm3MOViRq2ABk3eLoxwZELpH6zhl3FbAoJ7Vm1gofKx6El1Ib5z23NUEhF9AsGl7y+dzLe5Cw2AArGTA==} + engines: {node: '>=12'} + + d3-time-format@4.1.0: + resolution: {integrity: sha512-dJxPBlzC7NugB2PDLwo9Q8JiTR3M3e4/XANkreKSUxF8vvXKqm1Yfq4Q5dl8budlunRVlUUaDUgFt7eA8D6NLg==} + engines: {node: '>=12'} + + d3-time@3.1.0: + resolution: {integrity: sha512-VqKjzBLejbSMT4IgbmVgDjpkYrNWUYJnbCGo874u7MMKIWsILRX+OpX/gTk8MqjpT1A/c6HY2dCA77ZN0lkQ2Q==} + engines: {node: '>=12'} + + d3-timer@3.0.1: + resolution: {integrity: sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA==} + engines: {node: '>=12'} + + d@1.0.2: + resolution: {integrity: sha512-MOqHvMWF9/9MX6nza0KgvFH4HpMU0EF5uUDXqX/BtxtU8NfB0QzRtJ8Oe/6SuS4kbhyzVJwjd97EA4PKrzJ8bw==} + engines: {node: '>=0.12'} + + data-uri-to-buffer@4.0.1: + resolution: {integrity: sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==} + engines: {node: '>= 12'} + + date-fns@3.6.0: + resolution: {integrity: sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==} + + date-fns@4.1.0: + resolution: {integrity: sha512-Ukq0owbQXxa/U3EGtsdVBkR1w7KOQ5gIBqdH2hkvknzZPYvBxb/aa6E8L7tmjFtkwZBu3UXBbjIgPo/Ez4xaNg==} + + debug@4.3.4: + resolution: {integrity: sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==} + engines: {node: '>=6.0'} + peerDependencies: + supports-color: '*' + peerDependenciesMeta: + supports-color: + optional: true + + decimal.js-light@2.5.1: + resolution: {integrity: sha512-qIMFpTMZmny+MMIitAB6D7iVPEorVw6YQRWkvarTkT4tBeSLLiHzcwj6q0MmYSFCiVpiqPJTJEYIrpcPzVEIvg==} + + decompress-response@6.0.0: + resolution: {integrity: sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==} + engines: {node: '>=10'} + + deep-extend@0.6.0: + resolution: {integrity: sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==} + engines: {node: '>=4.0.0'} + + delegates@1.0.0: + resolution: {integrity: sha512-bd2L678uiWATM6m5Z1VzNCErI3jiGzt6HGY8OVICs40JQq/HALfbyNJmp0UDakEY4pMMaN0Ly5om/B1VI/+xfQ==} + + detect-libc@2.0.2: + resolution: {integrity: sha512-UX6sGumvvqSaXgdKGUsgZWqcUyIXZ/vZTrlRT/iobiKhGL0zL4d3osHj3uqllWJK+i+sixDS/3COVEOFbupFyw==} + engines: {node: '>=8'} + + detect-libc@2.0.3: + resolution: {integrity: sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==} + engines: {node: '>=8'} + + detect-node-es@1.1.0: + resolution: {integrity: sha512-ypdmJU/TbBby2Dxibuv7ZLW3Bs1QEmM7nHjEANfohJLvE0XVujisn1qPJcZxg+qDucsr+bP6fLD1rPS3AhJ7EQ==} + + didyoumean@1.2.2: + resolution: {integrity: sha512-gxtyfqMg7GKyhQmb056K7M3xszy/myH8w+B4RT+QXBQsvAOdc3XymqDDPHx1BgPgsdAA5SIifona89YtRATDzw==} + + diff@4.0.2: + resolution: {integrity: sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==} + engines: {node: '>=0.3.1'} + + difflib@0.2.4: + resolution: {integrity: sha512-9YVwmMb0wQHQNr5J9m6BSj6fk4pfGITGQOOs+D9Fl+INODWFOfvhIU1hNv6GgR1RBoC/9NJcwu77zShxV0kT7w==} + + dlv@1.1.3: + resolution: {integrity: sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==} + + dom-helpers@5.2.1: + resolution: {integrity: sha512-nRCa7CK3VTrM2NmGkIy4cbK7IZlgBE/PYMn55rrXefr5xXDP0LdtfPnblFDoVdcAfslJ7or6iqAUnx0CCGIWQA==} + + dreamopt@0.8.0: + resolution: {integrity: sha512-vyJTp8+mC+G+5dfgsY+r3ckxlz+QMX40VjPQsZc5gxVAxLmi64TBoVkP54A/pRAXMXsbu2GMMBrZPxNv23waMg==} + engines: {node: '>=0.4.0'} + + drizzle-kit@0.21.4: + resolution: {integrity: sha512-Nxcc1ONJLRgbhmR+azxjNF9Ly9privNLEIgW53c92whb4xp8jZLH1kMCh/54ci1mTMuYxPdOukqLwJ8wRudNwA==} + hasBin: true + + drizzle-orm@0.30.10: + resolution: {integrity: sha512-IRy/QmMWw9lAQHpwbUh1b8fcn27S/a9zMIzqea1WNOxK9/4EB8gIo+FZWLiPXzl2n9ixGSv8BhsLZiOppWEwBw==} + peerDependencies: + '@aws-sdk/client-rds-data': '>=3' + '@cloudflare/workers-types': '>=3' + '@electric-sql/pglite': '>=0.1.1' + '@libsql/client': '*' + '@neondatabase/serverless': '>=0.1' + '@op-engineering/op-sqlite': '>=2' + '@opentelemetry/api': ^1.4.1 + '@planetscale/database': '>=1' + '@types/better-sqlite3': '*' + '@types/pg': '*' + '@types/react': '>=18' + '@types/sql.js': '*' + '@vercel/postgres': '>=0.8.0' + '@xata.io/client': '*' + better-sqlite3: '>=7' + bun-types: '*' + expo-sqlite: '>=13.2.0' + knex: '*' + kysely: '*' + mysql2: '>=2' + pg: '>=8' + postgres: '>=3' + react: '>=18' + sql.js: '>=1' + sqlite3: '>=5' + peerDependenciesMeta: + '@aws-sdk/client-rds-data': + optional: true + '@cloudflare/workers-types': + optional: true + '@electric-sql/pglite': + optional: true + '@libsql/client': + optional: true + '@neondatabase/serverless': + optional: true + '@op-engineering/op-sqlite': + optional: true + '@opentelemetry/api': + optional: true + '@planetscale/database': + optional: true + '@types/better-sqlite3': + optional: true + '@types/pg': + optional: true + '@types/react': + optional: true + '@types/sql.js': + optional: true + '@vercel/postgres': + optional: true + '@xata.io/client': + optional: true + better-sqlite3: + optional: true + bun-types: + optional: true + expo-sqlite: + optional: true + knex: + optional: true + kysely: + optional: true + mysql2: + optional: true + pg: + optional: true + postgres: + optional: true + react: + optional: true + sql.js: + optional: true + sqlite3: + optional: true + + eastasianwidth@0.2.0: + resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==} + + emoji-regex@8.0.0: + resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} + + emoji-regex@9.2.2: + resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==} + + encoding@0.1.13: + resolution: {integrity: sha512-ETBauow1T35Y/WZMkio9jiM0Z5xjHHmJ4XmjZOq1l/dXz3lr2sRn87nJy20RupqSh1F2m3HHPSp8ShIPQJrJ3A==} + + end-of-stream@1.4.4: + resolution: {integrity: sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==} + + env-paths@2.2.1: + resolution: {integrity: sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==} + engines: {node: '>=6'} + + env-paths@3.0.0: + resolution: {integrity: sha512-dtJUTepzMW3Lm/NPxRf3wP4642UWhjL2sQxc+ym2YMj1m/H2zDNQOlezafzkHwn6sMstjHTwG6iQQsctDW/b1A==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + + err-code@2.0.3: + resolution: {integrity: sha512-2bmlRpNKBxT/CRmPOlyISQpNj+qSeYvcym/uT0Jx2bMOlKLtSy1ZmLuVxSEKKyor/N5yhvp/ZiG1oE3DEYMSFA==} + + es5-ext@0.10.64: + resolution: {integrity: sha512-p2snDhiLaXe6dahss1LddxqEm+SkuDvV8dnIQG0MWjyHpcMNfXKPE+/Cc0y+PhxJX3A4xGNeFCj5oc0BUh6deg==} + engines: {node: '>=0.10'} + + es6-iterator@2.0.3: + resolution: {integrity: sha512-zw4SRzoUkd+cl+ZoE15A9o1oQd920Bb0iOJMQkQhl3jNc03YqVjAhG7scf9C5KWRU/R13Orf588uCC6525o02g==} + + es6-symbol@3.1.4: + resolution: {integrity: sha512-U9bFFjX8tFiATgtkJ1zg25+KviIXpgRvRHS8sau3GfhVzThRQrOeksPeT0BWW2MNZs1OEWJ1DPXOQMn0KKRkvg==} + engines: {node: '>=0.12'} + + es6-weak-map@2.0.3: + resolution: {integrity: sha512-p5um32HOTO1kP+w7PRnB+5lQ43Z6muuMuIMffvDN8ZB4GcnjLBV6zGStpbASIMk4DCAvEaamhe2zhyCb/QXXsA==} + + esbuild-register@3.5.0: + resolution: {integrity: sha512-+4G/XmakeBAsvJuDugJvtyF1x+XJT4FMocynNpxrvEBViirpfUn2PgNpCHedfWhF4WokNsO/OvMKrmJOIJsI5A==} + peerDependencies: + esbuild: '>=0.12 <1' + + esbuild@0.18.20: + resolution: {integrity: sha512-ceqxoedUrcayh7Y7ZX6NdbbDzGROiyVBgC4PriJThBKSVPWnnFHZAkfI1lJT8QFkOwH4qOS2SJkS4wvpGl8BpA==} + engines: {node: '>=12'} + hasBin: true + + esbuild@0.19.12: + resolution: {integrity: sha512-aARqgq8roFBj054KvQr5f1sFu0D65G+miZRCuJyJ0G13Zwx7vRar5Zhn2tkQNzIXcBrNVsv/8stehpj+GAjgbg==} + engines: {node: '>=12'} + hasBin: true + + esniff@2.0.1: + resolution: {integrity: sha512-kTUIGKQ/mDPFoJ0oVfcmyJn4iBDRptjNVIzwIFR7tqWXdVI9xfA2RMwY/gbSpJG3lkdWNEjLap/NqVHZiJsdfg==} + engines: {node: '>=0.10'} + + event-emitter@0.3.5: + resolution: {integrity: sha512-D9rRn9y7kLPnJ+hMq7S/nhvoKwwvVJahBi2BPmx3bvbsEdK3W9ii8cBSGjP+72/LnM4n6fo3+dkCX5FeTQruXA==} + + eventemitter3@4.0.7: + resolution: {integrity: sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==} + + expand-template@2.0.3: + resolution: {integrity: sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==} + engines: {node: '>=6'} + + ext@1.7.0: + resolution: {integrity: sha512-6hxeJYaL110a9b5TEJSj0gojyHQAmA2ch5Os+ySCiA1QGdS697XWY1pzsrSjqA9LDEEgdB/KypIlR59RcLuHYw==} + + fast-equals@5.0.1: + resolution: {integrity: sha512-WF1Wi8PwwSY7/6Kx0vKXtw8RwuSGoM1bvDaJbu7MxDlR1vovZjIAKrnzyrThgAjm6JDTu0fVgWXDlMGspodfoQ==} + engines: {node: '>=6.0.0'} + + fast-glob@3.3.2: + resolution: {integrity: sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==} + engines: {node: '>=8.6.0'} + + fastq@1.17.1: + resolution: {integrity: sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==} + + fetch-blob@3.2.0: + resolution: {integrity: sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==} + engines: {node: ^12.20 || >= 14.13} + + file-uri-to-path@1.0.0: + resolution: {integrity: sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==} + + fill-range@7.0.1: + resolution: {integrity: sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==} + engines: {node: '>=8'} + + foreground-child@3.1.1: + resolution: {integrity: sha512-TMKDUnIte6bfb5nWv7V/caI169OHgvwjb7V4WkeUvbQQdjr5rWKqHFiKWb/fcOwB+CzBT+qbWjvj+DVwRskpIg==} + engines: {node: '>=14'} + + formdata-polyfill@4.0.10: + resolution: {integrity: sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==} + engines: {node: '>=12.20.0'} + + fs-constants@1.0.0: + resolution: {integrity: sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==} + + fs-minipass@2.1.0: + resolution: {integrity: sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==} + engines: {node: '>= 8'} + + fs-monkey@1.0.6: + resolution: {integrity: sha512-b1FMfwetIKymC0eioW7mTywihSQE4oLzQn1dB6rZB5fx/3NpNEdAWeCSMB+60/AeT0TCXsxzAlcYVEFCTAksWg==} + + fs.realpath@1.0.0: + resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} + + fsevents@2.3.3: + resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} + engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} + os: [darwin] + + function-bind@1.1.2: + resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} + + gauge@4.0.4: + resolution: {integrity: sha512-f9m+BEN5jkg6a0fZjleidjN51VE1X+mPFQ2DJ0uv1V39oCLCbsGe6yjbBnp7eK7z/+GAon99a3nHuqbuuthyPg==} + engines: {node: ^12.13.0 || ^14.15.0 || >=16.0.0} + deprecated: This package is no longer supported. + + get-nonce@1.0.1: + resolution: {integrity: sha512-FJhYRoDaiatfEkUK8HKlicmu/3SGFD51q3itKDGoSTysQJBnfOcxU5GxnhE1E6soB76MbT0MBtnKJuXyAx+96Q==} + engines: {node: '>=6'} + + get-tsconfig@4.7.5: + resolution: {integrity: sha512-ZCuZCnlqNzjb4QprAzXKdpp/gh6KTxSJuw3IBsPnV/7fV4NxC9ckB+vPTt8w7fJA0TaSD7c55BR47JD6MEDyDw==} + + github-from-package@0.0.0: + resolution: {integrity: sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==} + + glob-parent@5.1.2: + resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} + engines: {node: '>= 6'} + + glob-parent@6.0.2: + resolution: {integrity: sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==} + engines: {node: '>=10.13.0'} + + glob@10.3.14: + resolution: {integrity: sha512-4fkAqu93xe9Mk7le9v0y3VrPDqLKHarNi2s4Pv7f2yOvfhWfhc7hRPHC/JyqMqb8B/Dt/eGS4n7ykwf3fOsl8g==} + engines: {node: '>=16 || 14 >=14.17'} + hasBin: true + + glob@7.2.3: + resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==} + deprecated: Glob versions prior to v9 are no longer supported + + glob@8.1.0: + resolution: {integrity: sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==} + engines: {node: '>=12'} + deprecated: Glob versions prior to v9 are no longer supported + + graceful-fs@4.2.11: + resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==} + + hanji@0.0.5: + resolution: {integrity: sha512-Abxw1Lq+TnYiL4BueXqMau222fPSPMFtya8HdpWsz/xVAhifXou71mPh/kY2+08RgFcVccjG3uZHs6K5HAe3zw==} + + has-unicode@2.0.1: + resolution: {integrity: sha512-8Rf9Y83NBReMnx0gFzA8JImQACstCYWUplepDa9xprwwtmgEZUF0h/i5xSA625zB/I37EtrswSST6OXxwaaIJQ==} + + hasown@2.0.2: + resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} + engines: {node: '>= 0.4'} + + heap@0.2.7: + resolution: {integrity: sha512-2bsegYkkHO+h/9MGbn6KWcE45cHZgPANo5LXF7EvWdT0yT2EguSVO1nDgU5c8+ZOPwp2vMNa7YFsJhVcDR9Sdg==} + + http-cache-semantics@4.1.1: + resolution: {integrity: sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ==} + + http-proxy-agent@4.0.1: + resolution: {integrity: sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==} + engines: {node: '>= 6'} + + https-proxy-agent@5.0.1: + resolution: {integrity: sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==} + engines: {node: '>= 6'} + + humanize-ms@1.2.1: + resolution: {integrity: sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ==} + + iconv-lite@0.6.3: + resolution: {integrity: sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==} + engines: {node: '>=0.10.0'} + + ieee754@1.2.1: + resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} + + imurmurhash@0.1.4: + resolution: {integrity: sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==} + engines: {node: '>=0.8.19'} + + indent-string@4.0.0: + resolution: {integrity: sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==} + engines: {node: '>=8'} + + infer-owner@1.0.4: + resolution: {integrity: sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A==} + + inflight@1.0.6: + resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==} + deprecated: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful. + + inherits@2.0.4: + resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} + + ini@1.3.8: + resolution: {integrity: sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==} + + internmap@2.0.3: + resolution: {integrity: sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg==} + engines: {node: '>=12'} + + invariant@2.2.4: + resolution: {integrity: sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==} + + ip-address@9.0.5: + resolution: {integrity: sha512-zHtQzGojZXTwZTHQqra+ETKd4Sn3vgi7uBmlPoXVWZqYvuKmtI0l/VZTjqGmJY9x88GGOaZ9+G9ES8hC4T4X8g==} + engines: {node: '>= 12'} + + is-binary-path@2.1.0: + resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} + engines: {node: '>=8'} + + is-core-module@2.13.1: + resolution: {integrity: sha512-hHrIjvZsftOsvKSn2TRYl63zvxsgE0K+0mYMoH6gD4omR5IWB2KynivBQczo3+wF1cCkjzvptnI9Q0sPU66ilw==} + + is-extglob@2.1.1: + resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} + engines: {node: '>=0.10.0'} + + is-fullwidth-code-point@3.0.0: + resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} + engines: {node: '>=8'} + + is-glob@4.0.3: + resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} + engines: {node: '>=0.10.0'} + + is-lambda@1.0.1: + resolution: {integrity: sha512-z7CMFGNrENq5iFB9Bqo64Xk6Y9sg+epq1myIcdHaGnbMTYOxvzsEtdYqQUylB7LxfkvgrrjP32T6Ywciio9UIQ==} + + is-number@7.0.0: + resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} + engines: {node: '>=0.12.0'} + + is-promise@2.2.2: + resolution: {integrity: sha512-+lP4/6lKUBfQjZ2pdxThZvLUAafmZb8OAxFb8XXtiQmS35INgr85hdOGoEs124ez1FCnZJt6jau/T+alh58QFQ==} + + isexe@2.0.0: + resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} + + jackspeak@2.3.6: + resolution: {integrity: sha512-N3yCS/NegsOBokc8GAdM8UcmfsKiSS8cipheD/nivzr700H+nsMOxJjQnvwOcRYVuFkdH0wGUvW2WbXGmrZGbQ==} + engines: {node: '>=14'} + + jiti@1.21.0: + resolution: {integrity: sha512-gFqAIbuKyyso/3G2qhiO2OM6shY6EPP/R0+mkDbyspxKazh8BXDC5FiFsUjlczgdNz/vfra0da2y+aHrusLG/Q==} + hasBin: true + + js-base64@3.7.7: + resolution: {integrity: sha512-7rCnleh0z2CkXhH67J8K1Ytz0b2Y+yxTPL+/KOJoa20hfnVQ/3/T6W/KflYI4bRHRagNeXeU2bkNGI3v1oS/lw==} + + js-tokens@4.0.0: + resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} + + jsbn@1.1.0: + resolution: {integrity: sha512-4bYVV3aAMtDTTu4+xsDYa6sy9GyJ69/amsu9sYF2zqjiEoZA5xJi3BrfX3uY+/IekIu7MwdObdbDWpoZdBv3/A==} + + json-diff@0.9.0: + resolution: {integrity: sha512-cVnggDrVkAAA3OvFfHpFEhOnmcsUpleEKq4d4O8sQWWSH40MBrWstKigVB1kGrgLWzuom+7rRdaCsnBD6VyObQ==} + hasBin: true + + libsql@0.4.6: + resolution: {integrity: sha512-F5M+ltteK6dCcpjMahrkgT96uFJvVI8aQ4r9f2AzHQjC7BkAYtvfMSTWGvRBezRgMUIU2h1Sy0pF9nOGOD5iyA==} + os: [darwin, linux, win32] + + lilconfig@2.1.0: + resolution: {integrity: sha512-utWOt/GHzuUxnLKxB6dk81RoOeoNeHgbrXiuGk4yyF5qlRz+iIVWu56E2fqGHFrXz0QNUhLB/8nKqvRH66JKGQ==} + engines: {node: '>=10'} + + lilconfig@3.1.1: + resolution: {integrity: sha512-O18pf7nyvHTckunPWCV1XUNXU1piu01y2b7ATJ0ppkUkk8ocqVWBrYjJBCwHDjD/ZWcfyrA0P4gKhzWGi5EINQ==} + engines: {node: '>=14'} + + lines-and-columns@1.2.4: + resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} + + lodash.throttle@4.1.1: + resolution: {integrity: sha512-wIkUCfVKpVsWo3JSZlc+8MB5it+2AN5W8J7YVMST30UrvcQNZ1Okbj+rbVniijTWE6FGYy4XJq/rHkas8qJMLQ==} + + lodash@4.17.21: + resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} + + loose-envify@1.4.0: + resolution: {integrity: sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==} + hasBin: true + + lru-cache@10.2.2: + resolution: {integrity: sha512-9hp3Vp2/hFQUiIwKo8XCeFVnrg8Pk3TYNPIR7tJADKi5YfcF7vEaK7avFHTlSy3kOKYaJQaalfEo6YuXdceBOQ==} + engines: {node: 14 || >=16.14} + + lru-cache@6.0.0: + resolution: {integrity: sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==} + engines: {node: '>=10'} + + lru-queue@0.1.0: + resolution: {integrity: sha512-BpdYkt9EvGl8OfWHDQPISVpcl5xZthb+XPsbELj5AQXxIC8IriDZIQYjBJPEm5rS420sjZ0TLEzRcq5KdBhYrQ==} + + lucia@3.2.1: + resolution: {integrity: sha512-yIVBS/wU3R+8cLClh2ksBNxqHkAd0VUcjvib53azkSdRT1cPkKuFglkxFsghuspaioX+AHhmIECEkdOz/vIJsQ==} + + lucide-react@0.378.0: + resolution: {integrity: sha512-u6EPU8juLUk9ytRcyapkWI18epAv3RU+6+TC23ivjR0e+glWKBobFeSgRwOIJihzktILQuy6E0E80P2jVTDR5g==} + peerDependencies: + react: ^16.5.1 || ^17.0.0 || ^18.0.0 + + luxon@3.5.0: + resolution: {integrity: sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ==} + engines: {node: '>=12'} + + make-error@1.3.6: + resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==} + + make-fetch-happen@9.1.0: + resolution: {integrity: sha512-+zopwDy7DNknmwPQplem5lAZX/eCOzSvSNNcSKm5eVwTkOBzoktEfXsa9L23J/GIRhxRsaxzkPEhrJEpE2F4Gg==} + engines: {node: '>= 10'} + + memfs-browser@3.5.10302: + resolution: {integrity: sha512-JJTc/nh3ig05O0gBBGZjTCPOyydaTxNF0uHYBrcc1gHNnO+KIHIvo0Y1FKCJsaei6FCl8C6xfQomXqu+cuzkIw==} + + memfs@3.5.3: + resolution: {integrity: sha512-UERzLsxzllchadvbPs5aolHh65ISpKpM+ccLbOJ8/vvpBKmAWf+la7dXFy7Mr0ySHbdHrFv5kGFCUHHe6GFEmw==} + engines: {node: '>= 4.0.0'} + + memoizee@0.4.15: + resolution: {integrity: sha512-UBWmJpLZd5STPm7PMUlOw/TSy972M+z8gcyQ5veOnSDRREz/0bmpyTfKt3/51DhEBqCZQn1udM/5flcSPYhkdQ==} + + merge2@1.4.1: + resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==} + engines: {node: '>= 8'} + + micromatch@4.0.5: + resolution: {integrity: sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==} + engines: {node: '>=8.6'} + + mimic-response@3.1.0: + resolution: {integrity: sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==} + engines: {node: '>=10'} + + mini-svg-data-uri@1.4.4: + resolution: {integrity: sha512-r9deDe9p5FJUPZAk3A59wGH7Ii9YrjjWw0jmw/liSbHl2CHiyXj6FcDXDu2K3TjVAXqiJdaw3xxwlZZr9E6nHg==} + hasBin: true + + minimatch@3.1.2: + resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} + + minimatch@5.1.6: + resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==} + engines: {node: '>=10'} + + minimatch@9.0.4: + resolution: {integrity: sha512-KqWh+VchfxcMNRAJjj2tnsSJdNbHsVgnkBhTNrW7AjVo6OvLtxw8zfT9oLw1JSohlFzJ8jCoTgaoXvJ+kHt6fw==} + engines: {node: '>=16 || 14 >=14.17'} + + minimist@1.2.8: + resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} + + minipass-collect@1.0.2: + resolution: {integrity: sha512-6T6lH0H8OG9kITm/Jm6tdooIbogG9e0tLgpY6mphXSm/A9u8Nq1ryBG+Qspiub9LjWlBPsPS3tWQ/Botq4FdxA==} + engines: {node: '>= 8'} + + minipass-fetch@1.4.1: + resolution: {integrity: sha512-CGH1eblLq26Y15+Azk7ey4xh0J/XfJfrCox5LDJiKqI2Q2iwOLOKrlmIaODiSQS8d18jalF6y2K2ePUm0CmShw==} + engines: {node: '>=8'} + + minipass-flush@1.0.5: + resolution: {integrity: sha512-JmQSYYpPUqX5Jyn1mXaRwOda1uQ8HP5KAT/oDSLCzt1BYRhQU0/hDtsB1ufZfEEzMZ9aAVmsBw8+FWsIXlClWw==} + engines: {node: '>= 8'} + + minipass-pipeline@1.2.4: + resolution: {integrity: sha512-xuIq7cIOt09RPRJ19gdi4b+RiNvDFYe5JH+ggNvBqGqpQXcru3PcRmOZuHBKWK1Txf9+cQ+HMVN4d6z46LZP7A==} + engines: {node: '>=8'} + + minipass-sized@1.0.3: + resolution: {integrity: sha512-MbkQQ2CTiBMlA2Dm/5cY+9SWFEN8pzzOXi6rlM5Xxq0Yqbda5ZQy9sU75a673FE9ZK0Zsbr6Y5iP6u9nktfg2g==} + engines: {node: '>=8'} + + minipass@3.3.6: + resolution: {integrity: sha512-DxiNidxSEK+tHG6zOIklvNOwm3hvCrbUrdtzY74U6HKTJxvIDfOUL5W5P2Ghd3DTkhhKPYGqeNUIh5qcM4YBfw==} + engines: {node: '>=8'} + + minipass@5.0.0: + resolution: {integrity: sha512-3FnjYuehv9k6ovOEbyOswadCDPX1piCfhV8ncmYtHOjuPwylVWsghTLo7rabjC3Rx5xD4HDx8Wm1xnMF7S5qFQ==} + engines: {node: '>=8'} + + minipass@7.1.1: + resolution: {integrity: sha512-UZ7eQ+h8ywIRAW1hIEl2AqdwzJucU/Kp59+8kkZeSvafXhZjul247BvIJjEVFVeON6d7lM46XX1HXCduKAS8VA==} + engines: {node: '>=16 || 14 >=14.17'} + + minizlib@2.1.2: + resolution: {integrity: sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==} + engines: {node: '>= 8'} + + mkdirp-classic@0.5.3: + resolution: {integrity: sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==} + + mkdirp@1.0.4: + resolution: {integrity: sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==} + engines: {node: '>=10'} + hasBin: true + + ms@2.1.2: + resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} + + mz@2.7.0: + resolution: {integrity: sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==} + + nanoid@3.3.7: + resolution: {integrity: sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==} + engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} + hasBin: true + + napi-build-utils@1.0.2: + resolution: {integrity: sha512-ONmRUqK7zj7DWX0D9ADe03wbwOBZxNAfF20PlGfCWQcD3+/MakShIHrMqx9YwPTfxDdF1zLeL+RGZiR9kGMLdg==} + + negotiator@0.6.4: + resolution: {integrity: sha512-myRT3DiWPHqho5PrJaIRyaMv2kgYf0mUVgBNOYMuCH5Ki1yEiQaf/ZJuQ62nvpc44wL5WDbTX7yGJi1Neevw8w==} + engines: {node: '>= 0.6'} + + next-themes@0.3.0: + resolution: {integrity: sha512-/QHIrsYpd6Kfk7xakK4svpDI5mmXP0gfvCoJdGpZQ2TOrQZmsW0QxjaiLn8wbIKjtm4BTSqLoix4lxYYOnLJ/w==} + peerDependencies: + react: ^16.8 || ^17 || ^18 + react-dom: ^16.8 || ^17 || ^18 + + next-tick@1.1.0: + resolution: {integrity: sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ==} + + next@14.2.3: + resolution: {integrity: sha512-dowFkFTR8v79NPJO4QsBUtxv0g9BrS/phluVpMAt2ku7H+cbcBJlopXjkWlwxrk/xGqMemr7JkGPGemPrLLX7A==} + engines: {node: '>=18.17.0'} + hasBin: true + peerDependencies: + '@opentelemetry/api': ^1.1.0 + '@playwright/test': ^1.41.2 + react: ^18.2.0 + react-dom: ^18.2.0 + sass: ^1.3.0 + peerDependenciesMeta: + '@opentelemetry/api': + optional: true + '@playwright/test': + optional: true + sass: + optional: true + + node-abi@3.62.0: + resolution: {integrity: sha512-CPMcGa+y33xuL1E0TcNIu4YyaZCxnnvkVaEXrsosR3FxN+fV8xvb7Mzpb7IgKler10qeMkE6+Dp8qJhpzdq35g==} + engines: {node: '>=10'} + + node-addon-api@7.1.1: + resolution: {integrity: sha512-5m3bsyrjFWE1xf7nz7YXdN4udnVtXK6/Yfgn5qnahL6bCkf2yKt4k3nuTKAtT4r3IG8JNR2ncsIMdZuAzJjHQQ==} + + node-domexception@1.0.0: + resolution: {integrity: sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==} + engines: {node: '>=10.5.0'} + + node-fetch@3.3.2: + resolution: {integrity: sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + + node-gyp@8.4.1: + resolution: {integrity: sha512-olTJRgUtAb/hOXG0E93wZDs5YiJlgbXxTwQAFHyNlRsXQnYzUaF2aGgujZbw+hR8aF4ZG/rST57bWMWD16jr9w==} + engines: {node: '>= 10.12.0'} + hasBin: true + + nopt@5.0.0: + resolution: {integrity: sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==} + engines: {node: '>=6'} + hasBin: true + + normalize-path@3.0.0: + resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} + engines: {node: '>=0.10.0'} + + npmlog@6.0.2: + resolution: {integrity: sha512-/vBvz5Jfr9dT/aFWd0FIRf+T/Q2WBsLENygUaFUqstqsycmZAP/t5BvFJTK0viFmSUxiUKTUplWy5vt+rvKIxg==} + engines: {node: ^12.13.0 || ^14.15.0 || >=16.0.0} + deprecated: This package is no longer supported. + + object-assign@4.1.1: + resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} + engines: {node: '>=0.10.0'} + + object-hash@3.0.0: + resolution: {integrity: sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==} + engines: {node: '>= 6'} + + once@1.4.0: + resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} + + oslo@1.2.0: + resolution: {integrity: sha512-OoFX6rDsNcOQVAD2gQD/z03u4vEjWZLzJtwkmgfRF+KpQUXwdgEXErD7zNhyowmHwHefP+PM9Pw13pgpHMRlzw==} + + oslo@1.2.1: + resolution: {integrity: sha512-HfIhB5ruTdQv0XX2XlncWQiJ5SIHZ7NHZhVyHth0CSZ/xzge00etRyYy/3wp/Dsu+PkxMC+6+B2lS/GcKoewkA==} + + p-map@4.0.0: + resolution: {integrity: sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==} + engines: {node: '>=10'} + + path-is-absolute@1.0.1: + resolution: {integrity: sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==} + engines: {node: '>=0.10.0'} + + path-key@3.1.1: + resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} + engines: {node: '>=8'} + + path-parse@1.0.7: + resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==} + + path-scurry@1.11.0: + resolution: {integrity: sha512-LNHTaVkzaYaLGlO+0u3rQTz7QrHTFOuKyba9JMTQutkmtNew8dw8wOD7mTU/5fCPZzCWpfW0XnQKzY61P0aTaw==} + engines: {node: '>=16 || 14 >=14.17'} + + picocolors@1.0.0: + resolution: {integrity: sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==} + + picocolors@1.1.0: + resolution: {integrity: sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==} + + picomatch@2.3.1: + resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} + engines: {node: '>=8.6'} + + pify@2.3.0: + resolution: {integrity: sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==} + engines: {node: '>=0.10.0'} + + pirates@4.0.6: + resolution: {integrity: sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==} + engines: {node: '>= 6'} + + postcss-import@15.1.0: + resolution: {integrity: sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==} + engines: {node: '>=14.0.0'} + peerDependencies: + postcss: ^8.0.0 + + postcss-js@4.0.1: + resolution: {integrity: sha512-dDLF8pEO191hJMtlHFPRa8xsizHaM82MLfNkUHdUtVEV3tgTp5oj+8qbEqYM57SLfc74KSbw//4SeJma2LRVIw==} + engines: {node: ^12 || ^14 || >= 16} + peerDependencies: + postcss: ^8.4.21 + + postcss-load-config@4.0.2: + resolution: {integrity: sha512-bSVhyJGL00wMVoPUzAVAnbEoWyqRxkjv64tUl427SKnPrENtq6hJwUojroMz2VB+Q1edmi4IfrAPpami5VVgMQ==} + engines: {node: '>= 14'} + peerDependencies: + postcss: '>=8.0.9' + ts-node: '>=9.0.0' + peerDependenciesMeta: + postcss: + optional: true + ts-node: + optional: true + + postcss-nested@6.0.1: + resolution: {integrity: sha512-mEp4xPMi5bSWiMbsgoPfcP74lsWLHkQbZc3sY+jWYd65CUwXrUaTp0fmNpa01ZcETKlIgUdFN/MpS2xZtqL9dQ==} + engines: {node: '>=12.0'} + peerDependencies: + postcss: ^8.2.14 + + postcss-selector-parser@6.0.16: + resolution: {integrity: sha512-A0RVJrX+IUkVZbW3ClroRWurercFhieevHB38sr2+l9eUClMqome3LmEmnhlNy+5Mr2EYN6B2Kaw9wYdd+VHiw==} + engines: {node: '>=4'} + + postcss-value-parser@4.2.0: + resolution: {integrity: sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==} + + postcss@8.4.31: + resolution: {integrity: sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==} + engines: {node: ^10 || ^12 || >=14} + + postcss@8.4.47: + resolution: {integrity: sha512-56rxCq7G/XfB4EkXq9Egn5GCqugWvDFjafDOThIdMBsI15iqPqR5r15TfSr1YPYeEI19YeaXMCbY6u88Y76GLQ==} + engines: {node: ^10 || ^12 || >=14} + + prebuild-install@7.1.2: + resolution: {integrity: sha512-UnNke3IQb6sgarcZIDU3gbMeTp/9SSU1DAIkil7PrqG1vZlBtY5msYccSKSHDqa3hNg436IXK+SNImReuA1wEQ==} + engines: {node: '>=10'} + hasBin: true + + promise-inflight@1.0.1: + resolution: {integrity: sha512-6zWPyEOFaQBJYcGMHBKTKJ3u6TBsnMFOIZSa6ce1e/ZrrsOlnHRHbabMjLiBYKp+n44X9eUI6VUPaukCXHuG4g==} + peerDependencies: + bluebird: '*' + peerDependenciesMeta: + bluebird: + optional: true + + promise-limit@2.7.0: + resolution: {integrity: sha512-7nJ6v5lnJsXwGprnGXga4wx6d1POjvi5Qmf1ivTRxTjH4Z/9Czja/UCMLVmB9N93GeWOU93XaFaEt6jbuoagNw==} + + promise-retry@2.0.1: + resolution: {integrity: sha512-y+WKFlBR8BGXnsNlIHFGPZmyDf3DFMoLhaflAnyZgV6rG6xu+JwesTo2Q9R6XwYmtmwAFCkAk3e35jEdoeh/3g==} + engines: {node: '>=10'} + + prop-types@15.8.1: + resolution: {integrity: sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==} + + pump@3.0.0: + resolution: {integrity: sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==} + + queue-microtask@1.2.3: + resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} + + rc@1.2.8: + resolution: {integrity: sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==} + hasBin: true + + react-day-picker@8.10.1: + resolution: {integrity: sha512-TMx7fNbhLk15eqcMt+7Z7S2KF7mfTId/XJDjKE8f+IUcFn0l08/kI4FiYTL/0yuOLmEcbR4Fwe3GJf/NiiMnPA==} + peerDependencies: + date-fns: ^2.28.0 || ^3.0.0 + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + + react-dom@18.3.1: + resolution: {integrity: sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==} + peerDependencies: + react: ^18.3.1 + + react-hook-form@7.53.0: + resolution: {integrity: sha512-M1n3HhqCww6S2hxLxciEXy2oISPnAzxY7gvwVPrtlczTM/1dDadXgUxDpHMrMTblDOcm/AXtXxHwZ3jpg1mqKQ==} + engines: {node: '>=18.0.0'} + peerDependencies: + react: ^16.8.0 || ^17 || ^18 || ^19 + + react-if@4.1.5: + resolution: {integrity: sha512-Uk+Ub2gC83PAakuU4+7iLdTEP4LPi2ihNEPCtz/vr8SLGbzkMApbpYbkDZ5z9zYXurd0gg+EK/bpOLFFC1r1eQ==} + engines: {node: '>=12'} + peerDependencies: + react: ^16.x || ^17.x || ^18.x + + react-is@16.13.1: + resolution: {integrity: sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==} + + react-is@18.3.1: + resolution: {integrity: sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==} + + react-remove-scroll-bar@2.3.6: + resolution: {integrity: sha512-DtSYaao4mBmX+HDo5YWYdBWQwYIQQshUV/dVxFxK+KM26Wjwp1gZ6rv6OC3oujI6Bfu6Xyg3TwK533AQutsn/g==} + engines: {node: '>=10'} + peerDependencies: + '@types/react': ^16.8.0 || ^17.0.0 || ^18.0.0 + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + peerDependenciesMeta: + '@types/react': + optional: true + + react-remove-scroll@2.6.0: + resolution: {integrity: sha512-I2U4JVEsQenxDAKaVa3VZ/JeJZe0/2DxPWL8Tj8yLKctQJQiZM52pn/GWFpSp8dftjM3pSAHVJZscAnC/y+ySQ==} + engines: {node: '>=10'} + peerDependencies: + '@types/react': ^16.8.0 || ^17.0.0 || ^18.0.0 + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + peerDependenciesMeta: + '@types/react': + optional: true + + react-smooth@4.0.1: + resolution: {integrity: sha512-OE4hm7XqR0jNOq3Qmk9mFLyd6p2+j6bvbPJ7qlB7+oo0eNcL2l7WQzG6MBnT3EXY6xzkLMUBec3AfewJdA0J8w==} + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 + + react-style-singleton@2.2.1: + resolution: {integrity: sha512-ZWj0fHEMyWkHzKYUr2Bs/4zU6XLmq9HsgBURm7g5pAVfyn49DgUiNgY2d4lXRlYSiCif9YBGpQleewkcqddc7g==} + engines: {node: '>=10'} + peerDependencies: + '@types/react': ^16.8.0 || ^17.0.0 || ^18.0.0 + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + peerDependenciesMeta: + '@types/react': + optional: true + + react-timer-hook@3.0.7: + resolution: {integrity: sha512-ATpNcU+PQRxxfNBPVqce2+REtjGAlwmfoNQfcEBMZFxPj0r3GYdKhyPHdStvqrejejEi0QvqaJZjy2lBlFvAsA==} + peerDependencies: + react: '>=16.8.0' + + react-transition-group@4.4.5: + resolution: {integrity: sha512-pZcd1MCJoiKiBR2NRxeCRg13uCXbydPnmB4EOeRrY7480qNWO8IIgQG6zlDkm6uRMsURXPuKq0GWtiM59a5Q6g==} + peerDependencies: + react: '>=16.6.0' + react-dom: '>=16.6.0' + + react-transition-state@2.1.2: + resolution: {integrity: sha512-RkDYBkj1V1ZqBA5AwQPrMt2Uagwsx6b//GVJdRDhs/t0o66w2nhQiyHyFGQEI60mgtbaIdLm8yhBRCvhA+FxEg==} + peerDependencies: + react: '>=16.8.0' + react-dom: '>=16.8.0' + + react@18.3.1: + resolution: {integrity: sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==} + engines: {node: '>=0.10.0'} + + read-cache@1.0.0: + resolution: {integrity: sha512-Owdv/Ft7IjOgm/i0xvNDZ1LrRANRfew4b2prF3OWMQLxLfu3bS8FVhCsrSCMK4lR56Y9ya+AThoTpDCTxCmpRA==} + + readable-stream@3.6.2: + resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} + engines: {node: '>= 6'} + + readdirp@3.6.0: + resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} + engines: {node: '>=8.10.0'} + + recharts-scale@0.4.5: + resolution: {integrity: sha512-kivNFO+0OcUNu7jQquLXAxz1FIwZj8nrj+YkOKc5694NbjCvcT6aSZiIzNzd2Kul4o4rTto8QVR9lMNtxD4G1w==} + + recharts@2.13.3: + resolution: {integrity: sha512-YDZ9dOfK9t3ycwxgKbrnDlRC4BHdjlY73fet3a0C1+qGMjXVZe6+VXmpOIIhzkje5MMEL8AN4hLIe4AMskBzlA==} + engines: {node: '>=14'} + peerDependencies: + react: ^16.0.0 || ^17.0.0 || ^18.0.0 + react-dom: ^16.0.0 || ^17.0.0 || ^18.0.0 + + regenerator-runtime@0.14.1: + resolution: {integrity: sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==} + + regression@2.0.1: + resolution: {integrity: sha512-A4XYsc37dsBaNOgEjkJKzfJlE394IMmUPlI/p3TTI9u3T+2a+eox5Pr/CPUqF0eszeWZJPAc6QkroAhuUpWDJQ==} + + resolve-pkg-maps@1.0.0: + resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} + + resolve@1.22.8: + resolution: {integrity: sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==} + hasBin: true + + retry@0.12.0: + resolution: {integrity: sha512-9LkiTwjUh6rT555DtE9rTX+BKByPfrMzEAtnlEtdEwr3Nkffwiihqe2bWADg+OQRjt9gl6ICdmB/ZFDCGAtSow==} + engines: {node: '>= 4'} + + reusify@1.0.4: + resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==} + engines: {iojs: '>=1.0.0', node: '>=0.10.0'} + + rimraf@3.0.2: + resolution: {integrity: sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==} + deprecated: Rimraf versions prior to v4 are no longer supported + hasBin: true + + run-parallel@1.2.0: + resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} + + safe-buffer@5.2.1: + resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} + + safer-buffer@2.1.2: + resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==} + + scheduler@0.23.2: + resolution: {integrity: sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==} + + semver@7.6.2: + resolution: {integrity: sha512-FNAIBWCx9qcRhoHcgcJ0gvU7SN1lYU2ZXuSfl04bSC5OpvDHFyJCjdNHomPXxjQlCBU67YW64PzY7/VIEH7F2w==} + engines: {node: '>=10'} + hasBin: true + + set-blocking@2.0.0: + resolution: {integrity: sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==} + + shebang-command@2.0.0: + resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} + engines: {node: '>=8'} + + shebang-regex@3.0.0: + resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} + engines: {node: '>=8'} + + signal-exit@3.0.7: + resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==} + + signal-exit@4.1.0: + resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} + engines: {node: '>=14'} + + simple-concat@1.0.1: + resolution: {integrity: sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==} + + simple-get@4.0.1: + resolution: {integrity: sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==} + + sisteransi@1.0.5: + resolution: {integrity: sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==} + + smart-buffer@4.2.0: + resolution: {integrity: sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==} + engines: {node: '>= 6.0.0', npm: '>= 3.0.0'} + + socks-proxy-agent@6.2.1: + resolution: {integrity: sha512-a6KW9G+6B3nWZ1yB8G7pJwL3ggLy1uTzKAgCb7ttblwqdz9fMGJUuTy3uFzEP48FAs9FLILlmzDlE2JJhVQaXQ==} + engines: {node: '>= 10'} + + socks@2.8.3: + resolution: {integrity: sha512-l5x7VUUWbjVFbafGLxPWkYsHIhEvmF85tbIeFZWc8ZPtoMyybuEhL7Jye/ooC4/d48FgOjSJXgsF/AJPYCW8Zw==} + engines: {node: '>= 10.0.0', npm: '>= 3.0.0'} + + sonner@1.5.0: + resolution: {integrity: sha512-FBjhG/gnnbN6FY0jaNnqZOMmB73R+5IiyYAw8yBj7L54ER7HB3fOSE5OFiQiE2iXWxeXKvg6fIP4LtVppHEdJA==} + peerDependencies: + react: ^18.0.0 + react-dom: ^18.0.0 + + source-map-js@1.2.0: + resolution: {integrity: sha512-itJW8lvSA0TXEphiRoawsCksnlf8SyvmFzIhltqAHluXd88pkCd+cXJVHTDwdCr0IzwptSm035IHQktUu1QUMg==} + engines: {node: '>=0.10.0'} + + source-map-js@1.2.1: + resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==} + engines: {node: '>=0.10.0'} + + source-map-support@0.5.21: + resolution: {integrity: sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==} + + source-map@0.6.1: + resolution: {integrity: sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==} + engines: {node: '>=0.10.0'} + + sprintf-js@1.1.3: + resolution: {integrity: sha512-Oo+0REFV59/rz3gfJNKQiBlwfHaSESl1pcGyABQsnnIfWOFt6JNj5gCog2U6MLZ//IGYD+nA8nI+mTShREReaA==} + + sqlite3@5.1.7: + resolution: {integrity: sha512-GGIyOiFaG+TUra3JIfkI/zGP8yZYLPQ0pl1bH+ODjiX57sPhrLU5sQJn1y9bDKZUFYkX1crlrPfSYt0BKKdkog==} + + sqlite@5.1.1: + resolution: {integrity: sha512-oBkezXa2hnkfuJwUo44Hl9hS3er+YFtueifoajrgidvqsJRQFpc5fKoAkAor1O5ZnLoa28GBScfHXs8j0K358Q==} + + ssri@8.0.1: + resolution: {integrity: sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==} + engines: {node: '>= 8'} + + streamsearch@1.1.0: + resolution: {integrity: sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==} + engines: {node: '>=10.0.0'} + + string-width@4.2.3: + resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} + engines: {node: '>=8'} + + string-width@5.1.2: + resolution: {integrity: sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==} + engines: {node: '>=12'} + + string_decoder@1.3.0: + resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} + + strip-ansi@6.0.1: + resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} + engines: {node: '>=8'} + + strip-ansi@7.1.0: + resolution: {integrity: sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==} + engines: {node: '>=12'} + + strip-json-comments@2.0.1: + resolution: {integrity: sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==} + engines: {node: '>=0.10.0'} + + styled-jsx@5.1.1: + resolution: {integrity: sha512-pW7uC1l4mBZ8ugbiZrcIsiIvVx1UmTfw7UkC3Um2tmfUq9Bhk8IiyEIPl6F8agHgjzku6j0xQEZbfA5uSgSaCw==} + engines: {node: '>= 12.0.0'} + peerDependencies: + '@babel/core': '*' + babel-plugin-macros: '*' + react: '>= 16.8.0 || 17.x.x || ^18.0.0-0' + peerDependenciesMeta: + '@babel/core': + optional: true + babel-plugin-macros: + optional: true + + sucrase@3.35.0: + resolution: {integrity: sha512-8EbVDiu9iN/nESwxeSxDKe0dunta1GOlHufmSSXxMD2z2/tMZpDMpvXQGsc+ajGo8y2uYUmixaSRUc/QPoQ0GA==} + engines: {node: '>=16 || 14 >=14.17'} + hasBin: true + + supports-preserve-symlinks-flag@1.0.0: + resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} + engines: {node: '>= 0.4'} + + swr@2.2.5: + resolution: {integrity: sha512-QtxqyclFeAsxEUeZIYmsaQ0UjimSq1RZ9Un7I68/0ClKK/U3LoyQunwkQfJZr2fc22DfIXLNDc2wFyTEikCUpg==} + peerDependencies: + react: ^16.11.0 || ^17.0.0 || ^18.0.0 + + tabbable@6.2.0: + resolution: {integrity: sha512-Cat63mxsVJlzYvN51JmVXIgNoUokrIaT2zLclCXjRd8boZ0004U4KCs/sToJ75C6sdlByWxpYnb5Boif1VSFew==} + + tailwind-merge@2.5.3: + resolution: {integrity: sha512-d9ZolCAIzom1nf/5p4LdD5zvjmgSxY0BGgdSvmXIoMYAiPdAW/dSpP7joCDYFY7r/HkEa2qmPtkgsu0xjQeQtw==} + + tailwindcss-animate@1.0.7: + resolution: {integrity: sha512-bl6mpH3T7I3UFxuvDEXLxy/VuFxBk5bbzplh7tXI68mwMokNYd1t9qPBHlnyTwfa4JGC4zP516I1hYYtQ/vspA==} + peerDependencies: + tailwindcss: '>=3.0.0 || insiders' + + tailwindcss@3.4.13: + resolution: {integrity: sha512-KqjHOJKogOUt5Bs752ykCeiwvi0fKVkr5oqsFNt/8px/tA8scFPIlkygsf6jXrfCqGHz7VflA6+yytWuM+XhFw==} + engines: {node: '>=14.0.0'} + hasBin: true + + tar-fs@2.1.1: + resolution: {integrity: sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==} + + tar-stream@2.2.0: + resolution: {integrity: sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==} + engines: {node: '>=6'} + + tar@6.2.1: + resolution: {integrity: sha512-DZ4yORTwrbTj/7MZYq2w+/ZFdI6OZ/f9SFHR+71gIVUZhOQPHzVCLpvRnPgyaMpfWxxk/4ONva3GQSyNIKRv6A==} + engines: {node: '>=10'} + + thenify-all@1.6.0: + resolution: {integrity: sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==} + engines: {node: '>=0.8'} + + thenify@3.3.1: + resolution: {integrity: sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==} + + timers-ext@0.1.7: + resolution: {integrity: sha512-b85NUNzTSdodShTIbky6ZF02e8STtVVfD+fu4aXXShEELpozH+bCpJLYMPZbsABN2wDH7fJpqIoXxJpzbf0NqQ==} + + tiny-invariant@1.3.3: + resolution: {integrity: sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg==} + + to-regex-range@5.0.1: + resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} + engines: {node: '>=8.0'} + + ts-interface-checker@0.1.13: + resolution: {integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==} + + ts-node@10.9.2: + resolution: {integrity: sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==} + hasBin: true + peerDependencies: + '@swc/core': '>=1.2.50' + '@swc/wasm': '>=1.2.50' + '@types/node': '*' + typescript: '>=2.7' + peerDependenciesMeta: + '@swc/core': + optional: true + '@swc/wasm': + optional: true + + tslib@2.6.2: + resolution: {integrity: sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==} + + tunnel-agent@0.6.0: + resolution: {integrity: sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==} + + type@2.7.2: + resolution: {integrity: sha512-dzlvlNlt6AXU7EBSfpAscydQ7gXB+pPGsPnfJnZpiNJBDj7IaJzQlBZYGdEi4R9HmPdBv2XmWJ6YUtoTa7lmCw==} + + typescript@5.6.3: + resolution: {integrity: sha512-hjcS1mhfuyi4WW8IWtjP7brDrG2cuDZukyrYrSauoXGNgx0S7zceP07adYkJycEr56BOUTNPzbInooiN3fn1qw==} + engines: {node: '>=14.17'} + hasBin: true + + undici-types@6.19.8: + resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} + + unique-filename@1.1.1: + resolution: {integrity: sha512-Vmp0jIp2ln35UTXuryvjzkjGdRyf9b2lTXuSYUiPmzRcl3FDtYqAwOnTJkAngD9SWhnoJzDbTKwaOrZ+STtxNQ==} + + unique-slug@2.0.2: + resolution: {integrity: sha512-zoWr9ObaxALD3DOPfjPSqxt4fnZiWblxHIgeWqW8x7UqDzEtHEQLzji2cuJYQFCU6KmoJikOYAZlrTHHebjx2w==} + + use-callback-ref@1.3.2: + resolution: {integrity: sha512-elOQwe6Q8gqZgDA8mrh44qRTQqpIHDcZ3hXTLjBe1i4ph8XpNJnO+aQf3NaG+lriLopI4HMx9VjQLfPQ6vhnoA==} + engines: {node: '>=10'} + peerDependencies: + '@types/react': ^16.8.0 || ^17.0.0 || ^18.0.0 + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + peerDependenciesMeta: + '@types/react': + optional: true + + use-debounce@10.0.3: + resolution: {integrity: sha512-DxQSI9ZKso689WM1mjgGU3ozcxU1TJElBJ3X6S4SMzMNcm2lVH0AHmyXB+K7ewjz2BSUKJTDqTcwtSMRfB89dg==} + engines: {node: '>= 16.0.0'} + peerDependencies: + react: '*' + + use-sidecar@1.1.2: + resolution: {integrity: sha512-epTbsLuzZ7lPClpz2TyryBfztm7m+28DlEv2ZCQ3MDr5ssiwyOwGH/e5F9CkfWjJ1t4clvI58yF822/GUkjjhw==} + engines: {node: '>=10'} + peerDependencies: + '@types/react': ^16.9.0 || ^17.0.0 || ^18.0.0 + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + peerDependenciesMeta: + '@types/react': + optional: true + + use-sync-external-store@1.2.2: + resolution: {integrity: sha512-PElTlVMwpblvbNqQ82d2n6RjStvdSoNe9FG28kNfz3WiXilJm4DdNkEzRhCZuIDwY8U08WVihhGR5iRqAwfDiw==} + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + + util-deprecate@1.0.2: + resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} + + uuid@11.0.2: + resolution: {integrity: sha512-14FfcOJmqdjbBPdDjFQyk/SdT4NySW4eM0zcG+HqbHP5jzuH56xO3J1DGhgs/cEMCfwYi3HQI1gnTO62iaG+tQ==} + hasBin: true + + v8-compile-cache-lib@3.0.1: + resolution: {integrity: sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==} + + victory-vendor@36.9.2: + resolution: {integrity: sha512-PnpQQMuxlwYdocC8fIJqVXvkeViHYzotI+NJrCuav0ZYFoq912ZHBk3mCeuj+5/VpodOjPe1z0Fk2ihgzlXqjQ==} + + web-streams-polyfill@3.3.3: + resolution: {integrity: sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==} + engines: {node: '>= 8'} + + which@2.0.2: + resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} + engines: {node: '>= 8'} + hasBin: true + + wide-align@1.1.5: + resolution: {integrity: sha512-eDMORYaPNZ4sQIuuYPDHdQvf4gyCF9rEEV/yPxGfwPkRodwEgiMUUXTx/dex+Me0wxx53S+NgUHaP7y3MGlDmg==} + + wordwrap@1.0.0: + resolution: {integrity: sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==} + + wrap-ansi@7.0.0: + resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} + engines: {node: '>=10'} + + wrap-ansi@8.1.0: + resolution: {integrity: sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==} + engines: {node: '>=12'} + + wrappy@1.0.2: + resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} + + ws@8.18.0: + resolution: {integrity: sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw==} + engines: {node: '>=10.0.0'} + peerDependencies: + bufferutil: ^4.0.1 + utf-8-validate: '>=5.0.2' + peerDependenciesMeta: + bufferutil: + optional: true + utf-8-validate: + optional: true + + yallist@4.0.0: + resolution: {integrity: sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==} + + yaml@2.4.2: + resolution: {integrity: sha512-B3VqDZ+JAg1nZpaEmWtTXUlBneoGx6CPM9b0TENK6aoSu5t73dItudwdgmi6tHlIZZId4dZ9skcAQ2UbcyAeVA==} + engines: {node: '>= 14'} + hasBin: true + + yn@3.1.1: + resolution: {integrity: sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==} + engines: {node: '>=6'} + + zod@3.23.8: + resolution: {integrity: sha512-XBx9AXhXktjUqnepgTiE5flcKIYWi/rme0Eaj+5Y0lftuGBq+jyRu/md4WnuxqgP1ubdpNCsYEYPxrzVHD8d6g==} + +snapshots: + + '@alloc/quick-lru@5.2.0': {} + + '@babel/runtime@7.24.5': + dependencies: + regenerator-runtime: 0.14.1 + + '@biomejs/biome@1.9.3': + optionalDependencies: + '@biomejs/cli-darwin-arm64': 1.9.3 + '@biomejs/cli-darwin-x64': 1.9.3 + '@biomejs/cli-linux-arm64': 1.9.3 + '@biomejs/cli-linux-arm64-musl': 1.9.3 + '@biomejs/cli-linux-x64': 1.9.3 + '@biomejs/cli-linux-x64-musl': 1.9.3 + '@biomejs/cli-win32-arm64': 1.9.3 + '@biomejs/cli-win32-x64': 1.9.3 + + '@biomejs/cli-darwin-arm64@1.9.3': + optional: true + + '@biomejs/cli-darwin-x64@1.9.3': + optional: true + + '@biomejs/cli-linux-arm64-musl@1.9.3': + optional: true + + '@biomejs/cli-linux-arm64@1.9.3': + optional: true + + '@biomejs/cli-linux-x64-musl@1.9.3': + optional: true + + '@biomejs/cli-linux-x64@1.9.3': + optional: true + + '@biomejs/cli-win32-arm64@1.9.3': + optional: true + + '@biomejs/cli-win32-x64@1.9.3': + optional: true + + '@cspotcode/source-map-support@0.8.1': + dependencies: + '@jridgewell/trace-mapping': 0.3.9 + + '@emnapi/core@0.45.0': + dependencies: + tslib: 2.6.2 + optional: true + + '@emnapi/runtime@0.45.0': + dependencies: + tslib: 2.6.2 + optional: true + + '@esbuild-kit/core-utils@3.3.2': + dependencies: + esbuild: 0.18.20 + source-map-support: 0.5.21 + + '@esbuild-kit/esm-loader@2.6.5': + dependencies: + '@esbuild-kit/core-utils': 3.3.2 + get-tsconfig: 4.7.5 + + '@esbuild/aix-ppc64@0.19.12': + optional: true + + '@esbuild/android-arm64@0.18.20': + optional: true + + '@esbuild/android-arm64@0.19.12': + optional: true + + '@esbuild/android-arm@0.18.20': + optional: true + + '@esbuild/android-arm@0.19.12': + optional: true + + '@esbuild/android-x64@0.18.20': + optional: true + + '@esbuild/android-x64@0.19.12': + optional: true + + '@esbuild/darwin-arm64@0.18.20': + optional: true + + '@esbuild/darwin-arm64@0.19.12': + optional: true + + '@esbuild/darwin-x64@0.18.20': + optional: true + + '@esbuild/darwin-x64@0.19.12': + optional: true + + '@esbuild/freebsd-arm64@0.18.20': + optional: true + + '@esbuild/freebsd-arm64@0.19.12': + optional: true + + '@esbuild/freebsd-x64@0.18.20': + optional: true + + '@esbuild/freebsd-x64@0.19.12': + optional: true + + '@esbuild/linux-arm64@0.18.20': + optional: true + + '@esbuild/linux-arm64@0.19.12': + optional: true + + '@esbuild/linux-arm@0.18.20': + optional: true + + '@esbuild/linux-arm@0.19.12': + optional: true + + '@esbuild/linux-ia32@0.18.20': + optional: true + + '@esbuild/linux-ia32@0.19.12': + optional: true + + '@esbuild/linux-loong64@0.18.20': + optional: true + + '@esbuild/linux-loong64@0.19.12': + optional: true + + '@esbuild/linux-mips64el@0.18.20': + optional: true + + '@esbuild/linux-mips64el@0.19.12': + optional: true + + '@esbuild/linux-ppc64@0.18.20': + optional: true + + '@esbuild/linux-ppc64@0.19.12': + optional: true + + '@esbuild/linux-riscv64@0.18.20': + optional: true + + '@esbuild/linux-riscv64@0.19.12': + optional: true + + '@esbuild/linux-s390x@0.18.20': + optional: true + + '@esbuild/linux-s390x@0.19.12': + optional: true + + '@esbuild/linux-x64@0.18.20': + optional: true + + '@esbuild/linux-x64@0.19.12': + optional: true + + '@esbuild/netbsd-x64@0.18.20': + optional: true + + '@esbuild/netbsd-x64@0.19.12': + optional: true + + '@esbuild/openbsd-x64@0.18.20': + optional: true + + '@esbuild/openbsd-x64@0.19.12': + optional: true + + '@esbuild/sunos-x64@0.18.20': + optional: true + + '@esbuild/sunos-x64@0.19.12': + optional: true + + '@esbuild/win32-arm64@0.18.20': + optional: true + + '@esbuild/win32-arm64@0.19.12': + optional: true + + '@esbuild/win32-ia32@0.18.20': + optional: true + + '@esbuild/win32-ia32@0.19.12': + optional: true + + '@esbuild/win32-x64@0.18.20': + optional: true + + '@esbuild/win32-x64@0.19.12': + optional: true + + '@faker-js/faker@9.2.0': {} + + '@floating-ui/core@1.6.1': + dependencies: + '@floating-ui/utils': 0.2.8 + + '@floating-ui/dom@1.6.5': + dependencies: + '@floating-ui/core': 1.6.1 + '@floating-ui/utils': 0.2.8 + + '@floating-ui/react-dom@1.3.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@floating-ui/dom': 1.6.5 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + '@floating-ui/react-dom@2.0.9(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@floating-ui/dom': 1.6.5 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + '@floating-ui/react-dom@2.1.2(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@floating-ui/dom': 1.6.5 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + '@floating-ui/react@0.19.2(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@floating-ui/react-dom': 1.3.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + aria-hidden: 1.2.4 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + tabbable: 6.2.0 + + '@floating-ui/react@0.26.24(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@floating-ui/react-dom': 2.1.2(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@floating-ui/utils': 0.2.8 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + tabbable: 6.2.0 + + '@floating-ui/utils@0.2.8': {} + + '@gar/promisify@1.1.3': + optional: true + + '@headlessui/react@1.7.19(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@tanstack/react-virtual': 3.5.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + client-only: 0.0.1 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + '@headlessui/react@2.1.10(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@floating-ui/react': 0.26.24(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@react-aria/focus': 3.18.3(react@18.3.1) + '@react-aria/interactions': 3.22.3(react@18.3.1) + '@tanstack/react-virtual': 3.10.8(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + '@headlessui/tailwindcss@0.2.1(tailwindcss@3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3)))': + dependencies: + tailwindcss: 3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3)) + + '@hookform/resolvers@3.9.0(react-hook-form@7.53.0(react@18.3.1))': + dependencies: + react-hook-form: 7.53.0(react@18.3.1) + + '@isaacs/cliui@8.0.2': + dependencies: + string-width: 5.1.2 + string-width-cjs: string-width@4.2.3 + strip-ansi: 7.1.0 + strip-ansi-cjs: strip-ansi@6.0.1 + wrap-ansi: 8.1.0 + wrap-ansi-cjs: wrap-ansi@7.0.0 + + '@jridgewell/gen-mapping@0.3.5': + dependencies: + '@jridgewell/set-array': 1.2.1 + '@jridgewell/sourcemap-codec': 1.4.15 + '@jridgewell/trace-mapping': 0.3.25 + + '@jridgewell/resolve-uri@3.1.2': {} + + '@jridgewell/set-array@1.2.1': {} + + '@jridgewell/sourcemap-codec@1.4.15': {} + + '@jridgewell/trace-mapping@0.3.25': + dependencies: + '@jridgewell/resolve-uri': 3.1.2 + '@jridgewell/sourcemap-codec': 1.4.15 + + '@jridgewell/trace-mapping@0.3.9': + dependencies: + '@jridgewell/resolve-uri': 3.1.2 + '@jridgewell/sourcemap-codec': 1.4.15 + + '@libsql/client@0.14.0': + dependencies: + '@libsql/core': 0.14.0 + '@libsql/hrana-client': 0.7.0 + js-base64: 3.7.7 + libsql: 0.4.6 + promise-limit: 2.7.0 + transitivePeerDependencies: + - bufferutil + - utf-8-validate + + '@libsql/core@0.14.0': + dependencies: + js-base64: 3.7.7 + + '@libsql/darwin-arm64@0.4.6': + optional: true + + '@libsql/darwin-x64@0.4.6': + optional: true + + '@libsql/hrana-client@0.7.0': + dependencies: + '@libsql/isomorphic-fetch': 0.3.1 + '@libsql/isomorphic-ws': 0.1.5 + js-base64: 3.7.7 + node-fetch: 3.3.2 + transitivePeerDependencies: + - bufferutil + - utf-8-validate + + '@libsql/isomorphic-fetch@0.3.1': {} + + '@libsql/isomorphic-ws@0.1.5': + dependencies: + '@types/ws': 8.5.12 + ws: 8.18.0 + transitivePeerDependencies: + - bufferutil + - utf-8-validate + + '@libsql/linux-arm64-gnu@0.4.6': + optional: true + + '@libsql/linux-arm64-musl@0.4.6': + optional: true + + '@libsql/linux-x64-gnu@0.4.6': + optional: true + + '@libsql/linux-x64-musl@0.4.6': + optional: true + + '@libsql/win32-x64-msvc@0.4.6': + optional: true + + '@lucia-auth/adapter-drizzle@1.1.0(drizzle-orm@0.30.10(@libsql/client@0.14.0)(@types/better-sqlite3@7.6.11)(@types/react@18.3.11)(better-sqlite3@9.6.0)(react@18.3.1)(sqlite3@5.1.7))(lucia@3.2.1)': + dependencies: + drizzle-orm: 0.30.10(@libsql/client@0.14.0)(@types/better-sqlite3@7.6.11)(@types/react@18.3.11)(better-sqlite3@9.6.0)(react@18.3.1)(sqlite3@5.1.7) + lucia: 3.2.1 + + '@neon-rs/load@0.0.4': {} + + '@next/env@14.2.3': {} + + '@next/swc-darwin-arm64@14.2.3': + optional: true + + '@next/swc-darwin-x64@14.2.3': + optional: true + + '@next/swc-linux-arm64-gnu@14.2.3': + optional: true + + '@next/swc-linux-arm64-musl@14.2.3': + optional: true + + '@next/swc-linux-x64-gnu@14.2.3': + optional: true + + '@next/swc-linux-x64-musl@14.2.3': + optional: true + + '@next/swc-win32-arm64-msvc@14.2.3': + optional: true + + '@next/swc-win32-ia32-msvc@14.2.3': + optional: true + + '@next/swc-win32-x64-msvc@14.2.3': + optional: true + + '@node-rs/argon2-android-arm-eabi@1.7.0': + optional: true + + '@node-rs/argon2-android-arm64@1.7.0': + optional: true + + '@node-rs/argon2-darwin-arm64@1.7.0': + optional: true + + '@node-rs/argon2-darwin-x64@1.7.0': + optional: true + + '@node-rs/argon2-freebsd-x64@1.7.0': + optional: true + + '@node-rs/argon2-linux-arm-gnueabihf@1.7.0': + optional: true + + '@node-rs/argon2-linux-arm64-gnu@1.7.0': + optional: true + + '@node-rs/argon2-linux-arm64-musl@1.7.0': + optional: true + + '@node-rs/argon2-linux-x64-gnu@1.7.0': + optional: true + + '@node-rs/argon2-linux-x64-musl@1.7.0': + optional: true + + '@node-rs/argon2-wasm32-wasi@1.7.0': + dependencies: + '@emnapi/core': 0.45.0 + '@emnapi/runtime': 0.45.0 + '@tybys/wasm-util': 0.8.3 + memfs-browser: 3.5.10302 + optional: true + + '@node-rs/argon2-win32-arm64-msvc@1.7.0': + optional: true + + '@node-rs/argon2-win32-ia32-msvc@1.7.0': + optional: true + + '@node-rs/argon2-win32-x64-msvc@1.7.0': + optional: true + + '@node-rs/argon2@1.7.0': + optionalDependencies: + '@node-rs/argon2-android-arm-eabi': 1.7.0 + '@node-rs/argon2-android-arm64': 1.7.0 + '@node-rs/argon2-darwin-arm64': 1.7.0 + '@node-rs/argon2-darwin-x64': 1.7.0 + '@node-rs/argon2-freebsd-x64': 1.7.0 + '@node-rs/argon2-linux-arm-gnueabihf': 1.7.0 + '@node-rs/argon2-linux-arm64-gnu': 1.7.0 + '@node-rs/argon2-linux-arm64-musl': 1.7.0 + '@node-rs/argon2-linux-x64-gnu': 1.7.0 + '@node-rs/argon2-linux-x64-musl': 1.7.0 + '@node-rs/argon2-wasm32-wasi': 1.7.0 + '@node-rs/argon2-win32-arm64-msvc': 1.7.0 + '@node-rs/argon2-win32-ia32-msvc': 1.7.0 + '@node-rs/argon2-win32-x64-msvc': 1.7.0 + + '@node-rs/bcrypt-android-arm-eabi@1.9.0': + optional: true + + '@node-rs/bcrypt-android-arm64@1.9.0': + optional: true + + '@node-rs/bcrypt-darwin-arm64@1.9.0': + optional: true + + '@node-rs/bcrypt-darwin-x64@1.9.0': + optional: true + + '@node-rs/bcrypt-freebsd-x64@1.9.0': + optional: true + + '@node-rs/bcrypt-linux-arm-gnueabihf@1.9.0': + optional: true + + '@node-rs/bcrypt-linux-arm64-gnu@1.9.0': + optional: true + + '@node-rs/bcrypt-linux-arm64-musl@1.9.0': + optional: true + + '@node-rs/bcrypt-linux-x64-gnu@1.9.0': + optional: true + + '@node-rs/bcrypt-linux-x64-musl@1.9.0': + optional: true + + '@node-rs/bcrypt-wasm32-wasi@1.9.0': + dependencies: + '@emnapi/core': 0.45.0 + '@emnapi/runtime': 0.45.0 + '@tybys/wasm-util': 0.8.3 + memfs-browser: 3.5.10302 + optional: true + + '@node-rs/bcrypt-win32-arm64-msvc@1.9.0': + optional: true + + '@node-rs/bcrypt-win32-ia32-msvc@1.9.0': + optional: true + + '@node-rs/bcrypt-win32-x64-msvc@1.9.0': + optional: true + + '@node-rs/bcrypt@1.9.0': + optionalDependencies: + '@node-rs/bcrypt-android-arm-eabi': 1.9.0 + '@node-rs/bcrypt-android-arm64': 1.9.0 + '@node-rs/bcrypt-darwin-arm64': 1.9.0 + '@node-rs/bcrypt-darwin-x64': 1.9.0 + '@node-rs/bcrypt-freebsd-x64': 1.9.0 + '@node-rs/bcrypt-linux-arm-gnueabihf': 1.9.0 + '@node-rs/bcrypt-linux-arm64-gnu': 1.9.0 + '@node-rs/bcrypt-linux-arm64-musl': 1.9.0 + '@node-rs/bcrypt-linux-x64-gnu': 1.9.0 + '@node-rs/bcrypt-linux-x64-musl': 1.9.0 + '@node-rs/bcrypt-wasm32-wasi': 1.9.0 + '@node-rs/bcrypt-win32-arm64-msvc': 1.9.0 + '@node-rs/bcrypt-win32-ia32-msvc': 1.9.0 + '@node-rs/bcrypt-win32-x64-msvc': 1.9.0 + + '@nodelib/fs.scandir@2.1.5': + dependencies: + '@nodelib/fs.stat': 2.0.5 + run-parallel: 1.2.0 + + '@nodelib/fs.stat@2.0.5': {} + + '@nodelib/fs.walk@1.2.8': + dependencies: + '@nodelib/fs.scandir': 2.1.5 + fastq: 1.17.1 + + '@npmcli/fs@1.1.1': + dependencies: + '@gar/promisify': 1.1.3 + semver: 7.6.2 + optional: true + + '@npmcli/move-file@1.1.2': + dependencies: + mkdirp: 1.0.4 + rimraf: 3.0.2 + optional: true + + '@oslojs/asn1@1.0.0': + dependencies: + '@oslojs/binary': 1.0.0 + + '@oslojs/binary@1.0.0': {} + + '@oslojs/crypto@1.0.1': + dependencies: + '@oslojs/asn1': 1.0.0 + '@oslojs/binary': 1.0.0 + + '@oslojs/encoding@1.1.0': {} + + '@pkgjs/parseargs@0.11.0': + optional: true + + '@radix-ui/number@1.1.0': {} + + '@radix-ui/primitive@1.1.0': {} + + '@radix-ui/react-alert-dialog@1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-dialog': 1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-slot': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-arrow@1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-avatar@1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/react-context': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-collection@1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-slot': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-compose-refs@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-context@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-context@1.1.1(@types/react@18.3.11)(react@18.3.1)': + dependencies: + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-dialog@1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-dismissable-layer': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-focus-guards': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-focus-scope': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-id': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-portal': 1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-presence': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-slot': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.11)(react@18.3.1) + aria-hidden: 1.2.4 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + react-remove-scroll: 2.6.0(@types/react@18.3.11)(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-direction@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-dismissable-layer@1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-escape-keydown': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-dropdown-menu@2.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-id': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-menu': 2.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-focus-guards@1.1.1(@types/react@18.3.11)(react@18.3.1)': + dependencies: + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-focus-scope@1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-hover-card@1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-dismissable-layer': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-popper': 1.2.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-portal': 1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-presence': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-icons@1.3.0(react@18.3.1)': + dependencies: + react: 18.3.1 + + '@radix-ui/react-id@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-label@2.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-menu@2.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-collection': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-direction': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-dismissable-layer': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-focus-guards': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-focus-scope': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-id': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-popper': 1.2.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-portal': 1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-presence': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-roving-focus': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-slot': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + aria-hidden: 1.2.4 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + react-remove-scroll: 2.6.0(@types/react@18.3.11)(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-popper@1.2.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@floating-ui/react-dom': 2.0.9(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-arrow': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-rect': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-size': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/rect': 1.1.0 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-portal@1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-presence@1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-primitive@2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/react-slot': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-roving-focus@1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-collection': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-direction': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-id': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-scroll-area@1.2.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/number': 1.1.0 + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-direction': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-presence': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-select@2.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/number': 1.1.0 + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-collection': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-direction': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-dismissable-layer': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-focus-guards': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-focus-scope': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-id': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-popper': 1.2.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-portal': 1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-slot': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-previous': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-visually-hidden': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + aria-hidden: 1.2.4 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + react-remove-scroll: 2.6.0(@types/react@18.3.11)(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-slot@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-tabs@1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-context': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-direction': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-id': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-presence': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-roving-focus': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-toast@1.2.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/primitive': 1.1.0 + '@radix-ui/react-collection': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-context': 1.1.1(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-dismissable-layer': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-portal': 1.1.2(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-presence': 1.1.1(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.11)(react@18.3.1) + '@radix-ui/react-visually-hidden': 1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/react-use-callback-ref@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-use-controllable-state@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-use-escape-keydown@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + '@radix-ui/react-use-callback-ref': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-use-layout-effect@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-use-previous@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-use-rect@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + '@radix-ui/rect': 1.1.0 + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-use-size@1.1.0(@types/react@18.3.11)(react@18.3.1)': + dependencies: + '@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.11)(react@18.3.1) + react: 18.3.1 + optionalDependencies: + '@types/react': 18.3.11 + + '@radix-ui/react-visually-hidden@1.1.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.1)(@types/react@18.3.11)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + '@types/react-dom': 18.3.1 + + '@radix-ui/rect@1.1.0': {} + + '@react-aria/focus@3.18.3(react@18.3.1)': + dependencies: + '@react-aria/interactions': 3.22.3(react@18.3.1) + '@react-aria/utils': 3.25.3(react@18.3.1) + '@react-types/shared': 3.25.0(react@18.3.1) + '@swc/helpers': 0.5.5 + clsx: 2.1.1 + react: 18.3.1 + + '@react-aria/interactions@3.22.3(react@18.3.1)': + dependencies: + '@react-aria/ssr': 3.9.6(react@18.3.1) + '@react-aria/utils': 3.25.3(react@18.3.1) + '@react-types/shared': 3.25.0(react@18.3.1) + '@swc/helpers': 0.5.5 + react: 18.3.1 + + '@react-aria/ssr@3.9.6(react@18.3.1)': + dependencies: + '@swc/helpers': 0.5.5 + react: 18.3.1 + + '@react-aria/utils@3.25.3(react@18.3.1)': + dependencies: + '@react-aria/ssr': 3.9.6(react@18.3.1) + '@react-stately/utils': 3.10.4(react@18.3.1) + '@react-types/shared': 3.25.0(react@18.3.1) + '@swc/helpers': 0.5.5 + clsx: 2.1.1 + react: 18.3.1 + + '@react-stately/utils@3.10.4(react@18.3.1)': + dependencies: + '@swc/helpers': 0.5.5 + react: 18.3.1 + + '@react-types/shared@3.25.0(react@18.3.1)': + dependencies: + react: 18.3.1 + + '@remixicon/react@4.3.0(react@18.3.1)': + dependencies: + react: 18.3.1 + + '@swc/counter@0.1.3': {} + + '@swc/helpers@0.5.5': + dependencies: + '@swc/counter': 0.1.3 + tslib: 2.6.2 + + '@tailwindcss/forms@0.5.9(tailwindcss@3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3)))': + dependencies: + mini-svg-data-uri: 1.4.4 + tailwindcss: 3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3)) + + '@tanstack/react-table@8.20.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@tanstack/table-core': 8.20.5 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + '@tanstack/react-virtual@3.10.8(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@tanstack/virtual-core': 3.10.8 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + '@tanstack/react-virtual@3.5.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1)': + dependencies: + '@tanstack/virtual-core': 3.5.0 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + '@tanstack/table-core@8.20.5': {} + + '@tanstack/virtual-core@3.10.8': {} + + '@tanstack/virtual-core@3.5.0': {} + + '@tootallnate/once@1.1.2': + optional: true + + '@tremor/react@3.18.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(tailwindcss@3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3)))': + dependencies: + '@floating-ui/react': 0.19.2(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@headlessui/react': 1.7.19(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + '@headlessui/tailwindcss': 0.2.1(tailwindcss@3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3))) + date-fns: 3.6.0 + react: 18.3.1 + react-day-picker: 8.10.1(date-fns@3.6.0)(react@18.3.1) + react-dom: 18.3.1(react@18.3.1) + react-transition-state: 2.1.2(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + recharts: 2.13.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + tailwind-merge: 2.5.3 + transitivePeerDependencies: + - tailwindcss + + '@tsconfig/node10@1.0.11': {} + + '@tsconfig/node12@1.0.11': {} + + '@tsconfig/node14@1.0.3': {} + + '@tsconfig/node16@1.0.4': {} + + '@tybys/wasm-util@0.8.3': + dependencies: + tslib: 2.6.2 + optional: true + + '@types/better-sqlite3@7.6.11': + dependencies: + '@types/node': 20.16.11 + optional: true + + '@types/d3-array@3.2.1': {} + + '@types/d3-color@3.1.3': {} + + '@types/d3-ease@3.0.2': {} + + '@types/d3-interpolate@3.0.4': + dependencies: + '@types/d3-color': 3.1.3 + + '@types/d3-path@3.1.0': {} + + '@types/d3-scale@4.0.8': + dependencies: + '@types/d3-time': 3.0.3 + + '@types/d3-shape@3.1.6': + dependencies: + '@types/d3-path': 3.1.0 + + '@types/d3-time@3.0.3': {} + + '@types/d3-timer@3.0.2': {} + + '@types/lodash@4.17.13': {} + + '@types/luxon@3.4.2': {} + + '@types/node@20.16.11': + dependencies: + undici-types: 6.19.8 + + '@types/prop-types@15.7.12': {} + + '@types/react-dom@18.3.1': + dependencies: + '@types/react': 18.3.11 + + '@types/react@18.3.11': + dependencies: + '@types/prop-types': 15.7.12 + csstype: 3.1.3 + + '@types/ws@8.5.12': + dependencies: + '@types/node': 20.16.11 + + abbrev@1.1.1: + optional: true + + acorn-walk@8.3.4: + dependencies: + acorn: 8.14.0 + + acorn@8.14.0: {} + + agent-base@6.0.2: + dependencies: + debug: 4.3.4 + transitivePeerDependencies: + - supports-color + optional: true + + agentkeepalive@4.5.0: + dependencies: + humanize-ms: 1.2.1 + optional: true + + aggregate-error@3.1.0: + dependencies: + clean-stack: 2.2.0 + indent-string: 4.0.0 + optional: true + + ansi-regex@5.0.1: {} + + ansi-regex@6.0.1: {} + + ansi-styles@4.3.0: + dependencies: + color-convert: 2.0.1 + + ansi-styles@6.2.1: {} + + any-promise@1.3.0: {} + + anymatch@3.1.3: + dependencies: + normalize-path: 3.0.0 + picomatch: 2.3.1 + + aproba@2.0.0: + optional: true + + arctic@1.9.2: + dependencies: + oslo: 1.2.0 + + are-we-there-yet@3.0.1: + dependencies: + delegates: 1.0.0 + readable-stream: 3.6.2 + optional: true + + arg@4.1.3: {} + + arg@5.0.2: {} + + aria-hidden@1.2.4: + dependencies: + tslib: 2.6.2 + + balanced-match@1.0.2: {} + + base64-js@1.5.1: {} + + better-sqlite3@9.6.0: + dependencies: + bindings: 1.5.0 + prebuild-install: 7.1.2 + optional: true + + binary-extensions@2.3.0: {} + + bindings@1.5.0: + dependencies: + file-uri-to-path: 1.0.0 + + bl@4.1.0: + dependencies: + buffer: 5.7.1 + inherits: 2.0.4 + readable-stream: 3.6.2 + + brace-expansion@1.1.11: + dependencies: + balanced-match: 1.0.2 + concat-map: 0.0.1 + optional: true + + brace-expansion@2.0.1: + dependencies: + balanced-match: 1.0.2 + + braces@3.0.2: + dependencies: + fill-range: 7.0.1 + + buffer-from@1.1.2: {} + + buffer@5.7.1: + dependencies: + base64-js: 1.5.1 + ieee754: 1.2.1 + + busboy@1.6.0: + dependencies: + streamsearch: 1.1.0 + + cacache@15.3.0: + dependencies: + '@npmcli/fs': 1.1.1 + '@npmcli/move-file': 1.1.2 + chownr: 2.0.0 + fs-minipass: 2.1.0 + glob: 7.2.3 + infer-owner: 1.0.4 + lru-cache: 6.0.0 + minipass: 3.3.6 + minipass-collect: 1.0.2 + minipass-flush: 1.0.5 + minipass-pipeline: 1.2.4 + mkdirp: 1.0.4 + p-map: 4.0.0 + promise-inflight: 1.0.1 + rimraf: 3.0.2 + ssri: 8.0.1 + tar: 6.2.1 + unique-filename: 1.1.1 + transitivePeerDependencies: + - bluebird + optional: true + + camelcase-css@2.0.1: {} + + caniuse-lite@1.0.30001617: {} + + chokidar@3.6.0: + dependencies: + anymatch: 3.1.3 + braces: 3.0.2 + glob-parent: 5.1.2 + is-binary-path: 2.1.0 + is-glob: 4.0.3 + normalize-path: 3.0.0 + readdirp: 3.6.0 + optionalDependencies: + fsevents: 2.3.3 + + chownr@1.1.4: {} + + chownr@2.0.0: {} + + class-variance-authority@0.7.0: + dependencies: + clsx: 2.0.0 + + clean-stack@2.2.0: + optional: true + + cli-color@2.0.4: + dependencies: + d: 1.0.2 + es5-ext: 0.10.64 + es6-iterator: 2.0.3 + memoizee: 0.4.15 + timers-ext: 0.1.7 + + client-only@0.0.1: {} + + clsx@2.0.0: {} + + clsx@2.1.1: {} + + color-convert@2.0.1: + dependencies: + color-name: 1.1.4 + + color-name@1.1.4: {} + + color-support@1.1.3: + optional: true + + commander@4.1.1: {} + + commander@9.5.0: {} + + concat-map@0.0.1: + optional: true + + console-control-strings@1.1.0: + optional: true + + create-require@1.1.1: {} + + cross-spawn@7.0.3: + dependencies: + path-key: 3.1.1 + shebang-command: 2.0.0 + which: 2.0.2 + + cssesc@3.0.0: {} + + csstype@3.1.3: {} + + d3-array@3.2.4: + dependencies: + internmap: 2.0.3 + + d3-color@3.1.0: {} + + d3-ease@3.0.1: {} + + d3-format@3.1.0: {} + + d3-interpolate@3.0.1: + dependencies: + d3-color: 3.1.0 + + d3-path@3.1.0: {} + + d3-scale@4.0.2: + dependencies: + d3-array: 3.2.4 + d3-format: 3.1.0 + d3-interpolate: 3.0.1 + d3-time: 3.1.0 + d3-time-format: 4.1.0 + + d3-shape@3.2.0: + dependencies: + d3-path: 3.1.0 + + d3-time-format@4.1.0: + dependencies: + d3-time: 3.1.0 + + d3-time@3.1.0: + dependencies: + d3-array: 3.2.4 + + d3-timer@3.0.1: {} + + d@1.0.2: + dependencies: + es5-ext: 0.10.64 + type: 2.7.2 + + data-uri-to-buffer@4.0.1: {} + + date-fns@3.6.0: {} + + date-fns@4.1.0: {} + + debug@4.3.4: + dependencies: + ms: 2.1.2 + + decimal.js-light@2.5.1: {} + + decompress-response@6.0.0: + dependencies: + mimic-response: 3.1.0 + + deep-extend@0.6.0: {} + + delegates@1.0.0: + optional: true + + detect-libc@2.0.2: {} + + detect-libc@2.0.3: {} + + detect-node-es@1.1.0: {} + + didyoumean@1.2.2: {} + + diff@4.0.2: {} + + difflib@0.2.4: + dependencies: + heap: 0.2.7 + + dlv@1.1.3: {} + + dom-helpers@5.2.1: + dependencies: + '@babel/runtime': 7.24.5 + csstype: 3.1.3 + + dreamopt@0.8.0: + dependencies: + wordwrap: 1.0.0 + + drizzle-kit@0.21.4: + dependencies: + '@esbuild-kit/esm-loader': 2.6.5 + commander: 9.5.0 + env-paths: 3.0.0 + esbuild: 0.19.12 + esbuild-register: 3.5.0(esbuild@0.19.12) + glob: 8.1.0 + hanji: 0.0.5 + json-diff: 0.9.0 + zod: 3.23.8 + transitivePeerDependencies: + - supports-color + + drizzle-orm@0.30.10(@libsql/client@0.14.0)(@types/better-sqlite3@7.6.11)(@types/react@18.3.11)(better-sqlite3@9.6.0)(react@18.3.1)(sqlite3@5.1.7): + optionalDependencies: + '@libsql/client': 0.14.0 + '@types/better-sqlite3': 7.6.11 + '@types/react': 18.3.11 + better-sqlite3: 9.6.0 + react: 18.3.1 + sqlite3: 5.1.7 + + eastasianwidth@0.2.0: {} + + emoji-regex@8.0.0: {} + + emoji-regex@9.2.2: {} + + encoding@0.1.13: + dependencies: + iconv-lite: 0.6.3 + optional: true + + end-of-stream@1.4.4: + dependencies: + once: 1.4.0 + + env-paths@2.2.1: + optional: true + + env-paths@3.0.0: {} + + err-code@2.0.3: + optional: true + + es5-ext@0.10.64: + dependencies: + es6-iterator: 2.0.3 + es6-symbol: 3.1.4 + esniff: 2.0.1 + next-tick: 1.1.0 + + es6-iterator@2.0.3: + dependencies: + d: 1.0.2 + es5-ext: 0.10.64 + es6-symbol: 3.1.4 + + es6-symbol@3.1.4: + dependencies: + d: 1.0.2 + ext: 1.7.0 + + es6-weak-map@2.0.3: + dependencies: + d: 1.0.2 + es5-ext: 0.10.64 + es6-iterator: 2.0.3 + es6-symbol: 3.1.4 + + esbuild-register@3.5.0(esbuild@0.19.12): + dependencies: + debug: 4.3.4 + esbuild: 0.19.12 + transitivePeerDependencies: + - supports-color + + esbuild@0.18.20: + optionalDependencies: + '@esbuild/android-arm': 0.18.20 + '@esbuild/android-arm64': 0.18.20 + '@esbuild/android-x64': 0.18.20 + '@esbuild/darwin-arm64': 0.18.20 + '@esbuild/darwin-x64': 0.18.20 + '@esbuild/freebsd-arm64': 0.18.20 + '@esbuild/freebsd-x64': 0.18.20 + '@esbuild/linux-arm': 0.18.20 + '@esbuild/linux-arm64': 0.18.20 + '@esbuild/linux-ia32': 0.18.20 + '@esbuild/linux-loong64': 0.18.20 + '@esbuild/linux-mips64el': 0.18.20 + '@esbuild/linux-ppc64': 0.18.20 + '@esbuild/linux-riscv64': 0.18.20 + '@esbuild/linux-s390x': 0.18.20 + '@esbuild/linux-x64': 0.18.20 + '@esbuild/netbsd-x64': 0.18.20 + '@esbuild/openbsd-x64': 0.18.20 + '@esbuild/sunos-x64': 0.18.20 + '@esbuild/win32-arm64': 0.18.20 + '@esbuild/win32-ia32': 0.18.20 + '@esbuild/win32-x64': 0.18.20 + + esbuild@0.19.12: + optionalDependencies: + '@esbuild/aix-ppc64': 0.19.12 + '@esbuild/android-arm': 0.19.12 + '@esbuild/android-arm64': 0.19.12 + '@esbuild/android-x64': 0.19.12 + '@esbuild/darwin-arm64': 0.19.12 + '@esbuild/darwin-x64': 0.19.12 + '@esbuild/freebsd-arm64': 0.19.12 + '@esbuild/freebsd-x64': 0.19.12 + '@esbuild/linux-arm': 0.19.12 + '@esbuild/linux-arm64': 0.19.12 + '@esbuild/linux-ia32': 0.19.12 + '@esbuild/linux-loong64': 0.19.12 + '@esbuild/linux-mips64el': 0.19.12 + '@esbuild/linux-ppc64': 0.19.12 + '@esbuild/linux-riscv64': 0.19.12 + '@esbuild/linux-s390x': 0.19.12 + '@esbuild/linux-x64': 0.19.12 + '@esbuild/netbsd-x64': 0.19.12 + '@esbuild/openbsd-x64': 0.19.12 + '@esbuild/sunos-x64': 0.19.12 + '@esbuild/win32-arm64': 0.19.12 + '@esbuild/win32-ia32': 0.19.12 + '@esbuild/win32-x64': 0.19.12 + + esniff@2.0.1: + dependencies: + d: 1.0.2 + es5-ext: 0.10.64 + event-emitter: 0.3.5 + type: 2.7.2 + + event-emitter@0.3.5: + dependencies: + d: 1.0.2 + es5-ext: 0.10.64 + + eventemitter3@4.0.7: {} + + expand-template@2.0.3: {} + + ext@1.7.0: + dependencies: + type: 2.7.2 + + fast-equals@5.0.1: {} + + fast-glob@3.3.2: + dependencies: + '@nodelib/fs.stat': 2.0.5 + '@nodelib/fs.walk': 1.2.8 + glob-parent: 5.1.2 + merge2: 1.4.1 + micromatch: 4.0.5 + + fastq@1.17.1: + dependencies: + reusify: 1.0.4 + + fetch-blob@3.2.0: + dependencies: + node-domexception: 1.0.0 + web-streams-polyfill: 3.3.3 + + file-uri-to-path@1.0.0: {} + + fill-range@7.0.1: + dependencies: + to-regex-range: 5.0.1 + + foreground-child@3.1.1: + dependencies: + cross-spawn: 7.0.3 + signal-exit: 4.1.0 + + formdata-polyfill@4.0.10: + dependencies: + fetch-blob: 3.2.0 + + fs-constants@1.0.0: {} + + fs-minipass@2.1.0: + dependencies: + minipass: 3.3.6 + + fs-monkey@1.0.6: + optional: true + + fs.realpath@1.0.0: {} + + fsevents@2.3.3: + optional: true + + function-bind@1.1.2: {} + + gauge@4.0.4: + dependencies: + aproba: 2.0.0 + color-support: 1.1.3 + console-control-strings: 1.1.0 + has-unicode: 2.0.1 + signal-exit: 3.0.7 + string-width: 4.2.3 + strip-ansi: 6.0.1 + wide-align: 1.1.5 + optional: true + + get-nonce@1.0.1: {} + + get-tsconfig@4.7.5: + dependencies: + resolve-pkg-maps: 1.0.0 + + github-from-package@0.0.0: {} + + glob-parent@5.1.2: + dependencies: + is-glob: 4.0.3 + + glob-parent@6.0.2: + dependencies: + is-glob: 4.0.3 + + glob@10.3.14: + dependencies: + foreground-child: 3.1.1 + jackspeak: 2.3.6 + minimatch: 9.0.4 + minipass: 7.1.1 + path-scurry: 1.11.0 + + glob@7.2.3: + dependencies: + fs.realpath: 1.0.0 + inflight: 1.0.6 + inherits: 2.0.4 + minimatch: 3.1.2 + once: 1.4.0 + path-is-absolute: 1.0.1 + optional: true + + glob@8.1.0: + dependencies: + fs.realpath: 1.0.0 + inflight: 1.0.6 + inherits: 2.0.4 + minimatch: 5.1.6 + once: 1.4.0 + + graceful-fs@4.2.11: {} + + hanji@0.0.5: + dependencies: + lodash.throttle: 4.1.1 + sisteransi: 1.0.5 + + has-unicode@2.0.1: + optional: true + + hasown@2.0.2: + dependencies: + function-bind: 1.1.2 + + heap@0.2.7: {} + + http-cache-semantics@4.1.1: + optional: true + + http-proxy-agent@4.0.1: + dependencies: + '@tootallnate/once': 1.1.2 + agent-base: 6.0.2 + debug: 4.3.4 + transitivePeerDependencies: + - supports-color + optional: true + + https-proxy-agent@5.0.1: + dependencies: + agent-base: 6.0.2 + debug: 4.3.4 + transitivePeerDependencies: + - supports-color + optional: true + + humanize-ms@1.2.1: + dependencies: + ms: 2.1.2 + optional: true + + iconv-lite@0.6.3: + dependencies: + safer-buffer: 2.1.2 + optional: true + + ieee754@1.2.1: {} + + imurmurhash@0.1.4: + optional: true + + indent-string@4.0.0: + optional: true + + infer-owner@1.0.4: + optional: true + + inflight@1.0.6: + dependencies: + once: 1.4.0 + wrappy: 1.0.2 + + inherits@2.0.4: {} + + ini@1.3.8: {} + + internmap@2.0.3: {} + + invariant@2.2.4: + dependencies: + loose-envify: 1.4.0 + + ip-address@9.0.5: + dependencies: + jsbn: 1.1.0 + sprintf-js: 1.1.3 + optional: true + + is-binary-path@2.1.0: + dependencies: + binary-extensions: 2.3.0 + + is-core-module@2.13.1: + dependencies: + hasown: 2.0.2 + + is-extglob@2.1.1: {} + + is-fullwidth-code-point@3.0.0: {} + + is-glob@4.0.3: + dependencies: + is-extglob: 2.1.1 + + is-lambda@1.0.1: + optional: true + + is-number@7.0.0: {} + + is-promise@2.2.2: {} + + isexe@2.0.0: {} + + jackspeak@2.3.6: + dependencies: + '@isaacs/cliui': 8.0.2 + optionalDependencies: + '@pkgjs/parseargs': 0.11.0 + + jiti@1.21.0: {} + + js-base64@3.7.7: {} + + js-tokens@4.0.0: {} + + jsbn@1.1.0: + optional: true + + json-diff@0.9.0: + dependencies: + cli-color: 2.0.4 + difflib: 0.2.4 + dreamopt: 0.8.0 + + libsql@0.4.6: + dependencies: + '@neon-rs/load': 0.0.4 + detect-libc: 2.0.2 + optionalDependencies: + '@libsql/darwin-arm64': 0.4.6 + '@libsql/darwin-x64': 0.4.6 + '@libsql/linux-arm64-gnu': 0.4.6 + '@libsql/linux-arm64-musl': 0.4.6 + '@libsql/linux-x64-gnu': 0.4.6 + '@libsql/linux-x64-musl': 0.4.6 + '@libsql/win32-x64-msvc': 0.4.6 + + lilconfig@2.1.0: {} + + lilconfig@3.1.1: {} + + lines-and-columns@1.2.4: {} + + lodash.throttle@4.1.1: {} + + lodash@4.17.21: {} + + loose-envify@1.4.0: + dependencies: + js-tokens: 4.0.0 + + lru-cache@10.2.2: {} + + lru-cache@6.0.0: + dependencies: + yallist: 4.0.0 + optional: true + + lru-queue@0.1.0: + dependencies: + es5-ext: 0.10.64 + + lucia@3.2.1: + dependencies: + '@oslojs/crypto': 1.0.1 + '@oslojs/encoding': 1.1.0 + + lucide-react@0.378.0(react@18.3.1): + dependencies: + react: 18.3.1 + + luxon@3.5.0: {} + + make-error@1.3.6: {} + + make-fetch-happen@9.1.0: + dependencies: + agentkeepalive: 4.5.0 + cacache: 15.3.0 + http-cache-semantics: 4.1.1 + http-proxy-agent: 4.0.1 + https-proxy-agent: 5.0.1 + is-lambda: 1.0.1 + lru-cache: 6.0.0 + minipass: 3.3.6 + minipass-collect: 1.0.2 + minipass-fetch: 1.4.1 + minipass-flush: 1.0.5 + minipass-pipeline: 1.2.4 + negotiator: 0.6.4 + promise-retry: 2.0.1 + socks-proxy-agent: 6.2.1 + ssri: 8.0.1 + transitivePeerDependencies: + - bluebird + - supports-color + optional: true + + memfs-browser@3.5.10302: + dependencies: + memfs: 3.5.3 + optional: true + + memfs@3.5.3: + dependencies: + fs-monkey: 1.0.6 + optional: true + + memoizee@0.4.15: + dependencies: + d: 1.0.2 + es5-ext: 0.10.64 + es6-weak-map: 2.0.3 + event-emitter: 0.3.5 + is-promise: 2.2.2 + lru-queue: 0.1.0 + next-tick: 1.1.0 + timers-ext: 0.1.7 + + merge2@1.4.1: {} + + micromatch@4.0.5: + dependencies: + braces: 3.0.2 + picomatch: 2.3.1 + + mimic-response@3.1.0: {} + + mini-svg-data-uri@1.4.4: {} + + minimatch@3.1.2: + dependencies: + brace-expansion: 1.1.11 + optional: true + + minimatch@5.1.6: + dependencies: + brace-expansion: 2.0.1 + + minimatch@9.0.4: + dependencies: + brace-expansion: 2.0.1 + + minimist@1.2.8: {} + + minipass-collect@1.0.2: + dependencies: + minipass: 3.3.6 + optional: true + + minipass-fetch@1.4.1: + dependencies: + minipass: 3.3.6 + minipass-sized: 1.0.3 + minizlib: 2.1.2 + optionalDependencies: + encoding: 0.1.13 + optional: true + + minipass-flush@1.0.5: + dependencies: + minipass: 3.3.6 + optional: true + + minipass-pipeline@1.2.4: + dependencies: + minipass: 3.3.6 + optional: true + + minipass-sized@1.0.3: + dependencies: + minipass: 3.3.6 + optional: true + + minipass@3.3.6: + dependencies: + yallist: 4.0.0 + + minipass@5.0.0: {} + + minipass@7.1.1: {} + + minizlib@2.1.2: + dependencies: + minipass: 3.3.6 + yallist: 4.0.0 + + mkdirp-classic@0.5.3: {} + + mkdirp@1.0.4: {} + + ms@2.1.2: {} + + mz@2.7.0: + dependencies: + any-promise: 1.3.0 + object-assign: 4.1.1 + thenify-all: 1.6.0 + + nanoid@3.3.7: {} + + napi-build-utils@1.0.2: {} + + negotiator@0.6.4: + optional: true + + next-themes@0.3.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1): + dependencies: + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + next-tick@1.1.0: {} + + next@14.2.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1): + dependencies: + '@next/env': 14.2.3 + '@swc/helpers': 0.5.5 + busboy: 1.6.0 + caniuse-lite: 1.0.30001617 + graceful-fs: 4.2.11 + postcss: 8.4.31 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + styled-jsx: 5.1.1(react@18.3.1) + optionalDependencies: + '@next/swc-darwin-arm64': 14.2.3 + '@next/swc-darwin-x64': 14.2.3 + '@next/swc-linux-arm64-gnu': 14.2.3 + '@next/swc-linux-arm64-musl': 14.2.3 + '@next/swc-linux-x64-gnu': 14.2.3 + '@next/swc-linux-x64-musl': 14.2.3 + '@next/swc-win32-arm64-msvc': 14.2.3 + '@next/swc-win32-ia32-msvc': 14.2.3 + '@next/swc-win32-x64-msvc': 14.2.3 + transitivePeerDependencies: + - '@babel/core' + - babel-plugin-macros + + node-abi@3.62.0: + dependencies: + semver: 7.6.2 + + node-addon-api@7.1.1: {} + + node-domexception@1.0.0: {} + + node-fetch@3.3.2: + dependencies: + data-uri-to-buffer: 4.0.1 + fetch-blob: 3.2.0 + formdata-polyfill: 4.0.10 + + node-gyp@8.4.1: + dependencies: + env-paths: 2.2.1 + glob: 7.2.3 + graceful-fs: 4.2.11 + make-fetch-happen: 9.1.0 + nopt: 5.0.0 + npmlog: 6.0.2 + rimraf: 3.0.2 + semver: 7.6.2 + tar: 6.2.1 + which: 2.0.2 + transitivePeerDependencies: + - bluebird + - supports-color + optional: true + + nopt@5.0.0: + dependencies: + abbrev: 1.1.1 + optional: true + + normalize-path@3.0.0: {} + + npmlog@6.0.2: + dependencies: + are-we-there-yet: 3.0.1 + console-control-strings: 1.1.0 + gauge: 4.0.4 + set-blocking: 2.0.0 + optional: true + + object-assign@4.1.1: {} + + object-hash@3.0.0: {} + + once@1.4.0: + dependencies: + wrappy: 1.0.2 + + oslo@1.2.0: + dependencies: + '@node-rs/argon2': 1.7.0 + '@node-rs/bcrypt': 1.9.0 + + oslo@1.2.1: + dependencies: + '@node-rs/argon2': 1.7.0 + '@node-rs/bcrypt': 1.9.0 + + p-map@4.0.0: + dependencies: + aggregate-error: 3.1.0 + optional: true + + path-is-absolute@1.0.1: + optional: true + + path-key@3.1.1: {} + + path-parse@1.0.7: {} + + path-scurry@1.11.0: + dependencies: + lru-cache: 10.2.2 + minipass: 7.1.1 + + picocolors@1.0.0: {} + + picocolors@1.1.0: {} + + picomatch@2.3.1: {} + + pify@2.3.0: {} + + pirates@4.0.6: {} + + postcss-import@15.1.0(postcss@8.4.47): + dependencies: + postcss: 8.4.47 + postcss-value-parser: 4.2.0 + read-cache: 1.0.0 + resolve: 1.22.8 + + postcss-js@4.0.1(postcss@8.4.47): + dependencies: + camelcase-css: 2.0.1 + postcss: 8.4.47 + + postcss-load-config@4.0.2(postcss@8.4.47)(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3)): + dependencies: + lilconfig: 3.1.1 + yaml: 2.4.2 + optionalDependencies: + postcss: 8.4.47 + ts-node: 10.9.2(@types/node@20.16.11)(typescript@5.6.3) + + postcss-nested@6.0.1(postcss@8.4.47): + dependencies: + postcss: 8.4.47 + postcss-selector-parser: 6.0.16 + + postcss-selector-parser@6.0.16: + dependencies: + cssesc: 3.0.0 + util-deprecate: 1.0.2 + + postcss-value-parser@4.2.0: {} + + postcss@8.4.31: + dependencies: + nanoid: 3.3.7 + picocolors: 1.0.0 + source-map-js: 1.2.0 + + postcss@8.4.47: + dependencies: + nanoid: 3.3.7 + picocolors: 1.1.0 + source-map-js: 1.2.1 + + prebuild-install@7.1.2: + dependencies: + detect-libc: 2.0.3 + expand-template: 2.0.3 + github-from-package: 0.0.0 + minimist: 1.2.8 + mkdirp-classic: 0.5.3 + napi-build-utils: 1.0.2 + node-abi: 3.62.0 + pump: 3.0.0 + rc: 1.2.8 + simple-get: 4.0.1 + tar-fs: 2.1.1 + tunnel-agent: 0.6.0 + + promise-inflight@1.0.1: + optional: true + + promise-limit@2.7.0: {} + + promise-retry@2.0.1: + dependencies: + err-code: 2.0.3 + retry: 0.12.0 + optional: true + + prop-types@15.8.1: + dependencies: + loose-envify: 1.4.0 + object-assign: 4.1.1 + react-is: 16.13.1 + + pump@3.0.0: + dependencies: + end-of-stream: 1.4.4 + once: 1.4.0 + + queue-microtask@1.2.3: {} + + rc@1.2.8: + dependencies: + deep-extend: 0.6.0 + ini: 1.3.8 + minimist: 1.2.8 + strip-json-comments: 2.0.1 + + react-day-picker@8.10.1(date-fns@3.6.0)(react@18.3.1): + dependencies: + date-fns: 3.6.0 + react: 18.3.1 + + react-dom@18.3.1(react@18.3.1): + dependencies: + loose-envify: 1.4.0 + react: 18.3.1 + scheduler: 0.23.2 + + react-hook-form@7.53.0(react@18.3.1): + dependencies: + react: 18.3.1 + + react-if@4.1.5(react@18.3.1): + dependencies: + react: 18.3.1 + + react-is@16.13.1: {} + + react-is@18.3.1: {} + + react-remove-scroll-bar@2.3.6(@types/react@18.3.11)(react@18.3.1): + dependencies: + react: 18.3.1 + react-style-singleton: 2.2.1(@types/react@18.3.11)(react@18.3.1) + tslib: 2.6.2 + optionalDependencies: + '@types/react': 18.3.11 + + react-remove-scroll@2.6.0(@types/react@18.3.11)(react@18.3.1): + dependencies: + react: 18.3.1 + react-remove-scroll-bar: 2.3.6(@types/react@18.3.11)(react@18.3.1) + react-style-singleton: 2.2.1(@types/react@18.3.11)(react@18.3.1) + tslib: 2.6.2 + use-callback-ref: 1.3.2(@types/react@18.3.11)(react@18.3.1) + use-sidecar: 1.1.2(@types/react@18.3.11)(react@18.3.1) + optionalDependencies: + '@types/react': 18.3.11 + + react-smooth@4.0.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1): + dependencies: + fast-equals: 5.0.1 + prop-types: 15.8.1 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + react-transition-group: 4.4.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + + react-style-singleton@2.2.1(@types/react@18.3.11)(react@18.3.1): + dependencies: + get-nonce: 1.0.1 + invariant: 2.2.4 + react: 18.3.1 + tslib: 2.6.2 + optionalDependencies: + '@types/react': 18.3.11 + + react-timer-hook@3.0.7(react@18.3.1): + dependencies: + react: 18.3.1 + + react-transition-group@4.4.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1): + dependencies: + '@babel/runtime': 7.24.5 + dom-helpers: 5.2.1 + loose-envify: 1.4.0 + prop-types: 15.8.1 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + react-transition-state@2.1.2(react-dom@18.3.1(react@18.3.1))(react@18.3.1): + dependencies: + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + react@18.3.1: + dependencies: + loose-envify: 1.4.0 + + read-cache@1.0.0: + dependencies: + pify: 2.3.0 + + readable-stream@3.6.2: + dependencies: + inherits: 2.0.4 + string_decoder: 1.3.0 + util-deprecate: 1.0.2 + + readdirp@3.6.0: + dependencies: + picomatch: 2.3.1 + + recharts-scale@0.4.5: + dependencies: + decimal.js-light: 2.5.1 + + recharts@2.13.3(react-dom@18.3.1(react@18.3.1))(react@18.3.1): + dependencies: + clsx: 2.1.1 + eventemitter3: 4.0.7 + lodash: 4.17.21 + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + react-is: 18.3.1 + react-smooth: 4.0.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + recharts-scale: 0.4.5 + tiny-invariant: 1.3.3 + victory-vendor: 36.9.2 + + regenerator-runtime@0.14.1: {} + + regression@2.0.1: {} + + resolve-pkg-maps@1.0.0: {} + + resolve@1.22.8: + dependencies: + is-core-module: 2.13.1 + path-parse: 1.0.7 + supports-preserve-symlinks-flag: 1.0.0 + + retry@0.12.0: + optional: true + + reusify@1.0.4: {} + + rimraf@3.0.2: + dependencies: + glob: 7.2.3 + optional: true + + run-parallel@1.2.0: + dependencies: + queue-microtask: 1.2.3 + + safe-buffer@5.2.1: {} + + safer-buffer@2.1.2: + optional: true + + scheduler@0.23.2: + dependencies: + loose-envify: 1.4.0 + + semver@7.6.2: {} + + set-blocking@2.0.0: + optional: true + + shebang-command@2.0.0: + dependencies: + shebang-regex: 3.0.0 + + shebang-regex@3.0.0: {} + + signal-exit@3.0.7: + optional: true + + signal-exit@4.1.0: {} + + simple-concat@1.0.1: {} + + simple-get@4.0.1: + dependencies: + decompress-response: 6.0.0 + once: 1.4.0 + simple-concat: 1.0.1 + + sisteransi@1.0.5: {} + + smart-buffer@4.2.0: + optional: true + + socks-proxy-agent@6.2.1: + dependencies: + agent-base: 6.0.2 + debug: 4.3.4 + socks: 2.8.3 + transitivePeerDependencies: + - supports-color + optional: true + + socks@2.8.3: + dependencies: + ip-address: 9.0.5 + smart-buffer: 4.2.0 + optional: true + + sonner@1.5.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1): + dependencies: + react: 18.3.1 + react-dom: 18.3.1(react@18.3.1) + + source-map-js@1.2.0: {} + + source-map-js@1.2.1: {} + + source-map-support@0.5.21: + dependencies: + buffer-from: 1.1.2 + source-map: 0.6.1 + + source-map@0.6.1: {} + + sprintf-js@1.1.3: + optional: true + + sqlite3@5.1.7: + dependencies: + bindings: 1.5.0 + node-addon-api: 7.1.1 + prebuild-install: 7.1.2 + tar: 6.2.1 + optionalDependencies: + node-gyp: 8.4.1 + transitivePeerDependencies: + - bluebird + - supports-color + + sqlite@5.1.1: {} + + ssri@8.0.1: + dependencies: + minipass: 3.3.6 + optional: true + + streamsearch@1.1.0: {} + + string-width@4.2.3: + dependencies: + emoji-regex: 8.0.0 + is-fullwidth-code-point: 3.0.0 + strip-ansi: 6.0.1 + + string-width@5.1.2: + dependencies: + eastasianwidth: 0.2.0 + emoji-regex: 9.2.2 + strip-ansi: 7.1.0 + + string_decoder@1.3.0: + dependencies: + safe-buffer: 5.2.1 + + strip-ansi@6.0.1: + dependencies: + ansi-regex: 5.0.1 + + strip-ansi@7.1.0: + dependencies: + ansi-regex: 6.0.1 + + strip-json-comments@2.0.1: {} + + styled-jsx@5.1.1(react@18.3.1): + dependencies: + client-only: 0.0.1 + react: 18.3.1 + + sucrase@3.35.0: + dependencies: + '@jridgewell/gen-mapping': 0.3.5 + commander: 4.1.1 + glob: 10.3.14 + lines-and-columns: 1.2.4 + mz: 2.7.0 + pirates: 4.0.6 + ts-interface-checker: 0.1.13 + + supports-preserve-symlinks-flag@1.0.0: {} + + swr@2.2.5(react@18.3.1): + dependencies: + client-only: 0.0.1 + react: 18.3.1 + use-sync-external-store: 1.2.2(react@18.3.1) + + tabbable@6.2.0: {} + + tailwind-merge@2.5.3: {} + + tailwindcss-animate@1.0.7(tailwindcss@3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3))): + dependencies: + tailwindcss: 3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3)) + + tailwindcss@3.4.13(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3)): + dependencies: + '@alloc/quick-lru': 5.2.0 + arg: 5.0.2 + chokidar: 3.6.0 + didyoumean: 1.2.2 + dlv: 1.1.3 + fast-glob: 3.3.2 + glob-parent: 6.0.2 + is-glob: 4.0.3 + jiti: 1.21.0 + lilconfig: 2.1.0 + micromatch: 4.0.5 + normalize-path: 3.0.0 + object-hash: 3.0.0 + picocolors: 1.0.0 + postcss: 8.4.47 + postcss-import: 15.1.0(postcss@8.4.47) + postcss-js: 4.0.1(postcss@8.4.47) + postcss-load-config: 4.0.2(postcss@8.4.47)(ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3)) + postcss-nested: 6.0.1(postcss@8.4.47) + postcss-selector-parser: 6.0.16 + resolve: 1.22.8 + sucrase: 3.35.0 + transitivePeerDependencies: + - ts-node + + tar-fs@2.1.1: + dependencies: + chownr: 1.1.4 + mkdirp-classic: 0.5.3 + pump: 3.0.0 + tar-stream: 2.2.0 + + tar-stream@2.2.0: + dependencies: + bl: 4.1.0 + end-of-stream: 1.4.4 + fs-constants: 1.0.0 + inherits: 2.0.4 + readable-stream: 3.6.2 + + tar@6.2.1: + dependencies: + chownr: 2.0.0 + fs-minipass: 2.1.0 + minipass: 5.0.0 + minizlib: 2.1.2 + mkdirp: 1.0.4 + yallist: 4.0.0 + + thenify-all@1.6.0: + dependencies: + thenify: 3.3.1 + + thenify@3.3.1: + dependencies: + any-promise: 1.3.0 + + timers-ext@0.1.7: + dependencies: + es5-ext: 0.10.64 + next-tick: 1.1.0 + + tiny-invariant@1.3.3: {} + + to-regex-range@5.0.1: + dependencies: + is-number: 7.0.0 + + ts-interface-checker@0.1.13: {} + + ts-node@10.9.2(@types/node@20.16.11)(typescript@5.6.3): + dependencies: + '@cspotcode/source-map-support': 0.8.1 + '@tsconfig/node10': 1.0.11 + '@tsconfig/node12': 1.0.11 + '@tsconfig/node14': 1.0.3 + '@tsconfig/node16': 1.0.4 + '@types/node': 20.16.11 + acorn: 8.14.0 + acorn-walk: 8.3.4 + arg: 4.1.3 + create-require: 1.1.1 + diff: 4.0.2 + make-error: 1.3.6 + typescript: 5.6.3 + v8-compile-cache-lib: 3.0.1 + yn: 3.1.1 + + tslib@2.6.2: {} + + tunnel-agent@0.6.0: + dependencies: + safe-buffer: 5.2.1 + + type@2.7.2: {} + + typescript@5.6.3: {} + + undici-types@6.19.8: {} + + unique-filename@1.1.1: + dependencies: + unique-slug: 2.0.2 + optional: true + + unique-slug@2.0.2: + dependencies: + imurmurhash: 0.1.4 + optional: true + + use-callback-ref@1.3.2(@types/react@18.3.11)(react@18.3.1): + dependencies: + react: 18.3.1 + tslib: 2.6.2 + optionalDependencies: + '@types/react': 18.3.11 + + use-debounce@10.0.3(react@18.3.1): + dependencies: + react: 18.3.1 + + use-sidecar@1.1.2(@types/react@18.3.11)(react@18.3.1): + dependencies: + detect-node-es: 1.1.0 + react: 18.3.1 + tslib: 2.6.2 + optionalDependencies: + '@types/react': 18.3.11 + + use-sync-external-store@1.2.2(react@18.3.1): + dependencies: + react: 18.3.1 + + util-deprecate@1.0.2: {} + + uuid@11.0.2: {} + + v8-compile-cache-lib@3.0.1: {} + + victory-vendor@36.9.2: + dependencies: + '@types/d3-array': 3.2.1 + '@types/d3-ease': 3.0.2 + '@types/d3-interpolate': 3.0.4 + '@types/d3-scale': 4.0.8 + '@types/d3-shape': 3.1.6 + '@types/d3-time': 3.0.3 + '@types/d3-timer': 3.0.2 + d3-array: 3.2.4 + d3-ease: 3.0.1 + d3-interpolate: 3.0.1 + d3-scale: 4.0.2 + d3-shape: 3.2.0 + d3-time: 3.1.0 + d3-timer: 3.0.1 + + web-streams-polyfill@3.3.3: {} + + which@2.0.2: + dependencies: + isexe: 2.0.0 + + wide-align@1.1.5: + dependencies: + string-width: 4.2.3 + optional: true + + wordwrap@1.0.0: {} + + wrap-ansi@7.0.0: + dependencies: + ansi-styles: 4.3.0 + string-width: 4.2.3 + strip-ansi: 6.0.1 + + wrap-ansi@8.1.0: + dependencies: + ansi-styles: 6.2.1 + string-width: 5.1.2 + strip-ansi: 7.1.0 + + wrappy@1.0.2: {} + + ws@8.18.0: {} + + yallist@4.0.0: {} + + yaml@2.4.2: {} + + yn@3.1.1: {} + + zod@3.23.8: {} diff --git a/packages/reservation-platform/postcss.config.mjs b/packages/reservation-platform/postcss.config.mjs new file mode 100644 index 000000000..1a69fd2a4 --- /dev/null +++ b/packages/reservation-platform/postcss.config.mjs @@ -0,0 +1,8 @@ +/** @type {import('postcss-load-config').Config} */ +const config = { + plugins: { + tailwindcss: {}, + }, +}; + +export default config; diff --git a/packages/reservation-platform/public/next.svg b/packages/reservation-platform/public/next.svg new file mode 100644 index 000000000..5174b28c5 --- /dev/null +++ b/packages/reservation-platform/public/next.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/packages/reservation-platform/public/vercel.svg b/packages/reservation-platform/public/vercel.svg new file mode 100644 index 000000000..d2f842227 --- /dev/null +++ b/packages/reservation-platform/public/vercel.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/packages/reservation-platform/repomix-output.txt b/packages/reservation-platform/repomix-output.txt new file mode 100644 index 000000000..68f0be146 --- /dev/null +++ b/packages/reservation-platform/repomix-output.txt @@ -0,0 +1,9279 @@ +This file is a merged representation of the entire codebase, combining all repository files into a single document. +Generated by Repomix on: 2024-12-09T06:29:51.427Z + +================================================================ +File Summary +================================================================ + +Purpose: +-------- +This file contains a packed representation of the entire repository's contents. +It is designed to be easily consumable by AI systems for analysis, code review, +or other automated processes. + +File Format: +------------ +The content is organized as follows: +1. This summary section +2. Repository information +3. Repository structure +4. Multiple file entries, each consisting of: + a. A separator line (================) + b. The file path (File: path/to/file) + c. Another separator line + d. The full contents of the file + e. A blank line + +Usage Guidelines: +----------------- +- This file should be treated as read-only. Any changes should be made to the + original repository files, not this packed version. +- When processing this file, use the file path to distinguish + between different files in the repository. +- Be aware that this file may contain sensitive information. Handle it with + the same level of security as you would the original repository. + +Notes: +------ +- Some files may have been excluded based on .gitignore rules and Repomix's + configuration. +- Binary files are not included in this packed representation. Please refer to + the Repository Structure section for a complete list of file paths, including + binary files. + +Additional Info: +---------------- + +For more information about Repomix, visit: https://github.com/yamadashy/repomix + +================================================================ +Repository Structure +================================================================ +.dockerignore +.env.example +.gitignore +biome.json +components.json +docker/build.sh +docker/caddy/Caddyfile +docker/compose.yml +docker/deploy.sh +docker/images/.gitattributes +docker/save.sh +Dockerfile +drizzle.config.ts +drizzle/0000_overjoyed_strong_guy.sql +drizzle/meta/_journal.json +drizzle/meta/0000_snapshot.json +next.config.mjs +package.json +postcss.config.mjs +public/next.svg +public/vercel.svg +README.md +scripts/generate-data.js +src/app/admin/about/page.tsx +src/app/admin/admin-sidebar.tsx +src/app/admin/charts/printer-error-chart.tsx +src/app/admin/charts/printer-error-rate.tsx +src/app/admin/charts/printer-forecast.tsx +src/app/admin/charts/printer-utilization.tsx +src/app/admin/charts/printer-volume.tsx +src/app/admin/jobs/page.tsx +src/app/admin/layout.tsx +src/app/admin/page.tsx +src/app/admin/printers/columns.tsx +src/app/admin/printers/data-table.tsx +src/app/admin/printers/dialogs/create-printer.tsx +src/app/admin/printers/dialogs/delete-printer.tsx +src/app/admin/printers/dialogs/edit-printer.tsx +src/app/admin/printers/form.tsx +src/app/admin/printers/page.tsx +src/app/admin/settings/download/route.ts +src/app/admin/settings/page.tsx +src/app/admin/users/columns.tsx +src/app/admin/users/data-table.tsx +src/app/admin/users/dialog.tsx +src/app/admin/users/form.tsx +src/app/admin/users/page.tsx +src/app/api/job/[jobId]/remaining-time/route.ts +src/app/api/printers/route.ts +src/app/auth/login/callback/route.ts +src/app/auth/login/route.ts +src/app/globals.css +src/app/job/[jobId]/cancel-form.tsx +src/app/job/[jobId]/edit-comments.tsx +src/app/job/[jobId]/extend-form.tsx +src/app/job/[jobId]/finish-form.tsx +src/app/job/[jobId]/page.tsx +src/app/layout.tsx +src/app/my/jobs/columns.tsx +src/app/my/jobs/data-table.tsx +src/app/my/profile/page.tsx +src/app/not-found.tsx +src/app/page.tsx +src/app/printer/[printerId]/reserve/form.tsx +src/app/printer/[printerId]/reserve/page.tsx +src/components/data-card.tsx +src/components/dynamic-printer-cards.tsx +src/components/header/index.tsx +src/components/header/navigation.tsx +src/components/login-button.tsx +src/components/logout-button.tsx +src/components/personalized-cards.tsx +src/components/printer-availability-badge.tsx +src/components/printer-card/countdown.tsx +src/components/printer-card/index.tsx +src/components/ui/alert-dialog.tsx +src/components/ui/alert.tsx +src/components/ui/avatar.tsx +src/components/ui/badge.tsx +src/components/ui/breadcrumb.tsx +src/components/ui/button.tsx +src/components/ui/card.tsx +src/components/ui/chart.tsx +src/components/ui/dialog.tsx +src/components/ui/dropdown-menu.tsx +src/components/ui/form.tsx +src/components/ui/hover-card.tsx +src/components/ui/input.tsx +src/components/ui/label.tsx +src/components/ui/scroll-area.tsx +src/components/ui/select.tsx +src/components/ui/skeleton.tsx +src/components/ui/sonner.tsx +src/components/ui/table.tsx +src/components/ui/tabs.tsx +src/components/ui/textarea.tsx +src/components/ui/toast.tsx +src/components/ui/toaster.tsx +src/components/ui/use-toast.ts +src/server/actions/authentication/logout.ts +src/server/actions/printers.ts +src/server/actions/printJobs.ts +src/server/actions/timer.ts +src/server/actions/user/delete.ts +src/server/actions/user/update.ts +src/server/actions/users.ts +src/server/auth/index.ts +src/server/auth/oauth.ts +src/server/auth/permissions.ts +src/utils/analytics/error-rate.ts +src/utils/analytics/errors.ts +src/utils/analytics/forecast.ts +src/utils/analytics/utilization.ts +src/utils/analytics/volume.ts +src/utils/drizzle.ts +src/utils/errors.ts +src/utils/fetch.ts +src/utils/guard.ts +src/utils/printers.ts +src/utils/strings.ts +src/utils/styles.ts +tailwind.config.ts +tsconfig.json + +================================================================ +Repository Files +================================================================ + +================ +File: .dockerignore +================ +# Build and utility assets +docker/ +scripts/ + +# Ignore node_modules as they will be installed in the container +node_modules + +# Ignore build artifacts +.next + +# Ignore runtime data +db/ + +# Ignore local configuration files +.env +.env.example + +# Ignore version control files +.git +.gitignore + +# Ignore IDE/editor specific files +*.log +*.tmp +*.DS_Store +.vscode/ +.idea/ + +================ +File: .env.example +================ +# OAuth Configuration +OAUTH_CLIENT_ID=client_id +OAUTH_CLIENT_SECRET=client_secret + +================ +File: .gitignore +================ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# db folder +db/ + +# Env file +.env + + +# dependencies +/node_modules +/.pnp +.pnp.js +.yarn/install-state.gz + +# testing +/coverage + +# next.js +/.next/ +/out/ + +# production +/build + +# misc +.DS_Store +*.pem + +# debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# local env files +.env*.local + +# vercel +.vercel + +# typescript +*.tsbuildinfo +next-env.d.ts + +================ +File: biome.json +================ +{ + "$schema": "https://biomejs.dev/schemas/1.7.0/schema.json", + "organizeImports": { + "enabled": true + }, + "formatter": { + "enabled": true, + "lineWidth": 120 + }, + "linter": { + "enabled": true, + "rules": { + "recommended": true, + "correctness": { + "noUnusedImports": "error" + } + } + } +} + +================ +File: components.json +================ +{ + "$schema": "https://ui.shadcn.com/schema.json", + "style": "new-york", + "rsc": true, + "tsx": true, + "tailwind": { + "config": "tailwind.config.ts", + "css": "src/app/globals.css", + "baseColor": "neutral", + "cssVariables": true, + "prefix": "" + }, + "aliases": { + "components": "@/components", + "utils": "@/utils/styles" + } +} + +================ +File: docker/build.sh +================ +#!/bin/bash + +# Define image name +MYP_RP_IMAGE_NAME="myp-rp" + +# Function to build Docker image +build_image() { + local image_name=$1 + local dockerfile=$2 + local platform=$3 + + echo "Building $image_name Docker image for $platform..." + + docker buildx build --platform $platform -t ${image_name}:latest -f $dockerfile --load . + if [ $? -eq 0 ]; then + echo "$image_name Docker image built successfully" + else + echo "Error occurred while building $image_name Docker image" + exit 1 + fi +} + +# Create and use a builder instance (if not already created) +BUILDER_NAME="myp-rp-arm64-builder" +docker buildx create --name $BUILDER_NAME --use || docker buildx use $BUILDER_NAME + +# Build myp-rp image +build_image "$MYP_RP_IMAGE_NAME" "$PWD/Dockerfile" "linux/arm64" + +# Remove the builder instance +docker buildx rm $BUILDER_NAME + +================ +File: docker/caddy/Caddyfile +================ +{ + debug +} + +m040tbaraspi001.de040.corpintra.net, m040tbaraspi001.de040.corpinter.net { + reverse_proxy myp-rp:3000 + tls internal +} + +================ +File: docker/compose.yml +================ +services: + caddy: + image: caddy:2.8 + container_name: caddy + restart: unless-stopped + ports: + - 80:80 + - 443:443 + volumes: + - ./caddy/data:/data + - ./caddy/config:/config + - ./caddy/Caddyfile:/etc/caddy/Caddyfile:ro + myp-rp: + image: myp-rp:latest + container_name: myp-rp + env_file: "/srv/myp-env/github.env" + volumes: + - /srv/MYP-DB:/usr/src/app/db + restart: unless-stopped + +================ +File: docker/deploy.sh +================ +#!/bin/bash + +# Directory containing the Docker images +IMAGE_DIR="docker/images" + +# Load all Docker images from the tar.xz files in the IMAGE_DIR +echo "Loading Docker images from $IMAGE_DIR..." + +for image_file in "$IMAGE_DIR"/*.tar.xz; do + if [ -f "$image_file" ]; then + echo "Loading Docker image from $image_file..." + docker load -i "$image_file" + + # Check if the image loading was successful + if [ $? -ne 0 ]; then + echo "Error occurred while loading Docker image from $image_file" + exit 1 + fi + else + echo "No Docker image tar.xz files found in $IMAGE_DIR." + fi +done + +# Execute docker compose +echo "Running docker compose..." +docker compose -f "docker/compose.yml" up -d + +# Check if the operation was successful +if [ $? -eq 0 ]; then + echo "Docker compose executed successfully" +else + echo "Error occurred while executing docker compose" + exit 1 +fi + +echo "Deployment completed successfully" + +================ +File: docker/images/.gitattributes +================ +caddy_2.8.tar.xz filter=lfs diff=lfs merge=lfs -text +myp-rp_latest.tar.xz filter=lfs diff=lfs merge=lfs -text + +================ +File: docker/save.sh +================ +#!/bin/bash + +# Get image name as argument +IMAGE_NAME=$1 +PLATFORM="linux/arm64" + +# Define paths +IMAGE_DIR="docker/images" +IMAGE_FILE="${IMAGE_DIR}/${IMAGE_NAME//[:\/]/_}.tar" +COMPRESSED_FILE="${IMAGE_FILE}.xz" + +# Function to pull the image +pull_image() { + local image=$1 + if [[ $image == arm64v8/* ]]; then + echo "Pulling image $image without platform specification..." + docker pull $image + else + echo "Pulling image $image for platform $PLATFORM..." + docker pull --platform $PLATFORM $image + fi + return $? +} + +# Pull the image if it is not available locally +if ! docker image inspect ${IMAGE_NAME} &>/dev/null; then + if pull_image ${IMAGE_NAME}; then + echo "Image $IMAGE_NAME pulled successfully." + else + echo "Error occurred while pulling $IMAGE_NAME for platform $PLATFORM" + echo "Trying to pull $IMAGE_NAME without platform specification..." + + # Attempt to pull again without platform + if pull_image ${IMAGE_NAME}; then + echo "Image $IMAGE_NAME pulled successfully without platform." + else + echo "Error occurred while pulling $IMAGE_NAME without platform." + echo "Trying to pull arm64v8/${IMAGE_NAME} instead..." + + # Construct new image name + NEW_IMAGE_NAME="arm64v8/${IMAGE_NAME}" + if pull_image ${NEW_IMAGE_NAME}; then + echo "Image $NEW_IMAGE_NAME pulled successfully." + IMAGE_NAME=${NEW_IMAGE_NAME} # Update IMAGE_NAME to use the new one + else + echo "Error occurred while pulling $NEW_IMAGE_NAME" + exit 1 + fi + fi + fi +else + echo "Image $IMAGE_NAME found locally. Skipping pull." +fi + +# Save the Docker image +echo "Saving $IMAGE_NAME Docker image..." +docker save ${IMAGE_NAME} > $IMAGE_FILE + +# Compress the Docker image (overwriting if file exists) +echo "Compressing $IMAGE_FILE..." +xz -z --force $IMAGE_FILE + +if [ $? -eq 0 ]; then + echo "$IMAGE_NAME Docker image saved and compressed successfully as $COMPRESSED_FILE" +else + echo "Error occurred while compressing $IMAGE_NAME Docker image" + exit 1 +fi + +================ +File: Dockerfile +================ +FROM node:20-bookworm-slim + +# Create application directory +RUN mkdir -p /usr/src/app + +# Set environment variables +ENV PORT=3000 +ENV NEXT_TELEMETRY_DISABLED=1 + +WORKDIR /usr/src/app + +# Copy package.json and pnpm-lock.yaml +COPY package.json /usr/src/app +COPY pnpm-lock.yaml /usr/src/app + +# Install pnpm +RUN corepack enable pnpm + +# Install dependencies +RUN pnpm install + +# Copy the rest of the application code +COPY . /usr/src/app + +# Initialize Database, if it not already exists +RUN pnpm run db + +# Build the application +RUN pnpm run build + +EXPOSE 3000 + +# Start the application +CMD ["/bin/sh", "-c", "if [ ! -f ./db/sqlite.db ]; then pnpm db; fi && pnpm start"] + +================ +File: drizzle.config.ts +================ +import { defineConfig } from "drizzle-kit"; + +//@ts-ignore - better-sqlite driver throws an error even though its an valid value +export default defineConfig({ + dialect: "sqlite", + schema: "./src/server/db/schema.ts", + out: "./drizzle", + driver: "libsql", + dbCredentials: { + url: "file:./db/sqlite.db", + }, +}); + +================ +File: drizzle/0000_overjoyed_strong_guy.sql +================ +CREATE TABLE `printJob` ( + `id` text PRIMARY KEY NOT NULL, + `printerId` text NOT NULL, + `userId` text NOT NULL, + `startAt` integer NOT NULL, + `durationInMinutes` integer NOT NULL, + `comments` text, + `aborted` integer DEFAULT false NOT NULL, + `abortReason` text, + FOREIGN KEY (`printerId`) REFERENCES `printer`(`id`) ON UPDATE no action ON DELETE cascade, + FOREIGN KEY (`userId`) REFERENCES `user`(`id`) ON UPDATE no action ON DELETE cascade +); +--> statement-breakpoint +CREATE TABLE `printer` ( + `id` text PRIMARY KEY NOT NULL, + `name` text NOT NULL, + `description` text NOT NULL, + `status` integer DEFAULT 0 NOT NULL +); +--> statement-breakpoint +CREATE TABLE `session` ( + `id` text PRIMARY KEY NOT NULL, + `user_id` text NOT NULL, + `expires_at` integer NOT NULL, + FOREIGN KEY (`user_id`) REFERENCES `user`(`id`) ON UPDATE no action ON DELETE cascade +); +--> statement-breakpoint +CREATE TABLE `user` ( + `id` text PRIMARY KEY NOT NULL, + `github_id` integer NOT NULL, + `name` text, + `displayName` text, + `email` text NOT NULL, + `role` text DEFAULT 'guest' +); + +================ +File: drizzle/meta/_journal.json +================ +{ + "version": "6", + "dialect": "sqlite", + "entries": [ + { + "idx": 0, + "version": "6", + "when": 1715416514336, + "tag": "0000_overjoyed_strong_guy", + "breakpoints": true + } + ] +} + +================ +File: drizzle/meta/0000_snapshot.json +================ +{ + "version": "6", + "dialect": "sqlite", + "id": "791dc197-5254-4432-bd9f-1368d1a5aa6a", + "prevId": "00000000-0000-0000-0000-000000000000", + "tables": { + "printJob": { + "name": "printJob", + "columns": { + "id": { + "name": "id", + "type": "text", + "primaryKey": true, + "notNull": true, + "autoincrement": false + }, + "printerId": { + "name": "printerId", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "userId": { + "name": "userId", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "startAt": { + "name": "startAt", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "durationInMinutes": { + "name": "durationInMinutes", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "comments": { + "name": "comments", + "type": "text", + "primaryKey": false, + "notNull": false, + "autoincrement": false + }, + "aborted": { + "name": "aborted", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false, + "default": false + }, + "abortReason": { + "name": "abortReason", + "type": "text", + "primaryKey": false, + "notNull": false, + "autoincrement": false + } + }, + "indexes": {}, + "foreignKeys": { + "printJob_printerId_printer_id_fk": { + "name": "printJob_printerId_printer_id_fk", + "tableFrom": "printJob", + "tableTo": "printer", + "columnsFrom": [ + "printerId" + ], + "columnsTo": [ + "id" + ], + "onDelete": "cascade", + "onUpdate": "no action" + }, + "printJob_userId_user_id_fk": { + "name": "printJob_userId_user_id_fk", + "tableFrom": "printJob", + "tableTo": "user", + "columnsFrom": [ + "userId" + ], + "columnsTo": [ + "id" + ], + "onDelete": "cascade", + "onUpdate": "no action" + } + }, + "compositePrimaryKeys": {}, + "uniqueConstraints": {} + }, + "printer": { + "name": "printer", + "columns": { + "id": { + "name": "id", + "type": "text", + "primaryKey": true, + "notNull": true, + "autoincrement": false + }, + "name": { + "name": "name", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "description": { + "name": "description", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "status": { + "name": "status", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false, + "default": 0 + } + }, + "indexes": {}, + "foreignKeys": {}, + "compositePrimaryKeys": {}, + "uniqueConstraints": {} + }, + "session": { + "name": "session", + "columns": { + "id": { + "name": "id", + "type": "text", + "primaryKey": true, + "notNull": true, + "autoincrement": false + }, + "user_id": { + "name": "user_id", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "expires_at": { + "name": "expires_at", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false + } + }, + "indexes": {}, + "foreignKeys": { + "session_user_id_user_id_fk": { + "name": "session_user_id_user_id_fk", + "tableFrom": "session", + "tableTo": "user", + "columnsFrom": [ + "user_id" + ], + "columnsTo": [ + "id" + ], + "onDelete": "cascade", + "onUpdate": "no action" + } + }, + "compositePrimaryKeys": {}, + "uniqueConstraints": {} + }, + "user": { + "name": "user", + "columns": { + "id": { + "name": "id", + "type": "text", + "primaryKey": true, + "notNull": true, + "autoincrement": false + }, + "github_id": { + "name": "github_id", + "type": "integer", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "name": { + "name": "name", + "type": "text", + "primaryKey": false, + "notNull": false, + "autoincrement": false + }, + "displayName": { + "name": "displayName", + "type": "text", + "primaryKey": false, + "notNull": false, + "autoincrement": false + }, + "email": { + "name": "email", + "type": "text", + "primaryKey": false, + "notNull": true, + "autoincrement": false + }, + "role": { + "name": "role", + "type": "text", + "primaryKey": false, + "notNull": false, + "autoincrement": false, + "default": "'guest'" + } + }, + "indexes": {}, + "foreignKeys": {}, + "compositePrimaryKeys": {}, + "uniqueConstraints": {} + } + }, + "enums": {}, + "_meta": { + "schemas": {}, + "tables": {}, + "columns": {} + } +} + +================ +File: next.config.mjs +================ +/** @type {import('next').NextConfig} */ +const nextConfig = { + async headers() { + return [ + { + source: "/:path*", + headers: [ + { + key: "Access-Control-Allow-Origin", + value: "m040tbaraspi001.de040.corpintra.net", + }, + { + key: "Access-Control-Allow-Methods", + value: "GET, POST, PUT, DELETE, OPTIONS", + }, + { + key: "Access-Control-Allow-Headers", + value: "Content-Type, Authorization", + }, + ], + }, + ]; + }, +}; + +export default nextConfig; + +================ +File: package.json +================ +{ + "name": "myp-rp", + "version": "1.0.0", + "private": true, + "packageManager": "pnpm@9.12.1", + "scripts": { + "dev": "next dev", + "build": "next build", + "start": "next start", + "lint": "next lint", + "db:create-default": "mkdir -p db/", + "db:generate-sqlite": "pnpm drizzle-kit generate", + "db:clean": "rm -rf db/ drizzle/", + "db:migrate": "pnpm drizzle-kit migrate", + "db": "pnpm db:create-default && pnpm db:generate-sqlite && pnpm db:migrate", + "db:reset": "pnpm db:clean && pnpm db" + }, + "dependencies": { + "@faker-js/faker": "^9.2.0", + "@headlessui/react": "^2.1.10", + "@headlessui/tailwindcss": "^0.2.1", + "@hookform/resolvers": "^3.9.0", + "@libsql/client": "^0.14.0", + "@lucia-auth/adapter-drizzle": "^1.1.0", + "@radix-ui/react-alert-dialog": "^1.1.2", + "@radix-ui/react-avatar": "^1.1.1", + "@radix-ui/react-dialog": "^1.1.2", + "@radix-ui/react-dropdown-menu": "^2.1.2", + "@radix-ui/react-hover-card": "^1.1.2", + "@radix-ui/react-icons": "^1.3.0", + "@radix-ui/react-label": "^2.1.0", + "@radix-ui/react-scroll-area": "^1.2.0", + "@radix-ui/react-select": "^2.1.2", + "@radix-ui/react-slot": "^1.1.0", + "@radix-ui/react-tabs": "^1.1.1", + "@radix-ui/react-toast": "^1.2.2", + "@remixicon/react": "^4.3.0", + "@tanstack/react-table": "^8.20.5", + "@tremor/react": "^3.18.3", + "arctic": "^1.9.2", + "class-variance-authority": "^0.7.0", + "clsx": "^2.1.1", + "date-fns": "^4.1.0", + "drizzle-orm": "^0.30.10", + "lodash": "^4.17.21", + "lucia": "^3.2.1", + "lucide-react": "^0.378.0", + "luxon": "^3.5.0", + "next": "14.2.3", + "next-themes": "^0.3.0", + "oslo": "^1.2.1", + "react": "^18.3.1", + "react-dom": "^18.3.1", + "react-hook-form": "^7.53.0", + "react-if": "^4.1.5", + "react-timer-hook": "^3.0.7", + "recharts": "^2.13.3", + "regression": "^2.0.1", + "sonner": "^1.5.0", + "sqlite": "^5.1.1", + "sqlite3": "^5.1.7", + "swr": "^2.2.5", + "tailwind-merge": "^2.5.3", + "tailwindcss-animate": "^1.0.7", + "use-debounce": "^10.0.3", + "uuid": "^11.0.2", + "zod": "^3.23.8" + }, + "devDependencies": { + "@biomejs/biome": "^1.9.3", + "@tailwindcss/forms": "^0.5.9", + "@types/lodash": "^4.17.13", + "@types/luxon": "^3.4.2", + "@types/node": "^20.16.11", + "@types/react": "^18.3.11", + "@types/react-dom": "^18.3.1", + "drizzle-kit": "^0.21.4", + "postcss": "^8.4.47", + "tailwindcss": "^3.4.13", + "ts-node": "^10.9.2", + "typescript": "^5.6.3" + } +} + +================ +File: postcss.config.mjs +================ +/** @type {import('postcss-load-config').Config} */ +const config = { + plugins: { + tailwindcss: {}, + }, +}; + +export default config; + +================ +File: public/next.svg +================ + + +================ +File: public/vercel.svg +================ + + +================ +File: README.md +================ +# MYP - Manage Your Printer + +MYP (Manage Your Printer) ist eine Webanwendung zur Reservierung von 3D-Druckern. +Sie wurde im Rahmen des Abschlussprojektes der Fachinformatiker Ausbildung für Daten- und Prozessanalyse für die Technische Berufsausbildung des Mercedes-Benz Werkes Berlin-Marienfelde entwickelt. + +## Deployment + +### Voraussetzungen + +- Netzwerk auf Raspberry Pi ist eingerichtet +- Docker ist installiert + +### Schritte + +1. Docker-Container bauen (docker/build.sh) +2. Docker-Container speichern (docker/save.sh caddy:2.8 myp-rp:latest) +3. Docker-Container auf Raspberry Pi bereitstellen (docker/deploy.sh) + +## Entwicklerinformationen + +### Raspberry Pi Einstellungen + +Auf dem Raspberry Pi wurde Raspbian Lite installiert. +Unter /srv/* sind die Projektdateien zu finden. + +### Anmeldedaten + +``` +Benutzer: myp +Passwort: (persönlich bekannt) +``` + +================ +File: scripts/generate-data.js +================ +const sqlite3 = require("sqlite3"); +const faker = require("@faker-js/faker").faker; +const { random, sample, sampleSize, sum } = require("lodash"); +const { DateTime } = require("luxon"); +const { open } = require("sqlite"); +const { v4: uuidv4 } = require("uuid"); + +const dbPath = "./db/sqlite.db"; + +// Configuration for test data generation +let startDate = DateTime.fromISO("2024-10-08"); +let endDate = DateTime.fromISO("2024-11-08"); +let numberOfPrinters = 5; + +// Use weekday names for better readability and ease of setting trends +let avgPrintTimesPerDay = { + Monday: 4, + Tuesday: 2, + Wednesday: 5, + Thursday: 2, + Friday: 3, + Saturday: 0, + Sunday: 0, +}; // Average number of prints for each weekday + +let avgPrintDurationPerDay = { + Monday: 240, // Total average duration in minutes for Monday + Tuesday: 30, + Wednesday: 45, + Thursday: 40, + Friday: 120, + Saturday: 0, + Sunday: 0, +}; // Average total duration of prints for each weekday + +let printerUsage = { + "Drucker 1": 0.5, + "Drucker 2": 0.7, + "Drucker 3": 0.6, + "Drucker 4": 0.3, + "Drucker 5": 0.4, +}; // Usage percentages for each printer + +// **New Configurations for Error Rates** +let generalErrorRate = 0.05; // 5% chance any print job may fail +let printerErrorRates = { + "Drucker 1": 0.02, // 2% error rate for Printer 1 + "Drucker 2": 0.03, + "Drucker 3": 0.01, + "Drucker 4": 0.05, + "Drucker 5": 0.04, +}; // Error rates for each printer + +const holidays = []; // Example holidays +const existingJobs = []; + +const initDB = async () => { + console.log("Initializing database connection..."); + return open({ + filename: dbPath, + driver: sqlite3.Database, + }); +}; + +const createUser = (isPowerUser = false) => { + const name = [faker.person.firstName(), faker.person.lastName()]; + + const user = { + id: uuidv4(), + github_id: faker.number.int(), + username: `${name[0].slice(0, 2)}${name[1].slice(0, 6)}`.toUpperCase(), + displayName: `${name[0]} ${name[1]}`.toUpperCase(), + email: `${name[0]}.${name[1]}@example.com`, + role: sample(["user", "admin"]), + isPowerUser, + }; + console.log("Created user:", user); + return user; +}; + +const createPrinter = (index) => { + const printer = { + id: uuidv4(), + name: `Drucker ${index}`, + description: faker.lorem.sentence(), + status: random(0, 2), + }; + console.log("Created printer:", printer); + return printer; +}; + +const isPrinterAvailable = (printer, startAt, duration) => { + const endAt = startAt + duration * 60 * 1000; // Convert minutes to milliseconds + return !existingJobs.some((job) => { + const jobStart = job.startAt; + const jobEnd = job.startAt + job.durationInMinutes * 60 * 1000; + return ( + printer.id === job.printerId && + ((startAt >= jobStart && startAt < jobEnd) || + (endAt > jobStart && endAt <= jobEnd) || + (startAt <= jobStart && endAt >= jobEnd)) + ); + }); +}; + +const createPrintJob = (users, printers, startAt, duration) => { + const user = sample(users); + let printer; + + // Weighted selection based on printer usage + const printerNames = Object.keys(printerUsage); + const weightedPrinters = printers.filter((p) => printerNames.includes(p.name)); + + // Create a weighted array of printers based on usage percentages + const printerWeights = weightedPrinters.map((p) => ({ + printer: p, + weight: printerUsage[p.name], + })); + + const totalWeight = sum(printerWeights.map((pw) => pw.weight)); + const randomWeight = Math.random() * totalWeight; + let accumulatedWeight = 0; + for (const pw of printerWeights) { + accumulatedWeight += pw.weight; + if (randomWeight <= accumulatedWeight) { + printer = pw.printer; + break; + } + } + + if (!printer) { + printer = sample(printers); + } + + if (!isPrinterAvailable(printer, startAt, duration)) { + console.log("Printer not available, skipping job creation."); + return null; + } + + // **Determine if the job should be aborted based on error rates** + let aborted = false; + let abortReason = null; + + // Calculate the combined error rate + const printerErrorRate = printerErrorRates[printer.name] || 0; + const combinedErrorRate = 1 - (1 - generalErrorRate) * (1 - printerErrorRate); + + if (Math.random() < combinedErrorRate) { + aborted = true; + const errorMessages = [ + "Unbekannt", + "Keine Ahnung", + "Falsch gebucht", + "Filament gelöst", + "Druckabbruch", + "Düsenverstopfung", + "Schichthaftung fehlgeschlagen", + "Materialmangel", + "Dateifehler", + "Temperaturproblem", + "Mechanischer Fehler", + "Softwarefehler", + "Kalibrierungsfehler", + "Überhitzung", + ]; + abortReason = sample(errorMessages); // Generate a random abort reason + } + + const printJob = { + id: uuidv4(), + printerId: printer.id, + userId: user.id, + startAt, + durationInMinutes: duration, + comments: faker.lorem.sentence(), + aborted, + abortReason, + }; + console.log("Created print job:", printJob); + return printJob; +}; + +const generatePrintJobsForDay = async (users, printers, dayDate, totalJobsForDay, totalDurationForDay, db, dryRun) => { + console.log(`Generating print jobs for ${dayDate.toISODate()}...`); + + // Generate random durations that sum up approximately to totalDurationForDay + const durations = []; + let remainingDuration = totalDurationForDay; + for (let i = 0; i < totalJobsForDay; i++) { + const avgJobDuration = remainingDuration / (totalJobsForDay - i); + const jobDuration = Math.max( + Math.round(random(avgJobDuration * 0.8, avgJobDuration * 1.2)), + 5, // Minimum duration of 5 minutes + ); + durations.push(jobDuration); + remainingDuration -= jobDuration; + } + + // Shuffle durations to randomize job lengths + const shuffledDurations = sampleSize(durations, durations.length); + + for (let i = 0; i < totalJobsForDay; i++) { + const duration = shuffledDurations[i]; + + // Random start time between 8 AM and 6 PM, adjusted to avoid overlapping durations + const possibleStartHours = Array.from({ length: 10 }, (_, idx) => idx + 8); // 8 AM to 6 PM + let startAt; + let attempts = 0; + do { + const hour = sample(possibleStartHours); + const minute = random(0, 59); + startAt = dayDate.set({ hour, minute, second: 0, millisecond: 0 }).toMillis(); + attempts++; + if (attempts > 10) { + console.log("Unable to find available time slot, skipping job."); + break; + } + } while (!isPrinterAvailable(sample(printers), startAt, duration)); + + if (attempts > 10) continue; + + const printJob = createPrintJob(users, printers, startAt, duration); + if (printJob) { + if (!dryRun) { + await db.run( + `INSERT INTO printJob (id, printerId, userId, startAt, durationInMinutes, comments, aborted, abortReason) + VALUES (?, ?, ?, ?, ?, ?, ?, ?)`, + [ + printJob.id, + printJob.printerId, + printJob.userId, + printJob.startAt, + printJob.durationInMinutes, + printJob.comments, + printJob.aborted ? 1 : 0, + printJob.abortReason, + ], + ); + } + existingJobs.push(printJob); + console.log("Inserted print job into database:", printJob.id); + } + } +}; + +const generateTestData = async (dryRun = false) => { + console.log("Starting test data generation..."); + const db = await initDB(); + + // Generate users and printers + const users = [ + ...Array.from({ length: 7 }, () => createUser(false)), + ...Array.from({ length: 3 }, () => createUser(true)), + ]; + const printers = Array.from({ length: numberOfPrinters }, (_, index) => createPrinter(index + 1)); + + if (!dryRun) { + // Insert users into the database + for (const user of users) { + await db.run( + `INSERT INTO user (id, github_id, name, displayName, email, role) + VALUES (?, ?, ?, ?, ?, ?)`, + [user.id, user.github_id, user.username, user.displayName, user.email, user.role], + ); + console.log("Inserted user into database:", user.id); + } + + // Insert printers into the database + for (const printer of printers) { + await db.run( + `INSERT INTO printer (id, name, description, status) + VALUES (?, ?, ?, ?)`, + [printer.id, printer.name, printer.description, printer.status], + ); + console.log("Inserted printer into database:", printer.id); + } + } + + // Generate print jobs for each day within the specified date range + let currentDay = startDate; + while (currentDay <= endDate) { + const weekdayName = currentDay.toFormat("EEEE"); // Get weekday name (e.g., 'Monday') + if (holidays.includes(currentDay.toISODate()) || avgPrintTimesPerDay[weekdayName] === 0) { + console.log(`Skipping holiday or no jobs scheduled: ${currentDay.toISODate()}`); + currentDay = currentDay.plus({ days: 1 }); + continue; + } + + const totalJobsForDay = avgPrintTimesPerDay[weekdayName]; + const totalDurationForDay = avgPrintDurationPerDay[weekdayName]; + + await generatePrintJobsForDay(users, printers, currentDay, totalJobsForDay, totalDurationForDay, db, dryRun); + currentDay = currentDay.plus({ days: 1 }); + } + + if (!dryRun) { + await db.close(); + console.log("Database connection closed. Test data generation complete."); + } else { + console.log("Dry run complete. No data was written to the database."); + } +}; + +const setConfigurations = (config) => { + if (config.startDate) startDate = DateTime.fromISO(config.startDate); + if (config.endDate) endDate = DateTime.fromISO(config.endDate); + if (config.numberOfPrinters) numberOfPrinters = config.numberOfPrinters; + if (config.avgPrintTimesPerDay) avgPrintTimesPerDay = config.avgPrintTimesPerDay; + if (config.avgPrintDurationPerDay) avgPrintDurationPerDay = config.avgPrintDurationPerDay; + if (config.printerUsage) printerUsage = config.printerUsage; + if (config.generalErrorRate !== undefined) generalErrorRate = config.generalErrorRate; + if (config.printerErrorRates) printerErrorRates = config.printerErrorRates; +}; + +// Example usage +setConfigurations({ + startDate: "2024-10-08", + endDate: "2024-11-08", + numberOfPrinters: 6, + avgPrintTimesPerDay: { + Monday: 4, // High usage + Tuesday: 2, // Low usage + Wednesday: 3, // Low usage + Thursday: 2, // Low usage + Friday: 8, // High usage + Saturday: 0, + Sunday: 0, + }, + avgPrintDurationPerDay: { + Monday: 300, // High total duration + Tuesday: 60, // Low total duration + Wednesday: 90, + Thursday: 60, + Friday: 240, + Saturday: 0, + Sunday: 0, + }, + printerUsage: { + "Drucker 1": 2.3, + "Drucker 2": 1.7, + "Drucker 3": 0.1, + "Drucker 4": 1.5, + "Drucker 5": 2.4, + "Drucker 6": 0.3, + "Drucker 7": 0.9, + "Drucker 8": 0.1, + }, + generalErrorRate: 0.05, // 5% general error rate + printerErrorRates: { + "Drucker 1": 0.02, + "Drucker 2": 0.03, + "Drucker 3": 0.1, + "Drucker 4": 0.05, + "Drucker 5": 0.04, + "Drucker 6": 0.02, + "Drucker 7": 0.01, + "PrinteDrucker 8": 0.03, + }, +}); + +generateTestData(process.argv.includes("--dry-run")) + .then(() => { + console.log("Test data generation script finished."); + }) + .catch((err) => { + console.error("Error generating test data:", err); + }); + +================ +File: src/app/admin/about/page.tsx +================ +import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"; +import type { Metadata } from "next"; + +export const metadata: Metadata = { + title: "Über MYP", +}; + +export default async function AdminPage() { + return ( + + + Über MYP + + MYP — Manage Your Printer + + + +

+ MYP ist eine Webanwendung zur Reservierung von 3D-Druckern. Sie wurde im Rahmen des + Abschlussprojektes der Fachinformatiker Ausbildung für Daten- und Prozessanalyse für die Technische + Berufsausbildung des Mercedes-Benz Werkes Berlin-Marienfelde entwickelt. +

+

+ © 2024{" "} + + Torben Haack + +

+
+
+ ); +} + +================ +File: src/app/admin/admin-sidebar.tsx +================ +"use client"; + +import { cn } from "@/utils/styles"; +import { FileIcon, HeartIcon, LayoutDashboardIcon, PrinterIcon, UsersIcon, WrenchIcon } from "lucide-react"; +import Link from "next/link"; +import { usePathname } from "next/navigation"; + +interface AdminSite { + name: string; + path: string; + icon: React.ReactNode; +} + +export function AdminSidebar() { + const pathname = usePathname(); + const adminSites: AdminSite[] = [ + { + name: "Dashboard", + path: "/admin", + icon: , + }, + { + name: "Benutzer", + path: "/admin/users", + icon: , + }, + { + name: "Drucker", + path: "/admin/printers", + icon: , + }, + { + name: "Druckaufträge", + path: "/admin/jobs", + icon: , + }, + { + name: "Einstellungen", + path: "/admin/settings", + icon: , + }, + { + name: "Über MYP", + path: "/admin/about", + icon: , + }, + ]; + + return ( +
    + {adminSites.map((site) => ( +
  • + + {site.icon} + {site.name} + +
  • + ))} +
+ ); +} + +================ +File: src/app/admin/charts/printer-error-chart.tsx +================ +"use client"; + +import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"; +import { type ChartConfig, ChartContainer, ChartTooltip, ChartTooltipContent } from "@/components/ui/chart"; +import { Bar, BarChart, CartesianGrid, LabelList, XAxis, YAxis } from "recharts"; + +export const description = "Ein Säulendiagramm zur Darstellung der Abbruchgründe und ihrer Häufigkeit"; + +interface AbortReasonCountChartProps { + abortReasonCount: { + abortReason: string; + count: number; + }[]; +} + +const chartConfig = { + abortReason: { + label: "Abbruchgrund", + }, +} satisfies ChartConfig; + +export function AbortReasonCountChart({ abortReasonCount }: AbortReasonCountChartProps) { + // Transform data to fit the chart structure + const chartData = abortReasonCount.map((reason) => ({ + abortReason: reason.abortReason, + count: reason.count, + })); + + return ( + + + Abbruchgründe + Häufigkeit der Abbruchgründe für Druckaufträge + + + + + + value} + /> + `${value}`} /> + } /> + + `${value}`} + /> + + + + + + ); +} + +================ +File: src/app/admin/charts/printer-error-rate.tsx +================ +"use client"; +import { Bar, BarChart, CartesianGrid, LabelList, XAxis, YAxis } from "recharts"; + +import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"; +import { type ChartConfig, ChartContainer, ChartTooltip, ChartTooltipContent } from "@/components/ui/chart"; +import type { PrinterErrorRate } from "@/utils/analytics/error-rate"; + +export const description = "Ein Säulendiagramm zur Darstellung der Fehlerrate"; + +interface PrinterErrorRateChartProps { + printerErrorRate: PrinterErrorRate[]; +} + +const chartConfig = { + errorRate: { + label: "Fehlerrate", + }, +} satisfies ChartConfig; + +export function PrinterErrorRateChart({ printerErrorRate }: PrinterErrorRateChartProps) { + // Transform data to fit the chart structure + const chartData = printerErrorRate.map((printer) => ({ + printer: printer.name, + errorRate: printer.errorRate, + })); + + return ( + + + Fehlerrate + Fehlerrate der Drucker in Prozent + + + + + + value} + /> + `${value}%`} /> + } /> + + `${value}%`} + /> + + + + + + ); +} + +================ +File: src/app/admin/charts/printer-forecast.tsx +================ +"use client"; + +import { Card, CardContent, CardFooter, CardHeader, CardTitle } from "@/components/ui/card"; +import { type ChartConfig, ChartContainer, ChartTooltip, ChartTooltipContent } from "@/components/ui/chart"; +import { Area, AreaChart, CartesianGrid, XAxis, YAxis } from "recharts"; + +export const description = "Ein Bereichsdiagramm zur Darstellung der prognostizierten Nutzung pro Wochentag"; + +interface ForecastData { + day: number; // 0 for Sunday, 1 for Monday, ..., 6 for Saturday + usageMinutes: number; +} + +interface ForecastChartProps { + forecastData: ForecastData[]; +} + +const chartConfig = { + usage: { + label: "Prognostizierte Nutzung", + color: "hsl(var(--chart-1))", + }, +} satisfies ChartConfig; + +const daysOfWeek = ["Sonntag", "Montag", "Dienstag", "Mittwoch", "Donnerstag", "Freitag", "Samstag"]; + +export function ForecastPrinterUsageChart({ forecastData }: ForecastChartProps) { + // Transform and slice data to fit the chart structure + const chartData = forecastData.map((data) => ({ + //slice(1, forecastData.length - 1). + day: daysOfWeek[data.day], // Map day number to weekday name + usage: data.usageMinutes, + })); + + return ( + + + Prognostizierte Nutzung pro Wochentag + + + + + + + + } /> + + + + + +
+ Zeigt die prognostizierte Nutzungszeit pro Wochentag in Minuten. +
+
+ Besten Tage zur Wartung: {bestMaintenanceDays(forecastData)} +
+
+
+ ); +} + +function bestMaintenanceDays(forecastData: ForecastData[]) { + const sortedData = forecastData.map((a) => a).sort((a, b) => a.usageMinutes - b.usageMinutes); // Sort ascending + + const q1Index = Math.floor(sortedData.length * 0.33); + const q1 = sortedData[q1Index].usageMinutes; // First quartile (Q1) value + + const filteredData = sortedData.filter((data) => data.usageMinutes <= q1); + + return filteredData + .map((data) => { + const days = ["Sonntag", "Montag", "Dienstag", "Mittwoch", "Donnerstag", "Freitag", "Samstag"]; + return days[data.day]; + }) + .join(", "); +} + +================ +File: src/app/admin/charts/printer-utilization.tsx +================ +"use client"; + +import { TrendingUp } from "lucide-react"; +import * as React from "react"; +import { Label, Pie, PieChart } from "recharts"; + +import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from "@/components/ui/card"; +import { type ChartConfig, ChartContainer, ChartTooltip, ChartTooltipContent } from "@/components/ui/chart"; + +export const description = "Nutzung des Druckers"; + +interface ComponentProps { + data: { + printerId: string; + utilizationPercentage: number; + name: string; + }; +} + +const chartConfig = {} satisfies ChartConfig; + +export function PrinterUtilizationChart({ data }: ComponentProps) { + const totalUtilization = React.useMemo(() => data.utilizationPercentage, [data]); + const dataWithColor = { + ...data, + fill: "rgb(34 197 94)", + }; + const free = { + printerId: "-", + utilizationPercentage: 1 - data.utilizationPercentage, + name: "(Frei)", + fill: "rgb(212 212 212)", + }; + + return ( + + + {data.name} + Nutzung des ausgewählten Druckers + + + + + } /> + + + + + + +
+ Übersicht der Nutzung +
+
Aktuelle Auslastung des Druckers
+
+
+ ); +} + +================ +File: src/app/admin/charts/printer-volume.tsx +================ +"use client"; +import { Bar, BarChart, CartesianGrid, LabelList, XAxis } from "recharts"; + +import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from "@/components/ui/card"; +import { type ChartConfig, ChartContainer, ChartTooltip, ChartTooltipContent } from "@/components/ui/chart"; + +export const description = "Ein Balkendiagramm mit Beschriftung"; + +interface PrintVolumes { + today: number; + thisWeek: number; + thisMonth: number; +} + +const chartConfig = { + volume: { + label: "Volumen", + }, +} satisfies ChartConfig; + +interface PrinterVolumeChartProps { + printerVolume: PrintVolumes; +} + +export function PrinterVolumeChart({ printerVolume }: PrinterVolumeChartProps) { + const chartData = [ + { period: "Heute", volume: printerVolume.today, color: "hsl(var(--chart-1))" }, + { period: "Diese Woche", volume: printerVolume.thisWeek, color: "hsl(var(--chart-2))" }, + { period: "Diesen Monat", volume: printerVolume.thisMonth, color: "hsl(var(--chart-3))" }, + ]; + + return ( + + + Druckvolumen + Vergleich: Heute, Diese Woche, Diesen Monat + + + + + + value} + /> + } /> + + + + + + + +
+ Zeigt das Druckvolumen für heute, diese Woche und diesen Monat +
+
+
+ ); +} + +================ +File: src/app/admin/jobs/page.tsx +================ +import { columns } from "@/app/my/jobs/columns"; +import { JobsTable } from "@/app/my/jobs/data-table"; +import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"; +import { db } from "@/server/db"; +import { printJobs } from "@/server/db/schema"; +import { desc } from "drizzle-orm"; +import type { Metadata } from "next"; + +export const metadata: Metadata = { + title: "Alle Druckaufträge", +}; + +export default async function AdminJobsPage() { + const allJobs = await db.query.printJobs.findMany({ + orderBy: [desc(printJobs.startAt)], + with: { + user: true, + printer: true, + }, + }); + + return ( + + +
+ Druckaufträge + Alle Druckaufträge +
+
+ + + +
+ ); +} + +================ +File: src/app/admin/layout.tsx +================ +import { AdminSidebar } from "@/app/admin/admin-sidebar"; +import { validateRequest } from "@/server/auth"; +import { UserRole } from "@/server/auth/permissions"; +import { IS_NOT, guard } from "@/utils/guard"; +import { redirect } from "next/navigation"; + +interface AdminLayoutProps { + children: React.ReactNode; +} + +export const dynamic = "force-dynamic"; + +export default async function AdminLayout(props: AdminLayoutProps) { + const { children } = props; + const { user } = await validateRequest(); + + if (guard(user, IS_NOT, UserRole.ADMIN)) { + redirect("/"); + } + + return ( +
+
+

Admin

+
+
+ +
{children}
+
+
+ ); +} + +================ +File: src/app/admin/page.tsx +================ +import { AbortReasonCountChart } from "@/app/admin/charts/printer-error-chart"; +import { PrinterErrorRateChart } from "@/app/admin/charts/printer-error-rate"; +import { ForecastPrinterUsageChart } from "@/app/admin/charts/printer-forecast"; +import { PrinterUtilizationChart } from "@/app/admin/charts/printer-utilization"; +import { PrinterVolumeChart } from "@/app/admin/charts/printer-volume"; +import { DataCard } from "@/components/data-card"; +import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"; +import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs"; +import { db } from "@/server/db"; +import { calculatePrinterErrorRate } from "@/utils/analytics/error-rate"; +import { calculateAbortReasonsCount } from "@/utils/analytics/errors"; +import { forecastPrinterUsage } from "@/utils/analytics/forecast"; +import { calculatePrinterUtilization } from "@/utils/analytics/utilization"; +import { calculatePrintVolumes } from "@/utils/analytics/volume"; +import type { Metadata } from "next"; + +export const metadata: Metadata = { + title: "Admin Dashboard", +}; + +export const dynamic = "force-dynamic"; + +export default async function AdminPage() { + const currentDate = new Date(); + + const lastMonth = new Date(); + lastMonth.setDate(currentDate.getDate() - 31); + const printers = await db.query.printers.findMany({}); + const printJobs = await db.query.printJobs.findMany({ + where: (job, { gte }) => gte(job.startAt, lastMonth), + with: { + printer: true, + }, + }); + if (printJobs.length < 1) { + return ( + + + Druckaufträge + Zurzeit sind keine Druckaufträge verfügbar. + + +

Aktualisieren Sie die Seite oder prüfen Sie später erneut, ob neue Druckaufträge verfügbar sind.

+
+
+ ); + } + + const currentPrintJobs = printJobs.filter((job) => { + if (job.aborted) return false; + + const endAt = job.startAt.getTime() + job.durationInMinutes * 1000 * 60; + + return endAt > currentDate.getTime(); + }); + const occupiedPrinters = currentPrintJobs.map((job) => job.printer.id); + const freePrinters = printers.filter((printer) => !occupiedPrinters.includes(printer.id)); + const printerUtilization = calculatePrinterUtilization(printJobs); + const printerVolume = calculatePrintVolumes(printJobs); + const printerAbortReasons = calculateAbortReasonsCount(printJobs); + const printerErrorRate = calculatePrinterErrorRate(printJobs); + const printerForecast = forecastPrinterUsage(printJobs); + + return ( + <> + + + Allgemein + Druckerauslastung + Fehlerberichte + Prognosen + + +
+
+ +
+ + +
+
+ +
+
+ +
+ {printerUtilization.map((data) => ( + + ))} +
+
+ +
+
+ +
+
+ +
+
+
+ +
+
+ ({ + day: index, + usageMinutes, + }))} + /> +
+
+
+
+ + ); +} + +================ +File: src/app/admin/printers/columns.tsx +================ +"use client"; +import type { printers } from "@/server/db/schema"; +import type { ColumnDef } from "@tanstack/react-table"; +import type { InferSelectModel } from "drizzle-orm"; +import { ArrowUpDown, MoreHorizontal, PencilIcon } from "lucide-react"; + +import { EditPrinterDialogContent, EditPrinterDialogTrigger } from "@/app/admin/printers/dialogs/edit-printer"; +import { Button } from "@/components/ui/button"; +import { Dialog } from "@/components/ui/dialog"; +import { + DropdownMenu, + DropdownMenuContent, + DropdownMenuItem, + DropdownMenuLabel, + DropdownMenuTrigger, +} from "@/components/ui/dropdown-menu"; +import { type PrinterStatus, translatePrinterStatus } from "@/utils/printers"; +import { useState } from "react"; + +// This type is used to define the shape of our data. +// You can use a Zod schema here if you want. + +export const columns: ColumnDef>[] = [ + { + accessorKey: "id", + header: ({ column }) => { + return ( + + ); + }, + }, + { + accessorKey: "name", + header: "Name", + }, + { + accessorKey: "description", + header: "Beschreibung", + }, + { + accessorKey: "status", + header: "Status", + cell: ({ row }) => { + const status = row.getValue("status"); + const translated = translatePrinterStatus(status as PrinterStatus); + + return translated; + }, + }, + { + id: "actions", + cell: ({ row }) => { + const printer = row.original; + const [open, setOpen] = useState(false); + + return ( + + + + + + + Aktionen + ABC + + +
+ + Bearbeiten +
+
+
+
+
+ +
+ ); + }, + }, +]; + +================ +File: src/app/admin/printers/data-table.tsx +================ +"use client"; + +import { + type ColumnDef, + type ColumnFiltersState, + type SortingState, + type VisibilityState, + flexRender, + getCoreRowModel, + getFilteredRowModel, + getPaginationRowModel, + getSortedRowModel, + useReactTable, +} from "@tanstack/react-table"; + +import { Button } from "@/components/ui/button"; +import { + DropdownMenu, + DropdownMenuCheckboxItem, + DropdownMenuContent, + DropdownMenuTrigger, +} from "@/components/ui/dropdown-menu"; +import { Input } from "@/components/ui/input"; +import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/components/ui/table"; +import { SlidersHorizontalIcon } from "lucide-react"; +import { useState } from "react"; + +interface DataTableProps { + columns: ColumnDef[]; + data: TData[]; +} + +export function DataTable({ columns, data }: DataTableProps) { + const [sorting, setSorting] = useState([]); + const [columnFilters, setColumnFilters] = useState([]); + const [columnVisibility, setColumnVisibility] = useState({}); + + const table = useReactTable({ + data, + columns, + getCoreRowModel: getCoreRowModel(), + getPaginationRowModel: getPaginationRowModel(), + onSortingChange: setSorting, + getSortedRowModel: getSortedRowModel(), + onColumnFiltersChange: setColumnFilters, + getFilteredRowModel: getFilteredRowModel(), + onColumnVisibilityChange: setColumnVisibility, + state: { + sorting, + columnFilters, + columnVisibility, + }, + }); + + return ( +
+
+ table.getColumn("name")?.setFilterValue(event.target.value)} + className="max-w-sm" + /> + + + + + + {table + .getAllColumns() + .filter((column) => column.getCanHide()) + .map((column) => { + return ( + column.toggleVisibility(!!value)} + > + {column.id} + + ); + })} + + +
+ +
+ + + {table.getHeaderGroups().map((headerGroup) => ( + + {headerGroup.headers.map((header) => { + return ( + + {header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext())} + + ); + })} + + ))} + + + {table.getRowModel().rows?.length ? ( + table.getRowModel().rows.map((row) => ( + + {row.getVisibleCells().map((cell) => ( + {flexRender(cell.column.columnDef.cell, cell.getContext())} + ))} + + )) + ) : ( + + + Keine Ergebnisse gefunden. + + + )} + +
+
+
+ + +
+
+ ); +} + +================ +File: src/app/admin/printers/dialogs/create-printer.tsx +================ +"use client"; + +import { PrinterForm } from "@/app/admin/printers/form"; +import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogTrigger } from "@/components/ui/dialog"; +import { useState } from "react"; + +interface CreatePrinterDialogProps { + children: React.ReactNode; +} + +export function CreatePrinterDialog(props: CreatePrinterDialogProps) { + const { children } = props; + const [open, setOpen] = useState(false); + + return ( + + {children} + + + Drucker erstellen + + + + + ); +} + +================ +File: src/app/admin/printers/dialogs/delete-printer.tsx +================ +"use client"; + +import { + AlertDialog, + AlertDialogAction, + AlertDialogCancel, + AlertDialogContent, + AlertDialogDescription, + AlertDialogFooter, + AlertDialogHeader, + AlertDialogTitle, + AlertDialogTrigger, +} from "@/components/ui/alert-dialog"; +import { Button } from "@/components/ui/button"; +import { useToast } from "@/components/ui/use-toast"; +import { deletePrinter } from "@/server/actions/printers"; +import { TrashIcon } from "lucide-react"; + +interface DeletePrinterDialogProps { + printerId: string; + setOpen: (state: boolean) => void; +} +export function DeletePrinterDialog(props: DeletePrinterDialogProps) { + const { printerId, setOpen } = props; + const { toast } = useToast(); + + async function onSubmit() { + toast({ + description: "Drucker wird gelöscht...", + }); + try { + const result = await deletePrinter(printerId); + if (result?.error) { + toast({ + description: result.error, + variant: "destructive", + }); + } + toast({ + description: "Drucker wurde gelöscht.", + }); + setOpen(false); + } catch (error) { + if (error instanceof Error) { + toast({ + description: error.message, + variant: "destructive", + }); + } else { + toast({ + description: "Ein unbekannter Fehler ist aufgetreten.", + variant: "destructive", + }); + } + } + } + + return ( + + + + + + + Bist Du dir sicher? + + Diese Aktion kann nicht rückgängig gemacht werden. Der Drucker und die damit verbundenen Daten werden + unwiderruflich gelöscht. + + + + Abbrechen + + Ja, löschen + + + + + ); +} + +================ +File: src/app/admin/printers/dialogs/edit-printer.tsx +================ +import { PrinterForm } from "@/app/admin/printers/form"; +import { DialogContent, DialogHeader, DialogTitle, DialogTrigger } from "@/components/ui/dialog"; +import type { InferResultType } from "@/utils/drizzle"; + +interface EditPrinterDialogTriggerProps { + children: React.ReactNode; +} + +export function EditPrinterDialogTrigger(props: EditPrinterDialogTriggerProps) { + const { children } = props; + + return {children}; +} + +interface EditPrinterDialogContentProps { + printer: InferResultType<"printers">; + setOpen: (open: boolean) => void; +} +export function EditPrinterDialogContent(props: EditPrinterDialogContentProps) { + const { printer, setOpen } = props; + + return ( + + + Drucker bearbeiten + + + + ); +} + +================ +File: src/app/admin/printers/form.tsx +================ +"use client"; +import { DeletePrinterDialog } from "@/app/admin/printers/dialogs/delete-printer"; +import { Button } from "@/components/ui/button"; +import { DialogClose } from "@/components/ui/dialog"; +import { Form, FormControl, FormDescription, FormField, FormItem, FormLabel, FormMessage } from "@/components/ui/form"; +import { Input } from "@/components/ui/input"; +import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"; +import { useToast } from "@/components/ui/use-toast"; +import { createPrinter, updatePrinter } from "@/server/actions/printers"; +import type { InferResultType } from "@/utils/drizzle"; +import { zodResolver } from "@hookform/resolvers/zod"; +import { SaveIcon, XCircleIcon } from "lucide-react"; +import { useForm } from "react-hook-form"; +import { z } from "zod"; + +export const formSchema = z.object({ + name: z + .string() + .min(2, { + message: "Der Name muss mindestens 2 Zeichen lang sein.", + }) + .max(50), + description: z + .string() + .min(2, { + message: "Die Beschreibung muss mindestens 2 Zeichen lang sein.", + }) + .max(50), + status: z.coerce.number().int().min(0).max(1), +}); + +interface PrinterFormProps { + printer?: InferResultType<"printers">; + setOpen: (state: boolean) => void; +} + +export function PrinterForm(props: PrinterFormProps) { + const { printer, setOpen } = props; + const { toast } = useToast(); + + const form = useForm>({ + resolver: zodResolver(formSchema), + defaultValues: { + name: printer?.name ?? "", + description: printer?.description ?? "", + status: printer?.status ?? 0, + }, + }); + + // 2. Define a submit handler. + async function onSubmit(values: z.infer) { + // TODO: create or update + if (printer) { + toast({ + description: "Drucker wird aktualisiert...", + }); + + // Update + try { + const result = await updatePrinter(printer.id, { + description: values.description, + name: values.name, + status: values.status, + }); + if (result?.error) { + toast({ + description: result.error, + variant: "destructive", + }); + } + + setOpen(false); + + toast({ + description: "Drucker wurde aktualisiert.", + variant: "default", + }); + } catch (error) { + if (error instanceof Error) { + toast({ + description: error.message, + variant: "destructive", + }); + } else { + toast({ + description: "Ein unbekannter Fehler ist aufgetreten.", + variant: "destructive", + }); + } + } + } else { + toast({ + description: "Drucker wird erstellt...", + variant: "default", + }); + + // Create + try { + const result = await createPrinter({ + description: values.description, + name: values.name, + status: values.status, + }); + if (result?.error) { + toast({ + description: result.error, + variant: "destructive", + }); + } + + setOpen(false); + + toast({ + description: "Drucker wurde erstellt.", + variant: "default", + }); + } catch (error) { + if (error instanceof Error) { + toast({ + description: error.message, + variant: "destructive", + }); + } else { + toast({ + description: "Ein unbekannter Fehler ist aufgetreten.", + variant: "destructive", + }); + } + } + } + } + + return ( +
+ + ( + + Name + + + + Bitte gib einen eindeutigen Namen für den Drucker ein. + + + )} + /> + ( + + Beschreibung + + + + Füge eine kurze Beschreibung des Druckers hinzu. + + + )} + /> + ( + + Status + + Wähle den aktuellen Status des Druckers. + + + )} + /> +
+ {printer && } + {!printer && ( + + + + )} + +
+ + + ); +} + +================ +File: src/app/admin/printers/page.tsx +================ +import { columns } from "@/app/admin/printers/columns"; +import { DataTable } from "@/app/admin/printers/data-table"; +import { CreatePrinterDialog } from "@/app/admin/printers/dialogs/create-printer"; +import { Button } from "@/components/ui/button"; +import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"; +import { db } from "@/server/db"; +import { PlusCircleIcon } from "lucide-react"; + +export default async function AdminPage() { + const data = await db.query.printers.findMany(); + + return ( + + +
+ Druckerverwaltung + Suche, Bearbeite, Lösche und Erstelle Drucker +
+ + + +
+ + + +
+ ); +} + +================ +File: src/app/admin/settings/download/route.ts +================ +import fs from "node:fs"; + +export const dynamic = 'force-dynamic'; + +export async function GET() { + return new Response(fs.readFileSync("./db/sqlite.db")); +} + +================ +File: src/app/admin/settings/page.tsx +================ +import { Button } from "@/components/ui/button"; +import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"; +import Link from "next/link"; + +import type { Metadata } from "next"; + +export const metadata: Metadata = { + title: "Systemeinstellungen", +}; + +export default function AdminPage() { + return ( + + + Einstellungen + Systemeinstellungen + + +
+

Datenbank herunterladen

+ +
+
+
+ ); +} + +================ +File: src/app/admin/users/columns.tsx +================ +"use client"; + +import { type UserRole, translateUserRole } from "@/server/auth/permissions"; +import type { users } from "@/server/db/schema"; +import type { ColumnDef } from "@tanstack/react-table"; +import type { InferSelectModel } from "drizzle-orm"; +import { + ArrowUpDown, + MailIcon, + MessageCircleIcon, + MoreHorizontal, +} from "lucide-react"; + +import { Button } from "@/components/ui/button"; +import { + DropdownMenu, + DropdownMenuContent, + DropdownMenuItem, + DropdownMenuLabel, + DropdownMenuSeparator, + DropdownMenuTrigger, +} from "@/components/ui/dropdown-menu"; +import Link from "next/link"; +import { + EditUserDialogContent, + EditUserDialogRoot, + EditUserDialogTrigger, +} from "@/app/admin/users/dialog"; + +// This type is used to define the shape of our data. +// You can use a Zod schema here if you want. +export type User = { + id: string; + github_id: number; + username: string; + displayName: string; + email: string; + role: string; +}; + +export const columns: ColumnDef>[] = [ + { + accessorKey: "id", + header: ({ column }) => { + return ( + + ); + }, + }, + { + accessorKey: "github_id", + header: "GitHub ID", + }, + { + accessorKey: "username", + header: "Username", + }, + { + accessorKey: "displayName", + header: "Name", + }, + { + accessorKey: "email", + header: "E-Mail", + }, + { + accessorKey: "role", + header: "Rolle", + cell: ({ row }) => { + const role = row.getValue("role"); + const translated = translateUserRole(role as UserRole); + + return translated; + }, + }, + { + id: "actions", + cell: ({ row }) => { + const user = row.original; + + return ( + + + + + + + Aktionen + + + + Teams-Chat öffnen + + + + + + E-Mail schicken + + + + + + + + + + + ); + }, + }, +]; + +function generateTeamsChatURL(email: string) { + return `https://teams.microsoft.com/l/chat/0/0?users=${email}`; +} + +function generateEMailURL(email: string) { + return `mailto:${email}`; +} + +================ +File: src/app/admin/users/data-table.tsx +================ +"use client"; + +import { + type ColumnDef, + type ColumnFiltersState, + type SortingState, + type VisibilityState, + flexRender, + getCoreRowModel, + getFilteredRowModel, + getPaginationRowModel, + getSortedRowModel, + useReactTable, +} from "@tanstack/react-table"; + +import { Button } from "@/components/ui/button"; +import { + DropdownMenu, + DropdownMenuCheckboxItem, + DropdownMenuContent, + DropdownMenuTrigger, +} from "@/components/ui/dropdown-menu"; +import { Input } from "@/components/ui/input"; +import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/components/ui/table"; +import { SlidersHorizontalIcon } from "lucide-react"; +import { useState } from "react"; + +interface DataTableProps { + columns: ColumnDef[]; + data: TData[]; +} + +export function DataTable({ columns, data }: DataTableProps) { + const [sorting, setSorting] = useState([]); + const [columnFilters, setColumnFilters] = useState([]); + const [columnVisibility, setColumnVisibility] = useState({}); + + const table = useReactTable({ + data, + columns, + getCoreRowModel: getCoreRowModel(), + getPaginationRowModel: getPaginationRowModel(), + onSortingChange: setSorting, + getSortedRowModel: getSortedRowModel(), + onColumnFiltersChange: setColumnFilters, + getFilteredRowModel: getFilteredRowModel(), + onColumnVisibilityChange: setColumnVisibility, + state: { + sorting, + columnFilters, + columnVisibility, + }, + }); + + return ( +
+
+ table.getColumn("email")?.setFilterValue(event.target.value)} + className="max-w-sm" + /> + + + + + + {table + .getAllColumns() + .filter((column) => column.getCanHide()) + .map((column) => { + return ( + column.toggleVisibility(!!value)} + > + {column.id} + + ); + })} + + +
+ +
+ + + {table.getHeaderGroups().map((headerGroup) => ( + + {headerGroup.headers.map((header) => { + return ( + + {header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext())} + + ); + })} + + ))} + + + {table.getRowModel().rows?.length ? ( + table.getRowModel().rows.map((row) => ( + + {row.getVisibleCells().map((cell) => ( + {flexRender(cell.column.columnDef.cell, cell.getContext())} + ))} + + )) + ) : ( + + + Keine Ergebnisse gefunden. + + + )} + +
+
+
+ + +
+
+ ); +} + +================ +File: src/app/admin/users/dialog.tsx +================ +import type { User } from "@/app/admin/users/columns"; +import { ProfileForm } from "@/app/admin/users/form"; +import { + Dialog, + DialogContent, + DialogDescription, + DialogHeader, + DialogTitle, + DialogTrigger, +} from "@/components/ui/dialog"; +import { PencilIcon } from "lucide-react"; + +interface EditUserDialogRootProps { + children: React.ReactNode; +} + +export function EditUserDialogRoot(props: EditUserDialogRootProps) { + const { children } = props; + + return {children}; +} + +export function EditUserDialogTrigger() { + return ( + + + Benutzer bearbeiten + + ); +} + +interface EditUserDialogContentProps { + user: User; +} + +export function EditUserDialogContent(props: EditUserDialogContentProps) { + const { user } = props; + + if (!user) { + return; + } + + return ( + + + Benutzer bearbeiten + + Hinweis: In den seltensten Fällen sollten die Daten + eines Benutzers geändert werden. Dies kann zu unerwarteten Problemen + führen. + + + + + ); +} + +================ +File: src/app/admin/users/form.tsx +================ +"use client"; + +import type { User } from "@/app/admin/users/columns"; +import { + AlertDialog, + AlertDialogAction, + AlertDialogCancel, + AlertDialogContent, + AlertDialogDescription, + AlertDialogFooter, + AlertDialogHeader, + AlertDialogTitle, + AlertDialogTrigger, +} from "@/components/ui/alert-dialog"; +import { Button } from "@/components/ui/button"; +import { DialogClose } from "@/components/ui/dialog"; +import { + Form, + FormControl, + FormDescription, + FormField, + FormItem, + FormLabel, + FormMessage, +} from "@/components/ui/form"; +import { Input } from "@/components/ui/input"; +import { + Select, + SelectContent, + SelectItem, + SelectTrigger, + SelectValue, +} from "@/components/ui/select"; +import { useToast } from "@/components/ui/use-toast"; +import { deleteUser, updateUser } from "@/server/actions/users"; +import type { UserRole } from "@/server/auth/permissions"; +import { zodResolver } from "@hookform/resolvers/zod"; +import { SaveIcon, TrashIcon } from "lucide-react"; +import { useForm } from "react-hook-form"; +import { z } from "zod"; + +export const formSchema = z.object({ + username: z + .string() + .min(2, { + message: "Der Benutzername muss mindestens 2 Zeichen lang sein.", + }) + .max(50), + displayName: z + .string() + .min(2, { + message: "Der Anzeigename muss mindestens 2 Zeichen lang sein.", + }) + .max(50), + email: z.string().email(), + role: z.enum(["admin", "user", "guest"]), +}); + +interface ProfileFormProps { + user: User; +} + +export function ProfileForm(props: ProfileFormProps) { + const { user } = props; + const { toast } = useToast(); + + // 1. Define your form. + const form = useForm>({ + resolver: zodResolver(formSchema), + defaultValues: { + username: user.username, + displayName: user.displayName, + email: user.email, + role: user.role as UserRole, + }, + }); + + // 2. Define a submit handler. + async function onSubmit(values: z.infer) { + toast({ description: "Benutzerprofil wird aktualisiert..." }); + + await updateUser(user.id, values); + + toast({ description: "Benutzerprofil wurde aktualisiert." }); + } + + return ( +
+ + ( + + Benutzername + + + + + Nur in Ausnahmefällen sollte der Benutzername geändert werden. + + + + )} + /> + ( + + Anzeigename + + + + + Der Anzeigename darf frei verändert werden. + + + + )} + /> + ( + + E-Mail Adresse + + + + + Nur in Ausnahmefällen sollte die E-Mail Adresse geändert werden. + + + + )} + /> + ( + + Benutzerrolle + + + Die Benutzerrolle bestimmt die Berechtigungen des Benutzers. + + + + )} + /> +
+ + + + + + + Bist du dir sicher? + + Diese Aktion kann nicht rückgängig gemacht werden. Das + Benutzerprofil und die damit verbundenen Daten werden + unwiderruflich gelöscht. + + + + Abbrechen + { + toast({ description: "Benutzerprofil wird gelöscht..." }); + deleteUser(user.id); + toast({ description: "Benutzerprofil wurde gelöscht." }); + }} + > + Ja, löschen + + + + + + + +
+ + + ); +} + +================ +File: src/app/admin/users/page.tsx +================ +import { columns } from "@/app/admin/users/columns"; +import { DataTable } from "@/app/admin/users/data-table"; +import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"; +import { db } from "@/server/db"; + +import type { Metadata } from "next"; + +export const metadata: Metadata = { + title: "Alle Benutzer", +}; + +export default async function AdminPage() { + const data = await db.query.users.findMany(); + + return ( + + + Benutzerverwaltung + Suche, Bearbeite und Lösche Benutzer + + + + + + ); +} + +================ +File: src/app/api/job/[jobId]/remaining-time/route.ts +================ +import { db } from "@/server/db"; +import { printJobs } from "@/server/db/schema"; +import { eq } from "drizzle-orm"; + +export const dynamic = "force-dynamic"; + +interface RemainingTimeRouteProps { + params: { + jobId: string; + }; +} +export async function GET(request: Request, { params }: RemainingTimeRouteProps) { + // Trying to fix build error in container... + if (params.jobId === undefined) { + return Response.json({}); + } + + // Get the job details + const jobDetails = await db.query.printJobs.findFirst({ + where: eq(printJobs.id, params.jobId), + }); + + // Check if the job exists + if (!jobDetails) { + return Response.json({ + id: params.jobId, + error: "Job not found", + }); + } + + // Calculate the remaining time + const startAt = new Date(jobDetails.startAt).getTime(); + const endAt = startAt + jobDetails.durationInMinutes * 60 * 1000; + const remainingTime = Math.max(0, endAt - Date.now()); + + // Return the remaining time + return Response.json({ + id: params.jobId, + remainingTime, + }); +} + +================ +File: src/app/api/printers/route.ts +================ +import { getPrinters } from "@/server/actions/printers"; + +export const dynamic = "force-dynamic"; + +export async function GET() { + const printers = await getPrinters(); + + return Response.json(printers); +} + +================ +File: src/app/auth/login/callback/route.ts +================ +import { lucia } from "@/server/auth"; +import { type GitHubUserResult, github } from "@/server/auth/oauth"; +import { db } from "@/server/db"; +import { users } from "@/server/db/schema"; +import { OAuth2RequestError } from "arctic"; +import { eq } from "drizzle-orm"; +import { generateIdFromEntropySize } from "lucia"; +import { cookies } from "next/headers"; + +export const dynamic = "force-dynamic"; + +interface GithubEmailResponse { + email: string; + primary: boolean; + verified: boolean; + visibility: string; +} + +export async function GET(request: Request): Promise { + const url = new URL(request.url); + const code = url.searchParams.get("code"); + const state = url.searchParams.get("state"); + const storedState = cookies().get("github_oauth_state")?.value ?? null; + if (!code || !state || !storedState || state !== storedState) { + return new Response( + JSON.stringify({ + status_text: "Something is wrong", + data: { code, state, storedState }, + }), + { + status: 400, + }, + ); + } + + try { + const tokens = await github.validateAuthorizationCode(code); + const githubUserResponse = await fetch("https://git.i.mercedes-benz.com/api/v3/user", { + headers: { + Authorization: `Bearer ${tokens.accessToken}`, + }, + }); + const githubUser: GitHubUserResult = await githubUserResponse.json(); + + // Sometimes email can be null in the user query. + if (githubUser.email === null || githubUser.email === undefined) { + const githubEmailResponse = await fetch("https://git.i.mercedes-benz.com/api/v3/user/emails", { + headers: { + Authorization: `Bearer ${tokens.accessToken}`, + }, + }); + const githubUserEmail: GithubEmailResponse[] = await githubEmailResponse.json(); + githubUser.email = githubUserEmail[0].email; + } + const existingUser = await db.query.users.findFirst({ + where: eq(users.github_id, githubUser.id), + }); + + if (existingUser) { + const session = await lucia.createSession(existingUser.id, {}); + const sessionCookie = lucia.createSessionCookie(session.id); + cookies().set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes); + return new Response(null, { + status: 302, + headers: { + Location: "/", + }, + }); + } + + const userId = generateIdFromEntropySize(10); // 16 characters long + + await db.insert(users).values({ + id: userId, + github_id: githubUser.id, + username: githubUser.login, + displayName: githubUser.name, + email: githubUser.email, + }); + + const session = await lucia.createSession(userId, {}); + const sessionCookie = lucia.createSessionCookie(session.id); + cookies().set(sessionCookie.name, sessionCookie.value, { + ...sessionCookie.attributes, + secure: false, // Else cookie does not get set cause IT has not provided us an SSL certificate yet + }); + return new Response(null, { + status: 302, + headers: { + Location: "/", + }, + }); + } catch (e) { + // the specific error message depends on the provider + if (e instanceof OAuth2RequestError) { + // invalid code + return new Response( + JSON.stringify({ + status_text: "Invalid code", + error: JSON.stringify(e), + }), + { + status: 400, + }, + ); + } + return new Response(null, { + status: 500, + }); + } +} + +================ +File: src/app/auth/login/route.ts +================ +import { github } from "@/server/auth/oauth"; +import { generateState } from "arctic"; +import { cookies } from "next/headers"; + +export const dynamic = "force-dynamic"; + +export async function GET(): Promise { + const state = generateState(); + const url = await github.createAuthorizationURL(state, { + scopes: ["user"], + }); + const ONE_HOUR = 60 * 60; + + cookies().set("github_oauth_state", state, { + path: "/", + secure: false, //process.env.NODE_ENV === "production", -- can't be used until SSL certificate is provided by IT + httpOnly: true, + maxAge: ONE_HOUR, + sameSite: "lax", + }); + + return Response.redirect(url); +} + +================ +File: src/app/globals.css +================ +@tailwind base; +@tailwind components; +@tailwind utilities; + + +@layer base { + :root { + --background: 0 0% 100%; + --foreground: 222.2 84% 4.9%; + --card: 0 0% 100%; + --card-foreground: 222.2 84% 4.9%; + --popover: 0 0% 100%; + --popover-foreground: 222.2 84% 4.9%; + --primary: 221.2 83.2% 53.3%; + --primary-foreground: 210 40% 98%; + --secondary: 210 40% 96.1%; + --secondary-foreground: 222.2 47.4% 11.2%; + --muted: 210 40% 96.1%; + --muted-foreground: 215.4 16.3% 46.9%; + --accent: 210 40% 96.1%; + --accent-foreground: 222.2 47.4% 11.2%; + --destructive: 0 84.2% 60.2%; + --destructive-foreground: 210 40% 98%; + --border: 214.3 31.8% 91.4%; + --input: 214.3 31.8% 91.4%; + --ring: 221.2 83.2% 53.3%; + --radius: 0.75rem; + --chart-1: 12 76% 61%; + --chart-2: 173 58% 39%; + --chart-3: 197 37% 24%; + --chart-4: 43 74% 66%; + --chart-5: 27 87% 67%; + } + + .dark { + --background: 222.2 84% 4.9%; + --foreground: 210 40% 98%; + --card: 222.2 84% 4.9%; + --card-foreground: 210 40% 98%; + --popover: 222.2 84% 4.9%; + --popover-foreground: 210 40% 98%; + --primary: 217.2 91.2% 59.8%; + --primary-foreground: 222.2 47.4% 11.2%; + --secondary: 217.2 32.6% 17.5%; + --secondary-foreground: 210 40% 98%; + --muted: 217.2 32.6% 17.5%; + --muted-foreground: 215 20.2% 65.1%; + --accent: 217.2 32.6% 17.5%; + --accent-foreground: 210 40% 98%; + --destructive: 0 62.8% 30.6%; + --destructive-foreground: 210 40% 98%; + --border: 217.2 32.6% 17.5%; + --input: 217.2 32.6% 17.5%; + --ring: 224.3 76.3% 48%; + --chart-1: 220 70% 50%; + --chart-2: 160 60% 45%; + --chart-3: 30 80% 55%; + --chart-4: 280 65% 60%; + --chart-5: 340 75% 55%; + } +} + +================ +File: src/app/job/[jobId]/cancel-form.tsx +================ +"use client"; + +import { zodResolver } from "@hookform/resolvers/zod"; +import { useForm } from "react-hook-form"; +import { z } from "zod"; + +import { Button } from "@/components/ui/button"; +import { + Dialog, + DialogClose, + DialogContent, + DialogDescription, + DialogHeader, + DialogTitle, + DialogTrigger, +} from "@/components/ui/dialog"; +import { Form, FormControl, FormDescription, FormField, FormItem, FormLabel, FormMessage } from "@/components/ui/form"; +import { Input } from "@/components/ui/input"; +import { useToast } from "@/components/ui/use-toast"; +import { abortPrintJob } from "@/server/actions/printJobs"; +import { TriangleAlertIcon } from "lucide-react"; +import { useState } from "react"; + +const formSchema = z.object({ + abortReason: z + .string() + .min(1, { + message: "Bitte gebe einen Grund für den Abbruch an.", + }) + .max(255, { + message: "Der Grund darf maximal 255 Zeichen lang sein.", + }), +}); + +interface CancelFormProps { + jobId: string; +} + +export function CancelForm(props: CancelFormProps) { + const { jobId } = props; + const form = useForm>({ + resolver: zodResolver(formSchema), + defaultValues: { + abortReason: "", + }, + }); + const { toast } = useToast(); + const [open, setOpen] = useState(false); + + async function onSubmit(values: z.infer) { + toast({ + description: "Druckauftrag wird abgebrochen...", + }); + try { + const result = await abortPrintJob(jobId, values.abortReason); + if (result?.error) { + toast({ + description: result.error, + variant: "destructive", + }); + } + setOpen(false); + toast({ + description: "Druckauftrag wurde abgebrochen.", + }); + } catch (error) { + if (error instanceof Error) { + toast({ + description: error.message, + variant: "destructive", + }); + } else { + toast({ + description: "Ein unbekannter Fehler ist aufgetreten.", + variant: "destructive", + }); + } + } + } + + return ( + + + + + + + Druckauftrag abbrechen? + + Du bist dabei, den Druckauftrag abzubrechen. Bitte beachte, dass ein abgebrochener Druckauftrag nicht wieder + aufgenommen werden kann und der Drucker sich automatisch abschaltet. + + +
+ + ( + + Grund für den Abbruch + + + + + Bitte teile uns den Grund für den Abbruch des Druckauftrags mit. Wenn der Drucker eine Fehlermeldung + anzeigt, gib bitte nur diese Fehlermeldung an. + + + + )} + /> +
+ + + + +
+ + +
+
+ ); +} + +================ +File: src/app/job/[jobId]/edit-comments.tsx +================ +"use client"; + +import { Label } from "@/components/ui/label"; +import { Textarea } from "@/components/ui/textarea"; +import { useToast } from "@/components/ui/use-toast"; +import { updatePrintComments } from "@/server/actions/printJobs"; +import { useDebouncedCallback } from "use-debounce"; + +interface EditCommentsProps { + defaultValue: string | null; + jobId: string; + disabled?: boolean; +} +export function EditComments(props: EditCommentsProps) { + const { defaultValue, jobId, disabled } = props; + const { toast } = useToast(); + + const debounced = useDebouncedCallback(async (value) => { + try { + const result = await updatePrintComments(jobId, value); + if (result?.error) { + toast({ + description: result.error, + variant: "destructive", + }); + } + toast({ + description: "Anmerkungen wurden gespeichert.", + }); + } catch (error) { + if (error instanceof Error) { + toast({ + description: error.message, + variant: "destructive", + }); + } else { + toast({ + description: "Ein unbekannter Fehler ist aufgetreten.", + variant: "destructive", + }); + } + } + }, 1000); + + return ( +
+ +