mirror of
https://github.com/n8n-io/n8n.git
synced 2024-12-31 15:37:26 -08:00
386 lines
13 KiB
TypeScript
386 lines
13 KiB
TypeScript
import { Container } from 'typedi';
|
|
|
|
import type { Variables } from '@/databases/entities/variables';
|
|
import { VariablesRepository } from '@/databases/repositories/variables.repository';
|
|
import { generateNanoId } from '@/databases/utils/generators';
|
|
import { VariablesService } from '@/environments/variables/variables.service.ee';
|
|
import { CacheService } from '@/services/cache/cache.service';
|
|
|
|
import { createOwner, createUser } from './shared/db/users';
|
|
import * as testDb from './shared/test-db';
|
|
import type { SuperAgentTest } from './shared/types';
|
|
import * as utils from './shared/utils/';
|
|
|
|
let authOwnerAgent: SuperAgentTest;
|
|
let authMemberAgent: SuperAgentTest;
|
|
|
|
const testServer = utils.setupTestServer({ endpointGroups: ['variables'] });
|
|
const license = testServer.license;
|
|
|
|
async function createVariable(key: string, value: string) {
|
|
const result = await Container.get(VariablesRepository).save({
|
|
id: generateNanoId(),
|
|
key,
|
|
value,
|
|
});
|
|
await Container.get(VariablesService).updateCache();
|
|
return result;
|
|
}
|
|
|
|
async function getVariableByKey(key: string) {
|
|
return await Container.get(VariablesRepository).findOne({
|
|
where: {
|
|
key,
|
|
},
|
|
});
|
|
}
|
|
|
|
async function getVariableById(id: string) {
|
|
return await Container.get(VariablesRepository).findOne({
|
|
where: {
|
|
id,
|
|
},
|
|
});
|
|
}
|
|
|
|
beforeAll(async () => {
|
|
const owner = await createOwner();
|
|
authOwnerAgent = testServer.authAgentFor(owner);
|
|
const member = await createUser();
|
|
authMemberAgent = testServer.authAgentFor(member);
|
|
|
|
license.setDefaults({
|
|
features: ['feat:variables'],
|
|
// quota: {
|
|
// 'quota:maxVariables': -1,
|
|
// },
|
|
});
|
|
});
|
|
|
|
beforeEach(async () => {
|
|
await testDb.truncate(['Variables']);
|
|
});
|
|
|
|
// ----------------------------------------
|
|
// GET /variables - fetch all variables
|
|
// ----------------------------------------
|
|
describe('GET /variables', () => {
|
|
beforeEach(async () => {
|
|
await Promise.all([
|
|
createVariable('test1', 'value1'),
|
|
createVariable('test2', 'value2'),
|
|
createVariable('empty', ''),
|
|
]);
|
|
});
|
|
|
|
test('should return an empty array if there is nothing in the cache', async () => {
|
|
const cacheService = Container.get(CacheService);
|
|
const spy = jest.spyOn(cacheService, 'get').mockResolvedValueOnce(undefined);
|
|
const response = await authOwnerAgent.get('/variables');
|
|
expect(spy).toHaveBeenCalledTimes(1);
|
|
expect(response.statusCode).toBe(200);
|
|
expect(response.body.data.length).toBe(0);
|
|
});
|
|
|
|
test('should return all variables for an owner', async () => {
|
|
const response = await authOwnerAgent.get('/variables');
|
|
expect(response.statusCode).toBe(200);
|
|
expect(response.body.data.length).toBe(3);
|
|
});
|
|
|
|
test('should return all variables for a member', async () => {
|
|
const response = await authMemberAgent.get('/variables');
|
|
expect(response.statusCode).toBe(200);
|
|
expect(response.body.data.length).toBe(3);
|
|
});
|
|
|
|
describe('state:empty', () => {
|
|
test('only return empty variables', async () => {
|
|
const response = await authOwnerAgent.get('/variables').query({ state: 'empty' });
|
|
expect(response.statusCode).toBe(200);
|
|
expect(response.body.data.length).toBe(1);
|
|
expect(response.body.data[0]).toMatchObject({ key: 'empty', value: '', type: 'string' });
|
|
});
|
|
});
|
|
});
|
|
|
|
// ----------------------------------------
|
|
// GET /variables/:id - get a single variable
|
|
// ----------------------------------------
|
|
describe('GET /variables/:id', () => {
|
|
let var1: Variables, var2: Variables;
|
|
beforeEach(async () => {
|
|
[var1, var2] = await Promise.all([
|
|
createVariable('test1', 'value1'),
|
|
createVariable('test2', 'value2'),
|
|
]);
|
|
});
|
|
|
|
test('should return a single variable for an owner', async () => {
|
|
const response1 = await authOwnerAgent.get(`/variables/${var1.id}`);
|
|
expect(response1.statusCode).toBe(200);
|
|
expect(response1.body.data.key).toBe('test1');
|
|
|
|
const response2 = await authOwnerAgent.get(`/variables/${var2.id}`);
|
|
expect(response2.statusCode).toBe(200);
|
|
expect(response2.body.data.key).toBe('test2');
|
|
});
|
|
|
|
test('should return a single variable for a member', async () => {
|
|
const response1 = await authMemberAgent.get(`/variables/${var1.id}`);
|
|
expect(response1.statusCode).toBe(200);
|
|
expect(response1.body.data.key).toBe('test1');
|
|
|
|
const response2 = await authMemberAgent.get(`/variables/${var2.id}`);
|
|
expect(response2.statusCode).toBe(200);
|
|
expect(response2.body.data.key).toBe('test2');
|
|
});
|
|
});
|
|
|
|
// ----------------------------------------
|
|
// POST /variables - create a new variable
|
|
// ----------------------------------------
|
|
describe('POST /variables', () => {
|
|
const generatePayload = (i = 1) => ({
|
|
key: `create${i}`,
|
|
value: `createvalue${i}`,
|
|
});
|
|
const toCreate = generatePayload();
|
|
|
|
test('should create a new variable and return it for an owner', async () => {
|
|
const response = await authOwnerAgent.post('/variables').send(toCreate);
|
|
expect(response.statusCode).toBe(200);
|
|
expect(response.body.data.key).toBe(toCreate.key);
|
|
expect(response.body.data.value).toBe(toCreate.value);
|
|
|
|
const [byId, byKey] = await Promise.all([
|
|
getVariableById(response.body.data.id),
|
|
getVariableByKey(toCreate.key),
|
|
]);
|
|
|
|
expect(byId).not.toBeNull();
|
|
expect(byId!.key).toBe(toCreate.key);
|
|
expect(byId!.value).toBe(toCreate.value);
|
|
|
|
expect(byKey).not.toBeNull();
|
|
expect(byKey!.id).toBe(response.body.data.id);
|
|
expect(byKey!.value).toBe(toCreate.value);
|
|
});
|
|
|
|
test('should not create a new variable and return it for a member', async () => {
|
|
const response = await authMemberAgent.post('/variables').send(toCreate);
|
|
expect(response.statusCode).toBe(403);
|
|
expect(response.body.data?.key).not.toBe(toCreate.key);
|
|
expect(response.body.data?.value).not.toBe(toCreate.value);
|
|
|
|
const byKey = await getVariableByKey(toCreate.key);
|
|
expect(byKey).toBeNull();
|
|
});
|
|
|
|
test("POST /variables should not create a new variable and return it if the instance doesn't have a license", async () => {
|
|
license.disable('feat:variables');
|
|
const response = await authOwnerAgent.post('/variables').send(toCreate);
|
|
expect(response.statusCode).toBe(403);
|
|
expect(response.body.data?.key).not.toBe(toCreate.key);
|
|
expect(response.body.data?.value).not.toBe(toCreate.value);
|
|
|
|
const byKey = await getVariableByKey(toCreate.key);
|
|
expect(byKey).toBeNull();
|
|
});
|
|
|
|
test('should fail to create a new variable and if one with the same key exists', async () => {
|
|
await createVariable(toCreate.key, toCreate.value);
|
|
const response = await authOwnerAgent.post('/variables').send(toCreate);
|
|
expect(response.statusCode).toBe(500);
|
|
expect(response.body.data?.key).not.toBe(toCreate.key);
|
|
expect(response.body.data?.value).not.toBe(toCreate.value);
|
|
});
|
|
|
|
test('should not fail if variable limit not reached', async () => {
|
|
license.setQuota('quota:maxVariables', 5);
|
|
let i = 1;
|
|
let toCreate = generatePayload(i);
|
|
while (i < 3) {
|
|
await createVariable(toCreate.key, toCreate.value);
|
|
i++;
|
|
toCreate = generatePayload(i);
|
|
}
|
|
const response = await authOwnerAgent.post('/variables').send(toCreate);
|
|
expect(response.statusCode).toBe(200);
|
|
expect(response.body.data?.key).toBe(toCreate.key);
|
|
expect(response.body.data?.value).toBe(toCreate.value);
|
|
});
|
|
|
|
test('should fail if variable limit reached', async () => {
|
|
license.setQuota('quota:maxVariables', 5);
|
|
let i = 1;
|
|
let toCreate = generatePayload(i);
|
|
while (i < 6) {
|
|
await createVariable(toCreate.key, toCreate.value);
|
|
i++;
|
|
toCreate = generatePayload(i);
|
|
}
|
|
const response = await authOwnerAgent.post('/variables').send(toCreate);
|
|
expect(response.statusCode).toBe(400);
|
|
expect(response.body.data?.key).not.toBe(toCreate.key);
|
|
expect(response.body.data?.value).not.toBe(toCreate.value);
|
|
});
|
|
|
|
test('should fail if key too long', async () => {
|
|
const toCreate = {
|
|
// 51 'a's
|
|
key: 'aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa',
|
|
value: 'value',
|
|
};
|
|
const response = await authOwnerAgent.post('/variables').send(toCreate);
|
|
expect(response.statusCode).toBe(400);
|
|
expect(response.body.data?.key).not.toBe(toCreate.key);
|
|
expect(response.body.data?.value).not.toBe(toCreate.value);
|
|
});
|
|
|
|
test('should fail if value too long', async () => {
|
|
const toCreate = {
|
|
key: 'key',
|
|
// 256 'a's
|
|
value:
|
|
'aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa',
|
|
};
|
|
const response = await authOwnerAgent.post('/variables').send(toCreate);
|
|
expect(response.statusCode).toBe(400);
|
|
expect(response.body.data?.key).not.toBe(toCreate.key);
|
|
expect(response.body.data?.value).not.toBe(toCreate.value);
|
|
});
|
|
|
|
test("should fail if key contain's prohibited characters", async () => {
|
|
const toCreate = {
|
|
// 51 'a's
|
|
key: 'te$t',
|
|
value: 'value',
|
|
};
|
|
const response = await authOwnerAgent.post('/variables').send(toCreate);
|
|
expect(response.statusCode).toBe(400);
|
|
expect(response.body.data?.key).not.toBe(toCreate.key);
|
|
expect(response.body.data?.value).not.toBe(toCreate.value);
|
|
});
|
|
});
|
|
|
|
// ----------------------------------------
|
|
// PATCH /variables/:id - change a variable
|
|
// ----------------------------------------
|
|
describe('PATCH /variables/:id', () => {
|
|
const toModify = {
|
|
key: 'create1',
|
|
value: 'createvalue1',
|
|
};
|
|
|
|
test('should modify existing variable if use is an owner', async () => {
|
|
const variable = await createVariable('test1', 'value1');
|
|
const response = await authOwnerAgent.patch(`/variables/${variable.id}`).send(toModify);
|
|
expect(response.statusCode).toBe(200);
|
|
expect(response.body.data.key).toBe(toModify.key);
|
|
expect(response.body.data.value).toBe(toModify.value);
|
|
|
|
const [byId, byKey] = await Promise.all([
|
|
getVariableById(response.body.data.id),
|
|
getVariableByKey(toModify.key),
|
|
]);
|
|
|
|
expect(byId).not.toBeNull();
|
|
expect(byId!.key).toBe(toModify.key);
|
|
expect(byId!.value).toBe(toModify.value);
|
|
|
|
expect(byKey).not.toBeNull();
|
|
expect(byKey!.id).toBe(response.body.data.id);
|
|
expect(byKey!.value).toBe(toModify.value);
|
|
});
|
|
|
|
test('should modify existing variable if use is an owner', async () => {
|
|
const variable = await createVariable('test1', 'value1');
|
|
const response = await authOwnerAgent.patch(`/variables/${variable.id}`).send(toModify);
|
|
expect(response.statusCode).toBe(200);
|
|
expect(response.body.data.key).toBe(toModify.key);
|
|
expect(response.body.data.value).toBe(toModify.value);
|
|
|
|
const [byId, byKey] = await Promise.all([
|
|
getVariableById(response.body.data.id),
|
|
getVariableByKey(toModify.key),
|
|
]);
|
|
|
|
expect(byId).not.toBeNull();
|
|
expect(byId!.key).toBe(toModify.key);
|
|
expect(byId!.value).toBe(toModify.value);
|
|
|
|
expect(byKey).not.toBeNull();
|
|
expect(byKey!.id).toBe(response.body.data.id);
|
|
expect(byKey!.value).toBe(toModify.value);
|
|
});
|
|
|
|
test('should not modify existing variable if use is a member', async () => {
|
|
const variable = await createVariable('test1', 'value1');
|
|
const response = await authMemberAgent.patch(`/variables/${variable.id}`).send(toModify);
|
|
expect(response.statusCode).toBe(403);
|
|
expect(response.body.data?.key).not.toBe(toModify.key);
|
|
expect(response.body.data?.value).not.toBe(toModify.value);
|
|
|
|
const byId = await getVariableById(variable.id);
|
|
expect(byId).not.toBeNull();
|
|
expect(byId!.key).not.toBe(toModify.key);
|
|
expect(byId!.value).not.toBe(toModify.value);
|
|
});
|
|
|
|
test('should not modify existing variable if one with the same key exists', async () => {
|
|
const [var1] = await Promise.all([
|
|
createVariable('test1', 'value1'),
|
|
createVariable(toModify.key, toModify.value),
|
|
]);
|
|
const response = await authOwnerAgent.patch(`/variables/${var1.id}`).send(toModify);
|
|
expect(response.statusCode).toBe(500);
|
|
expect(response.body.data?.key).not.toBe(toModify.key);
|
|
expect(response.body.data?.value).not.toBe(toModify.value);
|
|
|
|
const byId = await getVariableById(var1.id);
|
|
expect(byId).not.toBeNull();
|
|
expect(byId!.key).toBe(var1.key);
|
|
expect(byId!.value).toBe(var1.value);
|
|
});
|
|
});
|
|
|
|
// ----------------------------------------
|
|
// DELETE /variables/:id - change a variable
|
|
// ----------------------------------------
|
|
describe('DELETE /variables/:id', () => {
|
|
test('should delete a single variable for an owner', async () => {
|
|
const [var1] = await Promise.all([
|
|
createVariable('test1', 'value1'),
|
|
createVariable('test2', 'value2'),
|
|
createVariable('test3', 'value3'),
|
|
]);
|
|
|
|
const delResponse = await authOwnerAgent.delete(`/variables/${var1.id}`);
|
|
expect(delResponse.statusCode).toBe(200);
|
|
|
|
const byId = await getVariableById(var1.id);
|
|
expect(byId).toBeNull();
|
|
|
|
const getResponse = await authOwnerAgent.get('/variables');
|
|
expect(getResponse.body.data.length).toBe(2);
|
|
});
|
|
|
|
test('should not delete a single variable for a member', async () => {
|
|
const [var1] = await Promise.all([
|
|
createVariable('test1', 'value1'),
|
|
createVariable('test2', 'value2'),
|
|
createVariable('test3', 'value3'),
|
|
]);
|
|
|
|
const delResponse = await authMemberAgent.delete(`/variables/${var1.id}`);
|
|
expect(delResponse.statusCode).toBe(403);
|
|
|
|
const byId = await getVariableById(var1.id);
|
|
expect(byId).not.toBeNull();
|
|
|
|
const getResponse = await authMemberAgent.get('/variables');
|
|
expect(getResponse.body.data.length).toBe(3);
|
|
});
|
|
});
|