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

@@ -7,11 +7,15 @@ import 'package:shadcn_ui/shadcn_ui.dart';
import 'package:superport_v2/core/common/models/paginated_result.dart';
import 'package:superport_v2/core/config/environment.dart';
import 'package:superport_v2/core/permissions/permission_manager.dart';
import 'package:superport_v2/core/permissions/permission_resources.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/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';
import '../../helpers/test_app.dart';
@@ -42,12 +46,14 @@ void main() {
testWidgets('결재 단계 액션은 승인 권한이 없으면 비활성화된다', (tester) async {
final repo = _StubApprovalRepository();
final templateRepo = _StubApprovalTemplateRepository();
final lookupRepo = _StubInventoryLookupRepository();
GetIt.I.registerSingleton<ApprovalRepository>(repo);
GetIt.I.registerSingleton<ApprovalTemplateRepository>(templateRepo);
GetIt.I.registerSingleton<InventoryLookupRepository>(lookupRepo);
final permissionManager = PermissionManager(
overrides: {
'/approvals/requests': {PermissionAction.view},
PermissionResources.approvals: {PermissionAction.view},
},
);
@@ -83,12 +89,14 @@ void main() {
testWidgets('승인 권한이 있으면 단계 액션을 실행할 수 있다', (tester) async {
final repo = _StubApprovalRepository();
final templateRepo = _StubApprovalTemplateRepository();
final lookupRepo = _StubInventoryLookupRepository();
GetIt.I.registerSingleton<ApprovalRepository>(repo);
GetIt.I.registerSingleton<ApprovalTemplateRepository>(templateRepo);
GetIt.I.registerSingleton<InventoryLookupRepository>(lookupRepo);
final permissionManager = PermissionManager(
overrides: {
'/approvals/requests': {
PermissionResources.approvals: {
PermissionAction.view,
PermissionAction.approve,
},
@@ -115,6 +123,44 @@ void main() {
expect(approveButton.onPressed, isNotNull);
expect(find.text('결재 권한이 없어 단계 행위를 실행할 수 없습니다.'), findsNothing);
});
testWidgets('canProceed가 false면 단계 버튼을 비활성화하고 이유를 안내한다', (tester) async {
final repo = _BlockingApprovalRepository();
final templateRepo = _StubApprovalTemplateRepository();
final lookupRepo = _StubInventoryLookupRepository();
GetIt.I.registerSingleton<ApprovalRepository>(repo);
GetIt.I.registerSingleton<ApprovalTemplateRepository>(templateRepo);
GetIt.I.registerSingleton<InventoryLookupRepository>(lookupRepo);
final permissionManager = PermissionManager(
overrides: {
PermissionResources.approvals: {
PermissionAction.view,
PermissionAction.approve,
},
},
);
await pumpApprovalPage(tester, permissionManager);
final rowFinder = find.byKey(const ValueKey('approval_row_1'));
expect(rowFinder, findsOneWidget);
await tester.tap(rowFinder);
await tester.pumpAndSettle();
final tabContext = tester.element(find.byType(TabBar));
final tabController = DefaultTabController.of(tabContext);
tabController.animateTo(1);
await tester.pumpAndSettle();
expect(find.textContaining('선행 단계가 완료되지 않았습니다.'), findsWidgets);
final approveButton = tester.widget<ShadButton>(
find.byKey(const ValueKey('step_action_100_approve')),
);
expect(approveButton.onPressed, isNull);
});
}
class _StubApprovalRepository implements ApprovalRepository {
@@ -194,6 +240,11 @@ class _StubApprovalRepository implements ApprovalRepository {
return _approval;
}
@override
Future<ApprovalProceedStatus> canProceed(int id) async {
return ApprovalProceedStatus(approvalId: id, canProceed: true);
}
@override
Future<Approval> create(ApprovalInput input) {
throw UnimplementedError();
@@ -215,6 +266,17 @@ class _StubApprovalRepository implements ApprovalRepository {
}
}
class _BlockingApprovalRepository extends _StubApprovalRepository {
@override
Future<ApprovalProceedStatus> canProceed(int id) async {
return ApprovalProceedStatus(
approvalId: id,
canProceed: false,
reason: '선행 단계가 완료되지 않았습니다. 관리자에게 문의하세요.',
);
}
}
class _StubApprovalTemplateRepository implements ApprovalTemplateRepository {
_StubApprovalTemplateRepository();
@@ -285,3 +347,45 @@ class _StubApprovalTemplateRepository implements ApprovalTemplateRepository {
throw UnimplementedError();
}
}
class _StubInventoryLookupRepository implements InventoryLookupRepository {
@override
Future<List<LookupItem>> fetchTransactionTypes({
bool activeOnly = true,
}) async {
return <LookupItem>[
LookupItem(id: 1, name: '입고', code: 'INBOUND'),
LookupItem(id: 2, name: '출고', code: 'OUTBOUND'),
];
}
@override
Future<List<LookupItem>> fetchTransactionStatuses({
bool activeOnly = true,
}) async {
return <LookupItem>[
LookupItem(id: 10, name: '승인대기', code: 'pending'),
LookupItem(id: 11, name: '진행중', code: 'in_progress'),
];
}
@override
Future<List<LookupItem>> fetchApprovalStatuses({
bool activeOnly = true,
}) async {
return <LookupItem>[
LookupItem(id: 20, name: '승인대기', code: 'pending'),
LookupItem(id: 21, name: '진행중', code: 'in_progress'),
];
}
@override
Future<List<LookupItem>> fetchApprovalActions({
bool activeOnly = true,
}) async {
return <LookupItem>[
LookupItem(id: 30, name: '승인', code: 'approve'),
LookupItem(id: 31, name: '반려', code: 'reject'),
];
}
}

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);
});
});
}

View File

@@ -7,6 +7,7 @@ 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/core/permissions/permission_resources.dart';
import 'package:superport_v2/features/approvals/domain/entities/approval.dart';
import 'package:superport_v2/features/approvals/step/domain/entities/approval_step_input.dart';
import 'package:superport_v2/features/approvals/step/domain/entities/approval_step_record.dart';
@@ -18,7 +19,9 @@ class _MockApprovalStepRepository extends Mock
Widget _buildApp(Widget child) {
final manager = PermissionManager(
overrides: {'/approvals/steps': PermissionAction.values.toSet()},
overrides: {
PermissionResources.approvalSteps: PermissionAction.values.toSet(),
},
);
return MaterialApp(
home: PermissionScope(