## 주요 변경사항 ### 아키텍처 개선 - Clean Architecture 패턴 적용 (Domain, Data, Presentation 레이어 분리) - Use Case 패턴 도입으로 비즈니스 로직 캡슐화 - Repository 패턴으로 데이터 접근 추상화 - 의존성 주입 구조 개선 ### 상태 관리 최적화 - 모든 Controller에서 불필요한 상태 관리 로직 제거 - 페이지네이션 로직 통일 및 간소화 - 에러 처리 로직 개선 (에러 메시지 한글화) - 로딩 상태 관리 최적화 ### Mock 서비스 제거 - MockDataService 완전 제거 - 모든 화면을 실제 API 전용으로 전환 - 불필요한 Mock 관련 코드 정리 ### UI/UX 개선 - Overview 화면 대시보드 기능 강화 - 라이선스 만료 알림 위젯 추가 - 사이드바 네비게이션 개선 - 일관된 UI 컴포넌트 사용 ### 코드 품질 - 중복 코드 제거 및 함수 추출 - 파일별 책임 분리 명확화 - 테스트 코드 업데이트 ## 영향 범위 - 모든 화면의 Controller 리팩토링 - API 통신 레이어 구조 개선 - 에러 처리 및 로깅 시스템 개선 ## 향후 계획 - 단위 테스트 커버리지 확대 - 통합 테스트 시나리오 추가 - 성능 모니터링 도구 통합
189 lines
5.7 KiB
Dart
189 lines
5.7 KiB
Dart
import 'package:flutter_test/flutter_test.dart';
|
|
import 'package:mockito/mockito.dart';
|
|
import 'package:mockito/annotations.dart';
|
|
import 'package:dartz/dartz.dart';
|
|
import 'package:superport/data/models/license/license_dto.dart';
|
|
import 'package:superport/data/repositories/license_repository.dart';
|
|
import 'package:superport/domain/usecases/base_usecase.dart';
|
|
import 'package:superport/domain/usecases/license/create_license_usecase.dart';
|
|
|
|
import 'create_license_usecase_test.mocks.dart';
|
|
|
|
@GenerateMocks([LicenseRepository])
|
|
void main() {
|
|
late CreateLicenseUseCase useCase;
|
|
late MockLicenseRepository mockRepository;
|
|
|
|
setUp(() {
|
|
mockRepository = MockLicenseRepository();
|
|
useCase = CreateLicenseUseCase(mockRepository);
|
|
});
|
|
|
|
group('CreateLicenseUseCase', () {
|
|
final validParams = CreateLicenseParams(
|
|
equipmentId: 1,
|
|
companyId: 1,
|
|
licenseType: 'maintenance',
|
|
startDate: DateTime(2025, 1, 1),
|
|
expiryDate: DateTime(2025, 12, 31),
|
|
description: 'Test license',
|
|
cost: 1000.0,
|
|
);
|
|
|
|
final mockLicense = LicenseDto(
|
|
id: 1,
|
|
equipmentId: 1,
|
|
companyId: 1,
|
|
licenseType: 'maintenance',
|
|
startDate: DateTime(2025, 1, 1),
|
|
expiryDate: DateTime(2025, 12, 31),
|
|
description: 'Test license',
|
|
cost: 1000.0,
|
|
status: 'active',
|
|
createdAt: DateTime.now(),
|
|
updatedAt: DateTime.now(),
|
|
);
|
|
|
|
test('라이선스 생성 성공', () async {
|
|
// arrange
|
|
when(mockRepository.createLicense(any))
|
|
.thenAnswer((_) async => mockLicense);
|
|
|
|
// act
|
|
final result = await useCase(validParams);
|
|
|
|
// assert
|
|
expect(result.isRight(), true);
|
|
result.fold(
|
|
(failure) => fail('Should not return failure'),
|
|
(license) => expect(license, equals(mockLicense)),
|
|
);
|
|
verify(mockRepository.createLicense(validParams.toMap())).called(1);
|
|
});
|
|
|
|
test('만료일이 시작일보다 이전인 경우 검증 실패', () async {
|
|
// arrange
|
|
final invalidParams = CreateLicenseParams(
|
|
equipmentId: 1,
|
|
companyId: 1,
|
|
licenseType: 'maintenance',
|
|
startDate: DateTime(2025, 12, 31),
|
|
expiryDate: DateTime(2025, 1, 1), // 시작일보다 이전
|
|
description: 'Test license',
|
|
cost: 1000.0,
|
|
);
|
|
|
|
// act
|
|
final result = await useCase(invalidParams);
|
|
|
|
// assert
|
|
expect(result.isLeft(), true);
|
|
result.fold(
|
|
(failure) {
|
|
expect(failure, isA<ValidationFailure>());
|
|
expect(failure.message, contains('만료일은 시작일 이후여야 합니다'));
|
|
},
|
|
(license) => fail('Should not return license'),
|
|
);
|
|
verifyNever(mockRepository.createLicense(any));
|
|
});
|
|
|
|
test('라이선스 기간이 30일 미만인 경우 검증 실패', () async {
|
|
// arrange
|
|
final invalidParams = CreateLicenseParams(
|
|
equipmentId: 1,
|
|
companyId: 1,
|
|
licenseType: 'maintenance',
|
|
startDate: DateTime(2025, 1, 1),
|
|
expiryDate: DateTime(2025, 1, 15), // 15일 기간
|
|
description: 'Test license',
|
|
cost: 1000.0,
|
|
);
|
|
|
|
// act
|
|
final result = await useCase(invalidParams);
|
|
|
|
// assert
|
|
expect(result.isLeft(), true);
|
|
result.fold(
|
|
(failure) {
|
|
expect(failure, isA<ValidationFailure>());
|
|
expect(failure.message, contains('라이선스 기간은 최소 30일 이상이어야 합니다'));
|
|
},
|
|
(license) => fail('Should not return license'),
|
|
);
|
|
verifyNever(mockRepository.createLicense(any));
|
|
});
|
|
|
|
test('Repository에서 예외 발생 시 ServerFailure 반환', () async {
|
|
// arrange
|
|
when(mockRepository.createLicense(any))
|
|
.thenThrow(Exception('Server error'));
|
|
|
|
// act
|
|
final result = await useCase(validParams);
|
|
|
|
// assert
|
|
expect(result.isLeft(), true);
|
|
result.fold(
|
|
(failure) {
|
|
expect(failure, isA<ServerFailure>());
|
|
expect(failure.message, contains('Server error'));
|
|
},
|
|
(license) => fail('Should not return license'),
|
|
);
|
|
verify(mockRepository.createLicense(validParams.toMap())).called(1);
|
|
});
|
|
|
|
test('파라미터를 올바른 Map으로 변환', () {
|
|
// arrange
|
|
final params = CreateLicenseParams(
|
|
equipmentId: 1,
|
|
companyId: 2,
|
|
licenseType: 'maintenance',
|
|
startDate: DateTime(2025, 1, 1),
|
|
expiryDate: DateTime(2025, 12, 31),
|
|
description: 'Test description',
|
|
cost: 5000.0,
|
|
);
|
|
|
|
// act
|
|
final map = params.toMap();
|
|
|
|
// assert
|
|
expect(map['equipment_id'], equals(1));
|
|
expect(map['company_id'], equals(2));
|
|
expect(map['license_type'], equals('maintenance'));
|
|
expect(map['start_date'], equals(DateTime(2025, 1, 1).toIso8601String()));
|
|
expect(map['expiry_date'], equals(DateTime(2025, 12, 31).toIso8601String()));
|
|
expect(map['description'], equals('Test description'));
|
|
expect(map['cost'], equals(5000.0));
|
|
});
|
|
|
|
test('옵셔널 파라미터가 null인 경우에도 정상 처리', () async {
|
|
// arrange
|
|
final paramsWithNulls = CreateLicenseParams(
|
|
equipmentId: 1,
|
|
companyId: 1,
|
|
licenseType: 'maintenance',
|
|
startDate: DateTime(2025, 1, 1),
|
|
expiryDate: DateTime(2025, 12, 31),
|
|
description: null,
|
|
cost: null,
|
|
);
|
|
|
|
when(mockRepository.createLicense(any))
|
|
.thenAnswer((_) async => mockLicense);
|
|
|
|
// act
|
|
final result = await useCase(paramsWithNulls);
|
|
|
|
// assert
|
|
expect(result.isRight(), true);
|
|
|
|
final map = paramsWithNulls.toMap();
|
|
expect(map['description'], isNull);
|
|
expect(map['cost'], isNull);
|
|
});
|
|
});
|
|
} |