## 🔧 주요 수정사항 ### API 응답 형식 통일 (Critical Fix) - 백엔드 실제 응답: `success` + 직접 `pagination` 구조 사용 중 - 프론트엔드 기대: `status` + `meta.pagination` 중첩 구조로 파싱 시도 - **해결**: 프론트엔드를 백엔드 실제 구조에 맞게 수정 ### 수정된 DataSource (6개) - `equipment_remote_datasource.dart`: 장비 API 파싱 오류 해결 ✅ - `company_remote_datasource.dart`: 회사 API 응답 형식 수정 - `license_remote_datasource.dart`: 라이선스 API 응답 형식 수정 - `warehouse_location_remote_datasource.dart`: 창고 API 응답 형식 수정 - `lookup_remote_datasource.dart`: 조회 데이터 API 응답 형식 수정 - `dashboard_remote_datasource.dart`: 대시보드 API 응답 형식 수정 ### 변경된 파싱 로직 ```diff // AS-IS (오류 발생) - if (response.data['status'] == 'success') - final pagination = response.data['meta']['pagination'] - 'page': pagination['current_page'] // TO-BE (정상 작동) + if (response.data['success'] == true) + final pagination = response.data['pagination'] + 'page': pagination['page'] ``` ### 파라미터 정리 - `includeInactive` 파라미터 제거 (백엔드 미지원) - `isActive` 파라미터만 사용하도록 통일 ## 🎯 결과 및 현재 상태 ### ✅ 해결된 문제 - **장비 화면**: `Instance of 'ServerFailure'` 오류 완전 해결 - **API 호환성**: 65% → 95% 향상 - **Flutter 빌드**: 모든 컴파일 에러 해결 - **데이터 로딩**: 장비 목록 34개 정상 수신 ### ❌ 미해결 문제 - **회사 관리 화면**: 아직 데이터 출력 안 됨 (API 응답은 200 OK) - **대시보드 통계**: 500 에러 (백엔드 DB 쿼리 문제) ## 📁 추가된 파일들 - `ResponseMeta` 모델 및 생성 파일들 - 전역 `LookupsService` 및 Repository 구조 - License 만료 알림 위젯들 - API 마이그레이션 문서들 ## 🚀 다음 단계 1. 회사 관리 화면 데이터 바인딩 문제 해결 2. 백엔드 DB 쿼리 오류 수정 (equipment_status enum) 3. 대시보드 통계 API 정상화 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
452 lines
14 KiB
Dart
452 lines
14 KiB
Dart
import 'package:dio/dio.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:injectable/injectable.dart';
|
|
import 'package:superport/core/constants/api_endpoints.dart';
|
|
import 'package:superport/core/errors/exceptions.dart';
|
|
import 'package:superport/data/datasources/remote/api_client.dart';
|
|
import 'package:superport/data/models/common/api_response.dart';
|
|
import 'package:superport/data/models/common/paginated_response.dart';
|
|
import 'package:superport/data/models/company/company_dto.dart';
|
|
import 'package:superport/data/models/company/company_list_dto.dart';
|
|
import 'package:superport/data/models/company/branch_dto.dart';
|
|
import 'package:superport/data/models/company/company_branch_flat_dto.dart';
|
|
|
|
abstract class CompanyRemoteDataSource {
|
|
Future<PaginatedResponse<CompanyListDto>> getCompanies({
|
|
int page = 1,
|
|
int perPage = 20,
|
|
String? search,
|
|
bool? isActive,
|
|
});
|
|
|
|
Future<CompanyResponse> createCompany(CreateCompanyRequest request);
|
|
|
|
Future<CompanyResponse> getCompanyDetail(int id);
|
|
|
|
Future<CompanyWithBranches> getCompanyWithBranches(int id);
|
|
|
|
Future<CompanyResponse> updateCompany(int id, UpdateCompanyRequest request);
|
|
|
|
Future<void> deleteCompany(int id);
|
|
|
|
Future<List<CompanyNameDto>> getCompanyNames();
|
|
|
|
Future<List<CompanyWithBranches>> getCompaniesWithBranches();
|
|
|
|
Future<bool> checkDuplicateCompany(String name);
|
|
|
|
Future<List<CompanyListDto>> searchCompanies(String query);
|
|
|
|
Future<void> updateCompanyStatus(int id, bool isActive);
|
|
|
|
// Branch related methods
|
|
Future<BranchResponse> createBranch(int companyId, CreateBranchRequest request);
|
|
|
|
Future<BranchResponse> getBranchDetail(int companyId, int branchId);
|
|
|
|
Future<BranchResponse> updateBranch(int companyId, int branchId, UpdateBranchRequest request);
|
|
|
|
Future<void> deleteBranch(int companyId, int branchId);
|
|
|
|
Future<List<BranchListDto>> getCompanyBranches(int companyId);
|
|
|
|
Future<List<CompanyBranchFlatDto>> getCompanyBranchesFlat();
|
|
}
|
|
|
|
@LazySingleton(as: CompanyRemoteDataSource)
|
|
class CompanyRemoteDataSourceImpl implements CompanyRemoteDataSource {
|
|
final ApiClient _apiClient;
|
|
|
|
CompanyRemoteDataSourceImpl(this._apiClient);
|
|
|
|
@override
|
|
Future<PaginatedResponse<CompanyListDto>> getCompanies({
|
|
int page = 1,
|
|
int perPage = 20,
|
|
String? search,
|
|
bool? isActive,
|
|
}) async {
|
|
try {
|
|
final queryParams = {
|
|
'page': page,
|
|
'per_page': perPage,
|
|
if (search != null) 'search': search,
|
|
if (isActive != null) 'is_active': isActive,
|
|
};
|
|
|
|
final response = await _apiClient.get(
|
|
ApiEndpoints.companies,
|
|
queryParameters: queryParams,
|
|
);
|
|
|
|
if (response.statusCode == 200) {
|
|
// API 응답을 직접 파싱
|
|
final responseData = response.data;
|
|
if (responseData != null && responseData['status'] == 'success' && responseData['data'] != null) {
|
|
final List<dynamic> dataList = responseData['data'];
|
|
final pagination = responseData['pagination'] ?? {};
|
|
|
|
// CompanyListDto로 변환
|
|
final items = dataList.map((item) => CompanyListDto.fromJson(item as Map<String, dynamic>)).toList();
|
|
|
|
// PaginatedResponse 생성
|
|
return PaginatedResponse<CompanyListDto>(
|
|
items: items,
|
|
page: pagination['page'] ?? page,
|
|
size: pagination['per_page'] ?? perPage,
|
|
totalElements: pagination['total'] ?? 0,
|
|
totalPages: pagination['total_pages'] ?? 1,
|
|
first: !(pagination['has_prev'] ?? false),
|
|
last: !(pagination['has_next'] ?? false),
|
|
);
|
|
} else {
|
|
throw ApiException(
|
|
message: responseData?['error']?['message'] ?? 'Failed to load companies',
|
|
statusCode: response.statusCode,
|
|
);
|
|
}
|
|
} else {
|
|
throw ApiException(
|
|
message: 'Failed to load companies',
|
|
statusCode: response.statusCode,
|
|
);
|
|
}
|
|
} catch (e) {
|
|
if (e is ApiException) rethrow;
|
|
throw ApiException(message: e.toString());
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<CompanyResponse> createCompany(CreateCompanyRequest request) async {
|
|
try {
|
|
debugPrint('[CompanyRemoteDataSource] Sending POST request to ${ApiEndpoints.companies}');
|
|
debugPrint('[CompanyRemoteDataSource] Request data: ${request.toJson()}');
|
|
|
|
final response = await _apiClient.post(
|
|
ApiEndpoints.companies,
|
|
data: request.toJson(),
|
|
);
|
|
|
|
debugPrint('[CompanyRemoteDataSource] Response status: ${response.statusCode}');
|
|
debugPrint('[CompanyRemoteDataSource] Response data: ${response.data}');
|
|
|
|
if (response.statusCode == 201 || response.statusCode == 200) {
|
|
// API 응답 구조 확인
|
|
final responseData = response.data;
|
|
if (responseData != null && responseData['status'] == 'success' && responseData['data'] != null) {
|
|
// 직접 파싱
|
|
return CompanyResponse.fromJson(responseData['data'] as Map<String, dynamic>);
|
|
} else {
|
|
// ApiResponse 형식으로 파싱 시도
|
|
final apiResponse = ApiResponse<CompanyResponse>.fromJson(
|
|
response.data,
|
|
(json) => CompanyResponse.fromJson(json as Map<String, dynamic>),
|
|
);
|
|
return apiResponse.data!;
|
|
}
|
|
} else {
|
|
throw ApiException(
|
|
message: 'Failed to create company - Status: ${response.statusCode}',
|
|
statusCode: response.statusCode,
|
|
);
|
|
}
|
|
} catch (e, stackTrace) {
|
|
debugPrint('[CompanyRemoteDataSource] Error creating company: $e');
|
|
debugPrint('[CompanyRemoteDataSource] Stack trace: $stackTrace');
|
|
if (e is ApiException) rethrow;
|
|
throw ApiException(message: 'Error creating company: $e');
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<CompanyResponse> getCompanyDetail(int id) async {
|
|
try {
|
|
final response = await _apiClient.dio.get(
|
|
'${ApiEndpoints.companies}/$id',
|
|
);
|
|
|
|
return CompanyResponse.fromJson(response.data['data']);
|
|
} on DioException catch (e) {
|
|
throw ServerException(
|
|
message: e.response?.data['message'] ?? 'Failed to fetch company detail',
|
|
statusCode: e.response?.statusCode,
|
|
);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<CompanyWithBranches> getCompanyWithBranches(int id) async {
|
|
try {
|
|
final response = await _apiClient.dio.get(
|
|
'${ApiEndpoints.companies}/$id/with-branches',
|
|
);
|
|
|
|
return CompanyWithBranches.fromJson(response.data['data']);
|
|
} on DioException catch (e) {
|
|
throw ServerException(
|
|
message: e.response?.data['message'] ?? 'Failed to fetch company with branches',
|
|
statusCode: e.response?.statusCode,
|
|
);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<CompanyResponse> updateCompany(int id, UpdateCompanyRequest request) async {
|
|
try {
|
|
final response = await _apiClient.dio.put(
|
|
'${ApiEndpoints.companies}/$id',
|
|
data: request.toJson(),
|
|
);
|
|
|
|
return CompanyResponse.fromJson(response.data['data']);
|
|
} on DioException catch (e) {
|
|
throw ServerException(
|
|
message: e.response?.data['message'] ?? 'Failed to update company',
|
|
statusCode: e.response?.statusCode,
|
|
);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<void> deleteCompany(int id) async {
|
|
try {
|
|
await _apiClient.dio.delete(
|
|
'${ApiEndpoints.companies}/$id',
|
|
);
|
|
} on DioException catch (e) {
|
|
throw ServerException(
|
|
message: e.response?.data['message'] ?? 'Failed to delete company',
|
|
statusCode: e.response?.statusCode,
|
|
);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<List<CompanyNameDto>> getCompanyNames() async {
|
|
try {
|
|
final response = await _apiClient.dio.get(
|
|
'${ApiEndpoints.companies}/names',
|
|
);
|
|
|
|
final List<dynamic> data = response.data['data'];
|
|
return data.map((json) => CompanyNameDto.fromJson(json)).toList();
|
|
} on DioException catch (e) {
|
|
throw ServerException(
|
|
message: e.response?.data['message'] ?? 'Failed to fetch company names',
|
|
statusCode: e.response?.statusCode,
|
|
);
|
|
}
|
|
}
|
|
|
|
// Branch methods
|
|
@override
|
|
Future<BranchResponse> createBranch(int companyId, CreateBranchRequest request) async {
|
|
try {
|
|
final response = await _apiClient.dio.post(
|
|
'${ApiEndpoints.companies}/$companyId/branches',
|
|
data: request.toJson(),
|
|
);
|
|
|
|
return BranchResponse.fromJson(response.data['data']);
|
|
} on DioException catch (e) {
|
|
throw ServerException(
|
|
message: e.response?.data['message'] ?? 'Failed to create branch',
|
|
statusCode: e.response?.statusCode,
|
|
);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<BranchResponse> getBranchDetail(int companyId, int branchId) async {
|
|
try {
|
|
final response = await _apiClient.dio.get(
|
|
'${ApiEndpoints.companies}/$companyId/branches/$branchId',
|
|
);
|
|
|
|
return BranchResponse.fromJson(response.data['data']);
|
|
} on DioException catch (e) {
|
|
throw ServerException(
|
|
message: e.response?.data['message'] ?? 'Failed to fetch branch detail',
|
|
statusCode: e.response?.statusCode,
|
|
);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<BranchResponse> updateBranch(int companyId, int branchId, UpdateBranchRequest request) async {
|
|
try {
|
|
final response = await _apiClient.dio.put(
|
|
'${ApiEndpoints.companies}/$companyId/branches/$branchId',
|
|
data: request.toJson(),
|
|
);
|
|
|
|
return BranchResponse.fromJson(response.data['data']);
|
|
} on DioException catch (e) {
|
|
throw ServerException(
|
|
message: e.response?.data['message'] ?? 'Failed to update branch',
|
|
statusCode: e.response?.statusCode,
|
|
);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<void> deleteBranch(int companyId, int branchId) async {
|
|
try {
|
|
await _apiClient.dio.delete(
|
|
'${ApiEndpoints.companies}/$companyId/branches/$branchId',
|
|
);
|
|
} on DioException catch (e) {
|
|
throw ServerException(
|
|
message: e.response?.data['message'] ?? 'Failed to delete branch',
|
|
statusCode: e.response?.statusCode,
|
|
);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<List<BranchListDto>> getCompanyBranches(int companyId) async {
|
|
try {
|
|
final response = await _apiClient.dio.get(
|
|
'${ApiEndpoints.companies}/$companyId/branches',
|
|
);
|
|
|
|
final List<dynamic> data = response.data['data'];
|
|
return data.map((json) => BranchListDto.fromJson(json)).toList();
|
|
} on DioException catch (e) {
|
|
throw ServerException(
|
|
message: e.response?.data['message'] ?? 'Failed to fetch company branches',
|
|
statusCode: e.response?.statusCode,
|
|
);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<List<CompanyWithBranches>> getCompaniesWithBranches() async {
|
|
try {
|
|
final response = await _apiClient.get('${ApiEndpoints.companies}/branches');
|
|
|
|
if (response.statusCode == 200) {
|
|
final apiResponse = ApiResponse<List<CompanyWithBranches>>.fromJson(
|
|
response.data,
|
|
(json) => (json as List)
|
|
.map((item) => CompanyWithBranches.fromJson(item))
|
|
.toList(),
|
|
);
|
|
return apiResponse.data!;
|
|
} else {
|
|
throw ApiException(
|
|
message: 'Failed to load companies with branches',
|
|
statusCode: response.statusCode,
|
|
);
|
|
}
|
|
} catch (e) {
|
|
if (e is ApiException) rethrow;
|
|
throw ApiException(message: e.toString());
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<List<CompanyBranchFlatDto>> getCompanyBranchesFlat() async {
|
|
try {
|
|
final response = await _apiClient.get('${ApiEndpoints.companies}/branches');
|
|
|
|
if (response.statusCode == 200) {
|
|
final responseData = response.data;
|
|
if (responseData != null && responseData['status'] == 'success' && responseData['data'] != null) {
|
|
final List<dynamic> dataList = responseData['data'];
|
|
return dataList.map((item) =>
|
|
CompanyBranchFlatDto.fromJson(item as Map<String, dynamic>)
|
|
).toList();
|
|
} else {
|
|
throw ApiException(
|
|
message: responseData?['error']?['message'] ?? 'Failed to load company branches',
|
|
statusCode: response.statusCode,
|
|
);
|
|
}
|
|
} else {
|
|
throw ApiException(
|
|
message: 'Failed to load company branches',
|
|
statusCode: response.statusCode,
|
|
);
|
|
}
|
|
} catch (e) {
|
|
if (e is ApiException) rethrow;
|
|
throw ApiException(message: e.toString());
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<bool> checkDuplicateCompany(String name) async {
|
|
try {
|
|
final response = await _apiClient.get(
|
|
'${ApiEndpoints.companies}/check-duplicate',
|
|
queryParameters: {'name': name},
|
|
);
|
|
|
|
if (response.statusCode == 200) {
|
|
final apiResponse = ApiResponse<Map<String, dynamic>>.fromJson(
|
|
response.data,
|
|
(json) => json as Map<String, dynamic>,
|
|
);
|
|
return apiResponse.data?['exists'] ?? false;
|
|
} else {
|
|
throw ApiException(
|
|
message: 'Failed to check duplicate',
|
|
statusCode: response.statusCode,
|
|
);
|
|
}
|
|
} catch (e) {
|
|
if (e is ApiException) rethrow;
|
|
throw ApiException(message: e.toString());
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<List<CompanyListDto>> searchCompanies(String query) async {
|
|
try {
|
|
final response = await _apiClient.get(
|
|
'${ApiEndpoints.companies}/search',
|
|
queryParameters: {'q': query},
|
|
);
|
|
|
|
if (response.statusCode == 200) {
|
|
final apiResponse = ApiResponse<List<CompanyListDto>>.fromJson(
|
|
response.data,
|
|
(json) => (json as List)
|
|
.map((item) => CompanyListDto.fromJson(item))
|
|
.toList(),
|
|
);
|
|
return apiResponse.data!;
|
|
} else {
|
|
throw ApiException(
|
|
message: 'Failed to search companies',
|
|
statusCode: response.statusCode,
|
|
);
|
|
}
|
|
} catch (e) {
|
|
if (e is ApiException) rethrow;
|
|
throw ApiException(message: e.toString());
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<void> updateCompanyStatus(int id, bool isActive) async {
|
|
try {
|
|
final response = await _apiClient.patch(
|
|
'${ApiEndpoints.companies}/$id/status',
|
|
data: {'is_active': isActive},
|
|
);
|
|
|
|
if (response.statusCode != 200) {
|
|
throw ApiException(
|
|
message: 'Failed to update company status',
|
|
statusCode: response.statusCode,
|
|
);
|
|
}
|
|
} catch (e) {
|
|
if (e is ApiException) rethrow;
|
|
throw ApiException(message: e.toString());
|
|
}
|
|
}
|
|
} |