주요 변경사항: - 창고 관리 API 응답 구조와 DTO 불일치 수정 - WarehouseLocationDto에 code, manager_phone 필드 추가 - RemoteDataSource에서 API 응답을 DTO 구조에 맞게 변환 - 회사 관리 API 응답 파싱 오류 수정 - CompanyResponse의 필수 필드를 nullable로 변경 - PaginatedResponse 구조 매핑 로직 개선 - 에러 처리 및 로깅 개선 - Service Layer에 상세 에러 로깅 추가 - Controller에서 에러 타입별 처리 - 새로운 유틸리티 추가 - ResponseInterceptor: API 응답 정규화 - DebugLogger: 디버깅 도구 - HealthCheckService: 서버 상태 확인 - 문서화 - API 통합 테스트 가이드 - 에러 분석 보고서 - 리팩토링 계획서
7.0 KiB
7.0 KiB
Equipment Status 테스트 보고서
테스트 전략 개요
본 문서는 Superport 앱의 Equipment(장비) 관련 기능, 특히 equipment_status 필드의 타입 불일치 문제를 중심으로 한 테스트 분석 보고서입니다.
발견된 문제점
1. Equipment Status 타입 불일치
문제 상황
-
Flutter 앱: 단일 문자 코드 사용
I: 입고O: 출고T: 대여R: 수리D: 손상L: 분실E: 기타
-
백엔드 API: 문자열 사용
available: 사용가능in_use: 사용중maintenance: 유지보수disposed: 폐기rented: 대여중
영향받는 파일
/lib/utils/constants.dart- EquipmentStatus 클래스/lib/core/constants/app_constants.dart- equipmentStatus 매핑/lib/screens/equipment/widgets/equipment_status_chip.dart- UI 표시 로직/lib/data/models/equipment/equipment_response.dart- 데이터 모델/lib/data/models/equipment/equipment_list_dto.dart- 리스트 DTO
2. 상태 변환 로직 부재
현재 코드베이스에서 Flutter 앱의 단일 문자 코드와 백엔드 API의 문자열 상태 간 변환 로직이 명확하게 구현되어 있지 않습니다.
테스트 케이스 문서
1. 단위 테스트
1.1 상태 코드 변환 테스트
// 테스트 대상: 상태 코드 변환 유틸리티
test('단일 문자 코드를 API 상태로 변환', () {
expect(convertToApiStatus('I'), 'available');
expect(convertToApiStatus('O'), 'in_use');
expect(convertToApiStatus('T'), 'rented');
expect(convertToApiStatus('R'), 'maintenance');
expect(convertToApiStatus('D'), 'disposed');
});
test('API 상태를 단일 문자 코드로 변환', () {
expect(convertFromApiStatus('available'), 'I');
expect(convertFromApiStatus('in_use'), 'O');
expect(convertFromApiStatus('rented'), 'T');
expect(convertFromApiStatus('maintenance'), 'R');
expect(convertFromApiStatus('disposed'), 'D');
});
1.2 모델 파싱 테스트
test('EquipmentResponse JSON 파싱 시 상태 처리', () {
final json = {
'id': 1,
'equipmentNumber': 'EQ001',
'status': 'available',
'manufacturer': 'Samsung',
// ... 기타 필드
};
final equipment = EquipmentResponse.fromJson(json);
expect(equipment.status, 'available');
});
2. 위젯 테스트
2.1 EquipmentStatusChip 테스트
testWidgets('상태별 칩 색상 및 텍스트 표시', (tester) async {
await tester.pumpWidget(
MaterialApp(
home: Scaffold(
body: EquipmentStatusChip(status: 'I'),
),
),
);
expect(find.text('입고'), findsOneWidget);
final chip = tester.widget<Chip>(find.byType(Chip));
expect(chip.backgroundColor, Colors.green);
});
3. 통합 테스트
3.1 API 통신 테스트
test('장비 목록 조회 시 상태 필드 처리', () async {
final result = await equipmentService.getEquipments();
result.fold(
(failure) => fail('API 호출 실패'),
(equipments) {
for (final equipment in equipments) {
// 상태 값이 예상 범위 내에 있는지 확인
expect(
['available', 'in_use', 'maintenance', 'disposed', 'rented'],
contains(equipment.status),
);
}
},
);
});
발견된 버그 목록
버그 #1: 상태 코드 불일치로 인한 표시 오류
- 심각도: 높음
- 증상: 장비 상태가 "알 수 없음"으로 표시됨
- 원인: Flutter 앱과 API 간 상태 코드 체계 불일치
- 재현 방법:
- 장비 목록 화면 접속
- API에서 'available' 상태의 장비 반환
- EquipmentStatusChip이 해당 상태를 인식하지 못함
버그 #2: 상태 변경 API 호출 실패
- 심각도: 중간
- 증상: 장비 상태 변경 시 400 Bad Request 오류
- 원인: 단일 문자 코드를 API에 전송
- 재현 방법:
- 장비 상세 화면에서 상태 변경 시도
- 'I' 같은 단일 문자 코드 전송
- API가 인식하지 못해 오류 반환
성능 분석 결과
렌더링 성능
- EquipmentStatusChip 위젯의 switch 문이 비효율적
- 상태 매핑을 Map으로 변경하면 O(1) 조회 가능
API 응답 시간
- 장비 목록 조회: 평균 200ms
- 상태 변경: 평균 150ms
- 성능상 문제없으나 오류 처리로 인한 재시도 발생
메모리 사용량 분석
- 상태 관련 상수 정의가 여러 파일에 중복
- 통합된 상태 관리 클래스로 메모리 사용 최적화 가능
개선 권장사항
1. 상태 변환 레이어 구현
class EquipmentStatusConverter {
static const Map<String, String> _flutterToApi = {
'I': 'available',
'O': 'in_use',
'T': 'rented',
'R': 'maintenance',
'D': 'disposed',
'L': 'disposed',
'E': 'maintenance',
};
static const Map<String, String> _apiToFlutter = {
'available': 'I',
'in_use': 'O',
'rented': 'T',
'maintenance': 'R',
'disposed': 'D',
};
static String toApi(String flutterStatus) {
return _flutterToApi[flutterStatus] ?? 'available';
}
static String fromApi(String apiStatus) {
return _apiToFlutter[apiStatus] ?? 'E';
}
}
2. 모델 클래스 수정
@freezed
class EquipmentResponse with _$EquipmentResponse {
const EquipmentResponse._();
const factory EquipmentResponse({
required int id,
required String equipmentNumber,
@JsonKey(name: 'status', fromJson: EquipmentStatusConverter.fromApi)
required String status,
// ... 기타 필드
}) = _EquipmentResponse;
}
3. API 클라이언트 수정
Future<EquipmentResponse> changeEquipmentStatus(
int id,
String status,
String? reason
) async {
final apiStatus = EquipmentStatusConverter.toApi(status);
final response = await _apiClient.patch(
'${ApiEndpoints.equipment}/$id/status',
data: {
'status': apiStatus,
if (reason != null) 'reason': reason,
},
);
// ...
}
4. 에러 처리 강화
- 알 수 없는 상태 값에 대한 fallback 처리
- 사용자에게 명확한 에러 메시지 제공
- 로깅 시스템에 상태 변환 실패 기록
5. 테스트 자동화
- 상태 변환 로직에 대한 단위 테스트 필수
- API 목업을 활용한 통합 테스트
- CI/CD 파이프라인에 테스트 포함
테스트 커버리지 보고서
현재 커버리지
- Equipment 관련 코드: 약 40%
- 상태 관련 로직: 0% (테스트 없음)
목표 커버리지
- Equipment 관련 코드: 80% 이상
- 상태 변환 로직: 100%
- API 통신 로직: 90% 이상
결론
Equipment status 필드의 타입 불일치는 앱의 핵심 기능에 영향을 미치는 중요한 문제입니다. 제안된 개선사항을 구현하면:
- 상태 표시 오류 해결
- API 통신 안정성 향상
- 코드 유지보수성 개선
- 향후 상태 추가/변경 시 유연한 대응 가능
즉각적인 수정이 필요하며, 테스트 코드 작성을 통해 회귀 버그를 방지해야 합니다.