mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-12 21:37:32 -08:00
feat(core): Allow admin creation (#7837)
https://linear.app/n8n/issue/PAY-1038
This commit is contained in:
parent
5ba5ed8e3c
commit
476806ebb0
|
@ -281,6 +281,7 @@ export class Server extends AbstractServer {
|
||||||
activeWorkflowRunner,
|
activeWorkflowRunner,
|
||||||
Container.get(RoleService),
|
Container.get(RoleService),
|
||||||
userService,
|
userService,
|
||||||
|
Container.get(License),
|
||||||
),
|
),
|
||||||
Container.get(SamlController),
|
Container.get(SamlController),
|
||||||
Container.get(SourceControlController),
|
Container.get(SourceControlController),
|
||||||
|
@ -296,6 +297,7 @@ export class Server extends AbstractServer {
|
||||||
internalHooks,
|
internalHooks,
|
||||||
externalHooks,
|
externalHooks,
|
||||||
Container.get(UserService),
|
Container.get(UserService),
|
||||||
|
Container.get(License),
|
||||||
postHog,
|
postHog,
|
||||||
),
|
),
|
||||||
Container.get(VariablesController),
|
Container.get(VariablesController),
|
||||||
|
|
|
@ -28,6 +28,7 @@ export class InvitationController {
|
||||||
private readonly internalHooks: IInternalHooksClass,
|
private readonly internalHooks: IInternalHooksClass,
|
||||||
private readonly externalHooks: IExternalHooksClass,
|
private readonly externalHooks: IExternalHooksClass,
|
||||||
private readonly userService: UserService,
|
private readonly userService: UserService,
|
||||||
|
private readonly license: License,
|
||||||
private readonly postHog?: PostHogClient,
|
private readonly postHog?: PostHogClient,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
|
@ -88,11 +89,26 @@ export class InvitationController {
|
||||||
`Request to send email invite(s) to user(s) failed because of an invalid email address: ${invite.email}`,
|
`Request to send email invite(s) to user(s) failed because of an invalid email address: ${invite.email}`,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (invite.role && !['member', 'admin'].includes(invite.role)) {
|
||||||
|
throw new BadRequestError(
|
||||||
|
`Cannot invite user with invalid role: ${invite.role}. Please ensure all invitees' roles are either 'member' or 'admin'.`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (invite.role === 'admin' && !this.license.isAdvancedPermissionsLicensed()) {
|
||||||
|
throw new UnauthorizedError(
|
||||||
|
'Cannot invite admin user without advanced permissions. Please upgrade to a license that includes this feature.',
|
||||||
|
);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const emails = req.body.map((e) => e.email);
|
const attributes = req.body.map(({ email, role }) => ({
|
||||||
|
email,
|
||||||
|
role: role ?? 'member',
|
||||||
|
}));
|
||||||
|
|
||||||
const { usersInvited, usersCreated } = await this.userService.inviteMembers(req.user, emails);
|
const { usersInvited, usersCreated } = await this.userService.inviteUsers(req.user, attributes);
|
||||||
|
|
||||||
await this.externalHooks.run('user.invited', [usersCreated]);
|
await this.externalHooks.run('user.invited', [usersCreated]);
|
||||||
|
|
||||||
|
|
|
@ -19,6 +19,7 @@ import { Logger } from '@/Logger';
|
||||||
import { UnauthorizedError } from '@/errors/response-errors/unauthorized.error';
|
import { UnauthorizedError } from '@/errors/response-errors/unauthorized.error';
|
||||||
import { NotFoundError } from '@/errors/response-errors/not-found.error';
|
import { NotFoundError } from '@/errors/response-errors/not-found.error';
|
||||||
import { BadRequestError } from '@/errors/response-errors/bad-request.error';
|
import { BadRequestError } from '@/errors/response-errors/bad-request.error';
|
||||||
|
import { License } from '@/License';
|
||||||
|
|
||||||
@Authorized()
|
@Authorized()
|
||||||
@RestController('/users')
|
@RestController('/users')
|
||||||
|
@ -32,6 +33,7 @@ export class UsersController {
|
||||||
private readonly activeWorkflowRunner: ActiveWorkflowRunner,
|
private readonly activeWorkflowRunner: ActiveWorkflowRunner,
|
||||||
private readonly roleService: RoleService,
|
private readonly roleService: RoleService,
|
||||||
private readonly userService: UserService,
|
private readonly userService: UserService,
|
||||||
|
private readonly license: License,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
static ERROR_MESSAGES = {
|
static ERROR_MESSAGES = {
|
||||||
|
@ -43,6 +45,7 @@ export class UsersController {
|
||||||
NO_ADMIN_ON_OWNER: 'Admin cannot change role on global owner',
|
NO_ADMIN_ON_OWNER: 'Admin cannot change role on global owner',
|
||||||
NO_OWNER_ON_OWNER: 'Owner cannot change role on global owner',
|
NO_OWNER_ON_OWNER: 'Owner cannot change role on global owner',
|
||||||
NO_USER_TO_OWNER: 'Cannot promote user to global owner',
|
NO_USER_TO_OWNER: 'Cannot promote user to global owner',
|
||||||
|
NO_ADMIN_IF_UNLICENSED: 'Admin role is not available without a license',
|
||||||
},
|
},
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
|
@ -336,6 +339,7 @@ export class UsersController {
|
||||||
NO_USER_TO_OWNER,
|
NO_USER_TO_OWNER,
|
||||||
NO_USER,
|
NO_USER,
|
||||||
NO_OWNER_ON_OWNER,
|
NO_OWNER_ON_OWNER,
|
||||||
|
NO_ADMIN_IF_UNLICENSED,
|
||||||
} = UsersController.ERROR_MESSAGES.CHANGE_ROLE;
|
} = UsersController.ERROR_MESSAGES.CHANGE_ROLE;
|
||||||
|
|
||||||
if (req.user.globalRole.scope === 'global' && req.user.globalRole.name === 'member') {
|
if (req.user.globalRole.scope === 'global' && req.user.globalRole.name === 'member') {
|
||||||
|
@ -364,6 +368,14 @@ export class UsersController {
|
||||||
throw new NotFoundError(NO_USER);
|
throw new NotFoundError(NO_USER);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
newRole.scope === 'global' &&
|
||||||
|
newRole.name === 'admin' &&
|
||||||
|
!this.license.isAdvancedPermissionsLicensed()
|
||||||
|
) {
|
||||||
|
throw new UnauthorizedError(NO_ADMIN_IF_UNLICENSED);
|
||||||
|
}
|
||||||
|
|
||||||
if (
|
if (
|
||||||
req.user.globalRole.scope === 'global' &&
|
req.user.globalRole.scope === 'global' &&
|
||||||
req.user.globalRole.name === 'admin' &&
|
req.user.globalRole.name === 'admin' &&
|
||||||
|
|
|
@ -296,7 +296,11 @@ export declare namespace PasswordResetRequest {
|
||||||
// ----------------------------------
|
// ----------------------------------
|
||||||
|
|
||||||
export declare namespace UserRequest {
|
export declare namespace UserRequest {
|
||||||
export type Invite = AuthenticatedRequest<{}, {}, Array<{ email: string }>>;
|
export type Invite = AuthenticatedRequest<
|
||||||
|
{},
|
||||||
|
{},
|
||||||
|
Array<{ email: string; role?: 'member' | 'admin' }>
|
||||||
|
>;
|
||||||
|
|
||||||
export type InviteResponse = {
|
export type InviteResponse = {
|
||||||
user: { id: string; email: string; inviteAcceptUrl?: string; emailSent: boolean };
|
user: { id: string; email: string; inviteAcceptUrl?: string; emailSent: boolean };
|
||||||
|
|
|
@ -238,18 +238,19 @@ export class UserService {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async inviteMembers(owner: User, emails: string[]) {
|
async inviteUsers(owner: User, attributes: Array<{ email: string; role: 'member' | 'admin' }>) {
|
||||||
const memberRole = await this.roleService.findGlobalMemberRole();
|
const memberRole = await this.roleService.findGlobalMemberRole();
|
||||||
|
const adminRole = await this.roleService.findGlobalAdminRole();
|
||||||
|
|
||||||
const existingUsers = await this.findMany({
|
const existingUsers = await this.findMany({
|
||||||
where: { email: In(emails) },
|
where: { email: In(attributes.map(({ email }) => email)) },
|
||||||
relations: ['globalRole'],
|
relations: ['globalRole'],
|
||||||
select: ['email', 'password', 'id'],
|
select: ['email', 'password', 'id'],
|
||||||
});
|
});
|
||||||
|
|
||||||
const existUsersEmails = existingUsers.map((user) => user.email);
|
const existUsersEmails = existingUsers.map((user) => user.email);
|
||||||
|
|
||||||
const toCreateUsers = emails.filter((email) => !existUsersEmails.includes(email));
|
const toCreateUsers = attributes.filter(({ email }) => !existUsersEmails.includes(email));
|
||||||
|
|
||||||
const pendingUsersToInvite = existingUsers.filter((email) => email.isPending);
|
const pendingUsersToInvite = existingUsers.filter((email) => email.isPending);
|
||||||
|
|
||||||
|
@ -264,10 +265,10 @@ export class UserService {
|
||||||
try {
|
try {
|
||||||
await this.getManager().transaction(async (transactionManager) =>
|
await this.getManager().transaction(async (transactionManager) =>
|
||||||
Promise.all(
|
Promise.all(
|
||||||
toCreateUsers.map(async (email) => {
|
toCreateUsers.map(async ({ email, role }) => {
|
||||||
const newUser = Object.assign(new User(), {
|
const newUser = Object.assign(new User(), {
|
||||||
email,
|
email,
|
||||||
globalRole: memberRole,
|
globalRole: role === 'member' ? memberRole : adminRole,
|
||||||
});
|
});
|
||||||
const savedUser = await transactionManager.save<User>(newUser);
|
const savedUser = await transactionManager.save<User>(newUser);
|
||||||
createdUsers.set(email, savedUser.id);
|
createdUsers.set(email, savedUser.id);
|
||||||
|
@ -285,6 +286,6 @@ export class UserService {
|
||||||
|
|
||||||
const usersInvited = await this.sendEmails(owner, Object.fromEntries(createdUsers));
|
const usersInvited = await this.sendEmails(owner, Object.fromEntries(createdUsers));
|
||||||
|
|
||||||
return { usersInvited, usersCreated: toCreateUsers };
|
return { usersInvited, usersCreated: toCreateUsers.map(({ email }) => email) };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
import validator from 'validator';
|
import validator from 'validator';
|
||||||
import type { SuperAgentTest } from 'supertest';
|
import type { SuperAgentTest } from 'supertest';
|
||||||
|
|
||||||
import type { Role } from '@db/entities/Role';
|
|
||||||
import type { User } from '@db/entities/User';
|
import type { User } from '@db/entities/User';
|
||||||
import { compareHash } from '@/UserManagement/UserManagementHelper';
|
import { compareHash } from '@/UserManagement/UserManagementHelper';
|
||||||
import { UserManagementMailer } from '@/UserManagement/email/UserManagementMailer';
|
import { UserManagementMailer } from '@/UserManagement/email/UserManagementMailer';
|
||||||
|
@ -18,74 +17,46 @@ import {
|
||||||
} from './shared/random';
|
} from './shared/random';
|
||||||
import * as testDb from './shared/testDb';
|
import * as testDb from './shared/testDb';
|
||||||
import * as utils from './shared/utils/';
|
import * as utils from './shared/utils/';
|
||||||
import { getAllRoles } from './shared/db/roles';
|
import { getGlobalAdminRole, getGlobalMemberRole } from './shared/db/roles';
|
||||||
import { createMember, createOwner, createUser, createUserShell } from './shared/db/users';
|
import { createMember, createOwner, createUser, createUserShell } from './shared/db/users';
|
||||||
import { ExternalHooks } from '@/ExternalHooks';
|
import { ExternalHooks } from '@/ExternalHooks';
|
||||||
import { InternalHooks } from '@/InternalHooks';
|
import { InternalHooks } from '@/InternalHooks';
|
||||||
|
import type { UserInvitationResponse } from './shared/utils/users';
|
||||||
let credentialOwnerRole: Role;
|
import {
|
||||||
let globalMemberRole: Role;
|
assertInviteUserSuccessResponse,
|
||||||
let workflowOwnerRole: Role;
|
assertInvitedUsersOnDb,
|
||||||
|
assertInviteUserErrorResponse,
|
||||||
let owner: User;
|
} from './shared/utils/users';
|
||||||
let member: User;
|
import { mocked } from 'jest-mock';
|
||||||
let authOwnerAgent: SuperAgentTest;
|
import { License } from '@/License';
|
||||||
let authlessAgent: SuperAgentTest;
|
|
||||||
|
|
||||||
mockInstance(InternalHooks);
|
mockInstance(InternalHooks);
|
||||||
|
|
||||||
|
const license = mockInstance(License, {
|
||||||
|
isAdvancedPermissionsLicensed: jest.fn().mockReturnValue(true),
|
||||||
|
isWithinUsersLimit: jest.fn().mockReturnValue(true),
|
||||||
|
});
|
||||||
|
|
||||||
const externalHooks = mockInstance(ExternalHooks);
|
const externalHooks = mockInstance(ExternalHooks);
|
||||||
const mailer = mockInstance(UserManagementMailer, { isEmailSetUp: true });
|
const mailer = mockInstance(UserManagementMailer, { isEmailSetUp: true });
|
||||||
|
|
||||||
const testServer = utils.setupTestServer({ endpointGroups: ['invitations'] });
|
const testServer = utils.setupTestServer({ endpointGroups: ['invitations'] });
|
||||||
|
|
||||||
type UserInvitationResponse = {
|
|
||||||
user: Pick<User, 'id' | 'email'> & { inviteAcceptUrl: string; emailSent: boolean };
|
|
||||||
error?: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
beforeAll(async () => {
|
|
||||||
const [_, fetchedGlobalMemberRole, fetchedWorkflowOwnerRole, fetchedCredentialOwnerRole] =
|
|
||||||
await getAllRoles();
|
|
||||||
|
|
||||||
credentialOwnerRole = fetchedCredentialOwnerRole;
|
|
||||||
globalMemberRole = fetchedGlobalMemberRole;
|
|
||||||
workflowOwnerRole = fetchedWorkflowOwnerRole;
|
|
||||||
});
|
|
||||||
|
|
||||||
beforeEach(async () => {
|
|
||||||
jest.resetAllMocks();
|
|
||||||
await testDb.truncate(['User', 'SharedCredentials', 'SharedWorkflow', 'Workflow', 'Credentials']);
|
|
||||||
owner = await createOwner();
|
|
||||||
member = await createMember();
|
|
||||||
authOwnerAgent = testServer.authAgentFor(owner);
|
|
||||||
authlessAgent = testServer.authlessAgent;
|
|
||||||
});
|
|
||||||
|
|
||||||
const assertInviteUserSuccessResponse = (data: UserInvitationResponse) => {
|
|
||||||
expect(validator.isUUID(data.user.id)).toBe(true);
|
|
||||||
expect(data.user.inviteAcceptUrl).toBeUndefined();
|
|
||||||
expect(data.user.email).toBeDefined();
|
|
||||||
expect(data.user.emailSent).toBe(true);
|
|
||||||
};
|
|
||||||
|
|
||||||
const assertInviteUserErrorResponse = (data: UserInvitationResponse) => {
|
|
||||||
expect(validator.isUUID(data.user.id)).toBe(true);
|
|
||||||
expect(data.user.inviteAcceptUrl).toBeDefined();
|
|
||||||
expect(data.user.email).toBeDefined();
|
|
||||||
expect(data.user.emailSent).toBe(false);
|
|
||||||
expect(data.error).toBeDefined();
|
|
||||||
};
|
|
||||||
|
|
||||||
const assertInvitedUsersOnDb = (user: User) => {
|
|
||||||
expect(user.firstName).toBeNull();
|
|
||||||
expect(user.lastName).toBeNull();
|
|
||||||
expect(user.personalizationAnswers).toBeNull();
|
|
||||||
expect(user.password).toBeNull();
|
|
||||||
expect(user.isPending).toBe(true);
|
|
||||||
};
|
|
||||||
|
|
||||||
describe('POST /invitations/:id/accept', () => {
|
describe('POST /invitations/:id/accept', () => {
|
||||||
test('should fill out a user shell', async () => {
|
let owner: User;
|
||||||
|
|
||||||
|
let authlessAgent: SuperAgentTest;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
await testDb.truncate(['User']);
|
||||||
|
|
||||||
|
owner = await createOwner();
|
||||||
|
|
||||||
|
authlessAgent = testServer.authlessAgent;
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should fill out a member shell', async () => {
|
||||||
|
const globalMemberRole = await getGlobalMemberRole();
|
||||||
const memberShell = await createUserShell(globalMemberRole);
|
const memberShell = await createUserShell(globalMemberRole);
|
||||||
|
|
||||||
const memberData = {
|
const memberData = {
|
||||||
|
@ -96,11 +67,9 @@ describe('POST /invitations/:id/accept', () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
const response = await authlessAgent
|
const response = await authlessAgent
|
||||||
.post(
|
.post(`/invitations/${memberShell.id}/accept`)
|
||||||
`/invitations/${memberShell.id}/
|
.send(memberData)
|
||||||
accept`,
|
.expect(200);
|
||||||
)
|
|
||||||
.send(memberData);
|
|
||||||
|
|
||||||
const {
|
const {
|
||||||
id,
|
id,
|
||||||
|
@ -121,21 +90,78 @@ describe('POST /invitations/:id/accept', () => {
|
||||||
expect(personalizationAnswers).toBeNull();
|
expect(personalizationAnswers).toBeNull();
|
||||||
expect(password).toBeUndefined();
|
expect(password).toBeUndefined();
|
||||||
expect(isPending).toBe(false);
|
expect(isPending).toBe(false);
|
||||||
expect(globalRole).toBeDefined();
|
expect(globalRole.scope).toBe('global');
|
||||||
|
expect(globalRole.name).toBe('member');
|
||||||
expect(apiKey).not.toBeDefined();
|
expect(apiKey).not.toBeDefined();
|
||||||
|
|
||||||
const authToken = utils.getAuthToken(response);
|
const authToken = utils.getAuthToken(response);
|
||||||
expect(authToken).toBeDefined();
|
expect(authToken).toBeDefined();
|
||||||
|
|
||||||
const member = await Container.get(UserRepository).findOneByOrFail({ id: memberShell.id });
|
const storedMember = await Container.get(UserRepository).findOneByOrFail({
|
||||||
expect(member.firstName).toBe(memberData.firstName);
|
id: memberShell.id,
|
||||||
expect(member.lastName).toBe(memberData.lastName);
|
});
|
||||||
expect(member.password).not.toBe(memberData.password);
|
|
||||||
|
expect(storedMember.firstName).toBe(memberData.firstName);
|
||||||
|
expect(storedMember.lastName).toBe(memberData.lastName);
|
||||||
|
expect(storedMember.password).not.toBe(memberData.password);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should fail with invalid inputs', async () => {
|
test('should fill out an admin shell', async () => {
|
||||||
|
const globalAdminRole = await getGlobalAdminRole();
|
||||||
|
const adminShell = await createUserShell(globalAdminRole);
|
||||||
|
|
||||||
|
const memberData = {
|
||||||
|
inviterId: owner.id,
|
||||||
|
firstName: randomName(),
|
||||||
|
lastName: randomName(),
|
||||||
|
password: randomValidPassword(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const response = await authlessAgent
|
||||||
|
.post(`/invitations/${adminShell.id}/accept`)
|
||||||
|
.send(memberData)
|
||||||
|
.expect(200);
|
||||||
|
|
||||||
|
const {
|
||||||
|
id,
|
||||||
|
email,
|
||||||
|
firstName,
|
||||||
|
lastName,
|
||||||
|
personalizationAnswers,
|
||||||
|
password,
|
||||||
|
globalRole,
|
||||||
|
isPending,
|
||||||
|
apiKey,
|
||||||
|
} = response.body.data;
|
||||||
|
|
||||||
|
expect(validator.isUUID(id)).toBe(true);
|
||||||
|
expect(email).toBeDefined();
|
||||||
|
expect(firstName).toBe(memberData.firstName);
|
||||||
|
expect(lastName).toBe(memberData.lastName);
|
||||||
|
expect(personalizationAnswers).toBeNull();
|
||||||
|
expect(password).toBeUndefined();
|
||||||
|
expect(isPending).toBe(false);
|
||||||
|
expect(globalRole.scope).toBe('global');
|
||||||
|
expect(globalRole.name).toBe('admin');
|
||||||
|
expect(apiKey).not.toBeDefined();
|
||||||
|
|
||||||
|
const authToken = utils.getAuthToken(response);
|
||||||
|
expect(authToken).toBeDefined();
|
||||||
|
|
||||||
|
const storedAdmin = await Container.get(UserRepository).findOneByOrFail({
|
||||||
|
id: adminShell.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(storedAdmin.firstName).toBe(memberData.firstName);
|
||||||
|
expect(storedAdmin.lastName).toBe(memberData.lastName);
|
||||||
|
expect(storedAdmin.password).not.toBe(memberData.password);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should fail with invalid payloads', async () => {
|
||||||
const memberShellEmail = randomEmail();
|
const memberShellEmail = randomEmail();
|
||||||
|
|
||||||
|
const globalMemberRole = await getGlobalMemberRole();
|
||||||
|
|
||||||
const memberShell = await Container.get(UserRepository).save({
|
const memberShell = await Container.get(UserRepository).save({
|
||||||
email: memberShellEmail,
|
email: memberShellEmail,
|
||||||
globalRole: globalMemberRole,
|
globalRole: globalMemberRole,
|
||||||
|
@ -171,51 +197,62 @@ describe('POST /invitations/:id/accept', () => {
|
||||||
];
|
];
|
||||||
|
|
||||||
for (const invalidPayload of invalidPayloads) {
|
for (const invalidPayload of invalidPayloads) {
|
||||||
const response = await authlessAgent
|
await authlessAgent
|
||||||
.post(`/invitations/${memberShell.id}/accept`)
|
.post(`/invitations/${memberShell.id}/accept`)
|
||||||
.send(invalidPayload);
|
.send(invalidPayload)
|
||||||
expect(response.statusCode).toBe(400);
|
.expect(400);
|
||||||
|
|
||||||
const storedUser = await Container.get(UserRepository).findOneOrFail({
|
const storedMemberShell = await Container.get(UserRepository).findOneOrFail({
|
||||||
where: { email: memberShellEmail },
|
where: { email: memberShellEmail },
|
||||||
});
|
});
|
||||||
|
|
||||||
expect(storedUser.firstName).toBeNull();
|
expect(storedMemberShell.firstName).toBeNull();
|
||||||
expect(storedUser.lastName).toBeNull();
|
expect(storedMemberShell.lastName).toBeNull();
|
||||||
expect(storedUser.password).toBeNull();
|
expect(storedMemberShell.password).toBeNull();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should fail with already accepted invite', async () => {
|
test('should fail with already accepted invite', async () => {
|
||||||
|
const globalMemberRole = await getGlobalMemberRole();
|
||||||
const member = await createUser({ globalRole: globalMemberRole });
|
const member = await createUser({ globalRole: globalMemberRole });
|
||||||
|
|
||||||
const newMemberData = {
|
const memberData = {
|
||||||
inviterId: owner.id,
|
inviterId: owner.id,
|
||||||
firstName: randomName(),
|
firstName: randomName(),
|
||||||
lastName: randomName(),
|
lastName: randomName(),
|
||||||
password: randomValidPassword(),
|
password: randomValidPassword(),
|
||||||
};
|
};
|
||||||
|
|
||||||
const response = await authlessAgent
|
await authlessAgent.post(`/invitations/${member.id}/accept`).send(memberData).expect(400);
|
||||||
.post(`/invitations/${member.id}/accept`)
|
|
||||||
.send(newMemberData);
|
|
||||||
|
|
||||||
expect(response.statusCode).toBe(400);
|
|
||||||
|
|
||||||
const storedMember = await Container.get(UserRepository).findOneOrFail({
|
const storedMember = await Container.get(UserRepository).findOneOrFail({
|
||||||
where: { email: member.email },
|
where: { email: member.email },
|
||||||
});
|
});
|
||||||
expect(storedMember.firstName).not.toBe(newMemberData.firstName);
|
|
||||||
expect(storedMember.lastName).not.toBe(newMemberData.lastName);
|
expect(storedMember.firstName).not.toBe(memberData.firstName);
|
||||||
|
expect(storedMember.lastName).not.toBe(memberData.lastName);
|
||||||
|
expect(storedMember.password).not.toBe(memberData.password);
|
||||||
|
|
||||||
const comparisonResult = await compareHash(member.password, storedMember.password);
|
const comparisonResult = await compareHash(member.password, storedMember.password);
|
||||||
|
|
||||||
expect(comparisonResult).toBe(false);
|
expect(comparisonResult).toBe(false);
|
||||||
expect(storedMember.password).not.toBe(newMemberData.password);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('POST /invitations', () => {
|
describe('POST /invitations', () => {
|
||||||
test('should fail with invalid inputs', async () => {
|
let owner: User;
|
||||||
|
let member: User;
|
||||||
|
let ownerAgent: SuperAgentTest;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
await testDb.truncate(['User']);
|
||||||
|
|
||||||
|
owner = await createOwner();
|
||||||
|
member = await createMember();
|
||||||
|
ownerAgent = testServer.authAgentFor(owner);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should fail with invalid payloads', async () => {
|
||||||
const invalidPayloads = [
|
const invalidPayloads = [
|
||||||
randomEmail(),
|
randomEmail(),
|
||||||
[randomEmail()],
|
[randomEmail()],
|
||||||
|
@ -226,48 +263,96 @@ describe('POST /invitations', () => {
|
||||||
|
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
invalidPayloads.map(async (invalidPayload) => {
|
invalidPayloads.map(async (invalidPayload) => {
|
||||||
const response = await authOwnerAgent.post('/invitations').send(invalidPayload);
|
await ownerAgent.post('/invitations').send(invalidPayload).expect(400);
|
||||||
expect(response.statusCode).toBe(400);
|
|
||||||
|
|
||||||
const users = await Container.get(UserRepository).find();
|
const usersCount = await Container.get(UserRepository).count();
|
||||||
expect(users.length).toBe(2); // DB unaffected
|
|
||||||
|
expect(usersCount).toBe(2); // DB unaffected
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should ignore an empty payload', async () => {
|
test('should return 200 on empty payload', async () => {
|
||||||
const response = await authOwnerAgent.post('/invitations').send([]);
|
const response = await ownerAgent.post('/invitations').send([]).expect(200);
|
||||||
|
|
||||||
const { data } = response.body;
|
expect(response.body.data).toStrictEqual([]);
|
||||||
|
|
||||||
expect(response.statusCode).toBe(200);
|
const usersCount = await Container.get(UserRepository).count();
|
||||||
expect(Array.isArray(data)).toBe(true);
|
|
||||||
expect(data.length).toBe(0);
|
|
||||||
|
|
||||||
const users = await Container.get(UserRepository).find();
|
expect(usersCount).toBe(2);
|
||||||
expect(users.length).toBe(2);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should succeed if emailing is not set up', async () => {
|
test('should return 200 if emailing is not set up', async () => {
|
||||||
mailer.invite.mockResolvedValueOnce({ emailSent: false });
|
mailer.invite.mockResolvedValue({ emailSent: false });
|
||||||
const usersToInvite = randomEmail();
|
|
||||||
const response = await authOwnerAgent.post('/invitations').send([{ email: usersToInvite }]);
|
const response = await ownerAgent
|
||||||
|
.post('/invitations')
|
||||||
|
.send([{ email: randomEmail() }])
|
||||||
|
.expect(200);
|
||||||
|
|
||||||
expect(response.statusCode).toBe(200);
|
|
||||||
expect(response.body.data).toBeInstanceOf(Array);
|
expect(response.body.data).toBeInstanceOf(Array);
|
||||||
expect(response.body.data.length).toBe(1);
|
expect(response.body.data.length).toBe(1);
|
||||||
|
|
||||||
const { user } = response.body.data[0];
|
const { user } = response.body.data[0];
|
||||||
|
|
||||||
expect(user.inviteAcceptUrl).toBeDefined();
|
expect(user.inviteAcceptUrl).toBeDefined();
|
||||||
|
|
||||||
const inviteUrl = new URL(user.inviteAcceptUrl);
|
const inviteUrl = new URL(user.inviteAcceptUrl);
|
||||||
|
|
||||||
expect(inviteUrl.searchParams.get('inviterId')).toBe(owner.id);
|
expect(inviteUrl.searchParams.get('inviterId')).toBe(owner.id);
|
||||||
expect(inviteUrl.searchParams.get('inviteeId')).toBe(user.id);
|
expect(inviteUrl.searchParams.get('inviteeId')).toBe(user.id);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('should create member shell', async () => {
|
||||||
|
mailer.invite.mockResolvedValue({ emailSent: false });
|
||||||
|
|
||||||
|
const response = await ownerAgent
|
||||||
|
.post('/invitations')
|
||||||
|
.send([{ email: randomEmail() }])
|
||||||
|
.expect(200);
|
||||||
|
|
||||||
|
const [result] = response.body.data as UserInvitationResponse[];
|
||||||
|
|
||||||
|
const storedUser = await Container.get(UserRepository).findOneByOrFail({
|
||||||
|
id: result.user.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
assertInvitedUsersOnDb(storedUser);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should create admin shell if licensed', async () => {
|
||||||
|
mailer.invite.mockResolvedValue({ emailSent: false });
|
||||||
|
|
||||||
|
const response = await ownerAgent
|
||||||
|
.post('/invitations')
|
||||||
|
.send([{ email: randomEmail(), role: 'admin' }])
|
||||||
|
.expect(200);
|
||||||
|
|
||||||
|
const [result] = response.body.data as UserInvitationResponse[];
|
||||||
|
|
||||||
|
const storedUser = await Container.get(UserRepository).findOneByOrFail({
|
||||||
|
id: result.user.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
assertInvitedUsersOnDb(storedUser);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should fail to create admin shell if not licensed', async () => {
|
||||||
|
license.isAdvancedPermissionsLicensed.mockReturnValue(false);
|
||||||
|
mailer.invite.mockResolvedValue({ emailSent: false });
|
||||||
|
|
||||||
|
await ownerAgent
|
||||||
|
.post('/invitations')
|
||||||
|
.send([{ email: randomEmail(), role: 'admin' }])
|
||||||
|
.expect(403);
|
||||||
|
});
|
||||||
|
|
||||||
test('should email invites and create user shells but ignore existing', async () => {
|
test('should email invites and create user shells but ignore existing', async () => {
|
||||||
const internalHooks = Container.get(InternalHooks);
|
externalHooks.run.mockClear();
|
||||||
|
|
||||||
mailer.invite.mockImplementation(async () => ({ emailSent: true }));
|
mailer.invite.mockResolvedValue({ emailSent: true });
|
||||||
|
|
||||||
|
const globalMemberRole = await getGlobalMemberRole();
|
||||||
const memberShell = await createUserShell(globalMemberRole);
|
const memberShell = await createUserShell(globalMemberRole);
|
||||||
|
|
||||||
const newUser = randomEmail();
|
const newUser = randomEmail();
|
||||||
|
@ -281,54 +366,54 @@ describe('POST /invitations', () => {
|
||||||
|
|
||||||
const payload = testEmails.map((email) => ({ email }));
|
const payload = testEmails.map((email) => ({ email }));
|
||||||
|
|
||||||
const response = await authOwnerAgent.post('/invitations').send(payload);
|
const response = await ownerAgent.post('/invitations').send(payload).expect(200);
|
||||||
|
|
||||||
expect(response.statusCode).toBe(200);
|
const internalHooks = Container.get(InternalHooks);
|
||||||
|
|
||||||
expect(internalHooks.onUserTransactionalEmail).toHaveBeenCalledTimes(usersToInvite.length);
|
expect(internalHooks.onUserTransactionalEmail).toHaveBeenCalledTimes(usersToInvite.length);
|
||||||
|
|
||||||
expect(externalHooks.run).toHaveBeenCalledTimes(1);
|
expect(externalHooks.run).toHaveBeenCalledTimes(1);
|
||||||
|
|
||||||
const [hookName, hookData] = externalHooks.run.mock.calls[0];
|
const [hookName, hookData] = externalHooks.run.mock.calls[0];
|
||||||
|
|
||||||
expect(hookName).toBe('user.invited');
|
expect(hookName).toBe('user.invited');
|
||||||
expect(hookData?.[0]).toStrictEqual(usersToCreate);
|
expect(hookData?.[0]).toStrictEqual(usersToCreate);
|
||||||
|
|
||||||
for (const invitationResponse of response.body.data as UserInvitationResponse[]) {
|
const result = response.body.data as UserInvitationResponse[];
|
||||||
|
|
||||||
|
for (const invitationResponse of result) {
|
||||||
|
assertInviteUserSuccessResponse(invitationResponse);
|
||||||
|
|
||||||
const storedUser = await Container.get(UserRepository).findOneByOrFail({
|
const storedUser = await Container.get(UserRepository).findOneByOrFail({
|
||||||
id: invitationResponse.user.id,
|
id: invitationResponse.user.id,
|
||||||
});
|
});
|
||||||
|
|
||||||
assertInviteUserSuccessResponse(invitationResponse);
|
|
||||||
|
|
||||||
assertInvitedUsersOnDb(storedUser);
|
assertInvitedUsersOnDb(storedUser);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const [onUserTransactionalEmailParameter] of internalHooks.onUserTransactionalEmail.mock
|
const calls = mocked(internalHooks).onUserTransactionalEmail.mock.calls;
|
||||||
.calls) {
|
|
||||||
|
for (const [onUserTransactionalEmailParameter] of calls) {
|
||||||
expect(onUserTransactionalEmailParameter.user_id).toBeDefined();
|
expect(onUserTransactionalEmailParameter.user_id).toBeDefined();
|
||||||
expect(onUserTransactionalEmailParameter.message_type).toBe('New user invite');
|
expect(onUserTransactionalEmailParameter.message_type).toBe('New user invite');
|
||||||
expect(onUserTransactionalEmailParameter.public_api).toBe(false);
|
expect(onUserTransactionalEmailParameter.public_api).toBe(false);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should return error when invite method throws an error', async () => {
|
test('should return 200 when invite method throws error', async () => {
|
||||||
const error = 'failed to send email';
|
|
||||||
|
|
||||||
mailer.invite.mockImplementation(async () => {
|
mailer.invite.mockImplementation(async () => {
|
||||||
throw new Error(error);
|
throw new Error('failed to send email');
|
||||||
});
|
});
|
||||||
|
|
||||||
const newUser = randomEmail();
|
const response = await ownerAgent
|
||||||
|
.post('/invitations')
|
||||||
const usersToCreate = [newUser];
|
.send([{ email: randomEmail() }])
|
||||||
|
.expect(200);
|
||||||
const payload = usersToCreate.map((email) => ({ email }));
|
|
||||||
|
|
||||||
const response = await authOwnerAgent.post('/invitations').send(payload);
|
|
||||||
|
|
||||||
expect(response.body.data).toBeInstanceOf(Array);
|
expect(response.body.data).toBeInstanceOf(Array);
|
||||||
expect(response.body.data.length).toBe(1);
|
expect(response.body.data.length).toBe(1);
|
||||||
expect(response.statusCode).toBe(200);
|
|
||||||
const invitationResponse = response.body.data[0];
|
const [invitationResponse] = response.body.data;
|
||||||
|
|
||||||
assertInviteUserErrorResponse(invitationResponse);
|
assertInviteUserErrorResponse(invitationResponse);
|
||||||
});
|
});
|
||||||
|
|
|
@ -250,6 +250,7 @@ export const setupTestServer = ({
|
||||||
Container.get(ActiveWorkflowRunner),
|
Container.get(ActiveWorkflowRunner),
|
||||||
Container.get(RS),
|
Container.get(RS),
|
||||||
Container.get(US),
|
Container.get(US),
|
||||||
|
Container.get(License),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
|
@ -268,6 +269,7 @@ export const setupTestServer = ({
|
||||||
Container.get(InternalHooks),
|
Container.get(InternalHooks),
|
||||||
Container.get(EHS),
|
Container.get(EHS),
|
||||||
Container.get(USE),
|
Container.get(USE),
|
||||||
|
Container.get(License),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
|
import validator from 'validator';
|
||||||
import type { PublicUser } from '@/Interfaces';
|
import type { PublicUser } from '@/Interfaces';
|
||||||
|
import type { User } from '@/databases/entities/User';
|
||||||
|
|
||||||
export const validateUser = (user: PublicUser) => {
|
export const validateUser = (user: PublicUser) => {
|
||||||
expect(typeof user.id).toBe('string');
|
expect(typeof user.id).toBe('string');
|
||||||
|
@ -13,3 +15,31 @@ export const validateUser = (user: PublicUser) => {
|
||||||
expect(user.password).toBeUndefined();
|
expect(user.password).toBeUndefined();
|
||||||
expect(user.globalRole).toBeDefined();
|
expect(user.globalRole).toBeDefined();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const assertInviteUserSuccessResponse = (data: UserInvitationResponse) => {
|
||||||
|
expect(validator.isUUID(data.user.id)).toBe(true);
|
||||||
|
expect(data.user.inviteAcceptUrl).toBeUndefined();
|
||||||
|
expect(data.user.email).toBeDefined();
|
||||||
|
expect(data.user.emailSent).toBe(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const assertInviteUserErrorResponse = (data: UserInvitationResponse) => {
|
||||||
|
expect(validator.isUUID(data.user.id)).toBe(true);
|
||||||
|
expect(data.user.inviteAcceptUrl).toBeDefined();
|
||||||
|
expect(data.user.email).toBeDefined();
|
||||||
|
expect(data.user.emailSent).toBe(false);
|
||||||
|
expect(data.error).toBeDefined();
|
||||||
|
};
|
||||||
|
|
||||||
|
export const assertInvitedUsersOnDb = (user: User) => {
|
||||||
|
expect(user.firstName).toBeNull();
|
||||||
|
expect(user.lastName).toBeNull();
|
||||||
|
expect(user.personalizationAnswers).toBeNull();
|
||||||
|
expect(user.password).toBeNull();
|
||||||
|
expect(user.isPending).toBe(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
export type UserInvitationResponse = {
|
||||||
|
user: Pick<User, 'id' | 'email'> & { inviteAcceptUrl: string; emailSent: boolean };
|
||||||
|
error?: string;
|
||||||
|
};
|
||||||
|
|
|
@ -18,9 +18,16 @@ import * as testDb from './shared/testDb';
|
||||||
import type { SuperAgentTest } from 'supertest';
|
import type { SuperAgentTest } from 'supertest';
|
||||||
import type { Role } from '@db/entities/Role';
|
import type { Role } from '@db/entities/Role';
|
||||||
import type { User } from '@db/entities/User';
|
import type { User } from '@db/entities/User';
|
||||||
|
import { License } from '@/License';
|
||||||
|
import { mockInstance } from '../shared/mocking';
|
||||||
|
|
||||||
const testServer = utils.setupTestServer({ endpointGroups: ['users'] });
|
const testServer = utils.setupTestServer({ endpointGroups: ['users'] });
|
||||||
|
|
||||||
|
const license = mockInstance(License, {
|
||||||
|
isAdvancedPermissionsLicensed: jest.fn().mockReturnValue(true),
|
||||||
|
isWithinUsersLimit: jest.fn().mockReturnValue(true),
|
||||||
|
});
|
||||||
|
|
||||||
describe('GET /users', () => {
|
describe('GET /users', () => {
|
||||||
let owner: User;
|
let owner: User;
|
||||||
let member: User;
|
let member: User;
|
||||||
|
@ -362,6 +369,7 @@ describe('PATCH /users/:id/role', () => {
|
||||||
NO_USER_TO_OWNER,
|
NO_USER_TO_OWNER,
|
||||||
NO_USER,
|
NO_USER,
|
||||||
NO_OWNER_ON_OWNER,
|
NO_OWNER_ON_OWNER,
|
||||||
|
NO_ADMIN_IF_UNLICENSED,
|
||||||
} = UsersController.ERROR_MESSAGES.CHANGE_ROLE;
|
} = UsersController.ERROR_MESSAGES.CHANGE_ROLE;
|
||||||
|
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
|
@ -518,6 +526,17 @@ describe('PATCH /users/:id/role', () => {
|
||||||
expect(response.body.message).toBe(NO_USER_TO_OWNER);
|
expect(response.body.message).toBe(NO_USER_TO_OWNER);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('should fail to promote member to admin if not licensed', async () => {
|
||||||
|
license.isAdvancedPermissionsLicensed.mockReturnValueOnce(false);
|
||||||
|
|
||||||
|
const response = await adminAgent.patch(`/users/${member.id}/role`).send({
|
||||||
|
newRole: { scope: 'global', name: 'admin' },
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(response.statusCode).toBe(403);
|
||||||
|
expect(response.body.message).toBe(NO_ADMIN_IF_UNLICENSED);
|
||||||
|
});
|
||||||
|
|
||||||
test('should be able to demote admin to member', async () => {
|
test('should be able to demote admin to member', async () => {
|
||||||
const response = await adminAgent.patch(`/users/${otherAdmin.id}/role`).send({
|
const response = await adminAgent.patch(`/users/${otherAdmin.id}/role`).send({
|
||||||
newRole: { scope: 'global', name: 'member' },
|
newRole: { scope: 'global', name: 'member' },
|
||||||
|
@ -556,7 +575,7 @@ describe('PATCH /users/:id/role', () => {
|
||||||
adminAgent = testServer.authAgentFor(admin);
|
adminAgent = testServer.authAgentFor(admin);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should be able to promote member to admin', async () => {
|
test('should be able to promote member to admin if licensed', async () => {
|
||||||
const response = await adminAgent.patch(`/users/${member.id}/role`).send({
|
const response = await adminAgent.patch(`/users/${member.id}/role`).send({
|
||||||
newRole: { scope: 'global', name: 'admin' },
|
newRole: { scope: 'global', name: 'admin' },
|
||||||
});
|
});
|
||||||
|
@ -613,7 +632,18 @@ describe('PATCH /users/:id/role', () => {
|
||||||
expect(response.body.message).toBe(NO_USER_TO_OWNER);
|
expect(response.body.message).toBe(NO_USER_TO_OWNER);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should be able to promote member to admin', async () => {
|
test('should fail to promote member to admin if not licensed', async () => {
|
||||||
|
license.isAdvancedPermissionsLicensed.mockReturnValueOnce(false);
|
||||||
|
|
||||||
|
const response = await ownerAgent.patch(`/users/${member.id}/role`).send({
|
||||||
|
newRole: { scope: 'global', name: 'admin' },
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(response.statusCode).toBe(403);
|
||||||
|
expect(response.body.message).toBe(NO_ADMIN_IF_UNLICENSED);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should be able to promote member to admin if licensed', async () => {
|
||||||
const response = await ownerAgent.patch(`/users/${member.id}/role`).send({
|
const response = await ownerAgent.patch(`/users/${member.id}/role`).send({
|
||||||
newRole: { scope: 'global', name: 'admin' },
|
newRole: { scope: 'global', name: 'admin' },
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in a new issue