2018-08-21 08:05:34 +02:00
|
|
|
import _ from 'lodash';
|
2021-10-29 17:33:05 +02:00
|
|
|
import path from 'path';
|
2024-09-29 12:47:10 +02:00
|
|
|
import { describe, expect, test, vi } from 'vitest';
|
2020-11-15 11:14:09 +01:00
|
|
|
|
|
|
|
import {
|
|
|
|
Config as AppConfig,
|
|
|
|
ROLES,
|
|
|
|
createAnonymousRemoteUser,
|
2021-10-29 17:33:05 +02:00
|
|
|
createRemoteUser,
|
|
|
|
parseConfigFile,
|
2020-11-15 11:14:09 +01:00
|
|
|
} from '@verdaccio/config';
|
2022-08-19 20:25:20 +02:00
|
|
|
import { getDefaultConfig } from '@verdaccio/config';
|
2024-09-29 12:47:10 +02:00
|
|
|
import { API_ERROR, CHARACTER_ENCODING, VerdaccioError, errorUtils } from '@verdaccio/core';
|
2024-10-20 19:26:36 +02:00
|
|
|
import { logger, setup } from '@verdaccio/logger';
|
2024-09-29 12:47:10 +02:00
|
|
|
import { aesDecrypt, verifyPayload } from '@verdaccio/signature';
|
|
|
|
import { Config, RemoteUser } from '@verdaccio/types';
|
|
|
|
import { getAuthenticatedMessage } from '@verdaccio/utils';
|
2021-10-29 17:33:05 +02:00
|
|
|
|
2020-09-25 20:16:16 +02:00
|
|
|
import {
|
|
|
|
ActionsAllowed,
|
2023-02-26 13:19:22 +01:00
|
|
|
AllowActionCallbackResponse,
|
2021-10-29 17:33:05 +02:00
|
|
|
Auth,
|
2020-09-25 20:16:16 +02:00
|
|
|
allow_action,
|
2021-10-29 17:33:05 +02:00
|
|
|
getApiToken,
|
2020-09-25 20:16:16 +02:00
|
|
|
getDefaultPlugins,
|
|
|
|
} from '../src';
|
2020-03-08 09:19:12 +01:00
|
|
|
|
2023-02-26 13:19:22 +01:00
|
|
|
setup({});
|
2018-08-21 08:05:34 +02:00
|
|
|
|
2020-03-03 23:59:19 +01:00
|
|
|
const parseConfigurationFile = (conf) => {
|
|
|
|
const { name, ext } = path.parse(conf);
|
|
|
|
const format = ext.startsWith('.') ? ext.substring(1) : 'yaml';
|
|
|
|
|
|
|
|
return path.join(__dirname, `./partials/config/${format}/security/${name}.${format}`);
|
|
|
|
};
|
|
|
|
|
2018-08-21 08:05:34 +02:00
|
|
|
describe('Auth utilities', () => {
|
2024-09-29 12:47:10 +02:00
|
|
|
vi.setConfig({ testTimeout: 20000 });
|
2019-07-27 07:20:30 +02:00
|
|
|
|
2018-08-21 08:05:34 +02:00
|
|
|
const parseConfigurationSecurityFile = (name) => {
|
|
|
|
return parseConfigurationFile(`security/${name}`);
|
|
|
|
};
|
|
|
|
|
|
|
|
function getConfig(configFileName: string, secret: string) {
|
|
|
|
const conf = parseConfigFile(parseConfigurationSecurityFile(configFileName));
|
2020-08-11 07:21:51 +02:00
|
|
|
// @ts-ignore
|
2022-08-19 20:25:20 +02:00
|
|
|
const secConf = _.merge(getDefaultConfig(), conf);
|
2024-09-29 12:47:10 +02:00
|
|
|
// @ts-expect-error
|
2018-08-21 08:05:34 +02:00
|
|
|
secConf.secret = secret;
|
|
|
|
const config: Config = new AppConfig(secConf);
|
|
|
|
|
|
|
|
return config;
|
|
|
|
}
|
|
|
|
|
2020-10-03 05:47:04 -07:00
|
|
|
async function getTokenByConfiguration(
|
2018-08-21 08:05:34 +02:00
|
|
|
configFileName: string,
|
|
|
|
username: string,
|
|
|
|
password: string,
|
|
|
|
secret = '12345',
|
|
|
|
methodToSpy: string,
|
2020-08-13 23:27:00 +02:00
|
|
|
methodNotBeenCalled: string
|
|
|
|
): Promise<string> {
|
2019-07-16 08:40:01 +02:00
|
|
|
const config: Config = getConfig(configFileName, secret);
|
2024-10-20 19:26:36 +02:00
|
|
|
const auth: Auth = new Auth(config, logger);
|
2022-09-16 08:02:08 +02:00
|
|
|
await auth.init();
|
2019-07-16 08:40:01 +02:00
|
|
|
// @ts-ignore
|
2024-09-29 12:47:10 +02:00
|
|
|
const spy = vi.spyOn(auth, methodToSpy);
|
2019-07-16 08:40:01 +02:00
|
|
|
// @ts-ignore
|
2024-09-29 12:47:10 +02:00
|
|
|
const spyNotCalled = vi.spyOn(auth, methodNotBeenCalled);
|
2019-07-16 08:40:01 +02:00
|
|
|
const user: RemoteUser = {
|
|
|
|
name: username,
|
2022-10-28 23:38:22 +02:00
|
|
|
real_groups: ['test', '$all', '$authenticated', '@all', '@authenticated', 'all'],
|
|
|
|
groups: ['company-role1', 'company-role2'],
|
2019-07-16 08:40:01 +02:00
|
|
|
};
|
|
|
|
const token = await getApiToken(auth, config, user, password);
|
|
|
|
expect(spy).toHaveBeenCalled();
|
|
|
|
expect(spy).toHaveBeenCalledTimes(1);
|
|
|
|
expect(spyNotCalled).not.toHaveBeenCalled();
|
|
|
|
expect(token).toBeDefined();
|
|
|
|
|
2020-10-03 05:47:04 -07:00
|
|
|
return token as string;
|
2018-08-21 08:05:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const verifyJWT = (token: string, user: string, password: string, secret: string) => {
|
|
|
|
const payload = verifyPayload(token, secret);
|
|
|
|
expect(payload.name).toBe(user);
|
|
|
|
expect(payload.groups).toBeDefined();
|
2022-10-28 23:38:22 +02:00
|
|
|
expect(payload.groups).toEqual([
|
|
|
|
'company-role1',
|
|
|
|
'company-role2',
|
|
|
|
'test',
|
|
|
|
'$all',
|
|
|
|
'$authenticated',
|
|
|
|
'@all',
|
|
|
|
'@authenticated',
|
|
|
|
'all',
|
|
|
|
]);
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(payload.real_groups).toBeDefined();
|
2022-10-28 23:38:22 +02:00
|
|
|
expect(payload.real_groups).toEqual([
|
|
|
|
'test',
|
|
|
|
'$all',
|
|
|
|
'$authenticated',
|
|
|
|
'@all',
|
|
|
|
'@authenticated',
|
|
|
|
'all',
|
|
|
|
]);
|
2018-08-21 08:05:34 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
const verifyAES = (token: string, user: string, password: string, secret: string) => {
|
2020-11-08 15:20:02 +01:00
|
|
|
// @ts-ignore
|
|
|
|
const payload = aesDecrypt(token, secret).toString(CHARACTER_ENCODING.UTF8);
|
2018-08-21 08:05:34 +02:00
|
|
|
const content = payload.split(':');
|
|
|
|
|
|
|
|
expect(content[0]).toBe(user);
|
|
|
|
expect(content[0]).toBe(password);
|
|
|
|
};
|
|
|
|
|
2020-09-25 20:16:16 +02:00
|
|
|
describe('getDefaultPlugins', () => {
|
|
|
|
test('authentication should fail by default (default)', () => {
|
2024-09-29 12:47:10 +02:00
|
|
|
const plugin = getDefaultPlugins({ trace: vi.fn() });
|
2020-09-25 20:16:16 +02:00
|
|
|
plugin.authenticate('foo', 'bar', (error: any) => {
|
2021-09-26 00:08:00 +02:00
|
|
|
expect(error).toEqual(errorUtils.getForbidden(API_ERROR.BAD_USERNAME_PASSWORD));
|
2020-09-25 20:16:16 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
test('add user should fail by default (default)', () => {
|
2024-09-29 12:47:10 +02:00
|
|
|
const plugin = getDefaultPlugins({ trace: vi.fn() });
|
2020-09-25 20:16:16 +02:00
|
|
|
// @ts-ignore
|
|
|
|
plugin.adduser('foo', 'bar', (error: any) => {
|
2021-09-26 00:08:00 +02:00
|
|
|
expect(error).toEqual(errorUtils.getForbidden(API_ERROR.BAD_USERNAME_PASSWORD));
|
2020-09-25 20:16:16 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('allow_action', () => {
|
|
|
|
describe('access/publish/unpublish and anonymous', () => {
|
|
|
|
const packageAccess = {
|
|
|
|
name: 'foo',
|
|
|
|
version: undefined,
|
|
|
|
access: ['foo'],
|
|
|
|
unpublish: false,
|
|
|
|
};
|
|
|
|
|
|
|
|
// const type = 'access';
|
|
|
|
test.each(['access', 'publish', 'unpublish'])(
|
|
|
|
'should restrict %s to anonymous users',
|
|
|
|
(type) => {
|
2024-09-29 12:47:10 +02:00
|
|
|
allow_action(type as ActionsAllowed, { trace: vi.fn() })(
|
2020-09-25 20:16:16 +02:00
|
|
|
createAnonymousRemoteUser(),
|
|
|
|
{
|
|
|
|
...packageAccess,
|
|
|
|
[type]: ['foo'],
|
|
|
|
},
|
|
|
|
(error: VerdaccioError | null, allowed: AllowActionCallbackResponse) => {
|
|
|
|
expect(error).not.toBeNull();
|
|
|
|
expect(allowed).toBeUndefined();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
test.each(['access', 'publish', 'unpublish'])(
|
|
|
|
'should allow %s to anonymous users',
|
|
|
|
(type) => {
|
2024-09-29 12:47:10 +02:00
|
|
|
allow_action(type as ActionsAllowed, { trace: vi.fn() })(
|
2020-09-25 20:16:16 +02:00
|
|
|
createAnonymousRemoteUser(),
|
|
|
|
{
|
|
|
|
...packageAccess,
|
|
|
|
[type]: [ROLES.$ANONYMOUS],
|
|
|
|
},
|
|
|
|
(error: VerdaccioError | null, allowed: AllowActionCallbackResponse) => {
|
|
|
|
expect(error).toBeNull();
|
|
|
|
expect(allowed).toBe(true);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
test.each(['access', 'publish', 'unpublish'])(
|
|
|
|
'should allow %s only if user is anonymous if the logged user has groups',
|
|
|
|
(type) => {
|
2024-09-29 12:47:10 +02:00
|
|
|
allow_action(type as ActionsAllowed, { trace: vi.fn() })(
|
2020-09-25 20:16:16 +02:00
|
|
|
createRemoteUser('juan', ['maintainer', 'admin']),
|
|
|
|
{
|
|
|
|
...packageAccess,
|
|
|
|
[type]: [ROLES.$ANONYMOUS],
|
|
|
|
},
|
|
|
|
(error: VerdaccioError | null, allowed: AllowActionCallbackResponse) => {
|
|
|
|
expect(error).not.toBeNull();
|
|
|
|
expect(allowed).toBeUndefined();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
test.each(['access', 'publish', 'unpublish'])(
|
|
|
|
'should allow %s only if user is anonymous match any other groups',
|
|
|
|
(type) => {
|
2024-09-29 12:47:10 +02:00
|
|
|
allow_action(type as ActionsAllowed, { trace: vi.fn() })(
|
2020-09-25 20:16:16 +02:00
|
|
|
createRemoteUser('juan', ['maintainer', 'admin']),
|
|
|
|
{
|
|
|
|
...packageAccess,
|
|
|
|
[type]: ['admin', 'some-other-group', ROLES.$ANONYMOUS],
|
|
|
|
},
|
|
|
|
(error: VerdaccioError | null, allowed: AllowActionCallbackResponse) => {
|
|
|
|
expect(error).toBeNull();
|
|
|
|
expect(allowed).toBe(true);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
test.each(['access', 'publish', 'unpublish'])(
|
|
|
|
'should not allow %s anonymous if other groups are defined and does not match',
|
|
|
|
(type) => {
|
2024-09-29 12:47:10 +02:00
|
|
|
allow_action(type as ActionsAllowed, { trace: vi.fn() })(
|
2020-09-25 20:16:16 +02:00
|
|
|
createRemoteUser('juan', ['maintainer', 'admin']),
|
|
|
|
{
|
|
|
|
...packageAccess,
|
|
|
|
[type]: ['bla-bla-group', 'some-other-group', ROLES.$ANONYMOUS],
|
|
|
|
},
|
|
|
|
(error: VerdaccioError | null, allowed: AllowActionCallbackResponse) => {
|
|
|
|
expect(error).not.toBeNull();
|
|
|
|
expect(allowed).toBeUndefined();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-10-28 23:38:22 +02:00
|
|
|
describe('createRemoteUser', () => {
|
|
|
|
test('create remote user', () => {
|
|
|
|
expect(createRemoteUser('test', [])).toEqual({
|
|
|
|
name: 'test',
|
|
|
|
real_groups: [],
|
|
|
|
groups: ['$all', '$authenticated', '@all', '@authenticated', 'all'],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
test('create remote user with groups', () => {
|
|
|
|
expect(createRemoteUser('test', ['group1', 'group2'])).toEqual({
|
|
|
|
name: 'test',
|
|
|
|
real_groups: ['group1', 'group2'],
|
|
|
|
groups: ['group1', 'group2', '$all', '$authenticated', '@all', '@authenticated', 'all'],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
test('create anonymous remote user', () => {
|
|
|
|
expect(createAnonymousRemoteUser()).toEqual({
|
|
|
|
name: undefined,
|
|
|
|
real_groups: [],
|
|
|
|
groups: ['$all', '$anonymous', '@all', '@anonymous'],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2018-08-21 08:05:34 +02:00
|
|
|
describe('getApiToken test', () => {
|
|
|
|
test('should sign token with aes and security missing', async () => {
|
2020-10-03 05:47:04 -07:00
|
|
|
const token = await getTokenByConfiguration(
|
2020-09-17 06:48:16 +02:00
|
|
|
'security-missing',
|
|
|
|
'test',
|
|
|
|
'test',
|
2020-10-03 05:47:04 -07:00
|
|
|
'b2df428b9929d3ace7c598bbf4e496b2',
|
2020-09-17 06:48:16 +02:00
|
|
|
'aesEncrypt',
|
|
|
|
'jwtEncrypt'
|
|
|
|
);
|
2018-08-21 08:05:34 +02:00
|
|
|
|
2020-10-03 05:47:04 -07:00
|
|
|
verifyAES(token, 'test', 'test', 'b2df428b9929d3ace7c598bbf4e496b2');
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(_.isString(token)).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
2018-09-21 17:34:12 +02:00
|
|
|
test('should sign token with aes and security empty', async () => {
|
2020-10-03 05:47:04 -07:00
|
|
|
const token = await getTokenByConfiguration(
|
2020-09-17 06:48:16 +02:00
|
|
|
'security-empty',
|
|
|
|
'test',
|
|
|
|
'test',
|
2020-10-03 05:47:04 -07:00
|
|
|
'b2df428b9929d3ace7c598bbf4e496b2',
|
2020-09-17 06:48:16 +02:00
|
|
|
'aesEncrypt',
|
|
|
|
'jwtEncrypt'
|
|
|
|
);
|
2018-08-21 08:05:34 +02:00
|
|
|
|
2020-10-03 05:47:04 -07:00
|
|
|
verifyAES(token, 'test', 'test', 'b2df428b9929d3ace7c598bbf4e496b2');
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(_.isString(token)).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should sign token with aes', async () => {
|
2020-10-03 05:47:04 -07:00
|
|
|
const token = await getTokenByConfiguration(
|
2020-09-17 06:48:16 +02:00
|
|
|
'security-basic',
|
|
|
|
'test',
|
|
|
|
'test',
|
2020-10-03 05:47:04 -07:00
|
|
|
'b2df428b9929d3ace7c598bbf4e496b2',
|
2020-09-17 06:48:16 +02:00
|
|
|
'aesEncrypt',
|
|
|
|
'jwtEncrypt'
|
|
|
|
);
|
2018-08-21 08:05:34 +02:00
|
|
|
|
2020-10-03 05:47:04 -07:00
|
|
|
verifyAES(token, 'test', 'test', 'b2df428b9929d3ace7c598bbf4e496b2');
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(_.isString(token)).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should sign token with legacy and jwt disabled', async () => {
|
2020-10-03 05:47:04 -07:00
|
|
|
const token = await getTokenByConfiguration(
|
2020-09-17 06:48:16 +02:00
|
|
|
'security-no-legacy',
|
|
|
|
'test',
|
|
|
|
'test',
|
2020-10-03 05:47:04 -07:00
|
|
|
'b2df428b9929d3ace7c598bbf4e496b2',
|
2020-09-17 06:48:16 +02:00
|
|
|
'aesEncrypt',
|
|
|
|
'jwtEncrypt'
|
|
|
|
);
|
2018-08-21 08:05:34 +02:00
|
|
|
|
|
|
|
expect(_.isString(token)).toBeTruthy();
|
2020-10-03 05:47:04 -07:00
|
|
|
verifyAES(token, 'test', 'test', 'b2df428b9929d3ace7c598bbf4e496b2');
|
2018-08-21 08:05:34 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
test('should sign token with legacy enabled and jwt enabled', async () => {
|
2020-10-03 05:47:04 -07:00
|
|
|
const token = await getTokenByConfiguration(
|
2020-09-17 06:48:16 +02:00
|
|
|
'security-jwt-legacy-enabled',
|
|
|
|
'test',
|
|
|
|
'test',
|
2020-10-03 05:47:04 -07:00
|
|
|
'b2df428b9929d3ace7c598bbf4e496b2',
|
2020-09-17 06:48:16 +02:00
|
|
|
'jwtEncrypt',
|
|
|
|
'aesEncrypt'
|
|
|
|
);
|
2018-08-21 08:05:34 +02:00
|
|
|
|
2020-10-03 05:47:04 -07:00
|
|
|
verifyJWT(token, 'test', 'test', 'b2df428b9929d3ace7c598bbf4e496b2');
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(_.isString(token)).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should sign token with jwt enabled', async () => {
|
2020-10-03 05:47:04 -07:00
|
|
|
const token = await getTokenByConfiguration(
|
2020-09-17 06:48:16 +02:00
|
|
|
'security-jwt',
|
|
|
|
'test',
|
|
|
|
'test',
|
2020-10-03 05:47:04 -07:00
|
|
|
'b2df428b9929d3ace7c598bbf4e496b2',
|
2020-09-17 06:48:16 +02:00
|
|
|
'jwtEncrypt',
|
|
|
|
'aesEncrypt'
|
|
|
|
);
|
2018-08-21 08:05:34 +02:00
|
|
|
|
2019-07-16 08:40:01 +02:00
|
|
|
expect(_.isString(token)).toBeTruthy();
|
2020-10-03 05:47:04 -07:00
|
|
|
verifyJWT(token, 'test', 'test', 'b2df428b9929d3ace7c598bbf4e496b2');
|
2018-08-21 08:05:34 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
test('should sign with jwt whether legacy is disabled', async () => {
|
2020-10-03 05:47:04 -07:00
|
|
|
const token = await getTokenByConfiguration(
|
2020-09-17 06:48:16 +02:00
|
|
|
'security-legacy-disabled',
|
|
|
|
'test',
|
|
|
|
'test',
|
2020-10-03 05:47:04 -07:00
|
|
|
'b2df428b9929d3ace7c598bbf4e496b2',
|
2020-09-17 06:48:16 +02:00
|
|
|
'jwtEncrypt',
|
|
|
|
'aesEncrypt'
|
|
|
|
);
|
2018-08-21 08:05:34 +02:00
|
|
|
|
|
|
|
expect(_.isString(token)).toBeTruthy();
|
2020-10-03 05:47:04 -07:00
|
|
|
verifyJWT(token, 'test', 'test', 'b2df428b9929d3ace7c598bbf4e496b2');
|
2018-08-21 08:05:34 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getAuthenticatedMessage test', () => {
|
|
|
|
test('should sign token with jwt enabled', () => {
|
2020-08-13 23:27:00 +02:00
|
|
|
expect(getAuthenticatedMessage('test')).toBe("you are authenticated as 'test'");
|
2018-08-21 08:05:34 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|