fix: API 응답 파싱 오류 수정 및 에러 처리 개선

주요 변경사항:
- 창고 관리 API 응답 구조와 DTO 불일치 수정
  - WarehouseLocationDto에 code, manager_phone 필드 추가
  - RemoteDataSource에서 API 응답을 DTO 구조에 맞게 변환
- 회사 관리 API 응답 파싱 오류 수정
  - CompanyResponse의 필수 필드를 nullable로 변경
  - PaginatedResponse 구조 매핑 로직 개선
- 에러 처리 및 로깅 개선
  - Service Layer에 상세 에러 로깅 추가
  - Controller에서 에러 타입별 처리
- 새로운 유틸리티 추가
  - ResponseInterceptor: API 응답 정규화
  - DebugLogger: 디버깅 도구
  - HealthCheckService: 서버 상태 확인
- 문서화
  - API 통합 테스트 가이드
  - 에러 분석 보고서
  - 리팩토링 계획서
This commit is contained in:
JiWoong Sul
2025-07-31 19:15:39 +09:00
parent ad2c699ff7
commit f08b7fec79
89 changed files with 10521 additions and 892 deletions

View File

@@ -62,7 +62,13 @@ class LicenseRemoteDataSourceImpl implements LicenseRemoteDataSource {
queryParameters: queryParams,
);
return LicenseListResponseDto.fromJson(response.data);
if (response.data != null && response.data['success'] == true && response.data['data'] != null) {
return LicenseListResponseDto.fromJson(response.data['data']);
} else {
throw ApiException(
message: response.data?['error']?['message'] ?? 'Failed to fetch licenses',
);
}
} catch (e) {
throw _handleError(e);
}
@@ -75,7 +81,13 @@ class LicenseRemoteDataSourceImpl implements LicenseRemoteDataSource {
'${ApiEndpoints.licenses}/$id',
);
return LicenseDto.fromJson(response.data);
if (response.data != null && response.data['success'] == true && response.data['data'] != null) {
return LicenseDto.fromJson(response.data['data']);
} else {
throw ApiException(
message: response.data?['error']?['message'] ?? 'Failed to fetch license',
);
}
} catch (e) {
throw _handleError(e);
}
@@ -89,7 +101,13 @@ class LicenseRemoteDataSourceImpl implements LicenseRemoteDataSource {
data: request.toJson(),
);
return LicenseDto.fromJson(response.data);
if (response.data != null && response.data['success'] == true && response.data['data'] != null) {
return LicenseDto.fromJson(response.data['data']);
} else {
throw ApiException(
message: response.data?['error']?['message'] ?? 'Failed to fetch license',
);
}
} catch (e) {
throw _handleError(e);
}
@@ -103,7 +121,13 @@ class LicenseRemoteDataSourceImpl implements LicenseRemoteDataSource {
data: request.toJson(),
);
return LicenseDto.fromJson(response.data);
if (response.data != null && response.data['success'] == true && response.data['data'] != null) {
return LicenseDto.fromJson(response.data['data']);
} else {
throw ApiException(
message: response.data?['error']?['message'] ?? 'Failed to fetch license',
);
}
} catch (e) {
throw _handleError(e);
}
@@ -128,7 +152,13 @@ class LicenseRemoteDataSourceImpl implements LicenseRemoteDataSource {
data: request.toJson(),
);
return LicenseDto.fromJson(response.data);
if (response.data != null && response.data['success'] == true && response.data['data'] != null) {
return LicenseDto.fromJson(response.data['data']);
} else {
throw ApiException(
message: response.data?['error']?['message'] ?? 'Failed to fetch license',
);
}
} catch (e) {
throw _handleError(e);
}
@@ -141,7 +171,13 @@ class LicenseRemoteDataSourceImpl implements LicenseRemoteDataSource {
'${ApiEndpoints.licenses}/$id/unassign',
);
return LicenseDto.fromJson(response.data);
if (response.data != null && response.data['success'] == true && response.data['data'] != null) {
return LicenseDto.fromJson(response.data['data']);
} else {
throw ApiException(
message: response.data?['error']?['message'] ?? 'Failed to fetch license',
);
}
} catch (e) {
throw _handleError(e);
}
@@ -165,7 +201,13 @@ class LicenseRemoteDataSourceImpl implements LicenseRemoteDataSource {
queryParameters: queryParams,
);
return ExpiringLicenseListDto.fromJson(response.data);
if (response.data != null && response.data['success'] == true && response.data['data'] != null) {
return ExpiringLicenseListDto.fromJson(response.data['data']);
} else {
throw ApiException(
message: response.data?['error']?['message'] ?? 'Failed to fetch expiring licenses',
);
}
} catch (e) {
throw _handleError(e);
}