feat: 결재·마스터 실연동 업데이트
This commit is contained in:
@@ -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'),
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
@@ -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')),
|
||||
|
||||
@@ -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);
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
@@ -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(
|
||||
|
||||
Reference in New Issue
Block a user