2018-08-21 08:05:34 +02:00
|
|
|
import path from 'path';
|
2020-08-11 07:21:51 +02:00
|
|
|
import request from 'supertest';
|
2020-03-03 23:59:19 +01:00
|
|
|
|
2020-09-17 06:48:16 +02:00
|
|
|
import {
|
2021-10-29 17:33:05 +02:00
|
|
|
API_ERROR,
|
2020-09-17 06:48:16 +02:00
|
|
|
HEADERS,
|
|
|
|
HEADER_TYPE,
|
2021-10-29 17:33:05 +02:00
|
|
|
HTTP_STATUS,
|
2020-09-17 06:48:16 +02:00
|
|
|
TOKEN_BASIC,
|
2021-10-29 17:33:05 +02:00
|
|
|
TOKEN_BEARER,
|
2021-09-26 00:08:00 +02:00
|
|
|
} from '@verdaccio/core';
|
2021-10-29 17:33:05 +02:00
|
|
|
import { logger, setup } from '@verdaccio/logger';
|
|
|
|
import { generateRamdonStorage, mockServer } from '@verdaccio/mock';
|
2020-09-17 06:48:16 +02:00
|
|
|
import {
|
|
|
|
DOMAIN_SERVERS,
|
|
|
|
addUser,
|
2021-10-29 17:33:05 +02:00
|
|
|
configExample,
|
2020-09-17 06:48:16 +02:00
|
|
|
getPackage,
|
|
|
|
loginUserToken,
|
|
|
|
} from '@verdaccio/mock';
|
2021-10-29 17:33:05 +02:00
|
|
|
import { buildToken, buildUserBuffer } from '@verdaccio/utils';
|
2018-08-21 08:05:34 +02:00
|
|
|
|
2020-08-11 07:21:51 +02:00
|
|
|
import endPointAPI from '../../src';
|
|
|
|
|
2018-08-21 08:05:34 +02:00
|
|
|
setup([]);
|
2020-03-03 23:59:19 +01:00
|
|
|
|
2018-08-21 08:05:34 +02:00
|
|
|
const credentials = { name: 'JotaJWT', password: 'secretPass' };
|
|
|
|
|
2019-07-16 08:40:01 +02:00
|
|
|
const FORBIDDEN_VUE = 'authorization required to access package vue';
|
2018-08-21 08:05:34 +02:00
|
|
|
|
|
|
|
describe('endpoint user auth JWT unit test', () => {
|
2019-05-20 07:41:12 +02:00
|
|
|
jest.setTimeout(20000);
|
2018-08-21 08:05:34 +02:00
|
|
|
let app;
|
|
|
|
let mockRegistry;
|
2019-05-19 21:37:43 +02:00
|
|
|
const FAKE_TOKEN: string = buildToken(TOKEN_BEARER, 'fake');
|
2018-08-21 08:05:34 +02:00
|
|
|
|
2021-06-13 09:14:04 +02:00
|
|
|
beforeAll(async function () {
|
2018-08-21 08:05:34 +02:00
|
|
|
const mockServerPort = 55546;
|
2020-03-03 23:59:19 +01:00
|
|
|
const store = generateRamdonStorage();
|
2020-08-13 23:27:00 +02:00
|
|
|
const configForTest = configExample(
|
|
|
|
{
|
|
|
|
storage: store,
|
|
|
|
uplinks: {
|
|
|
|
remote: {
|
|
|
|
url: `http://${DOMAIN_SERVERS}:${mockServerPort}`,
|
|
|
|
},
|
|
|
|
},
|
2020-11-08 15:20:02 +01:00
|
|
|
config_path: store,
|
2020-03-03 23:59:19 +01:00
|
|
|
},
|
2020-08-13 23:27:00 +02:00
|
|
|
'jwt.yaml',
|
|
|
|
__dirname
|
|
|
|
);
|
2020-03-03 23:59:19 +01:00
|
|
|
|
|
|
|
app = await endPointAPI(configForTest);
|
|
|
|
const binPath = require.resolve('verdaccio/bin/verdaccio');
|
|
|
|
const storePath = path.join(__dirname, '/mock/store');
|
|
|
|
mockRegistry = await mockServer(mockServerPort, { storePath, silence: true }).init(binPath);
|
2018-08-21 08:05:34 +02:00
|
|
|
});
|
|
|
|
|
2021-06-13 09:14:04 +02:00
|
|
|
afterAll(function () {
|
2020-03-03 23:59:19 +01:00
|
|
|
const [registry, pid] = mockRegistry;
|
|
|
|
registry.stop();
|
|
|
|
logger.info(`registry ${pid} has been stopped`);
|
2018-08-21 08:05:34 +02:00
|
|
|
});
|
|
|
|
|
2021-06-13 09:14:04 +02:00
|
|
|
test('should test add a new user with JWT enabled', async () => {
|
2018-08-21 08:05:34 +02:00
|
|
|
const [err, res] = await addUser(request(app), credentials.name, credentials);
|
|
|
|
expect(err).toBeNull();
|
|
|
|
expect(res.body.ok).toBeDefined();
|
|
|
|
expect(res.body.token).toBeDefined();
|
2019-06-13 06:58:43 +02:00
|
|
|
|
|
|
|
const { token } = res.body;
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(typeof token).toBe('string');
|
|
|
|
expect(res.body.ok).toMatch(`user '${credentials.name}' created`);
|
2019-06-13 06:58:43 +02:00
|
|
|
|
2018-08-21 08:05:34 +02:00
|
|
|
// testing JWT auth headers with token
|
|
|
|
// we need it here, because token is required
|
2019-08-10 13:38:06 +02:00
|
|
|
const [err1, resp1] = await getPackage(request(app), token, 'vue');
|
2020-03-03 23:59:19 +01:00
|
|
|
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(err1).toBeNull();
|
|
|
|
expect(resp1.body).toBeDefined();
|
|
|
|
expect(resp1.body.name).toMatch('vue');
|
|
|
|
|
2020-09-17 06:48:16 +02:00
|
|
|
const [err2, resp2] = await getPackage(
|
|
|
|
request(app),
|
|
|
|
FAKE_TOKEN,
|
|
|
|
'vue',
|
|
|
|
HTTP_STATUS.UNAUTHORIZED
|
|
|
|
);
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(err2).toBeNull();
|
2018-11-15 21:13:41 +01:00
|
|
|
expect(resp2.statusCode).toBe(HTTP_STATUS.UNAUTHORIZED);
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(resp2.body.error).toMatch(FORBIDDEN_VUE);
|
|
|
|
});
|
|
|
|
|
2021-06-13 09:14:04 +02:00
|
|
|
test('should emulate npm login when user already exist', async () => {
|
2018-08-21 08:05:34 +02:00
|
|
|
const credentials = { name: 'jwtUser2', password: 'secretPass' };
|
|
|
|
// creates an user
|
|
|
|
await addUser(request(app), credentials.name, credentials);
|
|
|
|
// it should fails conflict 409
|
|
|
|
await addUser(request(app), credentials.name, credentials, HTTP_STATUS.CONFLICT);
|
2019-06-13 06:58:43 +02:00
|
|
|
|
2018-08-21 08:05:34 +02:00
|
|
|
// npm will try to sign in sending credentials via basic auth header
|
|
|
|
const token = buildUserBuffer(credentials.name, credentials.password).toString('base64');
|
2019-07-16 08:40:01 +02:00
|
|
|
// put should exist in request
|
2021-06-13 09:14:04 +02:00
|
|
|
return new Promise((resolve) => {
|
|
|
|
// @ts-ignore
|
|
|
|
request(app)
|
|
|
|
.put(`/-/user/org.couchdb.user:${credentials.name}/-rev/undefined`)
|
|
|
|
.send(credentials)
|
|
|
|
.set(HEADERS.AUTHORIZATION, buildToken(TOKEN_BASIC, token))
|
|
|
|
.expect(HEADER_TYPE.CONTENT_TYPE, HEADERS.JSON_CHARSET)
|
|
|
|
.expect(HTTP_STATUS.CREATED)
|
|
|
|
.end(function (err, res) {
|
|
|
|
expect(err).toBeNull();
|
|
|
|
expect(res.body.ok).toBeDefined();
|
|
|
|
expect(res.body.token).toBeDefined();
|
|
|
|
|
|
|
|
resolve(res);
|
|
|
|
});
|
|
|
|
});
|
2018-08-21 08:05:34 +02:00
|
|
|
});
|
|
|
|
|
2021-06-13 09:14:04 +02:00
|
|
|
test('should fails on try to access with corrupted token', async () => {
|
2020-09-17 06:48:16 +02:00
|
|
|
const [err2, resp2] = await getPackage(
|
|
|
|
request(app),
|
|
|
|
FAKE_TOKEN,
|
|
|
|
'vue',
|
|
|
|
HTTP_STATUS.UNAUTHORIZED
|
|
|
|
);
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(err2).toBeNull();
|
2018-11-15 21:13:41 +01:00
|
|
|
expect(resp2.statusCode).toBe(HTTP_STATUS.UNAUTHORIZED);
|
2018-08-21 08:05:34 +02:00
|
|
|
expect(resp2.body.error).toMatch(FORBIDDEN_VUE);
|
|
|
|
});
|
|
|
|
|
2020-10-24 22:17:21 +11:00
|
|
|
test(
|
|
|
|
'should fails on login if user credentials are invalid even if jwt' +
|
|
|
|
' valid token is provided',
|
2021-06-13 09:14:04 +02:00
|
|
|
async () => {
|
2020-10-24 22:17:21 +11:00
|
|
|
const credentials = { name: 'newFailsUser', password: 'secretPass' };
|
|
|
|
const [err, res] = await addUser(request(app), credentials.name, credentials);
|
|
|
|
expect(err).toBeNull();
|
|
|
|
expect(res.body.ok).toBeDefined();
|
|
|
|
expect(res.body.token).toBeDefined();
|
|
|
|
|
|
|
|
const { token } = res.body;
|
|
|
|
expect(typeof token).toBe('string');
|
|
|
|
expect(res.body.ok).toMatch(`user '${credentials.name}' created`);
|
|
|
|
|
|
|
|
// we login when token is valid
|
|
|
|
const newCredentials = { name: 'newFailsUser', password: 'BAD_PASSWORD' };
|
|
|
|
const [err2, resp2] = await loginUserToken(
|
|
|
|
request(app),
|
|
|
|
newCredentials.name,
|
|
|
|
newCredentials,
|
|
|
|
token,
|
|
|
|
HTTP_STATUS.UNAUTHORIZED
|
|
|
|
);
|
|
|
|
expect(err2).toBeNull();
|
|
|
|
expect(resp2.statusCode).toBe(HTTP_STATUS.UNAUTHORIZED);
|
|
|
|
expect(resp2.body.error).toMatch(API_ERROR.BAD_USERNAME_PASSWORD);
|
|
|
|
}
|
|
|
|
);
|
2018-08-21 08:05:34 +02:00
|
|
|
});
|