refactor: Repository 패턴 적용 및 Clean Architecture 완성
## 주요 변경사항 ### 🏗️ Architecture - Repository 패턴 전면 도입 (인터페이스/구현체 분리) - Domain Layer에 Repository 인터페이스 정의 - Data Layer에 Repository 구현체 배치 - UseCase 의존성을 Service에서 Repository로 전환 ### 📦 Dependency Injection - GetIt 기반 DI Container 재구성 (lib/injection_container.dart) - Repository 인터페이스와 구현체 등록 - Service와 Repository 공존 (마이그레이션 기간) ### 🔄 Migration Status 완료: - License 모듈 (6개 UseCase) - Warehouse Location 모듈 (5개 UseCase) 진행중: - Auth 모듈 (2/5 UseCase) - Company 모듈 (1/6 UseCase) 대기: - User 모듈 (7개 UseCase) - Equipment 모듈 (4개 UseCase) ### 🎯 Controller 통합 - 중복 Controller 제거 (with_usecase 버전) - 단일 Controller로 통합 - UseCase 패턴 직접 적용 ### 🧹 코드 정리 - 임시 파일 제거 (test_*.md, task.md) - Node.js 아티팩트 제거 (package.json) - 불필요한 테스트 파일 정리 ### ✅ 테스트 개선 - Real API 중심 테스트 구조 - Mock 제거, 실제 API 엔드포인트 사용 - 통합 테스트 프레임워크 강화 ## 기술적 영향 - 의존성 역전 원칙 적용 - 레이어 간 결합도 감소 - 테스트 용이성 향상 - 확장성 및 유지보수성 개선 ## 다음 단계 1. User/Equipment 모듈 Repository 마이그레이션 2. Service Layer 점진적 제거 3. 캐싱 전략 구현 4. 성능 최적화
This commit is contained in:
@@ -128,7 +128,7 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
|
||||
// 기능 감지
|
||||
final features = await detectFeatures(metadata);
|
||||
_log('감지된 기능: ${features.items.map((f) => f.featureName).join(', ')}');
|
||||
_log('감지된 기능: ${features.map((f) => f.featureName).join(', ')}');
|
||||
|
||||
// 테스트 실행
|
||||
final result = await executeTests(features);
|
||||
@@ -200,7 +200,7 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
final companyService = getIt.get<CompanyService>();
|
||||
final companies = await companyService.getCompanies(page: 1, perPage: 1);
|
||||
|
||||
if (companies.isEmpty) {
|
||||
if (companies.items.isEmpty) {
|
||||
// 테스트용 회사 생성
|
||||
final companyData = await dataGenerator.generate(
|
||||
GenerationStrategy(
|
||||
@@ -214,7 +214,7 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
final company = await companyService.createCompany(companyData.data);
|
||||
testContext.setData('testCompanyId', company.id);
|
||||
} else {
|
||||
testContext.setData('testCompanyId', companies.first.id);
|
||||
testContext.setData('testCompanyId', companies.items.first.id);
|
||||
}
|
||||
} catch (e) {
|
||||
// 회사 생성은 선택사항이므로 에러 무시
|
||||
@@ -234,7 +234,7 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
perPage: 1,
|
||||
);
|
||||
|
||||
if (warehouses.isEmpty) {
|
||||
if (warehouses.items.isEmpty) {
|
||||
// 테스트용 창고 생성
|
||||
final warehouseData = await dataGenerator.generate(
|
||||
GenerationStrategy(
|
||||
@@ -249,7 +249,7 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
final warehouse = await warehouseService.createWarehouseLocation(warehouseData.data);
|
||||
testContext.setData('testWarehouseId', warehouse.id);
|
||||
} else {
|
||||
testContext.setData('testWarehouseId', warehouses.first.id);
|
||||
testContext.setData('testWarehouseId', warehouses.items.first.id);
|
||||
}
|
||||
}
|
||||
} catch (e) {
|
||||
@@ -266,7 +266,7 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
// createdIds를 resourceType별로 분류
|
||||
for (final id in createdIds) {
|
||||
final parts = id.split(':');
|
||||
if (parts.items.length == 2) {
|
||||
if (parts.length == 2) {
|
||||
final resourceType = parts[0];
|
||||
final resourceId = parts[1];
|
||||
resourcesByType.putIfAbsent(resourceType, () => []).add(resourceId);
|
||||
@@ -375,9 +375,9 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
);
|
||||
|
||||
testContext.setData('readResults', results);
|
||||
testContext.setData('readCount', results is List ? results.items.length : 1);
|
||||
testContext.setData('readCount', results is List ? results.length : 1);
|
||||
|
||||
_log('[READ] 성공: ${results is List ? results.items.length : 1}개 항목');
|
||||
_log('[READ] 성공: ${results is List ? results.length : 1}개 항목');
|
||||
} catch (e) {
|
||||
_log('[READ] 실패: $e');
|
||||
|
||||
@@ -496,7 +496,7 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
await performCreate(data);
|
||||
|
||||
final service = getService();
|
||||
final searchKeyword = data.data['name']?.toString().split(' ').items.first ?? 'test';
|
||||
final searchKeyword = data.data['name']?.toString().split(' ').first ?? 'test';
|
||||
|
||||
final results = await service.search(searchKeyword);
|
||||
testContext.setData('searchResults', results);
|
||||
@@ -511,9 +511,9 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
expect(searchResults, isNotNull, reason: '검색 결과가 없음');
|
||||
expect(searchResults, isA<List>(), reason: '올바른 검색 결과 형식이 아님');
|
||||
|
||||
if (searchResults.items.isNotEmpty) {
|
||||
if (searchResults.isNotEmpty) {
|
||||
// 검색 결과가 키워드를 포함하는지 확인
|
||||
final firstResult = searchResults.items.first;
|
||||
final firstResult = searchResults.first;
|
||||
expect(
|
||||
firstResult.toString().toLowerCase(),
|
||||
contains(searchKeyword.toLowerCase()),
|
||||
@@ -564,9 +564,9 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
expect(page2Results, isNotNull, reason: '두 번째 페이지 결과가 없음');
|
||||
|
||||
// 페이지별 결과가 다른지 확인 (데이터가 충분한 경우)
|
||||
if (page1Results.items.isNotEmpty && page2Results.items.isNotEmpty) {
|
||||
if (page1Results.isNotEmpty && page2Results.isNotEmpty) {
|
||||
expect(
|
||||
page1Results.items.first.id != page2Results.items.first.id,
|
||||
page1Results.first.id != page2Results.first.id,
|
||||
isTrue,
|
||||
reason: '페이지네이션이 올바르게 작동하지 않음',
|
||||
);
|
||||
@@ -658,7 +658,7 @@ abstract class BaseScreenTest extends ScreenTestFramework {
|
||||
final fixResult = await autoFixer.attemptAutoFix(diagnosis);
|
||||
|
||||
if (fixResult.success) {
|
||||
_log('자동 수정 성공: ${fixResult.executedActions.items.length}개 액션 적용');
|
||||
_log('자동 수정 성공: ${fixResult.executedActions.length}개 액션 적용');
|
||||
|
||||
// 수정 액션 적용 (AutoFixResult는 String 액션을 반환)
|
||||
// TODO: String 액션을 FixAction으로 변환하거나 별도 처리 필요
|
||||
|
||||
Reference in New Issue
Block a user