Neuer Abschnitt "Anvisiertes Nutzungsszenario" erklärt: - Einsatz in Flexnow für PDF-Dokumente (Urkunden, Zeugnisse, Bescheide) - Struktur der ~100 verknüpften XSL-Dateien - Typischer Entwicklungs-Workflow mit PDF-Diff-Prüfung - Wichtigkeit von RAM-sparsamem Design 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
8.8 KiB
CLAUDE.md
Spreche mit mir auf Deutsch! (Communicate with me in German!)
Projektübersicht
DocuMentor (ehemals xsl-validator) ist eine PySide6-basierte Desktop-Anwendung zur Verwaltung und Validierung von XSL-Transformationen mit XML-Dateien. Sie bietet eine GUI zur Konfiguration von Transformations-Toolchains (Saxon, Apache FOP, diff-pdf) und zur Verwaltung von PDF-Generierungsprojekten mit PostgreSQL-Datenbankintegration.
Anvisiertes Nutzungsszenario
Der primäre Einsatz ist die kontinuierliche Weiterentwicklung von PDF-Dokumenten in Flexnow (Software zur Prüfungsverwaltung). Dabei handelt es sich beispielsweise um amtliche Urkunden, Zeugnisse und Bescheide.
Die Basis bilden etwa 100 XSL-Dateien. Die meisten sind mittels <xsl:import/> bzw. <xsl:include/> miteinander verknüpft (ähnlich der Klassen-Vererbung). Daher können sich Änderungen in einer XSL-Datei auf (unerwartet) viele andere auswirken. Um diese Auswirkungen im Auge zu behalten, wird DocuMentor entwickelt.
Typischer Workflow:
- Entwickler führt benötigte Änderungen an den XSL-Dateien durch
- Entwickler startet die Transformation im DocuMentor und begutachtet die generierte PDF-Diff
- Prüfung: Wurden die richtigen PDF-Dateien geändert?
- Prüfung: Hat die Änderung der XSL-Dateien die erhoffte Änderung in den PDF-Dateien ergeben?
Diese Schritte können sich mehrfach wiederholen.
Da der DocuMentor permanent im Hintergrund läuft, ist ein sparsamer Umgang mit RAM wichtig.
PySide6-GUI
- Beim Erstellen neuer Dialoge und Fenster sollte immer eine entsprechende UI-Datei erstellt werden
- Der Entwickler sollte später in der Lage sein, den neuen Dialog bzw. Fenster über diese UI-Datei zu gestalten
- Aus der UI-Datei wird in Visual Studio Code über eine Erweiterung automatisch eine .py-Datei erzeugt
- Die automatisch generierte .py-Datei muss in den Code eingebunden und verwendet werden
Entwicklungskommandos
Paketverwaltung
Dieses Projekt verwendet den uv Paketmanager (nicht pip oder poetry):
uv sync # Abhängigkeiten installieren
uv run python src/main.py # Anwendung starten
uv run python test_hash_implementation.py # Hash-Tests ausführen
Linting
uv run ruff check # Code-Style prüfen (Zeilenlänge: 120)
uv run ruff format # Code formatieren
Architektur
Konfigurationssystem (src/conf.py)
Die Anwendung verwendet ein zentralisiertes Konfigurationsmodell mit Pydantic:
-
AppSettings: Globales Singleton (
app_settings), das die gesamte Anwendungskonfiguration speichert- Wird an plattformspezifischen Orten gespeichert:
- Linux:
~/.config/DocuMentor/config.json - Windows:
%APPDATA%\DocuMentor\config.json - macOS:
~/Library/Application Support/DocuMentor/config.json
- Linux:
- Enthält Listen von Tools:
java_vms,saxon_jars,apache_fops,diff_pdfs,xsl_dirs,postgresql_dbs
- Wird an plattformspezifischen Orten gespeichert:
-
ProjectData: Projektspezifische Einstellungen, die in
project.yamlim jeweiligen Projektverzeichnis gespeichert werden- Enthält hierarchische Baumstruktur von Transformationsknoten
- Verwendet
TreeNodeundXslFilezur Organisation - Jede
XmlFilehat eine optionalehashsum(blake2b) zur Änderungsverfolgung
Wichtige Datenmodelle
-
Tool-Konfigurationsmodelle (JavaVm, SaxonJar, ApacheFop, DiffPdf, XslDir, PostgreSqlDb):
- Jedes hat eine
idundversion - Speichert Pfade zu Binärdateien/Verzeichnissen
- Jedes hat eine
-
Project-Modell:
- Referenziert Tool-Konfigurationen über ID
- Verlinkt zu einem Projektverzeichnis mit
project.yaml - Hat Hilfsmethoden wie
getXsl(),getJavaVm()um IDs in Namen/Versionen aufzulösen
-
Baumstruktur (TreeNode → XslFile → XmlFile):
- Hierarchische Organisation von Transformations-Workflows
TreeNode: Organisationseinheit mitxslt_paramsund Kindknoten/-dateienXslFile: XSL-Stylesheet mit zugehörigen XML-Dateien und XSLT-ParameternXmlFile: XML-Eingabedatei mit optionalem blake2b-Hash
UI-Architektur (src/ui/)
Die Anwendung folgt einem spezifischen PySide6-Muster:
-
UI-Definitionsdateien (
*_ui.py): Automatisch generiert aus UI-Designer-Dateien- Diese Dateien definieren die UI-Struktur als Klassen (z.B.
Ui_MainWindow) - Sollten NICHT manuell bearbeitet werden
- Diese Dateien definieren die UI-Struktur als Klassen (z.B.
-
Implementierungsdateien (ohne
_uiSuffix): Tatsächliche Dialog-/Fenster-Implementierungen- Importieren und verwenden die entsprechende
*_ui.pyDatei - Enthalten Business-Logik und Signal/Slot-Verbindungen
- Beispiel:
MainWindow.pyverwendetUi_MainWindowausMainWinddow_ui.py
- Importieren und verwenden die entsprechende
Beim Erstellen neuer Dialoge:
- Immer zuerst eine entsprechende UI-Datei erstellen
- Die UI-Datei wird automatisch als
.py-Datei von einer VS Code Extension generiert - Die generierte UI-Klasse in der Implementierungsdatei importieren und verwenden
Hauptfenster (src/ui/MainWindow.py)
Zentrale Schaltstelle der Anwendung mit mehreren wichtigen Verantwortlichkeiten:
-
Projektverwaltung:
- Öffnet und verwaltet PDF-Transformationsprojekte
- Lädt/speichert
ProjectDataausproject.yamlDateien
-
Tree Widget: Zeigt hierarchische Struktur von Transformationsknoten an
- Kontextmenüs zum Hinzufügen/Bearbeiten/Löschen von Knoten, XSL-Dateien und XML-Dateien
- Drag-and-Drop-Unterstützung für XML-Dateien
-
PDF-Vergleichsansicht:
- Drei-Panel-Ansicht (Referenz, Diff, Neu)
- Alpha-Blending für visuellen Vergleich
- Zoom- und Pan-Funktionalität
-
Asynchrone Operationen:
XmlHashCalculatorThread: Hintergrund-blake2b-Hash-Berechnung für XML-DateienDatabaseTestThread(in PostgreSqlConfigDialog): Asynchrones Testen von Datenbankverbindungen
Hash-Berechnungssystem
Die Anwendung verwendet blake2b-Hashing zur Verfolgung von XML-Dateiänderungen:
- Automatisch: Hashes werden berechnet, wenn Projekte geladen werden (nur für Dateien ohne existierenden Hash)
- Asynchron: Hintergrund-Thread (
XmlHashCalculatorThread) um die UI reaktionsfähig zu halten - Format:
blake2b:<64-Zeichen-Hexdigest> - Speicherung: Persistiert in
project.yamlinnerhalb jedesXmlFile-Objekts - Details: Siehe
docs/blake2b_hash_implementation.md
Theme-System
Die Anwendung unterstützt mehrere Qt-Themes:
- Theme-Auswahlmenü wird dynamisch aus
QStyleFactory.keys()befüllt - Theme-Präferenz wird in
AppSettings.themegespeichert - Dark-Theme-Unterstützung via
qdarkthemePaket (aktuell in main.py auskommentiert)
Datenbankintegration
PostgreSQL-Integration mit Polars und ConnectorX:
- Konfiguration wird im
PostgreSqlDb-Modell mit SSL-Modus-Unterstützung gespeichert - SQL-Abfragen werden via
_execute_sql_query()im MainWindow ausgeführt - Ergebnisse werden in Polars DataFrames geladen
Wichtige Konventionen
Deutsche Sprache
Die Codebasis verwendet Deutsch für:
- UI-Texte und Labels
- Kommentare und Dokumentation
- Variablennamen wo kontextuell passend
- Log-Meldungen
Pfadbehandlung
- Immer
pathlib.Path-Objekte verwenden, keine Strings expanduser()undexpandvars()für Benutzer-/Umgebungspfade verwenden- Projektrelative Pfade werden als relativ gespeichert, zur Laufzeit gegen
project_diraufgelöst
ID-basierte Lookups
Konfigurationsentitäten (Tools, Datenbanken) werden in Projekten über ID referenziert. Die Hilfsmethoden des Project-Modells (getXsl(), getJavaVm(), etc.) verwenden, um IDs in Anzeigewerte aufzulösen.
Einstellungspersistenz
- Globale Einstellungen:
app_settings.save()nach Änderungen aufrufen - Projekteinstellungen:
project_data.writeSettings(project_dir)nach Änderungen aufrufen
Arbeiten mit der Codebasis
Neue Tool-Konfigurationen hinzufügen
- Modell zu
conf.pyhinzufügen (ähnlich wieJavaVm,SaxonJar) - Listenfeld zu
AppSettingshinzufügen - Konfigurationsdialog in
src/ui/erstellen (UI-Datei + Implementierung) - Zu
AppSettings.pyTabs hinzufügen Project-Modell aktualisieren, falls das Tool projektspezifisch sein soll
Neue Baumoperationen hinzufügen
- Aktion zum Kontextmenü in
_create_context_menu_for_type()hinzufügen - Handler-Methode implementieren nach Namensschema
_action_tree_node(),_action_xsl_file(), etc. - Baum nach Änderungen mit
_load_nodes_to_tree()aktualisieren self.project_data.writeSettings(self.project.project_dir)aufrufen um Änderungen zu persistieren
Projektstruktur modifizieren
Das ProjectData-Modell ist die Quelle der Wahrheit. Alle Änderungen an der Baumstruktur müssen:
- Die
project_data.nodesListe modifizieren project_data.writeSettings()aufrufen um zu persistieren- Baum mit
_load_nodes_to_tree()neu laden um Änderungen in der UI zu reflektieren