Merge pull request #2056 from automatisch/rest-create-flow
feat: Implement create flow rest API endpoint
This commit is contained in:
11
packages/backend/src/controllers/api/v1/flows/create-flow.js
Normal file
11
packages/backend/src/controllers/api/v1/flows/create-flow.js
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
import { renderObject } from '../../../../helpers/renderer.js';
|
||||||
|
|
||||||
|
export default async (request, response) => {
|
||||||
|
let flow = await request.currentUser.$relatedQuery('flows').insert({
|
||||||
|
name: 'Name your flow',
|
||||||
|
});
|
||||||
|
|
||||||
|
flow = await flow.createInitialSteps();
|
||||||
|
|
||||||
|
renderObject(response, flow, { status: 201 });
|
||||||
|
};
|
@@ -0,0 +1,41 @@
|
|||||||
|
import { describe, it, expect, beforeEach } from 'vitest';
|
||||||
|
import request from 'supertest';
|
||||||
|
|
||||||
|
import app from '../../../../app.js';
|
||||||
|
import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id.js';
|
||||||
|
import { createUser } from '../../../../../test/factories/user.js';
|
||||||
|
import createFlowMock from '../../../../../test/mocks/rest/api/v1/flows/create-flow.js';
|
||||||
|
import { createPermission } from '../../../../../test/factories/permission.js';
|
||||||
|
|
||||||
|
describe('POST /api/v1/flows', () => {
|
||||||
|
let currentUser, currentUserRole, token;
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
currentUser = await createUser();
|
||||||
|
currentUserRole = await currentUser.$relatedQuery('role');
|
||||||
|
|
||||||
|
token = await createAuthTokenByUserId(currentUser.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return created flow', async () => {
|
||||||
|
await createPermission({
|
||||||
|
action: 'create',
|
||||||
|
subject: 'Flow',
|
||||||
|
roleId: currentUserRole.id,
|
||||||
|
conditions: ['isCreator'],
|
||||||
|
});
|
||||||
|
|
||||||
|
const response = await request(app)
|
||||||
|
.post('/api/v1/flows')
|
||||||
|
.set('Authorization', token)
|
||||||
|
.expect(201);
|
||||||
|
|
||||||
|
const refetchedFlow = await currentUser
|
||||||
|
.$relatedQuery('flows')
|
||||||
|
.findById(response.body.data.id);
|
||||||
|
|
||||||
|
const expectedPayload = await createFlowMock(refetchedFlow);
|
||||||
|
|
||||||
|
expect(response.body).toMatchObject(expectedPayload);
|
||||||
|
});
|
||||||
|
});
|
@@ -1,5 +1,4 @@
|
|||||||
import createConnection from './mutations/create-connection.js';
|
import createConnection from './mutations/create-connection.js';
|
||||||
import createFlow from './mutations/create-flow.js';
|
|
||||||
import createRole from './mutations/create-role.ee.js';
|
import createRole from './mutations/create-role.ee.js';
|
||||||
import createStep from './mutations/create-step.js';
|
import createStep from './mutations/create-step.js';
|
||||||
import createUser from './mutations/create-user.ee.js';
|
import createUser from './mutations/create-user.ee.js';
|
||||||
@@ -23,6 +22,7 @@ import upsertSamlAuthProvidersRoleMappings from './mutations/upsert-saml-auth-pr
|
|||||||
import updateUser from './mutations/update-user.ee.js';
|
import updateUser from './mutations/update-user.ee.js';
|
||||||
import deleteStep from './mutations/delete-step.js';
|
import deleteStep from './mutations/delete-step.js';
|
||||||
import verifyConnection from './mutations/verify-connection.js';
|
import verifyConnection from './mutations/verify-connection.js';
|
||||||
|
import createFlow from './mutations/create-flow.js';
|
||||||
|
|
||||||
const mutationResolvers = {
|
const mutationResolvers = {
|
||||||
createConnection,
|
createConnection,
|
||||||
|
@@ -21,6 +21,10 @@ const authorizationList = {
|
|||||||
action: 'read',
|
action: 'read',
|
||||||
subject: 'Flow',
|
subject: 'Flow',
|
||||||
},
|
},
|
||||||
|
'POST /api/v1/flows/': {
|
||||||
|
action: 'create',
|
||||||
|
subject: 'Flow',
|
||||||
|
},
|
||||||
'GET /api/v1/steps/:stepId/connection': {
|
'GET /api/v1/steps/:stepId/connection': {
|
||||||
action: 'read',
|
action: 'read',
|
||||||
subject: 'Flow',
|
subject: 'Flow',
|
||||||
|
@@ -119,6 +119,22 @@ class Flow extends Base {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async createInitialSteps() {
|
||||||
|
await Step.query().insert({
|
||||||
|
flowId: this.id,
|
||||||
|
type: 'trigger',
|
||||||
|
position: 1,
|
||||||
|
});
|
||||||
|
|
||||||
|
await Step.query().insert({
|
||||||
|
flowId: this.id,
|
||||||
|
type: 'action',
|
||||||
|
position: 2,
|
||||||
|
});
|
||||||
|
|
||||||
|
return this.$query().withGraphFetched('steps');
|
||||||
|
}
|
||||||
|
|
||||||
async $beforeUpdate(opt, queryContext) {
|
async $beforeUpdate(opt, queryContext) {
|
||||||
await super.$beforeUpdate(opt, queryContext);
|
await super.$beforeUpdate(opt, queryContext);
|
||||||
|
|
||||||
|
@@ -4,11 +4,13 @@ import { authorizeUser } from '../../../helpers/authorization.js';
|
|||||||
import getFlowsAction from '../../../controllers/api/v1/flows/get-flows.js';
|
import getFlowsAction from '../../../controllers/api/v1/flows/get-flows.js';
|
||||||
import getFlowAction from '../../../controllers/api/v1/flows/get-flow.js';
|
import getFlowAction from '../../../controllers/api/v1/flows/get-flow.js';
|
||||||
import updateFlowAction from '../../../controllers/api/v1/flows/update-flow.js';
|
import updateFlowAction from '../../../controllers/api/v1/flows/update-flow.js';
|
||||||
|
import createFlowAction from '../../../controllers/api/v1/flows/create-flow.js';
|
||||||
|
|
||||||
const router = Router();
|
const router = Router();
|
||||||
|
|
||||||
router.get('/', authenticateUser, authorizeUser, getFlowsAction);
|
router.get('/', authenticateUser, authorizeUser, getFlowsAction);
|
||||||
router.get('/:flowId', authenticateUser, authorizeUser, getFlowAction);
|
router.get('/:flowId', authenticateUser, authorizeUser, getFlowAction);
|
||||||
|
router.post('/', authenticateUser, authorizeUser, createFlowAction);
|
||||||
router.patch('/:flowId', authenticateUser, authorizeUser, updateFlowAction);
|
router.patch('/:flowId', authenticateUser, authorizeUser, updateFlowAction);
|
||||||
|
|
||||||
export default router;
|
export default router;
|
||||||
|
35
packages/backend/test/mocks/rest/api/v1/flows/create-flow.js
Normal file
35
packages/backend/test/mocks/rest/api/v1/flows/create-flow.js
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
const createFlowMock = async (flow) => {
|
||||||
|
const data = {
|
||||||
|
id: flow.id,
|
||||||
|
active: flow.active,
|
||||||
|
name: flow.name,
|
||||||
|
status: flow.status,
|
||||||
|
createdAt: flow.createdAt.getTime(),
|
||||||
|
updatedAt: flow.updatedAt.getTime(),
|
||||||
|
steps: [
|
||||||
|
{
|
||||||
|
position: 1,
|
||||||
|
status: 'incomplete',
|
||||||
|
type: 'trigger',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
position: 2,
|
||||||
|
status: 'incomplete',
|
||||||
|
type: 'action',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
data: data,
|
||||||
|
meta: {
|
||||||
|
count: 1,
|
||||||
|
currentPage: null,
|
||||||
|
isArray: false,
|
||||||
|
totalPages: null,
|
||||||
|
type: 'Flow',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default createFlowMock;
|
Reference in New Issue
Block a user