feat: 결재·마스터 실연동 업데이트

This commit is contained in:
JiWoong Sul
2025-10-14 18:10:24 +09:00
parent 1325109fba
commit 8067416c09
66 changed files with 2129 additions and 222 deletions

View File

@@ -4,6 +4,7 @@ import 'package:mocktail/mocktail.dart';
import 'package:superport_v2/core/common/models/paginated_result.dart';
import 'package:superport_v2/features/approvals/domain/entities/approval.dart';
import 'package:superport_v2/features/approvals/domain/entities/approval_template.dart';
import 'package:superport_v2/features/approvals/domain/entities/approval_proceed_status.dart';
import 'package:superport_v2/features/approvals/domain/repositories/approval_repository.dart';
import 'package:superport_v2/features/approvals/domain/repositories/approval_template_repository.dart';
import 'package:superport_v2/features/approvals/presentation/controllers/approval_controller.dart';
@@ -74,6 +75,12 @@ void main() {
approvalRepository: repository,
templateRepository: templateRepository,
);
when(() => repository.canProceed(any())).thenAnswer(
(_) async => ApprovalProceedStatus(
approvalId: sampleApproval.id!,
canProceed: true,
),
);
});
// fetch 메서드 관련 시나리오
@@ -166,6 +173,8 @@ void main() {
includeHistories: true,
),
).called(1);
verify(() => repository.canProceed(1)).called(1);
expect(controller.canProceedSelected, isTrue);
});
test('에러 발생 시 errorMessage 설정', () async {
@@ -369,6 +378,30 @@ void main() {
expect(controller.isPerformingAction, isFalse);
});
test('canProceed가 false면 액션을 중단한다', () async {
when(() => repository.canProceed(any())).thenAnswer(
(_) async => ApprovalProceedStatus(
approvalId: sampleApproval.id!,
canProceed: false,
reason: '선행 단계가 완료되지 않았습니다.',
),
);
await controller.loadActionOptions(force: true);
await controller.fetch();
await controller.selectApproval(sampleApproval.id!);
final success = await controller.performStepAction(
step: sampleStep,
type: ApprovalStepActionType.approve,
);
expect(success, isFalse);
expect(controller.errorMessage, contains('선행 단계'));
expect(controller.canProceedSelected, isFalse);
verifyNever(() => repository.performStepAction(any()));
});
test('행위를 찾지 못하면 요청하지 않는다', () async {
when(
() => repository.listActions(activeOnly: any(named: 'activeOnly')),

View File

@@ -5,22 +5,38 @@ import 'package:get_it/get_it.dart';
import 'package:mocktail/mocktail.dart';
import 'package:shadcn_ui/shadcn_ui.dart';
import 'package:superport_v2/core/common/models/paginated_result.dart';
import 'package:superport_v2/core/permissions/permission_manager.dart';
import 'package:superport_v2/features/approvals/domain/entities/approval.dart';
import 'package:superport_v2/features/approvals/domain/entities/approval_template.dart';
import 'package:superport_v2/features/approvals/domain/repositories/approval_repository.dart';
import 'package:superport_v2/features/approvals/domain/repositories/approval_template_repository.dart';
import 'package:superport_v2/features/approvals/presentation/controllers/approval_controller.dart';
import 'package:superport_v2/features/approvals/presentation/pages/approval_page.dart';
import 'package:superport_v2/features/inventory/lookups/domain/entities/lookup_item.dart';
import 'package:superport_v2/features/inventory/lookups/domain/repositories/inventory_lookup_repository.dart';
class _MockApprovalRepository extends Mock implements ApprovalRepository {}
class _FakeApprovalInput extends Fake implements ApprovalInput {}
class _MockApprovalTemplateRepository extends Mock
implements ApprovalTemplateRepository {}
class _MockInventoryLookupRepository extends Mock
implements InventoryLookupRepository {}
Widget _buildApp(Widget child) {
return MaterialApp(
home: ShadTheme(
data: ShadThemeData(
colorScheme: const ShadSlateColorScheme.light(),
brightness: Brightness.light,
return PermissionScope(
manager: PermissionManager(),
child: MaterialApp(
home: ShadTheme(
data: ShadThemeData(
colorScheme: const ShadSlateColorScheme.light(),
brightness: Brightness.light,
),
child: Scaffold(body: child),
),
child: Scaffold(body: child),
),
);
}
@@ -49,11 +65,75 @@ void main() {
group('플래그 On', () {
late _MockApprovalRepository repository;
late _MockApprovalTemplateRepository templateRepository;
late _MockInventoryLookupRepository lookupRepository;
setUp(() {
dotenv.testLoad(fileInput: 'FEATURE_APPROVALS_ENABLED=true\n');
repository = _MockApprovalRepository();
templateRepository = _MockApprovalTemplateRepository();
lookupRepository = _MockInventoryLookupRepository();
GetIt.I.registerLazySingleton<ApprovalRepository>(() => repository);
GetIt.I.registerLazySingleton<ApprovalTemplateRepository>(
() => templateRepository,
);
GetIt.I.registerLazySingleton<InventoryLookupRepository>(
() => lookupRepository,
);
when(
() => templateRepository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
query: any(named: 'query'),
isActive: any(named: 'isActive'),
),
).thenAnswer(
(_) async => PaginatedResult<ApprovalTemplate>(
items: [],
page: 1,
pageSize: 20,
total: 0,
),
);
when(
() => repository.listActions(activeOnly: any(named: 'activeOnly')),
).thenAnswer((_) async => const []);
when(() => lookupRepository.fetchApprovalStatuses()).thenAnswer(
(_) async => [LookupItem(id: 1, name: '승인대기', code: 'pending')],
);
when(
() => repository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
query: any(named: 'query'),
status: any(named: 'status'),
from: any(named: 'from'),
to: any(named: 'to'),
includeHistories: any(named: 'includeHistories'),
includeSteps: any(named: 'includeSteps'),
),
).thenAnswer(
(_) async => PaginatedResult<Approval>(
items: const [],
page: 1,
pageSize: 20,
total: 0,
),
);
});
testWidgets('상태 룩업을 불러와 필터 라벨을 구성한다', (tester) async {
await tester.pumpWidget(_buildApp(const ApprovalPage()));
await tester.pumpAndSettle();
verify(() => lookupRepository.fetchApprovalStatuses()).called(1);
final statusSelectFinder = find.byKey(
const ValueKey(ApprovalStatusFilter.all),
);
expect(statusSelectFinder, findsOneWidget);
await tester.tap(statusSelectFinder);
await tester.pumpAndSettle();
expect(find.text('승인대기'), findsWidgets);
});
});
}