Some checks failed
CI / test (push) Has been cancelled
핵심 기능: - 단축근무·표준·반일 등 다양한 근무 패턴 (5개 프리셋 + 사용자 정의) - Windows 이벤트 뷰어 자동 출퇴근 감지 - 30분 단위 연장근무 적립/사용 시스템 - 1.0/0.5/0.25일 연차·반차·반반차 - 자동 점심·저녁·외출·자동 백업·화면 잠금 자동 외출 - 한국 공휴일 자동 등록 (음력 포함, holidays 패키지) - matplotlib 차트 기반 주간/월간/패턴 통계 - 미니 위젯 + 시스템 트레이 통합 - 한국어/English i18n - 자가 업데이트 (updater.exe + Gitea Releases) 아키텍처: - core/ (db, time_calculator, notifier, i18n, version, settings_keys) - ui/ (main_window + 9 dialogs + 3 controllers) - utils/ (backup, lock_detector, debug_log, updater_client, time_format) - tests/ (66 pytest 단위) + 통합/i18n GUI 검증 CI/CD: - .gitea/workflows/ci.yml: push 시 pytest + 통합 테스트 - .gitea/workflows/release.yml: v* 태그 push 시 두 .exe 자동 빌드 + Releases 첨부 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
111 lines
3.8 KiB
Python
111 lines
3.8 KiB
Python
"""
|
|
Database 단위 테스트 — 마이그레이션, 동기화, 헬퍼.
|
|
"""
|
|
import os
|
|
import sys
|
|
import tempfile
|
|
from datetime import date
|
|
|
|
import pytest
|
|
|
|
sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
|
|
|
|
from core.database import Database
|
|
|
|
|
|
@pytest.fixture
|
|
def fresh_db(tmp_path):
|
|
"""매 테스트마다 빈 DB."""
|
|
return Database(str(tmp_path / "test.db"))
|
|
|
|
|
|
class TestSettingsHelpers:
|
|
def test_get_setting_int_valid(self, fresh_db):
|
|
fresh_db.set_setting('foo', '42')
|
|
assert fresh_db.get_setting_int('foo') == 42
|
|
|
|
def test_get_setting_int_invalid_returns_default(self, fresh_db):
|
|
fresh_db.set_setting('foo', 'abc')
|
|
assert fresh_db.get_setting_int('foo', 99) == 99
|
|
|
|
def test_get_setting_int_missing_returns_default(self, fresh_db):
|
|
assert fresh_db.get_setting_int('missing', 7) == 7
|
|
|
|
def test_get_setting_bool_truthy(self, fresh_db):
|
|
fresh_db.set_setting('flag', 'true')
|
|
assert fresh_db.get_setting_bool('flag') is True
|
|
|
|
def test_get_setting_bool_falsy(self, fresh_db):
|
|
fresh_db.set_setting('flag', 'no')
|
|
assert fresh_db.get_setting_bool('flag') is False
|
|
|
|
def test_get_setting_float(self, fresh_db):
|
|
fresh_db.set_setting('rate', '3.14')
|
|
assert fresh_db.get_setting_float('rate') == 3.14
|
|
|
|
|
|
class TestSettingsAutoSync:
|
|
def test_work_minutes_to_work_hours_floor(self, fresh_db):
|
|
"""work_minutes 저장 시 work_hours는 floor 동기화 (450 → 7)"""
|
|
fresh_db.save_settings({'work_minutes': 450})
|
|
assert fresh_db.get_setting('work_hours') == '7'
|
|
|
|
def test_work_hours_to_work_minutes(self, fresh_db):
|
|
fresh_db.save_settings({'work_hours': 8})
|
|
assert fresh_db.get_setting('work_minutes') == '480'
|
|
|
|
def test_annual_leave_bidirectional(self, fresh_db):
|
|
fresh_db.save_settings({'annual_leave_days': 12})
|
|
assert fresh_db.get_setting('annual_leave_total') == '12'
|
|
|
|
|
|
class TestWorkMinutes:
|
|
def test_get_work_minutes_default(self, fresh_db):
|
|
assert fresh_db.get_work_minutes() == 480
|
|
|
|
def test_get_work_minutes_after_save(self, fresh_db):
|
|
fresh_db.save_settings({'work_minutes': 450})
|
|
assert fresh_db.get_work_minutes() == 450
|
|
|
|
|
|
class TestLeaveCalculation:
|
|
def test_leave_minutes_for_short_worker(self, fresh_db):
|
|
"""단축근무자(7h30m) 1일 연차 = 450분"""
|
|
fresh_db.save_settings({'work_minutes': 450})
|
|
today = date.today().isoformat()
|
|
fresh_db.add_leave_record(today, 'annual', 1.0)
|
|
assert fresh_db.get_today_leave_minutes() == 450
|
|
|
|
def test_half_day_leave(self, fresh_db):
|
|
today = date.today().isoformat()
|
|
fresh_db.add_leave_record(today, 'half', 0.5)
|
|
assert fresh_db.get_today_leave_minutes() == 240 # 8h * 0.5
|
|
|
|
|
|
class TestMigrationIdempotency:
|
|
def test_annual_leave_keys_migrated_sentinel(self, fresh_db):
|
|
assert fresh_db.get_setting('annual_leave_keys_migrated') == 'true'
|
|
|
|
def test_re_init_does_not_break(self, tmp_path):
|
|
path = str(tmp_path / "test.db")
|
|
db1 = Database(path)
|
|
db1.save_settings({'work_minutes': 450})
|
|
# 두 번째 init
|
|
db2 = Database(path)
|
|
assert db2.get_work_minutes() == 450
|
|
|
|
|
|
class TestConsecutiveOvertimeDays:
|
|
def test_no_records(self, fresh_db):
|
|
assert fresh_db.get_consecutive_overtime_days() == 0
|
|
|
|
def test_three_consecutive(self, fresh_db):
|
|
from datetime import date, timedelta
|
|
today = date.today()
|
|
for i in range(3):
|
|
d = (today - timedelta(days=i)).isoformat()
|
|
fresh_db.add_work_record(d, '09:00:00')
|
|
fresh_db.update_clock_out(d, '20:00:00', total_hours=11.0,
|
|
overtime_minutes=120, overtime_earned=120)
|
|
assert fresh_db.get_consecutive_overtime_days() == 3
|