2018-06-28 08:26:07 +02:00
|
|
|
import path from 'path';
|
2019-01-29 11:54:47 +08:00
|
|
|
import fs from 'fs';
|
2020-08-13 23:27:00 +02:00
|
|
|
import { Writable } from 'stream';
|
2020-03-03 23:59:19 +01:00
|
|
|
import { Config as AppConfig } from '@verdaccio/config';
|
|
|
|
import { Storage } from '@verdaccio/store';
|
|
|
|
|
2020-08-13 23:27:00 +02:00
|
|
|
import { Config } from '@verdaccio/types';
|
2020-11-15 11:14:09 +01:00
|
|
|
import { API_ERROR, HTTP_STATUS } from '@verdaccio/commons-api';
|
2020-08-13 23:27:00 +02:00
|
|
|
import { mockServer, configExample, DOMAIN_SERVERS, generateRamdonStorage } from '@verdaccio/mock';
|
2020-03-03 23:59:19 +01:00
|
|
|
|
2020-08-13 23:27:00 +02:00
|
|
|
import { setup, logger } from '@verdaccio/logger';
|
2018-02-19 19:29:14 +01:00
|
|
|
|
2019-02-24 23:20:25 +01:00
|
|
|
setup([]);
|
2018-02-19 19:29:14 +01:00
|
|
|
|
2019-07-16 08:40:01 +02:00
|
|
|
const mockServerPort = 55548;
|
2020-03-03 23:59:19 +01:00
|
|
|
|
2020-08-13 23:27:00 +02:00
|
|
|
const generateStorage = async function () {
|
2020-03-03 23:59:19 +01:00
|
|
|
const storagePath = generateRamdonStorage();
|
2020-08-13 23:27:00 +02:00
|
|
|
const storageConfig = configExample(
|
|
|
|
{
|
2020-11-08 15:20:02 +01:00
|
|
|
config_path: storagePath,
|
2020-08-13 23:27:00 +02:00
|
|
|
storage: storagePath,
|
|
|
|
uplinks: {
|
|
|
|
npmjs: {
|
|
|
|
url: `http://${DOMAIN_SERVERS}:${mockServerPort}`,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
'store.spec.yaml',
|
|
|
|
__dirname
|
|
|
|
);
|
2019-02-24 23:20:25 +01:00
|
|
|
|
2018-06-24 22:39:09 +02:00
|
|
|
const config: Config = new AppConfig(storageConfig);
|
2021-09-08 19:06:37 +02:00
|
|
|
const store: Storage = new Storage(config);
|
2019-07-16 08:40:01 +02:00
|
|
|
await store.init(config, []);
|
2018-02-19 19:29:14 +01:00
|
|
|
|
2018-06-24 22:39:09 +02:00
|
|
|
return store;
|
2021-03-14 08:42:46 +01:00
|
|
|
};
|
|
|
|
|
2020-08-13 23:27:00 +02:00
|
|
|
const generateSameUplinkStorage = async function () {
|
2020-03-03 23:59:19 +01:00
|
|
|
const storagePath = generateRamdonStorage();
|
2020-08-13 23:27:00 +02:00
|
|
|
const storageConfig = configExample(
|
|
|
|
{
|
2020-11-08 15:20:02 +01:00
|
|
|
config_path: storagePath,
|
2020-08-13 23:27:00 +02:00
|
|
|
storage: storagePath,
|
|
|
|
packages: {
|
|
|
|
jquery: {
|
|
|
|
access: ['$all'],
|
|
|
|
publish: ['$all'],
|
|
|
|
proxy: ['cached'],
|
|
|
|
},
|
|
|
|
'@jquery/*': {
|
|
|
|
access: ['$all'],
|
|
|
|
publish: ['$all'],
|
|
|
|
proxy: ['notcached'],
|
|
|
|
},
|
2020-01-11 08:39:58 +01:00
|
|
|
},
|
2020-08-13 23:27:00 +02:00
|
|
|
uplinks: {
|
|
|
|
cached: {
|
|
|
|
url: `http://${DOMAIN_SERVERS}:${mockServerPort}`,
|
|
|
|
cache: true,
|
|
|
|
},
|
|
|
|
notcached: {
|
|
|
|
url: `http://${DOMAIN_SERVERS}:${mockServerPort}`,
|
|
|
|
cache: false,
|
|
|
|
},
|
2020-03-03 23:59:19 +01:00
|
|
|
},
|
2020-08-13 23:27:00 +02:00
|
|
|
},
|
|
|
|
'store.spec.yaml',
|
|
|
|
__dirname
|
|
|
|
);
|
2020-01-11 08:39:58 +01:00
|
|
|
|
|
|
|
const config: Config = new AppConfig(storageConfig);
|
2021-09-08 19:06:37 +02:00
|
|
|
const store: Storage = new Storage(config);
|
2020-01-11 08:39:58 +01:00
|
|
|
await store.init(config, []);
|
|
|
|
|
|
|
|
return store;
|
2021-03-14 08:42:46 +01:00
|
|
|
};
|
2020-01-11 08:39:58 +01:00
|
|
|
|
2020-08-13 23:27:00 +02:00
|
|
|
const createNullStream = () =>
|
|
|
|
new Writable({
|
|
|
|
write: function (chunk, encoding, next) {
|
|
|
|
next();
|
|
|
|
},
|
|
|
|
});
|
2020-01-11 08:39:58 +01:00
|
|
|
|
2018-02-19 19:29:14 +01:00
|
|
|
describe('StorageTest', () => {
|
2018-06-24 22:39:09 +02:00
|
|
|
let mockRegistry;
|
2018-02-19 19:29:14 +01:00
|
|
|
|
2021-06-13 09:14:04 +02:00
|
|
|
beforeAll(async () => {
|
2020-03-03 23:59:19 +01:00
|
|
|
const binPath = require.resolve('verdaccio/bin/verdaccio');
|
|
|
|
const storePath = path.join(__dirname, '/mock/store');
|
|
|
|
mockRegistry = await mockServer(mockServerPort, { storePath, silence: true }).init(binPath);
|
2021-06-13 09:14:04 +02:00
|
|
|
return;
|
2018-06-24 22:39:09 +02:00
|
|
|
});
|
2018-02-19 19:29:14 +01: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-06-24 22:39:09 +02:00
|
|
|
});
|
2018-02-19 19:29:14 +01:00
|
|
|
|
2018-06-24 22:39:09 +02:00
|
|
|
test('should be defined', async () => {
|
2021-09-08 19:06:37 +02:00
|
|
|
const storage: Storage = await generateStorage();
|
2018-02-19 19:29:14 +01:00
|
|
|
|
|
|
|
expect(storage).toBeDefined();
|
2018-06-24 22:39:09 +02:00
|
|
|
});
|
2018-02-19 19:29:14 +01:00
|
|
|
|
2020-01-11 08:39:58 +01:00
|
|
|
describe('test getTarball', () => {
|
2020-10-24 22:17:21 +11:00
|
|
|
test.skip(
|
|
|
|
'should select right uplink given package.proxy for' + ' upstream tarballs',
|
|
|
|
async (done) => {
|
2021-09-08 19:06:37 +02:00
|
|
|
const storage: Storage = await generateSameUplinkStorage();
|
2020-10-24 22:17:21 +11:00
|
|
|
const notcachedSpy = jest.spyOn(storage.uplinks.notcached, 'fetchTarball');
|
|
|
|
const cachedSpy = jest.spyOn(storage.uplinks.cached, 'fetchTarball');
|
|
|
|
|
|
|
|
await new Promise((res, rej) => {
|
|
|
|
const reader = storage.getTarball('jquery', 'jquery-1.5.1.tgz');
|
|
|
|
reader.on('end', () => {
|
|
|
|
expect(notcachedSpy).toHaveBeenCalledTimes(0);
|
|
|
|
expect(cachedSpy).toHaveBeenCalledTimes(1);
|
|
|
|
expect(cachedSpy).toHaveBeenCalledWith(
|
|
|
|
'http://0.0.0.0:55548/jquery/-/jquery-1.5.1.tgz'
|
|
|
|
);
|
|
|
|
res();
|
|
|
|
});
|
|
|
|
reader.on('error', (err) => {
|
|
|
|
rej(err);
|
|
|
|
});
|
|
|
|
reader.pipe(createNullStream());
|
2020-01-11 08:39:58 +01:00
|
|
|
});
|
|
|
|
|
2020-10-24 22:17:21 +11:00
|
|
|
// Reset counters.
|
|
|
|
cachedSpy.mockClear();
|
|
|
|
notcachedSpy.mockClear();
|
|
|
|
|
|
|
|
await new Promise((res, rej) => {
|
|
|
|
const reader = storage.getTarball('@jquery/jquery', 'jquery-1.5.1.tgz');
|
|
|
|
reader.on('end', () => {
|
|
|
|
expect(cachedSpy).toHaveBeenCalledTimes(0);
|
|
|
|
expect(notcachedSpy).toHaveBeenCalledTimes(1);
|
|
|
|
expect(notcachedSpy).toHaveBeenCalledWith(
|
|
|
|
'http://0.0.0.0:55548/@jquery%2fjquery/-/jquery-1.5.1.tgz'
|
|
|
|
);
|
|
|
|
res();
|
|
|
|
});
|
|
|
|
reader.on('error', (err) => {
|
|
|
|
rej(err);
|
|
|
|
});
|
|
|
|
reader.pipe(createNullStream());
|
2020-01-11 08:39:58 +01:00
|
|
|
});
|
|
|
|
|
2020-10-24 22:17:21 +11:00
|
|
|
done();
|
|
|
|
}
|
|
|
|
);
|
2020-01-11 08:39:58 +01:00
|
|
|
});
|
|
|
|
|
2018-06-27 08:52:52 +02:00
|
|
|
describe('test _syncUplinksMetadata', () => {
|
2021-06-13 09:14:04 +02:00
|
|
|
test('should fetch from uplink jquery metadata from registry', async () => {
|
2021-09-08 19:06:37 +02:00
|
|
|
const storage: Storage = await generateStorage();
|
2018-02-19 19:29:14 +01:00
|
|
|
|
2021-06-13 09:14:04 +02:00
|
|
|
return new Promise((resolve) => {
|
|
|
|
// @ts-ignore
|
|
|
|
storage._syncUplinksMetadata('jquery', null, {}, (err, metadata) => {
|
|
|
|
expect(err).toBeNull();
|
|
|
|
expect(metadata).toBeDefined();
|
|
|
|
expect(metadata).toBeInstanceOf(Object);
|
|
|
|
resolve(metadata);
|
|
|
|
});
|
2018-06-27 08:52:52 +02:00
|
|
|
});
|
2018-06-24 22:39:09 +02:00
|
|
|
});
|
2018-02-19 19:29:14 +01:00
|
|
|
|
2021-06-13 09:14:04 +02:00
|
|
|
test('should fails on fetch from uplink non existing from registry', async () => {
|
2021-09-08 19:06:37 +02:00
|
|
|
const storage: Storage = await generateStorage();
|
2021-06-13 09:14:04 +02:00
|
|
|
return new Promise((resolve) => {
|
|
|
|
storage._syncUplinksMetadata('@verdaccio/404', null, {}, (err, _metadata, errors) => {
|
|
|
|
expect(err).not.toBeNull();
|
|
|
|
expect(errors).toBeInstanceOf(Array);
|
|
|
|
expect(errors[0][0].statusCode).toBe(HTTP_STATUS.NOT_FOUND);
|
|
|
|
expect(errors[0][0].message).toMatch(API_ERROR.NOT_PACKAGE_UPLINK);
|
|
|
|
resolve(errors);
|
|
|
|
});
|
2018-06-27 08:52:52 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-06-13 09:14:04 +02:00
|
|
|
test('should fails on fetch from uplink corrupted pkg from registry', async () => {
|
2021-09-08 19:06:37 +02:00
|
|
|
const storage: Storage = await generateStorage();
|
2021-06-13 09:14:04 +02:00
|
|
|
return new Promise((resolve) => {
|
|
|
|
// @ts-ignore
|
|
|
|
storage._syncUplinksMetadata('corrupted-package', null, {}, (err, metadata, errors) => {
|
|
|
|
expect(err).not.toBeNull();
|
|
|
|
expect(errors).toBeInstanceOf(Array);
|
|
|
|
expect(errors[0][0].statusCode).toBe(HTTP_STATUS.INTERNAL_ERROR);
|
|
|
|
expect(errors[0][0].message).toMatch(API_ERROR.BAD_STATUS_CODE);
|
|
|
|
resolve(errors);
|
|
|
|
});
|
2018-06-27 08:52:52 +02:00
|
|
|
});
|
2018-06-24 22:39:09 +02:00
|
|
|
});
|
2019-01-29 11:54:47 +08:00
|
|
|
|
2020-03-03 23:59:19 +01:00
|
|
|
test.skip('should not touch if the package exists and has no uplinks', async (done) => {
|
|
|
|
const storagePath = generateRamdonStorage();
|
2021-09-08 19:06:37 +02:00
|
|
|
const storage: Storage = (await generateStorage()) as Storage;
|
2019-05-19 23:07:17 +02:00
|
|
|
const metadataSource = path.join(__dirname, '../../partials/metadata');
|
2019-01-30 11:21:36 +08:00
|
|
|
const metadataPath = path.join(storagePath, 'npm_test/package.json');
|
2019-07-16 08:40:01 +02:00
|
|
|
|
2019-01-30 11:21:36 +08:00
|
|
|
fs.mkdirSync(path.join(storagePath, 'npm_test'));
|
2019-01-29 14:03:04 +08:00
|
|
|
fs.writeFileSync(metadataPath, fs.readFileSync(metadataSource));
|
2019-01-30 11:21:36 +08:00
|
|
|
const metadata = JSON.parse(fs.readFileSync(metadataPath).toString());
|
2019-08-16 21:20:18 +02:00
|
|
|
// @ts-ignore
|
2019-01-30 11:21:36 +08:00
|
|
|
storage.localStorage.updateVersions = jest.fn(storage.localStorage.updateVersions);
|
2019-02-24 23:20:25 +01:00
|
|
|
expect(metadata).toBeDefined();
|
2019-01-29 11:54:47 +08:00
|
|
|
storage._syncUplinksMetadata('npm_test', metadata, {}, (err) => {
|
2019-01-30 11:21:36 +08:00
|
|
|
expect(err).toBeNull();
|
2019-08-16 21:20:18 +02:00
|
|
|
// @ts-ignore
|
2019-01-30 11:21:36 +08:00
|
|
|
expect(storage.localStorage.updateVersions).not.toHaveBeenCalled();
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
2018-02-19 19:29:14 +01:00
|
|
|
});
|
|
|
|
});
|