@gadmin2n/schematics 0.0.88 → 0.0.89
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/lib/application/files/gadmin2-game-angle-demo/.dockerignore +16 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/Dockerfile.codegen +40 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/Dockerfile.server +76 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/Dockerfile.web +53 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/Jenkinsfile +219 -33
- package/dist/lib/application/files/gadmin2-game-angle-demo/compose-ctl.sh +250 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/config/prisma/workflow.prisma +4 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/dev/postgres/init.sql +12 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/docker-compose.md +170 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/docker-compose.yml +254 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/package.json +8 -7
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/lib/page-helpers.ts +1 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/prismaModels.ts +1 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/agenda.seed.ts +39 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/audit.seed.ts +40 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/bootstrap.ts +56 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/canvas.seed.ts +39 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/{scripts/sync-data-mngt-pages.ts → seed/data-mngt.seed.ts} +36 -20
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/game.seed.ts +44 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/index.ts +30 -6
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/permission.seed.ts +130 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/workflow-event-trigger.ts +60 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/workflow-node-types.ts +11 -25
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/workflow.seed.ts +108 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/main.ts +1 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/agendaJob/agendaJob.controller.spec.ts +31 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/audit/audit.controller.spec.ts +31 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/audit/audit.service.spec.ts +41 -57
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/game/game.controller.spec.ts +31 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/game/game.service.spec.ts +309 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/page/page.controller.spec.ts +31 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/page/page.service.spec.ts +315 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/pageResource/pageResource.controller.spec.ts +31 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/pageResource/pageResource.service.spec.ts +312 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/resource/resource.controller.spec.ts +31 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/resource/resource.service.spec.ts +317 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/role/role.controller.spec.ts +31 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/role/role.service.spec.ts +309 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/rolePages/rolePages.controller.spec.ts +31 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/rolePages/rolePages.service.spec.ts +299 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/roleResource/roleResource.controller.spec.ts +31 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/roleResource/roleResource.service.spec.ts +307 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/user/user.controller.spec.ts +31 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/user/user.service.spec.ts +309 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/dsl-validate.util.spec.ts +205 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/dsl-validate.util.ts +116 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/temporal.service.spec.ts +158 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/temporal.service.ts +110 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/webhook-signature.util.spec.ts +79 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/webhook-signature.util.ts +54 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/workflow.controller.ts +34 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/workflow.service.spec.ts +457 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflow/workflow.service.ts +241 -4
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowEventOutbox/workflowEventOutbox.controller.spec.ts +34 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowEventOutbox/workflowEventOutbox.service.spec.ts +24 -30
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowNodeInstance/workflowNodeInstance.controller.spec.ts +34 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowNodeInstance/workflowNodeInstance.service.spec.ts +36 -36
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowNodeType/workflowNodeType.controller.spec.ts +34 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/workflowNodeType/workflowNodeType.service.spec.ts +48 -24
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/README.md +312 -3
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/TODO.md +152 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/.dockerignore +12 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/Dockerfile +79 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/GRACEFUL-DEPLOYMENT.md +270 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/activities/index.ts +1 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/activities/reporting.ts +23 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/index.ts +70 -5
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/outbox-poller.ts +246 -90
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/tests/cron-trigger-workflow.test.ts +20 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/worker/src/workflows/dsl-workflow.ts +96 -8
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/nginx.conf +74 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/ElementInspector.tsx +18 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/promptGenerator.ts +1 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/form.tsx +1 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/locales/en/common.json +3 -3
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/locales/zh_CN/common.json +3 -3
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/plugins/devShellPlugin.ts +4 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasEditPage.tsx +9 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasListPage.tsx +156 -139
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasPage.tsx +14 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasToolbar.tsx +62 -0
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/PublishModal.tsx +4 -6
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/canvasApi.ts +18 -27
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/canvasDefaults.ts +32 -11
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/demos.ts +48 -61
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas-page/index.tsx +3 -6
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/workflow/components/DslView.tsx +16 -16
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/workflow/editor.tsx +28 -35
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/workflow/instance-detail.tsx +34 -3
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/workflow/show.tsx +1 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/workflow/types.ts +1 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/styles/antd.css +6 -0
- package/package.json +1 -1
- package/dist/lib/application/files/gadmin2-game-angle-demo/.gitattributes +0 -2
- package/dist/lib/application/files/gadmin2-game-angle-demo/Dockerfile +0 -63
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/sync-resources.ts +0 -100
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/permissions.ts +0 -302
- package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/canvas/canvas.controller.spec.ts +0 -20
- package/dist/lib/application/files/gadmin2-game-angle-demo/temporal/sql/create-event-trigger.sql +0 -87
- /package/dist/lib/application/files/gadmin2-game-angle-demo/{GRACEFUL-DEPLOYMENT.md → server/GRACEFUL-DEPLOYMENT.md} +0 -0
|
@@ -1,15 +1,44 @@
|
|
|
1
|
+
/// <reference types="jest" />
|
|
2
|
+
import { ConfigService } from '@nestjs/config';
|
|
1
3
|
import { Test, TestingModule } from '@nestjs/testing';
|
|
4
|
+
import { WINSTON_MODULE_PROVIDER } from 'nest-winston';
|
|
2
5
|
import { RoleResourceController } from './roleResource.controller';
|
|
3
6
|
import { RoleResourceService } from './roleResource.service';
|
|
4
7
|
|
|
8
|
+
// Mock RoleResourceService:用空对象屏蔽真实 Service 的依赖(PrismaService 等),
|
|
9
|
+
// 避免 NestJS 在测试模块编译时去解析 Service 的构造参数。
|
|
10
|
+
const mockRoleResourceService = {};
|
|
11
|
+
|
|
12
|
+
// Mock ConfigService
|
|
13
|
+
const mockConfigService = {
|
|
14
|
+
get: jest.fn(),
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
// Mock Logger
|
|
18
|
+
const mockLogger = {
|
|
19
|
+
child: jest.fn().mockReturnThis(),
|
|
20
|
+
info: jest.fn(),
|
|
21
|
+
error: jest.fn(),
|
|
22
|
+
warn: jest.fn(),
|
|
23
|
+
debug: jest.fn(),
|
|
24
|
+
};
|
|
25
|
+
|
|
5
26
|
describe('RoleResourceController', () => {
|
|
6
27
|
let controller: RoleResourceController;
|
|
7
28
|
|
|
8
29
|
beforeEach(async () => {
|
|
9
30
|
const module: TestingModule = await Test.createTestingModule({
|
|
10
31
|
controllers: [RoleResourceController],
|
|
11
|
-
providers: [
|
|
12
|
-
|
|
32
|
+
providers: [
|
|
33
|
+
{ provide: RoleResourceService, useValue: mockRoleResourceService },
|
|
34
|
+
{ provide: ConfigService, useValue: mockConfigService },
|
|
35
|
+
{ provide: WINSTON_MODULE_PROVIDER, useValue: mockLogger },
|
|
36
|
+
],
|
|
37
|
+
})
|
|
38
|
+
// 兜底:自动 mock 任何未显式提供的依赖
|
|
39
|
+
// (例如 ACGuard 内部的 __roles_builder__ token)
|
|
40
|
+
.useMocker(() => ({}))
|
|
41
|
+
.compile();
|
|
13
42
|
|
|
14
43
|
controller = module.get<RoleResourceController>(RoleResourceController);
|
|
15
44
|
});
|
|
@@ -1,18 +1,324 @@
|
|
|
1
|
+
/// <reference types="jest" />
|
|
2
|
+
import { ConfigService } from '@nestjs/config';
|
|
1
3
|
import { Test, TestingModule } from '@nestjs/testing';
|
|
4
|
+
import { WINSTON_MODULE_PROVIDER } from 'nest-winston';
|
|
5
|
+
|
|
6
|
+
// Mock nestjs-prisma 模块
|
|
7
|
+
jest.mock('nestjs-prisma', () => ({
|
|
8
|
+
PrismaService: jest.fn(),
|
|
9
|
+
}));
|
|
10
|
+
|
|
11
|
+
import { PrismaService } from 'nestjs-prisma';
|
|
2
12
|
import { RoleResourceService } from './roleResource.service';
|
|
3
13
|
|
|
14
|
+
// Mock PrismaService
|
|
15
|
+
const mockPrismaService = {
|
|
16
|
+
roleResource: {
|
|
17
|
+
create: jest.fn(),
|
|
18
|
+
createMany: jest.fn(),
|
|
19
|
+
findMany: jest.fn(),
|
|
20
|
+
findUnique: jest.fn(),
|
|
21
|
+
update: jest.fn(),
|
|
22
|
+
updateMany: jest.fn(),
|
|
23
|
+
delete: jest.fn(),
|
|
24
|
+
deleteMany: jest.fn(),
|
|
25
|
+
count: jest.fn(),
|
|
26
|
+
groupBy: jest.fn(),
|
|
27
|
+
aggregate: jest.fn(),
|
|
28
|
+
},
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
// Mock Logger
|
|
32
|
+
const mockLogger = {
|
|
33
|
+
child: jest.fn().mockReturnThis(),
|
|
34
|
+
info: jest.fn(),
|
|
35
|
+
error: jest.fn(),
|
|
36
|
+
warn: jest.fn(),
|
|
37
|
+
debug: jest.fn(),
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
// Mock ConfigService
|
|
41
|
+
const mockConfigService = {
|
|
42
|
+
get: jest.fn(),
|
|
43
|
+
};
|
|
44
|
+
|
|
4
45
|
describe('RoleResourceService', () => {
|
|
5
46
|
let service: RoleResourceService;
|
|
47
|
+
let prisma: typeof mockPrismaService;
|
|
6
48
|
|
|
7
49
|
beforeEach(async () => {
|
|
50
|
+
// 每个测试前重置所有 mock
|
|
51
|
+
jest.clearAllMocks();
|
|
52
|
+
|
|
8
53
|
const module: TestingModule = await Test.createTestingModule({
|
|
9
|
-
providers: [
|
|
54
|
+
providers: [
|
|
55
|
+
RoleResourceService,
|
|
56
|
+
{ provide: PrismaService, useValue: mockPrismaService },
|
|
57
|
+
{ provide: ConfigService, useValue: mockConfigService },
|
|
58
|
+
{ provide: WINSTON_MODULE_PROVIDER, useValue: mockLogger },
|
|
59
|
+
],
|
|
10
60
|
}).compile();
|
|
11
61
|
|
|
12
62
|
service = module.get<RoleResourceService>(RoleResourceService);
|
|
63
|
+
prisma = mockPrismaService;
|
|
13
64
|
});
|
|
14
65
|
|
|
15
66
|
it('should be defined', () => {
|
|
16
67
|
expect(service).toBeDefined();
|
|
17
68
|
});
|
|
69
|
+
|
|
70
|
+
describe('createOne', () => {
|
|
71
|
+
it('should create a single roleResource record', async () => {
|
|
72
|
+
const createArgs = {
|
|
73
|
+
data: {
|
|
74
|
+
actions: {},
|
|
75
|
+
creator: 'test_creator',
|
|
76
|
+
},
|
|
77
|
+
} as any;
|
|
78
|
+
const expectedResult = { id: 1, ...createArgs.data };
|
|
79
|
+
|
|
80
|
+
prisma.roleResource.create.mockResolvedValue(expectedResult);
|
|
81
|
+
|
|
82
|
+
const result = await service.createOne(createArgs);
|
|
83
|
+
|
|
84
|
+
expect(prisma.roleResource.create).toHaveBeenCalledWith(createArgs);
|
|
85
|
+
expect(prisma.roleResource.create).toHaveBeenCalledTimes(1);
|
|
86
|
+
expect(result).toEqual(expectedResult);
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
it('should throw error when create fails', async () => {
|
|
90
|
+
const createArgs = {
|
|
91
|
+
data: {
|
|
92
|
+
actions: {},
|
|
93
|
+
creator: 'test_creator',
|
|
94
|
+
},
|
|
95
|
+
} as any;
|
|
96
|
+
const error = new Error('Database connection failed');
|
|
97
|
+
|
|
98
|
+
prisma.roleResource.create.mockRejectedValue(error);
|
|
99
|
+
|
|
100
|
+
await expect(service.createOne(createArgs)).rejects.toThrow(
|
|
101
|
+
'Database connection failed',
|
|
102
|
+
);
|
|
103
|
+
});
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
describe('createMany', () => {
|
|
107
|
+
it('should create multiple roleResource records with skipDuplicates', async () => {
|
|
108
|
+
const data = [
|
|
109
|
+
{
|
|
110
|
+
actions: {},
|
|
111
|
+
creator: 'test_creator',
|
|
112
|
+
},
|
|
113
|
+
{
|
|
114
|
+
actions: {},
|
|
115
|
+
creator: 'test_creator',
|
|
116
|
+
},
|
|
117
|
+
] as any;
|
|
118
|
+
const expectedResult = { count: 2 };
|
|
119
|
+
|
|
120
|
+
prisma.roleResource.createMany.mockResolvedValue(expectedResult);
|
|
121
|
+
|
|
122
|
+
const result = await service.createMany(data);
|
|
123
|
+
|
|
124
|
+
expect(prisma.roleResource.createMany).toHaveBeenCalledWith({
|
|
125
|
+
data,
|
|
126
|
+
skipDuplicates: true,
|
|
127
|
+
});
|
|
128
|
+
expect(result).toEqual(expectedResult);
|
|
129
|
+
});
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
describe('findMany', () => {
|
|
133
|
+
it('should return entities with itemCount', async () => {
|
|
134
|
+
const findArgs = {
|
|
135
|
+
where: {},
|
|
136
|
+
take: 10,
|
|
137
|
+
skip: 0,
|
|
138
|
+
} as any;
|
|
139
|
+
const mockEntities = [{ id: 1 }, { id: 2 }];
|
|
140
|
+
|
|
141
|
+
prisma.roleResource.count.mockResolvedValue(2);
|
|
142
|
+
prisma.roleResource.findMany.mockResolvedValue(mockEntities);
|
|
143
|
+
|
|
144
|
+
const result = await service.findMany(findArgs);
|
|
145
|
+
|
|
146
|
+
expect(prisma.roleResource.count).toHaveBeenCalledWith({
|
|
147
|
+
where: findArgs.where,
|
|
148
|
+
});
|
|
149
|
+
expect(prisma.roleResource.findMany).toHaveBeenCalledWith(findArgs);
|
|
150
|
+
expect(result).toEqual({
|
|
151
|
+
itemCount: 2,
|
|
152
|
+
entities: mockEntities,
|
|
153
|
+
});
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
it('should return empty result when no records found', async () => {
|
|
157
|
+
const findArgs = { where: {} } as any;
|
|
158
|
+
|
|
159
|
+
prisma.roleResource.count.mockResolvedValue(0);
|
|
160
|
+
prisma.roleResource.findMany.mockResolvedValue([]);
|
|
161
|
+
|
|
162
|
+
const result = await service.findMany(findArgs);
|
|
163
|
+
|
|
164
|
+
expect(result).toEqual({ itemCount: 0, entities: [] });
|
|
165
|
+
});
|
|
166
|
+
});
|
|
167
|
+
|
|
168
|
+
describe('findUnique', () => {
|
|
169
|
+
it('should find a single roleResource record by id', async () => {
|
|
170
|
+
const id = 1;
|
|
171
|
+
const select = { id: true };
|
|
172
|
+
const expectedResult = { id: 1 };
|
|
173
|
+
|
|
174
|
+
prisma.roleResource.findUnique.mockResolvedValue(expectedResult);
|
|
175
|
+
|
|
176
|
+
const result = await service.findUnique(id, select);
|
|
177
|
+
|
|
178
|
+
expect(prisma.roleResource.findUnique).toHaveBeenCalledWith({
|
|
179
|
+
where: { id },
|
|
180
|
+
select,
|
|
181
|
+
});
|
|
182
|
+
expect(result).toEqual(expectedResult);
|
|
183
|
+
});
|
|
184
|
+
|
|
185
|
+
it('should return null when record not found', async () => {
|
|
186
|
+
prisma.roleResource.findUnique.mockResolvedValue(null);
|
|
187
|
+
|
|
188
|
+
const result = await service.findUnique(1, { id: true });
|
|
189
|
+
|
|
190
|
+
expect(result).toBeNull();
|
|
191
|
+
});
|
|
192
|
+
});
|
|
193
|
+
|
|
194
|
+
describe('updateUnique', () => {
|
|
195
|
+
it('should update a single roleResource record', async () => {
|
|
196
|
+
const id = 1;
|
|
197
|
+
const data = {
|
|
198
|
+
actions: {},
|
|
199
|
+
creator: 'test_creator',
|
|
200
|
+
} as any;
|
|
201
|
+
const expectedResult = { id: 1, ...data };
|
|
202
|
+
|
|
203
|
+
prisma.roleResource.update.mockResolvedValue(expectedResult);
|
|
204
|
+
|
|
205
|
+
const result = await service.updateUnique(id, data);
|
|
206
|
+
|
|
207
|
+
expect(prisma.roleResource.update).toHaveBeenCalledWith({
|
|
208
|
+
where: { id },
|
|
209
|
+
data,
|
|
210
|
+
});
|
|
211
|
+
expect(result).toEqual(expectedResult);
|
|
212
|
+
});
|
|
213
|
+
});
|
|
214
|
+
|
|
215
|
+
describe('updateMany', () => {
|
|
216
|
+
it('should update multiple roleResource records', async () => {
|
|
217
|
+
const updateArgs = {
|
|
218
|
+
where: {},
|
|
219
|
+
data: {
|
|
220
|
+
actions: {},
|
|
221
|
+
creator: 'test_creator',
|
|
222
|
+
},
|
|
223
|
+
} as any;
|
|
224
|
+
const expectedResult = { count: 5 };
|
|
225
|
+
|
|
226
|
+
prisma.roleResource.updateMany.mockResolvedValue(expectedResult);
|
|
227
|
+
|
|
228
|
+
const result = await service.updateMany(updateArgs);
|
|
229
|
+
|
|
230
|
+
expect(prisma.roleResource.updateMany).toHaveBeenCalledWith(updateArgs);
|
|
231
|
+
expect(result).toEqual(expectedResult);
|
|
232
|
+
});
|
|
233
|
+
});
|
|
234
|
+
|
|
235
|
+
describe('deleteUnique', () => {
|
|
236
|
+
it('should delete a single roleResource record and return count', async () => {
|
|
237
|
+
const id = 1;
|
|
238
|
+
|
|
239
|
+
prisma.roleResource.delete.mockResolvedValue({ id: 1 });
|
|
240
|
+
|
|
241
|
+
const result = await service.deleteUnique(id);
|
|
242
|
+
|
|
243
|
+
expect(prisma.roleResource.delete).toHaveBeenCalledWith({
|
|
244
|
+
where: { id },
|
|
245
|
+
});
|
|
246
|
+
expect(result).toEqual({ count: 1 });
|
|
247
|
+
});
|
|
248
|
+
});
|
|
249
|
+
|
|
250
|
+
describe('deleteMany', () => {
|
|
251
|
+
it('should delete multiple roleResource records', async () => {
|
|
252
|
+
const deleteArgs = { where: {} } as any;
|
|
253
|
+
const expectedResult = { count: 3 };
|
|
254
|
+
|
|
255
|
+
prisma.roleResource.deleteMany.mockResolvedValue(expectedResult);
|
|
256
|
+
|
|
257
|
+
const result = await service.deleteMany(deleteArgs);
|
|
258
|
+
|
|
259
|
+
expect(prisma.roleResource.deleteMany).toHaveBeenCalledWith(deleteArgs);
|
|
260
|
+
expect(result).toEqual(expectedResult);
|
|
261
|
+
});
|
|
262
|
+
});
|
|
263
|
+
|
|
264
|
+
describe('count', () => {
|
|
265
|
+
it('should count roleResource records with filter', async () => {
|
|
266
|
+
const countArgs = { where: {} } as any;
|
|
267
|
+
|
|
268
|
+
prisma.roleResource.count.mockResolvedValue(10);
|
|
269
|
+
|
|
270
|
+
const result = await service.count(countArgs);
|
|
271
|
+
|
|
272
|
+
expect(prisma.roleResource.count).toHaveBeenCalledWith(countArgs);
|
|
273
|
+
expect(result).toBe(10);
|
|
274
|
+
});
|
|
275
|
+
|
|
276
|
+
it('should count all records when no filter provided', async () => {
|
|
277
|
+
prisma.roleResource.count.mockResolvedValue(100);
|
|
278
|
+
|
|
279
|
+
const result = await service.count();
|
|
280
|
+
|
|
281
|
+
expect(prisma.roleResource.count).toHaveBeenCalledWith(undefined);
|
|
282
|
+
expect(result).toBe(100);
|
|
283
|
+
});
|
|
284
|
+
});
|
|
285
|
+
|
|
286
|
+
describe('groupBy', () => {
|
|
287
|
+
it('should group roleResource records by specified field', async () => {
|
|
288
|
+
const groupByArgs = {
|
|
289
|
+
by: ['id'],
|
|
290
|
+
_count: { id: true },
|
|
291
|
+
} as any;
|
|
292
|
+
const expectedResult = [{ id: 1, _count: { id: 50 } }];
|
|
293
|
+
|
|
294
|
+
prisma.roleResource.groupBy.mockResolvedValue(expectedResult);
|
|
295
|
+
|
|
296
|
+
const result = await service.groupBy(groupByArgs);
|
|
297
|
+
|
|
298
|
+
expect(prisma.roleResource.groupBy).toHaveBeenCalledWith(groupByArgs);
|
|
299
|
+
expect(result).toEqual(expectedResult);
|
|
300
|
+
});
|
|
301
|
+
});
|
|
302
|
+
|
|
303
|
+
describe('aggregate', () => {
|
|
304
|
+
it('should aggregate roleResource records', async () => {
|
|
305
|
+
const aggregateArgs = {
|
|
306
|
+
_count: true,
|
|
307
|
+
_max: { id: true },
|
|
308
|
+
_min: { id: true },
|
|
309
|
+
} as any;
|
|
310
|
+
const expectedResult = {
|
|
311
|
+
_count: 100,
|
|
312
|
+
_max: { id: 1 },
|
|
313
|
+
_min: { id: 1 },
|
|
314
|
+
};
|
|
315
|
+
|
|
316
|
+
prisma.roleResource.aggregate.mockResolvedValue(expectedResult);
|
|
317
|
+
|
|
318
|
+
const result = await service.aggregate(aggregateArgs);
|
|
319
|
+
|
|
320
|
+
expect(prisma.roleResource.aggregate).toHaveBeenCalledWith(aggregateArgs);
|
|
321
|
+
expect(result).toEqual(expectedResult);
|
|
322
|
+
});
|
|
323
|
+
});
|
|
18
324
|
});
|
|
@@ -1,15 +1,44 @@
|
|
|
1
|
+
/// <reference types="jest" />
|
|
2
|
+
import { ConfigService } from '@nestjs/config';
|
|
1
3
|
import { Test, TestingModule } from '@nestjs/testing';
|
|
4
|
+
import { WINSTON_MODULE_PROVIDER } from 'nest-winston';
|
|
2
5
|
import { UserController } from './user.controller';
|
|
3
6
|
import { UserService } from './user.service';
|
|
4
7
|
|
|
8
|
+
// Mock UserService:用空对象屏蔽真实 Service 的依赖(PrismaService 等),
|
|
9
|
+
// 避免 NestJS 在测试模块编译时去解析 Service 的构造参数。
|
|
10
|
+
const mockUserService = {};
|
|
11
|
+
|
|
12
|
+
// Mock ConfigService
|
|
13
|
+
const mockConfigService = {
|
|
14
|
+
get: jest.fn(),
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
// Mock Logger
|
|
18
|
+
const mockLogger = {
|
|
19
|
+
child: jest.fn().mockReturnThis(),
|
|
20
|
+
info: jest.fn(),
|
|
21
|
+
error: jest.fn(),
|
|
22
|
+
warn: jest.fn(),
|
|
23
|
+
debug: jest.fn(),
|
|
24
|
+
};
|
|
25
|
+
|
|
5
26
|
describe('UserController', () => {
|
|
6
27
|
let controller: UserController;
|
|
7
28
|
|
|
8
29
|
beforeEach(async () => {
|
|
9
30
|
const module: TestingModule = await Test.createTestingModule({
|
|
10
31
|
controllers: [UserController],
|
|
11
|
-
providers: [
|
|
12
|
-
|
|
32
|
+
providers: [
|
|
33
|
+
{ provide: UserService, useValue: mockUserService },
|
|
34
|
+
{ provide: ConfigService, useValue: mockConfigService },
|
|
35
|
+
{ provide: WINSTON_MODULE_PROVIDER, useValue: mockLogger },
|
|
36
|
+
],
|
|
37
|
+
})
|
|
38
|
+
// 兜底:自动 mock 任何未显式提供的依赖
|
|
39
|
+
// (例如 ACGuard 内部的 __roles_builder__ token)
|
|
40
|
+
.useMocker(() => ({}))
|
|
41
|
+
.compile();
|
|
13
42
|
|
|
14
43
|
controller = module.get<UserController>(UserController);
|
|
15
44
|
});
|