mirror of
https://github.com/verdaccio/verdaccio.git
synced 2025-01-06 22:40:26 -05:00
274 lines
8.6 KiB
TypeScript
274 lines
8.6 KiB
TypeScript
|
/* eslint-disable jest/no-mocks-import */
|
||
|
import fs from 'fs';
|
||
|
import path from 'path';
|
||
|
|
||
|
import { assign } from 'lodash';
|
||
|
import { ILocalData, PluginOptions, Token } from '@verdaccio/types';
|
||
|
|
||
|
import LocalDatabase from '../src/local-database';
|
||
|
import { ILocalFSPackageManager } from '../src/local-fs';
|
||
|
import * as pkgUtils from '../src/pkg-utils';
|
||
|
|
||
|
// FIXME: remove this mocks imports
|
||
|
import Config from './__mocks__/Config';
|
||
|
import logger from './__mocks__/Logger';
|
||
|
|
||
|
const optionsPlugin: PluginOptions<{}> = {
|
||
|
logger,
|
||
|
config: new Config(),
|
||
|
};
|
||
|
|
||
|
let locaDatabase: ILocalData<{}>;
|
||
|
let loadPrivatePackages;
|
||
|
|
||
|
describe('Local Database', () => {
|
||
|
beforeEach(() => {
|
||
|
const writeMock = jest.spyOn(fs, 'writeFileSync').mockImplementation();
|
||
|
loadPrivatePackages = jest.spyOn(pkgUtils, 'loadPrivatePackages').mockReturnValue({ list: [], secret: '' });
|
||
|
locaDatabase = new LocalDatabase(optionsPlugin.config, optionsPlugin.logger);
|
||
|
(locaDatabase as LocalDatabase).clean();
|
||
|
writeMock.mockClear();
|
||
|
});
|
||
|
|
||
|
afterEach(() => {
|
||
|
jest.clearAllMocks();
|
||
|
});
|
||
|
|
||
|
test('should create an instance', () => {
|
||
|
expect(optionsPlugin.logger.error).not.toHaveBeenCalled();
|
||
|
expect(locaDatabase).toBeDefined();
|
||
|
});
|
||
|
|
||
|
test('should display log error if fails on load database', () => {
|
||
|
loadPrivatePackages.mockImplementation(() => {
|
||
|
throw Error();
|
||
|
});
|
||
|
|
||
|
new LocalDatabase(optionsPlugin.config, optionsPlugin.logger);
|
||
|
|
||
|
expect(optionsPlugin.logger.error).toHaveBeenCalled();
|
||
|
expect(optionsPlugin.logger.error).toHaveBeenCalledTimes(2);
|
||
|
});
|
||
|
|
||
|
describe('should create set secret', () => {
|
||
|
test('should create get secret', async () => {
|
||
|
const secretKey = await locaDatabase.getSecret();
|
||
|
|
||
|
expect(secretKey).toBeDefined();
|
||
|
expect(typeof secretKey === 'string').toBeTruthy();
|
||
|
});
|
||
|
|
||
|
test('should create set secret', async () => {
|
||
|
await locaDatabase.setSecret(optionsPlugin.config.checkSecretKey(''));
|
||
|
|
||
|
expect(optionsPlugin.config.secret).toBeDefined();
|
||
|
expect(typeof optionsPlugin.config.secret === 'string').toBeTruthy();
|
||
|
|
||
|
const fetchedSecretKey = await locaDatabase.getSecret();
|
||
|
expect(optionsPlugin.config.secret).toBe(fetchedSecretKey);
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('getPackageStorage', () => {
|
||
|
test('should get default storage', () => {
|
||
|
const pkgName = 'someRandomePackage';
|
||
|
const storage = locaDatabase.getPackageStorage(pkgName);
|
||
|
expect(storage).toBeDefined();
|
||
|
|
||
|
if (storage) {
|
||
|
const storagePath = path.normalize((storage as ILocalFSPackageManager).path).toLowerCase();
|
||
|
expect(storagePath).toBe(path.normalize(path.join(__dirname, '__fixtures__', optionsPlugin.config.storage || '', pkgName)).toLowerCase());
|
||
|
}
|
||
|
});
|
||
|
|
||
|
test('should use custom storage', () => {
|
||
|
const pkgName = 'local-private-custom-storage';
|
||
|
const storage = locaDatabase.getPackageStorage(pkgName);
|
||
|
|
||
|
expect(storage).toBeDefined();
|
||
|
|
||
|
if (storage) {
|
||
|
const storagePath = path.normalize((storage as ILocalFSPackageManager).path).toLowerCase();
|
||
|
expect(storagePath).toBe(
|
||
|
path.normalize(path.join(__dirname, '__fixtures__', optionsPlugin.config.storage || '', 'private_folder', pkgName)).toLowerCase()
|
||
|
);
|
||
|
}
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('Database CRUD', () => {
|
||
|
test('should add an item to database', (done) => {
|
||
|
const pgkName = 'jquery';
|
||
|
locaDatabase.get((err, data) => {
|
||
|
expect(err).toBeNull();
|
||
|
expect(data).toHaveLength(0);
|
||
|
|
||
|
locaDatabase.add(pgkName, (err) => {
|
||
|
expect(err).toBeNull();
|
||
|
locaDatabase.get((err, data) => {
|
||
|
expect(err).toBeNull();
|
||
|
expect(data).toHaveLength(1);
|
||
|
done();
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
|
||
|
test('should remove an item to database', (done) => {
|
||
|
const pgkName = 'jquery';
|
||
|
locaDatabase.get((err, data) => {
|
||
|
expect(err).toBeNull();
|
||
|
expect(data).toHaveLength(0);
|
||
|
locaDatabase.add(pgkName, (err) => {
|
||
|
expect(err).toBeNull();
|
||
|
locaDatabase.remove(pgkName, (err) => {
|
||
|
expect(err).toBeNull();
|
||
|
locaDatabase.get((err, data) => {
|
||
|
expect(err).toBeNull();
|
||
|
expect(data).toHaveLength(0);
|
||
|
done();
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('search', () => {
|
||
|
const onPackageMock = jest.fn((item, cb) => cb());
|
||
|
const validatorMock = jest.fn(() => true);
|
||
|
const callSearch = (db, numberTimesCalled, cb): void => {
|
||
|
db.search(
|
||
|
onPackageMock,
|
||
|
function onEnd() {
|
||
|
expect(onPackageMock).toHaveBeenCalledTimes(numberTimesCalled);
|
||
|
expect(validatorMock).toHaveBeenCalledTimes(numberTimesCalled);
|
||
|
cb();
|
||
|
},
|
||
|
validatorMock
|
||
|
);
|
||
|
};
|
||
|
|
||
|
test('should find scoped packages', (done) => {
|
||
|
const scopedPackages = ['@pkg1/test'];
|
||
|
const stats = { mtime: new Date() };
|
||
|
jest.spyOn(fs, 'stat').mockImplementation((_, cb) => cb(null, stats as fs.Stats));
|
||
|
jest.spyOn(fs, 'readdir').mockImplementation((storePath, cb) => cb(null, storePath.match('test-storage') ? scopedPackages : []));
|
||
|
|
||
|
callSearch(locaDatabase, 1, done);
|
||
|
});
|
||
|
|
||
|
test('should find non scoped packages', (done) => {
|
||
|
const nonScopedPackages = ['pkg1', 'pkg2'];
|
||
|
const stats = { mtime: new Date() };
|
||
|
jest.spyOn(fs, 'stat').mockImplementation((_, cb) => cb(null, stats as fs.Stats));
|
||
|
jest.spyOn(fs, 'readdir').mockImplementation((storePath, cb) => cb(null, storePath.match('test-storage') ? nonScopedPackages : []));
|
||
|
|
||
|
const db = new LocalDatabase(
|
||
|
assign({}, optionsPlugin.config, {
|
||
|
// clean up this, it creates noise
|
||
|
packages: {},
|
||
|
}),
|
||
|
optionsPlugin.logger
|
||
|
);
|
||
|
|
||
|
callSearch(db, 2, done);
|
||
|
});
|
||
|
|
||
|
test('should fails on read the storage', (done) => {
|
||
|
const spyInstance = jest.spyOn(fs, 'readdir').mockImplementation((_, cb) => cb(Error('fails'), null));
|
||
|
|
||
|
const db = new LocalDatabase(
|
||
|
assign({}, optionsPlugin.config, {
|
||
|
// clean up this, it creates noise
|
||
|
packages: {},
|
||
|
}),
|
||
|
optionsPlugin.logger
|
||
|
);
|
||
|
|
||
|
callSearch(db, 0, done);
|
||
|
spyInstance.mockRestore();
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('token', () => {
|
||
|
let token: Token;
|
||
|
|
||
|
beforeEach(() => {
|
||
|
(locaDatabase as LocalDatabase).tokenDb = {
|
||
|
put: jest.fn().mockImplementation((key, value, cb) => cb()),
|
||
|
del: jest.fn().mockImplementation((key, cb) => cb()),
|
||
|
createReadStream: jest.fn(),
|
||
|
};
|
||
|
|
||
|
token = {
|
||
|
user: 'someUser',
|
||
|
viewToken: 'viewToken',
|
||
|
key: 'someHash',
|
||
|
readonly: true,
|
||
|
createdTimestamp: new Date().getTime(),
|
||
|
};
|
||
|
});
|
||
|
|
||
|
test('should save token', async (done) => {
|
||
|
const db = (locaDatabase as LocalDatabase).tokenDb;
|
||
|
|
||
|
await locaDatabase.saveToken(token);
|
||
|
|
||
|
expect(db.put).toHaveBeenCalledWith('someUser:someHash', token, expect.anything());
|
||
|
done();
|
||
|
});
|
||
|
|
||
|
test('should delete token', async (done) => {
|
||
|
const db = (locaDatabase as LocalDatabase).tokenDb;
|
||
|
|
||
|
await locaDatabase.deleteToken('someUser', 'someHash');
|
||
|
|
||
|
expect(db.del).toHaveBeenCalledWith('someUser:someHash', expect.anything());
|
||
|
done();
|
||
|
});
|
||
|
|
||
|
test('should get tokens', async () => {
|
||
|
const db = (locaDatabase as LocalDatabase).tokenDb;
|
||
|
const events = { on: {}, once: {} };
|
||
|
const stream = {
|
||
|
on: (event, cb): void => {
|
||
|
events.on[event] = cb;
|
||
|
},
|
||
|
once: (event, cb): void => {
|
||
|
events.once[event] = cb;
|
||
|
},
|
||
|
};
|
||
|
db.createReadStream.mockImplementation(() => stream);
|
||
|
setTimeout(() => events.on['data']({ value: token }));
|
||
|
setTimeout(() => events.once['end']());
|
||
|
|
||
|
const tokens = await locaDatabase.readTokens({ user: 'someUser' });
|
||
|
|
||
|
expect(db.createReadStream).toHaveBeenCalledWith({
|
||
|
gte: 'someUser:',
|
||
|
lte: 't',
|
||
|
});
|
||
|
expect(tokens).toHaveLength(1);
|
||
|
expect(tokens[0]).toBe(token);
|
||
|
});
|
||
|
|
||
|
test('should fail getting tokens if something goes wrong', async () => {
|
||
|
const db = (locaDatabase as LocalDatabase).tokenDb;
|
||
|
const events = { on: {}, once: {} };
|
||
|
const stream = {
|
||
|
on: (event, cb): void => {
|
||
|
events.on[event] = cb;
|
||
|
},
|
||
|
once: (event, cb): void => {
|
||
|
events.once[event] = cb;
|
||
|
},
|
||
|
};
|
||
|
db.createReadStream.mockImplementation(() => stream);
|
||
|
setTimeout(() => events.once['error'](new Error('Unexpected error!')));
|
||
|
|
||
|
await expect(locaDatabase.readTokens({ user: 'someUser' })).rejects.toThrow('Unexpected error!');
|
||
|
});
|
||
|
});
|
||
|
});
|