mirror of
https://github.com/verdaccio/verdaccio.git
synced 2025-01-20 22:52:46 -05:00
fix: official package - cannot be synced (#3919)
* fix: official package - cannot be synced * coverage
This commit is contained in:
parent
979f2ff210
commit
f859d2b1ae
7 changed files with 113 additions and 64 deletions
7
.changeset/rich-bananas-chew.md
Normal file
7
.changeset/rich-bananas-chew.md
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
---
|
||||||
|
'@verdaccio/api': patch
|
||||||
|
'@verdaccio/core': patch
|
||||||
|
'@verdaccio/middleware': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
fix: official package "-" cannot be synced
|
|
@ -50,7 +50,6 @@ export default function (config: Config, auth: Auth, storage: Storage): Router {
|
||||||
app.use(encodeScopePackage);
|
app.use(encodeScopePackage);
|
||||||
// for "npm whoami"
|
// for "npm whoami"
|
||||||
whoami(app);
|
whoami(app);
|
||||||
pkg(app, auth, storage);
|
|
||||||
profile(app, auth, config);
|
profile(app, auth, config);
|
||||||
// @deprecated endpoint, 404 by default
|
// @deprecated endpoint, 404 by default
|
||||||
search(app);
|
search(app);
|
||||||
|
@ -62,5 +61,6 @@ export default function (config: Config, auth: Auth, storage: Storage): Router {
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
v1Search(app, auth, storage);
|
v1Search(app, auth, storage);
|
||||||
token(app, auth, storage, config);
|
token(app, auth, storage, config);
|
||||||
|
pkg(app, auth, storage);
|
||||||
return app;
|
return app;
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,10 @@ import {
|
||||||
|
|
||||||
describe('validatePackage', () => {
|
describe('validatePackage', () => {
|
||||||
test('should validate package names', () => {
|
test('should validate package names', () => {
|
||||||
|
expect(validatePackage('-')).toBeTruthy();
|
||||||
|
expect(validatePackage('--')).toBeTruthy();
|
||||||
|
expect(validatePackage('a')).toBeTruthy();
|
||||||
|
expect(validatePackage('a-')).toBeTruthy();
|
||||||
expect(validatePackage('package-name')).toBeTruthy();
|
expect(validatePackage('package-name')).toBeTruthy();
|
||||||
expect(validatePackage('@scope/package-name')).toBeTruthy();
|
expect(validatePackage('@scope/package-name')).toBeTruthy();
|
||||||
});
|
});
|
||||||
|
@ -21,6 +25,7 @@ describe('validatePackage', () => {
|
||||||
expect(validatePackage('node_modules')).toBeFalsy();
|
expect(validatePackage('node_modules')).toBeFalsy();
|
||||||
expect(validatePackage('__proto__')).toBeFalsy();
|
expect(validatePackage('__proto__')).toBeFalsy();
|
||||||
expect(validatePackage('favicon.ico')).toBeFalsy();
|
expect(validatePackage('favicon.ico')).toBeFalsy();
|
||||||
|
expect(validatePackage('%')).toBeFalsy();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -75,6 +80,7 @@ describe('validateName', () => {
|
||||||
test('good ones', () => {
|
test('good ones', () => {
|
||||||
expect(validateName('verdaccio')).toBeTruthy();
|
expect(validateName('verdaccio')).toBeTruthy();
|
||||||
expect(validateName('some.weird.package-zzz')).toBeTruthy();
|
expect(validateName('some.weird.package-zzz')).toBeTruthy();
|
||||||
|
expect(validateName('--0.0.1.tgz')).toBeTruthy();
|
||||||
expect(validateName('old-package@0.1.2.tgz')).toBeTruthy();
|
expect(validateName('old-package@0.1.2.tgz')).toBeTruthy();
|
||||||
// fix https://github.com/verdaccio/verdaccio/issues/1400
|
// fix https://github.com/verdaccio/verdaccio/issues/1400
|
||||||
expect(validateName('-build-infra')).toBeTruthy();
|
expect(validateName('-build-infra')).toBeTruthy();
|
||||||
|
|
|
@ -1,27 +1,17 @@
|
||||||
import { errorUtils } from '@verdaccio/core';
|
import { errorUtils, validationUtils } from '@verdaccio/core';
|
||||||
import {
|
|
||||||
validateName as utilValidateName,
|
|
||||||
validatePackage as utilValidatePackage,
|
|
||||||
} from '@verdaccio/utils';
|
|
||||||
|
|
||||||
export function validateName(_req, _res, next, value: string, name: string) {
|
export function validateName(_req, _res, next, value: string, name: string) {
|
||||||
if (value === '-') {
|
if (validationUtils.validateName(value)) {
|
||||||
// special case in couchdb usually
|
|
||||||
next('route');
|
|
||||||
} else if (utilValidateName(value)) {
|
|
||||||
next();
|
next();
|
||||||
} else {
|
} else {
|
||||||
next(errorUtils.getForbidden('invalid ' + name));
|
next(errorUtils.getBadRequest('invalid ' + name));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function validatePackage(_req, _res, next, value: string, name: string) {
|
export function validatePackage(_req, _res, next, value: string, name: string) {
|
||||||
if (value === '-') {
|
if (validationUtils.validatePackage(value)) {
|
||||||
// special case in couchdb usually
|
|
||||||
next('route');
|
|
||||||
} else if (utilValidatePackage(value)) {
|
|
||||||
next();
|
next();
|
||||||
} else {
|
} else {
|
||||||
next(errorUtils.getForbidden('invalid ' + name));
|
next(errorUtils.getBadRequest('invalid ' + name));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,55 +2,9 @@ import request from 'supertest';
|
||||||
|
|
||||||
import { HTTP_STATUS } from '@verdaccio/core';
|
import { HTTP_STATUS } from '@verdaccio/core';
|
||||||
|
|
||||||
import { match, validateName, validatePackage } from '../src';
|
import { match } from '../src';
|
||||||
import { getApp } from './helper';
|
import { getApp } from './helper';
|
||||||
|
|
||||||
describe('validate params', () => {
|
|
||||||
test('should validate package name', async () => {
|
|
||||||
const app = getApp([]);
|
|
||||||
// @ts-ignore
|
|
||||||
app.param('package', validatePackage);
|
|
||||||
app.get('/pkg/:package', (req, res) => {
|
|
||||||
res.status(HTTP_STATUS.OK).json({});
|
|
||||||
});
|
|
||||||
|
|
||||||
return request(app).get('/pkg/react').expect(HTTP_STATUS.OK);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('should fails validate package name', async () => {
|
|
||||||
const app = getApp([]);
|
|
||||||
// @ts-ignore
|
|
||||||
app.param('package', validatePackage);
|
|
||||||
app.get('/pkg/:package', (req, res) => {
|
|
||||||
res.status(HTTP_STATUS.OK).json({});
|
|
||||||
});
|
|
||||||
|
|
||||||
return request(app).get('/pkg/node_modules').expect(HTTP_STATUS.FORBIDDEN);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('should fails file name package name', async () => {
|
|
||||||
const app = getApp([]);
|
|
||||||
// @ts-ignore
|
|
||||||
app.param('filename', validateName);
|
|
||||||
app.get('/file/:filename', (req, res) => {
|
|
||||||
res.status(HTTP_STATUS.OK).json({});
|
|
||||||
});
|
|
||||||
|
|
||||||
return request(app).get('/file/__proto__').expect(HTTP_STATUS.FORBIDDEN);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('should validate file name package name', async () => {
|
|
||||||
const app = getApp([]);
|
|
||||||
// @ts-ignore
|
|
||||||
app.param('filename', validateName);
|
|
||||||
app.get('/file/:filename', (req, res) => {
|
|
||||||
res.status(HTTP_STATUS.OK).json({});
|
|
||||||
});
|
|
||||||
|
|
||||||
return request(app).get('/file/react.tar.gz').expect(HTTP_STATUS.OK);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('match', () => {
|
describe('match', () => {
|
||||||
test('should not match middleware', async () => {
|
test('should not match middleware', async () => {
|
||||||
const app = getApp([]);
|
const app = getApp([]);
|
||||||
|
|
92
packages/middleware/test/validation.spec.ts
Normal file
92
packages/middleware/test/validation.spec.ts
Normal file
|
@ -0,0 +1,92 @@
|
||||||
|
import request from 'supertest';
|
||||||
|
|
||||||
|
import { HTTP_STATUS } from '@verdaccio/core';
|
||||||
|
|
||||||
|
import { validateName, validatePackage } from '../src';
|
||||||
|
import { getApp } from './helper';
|
||||||
|
|
||||||
|
describe('validate package name middleware', () => {
|
||||||
|
test.each(['jquery', '-'])('%s should be valid package name', (pkg) => {
|
||||||
|
const app = getApp([]);
|
||||||
|
app.param('pkg', validatePackage);
|
||||||
|
app.get('/:pkg', (_req, res) => {
|
||||||
|
res.status(HTTP_STATUS.OK).json({});
|
||||||
|
});
|
||||||
|
|
||||||
|
return request(app).get(`/${pkg}`).expect(HTTP_STATUS.OK);
|
||||||
|
});
|
||||||
|
|
||||||
|
test.each(['node_modules', '%'])('%s should be invalid package name', (pkg) => {
|
||||||
|
const app = getApp([]);
|
||||||
|
app.param('pkg', validatePackage);
|
||||||
|
app.get('/:pkg', (_req, res) => {
|
||||||
|
res.status(HTTP_STATUS.OK).json({});
|
||||||
|
});
|
||||||
|
|
||||||
|
return request(app).get(`/${pkg}`).expect(HTTP_STATUS.BAD_REQUEST);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should validate package name double level', async () => {
|
||||||
|
const app = getApp([]);
|
||||||
|
// @ts-ignore
|
||||||
|
app.param('package', validatePackage);
|
||||||
|
app.get('/pkg/:package', (req, res) => {
|
||||||
|
res.status(HTTP_STATUS.OK).json({});
|
||||||
|
});
|
||||||
|
|
||||||
|
return request(app).get('/pkg/react').expect(HTTP_STATUS.OK);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should fails validate package name double level', async () => {
|
||||||
|
const app = getApp([]);
|
||||||
|
// @ts-ignore
|
||||||
|
app.param('package', validatePackage);
|
||||||
|
app.get('/pkg/:package', (req, res) => {
|
||||||
|
res.status(HTTP_STATUS.OK).json({});
|
||||||
|
});
|
||||||
|
|
||||||
|
return request(app).get('/pkg/node_modules').expect(HTTP_STATUS.BAD_REQUEST);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('validate file name name middleware', () => {
|
||||||
|
test.each(['old-package@0.1.2.tgz', '--0.0.1.tgz'])('%s should be valid file name', (pkg) => {
|
||||||
|
const app = getApp([]);
|
||||||
|
app.param('pkg', validateName);
|
||||||
|
app.get('/:pkg', (_req, res) => {
|
||||||
|
res.status(HTTP_STATUS.OK).json({});
|
||||||
|
});
|
||||||
|
|
||||||
|
return request(app).get(`/${pkg}`).expect(HTTP_STATUS.OK);
|
||||||
|
});
|
||||||
|
|
||||||
|
test.each(['some%2Fthing', '.bin'])('%s should be invalid package name', (pkg) => {
|
||||||
|
const app = getApp([]);
|
||||||
|
app.param('pkg', validateName);
|
||||||
|
app.get('/:pkg', (_req, res) => {
|
||||||
|
res.status(HTTP_STATUS.OK).json({});
|
||||||
|
});
|
||||||
|
|
||||||
|
return request(app).get(`/${pkg}`).expect(HTTP_STATUS.BAD_REQUEST);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should fails file name package name', async () => {
|
||||||
|
const app = getApp([]);
|
||||||
|
app.param('filename', validateName);
|
||||||
|
app.get('/file/:filename', (req, res) => {
|
||||||
|
res.status(HTTP_STATUS.OK).json({});
|
||||||
|
});
|
||||||
|
|
||||||
|
return request(app).get('/file/__proto__').expect(HTTP_STATUS.BAD_REQUEST);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should validate file name package name', async () => {
|
||||||
|
const app = getApp([]);
|
||||||
|
app.param('filename', validateName);
|
||||||
|
app.get('/file/:filename', (req, res) => {
|
||||||
|
res.status(HTTP_STATUS.OK).json({});
|
||||||
|
});
|
||||||
|
|
||||||
|
return request(app).get('/file/react.tar.gz').expect(HTTP_STATUS.OK);
|
||||||
|
});
|
||||||
|
});
|
|
@ -4,7 +4,7 @@ module.exports = Object.assign({}, config, {
|
||||||
coverageThreshold: {
|
coverageThreshold: {
|
||||||
global: {
|
global: {
|
||||||
// FIXME: increase to 90
|
// FIXME: increase to 90
|
||||||
lines: 85,
|
lines: 84,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
Loading…
Add table
Reference in a new issue