2021-05-08 09:47:30 +02:00
|
|
|
import nock from 'nock';
|
2021-10-29 17:33:05 +02:00
|
|
|
import path from 'path';
|
|
|
|
|
2021-05-08 09:47:30 +02:00
|
|
|
import { Config, parseConfigFile } from '@verdaccio/config';
|
2023-08-06 17:42:20 +02:00
|
|
|
import { logger, setup } from '@verdaccio/logger';
|
2021-10-29 17:33:05 +02:00
|
|
|
|
2022-07-29 20:51:45 +02:00
|
|
|
import { ProxyStorage } from '../src';
|
|
|
|
|
|
|
|
setup();
|
2021-05-08 09:47:30 +02:00
|
|
|
|
|
|
|
const getConf = (name) => path.join(__dirname, '/conf', name);
|
|
|
|
|
2022-07-29 20:51:45 +02:00
|
|
|
// // mock to get the headers fixed value
|
|
|
|
jest.mock('crypto', () => {
|
2021-05-08 09:47:30 +02:00
|
|
|
return {
|
2022-07-29 20:51:45 +02:00
|
|
|
randomBytes: (): { toString: () => string } => {
|
|
|
|
return {
|
|
|
|
toString: (): string => 'foo-random-bytes',
|
|
|
|
};
|
|
|
|
},
|
|
|
|
pseudoRandomBytes: (): { toString: () => string } => {
|
|
|
|
return {
|
|
|
|
toString: (): string => 'foo-phseudo-bytes',
|
|
|
|
};
|
2021-05-08 09:47:30 +02:00
|
|
|
},
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
2022-07-29 20:51:45 +02:00
|
|
|
describe('tarball proxy', () => {
|
2021-05-08 09:47:30 +02:00
|
|
|
beforeEach(() => {
|
|
|
|
nock.cleanAll();
|
2021-09-08 19:06:37 +02:00
|
|
|
nock.abortPendingRequests();
|
|
|
|
jest.clearAllMocks();
|
2021-05-08 09:47:30 +02:00
|
|
|
});
|
|
|
|
const defaultRequestOptions = {
|
2022-07-29 20:51:45 +02:00
|
|
|
url: 'https://registry.verdaccio.org',
|
2021-05-08 09:47:30 +02:00
|
|
|
};
|
2022-07-29 20:51:45 +02:00
|
|
|
|
2021-05-08 09:47:30 +02:00
|
|
|
const proxyPath = getConf('proxy1.yaml');
|
|
|
|
const conf = new Config(parseConfigFile(proxyPath));
|
2021-05-08 19:21:21 +02:00
|
|
|
|
2023-02-12 20:26:18 +01:00
|
|
|
describe('fetchTarball', () => {
|
2022-07-29 20:51:45 +02:00
|
|
|
test('get file tarball fetch', (done) => {
|
|
|
|
nock('https://registry.verdaccio.org')
|
2021-05-08 19:21:21 +02:00
|
|
|
.get('/jquery/-/jquery-0.0.1.tgz')
|
|
|
|
.replyWithFile(201, path.join(__dirname, 'partials/jquery-0.0.1.tgz'));
|
2023-08-06 17:42:20 +02:00
|
|
|
const prox1 = new ProxyStorage(defaultRequestOptions, conf, logger);
|
2023-02-12 20:26:18 +01:00
|
|
|
const stream = prox1.fetchTarball(
|
2022-07-29 20:51:45 +02:00
|
|
|
'https://registry.verdaccio.org/jquery/-/jquery-0.0.1.tgz',
|
|
|
|
{}
|
|
|
|
);
|
|
|
|
stream.on('response', () => {
|
2021-05-08 09:47:30 +02:00
|
|
|
done();
|
|
|
|
});
|
2022-07-29 20:51:45 +02:00
|
|
|
stream.on('error', (err) => {
|
|
|
|
done(err);
|
|
|
|
});
|
2021-05-08 09:47:30 +02:00
|
|
|
});
|
|
|
|
|
2022-07-29 20:51:45 +02:00
|
|
|
test.skip('get file tarball handle retries', (done) => {
|
|
|
|
nock('https://registry.verdaccio.org')
|
2021-05-08 19:21:21 +02:00
|
|
|
.get('/jquery/-/jquery-0.0.1.tgz')
|
2022-07-29 20:51:45 +02:00
|
|
|
.twice()
|
|
|
|
.reply(500, 'some-text')
|
|
|
|
.get('/jquery/-/jquery-0.0.1.tgz')
|
|
|
|
.once()
|
|
|
|
.replyWithFile(201, path.join(__dirname, 'partials/jquery-0.0.1.tgz'));
|
2021-05-08 09:47:30 +02:00
|
|
|
const prox1 = new ProxyStorage(defaultRequestOptions, conf);
|
2023-02-12 20:26:18 +01:00
|
|
|
const stream = prox1.fetchTarball(
|
2022-07-29 20:51:45 +02:00
|
|
|
'https://registry.verdaccio.org/jquery/-/jquery-0.0.1.tgz',
|
|
|
|
{ retry: { limit: 2 } }
|
|
|
|
);
|
|
|
|
stream.on('error', () => {
|
|
|
|
// FIXME: stream should have handle 2 retry
|
2021-05-08 09:47:30 +02:00
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2022-07-29 20:51:45 +02:00
|
|
|
// test('get file tarball correct content-length', (done) => {
|
|
|
|
// nock(domain)
|
|
|
|
// .get('/jquery/-/jquery-0.0.1.tgz')
|
|
|
|
// // types does not match here with documentation
|
|
|
|
// // @ts-expect-error
|
|
|
|
// .replyWithFile(201, path.join(__dirname, 'partials/jquery-0.0.1.tgz'), {
|
|
|
|
// [HEADER_TYPE.CONTENT_LENGTH]: 277,
|
|
|
|
// });
|
|
|
|
// const prox1 = new ProxyStorage(defaultRequestOptions, conf);
|
|
|
|
// const stream = prox1.fetchTarball('https://registry.npmjs.org/jquery/-/jquery-0.0.1.tgz');
|
|
|
|
// stream.on(HEADER_TYPE.CONTENT_LENGTH, (data) => {
|
|
|
|
// expect(data).toEqual('277');
|
|
|
|
// done();
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
|
|
|
|
// describe('error handling', () => {
|
|
|
|
// test('should be offline uplink', (done) => {
|
|
|
|
// const tarball = 'https://registry.npmjs.org/jquery/-/jquery-0.0.1.tgz';
|
|
|
|
// nock(domain).get('/jquery/-/jquery-0.0.1.tgz').times(100).replyWithError('some error');
|
|
|
|
// const proxy = new ProxyStorage(defaultRequestOptions, conf);
|
|
|
|
// const stream = proxy.fetchTarball(tarball);
|
|
|
|
// // to test a uplink is offline we have to be try 3 times
|
|
|
|
// // the default failed request are set to 2
|
|
|
|
// process.nextTick(function () {
|
|
|
|
// stream.on('error', function (err) {
|
|
|
|
// expect(err).not.toBeNull();
|
|
|
|
// // expect(err.statusCode).toBe(404);
|
|
|
|
// expect(proxy.failed_requests).toBe(1);
|
|
|
|
|
|
|
|
// const streamSecondTry = proxy.fetchTarball(tarball);
|
|
|
|
// streamSecondTry.on('error', function (err) {
|
|
|
|
// expect(err).not.toBeNull();
|
|
|
|
// /*
|
|
|
|
// code: 'ENOTFOUND',
|
|
|
|
// errno: 'ENOTFOUND',
|
|
|
|
// */
|
|
|
|
// // expect(err.statusCode).toBe(404);
|
|
|
|
// expect(proxy.failed_requests).toBe(2);
|
|
|
|
// const streamThirdTry = proxy.fetchTarball(tarball);
|
|
|
|
// streamThirdTry.on('error', function (err: VerdaccioError) {
|
|
|
|
// expect(err).not.toBeNull();
|
|
|
|
// expect(err.statusCode).toBe(HTTP_STATUS.INTERNAL_ERROR);
|
|
|
|
// expect(proxy.failed_requests).toBe(2);
|
|
|
|
// expect(err.message).toMatch(API_ERROR.UPLINK_OFFLINE);
|
|
|
|
// done();
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
|
|
|
|
// test('not found tarball', (done) => {
|
|
|
|
// nock(domain).get('/jquery/-/jquery-0.0.1.tgz').reply(404);
|
|
|
|
// const prox1 = new ProxyStorage(defaultRequestOptions, conf);
|
|
|
|
// const stream = prox1.fetchTarball('https://registry.npmjs.org/jquery/-/jquery-0.0.1.tgz');
|
|
|
|
// stream.on('error', (response) => {
|
|
|
|
// expect(response).toEqual(errorUtils.getNotFound(API_ERROR.NOT_FILE_UPLINK));
|
|
|
|
// done();
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
|
|
|
|
// test('fail tarball request', (done) => {
|
|
|
|
// nock(domain).get('/jquery/-/jquery-0.0.1.tgz').replyWithError('boom file');
|
|
|
|
// const prox1 = new ProxyStorage(defaultRequestOptions, conf);
|
|
|
|
// const stream = prox1.fetchTarball('https://registry.npmjs.org/jquery/-/jquery-0.0.1.tgz');
|
|
|
|
// stream.on('error', (response) => {
|
|
|
|
// expect(response).toEqual(Error('boom file'));
|
|
|
|
// done();
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
|
|
|
|
// test('bad uplink request', (done) => {
|
|
|
|
// nock(domain).get('/jquery/-/jquery-0.0.1.tgz').reply(409);
|
|
|
|
// const prox1 = new ProxyStorage(defaultRequestOptions, conf);
|
|
|
|
// const stream = prox1.fetchTarball('https://registry.npmjs.org/jquery/-/jquery-0.0.1.tgz');
|
|
|
|
// stream.on('error', (response) => {
|
|
|
|
// expect(response).toEqual(errorUtils.getInternalError(`bad uplink status code: 409`));
|
|
|
|
// done();
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
|
|
|
|
// test('content length header mismatch', (done) => {
|
|
|
|
// nock(domain)
|
|
|
|
// .get('/jquery/-/jquery-0.0.1.tgz')
|
|
|
|
// // types does not match here with documentation
|
|
|
|
// // @ts-expect-error
|
|
|
|
// .replyWithFile(201, path.join(__dirname, 'partials/jquery-0.0.1.tgz'), {
|
|
|
|
// [HEADER_TYPE.CONTENT_LENGTH]: 0,
|
|
|
|
// });
|
|
|
|
// const prox1 = new ProxyStorage(defaultRequestOptions, conf);
|
|
|
|
// const stream = prox1.fetchTarball('https://registry.npmjs.org/jquery/-/jquery-0.0.1.tgz');
|
|
|
|
// stream.on('error', (response) => {
|
|
|
|
// expect(response).toEqual(errorUtils.getInternalError(API_ERROR.CONTENT_MISMATCH));
|
|
|
|
// done();
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
// });
|
|
|
|
// });
|