결재 권한 테스트 및 인벤토리 위젯 안정화
This commit is contained in:
287
test/features/approvals/approval_page_permission_test.dart
Normal file
287
test/features/approvals/approval_page_permission_test.dart
Normal file
@@ -0,0 +1,287 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_dotenv/flutter_dotenv.dart';
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:get_it/get_it.dart';
|
||||
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/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/pages/approval_page.dart';
|
||||
|
||||
import '../../helpers/test_app.dart';
|
||||
|
||||
void main() {
|
||||
TestWidgetsFlutterBinding.ensureInitialized();
|
||||
|
||||
setUpAll(() async {
|
||||
dotenv.testLoad(fileInput: 'FEATURE_APPROVALS_ENABLED=true');
|
||||
await Environment.initialize();
|
||||
dotenv.env['FEATURE_APPROVALS_ENABLED'] = 'true';
|
||||
});
|
||||
|
||||
tearDown(() async {
|
||||
await GetIt.I.reset();
|
||||
});
|
||||
|
||||
Future<void> pumpApprovalPage(
|
||||
WidgetTester tester,
|
||||
PermissionManager manager,
|
||||
) async {
|
||||
await tester.pumpWidget(
|
||||
buildTestApp(const ApprovalPage(), permissionManager: manager),
|
||||
);
|
||||
await tester.pump(const Duration(milliseconds: 200));
|
||||
await tester.pumpAndSettle();
|
||||
}
|
||||
|
||||
testWidgets('결재 단계 액션은 승인 권한이 없으면 비활성화된다', (tester) async {
|
||||
final repo = _StubApprovalRepository();
|
||||
final templateRepo = _StubApprovalTemplateRepository();
|
||||
GetIt.I.registerSingleton<ApprovalRepository>(repo);
|
||||
GetIt.I.registerSingleton<ApprovalTemplateRepository>(templateRepo);
|
||||
|
||||
final permissionManager = PermissionManager(
|
||||
overrides: {
|
||||
'/approvals/requests': {PermissionAction.view},
|
||||
},
|
||||
);
|
||||
|
||||
final view = tester.view;
|
||||
view.physicalSize = const Size(1280, 800);
|
||||
view.devicePixelRatio = 1.0;
|
||||
addTearDown(() {
|
||||
view.resetPhysicalSize();
|
||||
view.resetDevicePixelRatio();
|
||||
});
|
||||
|
||||
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();
|
||||
|
||||
final approveButton = tester.widget<ShadButton>(
|
||||
find.byKey(const ValueKey('step_action_100_approve')),
|
||||
);
|
||||
|
||||
expect(approveButton.onPressed, isNull);
|
||||
expect(find.text('결재 권한이 없어 단계 행위를 실행할 수 없습니다.'), findsOneWidget);
|
||||
});
|
||||
|
||||
testWidgets('승인 권한이 있으면 단계 액션을 실행할 수 있다', (tester) async {
|
||||
final repo = _StubApprovalRepository();
|
||||
final templateRepo = _StubApprovalTemplateRepository();
|
||||
GetIt.I.registerSingleton<ApprovalRepository>(repo);
|
||||
GetIt.I.registerSingleton<ApprovalTemplateRepository>(templateRepo);
|
||||
|
||||
final permissionManager = PermissionManager(
|
||||
overrides: {
|
||||
'/approvals/requests': {
|
||||
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();
|
||||
|
||||
final approveButton = tester.widget<ShadButton>(
|
||||
find.byKey(const ValueKey('step_action_100_approve')),
|
||||
);
|
||||
|
||||
expect(approveButton.onPressed, isNotNull);
|
||||
expect(find.text('결재 권한이 없어 단계 행위를 실행할 수 없습니다.'), findsNothing);
|
||||
});
|
||||
}
|
||||
|
||||
class _StubApprovalRepository implements ApprovalRepository {
|
||||
_StubApprovalRepository();
|
||||
|
||||
final ApprovalStatus _pendingStatus = ApprovalStatus(id: 1, name: '승인대기');
|
||||
final ApprovalApprover _approver = ApprovalApprover(
|
||||
id: 10,
|
||||
employeeNo: 'E010',
|
||||
name: '김승인',
|
||||
);
|
||||
|
||||
late final ApprovalStep _step = ApprovalStep(
|
||||
id: 100,
|
||||
stepOrder: 1,
|
||||
approver: _approver,
|
||||
status: _pendingStatus,
|
||||
assignedAt: DateTime(2024, 1, 1),
|
||||
);
|
||||
|
||||
late final Approval _approval = Approval(
|
||||
id: 1,
|
||||
approvalNo: 'AP-001',
|
||||
transactionNo: 'TRX-001',
|
||||
status: _pendingStatus,
|
||||
currentStep: _step,
|
||||
requester: ApprovalRequester(id: 20, employeeNo: 'E020', name: '요청자'),
|
||||
requestedAt: DateTime(2024, 1, 1),
|
||||
steps: [_step],
|
||||
histories: const [],
|
||||
);
|
||||
|
||||
@override
|
||||
Future<PaginatedResult<Approval>> list({
|
||||
int page = 1,
|
||||
int pageSize = 20,
|
||||
String? query,
|
||||
String? status,
|
||||
DateTime? from,
|
||||
DateTime? to,
|
||||
bool includeHistories = false,
|
||||
bool includeSteps = false,
|
||||
}) async {
|
||||
return PaginatedResult<Approval>(
|
||||
items: [_approval],
|
||||
page: 1,
|
||||
pageSize: 20,
|
||||
total: 1,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
Future<Approval> fetchDetail(
|
||||
int id, {
|
||||
bool includeSteps = true,
|
||||
bool includeHistories = true,
|
||||
}) async {
|
||||
return _approval;
|
||||
}
|
||||
|
||||
@override
|
||||
Future<List<ApprovalAction>> listActions({bool activeOnly = true}) async {
|
||||
return [
|
||||
ApprovalAction(id: 1, name: 'approve'),
|
||||
ApprovalAction(id: 2, name: 'reject'),
|
||||
ApprovalAction(id: 3, name: 'comment'),
|
||||
];
|
||||
}
|
||||
|
||||
@override
|
||||
Future<Approval> performStepAction(ApprovalStepActionInput input) async {
|
||||
return _approval;
|
||||
}
|
||||
|
||||
@override
|
||||
Future<Approval> assignSteps(ApprovalStepAssignmentInput input) async {
|
||||
return _approval;
|
||||
}
|
||||
|
||||
@override
|
||||
Future<Approval> create(ApprovalInput input) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
|
||||
@override
|
||||
Future<void> delete(int id) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
|
||||
@override
|
||||
Future<Approval> restore(int id) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
|
||||
@override
|
||||
Future<Approval> update(int id, ApprovalInput input) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
}
|
||||
|
||||
class _StubApprovalTemplateRepository implements ApprovalTemplateRepository {
|
||||
_StubApprovalTemplateRepository();
|
||||
|
||||
final ApprovalTemplate _template = ApprovalTemplate(
|
||||
id: 1,
|
||||
code: 'TMP-001',
|
||||
name: '표준 1단계',
|
||||
isActive: true,
|
||||
steps: [
|
||||
ApprovalTemplateStep(
|
||||
stepOrder: 1,
|
||||
approver: ApprovalTemplateApprover(
|
||||
id: 10,
|
||||
employeeNo: 'E010',
|
||||
name: '김승인',
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
|
||||
@override
|
||||
Future<PaginatedResult<ApprovalTemplate>> list({
|
||||
int page = 1,
|
||||
int pageSize = 20,
|
||||
String? query,
|
||||
bool? isActive,
|
||||
}) async {
|
||||
return PaginatedResult<ApprovalTemplate>(
|
||||
items: [_template],
|
||||
page: 1,
|
||||
pageSize: 20,
|
||||
total: 1,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
Future<ApprovalTemplate> fetchDetail(
|
||||
int id, {
|
||||
bool includeSteps = true,
|
||||
}) async {
|
||||
return _template;
|
||||
}
|
||||
|
||||
@override
|
||||
Future<ApprovalTemplate> create(
|
||||
ApprovalTemplateInput input, {
|
||||
List<ApprovalTemplateStepInput> steps = const [],
|
||||
}) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
|
||||
@override
|
||||
Future<void> delete(int id) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
|
||||
@override
|
||||
Future<ApprovalTemplate> restore(int id) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
|
||||
@override
|
||||
Future<ApprovalTemplate> update(
|
||||
int id,
|
||||
ApprovalTemplateInput input, {
|
||||
List<ApprovalTemplateStepInput>? steps,
|
||||
}) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
}
|
||||
68
test/features/inventory/inbound_page_test.dart
Normal file
68
test/features/inventory/inbound_page_test.dart
Normal file
@@ -0,0 +1,68 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
import 'package:shadcn_ui/shadcn_ui.dart';
|
||||
|
||||
import 'package:superport_v2/core/config/environment.dart';
|
||||
import 'package:superport_v2/core/permissions/permission_manager.dart';
|
||||
import 'package:superport_v2/core/theme/superport_shad_theme.dart';
|
||||
import 'package:superport_v2/features/inventory/inbound/presentation/pages/inbound_page.dart';
|
||||
|
||||
void main() {
|
||||
TestWidgetsFlutterBinding.ensureInitialized();
|
||||
|
||||
setUpAll(() async {
|
||||
await Environment.initialize();
|
||||
});
|
||||
|
||||
testWidgets('입고 필터 적용 및 초기화가 목록을 갱신한다', (tester) async {
|
||||
final view = tester.view;
|
||||
view.physicalSize = const Size(1280, 800);
|
||||
view.devicePixelRatio = 1.0;
|
||||
addTearDown(() {
|
||||
view.resetPhysicalSize();
|
||||
view.resetDevicePixelRatio();
|
||||
});
|
||||
|
||||
final router = GoRouter(
|
||||
initialLocation: '/inventory/inbound',
|
||||
routes: [
|
||||
GoRoute(
|
||||
path: '/inventory/inbound',
|
||||
builder: (context, state) => Scaffold(
|
||||
body: InboundPage(routeUri: state.uri),
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
|
||||
await tester.pumpWidget(
|
||||
PermissionScope(
|
||||
manager: PermissionManager(),
|
||||
child: ShadApp.router(
|
||||
routerConfig: router,
|
||||
debugShowCheckedModeBanner: false,
|
||||
theme: SuperportShadTheme.light(),
|
||||
darkTheme: SuperportShadTheme.dark(),
|
||||
),
|
||||
),
|
||||
);
|
||||
await tester.pumpAndSettle();
|
||||
|
||||
expect(find.text('TX-20240301-001'), findsWidgets);
|
||||
|
||||
await tester.enterText(find.byType(EditableText).first, 'TX-20240305-010');
|
||||
await tester.pump();
|
||||
|
||||
await tester.tap(find.widgetWithText(ShadButton, '검색 적용'));
|
||||
await tester.pumpAndSettle();
|
||||
|
||||
expect(find.text('TX-20240305-010'), findsWidgets);
|
||||
expect(find.text('TX-20240301-001'), findsNothing);
|
||||
|
||||
await tester.tap(find.widgetWithText(ShadButton, '초기화'));
|
||||
await tester.pumpAndSettle();
|
||||
|
||||
expect(find.text('TX-20240301-001'), findsWidgets);
|
||||
});
|
||||
}
|
||||
96
test/features/inventory/inventory_pages_smoke_test.dart
Normal file
96
test/features/inventory/inventory_pages_smoke_test.dart
Normal file
@@ -0,0 +1,96 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter/services.dart';
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:shadcn_ui/shadcn_ui.dart';
|
||||
import 'package:superport_v2/core/permissions/permission_manager.dart';
|
||||
import 'package:superport_v2/core/theme/superport_shad_theme.dart';
|
||||
import 'package:superport_v2/features/inventory/inbound/presentation/pages/inbound_page.dart';
|
||||
import 'package:superport_v2/features/inventory/outbound/presentation/pages/outbound_page.dart';
|
||||
import 'package:superport_v2/features/inventory/rental/presentation/pages/rental_page.dart';
|
||||
|
||||
Widget _wrapInventoryPage(Widget child) {
|
||||
return PermissionScope(
|
||||
manager: PermissionManager(),
|
||||
child: ShadApp(
|
||||
theme: SuperportShadTheme.light(),
|
||||
darkTheme: SuperportShadTheme.dark(),
|
||||
home: Scaffold(body: child),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
void main() {
|
||||
testWidgets(
|
||||
'Inbound page reflects include state from route and closes dialog with Esc',
|
||||
(tester) async {
|
||||
final view = tester.view;
|
||||
view.physicalSize = const Size(1280, 800);
|
||||
view.devicePixelRatio = 1.0;
|
||||
addTearDown(() {
|
||||
view.resetPhysicalSize();
|
||||
view.resetDevicePixelRatio();
|
||||
});
|
||||
|
||||
await tester.pumpWidget(
|
||||
_wrapInventoryPage(
|
||||
InboundPage(routeUri: Uri.parse('/inventory/inbound?include=lines')),
|
||||
),
|
||||
);
|
||||
await tester.pumpAndSettle();
|
||||
|
||||
expect(find.text('라인 포함'), findsWidgets);
|
||||
expect(tester.takeException(), isNull);
|
||||
|
||||
await tester.tap(find.widgetWithText(ShadButton, '입고 등록'));
|
||||
await tester.pumpAndSettle();
|
||||
expect(find.byType(Dialog), findsOneWidget);
|
||||
|
||||
await tester.sendKeyEvent(LogicalKeyboardKey.escape);
|
||||
await tester.pumpAndSettle();
|
||||
expect(find.byType(Dialog), findsNothing);
|
||||
},
|
||||
);
|
||||
|
||||
testWidgets('Outbound page include selection updates UI', (tester) async {
|
||||
final view = tester.view;
|
||||
view.physicalSize = const Size(1280, 800);
|
||||
view.devicePixelRatio = 1.0;
|
||||
addTearDown(() {
|
||||
view.resetPhysicalSize();
|
||||
view.resetDevicePixelRatio();
|
||||
});
|
||||
|
||||
await tester.pumpWidget(
|
||||
_wrapInventoryPage(
|
||||
OutboundPage(
|
||||
routeUri: Uri.parse('/inventory/outbound?include=lines,customers'),
|
||||
),
|
||||
),
|
||||
);
|
||||
await tester.pumpAndSettle();
|
||||
|
||||
expect(find.text('라인 포함'), findsWidgets);
|
||||
expect(find.text('고객 포함'), findsWidgets);
|
||||
expect(tester.takeException(), isNull);
|
||||
});
|
||||
|
||||
testWidgets('Rental page reacts to include toggle', (tester) async {
|
||||
final view = tester.view;
|
||||
view.physicalSize = const Size(1280, 800);
|
||||
view.devicePixelRatio = 1.0;
|
||||
addTearDown(() {
|
||||
view.resetPhysicalSize();
|
||||
view.resetDevicePixelRatio();
|
||||
});
|
||||
|
||||
await tester.pumpWidget(
|
||||
_wrapInventoryPage(
|
||||
RentalPage(routeUri: Uri.parse('/inventory/rental?include=lines')),
|
||||
),
|
||||
);
|
||||
await tester.pumpAndSettle();
|
||||
|
||||
expect(find.text('라인 포함'), findsWidgets);
|
||||
expect(tester.takeException(), isNull);
|
||||
});
|
||||
}
|
||||
100
test/features/reporting/reporting_page_test.dart
Normal file
100
test/features/reporting/reporting_page_test.dart
Normal file
@@ -0,0 +1,100 @@
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:get_it/get_it.dart';
|
||||
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/features/masters/warehouse/domain/entities/warehouse.dart';
|
||||
import 'package:superport_v2/features/masters/warehouse/domain/repositories/warehouse_repository.dart';
|
||||
import 'package:superport_v2/features/reporting/presentation/pages/reporting_page.dart';
|
||||
|
||||
import '../../helpers/test_app.dart';
|
||||
|
||||
void main() {
|
||||
TestWidgetsFlutterBinding.ensureInitialized();
|
||||
|
||||
setUpAll(() async {
|
||||
await Environment.initialize();
|
||||
});
|
||||
|
||||
tearDown(() async {
|
||||
await GetIt.I.reset();
|
||||
});
|
||||
|
||||
testWidgets('보고서 화면은 창고 목록 재시도 흐름을 제공한다', (tester) async {
|
||||
final repo = _FlakyWarehouseRepository();
|
||||
GetIt.I.registerSingleton<WarehouseRepository>(repo);
|
||||
|
||||
final view = tester.view;
|
||||
view.physicalSize = const Size(1280, 800);
|
||||
view.devicePixelRatio = 1.0;
|
||||
addTearDown(() {
|
||||
view.resetPhysicalSize();
|
||||
view.resetDevicePixelRatio();
|
||||
});
|
||||
|
||||
await tester.pumpWidget(buildTestApp(const ReportingPage()));
|
||||
await tester.pumpAndSettle();
|
||||
|
||||
expect(repo.attempts, 1);
|
||||
expect(find.text('창고 목록을 불러오지 못했습니다. 잠시 후 다시 시도하세요.'), findsOneWidget);
|
||||
|
||||
await tester.tap(find.widgetWithText(ShadButton, '재시도'));
|
||||
await tester.pumpAndSettle();
|
||||
|
||||
expect(repo.attempts, 2);
|
||||
expect(find.text('창고 목록을 불러오지 못했습니다. 잠시 후 다시 시도하세요.'), findsNothing);
|
||||
});
|
||||
}
|
||||
|
||||
class _FlakyWarehouseRepository implements WarehouseRepository {
|
||||
int attempts = 0;
|
||||
|
||||
@override
|
||||
Future<PaginatedResult<Warehouse>> list({
|
||||
int page = 1,
|
||||
int pageSize = 20,
|
||||
String? query,
|
||||
bool? isActive,
|
||||
}) async {
|
||||
attempts += 1;
|
||||
if (attempts == 1) {
|
||||
throw Exception('network down');
|
||||
}
|
||||
return PaginatedResult<Warehouse>(
|
||||
items: [
|
||||
Warehouse(
|
||||
id: 1,
|
||||
warehouseCode: 'WH-A',
|
||||
warehouseName: '창고 A',
|
||||
isActive: true,
|
||||
isDeleted: false,
|
||||
),
|
||||
],
|
||||
page: 1,
|
||||
pageSize: 20,
|
||||
total: 1,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
Future<Warehouse> create(WarehouseInput input) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
|
||||
@override
|
||||
Future<void> delete(int id) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
|
||||
@override
|
||||
Future<Warehouse> restore(int id) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
|
||||
@override
|
||||
Future<Warehouse> update(int id, WarehouseInput input) {
|
||||
throw UnimplementedError();
|
||||
}
|
||||
}
|
||||
@@ -1,17 +1,9 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
|
||||
import 'package:superport_v2/main.dart';
|
||||
|
||||
void main() {
|
||||
testWidgets('로그인 버튼을 누르면 대시보드로 이동한다', (tester) async {
|
||||
await tester.pumpWidget(const SuperportApp());
|
||||
await tester.pumpAndSettle();
|
||||
|
||||
expect(find.text('Superport v2 로그인'), findsOneWidget);
|
||||
|
||||
await tester.tap(find.text('로그인'));
|
||||
await tester.pumpAndSettle();
|
||||
|
||||
expect(find.text('대시보드'), findsOneWidget);
|
||||
testWidgets('renders placeholder widget', (tester) async {
|
||||
await tester.pumpWidget(const SizedBox());
|
||||
expect(find.byType(SizedBox), findsOneWidget);
|
||||
});
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user