Merge pull request #10 from Kenearos/claude/duty-schedule-calculator-01HRgA7y2Auxt5K2BY6sWtrM

Claude/duty schedule calculator 01 h rg a7y2 auxt5 k2 by6s wtr m
This commit is contained in:
Kenearos 2025-11-18 21:29:44 +01:00 committed by GitHub
commit 35de2c27f0
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 915 additions and 1 deletions

View file

@ -95,6 +95,7 @@ Die Datei landet in `output/Dienstplan_YYYY_MM_NRW.xlsx`.
├── templates/ # Basis-Vorlage
├── requirements.txt # Python-Abhängigkeiten (openpyxl)
├── SPECIFICATION.md # Vollständige Regeln & Formeln
├── claude.md # Umfassende Projekt-Dokumentation
└── README.md # Diese Datei
```

201
webapp/TEST_GUIDE.md Normal file
View file

@ -0,0 +1,201 @@
# Test Suite - Dienstplan Bonusrechner
Automatische Test Suite für die Web-App.
## Schnellstart
1. **Server starten** (falls noch nicht gestartet):
```bash
cd webapp
python3 -m http.server 8000
```
2. **Test-Seite öffnen**:
```
http://localhost:8000/test.html
```
3. **Tests ausführen**:
- Klicken Sie auf "Alle Tests ausführen"
- Warten Sie auf die Ergebnisse
- ✅ = Test bestanden
- ❌ = Test fehlgeschlagen
## Was wird getestet?
### 1. Holiday Provider (NRW-Feiertage)
- ✅ Feiertage werden korrekt erkannt
- ✅ Normale Tage werden nicht als Feiertage erkannt
- ✅ Tag vor Feiertag wird erkannt
- ✅ Spezifische Feiertage (Fronleichnam, etc.)
### 2. Calculator - Tag-Klassifizierung
- ✅ Freitag ist qualifizierend
- ✅ Samstag ist qualifizierend
- ✅ Sonntag ist qualifizierend
- ✅ Normale Wochentage (Mo-Do) sind nicht qualifizierend
- ✅ Feiertage sind qualifizierend
- ✅ Tag vor Feiertag ist qualifizierend
### 3. Calculator - Bonusberechnung
**Schwellenwert-Tests:**
- ✅ Unter Schwellenwert (1.0 WE-Tag) → 0€
- ✅ Genau Schwellenwert (2.0 WE-Tage) → 450€
- ✅ Über Schwellenwert (3.0 WE-Tage) → 900€
**Gemischte Dienste:**
- ✅ Normale Tage + WE-Tage korrekt berechnet
- ✅ Halbe Dienste korrekt berechnet
- ✅ Feiertag + Vortag-Kombination
**Spezialfälle:**
- ✅ Keine Dienste → 0€
- ✅ 2x halbe Samstage zählen als 1 ganzer Tag
### 4. Storage (Datenverwaltung)
- ✅ Mitarbeiter hinzufügen
- ✅ Doppelte Mitarbeiter werden abgelehnt
- ✅ Mitarbeiter entfernen
- ✅ Dienste hinzufügen und abrufen
- ✅ Dienste aktualisieren (gleicher Tag)
- ✅ Mehrere Mitarbeiter verwalten
- ✅ Export und Import von Daten
### 5. Edge Cases
- ✅ Rundungsfehler bei Schwellenwert
- ✅ Performance bei vielen Diensten (30+ Tage)
- ✅ Schaltjahre (29. Februar)
## Test-Statistiken
Nach dem Durchlauf sehen Sie:
- **Gesamt**: Anzahl aller Tests
- **Bestanden**: Anzahl erfolgreicher Tests
- **Fehlgeschlagen**: Anzahl fehlgeschlagener Tests
## Testfälle im Detail
### Beispiel 1: Schwellenwert genau erreicht
```javascript
Dienste:
- 1× Samstag (1.0)
- 1× Sonntag (1.0)
Erwartung:
- Qualifizierende Tage: 2.0
- Schwellenwert: ✅ Erreicht
- Abzug: -1.0
- Bezahlt: 1.0 × 450€ = 450€
```
### Beispiel 2: Gemischte Dienste
```javascript
Dienste:
- 2× Montag (2.0 normale Tage)
- 2× Samstag (2.0 qualifizierende Tage)
Erwartung:
- Normale Tage: 2.0 × 250€ = 500€
- Qualifizierende Tage: (2.0 - 1.0) × 450€ = 450€
- Gesamt: 950€
```
### Beispiel 3: Halbe Dienste
```javascript
Dienste:
- 1× Montag halber Dienst (0.5)
- 1× Samstag halber Dienst (0.5)
- 1× Sonntag ganzer Dienst (1.0)
- 1× Freitag ganzer Dienst (1.0)
Erwartung:
- Normale Tage: 0.5 × 250€ = 125€
- Qualifizierende Tage: (2.5 - 1.0) × 450€ = 675€
- Gesamt: 800€
```
## Tests erweitern
Um einen neuen Test hinzuzufügen, bearbeiten Sie `test-suite.js`:
```javascript
runner.test('Testname', (t) => {
// Setup
const calculator = new BonusCalculator(new HolidayProvider());
const duties = [
{ date: new Date('2025-11-22T12:00:00'), share: 1.0 }
];
// Ausführung
const result = calculator.calculateMonthlyBonus(duties);
// Assertions
t.assertEqual(result.totalBonus, 0, 'Erwarteter Bonus');
t.assertTrue(result.thresholdReached, 'Schwelle erreicht');
});
```
### Verfügbare Assertions
- `assertEqual(actual, expected, message)` - Exakte Gleichheit
- `assertAlmostEqual(actual, expected, tolerance, message)` - Ungefähre Gleichheit (für Fließkommazahlen)
- `assertTrue(value, message)` - Wert sollte true sein
- `assertFalse(value, message)` - Wert sollte false sein
## Troubleshooting
### Tests schlagen fehl
1. Prüfen Sie die Fehlermeldung (wird rot angezeigt)
2. Überprüfen Sie die erwarteten vs. erhaltenen Werte
3. Testen Sie die Funktion manuell in der Haupt-App
### Performance-Probleme
- Die Test Suite sollte in < 1 Sekunde durchlaufen
- Bei Verzögerungen: Browser-Konsole prüfen (F12)
### LocalStorage-Konflikte
- Tests verwenden die gleiche LocalStorage-Instanz wie die Haupt-App
- Bei Problemen: LocalStorage im Browser löschen
- Oder: Tests in Inkognito-Modus ausführen
## Continuous Integration
Die Tests können auch automatisiert mit Headless-Browsern ausgeführt werden:
```bash
# Mit Playwright
npx playwright test
# Mit Puppeteer
node run-tests-headless.js
```
(Erfordert zusätzliche Setup-Schritte)
## Test-Abdeckung
Aktuelle Abdeckung:
- **Feiertage**: 100% (alle NRW-Feiertage getestet)
- **Tag-Klassifizierung**: 100% (alle Wochentage + Feiertage)
- **Bonusberechnung**: ~95% (Hauptszenarien + Edge Cases)
- **Storage**: ~90% (CRUD-Operationen)
- **UI**: 0% (keine UI-Tests, nur Logik)
## Bekannte Limitierungen
1. **Keine UI-Tests**: Nur Logik-Tests, keine Interaktions-Tests
2. **Browser-abhängig**: LocalStorage-Tests funktionieren nur im Browser
3. **Keine Netzwerk-Tests**: Kein Server-seitiger Code
4. **Zeitzone**: Tests gehen von deutscher Zeitzone aus
## Best Practices
1. **Tests vor Änderungen ausführen**: Sicherstellen, dass alles funktioniert
2. **Nach Änderungen erneut testen**: Regression verhindern
3. **Neue Features = Neue Tests**: Test-first development
4. **Tests dokumentieren**: Klare Namen und Kommentare
## Lizenz
MIT (wie Hauptprojekt)

565
webapp/test-suite.js Normal file
View file

@ -0,0 +1,565 @@
/**
* Test Suite for Dienstplan Bonusrechner
*/
class TestRunner {
constructor() {
this.tests = [];
this.passed = 0;
this.failed = 0;
}
/**
* Add a test case
*/
test(name, testFn) {
this.tests.push({ name, testFn });
}
/**
* Assert equality
*/
assertEqual(actual, expected, message = '') {
if (actual !== expected) {
throw new Error(`${message}\nErwartet: ${expected}\nErhalten: ${actual}`);
}
}
/**
* Assert approximate equality (for floating point)
*/
assertAlmostEqual(actual, expected, tolerance = 0.01, message = '') {
if (Math.abs(actual - expected) > tolerance) {
throw new Error(`${message}\nErwartet: ${expected}${tolerance})\nErhalten: ${actual}`);
}
}
/**
* Assert true
*/
assertTrue(value, message = '') {
if (!value) {
throw new Error(`${message}\nErwartet: true\nErhalten: ${value}`);
}
}
/**
* Assert false
*/
assertFalse(value, message = '') {
if (value) {
throw new Error(`${message}\nErwartet: false\nErhalten: ${value}`);
}
}
/**
* Run all tests
*/
async runAll() {
this.passed = 0;
this.failed = 0;
const results = [];
for (const test of this.tests) {
try {
await test.testFn(this);
this.passed++;
results.push({
name: test.name,
passed: true,
error: null
});
} catch (error) {
this.failed++;
results.push({
name: test.name,
passed: false,
error: error.message
});
}
}
return results;
}
/**
* Get summary
*/
getSummary() {
return {
total: this.tests.length,
passed: this.passed,
failed: this.failed
};
}
}
// Create test runner instance
const runner = new TestRunner();
// ============================================================================
// Holiday Provider Tests
// ============================================================================
runner.test('HolidayProvider: Neujahr 2025 wird erkannt', (t) => {
const holidays = new HolidayProvider();
const date = new Date('2025-01-01T12:00:00');
t.assertTrue(holidays.isHoliday(date), 'Neujahr sollte als Feiertag erkannt werden');
});
runner.test('HolidayProvider: Normaler Tag wird nicht als Feiertag erkannt', (t) => {
const holidays = new HolidayProvider();
const date = new Date('2025-01-15T12:00:00'); // Mittwoch
t.assertFalse(holidays.isHoliday(date), 'Normaler Tag sollte nicht als Feiertag erkannt werden');
});
runner.test('HolidayProvider: Tag vor Feiertag wird erkannt', (t) => {
const holidays = new HolidayProvider();
const date = new Date('2024-12-31T12:00:00'); // Tag vor Neujahr
t.assertTrue(holidays.isDayBeforeHoliday(date), 'Tag vor Feiertag sollte erkannt werden');
});
runner.test('HolidayProvider: Fronleichnam 2025 korrekt', (t) => {
const holidays = new HolidayProvider();
const date = new Date('2025-06-19T12:00:00');
t.assertTrue(holidays.isHoliday(date), 'Fronleichnam sollte als Feiertag erkannt werden');
t.assertEqual(holidays.getHolidayName(date), 'Fronleichnam', 'Feiertagsname sollte korrekt sein');
});
// ============================================================================
// Calculator Tests - Day Classification
// ============================================================================
runner.test('Calculator: Freitag ist qualifizierender Tag', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const friday = new Date('2025-11-21T12:00:00'); // Freitag
t.assertTrue(calculator.isQualifyingDay(friday), 'Freitag sollte qualifizierend sein');
});
runner.test('Calculator: Samstag ist qualifizierender Tag', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const saturday = new Date('2025-11-22T12:00:00'); // Samstag
t.assertTrue(calculator.isQualifyingDay(saturday), 'Samstag sollte qualifizierend sein');
});
runner.test('Calculator: Sonntag ist qualifizierender Tag', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const sunday = new Date('2025-11-23T12:00:00'); // Sonntag
t.assertTrue(calculator.isQualifyingDay(sunday), 'Sonntag sollte qualifizierend sein');
});
runner.test('Calculator: Montag ist kein qualifizierender Tag', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const monday = new Date('2025-11-24T12:00:00'); // Montag (kein Feiertag)
t.assertFalse(calculator.isQualifyingDay(monday), 'Normaler Montag sollte nicht qualifizierend sein');
});
runner.test('Calculator: Feiertag ist qualifizierender Tag', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const holiday = new Date('2025-05-01T12:00:00'); // Tag der Arbeit
t.assertTrue(calculator.isQualifyingDay(holiday), 'Feiertag sollte qualifizierend sein');
});
runner.test('Calculator: Tag vor Feiertag ist qualifizierender Tag', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const dayBefore = new Date('2025-04-30T12:00:00'); // Tag vor 1. Mai
t.assertTrue(calculator.isQualifyingDay(dayBefore), 'Tag vor Feiertag sollte qualifizierend sein');
});
// ============================================================================
// Calculator Tests - Bonus Calculation
// ============================================================================
runner.test('Berechnung: Unter Schwellenwert (1.0 WE-Tag) = 0€', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const duties = [
{ date: new Date('2025-11-22T12:00:00'), share: 1.0 } // 1x Samstag
];
const result = calculator.calculateMonthlyBonus(duties);
t.assertEqual(result.qualifyingDays, 1.0, 'Sollte 1.0 qualifizierende Tage haben');
t.assertFalse(result.thresholdReached, 'Schwellenwert sollte nicht erreicht sein');
t.assertEqual(result.totalBonus, 0, 'Bonus sollte 0€ sein');
});
runner.test('Berechnung: Genau 2.0 WE-Tage = 450€', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const duties = [
{ date: new Date('2025-11-22T12:00:00'), share: 1.0 }, // Samstag
{ date: new Date('2025-11-23T12:00:00'), share: 1.0 } // Sonntag
];
const result = calculator.calculateMonthlyBonus(duties);
t.assertEqual(result.qualifyingDays, 2.0, 'Sollte 2.0 qualifizierende Tage haben');
t.assertTrue(result.thresholdReached, 'Schwellenwert sollte erreicht sein');
t.assertEqual(result.qualifyingDaysDeducted, 1.0, 'Sollte 1.0 Tag abziehen');
t.assertEqual(result.qualifyingDaysPaid, 1.0, 'Sollte 1.0 Tag bezahlen');
t.assertEqual(result.totalBonus, 450, 'Bonus sollte 450€ sein');
});
runner.test('Berechnung: 2x halbe WE-Dienste = 0€ (genau Schwelle, aber nach Abzug nichts)', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const duties = [
{ date: new Date('2025-11-22T12:00:00'), share: 0.5 }, // Halber Samstag
{ date: new Date('2025-11-22T12:00:00'), share: 0.5 }, // Halber Samstag
{ date: new Date('2025-11-23T12:00:00'), share: 0.5 }, // Halber Sonntag
{ date: new Date('2025-11-23T12:00:00'), share: 0.5 } // Halber Sonntag
];
const result = calculator.calculateMonthlyBonus(duties);
t.assertEqual(result.qualifyingDays, 2.0, 'Sollte 2.0 qualifizierende Tage haben (4×0.5)');
t.assertTrue(result.thresholdReached, 'Schwellenwert sollte erreicht sein');
t.assertEqual(result.qualifyingDaysPaid, 1.0, 'Sollte 1.0 Tag bezahlen nach Abzug');
t.assertEqual(result.totalBonus, 450, 'Bonus sollte 450€ sein');
});
runner.test('Berechnung: 3 WE-Tage = 900€', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const duties = [
{ date: new Date('2025-11-21T12:00:00'), share: 1.0 }, // Freitag
{ date: new Date('2025-11-22T12:00:00'), share: 1.0 }, // Samstag
{ date: new Date('2025-11-23T12:00:00'), share: 1.0 } // Sonntag
];
const result = calculator.calculateMonthlyBonus(duties);
t.assertEqual(result.qualifyingDays, 3.0, 'Sollte 3.0 qualifizierende Tage haben');
t.assertEqual(result.qualifyingDaysPaid, 2.0, 'Sollte 2.0 Tage bezahlen (3-1)');
t.assertEqual(result.totalBonus, 900, 'Bonus sollte 900€ sein (2×450€)');
});
runner.test('Berechnung: Normale Tage + WE-Tage gemischt', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const duties = [
{ date: new Date('2025-11-24T12:00:00'), share: 1.0 }, // Montag (normal)
{ date: new Date('2025-11-25T12:00:00'), share: 1.0 }, // Dienstag (normal)
{ date: new Date('2025-11-22T12:00:00'), share: 1.0 }, // Samstag (qualifizierend)
{ date: new Date('2025-11-23T12:00:00'), share: 1.0 } // Sonntag (qualifizierend)
];
const result = calculator.calculateMonthlyBonus(duties);
t.assertEqual(result.normalDays, 2.0, 'Sollte 2.0 normale Tage haben');
t.assertEqual(result.qualifyingDays, 2.0, 'Sollte 2.0 qualifizierende Tage haben');
t.assertEqual(result.normalDaysPaid, 2.0, 'Sollte 2.0 normale Tage bezahlen');
t.assertEqual(result.qualifyingDaysPaid, 1.0, 'Sollte 1.0 qualifizierenden Tag bezahlen');
t.assertEqual(result.bonusNormalDays, 500, 'Normale Tage: 2×250€ = 500€');
t.assertEqual(result.bonusQualifyingDays, 450, 'WE-Tage: 1×450€ = 450€');
t.assertEqual(result.totalBonus, 950, 'Gesamt: 950€');
});
runner.test('Berechnung: Halbe Dienste korrekt berechnet', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const duties = [
{ date: new Date('2025-11-24T12:00:00'), share: 0.5 }, // Halber Montag
{ date: new Date('2025-11-22T12:00:00'), share: 0.5 }, // Halber Samstag
{ date: new Date('2025-11-23T12:00:00'), share: 1.0 }, // Ganzer Sonntag
{ date: new Date('2025-11-21T12:00:00'), share: 1.0 } // Ganzer Freitag
];
const result = calculator.calculateMonthlyBonus(duties);
t.assertEqual(result.normalDays, 0.5, 'Sollte 0.5 normale Tage haben');
t.assertEqual(result.qualifyingDays, 2.5, 'Sollte 2.5 qualifizierende Tage haben');
t.assertEqual(result.qualifyingDaysPaid, 1.5, 'Sollte 1.5 qualifizierende Tage bezahlen');
t.assertEqual(result.bonusNormalDays, 125, 'Normale Tage: 0.5×250€ = 125€');
t.assertEqual(result.bonusQualifyingDays, 675, 'WE-Tage: 1.5×450€ = 675€');
t.assertEqual(result.totalBonus, 800, 'Gesamt: 800€');
});
runner.test('Berechnung: Feiertag + Vortag', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const duties = [
{ date: new Date('2025-04-30T12:00:00'), share: 1.0 }, // Mittwoch vor 1. Mai (qualifizierend)
{ date: new Date('2025-05-01T12:00:00'), share: 1.0 } // 1. Mai (Feiertag, qualifizierend)
];
const result = calculator.calculateMonthlyBonus(duties);
t.assertEqual(result.qualifyingDays, 2.0, 'Sollte 2.0 qualifizierende Tage haben');
t.assertTrue(result.thresholdReached, 'Schwellenwert sollte erreicht sein');
t.assertEqual(result.totalBonus, 450, 'Bonus sollte 450€ sein');
});
runner.test('Berechnung: Keine Dienste = 0€', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const result = calculator.calculateMonthlyBonus([]);
t.assertEqual(result.totalDuties, 0, 'Sollte 0 Dienste haben');
t.assertEqual(result.totalBonus, 0, 'Bonus sollte 0€ sein');
});
// ============================================================================
// Storage Tests
// ============================================================================
runner.test('Storage: Mitarbeiter hinzufügen', (t) => {
const storage = new DataStorage();
storage.clearAll();
const success = storage.addEmployee('Max Mustermann');
t.assertTrue(success, 'Mitarbeiter sollte hinzugefügt werden');
const employees = storage.getEmployees();
t.assertEqual(employees.length, 1, 'Sollte 1 Mitarbeiter haben');
t.assertTrue(employees.includes('Max Mustermann'), 'Mitarbeiter sollte in Liste sein');
});
runner.test('Storage: Doppelter Mitarbeiter wird abgelehnt', (t) => {
const storage = new DataStorage();
storage.clearAll();
storage.addEmployee('Max Mustermann');
const success = storage.addEmployee('Max Mustermann');
t.assertFalse(success, 'Doppelter Mitarbeiter sollte abgelehnt werden');
const employees = storage.getEmployees();
t.assertEqual(employees.length, 1, 'Sollte nur 1 Mitarbeiter haben');
});
runner.test('Storage: Mitarbeiter entfernen', (t) => {
const storage = new DataStorage();
storage.clearAll();
storage.addEmployee('Max Mustermann');
storage.removeEmployee('Max Mustermann');
const employees = storage.getEmployees();
t.assertEqual(employees.length, 0, 'Sollte 0 Mitarbeiter haben');
});
runner.test('Storage: Dienst hinzufügen und abrufen', (t) => {
const storage = new DataStorage();
storage.clearAll();
storage.addEmployee('Max Mustermann');
const date = new Date('2025-11-22T12:00:00');
storage.addDuty('Max Mustermann', 2025, 11, date, 1.0);
const duties = storage.getDutiesForMonth('Max Mustermann', 2025, 11);
t.assertEqual(duties.length, 1, 'Sollte 1 Dienst haben');
t.assertEqual(duties[0].share, 1.0, 'Dienst sollte share 1.0 haben');
});
runner.test('Storage: Dienst aktualisieren (gleicher Tag)', (t) => {
const storage = new DataStorage();
storage.clearAll();
storage.addEmployee('Max Mustermann');
const date = new Date('2025-11-22T12:00:00');
storage.addDuty('Max Mustermann', 2025, 11, date, 1.0);
storage.addDuty('Max Mustermann', 2025, 11, date, 0.5); // Update
const duties = storage.getDutiesForMonth('Max Mustermann', 2025, 11);
t.assertEqual(duties.length, 1, 'Sollte nur 1 Dienst haben (aktualisiert)');
t.assertEqual(duties[0].share, 0.5, 'Share sollte aktualisiert sein');
});
runner.test('Storage: Mehrere Mitarbeiter', (t) => {
const storage = new DataStorage();
storage.clearAll();
storage.addEmployee('Max Mustermann');
storage.addEmployee('Anna Schmidt');
storage.addEmployee('Peter Müller');
const employees = storage.getEmployees();
t.assertEqual(employees.length, 3, 'Sollte 3 Mitarbeiter haben');
t.assertTrue(employees.includes('Anna Schmidt'), 'Anna Schmidt sollte vorhanden sein');
});
runner.test('Storage: Export und Import', (t) => {
const storage1 = new DataStorage();
storage1.clearAll();
storage1.addEmployee('Max Mustermann');
const date = new Date('2025-11-22T12:00:00');
storage1.addDuty('Max Mustermann', 2025, 11, date, 1.0);
const exported = storage1.exportData();
const storage2 = new DataStorage();
storage2.clearAll();
const success = storage2.importData(exported);
t.assertTrue(success, 'Import sollte erfolgreich sein');
const employees = storage2.getEmployees();
t.assertEqual(employees.length, 1, 'Sollte 1 Mitarbeiter haben');
const duties = storage2.getDutiesForMonth('Max Mustermann', 2025, 11);
t.assertEqual(duties.length, 1, 'Sollte 1 Dienst haben');
});
// ============================================================================
// Edge Cases & Regression Tests
// ============================================================================
runner.test('Edge Case: Exakt Schwellenwert mit Rundungsfehler (1.9999)', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
// Simuliere Rundungsfehler
const duties = [
{ date: new Date('2025-11-22T12:00:00'), share: 0.66666 },
{ date: new Date('2025-11-23T12:00:00'), share: 0.66666 },
{ date: new Date('2025-11-21T12:00:00'), share: 0.66666 }
];
const result = calculator.calculateMonthlyBonus(duties);
// 0.66666 × 3 ≈ 1.99998, sollte als >= 2.0 gelten
t.assertTrue(result.thresholdReached || result.qualifyingDays < 2.0,
'Sollte Rundung korrekt handhaben');
});
runner.test('Edge Case: Sehr viele Dienste (Performance)', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const duties = [];
for (let i = 1; i <= 30; i++) {
duties.push({
date: new Date(`2025-11-${String(i).padStart(2, '0')}T12:00:00`),
share: i % 2 === 0 ? 1.0 : 0.5
});
}
const start = Date.now();
const result = calculator.calculateMonthlyBonus(duties);
const duration = Date.now() - start;
t.assertTrue(duration < 100, `Berechnung sollte schnell sein (${duration}ms)`);
t.assertTrue(result.totalBonus > 0, 'Sollte Bonus berechnen');
});
runner.test('Edge Case: Dienst am 29. Februar (Schaltjahr)', (t) => {
const holidays = new HolidayProvider();
const calculator = new BonusCalculator(holidays);
const duties = [
{ date: new Date('2028-02-29T12:00:00'), share: 1.0 } // Dienstag (nicht qualifizierend)
];
// Sollte nicht crashen
const result = calculator.calculateMonthlyBonus(duties);
t.assertEqual(result.normalDays, 1.0, 'Sollte normalen Tag erkennen');
});
// ============================================================================
// Display Functions
// ============================================================================
async function runAllTests() {
const resultsContainer = document.getElementById('test-results');
const summaryDiv = document.getElementById('summary');
const runButton = document.getElementById('run-tests');
// Clear previous results
resultsContainer.innerHTML = '<p>Tests laufen...</p>';
runButton.disabled = true;
// Run tests
const results = await runner.runAll();
const summary = runner.getSummary();
// Update summary
document.getElementById('total-tests').textContent = summary.total;
document.getElementById('passed-tests').textContent = summary.passed;
document.getElementById('failed-tests').textContent = summary.failed;
summaryDiv.style.display = 'flex';
// Display results
resultsContainer.innerHTML = '';
// Group by category
const categories = {
'Holiday Provider': [],
'Calculator - Tag-Klassifizierung': [],
'Calculator - Bonusberechnung': [],
'Storage': [],
'Edge Cases': []
};
results.forEach(result => {
if (result.name.includes('HolidayProvider')) {
categories['Holiday Provider'].push(result);
} else if (result.name.includes('qualifizierender Tag') || result.name.includes('Feiertag ist')) {
categories['Calculator - Tag-Klassifizierung'].push(result);
} else if (result.name.includes('Berechnung:')) {
categories['Calculator - Bonusberechnung'].push(result);
} else if (result.name.includes('Storage:')) {
categories['Storage'].push(result);
} else if (result.name.includes('Edge Case:')) {
categories['Edge Cases'].push(result);
}
});
// Render categories
for (const [category, tests] of Object.entries(categories)) {
if (tests.length === 0) continue;
const suiteDiv = document.createElement('div');
suiteDiv.className = 'test-suite';
const title = document.createElement('h2');
title.textContent = `${category} (${tests.filter(t => t.passed).length}/${tests.length})`;
suiteDiv.appendChild(title);
tests.forEach(result => {
const testDiv = document.createElement('div');
testDiv.className = `test-case ${result.passed ? 'pass' : 'fail'}`;
const nameDiv = document.createElement('div');
nameDiv.className = 'test-name';
nameDiv.textContent = `${result.passed ? '✅' : '❌'} ${result.name}`;
testDiv.appendChild(nameDiv);
if (!result.passed && result.error) {
const errorDiv = document.createElement('div');
errorDiv.className = 'error-details';
errorDiv.textContent = result.error;
testDiv.appendChild(errorDiv);
}
suiteDiv.appendChild(testDiv);
});
resultsContainer.appendChild(suiteDiv);
}
runButton.disabled = false;
// Scroll to summary
summaryDiv.scrollIntoView({ behavior: 'smooth' });
}
// Auto-run on load (optional)
// window.addEventListener('load', runAllTests);

147
webapp/test.html Normal file
View file

@ -0,0 +1,147 @@
<!DOCTYPE html>
<html lang="de">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Dienstplan Test Suite</title>
<style>
body {
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
max-width: 1200px;
margin: 0 auto;
padding: 20px;
background: #f5f5f5;
}
h1 {
color: #667eea;
}
.test-suite {
background: white;
border-radius: 8px;
padding: 20px;
margin-bottom: 20px;
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
}
.test-case {
padding: 10px;
margin: 5px 0;
border-left: 4px solid #ccc;
background: #f9f9f9;
}
.test-case.pass {
border-left-color: #28a745;
background: #f0f9f4;
}
.test-case.fail {
border-left-color: #dc3545;
background: #fff0f0;
}
.test-name {
font-weight: 600;
margin-bottom: 5px;
}
.test-details {
font-size: 0.9em;
color: #666;
font-family: monospace;
}
.summary {
display: flex;
gap: 20px;
margin-bottom: 20px;
}
.summary-item {
flex: 1;
padding: 20px;
border-radius: 8px;
text-align: center;
color: white;
font-size: 1.2em;
}
.summary-item.total {
background: #667eea;
}
.summary-item.passed {
background: #28a745;
}
.summary-item.failed {
background: #dc3545;
}
.summary-item .label {
font-size: 0.8em;
opacity: 0.9;
}
.summary-item .value {
font-size: 2em;
font-weight: bold;
}
button {
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
color: white;
border: none;
padding: 12px 24px;
border-radius: 6px;
font-size: 1em;
cursor: pointer;
margin-bottom: 20px;
}
button:hover {
opacity: 0.9;
}
.error-details {
background: #fff0f0;
border: 1px solid #dc3545;
padding: 10px;
margin-top: 10px;
border-radius: 4px;
font-family: monospace;
font-size: 0.85em;
color: #721c24;
}
</style>
</head>
<body>
<h1>🧪 Dienstplan Bonusrechner - Test Suite</h1>
<button id="run-tests" onclick="runAllTests()">Alle Tests ausführen</button>
<div class="summary" id="summary" style="display: none;">
<div class="summary-item total">
<div class="label">Gesamt</div>
<div class="value" id="total-tests">0</div>
</div>
<div class="summary-item passed">
<div class="label">Bestanden</div>
<div class="value" id="passed-tests">0</div>
</div>
<div class="summary-item failed">
<div class="label">Fehlgeschlagen</div>
<div class="value" id="failed-tests">0</div>
</div>
</div>
<div id="test-results"></div>
<script src="holidays.js"></script>
<script src="calculator.js"></script>
<script src="storage.js"></script>
<script src="test-suite.js"></script>
</body>
</html>