import 'package:dartz/dartz.dart'; import 'package:injectable/injectable.dart'; import '../../core/errors/failures.dart'; import '../../domain/repositories/company_repository.dart'; import '../../models/company_model.dart'; import '../../models/address_model.dart'; import '../datasources/remote/company_remote_datasource.dart'; import '../models/common/paginated_response.dart'; import '../models/company/company_dto.dart'; import '../models/company/branch_dto.dart'; import '../models/company/company_list_dto.dart'; /// 회사 관리 Repository 구현체 /// 회사 및 지점 정보 CRUD 작업을 처리하며 도메인 모델과 API DTO 간 변환을 담당 @Injectable(as: CompanyRepository) class CompanyRepositoryImpl implements CompanyRepository { final CompanyRemoteDataSource remoteDataSource; CompanyRepositoryImpl({required this.remoteDataSource}); @override Future>> getCompanies({ int? page, int? limit, String? search, CompanyType? companyType, String? sortBy, String? sortOrder, }) async { try { final result = await remoteDataSource.getCompanies( page: page ?? 1, perPage: limit ?? 20, search: search, isActive: null, // companyType에 따른 필터링 로직 필요 시 추가 ); // DTO를 도메인 모델로 변환 final companies = result.items.map((dto) => _mapDtoToDomain(dto)).toList(); final paginatedResult = PaginatedResponse( items: companies, page: result.page, size: result.size, totalElements: result.totalElements, totalPages: result.totalPages, first: result.first, last: result.last, ); return Right(paginatedResult); } catch (e) { return Left(ServerFailure( message: '회사 목록 조회 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future> getCompanyById(int id) async { try { final result = await remoteDataSource.getCompanyWithBranches(id); final company = _mapDetailDtoToDomain(result); return Right(company); } catch (e) { if (e.toString().contains('404')) { return Left(NotFoundFailure( message: '해당 회사를 찾을 수 없습니다.', resourceType: 'Company', resourceId: id.toString(), )); } return Left(ServerFailure( message: '회사 상세 정보 조회 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future> createCompany(Company company) async { try { final request = _mapDomainToCreateRequest(company); final result = await remoteDataSource.createCompany(request); final createdCompany = _mapResponseToDomain(result); return Right(createdCompany); } catch (e) { if (e.toString().contains('중복')) { return Left(DuplicateFailure( message: '이미 존재하는 회사명입니다.', field: 'name', value: company.name, )); } if (e.toString().contains('유효성')) { return Left(ValidationFailure( message: '입력 데이터가 올바르지 않습니다.', )); } return Left(ServerFailure( message: '회사 생성 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future> updateCompany(int id, Company company) async { try { final request = _mapDomainToUpdateRequest(company); final result = await remoteDataSource.updateCompany(id, request); final updatedCompany = _mapResponseToDomain(result); return Right(updatedCompany); } catch (e) { if (e.toString().contains('404')) { return Left(NotFoundFailure( message: '수정할 회사를 찾을 수 없습니다.', resourceType: 'Company', resourceId: id.toString(), )); } if (e.toString().contains('중복')) { return Left(DuplicateFailure( message: '이미 존재하는 회사명입니다.', field: 'name', value: company.name, )); } return Left(ServerFailure( message: '회사 정보 수정 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future> deleteCompany(int id) async { try { await remoteDataSource.deleteCompany(id); return const Right(null); } catch (e) { if (e.toString().contains('404')) { return Left(NotFoundFailure( message: '삭제할 회사를 찾을 수 없습니다.', resourceType: 'Company', resourceId: id.toString(), )); } if (e.toString().contains('참조')) { return Left(BusinessFailure( message: '해당 회사에 연결된 데이터가 있어 삭제할 수 없습니다.', )); } return Left(ServerFailure( message: '회사 삭제 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future> toggleCompanyStatus(int id) async { try { // 현재 회사 정보 조회 final currentCompany = await remoteDataSource.getCompanyDetail(id); final newStatus = !currentCompany.isActive; // 상태 업데이트 await remoteDataSource.updateCompanyStatus(id, newStatus); // 업데이트된 회사 정보 재조회 final updatedCompany = await remoteDataSource.getCompanyDetail(id); final company = _mapResponseToDomain(updatedCompany); return Right(company); } catch (e) { if (e.toString().contains('404')) { return Left(NotFoundFailure( message: '상태를 변경할 회사를 찾을 수 없습니다.', resourceType: 'Company', resourceId: id.toString(), )); } return Left(ServerFailure( message: '회사 상태 변경 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future> createBranch(int companyId, Branch branch) async { try { final request = _mapBranchToCreateRequest(branch); final result = await remoteDataSource.createBranch(companyId, request); final createdBranch = _mapBranchResponseToDomain(result); return Right(createdBranch); } catch (e) { if (e.toString().contains('404')) { return Left(NotFoundFailure( message: '해당 회사를 찾을 수 없습니다.', resourceType: 'Company', resourceId: companyId.toString(), )); } return Left(ServerFailure( message: '지점 생성 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future> updateBranch(int companyId, int branchId, Branch branch) async { try { final request = _mapBranchToUpdateRequest(branch); final result = await remoteDataSource.updateBranch(companyId, branchId, request); final updatedBranch = _mapBranchResponseToDomain(result); return Right(updatedBranch); } catch (e) { if (e.toString().contains('404')) { return Left(NotFoundFailure( message: '수정할 지점을 찾을 수 없습니다.', resourceType: 'Branch', resourceId: branchId.toString(), )); } return Left(ServerFailure( message: '지점 정보 수정 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future> deleteBranch(int companyId, int branchId) async { try { await remoteDataSource.deleteBranch(companyId, branchId); return const Right(null); } catch (e) { if (e.toString().contains('404')) { return Left(NotFoundFailure( message: '삭제할 지점을 찾을 수 없습니다.', resourceType: 'Branch', resourceId: branchId.toString(), )); } return Left(ServerFailure( message: '지점 삭제 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future>> searchCompanyNames(String query, {int? limit}) async { try { final companies = await remoteDataSource.searchCompanies(query); final names = companies.map((company) => company.name).take(limit ?? 10).toList(); return Right(names); } catch (e) { return Left(ServerFailure( message: '회사명 검색 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future>> getCompanyCountByType() async { // TODO: API에서 회사 유형별 통계 기능이 구현되면 추가 return const Left(ServerFailure( message: '회사 유형별 통계 기능이 아직 구현되지 않았습니다.', )); } @override Future> hasLinkedUsers(int companyId) async { // TODO: 회사에 연결된 사용자 존재 여부 확인 API 구현 필요 try { // 임시로 false 반환 - API 구현 후 수정 필요 return const Right(false); } catch (e) { return Left(ServerFailure( message: '연결된 사용자 확인 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future> hasLinkedEquipment(int companyId) async { // TODO: 회사에 연결된 장비 존재 여부 확인 API 구현 필요 try { // 임시로 false 반환 - API 구현 후 수정 필요 return const Right(false); } catch (e) { return Left(ServerFailure( message: '연결된 장비 확인 중 오류가 발생했습니다: ${e.toString()}', )); } } @override Future> isDuplicateCompanyName(String name, {int? excludeId}) async { try { final isDuplicate = await remoteDataSource.checkDuplicateCompany(name); // excludeId가 있는 경우 해당 ID 제외 로직 추가 필요 return Right(isDuplicate); } catch (e) { return Left(ServerFailure( message: '중복 회사명 확인 중 오류가 발생했습니다: ${e.toString()}', )); } } // Private 매퍼 메서드들 Company _mapDtoToDomain(CompanyListDto dto) { return Company( id: dto.id, name: dto.name, address: Address.fromFullAddress(dto.address ?? ''), contactName: dto.contactName, contactPosition: null, // CompanyListDto에 없음 contactPhone: dto.contactPhone, contactEmail: dto.contactEmail, companyTypes: _parseCompanyTypes(dto.companyTypes), remark: null, // CompanyListDto에 없음 branches: [], // 목록에서는 지점 정보 비어있음 ); } Company _mapDetailDtoToDomain(CompanyWithBranches dto) { return Company( id: dto.company.id, name: dto.company.name, address: Address.fromFullAddress(dto.company.address ?? ''), contactName: dto.company.contactName, contactPosition: dto.company.contactPosition, contactPhone: dto.company.contactPhone, contactEmail: dto.company.contactEmail, companyTypes: _parseCompanyTypes(dto.company.companyTypes), remark: dto.company.remark, branches: dto.branches.map((branch) => _mapBranchDtoToDomain(branch)).toList(), ); } Company _mapResponseToDomain(CompanyResponse response) { return Company( id: response.id, name: response.name, address: Address.fromFullAddress(response.address ?? ''), contactName: response.contactName, contactPosition: response.contactPosition, contactPhone: response.contactPhone, contactEmail: response.contactEmail, companyTypes: _parseCompanyTypes(response.companyTypes), remark: response.remark, branches: [], // CompanyResponse에서는 지점 정보 따로 조회 ); } Branch _mapBranchDtoToDomain(BranchListDto dto) { return Branch( id: dto.id, companyId: dto.companyId, name: dto.branchName, address: Address.fromFullAddress(dto.address ?? ''), contactName: dto.managerName, contactPosition: null, // BranchListDto에 없음 contactPhone: dto.phone, contactEmail: null, // BranchListDto에 없음 remark: null, // BranchListDto에 없음 ); } Branch _mapBranchResponseToDomain(BranchResponse response) { return Branch( id: response.id, companyId: response.companyId, name: response.branchName, address: Address.fromFullAddress(response.address ?? ''), contactName: response.managerName, contactPosition: null, contactPhone: response.phone, contactEmail: null, remark: response.remark, ); } /// API에서 받은 문자열 리스트를 CompanyType enum 리스트로 변환 /// 지원하는 형식: ['customer', 'partner'] 또는 ['고객사', '파트너사'] List _parseCompanyTypes(List? types) { if (types == null || types.isEmpty) return [CompanyType.customer]; return types.map((type) { final lowerType = type.toLowerCase().trim(); // API 문자열 형식 매칭 if (lowerType == 'partner' || lowerType.contains('partner') || lowerType == '파트너사') { return CompanyType.partner; } // 기본값은 customer return CompanyType.customer; }).toList(); } /// CompanyType enum을 API 문자열로 변환 String _mapCompanyTypeToApiString(CompanyType type) { switch (type) { case CompanyType.partner: return 'partner'; case CompanyType.customer: return 'customer'; } } CreateCompanyRequest _mapDomainToCreateRequest(Company company) { return CreateCompanyRequest( name: company.name, address: company.address.toString(), contactName: company.contactName ?? '', contactPosition: company.contactPosition ?? '', contactPhone: company.contactPhone ?? '', contactEmail: company.contactEmail ?? '', companyTypes: company.companyTypes.map((type) => _mapCompanyTypeToApiString(type)).toList(), remark: company.remark, ); } UpdateCompanyRequest _mapDomainToUpdateRequest(Company company) { return UpdateCompanyRequest( name: company.name, address: company.address.toString(), contactName: company.contactName, contactPosition: company.contactPosition, contactPhone: company.contactPhone, contactEmail: company.contactEmail, companyTypes: company.companyTypes.map((type) => _mapCompanyTypeToApiString(type)).toList(), remark: company.remark, isActive: null, // UpdateCompanyRequest에서 필요한 경우 추가 ); } CreateBranchRequest _mapBranchToCreateRequest(Branch branch) { return CreateBranchRequest( branchName: branch.name, address: branch.address.toString(), phone: branch.contactPhone ?? '', managerName: branch.contactName, managerPhone: null, // Branch에 없음 remark: branch.remark, ); } UpdateBranchRequest _mapBranchToUpdateRequest(Branch branch) { return UpdateBranchRequest( branchName: branch.name, address: branch.address.toString(), phone: branch.contactPhone, managerName: branch.contactName, managerPhone: null, // Branch에 없음 remark: branch.remark, ); } }