refactor: 프로젝트 구조 개선 및 테스트 시스템 강화
주요 변경사항: - CLAUDE.md: 프로젝트 규칙 v2.0으로 업데이트, 아키텍처 명확화 - 불필요한 문서 제거: NEXT_TASKS.md, TEST_PROGRESS.md, test_results 파일들 - 테스트 시스템 개선: 실제 API 테스트 스위트 추가 (15개 새 테스트 파일) - License 관리: DTO 모델 개선, API 응답 처리 최적화 - 에러 처리: Interceptor 로직 강화, 상세 로깅 추가 - Company/User/Warehouse 테스트: 자동화 테스트 안정성 향상 - Phone Utils: 전화번호 포맷팅 로직 개선 - Overview Controller: 대시보드 데이터 로딩 최적화 - Analysis Options: Flutter 린트 규칙 추가 테스트 개선: - company_real_api_test.dart: 실제 API 회사 관리 테스트 - equipment_in/out_real_api_test.dart: 장비 입출고 API 테스트 - license_real_api_test.dart: 라이선스 관리 API 테스트 - user_real_api_test.dart: 사용자 관리 API 테스트 - warehouse_location_real_api_test.dart: 창고 위치 API 테스트 - filter_sort_test.dart: 필터링/정렬 기능 테스트 - pagination_test.dart: 페이지네이션 테스트 - interactive_search_test.dart: 검색 기능 테스트 - overview_dashboard_test.dart: 대시보드 통합 테스트 코드 품질: - 모든 서비스에 에러 처리 강화 - DTO 모델 null safety 개선 - 테스트 커버리지 확대 - 불필요한 로그 파일 제거로 리포지토리 정리 Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
@@ -52,12 +52,17 @@ class OverviewController extends ChangeNotifier {
|
||||
|
||||
// 데이터 로드
|
||||
Future<void> loadData() async {
|
||||
await Future.wait([
|
||||
_loadOverviewStats(),
|
||||
_loadRecentActivities(),
|
||||
_loadEquipmentStatus(),
|
||||
_loadExpiringLicenses(),
|
||||
]);
|
||||
try {
|
||||
await Future.wait([
|
||||
_loadOverviewStats(),
|
||||
_loadRecentActivities(),
|
||||
_loadEquipmentStatus(),
|
||||
_loadExpiringLicenses(),
|
||||
], eagerError: false); // 하나의 작업이 실패해도 다른 작업 계속 진행
|
||||
} catch (e) {
|
||||
DebugLogger.logError('대시보드 데이터 로드 중 오류', error: e);
|
||||
// 개별 에러는 각 메서드에서 처리하므로 여기서는 로그만 남김
|
||||
}
|
||||
}
|
||||
|
||||
// 대시보드 데이터 로드 (loadData의 alias)
|
||||
@@ -71,16 +76,55 @@ class OverviewController extends ChangeNotifier {
|
||||
_statsError = null;
|
||||
notifyListeners();
|
||||
|
||||
final result = await _dashboardService.getOverviewStats();
|
||||
|
||||
result.fold(
|
||||
(failure) {
|
||||
_statsError = failure.message;
|
||||
},
|
||||
(stats) {
|
||||
_overviewStats = stats;
|
||||
},
|
||||
);
|
||||
try {
|
||||
final result = await _dashboardService.getOverviewStats();
|
||||
|
||||
result.fold(
|
||||
(failure) {
|
||||
_statsError = failure.message;
|
||||
DebugLogger.logError('Overview 통계 로드 실패', error: failure.message);
|
||||
// 실패 시 기본값 설정
|
||||
_overviewStats = OverviewStats(
|
||||
totalCompanies: 0,
|
||||
activeCompanies: 0,
|
||||
totalUsers: 0,
|
||||
activeUsers: 0,
|
||||
totalEquipment: 0,
|
||||
availableEquipment: 0,
|
||||
inUseEquipment: 0,
|
||||
maintenanceEquipment: 0,
|
||||
totalLicenses: 0,
|
||||
activeLicenses: 0,
|
||||
expiringLicensesCount: 0,
|
||||
expiredLicensesCount: 0,
|
||||
totalWarehouseLocations: 0,
|
||||
activeWarehouseLocations: 0,
|
||||
);
|
||||
},
|
||||
(stats) {
|
||||
_overviewStats = stats;
|
||||
},
|
||||
);
|
||||
} catch (e) {
|
||||
_statsError = '통계 데이터를 불러올 수 없습니다';
|
||||
_overviewStats = OverviewStats(
|
||||
totalCompanies: 0,
|
||||
activeCompanies: 0,
|
||||
totalUsers: 0,
|
||||
activeUsers: 0,
|
||||
totalEquipment: 0,
|
||||
availableEquipment: 0,
|
||||
inUseEquipment: 0,
|
||||
maintenanceEquipment: 0,
|
||||
totalLicenses: 0,
|
||||
activeLicenses: 0,
|
||||
expiringLicensesCount: 0,
|
||||
expiredLicensesCount: 0,
|
||||
totalWarehouseLocations: 0,
|
||||
activeWarehouseLocations: 0,
|
||||
);
|
||||
DebugLogger.logError('Overview 통계 로드 예외', error: e);
|
||||
}
|
||||
|
||||
_isLoadingStats = false;
|
||||
notifyListeners();
|
||||
@@ -91,16 +135,24 @@ class OverviewController extends ChangeNotifier {
|
||||
_activitiesError = null;
|
||||
notifyListeners();
|
||||
|
||||
final result = await _dashboardService.getRecentActivities();
|
||||
|
||||
result.fold(
|
||||
(failure) {
|
||||
_activitiesError = failure.message;
|
||||
},
|
||||
(activities) {
|
||||
_recentActivities = activities;
|
||||
},
|
||||
);
|
||||
try {
|
||||
final result = await _dashboardService.getRecentActivities();
|
||||
|
||||
result.fold(
|
||||
(failure) {
|
||||
_activitiesError = failure.message;
|
||||
_recentActivities = []; // 실패 시 빈 리스트
|
||||
DebugLogger.logError('최근 활동 로드 실패', error: failure.message);
|
||||
},
|
||||
(activities) {
|
||||
_recentActivities = activities ?? [];
|
||||
},
|
||||
);
|
||||
} catch (e) {
|
||||
_activitiesError = '최근 활동을 불러올 수 없습니다';
|
||||
_recentActivities = [];
|
||||
DebugLogger.logError('최근 활동 로드 예외', error: e);
|
||||
}
|
||||
|
||||
_isLoadingActivities = false;
|
||||
notifyListeners();
|
||||
@@ -113,23 +165,41 @@ class OverviewController extends ChangeNotifier {
|
||||
|
||||
DebugLogger.log('장비 상태 분포 로드 시작', tag: 'DASHBOARD');
|
||||
|
||||
final result = await _dashboardService.getEquipmentStatusDistribution();
|
||||
|
||||
result.fold(
|
||||
(failure) {
|
||||
_equipmentStatusError = failure.message;
|
||||
DebugLogger.logError('장비 상태 분포 로드 실패', error: failure.message);
|
||||
},
|
||||
(status) {
|
||||
_equipmentStatus = status;
|
||||
DebugLogger.log('장비 상태 분포 로드 성공', tag: 'DASHBOARD', data: {
|
||||
'available': status.available,
|
||||
'inUse': status.inUse,
|
||||
'maintenance': status.maintenance,
|
||||
'disposed': status.disposed,
|
||||
});
|
||||
},
|
||||
);
|
||||
try {
|
||||
final result = await _dashboardService.getEquipmentStatusDistribution();
|
||||
|
||||
result.fold(
|
||||
(failure) {
|
||||
_equipmentStatusError = failure.message;
|
||||
DebugLogger.logError('장비 상태 분포 로드 실패', error: failure.message);
|
||||
// 실패 시 기본값 설정
|
||||
_equipmentStatus = EquipmentStatusDistribution(
|
||||
available: 0,
|
||||
inUse: 0,
|
||||
maintenance: 0,
|
||||
disposed: 0,
|
||||
);
|
||||
},
|
||||
(status) {
|
||||
_equipmentStatus = status;
|
||||
DebugLogger.log('장비 상태 분포 로드 성공', tag: 'DASHBOARD', data: {
|
||||
'available': status.available,
|
||||
'inUse': status.inUse,
|
||||
'maintenance': status.maintenance,
|
||||
'disposed': status.disposed,
|
||||
});
|
||||
},
|
||||
);
|
||||
} catch (e) {
|
||||
_equipmentStatusError = '장비 상태를 불러올 수 없습니다';
|
||||
_equipmentStatus = EquipmentStatusDistribution(
|
||||
available: 0,
|
||||
inUse: 0,
|
||||
maintenance: 0,
|
||||
disposed: 0,
|
||||
);
|
||||
DebugLogger.logError('장비 상태 로드 예외', error: e);
|
||||
}
|
||||
|
||||
_isLoadingEquipmentStatus = false;
|
||||
notifyListeners();
|
||||
@@ -140,16 +210,24 @@ class OverviewController extends ChangeNotifier {
|
||||
_licensesError = null;
|
||||
notifyListeners();
|
||||
|
||||
final result = await _dashboardService.getExpiringLicenses(days: 30);
|
||||
|
||||
result.fold(
|
||||
(failure) {
|
||||
_licensesError = failure.message;
|
||||
},
|
||||
(licenses) {
|
||||
_expiringLicenses = licenses;
|
||||
},
|
||||
);
|
||||
try {
|
||||
final result = await _dashboardService.getExpiringLicenses(days: 30);
|
||||
|
||||
result.fold(
|
||||
(failure) {
|
||||
_licensesError = failure.message;
|
||||
_expiringLicenses = []; // 실패 시 빈 리스트
|
||||
DebugLogger.logError('만료 라이선스 로드 실패', error: failure.message);
|
||||
},
|
||||
(licenses) {
|
||||
_expiringLicenses = licenses ?? [];
|
||||
},
|
||||
);
|
||||
} catch (e) {
|
||||
_licensesError = '라이선스 정보를 불러올 수 없습니다';
|
||||
_expiringLicenses = [];
|
||||
DebugLogger.logError('만료 라이선스 로드 예외', error: e);
|
||||
}
|
||||
|
||||
_isLoadingLicenses = false;
|
||||
notifyListeners();
|
||||
|
||||
@@ -125,13 +125,13 @@ class _OverviewScreenRedesignState extends State<OverviewScreenRedesign> {
|
||||
),
|
||||
_buildStatCard(
|
||||
'입고 장비',
|
||||
'${_controller.overviewStats?.availableEquipment ?? 0}',
|
||||
'${_controller.equipmentStatus?.available ?? 0}',
|
||||
Icons.inventory,
|
||||
ShadcnTheme.success,
|
||||
),
|
||||
_buildStatCard(
|
||||
'출고 장비',
|
||||
'${_controller.overviewStats?.inUseEquipment ?? 0}',
|
||||
'${_controller.equipmentStatus?.inUse ?? 0}',
|
||||
Icons.local_shipping,
|
||||
ShadcnTheme.warning,
|
||||
),
|
||||
@@ -300,7 +300,7 @@ class _OverviewScreenRedesignState extends State<OverviewScreenRedesign> {
|
||||
const SizedBox(height: 16),
|
||||
Consumer<OverviewController>(
|
||||
builder: (context, controller, child) {
|
||||
final activities = controller.recentActivities ?? [];
|
||||
final activities = controller.recentActivities;
|
||||
if (activities.isEmpty) {
|
||||
return Padding(
|
||||
padding: const EdgeInsets.symmetric(vertical: 20),
|
||||
@@ -435,15 +435,19 @@ class _OverviewScreenRedesignState extends State<OverviewScreenRedesign> {
|
||||
|
||||
Widget _buildActivityItem(dynamic activity) {
|
||||
// 아이콘 매핑
|
||||
IconData getActivityIcon(String type) {
|
||||
switch (type) {
|
||||
IconData getActivityIcon(String? type) {
|
||||
switch (type?.toLowerCase()) {
|
||||
case 'equipment_in':
|
||||
case '장비 입고':
|
||||
return Icons.inventory;
|
||||
case 'equipment_out':
|
||||
case '장비 출고':
|
||||
return Icons.local_shipping;
|
||||
case 'company':
|
||||
case '회사':
|
||||
return Icons.business;
|
||||
case 'user':
|
||||
case '사용자':
|
||||
return Icons.person_add;
|
||||
default:
|
||||
return Icons.settings;
|
||||
@@ -451,23 +455,31 @@ class _OverviewScreenRedesignState extends State<OverviewScreenRedesign> {
|
||||
}
|
||||
|
||||
// 색상 매핑
|
||||
Color getActivityColor(String type) {
|
||||
switch (type) {
|
||||
Color getActivityColor(String? type) {
|
||||
switch (type?.toLowerCase()) {
|
||||
case 'equipment_in':
|
||||
case '장비 입고':
|
||||
return ShadcnTheme.success;
|
||||
case 'equipment_out':
|
||||
case '장비 출고':
|
||||
return ShadcnTheme.warning;
|
||||
case 'company':
|
||||
case '회사':
|
||||
return ShadcnTheme.info;
|
||||
case 'user':
|
||||
case '사용자':
|
||||
return ShadcnTheme.primary;
|
||||
default:
|
||||
return ShadcnTheme.mutedForeground;
|
||||
}
|
||||
}
|
||||
|
||||
final color = getActivityColor(activity.activityType);
|
||||
final activityType = activity.activityType ?? '';
|
||||
final color = getActivityColor(activityType);
|
||||
final dateFormat = DateFormat('MM/dd HH:mm');
|
||||
final timestamp = activity.timestamp ?? DateTime.now();
|
||||
final entityName = activity.entityName ?? '이름 없음';
|
||||
final description = activity.description ?? '설명 없음';
|
||||
|
||||
return Padding(
|
||||
padding: const EdgeInsets.symmetric(vertical: 8),
|
||||
@@ -480,7 +492,7 @@ class _OverviewScreenRedesignState extends State<OverviewScreenRedesign> {
|
||||
borderRadius: BorderRadius.circular(6),
|
||||
),
|
||||
child: Icon(
|
||||
getActivityIcon(activity.activityType),
|
||||
getActivityIcon(activityType),
|
||||
color: color,
|
||||
size: 16,
|
||||
),
|
||||
@@ -491,18 +503,20 @@ class _OverviewScreenRedesignState extends State<OverviewScreenRedesign> {
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
activity.entityName,
|
||||
entityName,
|
||||
style: ShadcnTheme.bodyMedium,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
Text(
|
||||
activity.description,
|
||||
description,
|
||||
style: ShadcnTheme.bodySmall,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
Text(
|
||||
dateFormat.format(activity.timestamp),
|
||||
dateFormat.format(timestamp),
|
||||
style: ShadcnTheme.bodySmall,
|
||||
),
|
||||
],
|
||||
|
||||
Reference in New Issue
Block a user