Merge pull request #2056 from automatisch/rest-create-flow

feat: Implement create flow rest API endpoint
This commit is contained in:
Ömer Faruk Aydın
2024-09-02 15:21:46 +03:00
committed by GitHub
7 changed files with 110 additions and 1 deletions

View 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 });
};

View File

@@ -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);
});
});

View File

@@ -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,

View File

@@ -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',

View File

@@ -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);

View File

@@ -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;

View 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;