refactor(core): Use DI in execution services (no-changelog) (#8358)

This commit is contained in:
Iván Ovejero 2024-01-17 15:42:19 +01:00 committed by GitHub
parent 01280815c9
commit 2eb829a6b4
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
9 changed files with 101 additions and 106 deletions

View file

@ -35,7 +35,7 @@ import * as WorkflowExecuteAdditionalData from '@/WorkflowExecuteAdditionalData'
import type { WorkflowEntity } from '@db/entities/WorkflowEntity'; import type { WorkflowEntity } from '@db/entities/WorkflowEntity';
import { ActiveExecutions } from '@/ActiveExecutions'; import { ActiveExecutions } from '@/ActiveExecutions';
import { ExecutionsService } from './executions/executions.service'; import { ExecutionService } from './executions/execution.service';
import { import {
STARTING_NODES, STARTING_NODES,
WORKFLOW_REACTIVATE_INITIAL_TIMEOUT, WORKFLOW_REACTIVATE_INITIAL_TIMEOUT,
@ -74,7 +74,7 @@ export class ActiveWorkflowRunner {
private readonly workflowRepository: WorkflowRepository, private readonly workflowRepository: WorkflowRepository,
private readonly multiMainSetup: MultiMainSetup, private readonly multiMainSetup: MultiMainSetup,
private readonly activationErrorsService: ActivationErrorsService, private readonly activationErrorsService: ActivationErrorsService,
private readonly executionService: ExecutionsService, private readonly executionService: ExecutionService,
private readonly workflowStaticDataService: WorkflowStaticDataService, private readonly workflowStaticDataService: WorkflowStaticDataService,
private readonly activeWorkflowsService: ActiveWorkflowsService, private readonly activeWorkflowsService: ActiveWorkflowsService,
) {} ) {}

View file

@ -1,41 +1,38 @@
import Container from 'typedi'; import { ExecutionService } from './execution.service';
import type { User } from '@db/entities/User';
import { ExecutionsService } from './executions.service';
import type { ExecutionRequest } from './execution.request'; import type { ExecutionRequest } from './execution.request';
import type { IExecutionResponse, IExecutionFlattedResponse } from '@/Interfaces'; import type { IExecutionResponse, IExecutionFlattedResponse } from '@/Interfaces';
import { EnterpriseWorkflowService } from '../workflows/workflow.service.ee'; import { EnterpriseWorkflowService } from '../workflows/workflow.service.ee';
import type { WorkflowWithSharingsAndCredentials } from '@/workflows/workflows.types'; import type { WorkflowWithSharingsAndCredentials } from '@/workflows/workflows.types';
import { WorkflowRepository } from '@/databases/repositories/workflow.repository'; import { WorkflowRepository } from '@/databases/repositories/workflow.repository';
import { WorkflowSharingService } from '@/workflows/workflowSharing.service'; import { Service } from 'typedi';
export class EnterpriseExecutionsService extends ExecutionsService { @Service()
/** export class EnterpriseExecutionsService {
* Function to get the workflow Ids for a User regardless of role constructor(
*/ private readonly executionService: ExecutionService,
static async getWorkflowIdsForUser(user: User): Promise<string[]> { private readonly workflowRepository: WorkflowRepository,
// Get all workflows private readonly enterpriseWorkflowService: EnterpriseWorkflowService,
return Container.get(WorkflowSharingService).getSharedWorkflowIds(user); ) {}
}
static async getExecution( async getExecution(
req: ExecutionRequest.Get, req: ExecutionRequest.Get,
sharedWorkflowIds: string[],
): Promise<IExecutionResponse | IExecutionFlattedResponse | undefined> { ): Promise<IExecutionResponse | IExecutionFlattedResponse | undefined> {
const execution = await super.getExecution(req); const execution = await this.executionService.getExecution(req, sharedWorkflowIds);
if (!execution) return; if (!execution) return;
const relations = ['shared', 'shared.user', 'shared.role']; const relations = ['shared', 'shared.user', 'shared.role'];
const workflow = (await Container.get(WorkflowRepository).get( const workflow = (await this.workflowRepository.get(
{ id: execution.workflowId }, { id: execution.workflowId },
{ relations }, { relations },
)) as WorkflowWithSharingsAndCredentials; )) as WorkflowWithSharingsAndCredentials;
if (!workflow) return; if (!workflow) return;
const enterpriseWorkflowService = Container.get(EnterpriseWorkflowService); this.enterpriseWorkflowService.addOwnerAndSharings(workflow);
await this.enterpriseWorkflowService.addCredentialsToWorkflow(workflow, req.user);
enterpriseWorkflowService.addOwnerAndSharings(workflow);
await enterpriseWorkflowService.addCredentialsToWorkflow(workflow, req.user);
execution.workflowData = { execution.workflowData = {
...execution.workflowData, ...execution.workflowData,

View file

@ -1,4 +1,4 @@
import { Container, Service } from 'typedi'; import { Service } from 'typedi';
import { validate as jsonSchemaValidate } from 'jsonschema'; import { validate as jsonSchemaValidate } from 'jsonschema';
import type { import type {
IWorkflowBase, IWorkflowBase,
@ -12,12 +12,10 @@ import { ApplicationError, jsonParse, Workflow, WorkflowOperationError } from 'n
import { ActiveExecutions } from '@/ActiveExecutions'; import { ActiveExecutions } from '@/ActiveExecutions';
import config from '@/config'; import config from '@/config';
import type { User } from '@db/entities/User';
import type { import type {
ExecutionPayload, ExecutionPayload,
IExecutionFlattedResponse, IExecutionFlattedResponse,
IExecutionResponse, IExecutionResponse,
IExecutionsListResponse,
IWorkflowDb, IWorkflowDb,
IWorkflowExecutionDataProcess, IWorkflowExecutionDataProcess,
} from '@/Interfaces'; } from '@/Interfaces';
@ -33,7 +31,6 @@ import { WorkflowRepository } from '@db/repositories/workflow.repository';
import { Logger } from '@/Logger'; import { Logger } from '@/Logger';
import { InternalServerError } from '@/errors/response-errors/internal-server.error'; import { InternalServerError } from '@/errors/response-errors/internal-server.error';
import { NotFoundError } from '@/errors/response-errors/not-found.error'; import { NotFoundError } from '@/errors/response-errors/not-found.error';
import { WorkflowSharingService } from '@/workflows/workflowSharing.service';
const schemaGetExecutionsQueryFilter = { const schemaGetExecutionsQueryFilter = {
$id: '/IGetExecutionsQueryFilter', $id: '/IGetExecutionsQueryFilter',
@ -71,21 +68,18 @@ const schemaGetExecutionsQueryFilter = {
const allowedExecutionsQueryFilterFields = Object.keys(schemaGetExecutionsQueryFilter.properties); const allowedExecutionsQueryFilterFields = Object.keys(schemaGetExecutionsQueryFilter.properties);
@Service() @Service()
export class ExecutionsService { export class ExecutionService {
/** constructor(
* Function to get the workflow Ids for a User private readonly logger: Logger,
* Overridden in EE version to ignore roles private readonly queue: Queue,
*/ private readonly activeExecutions: ActiveExecutions,
static async getWorkflowIdsForUser(user: User): Promise<string[]> { private readonly executionRepository: ExecutionRepository,
// Get all workflows using owner role private readonly workflowRepository: WorkflowRepository,
return Container.get(WorkflowSharingService).getSharedWorkflowIds(user, ['owner']); private readonly nodeTypes: NodeTypes,
} ) {}
static async getExecutionsList(req: ExecutionRequest.GetAll): Promise<IExecutionsListResponse> { async getExecutionsList(req: ExecutionRequest.GetAll, sharedWorkflowIds: string[]) {
const sharedWorkflowIds = await this.getWorkflowIdsForUser(req.user);
if (sharedWorkflowIds.length === 0) { if (sharedWorkflowIds.length === 0) {
// return early since without shared workflows there can be no hits
// (note: getSharedWorkflowIds() returns _all_ workflow ids for global owners)
return { return {
count: 0, count: 0,
estimated: false, estimated: false,
@ -107,7 +101,7 @@ export class ExecutionsService {
} }
} }
} catch (error) { } catch (error) {
Container.get(Logger).error('Failed to parse filter', { this.logger.error('Failed to parse filter', {
userId: req.user.id, userId: req.user.id,
filter: req.query.filter, filter: req.query.filter,
}); });
@ -118,7 +112,7 @@ export class ExecutionsService {
// safeguard against querying workflowIds not shared with the user // safeguard against querying workflowIds not shared with the user
const workflowId = filter?.workflowId?.toString(); const workflowId = filter?.workflowId?.toString();
if (workflowId !== undefined && !sharedWorkflowIds.includes(workflowId)) { if (workflowId !== undefined && !sharedWorkflowIds.includes(workflowId)) {
Container.get(Logger).verbose( this.logger.verbose(
`User ${req.user.id} attempted to query non-shared workflow ${workflowId}`, `User ${req.user.id} attempted to query non-shared workflow ${workflowId}`,
); );
return { return {
@ -135,26 +129,21 @@ export class ExecutionsService {
const executingWorkflowIds: string[] = []; const executingWorkflowIds: string[] = [];
if (config.getEnv('executions.mode') === 'queue') { if (config.getEnv('executions.mode') === 'queue') {
const queue = Container.get(Queue); const currentJobs = await this.queue.getJobs(['active', 'waiting']);
const currentJobs = await queue.getJobs(['active', 'waiting']);
executingWorkflowIds.push(...currentJobs.map(({ data }) => data.executionId)); executingWorkflowIds.push(...currentJobs.map(({ data }) => data.executionId));
} }
// We may have manual executions even with queue so we must account for these. // We may have manual executions even with queue so we must account for these.
executingWorkflowIds.push( executingWorkflowIds.push(...this.activeExecutions.getActiveExecutions().map(({ id }) => id));
...Container.get(ActiveExecutions)
.getActiveExecutions()
.map(({ id }) => id),
);
const { count, estimated } = await Container.get(ExecutionRepository).countExecutions( const { count, estimated } = await this.executionRepository.countExecutions(
filter, filter,
sharedWorkflowIds, sharedWorkflowIds,
executingWorkflowIds, executingWorkflowIds,
req.user.hasGlobalScope('workflow:list'), req.user.hasGlobalScope('workflow:list'),
); );
const formattedExecutions = await Container.get(ExecutionRepository).searchExecutions( const formattedExecutions = await this.executionRepository.searchExecutions(
filter, filter,
limit, limit,
executingWorkflowIds, executingWorkflowIds,
@ -171,26 +160,20 @@ export class ExecutionsService {
}; };
} }
static async getExecution( async getExecution(
req: ExecutionRequest.Get, req: ExecutionRequest.Get,
sharedWorkflowIds: string[],
): Promise<IExecutionResponse | IExecutionFlattedResponse | undefined> { ): Promise<IExecutionResponse | IExecutionFlattedResponse | undefined> {
const sharedWorkflowIds = await this.getWorkflowIdsForUser(req.user);
if (!sharedWorkflowIds.length) return undefined; if (!sharedWorkflowIds.length) return undefined;
const { id: executionId } = req.params; const { id: executionId } = req.params;
const execution = await Container.get(ExecutionRepository).findIfShared( const execution = await this.executionRepository.findIfShared(executionId, sharedWorkflowIds);
executionId,
sharedWorkflowIds,
);
if (!execution) { if (!execution) {
Container.get(Logger).info( this.logger.info('Attempt to read execution was blocked due to insufficient permissions', {
'Attempt to read execution was blocked due to insufficient permissions',
{
userId: req.user.id, userId: req.user.id,
executionId, executionId,
}, });
);
return undefined; return undefined;
} }
@ -201,18 +184,17 @@ export class ExecutionsService {
return execution; return execution;
} }
static async retryExecution(req: ExecutionRequest.Retry): Promise<boolean> { async retryExecution(req: ExecutionRequest.Retry, sharedWorkflowIds: string[]) {
const sharedWorkflowIds = await this.getWorkflowIdsForUser(req.user);
if (!sharedWorkflowIds.length) return false; if (!sharedWorkflowIds.length) return false;
const { id: executionId } = req.params; const { id: executionId } = req.params;
const execution = (await Container.get(ExecutionRepository).findIfShared( const execution = (await this.executionRepository.findIfShared(
executionId, executionId,
sharedWorkflowIds, sharedWorkflowIds,
)) as unknown as IExecutionResponse; )) as unknown as IExecutionResponse;
if (!execution) { if (!execution) {
Container.get(Logger).info( this.logger.info(
'Attempt to retry an execution was blocked due to insufficient permissions', 'Attempt to retry an execution was blocked due to insufficient permissions',
{ {
userId: req.user.id, userId: req.user.id,
@ -260,7 +242,7 @@ export class ExecutionsService {
// Loads the currently saved workflow to execute instead of the // Loads the currently saved workflow to execute instead of the
// one saved at the time of the execution. // one saved at the time of the execution.
const workflowId = execution.workflowData.id; const workflowId = execution.workflowData.id;
const workflowData = (await Container.get(WorkflowRepository).findOneBy({ const workflowData = (await this.workflowRepository.findOneBy({
id: workflowId, id: workflowId,
})) as IWorkflowBase; })) as IWorkflowBase;
@ -272,14 +254,14 @@ export class ExecutionsService {
} }
data.workflowData = workflowData; data.workflowData = workflowData;
const nodeTypes = Container.get(NodeTypes);
const workflowInstance = new Workflow({ const workflowInstance = new Workflow({
id: workflowData.id, id: workflowData.id,
name: workflowData.name, name: workflowData.name,
nodes: workflowData.nodes, nodes: workflowData.nodes,
connections: workflowData.connections, connections: workflowData.connections,
active: false, active: false,
nodeTypes, nodeTypes: this.nodeTypes,
staticData: undefined, staticData: undefined,
settings: workflowData.settings, settings: workflowData.settings,
}); });
@ -289,14 +271,11 @@ export class ExecutionsService {
// Find the data of the last executed node in the new workflow // Find the data of the last executed node in the new workflow
const node = workflowInstance.getNode(stack.node.name); const node = workflowInstance.getNode(stack.node.name);
if (node === null) { if (node === null) {
Container.get(Logger).error( this.logger.error('Failed to retry an execution because a node could not be found', {
'Failed to retry an execution because a node could not be found',
{
userId: req.user.id, userId: req.user.id,
executionId, executionId,
nodeName: stack.node.name, nodeName: stack.node.name,
}, });
);
throw new WorkflowOperationError( throw new WorkflowOperationError(
`Could not find the node "${stack.node.name}" in workflow. It probably got deleted or renamed. Without it the workflow can sadly not be retried.`, `Could not find the node "${stack.node.name}" in workflow. It probably got deleted or renamed. Without it the workflow can sadly not be retried.`,
); );
@ -310,8 +289,7 @@ export class ExecutionsService {
const workflowRunner = new WorkflowRunner(); const workflowRunner = new WorkflowRunner();
const retriedExecutionId = await workflowRunner.run(data); const retriedExecutionId = await workflowRunner.run(data);
const executionData = const executionData = await this.activeExecutions.getPostExecutePromise(retriedExecutionId);
await Container.get(ActiveExecutions).getPostExecutePromise(retriedExecutionId);
if (!executionData) { if (!executionData) {
throw new ApplicationError('The retry did not start for an unknown reason.'); throw new ApplicationError('The retry did not start for an unknown reason.');
@ -320,8 +298,7 @@ export class ExecutionsService {
return !!executionData.finished; return !!executionData.finished;
} }
static async deleteExecutions(req: ExecutionRequest.Delete): Promise<void> { async deleteExecutions(req: ExecutionRequest.Delete, sharedWorkflowIds: string[]) {
const sharedWorkflowIds = await this.getWorkflowIdsForUser(req.user);
if (sharedWorkflowIds.length === 0) { if (sharedWorkflowIds.length === 0) {
// return early since without shared workflows there can be no hits // return early since without shared workflows there can be no hits
// (note: getSharedWorkflowIds() returns _all_ workflow ids for global owners) // (note: getSharedWorkflowIds() returns _all_ workflow ids for global owners)
@ -342,14 +319,10 @@ export class ExecutionsService {
} }
} }
return Container.get(ExecutionRepository).deleteExecutionsByFilter( return this.executionRepository.deleteExecutionsByFilter(requestFilters, sharedWorkflowIds, {
requestFilters,
sharedWorkflowIds,
{
deleteBefore, deleteBefore,
ids, ids,
}, });
);
} }
async createErrorExecution( async createErrorExecution(
@ -358,7 +331,7 @@ export class ExecutionsService {
workflowData: IWorkflowDb, workflowData: IWorkflowDb,
workflow: Workflow, workflow: Workflow,
mode: WorkflowExecuteMode, mode: WorkflowExecuteMode,
): Promise<void> { ) {
const saveDataErrorExecutionDisabled = const saveDataErrorExecutionDisabled =
workflowData?.settings?.saveDataErrorExecution === 'none'; workflowData?.settings?.saveDataErrorExecution === 'none';
@ -420,6 +393,6 @@ export class ExecutionsService {
status: 'error', status: 'error',
}; };
await Container.get(ExecutionRepository).createNewExecution(fullExecutionData); await this.executionRepository.createNewExecution(fullExecutionData);
} }
} }

View file

@ -1,31 +1,53 @@
import { ExecutionRequest } from './execution.request'; import { ExecutionRequest } from './execution.request';
import { ExecutionsService } from './executions.service'; import { ExecutionService } from './execution.service';
import { Authorized, Get, Post, RestController } from '@/decorators'; import { Authorized, Get, Post, RestController } from '@/decorators';
import { EnterpriseExecutionsService } from './executions.service.ee'; import { EnterpriseExecutionsService } from './execution.service.ee';
import { isSharingEnabled } from '@/UserManagement/UserManagementHelper'; import { isSharingEnabled } from '@/UserManagement/UserManagementHelper';
import { WorkflowSharingService } from '@/workflows/workflowSharing.service';
import type { User } from '@/databases/entities/User';
@Authorized() @Authorized()
@RestController('/executions') @RestController('/executions')
export class ExecutionsController { export class ExecutionsController {
constructor(
private readonly executionService: ExecutionService,
private readonly enterpriseExecutionService: EnterpriseExecutionsService,
private readonly workflowSharingService: WorkflowSharingService,
) {}
private async getAccessibleWorkflowIds(user: User) {
return isSharingEnabled()
? this.workflowSharingService.getSharedWorkflowIds(user)
: this.workflowSharingService.getSharedWorkflowIds(user, ['owner']);
}
@Get('/') @Get('/')
async getExecutionsList(req: ExecutionRequest.GetAll) { async getExecutionsList(req: ExecutionRequest.GetAll) {
return ExecutionsService.getExecutionsList(req); const workflowIds = await this.getAccessibleWorkflowIds(req.user);
return this.executionService.getExecutionsList(req, workflowIds);
} }
@Get('/:id') @Get('/:id')
async getExecution(req: ExecutionRequest.Get) { async getExecution(req: ExecutionRequest.Get) {
const workflowIds = await this.getAccessibleWorkflowIds(req.user);
return isSharingEnabled() return isSharingEnabled()
? EnterpriseExecutionsService.getExecution(req) ? this.enterpriseExecutionService.getExecution(req, workflowIds)
: ExecutionsService.getExecution(req); : this.executionService.getExecution(req, workflowIds);
} }
@Post('/:id/retry') @Post('/:id/retry')
async retryExecution(req: ExecutionRequest.Retry) { async retryExecution(req: ExecutionRequest.Retry) {
return ExecutionsService.retryExecution(req); const workflowIds = await this.getAccessibleWorkflowIds(req.user);
return this.executionService.retryExecution(req, workflowIds);
} }
@Post('/delete') @Post('/delete')
async deleteExecutions(req: ExecutionRequest.Delete) { async deleteExecutions(req: ExecutionRequest.Delete) {
return ExecutionsService.deleteExecutions(req); const workflowIds = await this.getAccessibleWorkflowIds(req.user);
return this.executionService.deleteExecutions(req, workflowIds);
} }
} }

View file

@ -23,14 +23,14 @@ import { setSchedulerAsLoadedNode } from './shared/utils';
import * as testDb from './shared/testDb'; import * as testDb from './shared/testDb';
import { createOwner } from './shared/db/users'; import { createOwner } from './shared/db/users';
import { createWorkflow } from './shared/db/workflows'; import { createWorkflow } from './shared/db/workflows';
import { ExecutionsService } from '@/executions/executions.service'; import { ExecutionService } from '@/executions/execution.service';
import { WorkflowService } from '@/workflows/workflow.service'; import { WorkflowService } from '@/workflows/workflow.service';
import { ActiveWorkflowsService } from '@/services/activeWorkflows.service'; import { ActiveWorkflowsService } from '@/services/activeWorkflows.service';
mockInstance(ActiveExecutions); mockInstance(ActiveExecutions);
mockInstance(Push); mockInstance(Push);
mockInstance(SecretsHelper); mockInstance(SecretsHelper);
mockInstance(ExecutionsService); mockInstance(ExecutionService);
mockInstance(WorkflowService); mockInstance(WorkflowService);
const webhookService = mockInstance(WebhookService); const webhookService = mockInstance(WebhookService);

View file

@ -6,6 +6,9 @@ import { createWorkflow } from './shared/db/workflows';
import * as testDb from './shared/testDb'; import * as testDb from './shared/testDb';
import { setupTestServer } from './shared/utils'; import { setupTestServer } from './shared/utils';
import { mockInstance } from '../shared/mocking'; import { mockInstance } from '../shared/mocking';
import { EnterpriseExecutionsService } from '@/executions/execution.service.ee';
mockInstance(EnterpriseExecutionsService);
mockInstance(Push); mockInstance(Push);
let testServer = setupTestServer({ endpointGroups: ['executions'] }); let testServer = setupTestServer({ endpointGroups: ['executions'] });

View file

@ -18,7 +18,7 @@ import { createWorkflow, createWorkflowWithTrigger } from '../shared/db/workflow
import { createTag } from '../shared/db/tags'; import { createTag } from '../shared/db/tags';
import { mockInstance } from '../../shared/mocking'; import { mockInstance } from '../../shared/mocking';
import { Push } from '@/push'; import { Push } from '@/push';
import { ExecutionsService } from '@/executions/executions.service'; import { ExecutionService } from '@/executions/execution.service';
let workflowOwnerRole: Role; let workflowOwnerRole: Role;
let owner: User; let owner: User;
@ -31,7 +31,7 @@ const testServer = utils.setupTestServer({ endpointGroups: ['publicApi'] });
const license = testServer.license; const license = testServer.license;
mockInstance(Push); mockInstance(Push);
mockInstance(ExecutionsService); mockInstance(ExecutionService);
beforeAll(async () => { beforeAll(async () => {
const [globalOwnerRole, globalMemberRole, fetchedWorkflowOwnerRole] = await getAllRoles(); const [globalOwnerRole, globalMemberRole, fetchedWorkflowOwnerRole] = await getAllRoles();

View file

@ -18,7 +18,7 @@ import { SettingsRepository } from '@db/repositories/settings.repository';
import { mockNodeTypesData } from '../../../unit/Helpers'; import { mockNodeTypesData } from '../../../unit/Helpers';
import { MultiMainSetup } from '@/services/orchestration/main/MultiMainSetup.ee'; import { MultiMainSetup } from '@/services/orchestration/main/MultiMainSetup.ee';
import { mockInstance } from '../../../shared/mocking'; import { mockInstance } from '../../../shared/mocking';
import { ExecutionsService } from '@/executions/executions.service'; import { ExecutionService } from '@/executions/execution.service';
export { setupTestServer } from './testServer'; export { setupTestServer } from './testServer';
@ -32,7 +32,7 @@ export { setupTestServer } from './testServer';
export async function initActiveWorkflowRunner() { export async function initActiveWorkflowRunner() {
mockInstance(MultiMainSetup); mockInstance(MultiMainSetup);
mockInstance(ExecutionsService); mockInstance(ExecutionService);
const { ActiveWorkflowRunner } = await import('@/ActiveWorkflowRunner'); const { ActiveWorkflowRunner } = await import('@/ActiveWorkflowRunner');
const workflowRunner = Container.get(ActiveWorkflowRunner); const workflowRunner = Container.get(ActiveWorkflowRunner);
await workflowRunner.init(); await workflowRunner.init();

View file

@ -18,10 +18,10 @@ import * as testDb from './shared/testDb';
import type { SuperAgentTest } from 'supertest'; import type { SuperAgentTest } from 'supertest';
import type { Role } from '@db/entities/Role'; import type { Role } from '@db/entities/Role';
import type { User } from '@db/entities/User'; import type { User } from '@db/entities/User';
import { ExecutionsService } from '@/executions/executions.service'; import { ExecutionService } from '@/executions/execution.service';
import { mockInstance } from '../shared/mocking'; import { mockInstance } from '../shared/mocking';
mockInstance(ExecutionsService); mockInstance(ExecutionService);
const testServer = utils.setupTestServer({ const testServer = utils.setupTestServer({
endpointGroups: ['users'], endpointGroups: ['users'],