n8n
Version:
n8n Workflow Automation Tool
335 lines • 16.8 kB
JavaScript
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
return c > 3 && r && Object.defineProperty(target, key, r), r;
};
var __importStar = (this && this.__importStar) || (function () {
var ownKeys = function(o) {
ownKeys = Object.getOwnPropertyNames || function (o) {
var ar = [];
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
return ar;
};
return ownKeys(o);
};
return function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
__setModuleDefault(result, mod);
return result;
};
})();
var __metadata = (this && this.__metadata) || function (k, v) {
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
};
var __param = (this && this.__param) || function (paramIndex, decorator) {
return function (target, key) { decorator(target, key, paramIndex); }
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.CredentialsController = void 0;
const api_types_1 = require("@n8n/api-types");
const config_1 = require("@n8n/config");
const typeorm_1 = require("@n8n/typeorm");
const n8n_core_1 = require("n8n-core");
const n8n_workflow_1 = require("n8n-workflow");
const zod_1 = require("zod");
const shared_credentials_1 = require("../databases/entities/shared-credentials");
const project_relation_repository_1 = require("../databases/repositories/project-relation.repository");
const shared_credentials_repository_1 = require("../databases/repositories/shared-credentials.repository");
const Db = __importStar(require("../db"));
const decorators_1 = require("../decorators");
const args_1 = require("../decorators/args");
const bad_request_error_1 = require("../errors/response-errors/bad-request.error");
const forbidden_error_1 = require("../errors/response-errors/forbidden.error");
const not_found_error_1 = require("../errors/response-errors/not-found.error");
const event_service_1 = require("../events/event.service");
const license_1 = require("../license");
const middlewares_1 = require("../middlewares");
const naming_service_1 = require("../services/naming.service");
const email_1 = require("../user-management/email");
const utils = __importStar(require("../utils"));
const credentials_service_1 = require("./credentials.service");
const credentials_service_ee_1 = require("./credentials.service.ee");
let CredentialsController = class CredentialsController {
constructor(globalConfig, credentialsService, enterpriseCredentialsService, namingService, license, logger, userManagementMailer, sharedCredentialsRepository, projectRelationRepository, eventService) {
this.globalConfig = globalConfig;
this.credentialsService = credentialsService;
this.enterpriseCredentialsService = enterpriseCredentialsService;
this.namingService = namingService;
this.license = license;
this.logger = logger;
this.userManagementMailer = userManagementMailer;
this.sharedCredentialsRepository = sharedCredentialsRepository;
this.projectRelationRepository = projectRelationRepository;
this.eventService = eventService;
}
async getMany(req, _res, query) {
const credentials = await this.credentialsService.getMany(req.user, {
listQueryOptions: req.listQueryOptions,
includeScopes: query.includeScopes,
includeData: query.includeData,
});
credentials.forEach((c) => {
delete c.shared;
});
return credentials;
}
async getProjectCredentials(req) {
const options = zod_1.z
.union([zod_1.z.object({ workflowId: zod_1.z.string() }), zod_1.z.object({ projectId: zod_1.z.string() })])
.parse(req.query);
return await this.credentialsService.getCredentialsAUserCanUseInAWorkflow(req.user, options);
}
async generateUniqueName(_req, _res, query) {
const requestedName = query.name ?? this.globalConfig.credentials.defaultName;
return {
name: await this.namingService.getUniqueCredentialName(requestedName),
};
}
async getOne(req, _res, credentialId, query) {
const { shared, ...credential } = this.license.isSharingEnabled()
? await this.enterpriseCredentialsService.getOne(req.user, credentialId, query.includeData)
: await this.credentialsService.getOne(req.user, credentialId, query.includeData);
const scopes = await this.credentialsService.getCredentialScopes(req.user, req.params.credentialId);
return { ...credential, scopes };
}
async testCredentials(req) {
const { credentials } = req.body;
const storedCredential = await this.sharedCredentialsRepository.findCredentialForUser(credentials.id, req.user, ['credential:read']);
if (!storedCredential) {
throw new forbidden_error_1.ForbiddenError();
}
const mergedCredentials = (0, n8n_workflow_1.deepCopy)(credentials);
const decryptedData = this.credentialsService.decrypt(storedCredential, true);
await this.credentialsService.replaceCredentialContentsForSharee(req.user, storedCredential, decryptedData, mergedCredentials);
if (mergedCredentials.data) {
mergedCredentials.data = this.credentialsService.unredact(mergedCredentials.data, decryptedData);
}
return await this.credentialsService.test(req.user.id, mergedCredentials);
}
async createCredentials(req, _, payload) {
const newCredential = await this.credentialsService.createUnmanagedCredential(payload, req.user);
const project = await this.sharedCredentialsRepository.findCredentialOwningProject(newCredential.id);
this.eventService.emit('credentials-created', {
user: req.user,
credentialType: newCredential.type,
credentialId: newCredential.id,
publicApi: false,
projectId: project?.id,
projectType: project?.type,
});
return newCredential;
}
async updateCredentials(req) {
const { body, user, params: { credentialId }, } = req;
const credential = await this.sharedCredentialsRepository.findCredentialForUser(credentialId, user, ['credential:update']);
if (!credential) {
this.logger.info('Attempt to update credential blocked due to lack of permissions', {
credentialId,
userId: user.id,
});
throw new not_found_error_1.NotFoundError('Credential to be updated not found. You can only update credentials owned by you');
}
if (credential.isManaged) {
throw new bad_request_error_1.BadRequestError('Managed credentials cannot be updated');
}
const decryptedData = this.credentialsService.decrypt(credential, true);
delete body.data?.oauthTokenData;
const preparedCredentialData = await this.credentialsService.prepareUpdateData(req.body, decryptedData);
const newCredentialData = this.credentialsService.createEncryptedData({
id: credential.id,
name: preparedCredentialData.name,
type: preparedCredentialData.type,
data: preparedCredentialData.data,
});
const responseData = await this.credentialsService.update(credentialId, newCredentialData);
if (responseData === null) {
throw new not_found_error_1.NotFoundError(`Credential ID "${credentialId}" could not be found to be updated.`);
}
const { data, shared, ...rest } = responseData;
this.logger.debug('Credential updated', { credentialId });
this.eventService.emit('credentials-updated', {
user: req.user,
credentialType: credential.type,
credentialId: credential.id,
});
const scopes = await this.credentialsService.getCredentialScopes(req.user, credential.id);
return { ...rest, scopes };
}
async deleteCredentials(req) {
const { credentialId } = req.params;
const credential = await this.sharedCredentialsRepository.findCredentialForUser(credentialId, req.user, ['credential:delete']);
if (!credential) {
this.logger.info('Attempt to delete credential blocked due to lack of permissions', {
credentialId,
userId: req.user.id,
});
throw new not_found_error_1.NotFoundError('Credential to be deleted not found. You can only removed credentials owned by you');
}
await this.credentialsService.delete(req.user, credential.id);
this.eventService.emit('credentials-deleted', {
user: req.user,
credentialType: credential.type,
credentialId: credential.id,
});
return true;
}
async shareCredentials(req) {
const { credentialId } = req.params;
const { shareWithIds } = req.body;
if (!Array.isArray(shareWithIds) ||
!shareWithIds.every((userId) => typeof userId === 'string')) {
throw new bad_request_error_1.BadRequestError('Bad request');
}
const credential = await this.sharedCredentialsRepository.findCredentialForUser(credentialId, req.user, ['credential:share']);
if (!credential) {
throw new forbidden_error_1.ForbiddenError();
}
let amountRemoved = null;
let newShareeIds = [];
await Db.transaction(async (trx) => {
const currentProjectIds = credential.shared
.filter((sc) => sc.role === 'credential:user')
.map((sc) => sc.projectId);
const newProjectIds = shareWithIds;
const toShare = utils.rightDiff([currentProjectIds, (id) => id], [newProjectIds, (id) => id]);
const toUnshare = utils.rightDiff([newProjectIds, (id) => id], [currentProjectIds, (id) => id]);
const deleteResult = await trx.delete(shared_credentials_1.SharedCredentials, {
credentialsId: credentialId,
projectId: (0, typeorm_1.In)(toUnshare),
});
await this.enterpriseCredentialsService.shareWithProjects(req.user, credential.id, toShare, trx);
if (deleteResult.affected) {
amountRemoved = deleteResult.affected;
}
newShareeIds = toShare;
});
this.eventService.emit('credentials-shared', {
user: req.user,
credentialType: credential.type,
credentialId: credential.id,
userIdSharer: req.user.id,
userIdsShareesAdded: newShareeIds,
shareesRemoved: amountRemoved,
});
const projectsRelations = await this.projectRelationRepository.findBy({
projectId: (0, typeorm_1.In)(newShareeIds),
role: 'project:personalOwner',
});
await this.userManagementMailer.notifyCredentialsShared({
sharer: req.user,
newShareeIds: projectsRelations.map((pr) => pr.userId),
credentialsName: credential.name,
});
}
async transfer(req) {
const body = zod_1.z.object({ destinationProjectId: zod_1.z.string() }).parse(req.body);
return await this.enterpriseCredentialsService.transferOne(req.user, req.params.credentialId, body.destinationProjectId);
}
};
exports.CredentialsController = CredentialsController;
__decorate([
(0, decorators_1.Get)('/', { middlewares: middlewares_1.listQueryMiddleware }),
__param(2, args_1.Query),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object, Object, api_types_1.CredentialsGetManyRequestQuery]),
__metadata("design:returntype", Promise)
], CredentialsController.prototype, "getMany", null);
__decorate([
(0, decorators_1.Get)('/for-workflow'),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object]),
__metadata("design:returntype", Promise)
], CredentialsController.prototype, "getProjectCredentials", null);
__decorate([
(0, decorators_1.Get)('/new'),
__param(2, args_1.Query),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object, Object, api_types_1.GenerateCredentialNameRequestQuery]),
__metadata("design:returntype", Promise)
], CredentialsController.prototype, "generateUniqueName", null);
__decorate([
(0, decorators_1.Get)('/:credentialId'),
(0, decorators_1.ProjectScope)('credential:read'),
__param(2, (0, args_1.Param)('credentialId')),
__param(3, args_1.Query),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object, Object, String, api_types_1.CredentialsGetOneRequestQuery]),
__metadata("design:returntype", Promise)
], CredentialsController.prototype, "getOne", null);
__decorate([
(0, decorators_1.Post)('/test'),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object]),
__metadata("design:returntype", Promise)
], CredentialsController.prototype, "testCredentials", null);
__decorate([
(0, decorators_1.Post)('/'),
__param(2, args_1.Body),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object, Response,
api_types_1.CreateCredentialDto]),
__metadata("design:returntype", Promise)
], CredentialsController.prototype, "createCredentials", null);
__decorate([
(0, decorators_1.Patch)('/:credentialId'),
(0, decorators_1.ProjectScope)('credential:update'),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object]),
__metadata("design:returntype", Promise)
], CredentialsController.prototype, "updateCredentials", null);
__decorate([
(0, decorators_1.Delete)('/:credentialId'),
(0, decorators_1.ProjectScope)('credential:delete'),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object]),
__metadata("design:returntype", Promise)
], CredentialsController.prototype, "deleteCredentials", null);
__decorate([
(0, decorators_1.Licensed)('feat:sharing'),
(0, decorators_1.Put)('/:credentialId/share'),
(0, decorators_1.ProjectScope)('credential:share'),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object]),
__metadata("design:returntype", Promise)
], CredentialsController.prototype, "shareCredentials", null);
__decorate([
(0, decorators_1.Put)('/:credentialId/transfer'),
(0, decorators_1.ProjectScope)('credential:move'),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object]),
__metadata("design:returntype", Promise)
], CredentialsController.prototype, "transfer", null);
exports.CredentialsController = CredentialsController = __decorate([
(0, decorators_1.RestController)('/credentials'),
__metadata("design:paramtypes", [config_1.GlobalConfig,
credentials_service_1.CredentialsService,
credentials_service_ee_1.EnterpriseCredentialsService,
naming_service_1.NamingService,
license_1.License,
n8n_core_1.Logger,
email_1.UserManagementMailer,
shared_credentials_repository_1.SharedCredentialsRepository,
project_relation_repository_1.ProjectRelationRepository,
event_service_1.EventService])
], CredentialsController);
//# sourceMappingURL=credentials.controller.js.map
;