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

@@ -0,0 +1,119 @@
import 'package:dio/dio.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:mocktail/mocktail.dart';
import 'package:superport_v2/core/network/api_client.dart';
import 'package:superport_v2/features/masters/customer/data/repositories/customer_repository_remote.dart';
class _MockApiClient extends Mock implements ApiClient {}
void main() {
late ApiClient apiClient;
late CustomerRepositoryRemote repository;
setUpAll(() {
registerFallbackValue(Options());
registerFallbackValue(CancelToken());
});
setUp(() {
apiClient = _MockApiClient();
repository = CustomerRepositoryRemote(apiClient: apiClient);
});
test('list 호출 시 필터를 쿼리에 포함한다', () async {
when(
() => apiClient.get<Map<String, dynamic>>(
any(),
query: any(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
).thenAnswer(
(_) async => Response<Map<String, dynamic>>(
data: {
'items': [
{'id': 1, 'customer_code': 'C-001', 'customer_name': '슈퍼포트'},
],
'page': 2,
'page_size': 30,
'total': 1,
},
requestOptions: RequestOptions(path: '/api/v1/customers'),
statusCode: 200,
),
);
final result = await repository.list(
page: 2,
pageSize: 30,
query: 'sup',
isPartner: true,
isGeneral: false,
isActive: true,
);
expect(result.items, isNotEmpty);
final verification = verify(
() => apiClient.get<Map<String, dynamic>>(
captureAny(),
query: captureAny(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
);
final path = verification.captured[0] as String;
final query = verification.captured[1] as Map<String, dynamic>;
expect(path, equals('/api/v1/customers'));
expect(query['page'], 2);
expect(query['page_size'], 30);
expect(query['q'], 'sup');
expect(query['is_partner'], true);
expect(query['is_general'], false);
expect(query['is_active'], true);
});
test('fetchDetail은 include=zipcode 파라미터를 전달한다', () async {
when(
() => apiClient.get<Map<String, dynamic>>(
any(),
query: any(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
).thenAnswer(
(_) async => Response<Map<String, dynamic>>(
data: {
'data': {
'id': 10,
'customer_code': 'C-010',
'customer_name': '테스트 고객',
'zipcode': {'zipcode': '06000'},
},
},
requestOptions: RequestOptions(path: '/api/v1/customers/10'),
statusCode: 200,
),
);
final customer = await repository.fetchDetail(10);
expect(customer.customerCode, 'C-010');
final verification = verify(
() => apiClient.get<Map<String, dynamic>>(
captureAny(),
query: captureAny(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
);
final path = verification.captured[0] as String;
final query = verification.captured[1] as Map<String, dynamic>;
expect(path, equals('/api/v1/customers/10'));
expect(query['include'], 'zipcode');
});
}

View File

@@ -0,0 +1,65 @@
import 'package:dio/dio.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:mocktail/mocktail.dart';
import 'package:superport_v2/core/network/api_client.dart';
import 'package:superport_v2/features/masters/group/data/repositories/group_repository_remote.dart';
class _MockApiClient extends Mock implements ApiClient {}
void main() {
late ApiClient apiClient;
late GroupRepositoryRemote repository;
setUpAll(() {
registerFallbackValue(Options());
registerFallbackValue(CancelToken());
});
setUp(() {
apiClient = _MockApiClient();
repository = GroupRepositoryRemote(apiClient: apiClient);
});
test('include 옵션이 쿼리 파라미터에 반영된다', () async {
when(
() => apiClient.get<Map<String, dynamic>>(
any(),
query: any(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
).thenAnswer(
(_) async => Response<Map<String, dynamic>>(
data: {
'items': [
{
'id': 1,
'group_name': '관리자',
'is_default': true,
'is_active': true,
},
],
},
requestOptions: RequestOptions(path: '/api/v1/groups'),
statusCode: 200,
),
);
await repository.list(includePermissions: true, includeEmployees: true);
final captured = verify(
() => apiClient.get<Map<String, dynamic>>(
captureAny(),
query: captureAny(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
).captured;
final path = captured[0] as String;
final query = captured[1] as Map<String, dynamic>;
expect(path, equals('/api/v1/groups'));
expect(query['include'], 'permissions,employees');
});
}

View File

@@ -2,12 +2,18 @@ import 'package:flutter_test/flutter_test.dart';
import 'package:mocktail/mocktail.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/masters/group/domain/entities/group.dart';
import 'package:superport_v2/features/masters/group/domain/repositories/group_repository.dart';
import 'package:superport_v2/features/masters/group/presentation/controllers/group_controller.dart';
import 'package:superport_v2/features/masters/group_permission/domain/entities/group_permission.dart';
import 'package:superport_v2/features/masters/group_permission/domain/repositories/group_permission_repository.dart';
class _MockGroupRepository extends Mock implements GroupRepository {}
class _MockGroupPermissionRepository extends Mock
implements GroupPermissionRepository {}
class _FakeGroupInput extends Fake implements GroupInput {}
void main() {
@@ -126,6 +132,65 @@ void main() {
expect(controller.statusFilter, GroupStatusFilter.activeOnly);
});
group('permission sync', () {
late _MockGroupPermissionRepository permissionRepository;
late PermissionManager permissionManager;
setUp(() {
permissionRepository = _MockGroupPermissionRepository();
permissionManager = PermissionManager();
when(
() => permissionRepository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
groupId: any(named: 'groupId'),
menuId: any(named: 'menuId'),
isActive: any(named: 'isActive'),
includeDeleted: any(named: 'includeDeleted'),
),
).thenAnswer(
(_) async => PaginatedResult<GroupPermission>(
items: const <GroupPermission>[],
page: 1,
pageSize: 200,
total: 0,
),
);
});
test('그룹 생성 후 권한 동기화를 시도한다', () async {
when(() => repository.create(any())).thenAnswer((_) async => sampleGroup);
when(
() => repository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
query: any(named: 'query'),
isDefault: any(named: 'isDefault'),
isActive: any(named: 'isActive'),
),
).thenAnswer((_) async => createResult());
final controllerWithSync = GroupController(
repository: repository,
permissionRepository: permissionRepository,
permissionManager: permissionManager,
);
await controllerWithSync.create(GroupInput(groupName: '신규 그룹'));
verify(
() => permissionRepository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
groupId: sampleGroup.id,
menuId: any(named: 'menuId'),
isActive: any(named: 'isActive'),
includeDeleted: any(named: 'includeDeleted'),
),
).called(greaterThanOrEqualTo(1));
});
});
group('mutations', () {
setUp(() {
when(

View File

@@ -0,0 +1,115 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:mocktail/mocktail.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/masters/group_permission/application/permission_synchronizer.dart';
import 'package:superport_v2/features/masters/group_permission/domain/entities/group_permission.dart';
import 'package:superport_v2/features/masters/group_permission/domain/repositories/group_permission_repository.dart';
class _MockGroupPermissionRepository extends Mock
implements GroupPermissionRepository {}
void main() {
TestWidgetsFlutterBinding.ensureInitialized();
test('그룹 권한을 모두 불러와 PermissionManager에 적용한다', () async {
final repository = _MockGroupPermissionRepository();
final manager = PermissionManager();
final synchronizer = PermissionSynchronizer(
repository: repository,
manager: manager,
pageSize: 1,
);
final permissionPage1 = GroupPermission(
id: 1,
group: GroupPermissionGroup(id: 1, groupName: '관리자'),
menu: GroupPermissionMenu(
id: 10,
menuCode: 'INBOUND',
menuName: '입고',
path: '/inventory/inbound',
),
canCreate: true,
canRead: true,
canUpdate: false,
canDelete: false,
);
final permissionPage2 = GroupPermission(
id: 2,
group: GroupPermissionGroup(id: 1, groupName: '관리자'),
menu: GroupPermissionMenu(
id: 11,
menuCode: 'OUTBOUND',
menuName: '출고',
path: '/inventory/outbound',
),
canCreate: false,
canRead: true,
canUpdate: true,
canDelete: false,
);
when(
() => repository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
groupId: any(named: 'groupId'),
menuId: any(named: 'menuId'),
isActive: any(named: 'isActive'),
includeDeleted: any(named: 'includeDeleted'),
),
).thenAnswer((invocation) async {
final page = invocation.namedArguments[#page] as int;
if (page == 1) {
return PaginatedResult<GroupPermission>(
items: [permissionPage1],
page: 1,
pageSize: 1,
total: 2,
);
}
return PaginatedResult<GroupPermission>(
items: [permissionPage2],
page: 2,
pageSize: 1,
total: 2,
);
});
await synchronizer.syncForGroup(1);
verify(
() => repository.list(
page: any(named: 'page'),
pageSize: 1,
groupId: 1,
menuId: null,
isActive: true,
includeDeleted: false,
),
).called(greaterThanOrEqualTo(1));
expect(
manager.can(
PermissionResources.stockTransactions,
PermissionAction.create,
),
isTrue,
);
expect(
manager.can(PermissionResources.stockTransactions, PermissionAction.edit),
isTrue,
);
expect(
manager.can(
PermissionResources.stockTransactions,
PermissionAction.delete,
),
isFalse,
);
});
}

View File

@@ -0,0 +1,71 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:superport_v2/core/permissions/permission_manager.dart';
import 'package:superport_v2/core/permissions/permission_resources.dart';
import 'package:superport_v2/features/masters/group_permission/domain/entities/group_permission.dart';
import 'package:superport_v2/features/masters/group_permission/domain/mappers/group_permission_mapper.dart';
void main() {
test('메뉴 경로 기준으로 권한 맵을 생성한다', () {
final permissions = [
GroupPermission(
id: 1,
group: GroupPermissionGroup(id: 1, groupName: '관리자'),
menu: GroupPermissionMenu(
id: 10,
menuCode: 'INBOUND',
menuName: '입고',
path: '/inventory/inbound',
),
canCreate: true,
canRead: true,
canUpdate: false,
canDelete: false,
),
GroupPermission(
id: 2,
group: GroupPermissionGroup(id: 1, groupName: '관리자'),
menu: GroupPermissionMenu(
id: 11,
menuCode: 'OUTBOUND',
menuName: '출고',
path: '/inventory/outbound',
),
canCreate: false,
canRead: true,
canUpdate: true,
canDelete: true,
),
GroupPermission(
id: 3,
group: GroupPermissionGroup(id: 1, groupName: '관리자'),
menu: GroupPermissionMenu(
id: 12,
menuCode: 'NO_PATH',
menuName: '경로없음',
path: null,
),
canCreate: true,
canRead: true,
canUpdate: true,
canDelete: true,
),
];
final map = buildPermissionMap(permissions);
expect(map.length, 1);
final stockPermissions =
map[PermissionResources.stockTransactions] ?? <PermissionAction>{};
expect(
stockPermissions,
containsAll(<PermissionAction>{
PermissionAction.view,
PermissionAction.create,
PermissionAction.edit,
PermissionAction.delete,
}),
);
expect(map.containsKey('NO_PATH'), isFalse);
});
}

View File

@@ -2,6 +2,7 @@ import 'package:flutter_test/flutter_test.dart';
import 'package:mocktail/mocktail.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/masters/group/domain/entities/group.dart';
import 'package:superport_v2/features/masters/group/domain/repositories/group_repository.dart';
import 'package:superport_v2/features/masters/group_permission/domain/entities/group_permission.dart';
@@ -24,11 +25,17 @@ void main() {
late _MockPermissionRepository permissionRepository;
late _MockGroupRepository groupRepository;
late _MockMenuRepository menuRepository;
late PermissionManager permissionManager;
final samplePermission = GroupPermission(
id: 1,
group: GroupPermissionGroup(id: 1, groupName: '관리자'),
menu: GroupPermissionMenu(id: 10, menuName: '대시보드'),
menu: GroupPermissionMenu(
id: 10,
menuCode: 'DASHBOARD',
menuName: '대시보드',
path: '/dashboard',
),
canCreate: true,
canRead: true,
canUpdate: false,
@@ -52,10 +59,22 @@ void main() {
permissionRepository = _MockPermissionRepository();
groupRepository = _MockGroupRepository();
menuRepository = _MockMenuRepository();
permissionManager = PermissionManager();
when(
() => permissionRepository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
groupId: any(named: 'groupId'),
menuId: any(named: 'menuId'),
isActive: any(named: 'isActive'),
includeDeleted: any(named: 'includeDeleted'),
),
).thenAnswer((_) async => createResult([samplePermission]));
controller = GroupPermissionController(
permissionRepository: permissionRepository,
groupRepository: groupRepository,
menuRepository: menuRepository,
permissionManager: permissionManager,
);
});
@@ -68,6 +87,8 @@ void main() {
query: any(named: 'query'),
isDefault: any(named: 'isDefault'),
isActive: any(named: 'isActive'),
includePermissions: any(named: 'includePermissions'),
includeEmployees: any(named: 'includeEmployees'),
),
).thenAnswer(
(_) async => PaginatedResult<Group>(

View File

@@ -6,6 +6,7 @@ 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/masters/group/domain/entities/group.dart';
import 'package:superport_v2/features/masters/group/domain/repositories/group_repository.dart';
import 'package:superport_v2/features/masters/group_permission/domain/entities/group_permission.dart';
@@ -24,13 +25,16 @@ class _MockMenuRepository extends Mock implements MenuRepository {}
class _FakeGroupPermissionInput extends Fake implements GroupPermissionInput {}
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),
),
);
}
@@ -80,6 +84,8 @@ void main() {
query: any(named: 'query'),
isDefault: any(named: 'isDefault'),
isActive: any(named: 'isActive'),
includePermissions: any(named: 'includePermissions'),
includeEmployees: any(named: 'includeEmployees'),
),
).thenAnswer(
(_) async => PaginatedResult<Group>(
@@ -125,7 +131,12 @@ void main() {
GroupPermission(
id: 1,
group: GroupPermissionGroup(id: 1, groupName: '관리자'),
menu: GroupPermissionMenu(id: 10, menuName: '대시보드'),
menu: GroupPermissionMenu(
id: 10,
menuCode: 'DASHBOARD',
menuName: '대시보드',
path: '/dashboard',
),
canCreate: true,
canRead: true,
),
@@ -201,7 +212,12 @@ void main() {
GroupPermission(
id: 5,
group: GroupPermissionGroup(id: 1, groupName: '관리자'),
menu: GroupPermissionMenu(id: 10, menuName: '대시보드'),
menu: GroupPermissionMenu(
id: 10,
menuCode: 'DASHBOARD',
menuName: '대시보드',
path: '/dashboard',
),
canCreate: true,
canRead: true,
),
@@ -226,7 +242,9 @@ void main() {
),
menu: GroupPermissionMenu(
id: capturedInput!.menuId,
menuCode: 'DASHBOARD',
menuName: '대시보드',
path: '/dashboard',
),
canCreate: capturedInput!.canCreate,
canRead: capturedInput!.canRead,

View File

@@ -0,0 +1,86 @@
import 'package:dio/dio.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:mocktail/mocktail.dart';
import 'package:superport_v2/core/network/api_client.dart';
import 'package:superport_v2/features/masters/product/data/repositories/product_repository_remote.dart';
class _MockApiClient extends Mock implements ApiClient {}
void main() {
late ApiClient apiClient;
late ProductRepositoryRemote repository;
setUpAll(() {
registerFallbackValue(Options());
registerFallbackValue(CancelToken());
});
setUp(() {
apiClient = _MockApiClient();
repository = ProductRepositoryRemote(apiClient: apiClient);
});
test('list는 include 파라미터와 필터를 전달한다', () async {
when(
() => apiClient.get<Map<String, dynamic>>(
any(),
query: any(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
).thenAnswer(
(_) async => Response<Map<String, dynamic>>(
data: {
'items': [
{
'id': 1,
'product_code': 'P-001',
'product_name': '샘플',
'vendor': {
'id': 10,
'vendor_code': 'V-010',
'vendor_name': '테스트 벤더',
},
'uom': {'id': 5, 'uom_name': 'EA'},
},
],
'page': 1,
'page_size': 20,
'total': 1,
},
requestOptions: RequestOptions(path: '/api/v1/products'),
statusCode: 200,
),
);
final result = await repository.list(
page: 3,
pageSize: 40,
query: 'gear',
vendorId: 10,
uomId: 5,
isActive: false,
);
expect(result.items, isNotEmpty);
final verification = verify(
() => apiClient.get<Map<String, dynamic>>(
captureAny(),
query: captureAny(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
);
final query = verification.captured[1] as Map<String, dynamic>;
expect(query['include'], 'vendor,uom');
expect(query['vendor_id'], 10);
expect(query['uom_id'], 5);
expect(query['is_active'], false);
expect(query['page'], 3);
expect(query['page_size'], 40);
expect(query['q'], 'gear');
});
}

View File

@@ -154,6 +154,33 @@ void main() {
).called(1);
});
testWidgets('목록이 비어 있으면 안내 문구를 표시한다', (tester) async {
when(
() => productRepository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
query: any(named: 'query'),
vendorId: any(named: 'vendorId'),
uomId: any(named: 'uomId'),
isActive: any(named: 'isActive'),
),
).thenAnswer(
(_) async => PaginatedResult<Product>(
items: const [],
page: 1,
pageSize: 20,
total: 0,
),
);
await tester.pumpWidget(
_buildApp(ProductPage(routeUri: Uri(path: '/masters/products'))),
);
await tester.pumpAndSettle();
expect(find.text('조건에 맞는 제품이 없습니다.'), findsOneWidget);
});
testWidgets('폼 검증: 필수값 미입력 시 에러 메시지를 표시한다', (tester) async {
when(
() => productRepository.list(

View File

@@ -0,0 +1,65 @@
import 'package:dio/dio.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:mocktail/mocktail.dart';
import 'package:superport_v2/core/network/api_client.dart';
import 'package:superport_v2/features/masters/user/data/repositories/user_repository_remote.dart';
class _MockApiClient extends Mock implements ApiClient {}
void main() {
late ApiClient apiClient;
late UserRepositoryRemote repository;
setUpAll(() {
registerFallbackValue(Options());
registerFallbackValue(CancelToken());
});
setUp(() {
apiClient = _MockApiClient();
repository = UserRepositoryRemote(apiClient: apiClient);
});
test('목록 조회 시 include=group 파라미터를 전달한다', () async {
when(
() => apiClient.get<Map<String, dynamic>>(
any(),
query: any(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
).thenAnswer(
(_) async => Response<Map<String, dynamic>>(
data: {
'items': [
{
'id': 1,
'employee_no': 'E-001',
'employee_name': '홍길동',
'group': {'id': 2, 'group_name': '관리자'},
},
],
},
requestOptions: RequestOptions(path: '/api/v1/employees'),
statusCode: 200,
),
);
await repository.list();
final captured = verify(
() => apiClient.get<Map<String, dynamic>>(
captureAny(),
query: captureAny(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
).captured;
final path = captured[0] as String;
final query = captured[1] as Map<String, dynamic>;
expect(path, equals('/api/v1/employees'));
expect(query['include'], 'group');
});
}

View File

@@ -2,8 +2,11 @@ import 'package:flutter_test/flutter_test.dart';
import 'package:mocktail/mocktail.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/masters/group/domain/entities/group.dart';
import 'package:superport_v2/features/masters/group/domain/repositories/group_repository.dart';
import 'package:superport_v2/features/masters/group_permission/domain/entities/group_permission.dart';
import 'package:superport_v2/features/masters/group_permission/domain/repositories/group_permission_repository.dart';
import 'package:superport_v2/features/masters/user/domain/entities/user.dart';
import 'package:superport_v2/features/masters/user/domain/repositories/user_repository.dart';
import 'package:superport_v2/features/masters/user/presentation/controllers/user_controller.dart';
@@ -12,12 +15,17 @@ class _MockUserRepository extends Mock implements UserRepository {}
class _MockGroupRepository extends Mock implements GroupRepository {}
class _MockGroupPermissionRepository extends Mock
implements GroupPermissionRepository {}
class _FakeUserInput extends Fake implements UserInput {}
void main() {
late UserController controller;
late _MockUserRepository userRepository;
late _MockGroupRepository groupRepository;
late _MockGroupPermissionRepository permissionRepository;
late PermissionManager permissionManager;
final sampleUser = UserAccount(
id: 1,
@@ -44,9 +52,43 @@ void main() {
setUp(() {
userRepository = _MockUserRepository();
groupRepository = _MockGroupRepository();
permissionRepository = _MockGroupPermissionRepository();
permissionManager = PermissionManager();
when(
() => permissionRepository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
groupId: any(named: 'groupId'),
menuId: any(named: 'menuId'),
isActive: any(named: 'isActive'),
includeDeleted: any(named: 'includeDeleted'),
),
).thenAnswer(
(_) async => PaginatedResult<GroupPermission>(
items: [
GroupPermission(
id: 1,
group: GroupPermissionGroup(id: 1, groupName: '관리자'),
menu: GroupPermissionMenu(
id: 10,
menuCode: 'DASHBOARD',
menuName: '대시보드',
path: '/dashboard',
),
canCreate: true,
canRead: true,
),
],
page: 1,
pageSize: 200,
total: 1,
),
);
controller = UserController(
userRepository: userRepository,
groupRepository: groupRepository,
permissionRepository: permissionRepository,
permissionManager: permissionManager,
);
});
@@ -58,6 +100,8 @@ void main() {
query: any(named: 'query'),
isDefault: any(named: 'isDefault'),
isActive: any(named: 'isActive'),
includePermissions: any(named: 'includePermissions'),
includeEmployees: any(named: 'includeEmployees'),
),
).thenAnswer(
(_) async => PaginatedResult<Group>(
@@ -82,6 +126,8 @@ void main() {
query: any(named: 'query'),
isDefault: any(named: 'isDefault'),
isActive: any(named: 'isActive'),
includePermissions: any(named: 'includePermissions'),
includeEmployees: any(named: 'includeEmployees'),
),
).thenAnswer(
(_) async => PaginatedResult<Group>(
@@ -195,6 +241,24 @@ void main() {
verify(() => userRepository.delete(1)).called(1);
});
test('delete 이후 권한 동기화를 시도한다', () async {
when(() => userRepository.delete(any())).thenAnswer((_) async {});
await controller.fetch();
await controller.delete(sampleUser.id!);
verify(
() => permissionRepository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
groupId: sampleUser.group!.id,
menuId: any(named: 'menuId'),
isActive: any(named: 'isActive'),
includeDeleted: any(named: 'includeDeleted'),
),
).called(greaterThanOrEqualTo(1));
});
test('restore 성공', () async {
when(
() => userRepository.restore(any()),

View File

@@ -6,8 +6,11 @@ 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/masters/group/domain/entities/group.dart';
import 'package:superport_v2/features/masters/group/domain/repositories/group_repository.dart';
import 'package:superport_v2/features/masters/group_permission/domain/entities/group_permission.dart';
import 'package:superport_v2/features/masters/group_permission/domain/repositories/group_permission_repository.dart';
import 'package:superport_v2/features/masters/user/domain/entities/user.dart';
import 'package:superport_v2/features/masters/user/domain/repositories/user_repository.dart';
import 'package:superport_v2/features/masters/user/presentation/pages/user_page.dart';
@@ -16,16 +19,22 @@ class _MockUserRepository extends Mock implements UserRepository {}
class _MockGroupRepository extends Mock implements GroupRepository {}
class _MockGroupPermissionRepository extends Mock
implements GroupPermissionRepository {}
class _FakeUserInput extends Fake implements UserInput {}
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),
),
);
}
@@ -55,12 +64,17 @@ void main() {
group('플래그 On', () {
late _MockUserRepository userRepository;
late _MockGroupRepository groupRepository;
late _MockGroupPermissionRepository permissionRepository;
setUp(() {
dotenv.testLoad(fileInput: 'FEATURE_USERS_ENABLED=true\n');
userRepository = _MockUserRepository();
groupRepository = _MockGroupRepository();
permissionRepository = _MockGroupPermissionRepository();
GetIt.I.registerLazySingleton<UserRepository>(() => userRepository);
GetIt.I.registerLazySingleton<GroupRepository>(() => groupRepository);
GetIt.I.registerLazySingleton<GroupPermissionRepository>(
() => permissionRepository,
);
when(
() => groupRepository.list(
@@ -69,6 +83,8 @@ void main() {
query: any(named: 'query'),
isDefault: any(named: 'isDefault'),
isActive: any(named: 'isActive'),
includePermissions: any(named: 'includePermissions'),
includeEmployees: any(named: 'includeEmployees'),
),
).thenAnswer(
(_) async => PaginatedResult<Group>(
@@ -78,6 +94,37 @@ void main() {
total: 1,
),
);
when(
() => permissionRepository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
groupId: any(named: 'groupId'),
menuId: any(named: 'menuId'),
isActive: any(named: 'isActive'),
includeDeleted: any(named: 'includeDeleted'),
),
).thenAnswer(
(_) async => PaginatedResult<GroupPermission>(
items: [
GroupPermission(
id: 1,
group: GroupPermissionGroup(id: 1, groupName: '관리자'),
menu: GroupPermissionMenu(
id: 10,
menuCode: 'DASHBOARD',
menuName: '대시보드',
path: '/dashboard',
),
canCreate: true,
canRead: true,
),
],
page: 1,
pageSize: 200,
total: 1,
),
);
});
testWidgets('목록 조회 후 테이블 렌더', (tester) async {

View File

@@ -84,6 +84,35 @@ void main() {
).called(1);
});
testWidgets('목록이 비어 있으면 안내 문구를 표시한다', (tester) async {
dotenv.testLoad(fileInput: 'FEATURE_VENDORS_ENABLED=true\n');
final repository = _MockVendorRepository();
GetIt.I.registerLazySingleton<VendorRepository>(() => repository);
when(
() => repository.list(
page: any(named: 'page'),
pageSize: any(named: 'pageSize'),
query: any(named: 'query'),
isActive: any(named: 'isActive'),
),
).thenAnswer(
(_) async => PaginatedResult<Vendor>(
items: const [],
page: 1,
pageSize: 20,
total: 0,
),
);
await tester.pumpWidget(
_buildApp(VendorPage(routeUri: Uri(path: '/masters/vendors'))),
);
await tester.pumpAndSettle();
expect(find.text('조건에 맞는 벤더가 없습니다.'), findsOneWidget);
});
testWidgets('신규 등록 폼에서 필수값 미입력 시 검증 메시지를 보여준다', (tester) async {
dotenv.testLoad(fileInput: 'FEATURE_VENDORS_ENABLED=true\n');
final repository = _MockVendorRepository();

View File

@@ -0,0 +1,60 @@
import 'package:dio/dio.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:mocktail/mocktail.dart';
import 'package:superport_v2/core/network/api_client.dart';
import 'package:superport_v2/features/masters/warehouse/data/repositories/warehouse_repository_remote.dart';
class _MockApiClient extends Mock implements ApiClient {}
void main() {
late ApiClient apiClient;
late WarehouseRepositoryRemote repository;
setUpAll(() {
registerFallbackValue(Options());
registerFallbackValue(CancelToken());
});
setUp(() {
apiClient = _MockApiClient();
repository = WarehouseRepositoryRemote(apiClient: apiClient);
});
test('include=zipcode 파라미터를 기본으로 전달한다', () async {
when(
() => apiClient.get<Map<String, dynamic>>(
any(),
query: any(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
).thenAnswer(
(_) async => Response<Map<String, dynamic>>(
data: {
'items': [
{'id': 1, 'warehouse_code': 'WH-001', 'warehouse_name': '1센터'},
],
},
requestOptions: RequestOptions(path: '/api/v1/warehouses'),
statusCode: 200,
),
);
await repository.list();
final captured = verify(
() => apiClient.get<Map<String, dynamic>>(
captureAny(),
query: captureAny(named: 'query'),
options: any(named: 'options'),
cancelToken: any(named: 'cancelToken'),
),
).captured;
final path = captured[0] as String;
final query = captured[1] as Map<String, dynamic>;
expect(path, equals('/api/v1/warehouses'));
expect(query['include'], 'zipcode');
});
}

View File

@@ -48,6 +48,7 @@ void main() {
pageSize: any(named: 'pageSize'),
query: any(named: 'query'),
isActive: any(named: 'isActive'),
includeZipcode: any(named: 'includeZipcode'),
),
).thenAnswer((_) async => createResult());
@@ -55,8 +56,13 @@ void main() {
expect(controller.result?.items, isNotEmpty);
verify(
() =>
repository.list(page: 1, pageSize: 20, query: null, isActive: null),
() => repository.list(
page: 1,
pageSize: 20,
query: null,
isActive: null,
includeZipcode: true,
),
).called(1);
});
@@ -67,6 +73,7 @@ void main() {
pageSize: any(named: 'pageSize'),
query: any(named: 'query'),
isActive: any(named: 'isActive'),
includeZipcode: any(named: 'includeZipcode'),
),
).thenThrow(Exception('fail'));
@@ -92,6 +99,7 @@ void main() {
pageSize: any(named: 'pageSize'),
query: any(named: 'query'),
isActive: any(named: 'isActive'),
includeZipcode: any(named: 'includeZipcode'),
),
).thenAnswer((_) async => createResult());
});

View File

@@ -92,6 +92,7 @@ void main() {
pageSize: any(named: 'pageSize'),
query: any(named: 'query'),
isActive: any(named: 'isActive'),
includeZipcode: any(named: 'includeZipcode'),
),
).thenAnswer(
(_) async => PaginatedResult<Warehouse>(
@@ -116,8 +117,13 @@ void main() {
expect(find.text('WH-001'), findsOneWidget);
verify(
() =>
repository.list(page: 1, pageSize: 20, query: null, isActive: null),
() => repository.list(
page: 1,
pageSize: 20,
query: null,
isActive: null,
includeZipcode: true,
),
).called(1);
});
@@ -128,6 +134,7 @@ void main() {
pageSize: any(named: 'pageSize'),
query: any(named: 'query'),
isActive: any(named: 'isActive'),
includeZipcode: any(named: 'includeZipcode'),
),
).thenAnswer(
(_) async => PaginatedResult<Warehouse>(