mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-17 18:12:04 +00:00
* first commit for postgres migration * (not working) * sqlite migration * quicksave * fix tests * fix pg test * fix postgres * fix variables import * fix execution saving * add user settings fix * change migration to single lines * patch preferences endpoint * cleanup * improve variable import * cleanup unusued code * Update packages/cli/src/PublicApi/v1/handlers/workflows/workflows.handler.ts Co-authored-by: Omar Ajoue <krynble@gmail.com> * address review notes * fix var update/import * refactor: Separate execution data to its own table (#6323) * wip: Temporary migration process * refactor: Create boilerplate repository methods for executions * fix: Lint issues * refactor: Added search endpoint to repository * refactor: Make the execution list work again * wip: Updating how we create and update executions everywhere * fix: Lint issues and remove most of the direct access to execution model * refactor: Remove includeWorkflowData flag and fix more tests * fix: Lint issues * fix: Fixed ordering of executions for FE, removed transaction when saving execution and removed unnecessary update * refactor: Add comment about missing feature * refactor: Refactor counting executions * refactor: Add migration for other dbms and fix issues found * refactor: Fix lint issues * refactor: Remove unnecessary comment and auto inject repo to internal hooks * refactor: remove type assertion * fix: Fix broken tests * fix: Remove unnecessary import * Remove unnecessary toString() call Co-authored-by: Iván Ovejero <ivov.src@gmail.com> * fix: Address comments after review * refactor: Remove unused import * fix: Lint issues * fix: Add correct migration files --------- Co-authored-by: Iván Ovejero <ivov.src@gmail.com> * remove null values from credential export * fix: Fix an issue with queue mode where all running execution would be returned * fix: Update n8n node to allow for workflow ids with letters * set upstream on set branch * remove typo * add nodeAccess to credentials * fix unsaved run check for undefined id * fix(core): Rename version control feature to source control (#6480) * rename versionControl to sourceControl * fix source control tooltip wording --------- Co-authored-by: Romain Minaud <romain.minaud@gmail.com> * fix(editor): Pay 548 hide the set up version control button (#6485) * feat(DebugHelper Node): Fix and include in main app (#6406) * improve node a bit * fixing continueOnFail() ton contain error in json * improve pairedItem * fix random data returning object results * fix nanoId length typo * update pnpm-lock file --------- Co-authored-by: Marcus <marcus@n8n.io> * fix(editor): Remove setup source control CTA button * fix(editor): Remove setup source control CTA button --------- Co-authored-by: Michael Auerswald <michael.auerswald@gmail.com> Co-authored-by: Marcus <marcus@n8n.io> * fix(editor): Update source control docs links (#6488) * feat(DebugHelper Node): Fix and include in main app (#6406) * improve node a bit * fixing continueOnFail() ton contain error in json * improve pairedItem * fix random data returning object results * fix nanoId length typo * update pnpm-lock file --------- Co-authored-by: Marcus <marcus@n8n.io> * feat(editor): Replace root events with event bus events (no-changelog) (#6454) * feat: replace root events with event bus events * fix: prevent cypress from replacing global with globalThis in import path * feat: remove emitter mixin * fix: replace component events with event bus * fix: fix linting issue * fix: fix breaking expression switch * chore: prettify ndv e2e suite code * fix(editor): Update source control docs links --------- Co-authored-by: Michael Auerswald <michael.auerswald@gmail.com> Co-authored-by: Marcus <marcus@n8n.io> Co-authored-by: Alex Grozav <alex@grozav.com> * fix tag endpoint regex --------- Co-authored-by: Omar Ajoue <krynble@gmail.com> Co-authored-by: Iván Ovejero <ivov.src@gmail.com> Co-authored-by: Romain Minaud <romain.minaud@gmail.com> Co-authored-by: Csaba Tuncsik <csaba@n8n.io> Co-authored-by: Marcus <marcus@n8n.io> Co-authored-by: Alex Grozav <alex@grozav.com>
159 lines
4.6 KiB
TypeScript
159 lines
4.6 KiB
TypeScript
/* eslint-disable @typescript-eslint/no-unsafe-return */
|
|
/* eslint-disable @typescript-eslint/no-unsafe-call */
|
|
/* eslint-disable @typescript-eslint/no-unsafe-member-access */
|
|
/* eslint-disable @typescript-eslint/no-unsafe-argument */
|
|
/* eslint-disable @typescript-eslint/no-unsafe-assignment */
|
|
/* eslint-disable @typescript-eslint/naming-convention */
|
|
import { Router } from 'express';
|
|
import type { Request } from 'express';
|
|
import bodyParser from 'body-parser';
|
|
import { v4 as uuid } from 'uuid';
|
|
import { Container } from 'typedi';
|
|
import config from '@/config';
|
|
import * as Db from '@/Db';
|
|
import type { Role } from '@db/entities/Role';
|
|
import { RoleRepository } from '@db/repositories';
|
|
import { hashPassword } from '@/UserManagement/UserManagementHelper';
|
|
import { eventBus } from '@/eventbus/MessageEventBus/MessageEventBus';
|
|
import { License } from '../License';
|
|
import { LICENSE_FEATURES } from '@/constants';
|
|
|
|
if (process.env.E2E_TESTS !== 'true') {
|
|
console.error('E2E endpoints only allowed during E2E tests');
|
|
process.exit(1);
|
|
}
|
|
|
|
const enabledFeatures = {
|
|
[LICENSE_FEATURES.SHARING]: true, //default to true here instead of setting it in config/index.ts for e2e
|
|
[LICENSE_FEATURES.LDAP]: false,
|
|
[LICENSE_FEATURES.SAML]: false,
|
|
[LICENSE_FEATURES.LOG_STREAMING]: false,
|
|
[LICENSE_FEATURES.ADVANCED_EXECUTION_FILTERS]: false,
|
|
[LICENSE_FEATURES.SOURCE_CONTROL]: false,
|
|
};
|
|
|
|
type Feature = keyof typeof enabledFeatures;
|
|
|
|
Container.get(License).isFeatureEnabled = (feature: Feature) => enabledFeatures[feature] ?? false;
|
|
|
|
const tablesToTruncate = [
|
|
'auth_identity',
|
|
'auth_provider_sync_history',
|
|
'event_destinations',
|
|
'shared_workflow',
|
|
'shared_credentials',
|
|
'webhook_entity',
|
|
'workflows_tags',
|
|
'credentials_entity',
|
|
'tag_entity',
|
|
'workflow_statistics',
|
|
'workflow_entity',
|
|
'execution_entity',
|
|
'settings',
|
|
'installed_packages',
|
|
'installed_nodes',
|
|
'user',
|
|
'role',
|
|
'variables',
|
|
];
|
|
|
|
const truncateAll = async () => {
|
|
const connection = Db.getConnection();
|
|
|
|
for (const table of tablesToTruncate) {
|
|
try {
|
|
await connection.query(
|
|
`DELETE FROM ${table}; DELETE FROM sqlite_sequence WHERE name=${table};`,
|
|
);
|
|
} catch (error) {
|
|
console.warn('Dropping Table for E2E Reset error: ', error);
|
|
}
|
|
}
|
|
};
|
|
|
|
const setupUserManagement = async () => {
|
|
const connection = Db.getConnection();
|
|
await connection.query('INSERT INTO role (name, scope) VALUES ("owner", "global");');
|
|
const instanceOwnerRole = (await connection.query(
|
|
'SELECT last_insert_rowid() as insertId',
|
|
)) as Array<{ insertId: number }>;
|
|
|
|
const roles: Array<[Role['name'], Role['scope']]> = [
|
|
['member', 'global'],
|
|
['owner', 'workflow'],
|
|
['owner', 'credential'],
|
|
['user', 'credential'],
|
|
['editor', 'workflow'],
|
|
];
|
|
|
|
await Promise.all(
|
|
roles.map(async ([name, scope]) =>
|
|
connection.query(`INSERT INTO role (name, scope) VALUES ("${name}", "${scope}");`),
|
|
),
|
|
);
|
|
await connection.query(
|
|
`INSERT INTO user (id, globalRoleId) values ("${uuid()}", ${instanceOwnerRole[0].insertId})`,
|
|
);
|
|
await connection.query(
|
|
"INSERT INTO \"settings\" (key, value, loadOnStartup) values ('userManagement.isInstanceOwnerSetUp', 'false', true), ('userManagement.skipInstanceOwnerSetup', 'false', true)",
|
|
);
|
|
|
|
config.set('userManagement.isInstanceOwnerSetUp', false);
|
|
};
|
|
|
|
const resetLogStreaming = async () => {
|
|
enabledFeatures[LICENSE_FEATURES.LOG_STREAMING] = false;
|
|
for (const id in eventBus.destinations) {
|
|
await eventBus.removeDestination(id);
|
|
}
|
|
};
|
|
|
|
export const e2eController = Router();
|
|
|
|
e2eController.post('/db/reset', async (req, res) => {
|
|
await resetLogStreaming();
|
|
await truncateAll();
|
|
await setupUserManagement();
|
|
|
|
res.writeHead(204).end();
|
|
});
|
|
|
|
e2eController.post('/db/setup-owner', bodyParser.json(), async (req, res) => {
|
|
if (config.get('userManagement.isInstanceOwnerSetUp')) {
|
|
res.writeHead(500).send({ error: 'Owner already setup' });
|
|
return;
|
|
}
|
|
|
|
const globalRole = await Container.get(RoleRepository).findGlobalOwnerRoleOrFail();
|
|
|
|
const owner = await Db.collections.User.findOneByOrFail({ globalRoleId: globalRole.id });
|
|
|
|
await Db.collections.User.update(owner.id, {
|
|
email: req.body.email,
|
|
password: await hashPassword(req.body.password),
|
|
firstName: req.body.firstName,
|
|
lastName: req.body.lastName,
|
|
});
|
|
|
|
await Db.collections.Settings.update(
|
|
{ key: 'userManagement.isInstanceOwnerSetUp' },
|
|
{ value: 'true' },
|
|
);
|
|
|
|
config.set('userManagement.isInstanceOwnerSetUp', true);
|
|
|
|
res.writeHead(204).end();
|
|
});
|
|
|
|
e2eController.patch(
|
|
'/feature/:feature',
|
|
bodyParser.json(),
|
|
async (req: Request<{ feature: Feature }>, res) => {
|
|
const { feature } = req.params;
|
|
const { enabled } = req.body;
|
|
|
|
enabledFeatures[feature] = enabled === undefined || enabled === true;
|
|
res.writeHead(204).end();
|
|
},
|
|
);
|