import _ from 'lodash'; import request from 'supertest'; import { Package } from '@verdaccio/types'; import { HEADERS, HEADER_TYPE, HTTP_STATUS, TOKEN_BEARER } from '../../../src/lib/constants'; import { generateRandomHexString } from '../../../src/lib/crypto-utils'; import { buildToken, encodeScopedUri } from '../../../src/lib/utils'; import { getTaggedVersionFromPackage } from './expects'; // API Helpers // This file should contain utilities to avoid repeated task over API unit testing, // Please, comply with the following: // - Promisify everything // - Encourage using constants or create new ones if it's needed // - // @ts-ignore or any is fine if there is no other way export function putPackage(request: any, pkgName: string, publishMetadata: Package, token?: string): Promise { return new Promise((resolve) => { let put = request.put(pkgName).set(HEADER_TYPE.CONTENT_TYPE, HEADERS.JSON).send(JSON.stringify(publishMetadata)); if (_.isEmpty(token) === false) { expect(token).toBeDefined(); put.set(HEADERS.AUTHORIZATION, buildToken(TOKEN_BEARER, token as string)); } put .set('accept', 'gzip') .set('accept-encoding', HEADERS.JSON) .expect(HTTP_STATUS.CREATED) .end(function (err, res) { resolve([err, res]); }); }); } export function deletePackage(request: any, pkgName: string, token?: string): Promise { return new Promise((resolve) => { let del = request.put(`/${encodeScopedUri(pkgName)}/-rev/${generateRandomHexString(8)}`).set(HEADER_TYPE.CONTENT_TYPE, HEADERS.JSON); if (_.isNil(token) === false) { del.set(HEADERS.AUTHORIZATION, buildToken(TOKEN_BEARER, token as string)); } del .set('accept-encoding', HEADERS.JSON) .expect(HTTP_STATUS.CREATED) .end(function (err, res) { resolve([err, res]); }); }); } export function getPackage(request: any, token: string, pkgName: string, statusCode: number = HTTP_STATUS.OK): Promise { return new Promise((resolve) => { let getRequest = request.get(`/${pkgName}`); // token is a string if (token !== '') { getRequest.set(HEADERS.AUTHORIZATION, buildToken(TOKEN_BEARER, token)); } getRequest .expect(HEADER_TYPE.CONTENT_TYPE, HEADERS.JSON_CHARSET) .expect(statusCode) .end(function (err, res) { resolve([err, res]); }); }); } export function loginUserToken(request: any, user: string, credentials: any, token: string, statusCode: number = HTTP_STATUS.CREATED): Promise { // $FlowFixMe return new Promise((resolve) => { request .put(`/-/user/org.couchdb.user:${user}`) .send(credentials) .set('authorization', buildToken(TOKEN_BEARER, token)) .expect(HEADER_TYPE.CONTENT_TYPE, HEADERS.JSON_CHARSET) .expect(statusCode) .end(function (err, res) { return resolve([err, res]); }); }); } export function addUser(request: any, user: string, credentials: any, statusCode: number = HTTP_STATUS.CREATED): Promise { // $FlowFixMe return new Promise((resolve) => { request .put(`/-/user/org.couchdb.user:${user}`) .send(credentials) .expect(HEADER_TYPE.CONTENT_TYPE, HEADERS.JSON_CHARSET) .expect(statusCode) .end(function (err, res) { return resolve([err, res]); }); }); } export async function getNewToken(request: any, credentials: any): Promise { /* eslint no-async-promise-executor: 0 */ return new Promise(async (resolve) => { const [err, res] = await addUser(request, credentials.name, credentials); expect(err).toBeNull(); const { token, ok } = res.body; expect(ok).toBeDefined(); expect(token).toBeDefined(); expect(typeof token).toBe('string'); resolve(token); }); } export function getProfile(request: any, token: string, statusCode: number = HTTP_STATUS.OK): Promise { // $FlowFixMe return new Promise((resolve) => { request .get(`/-/npm/v1/user`) .set('authorization', buildToken(TOKEN_BEARER, token)) .expect(HEADER_TYPE.CONTENT_TYPE, HEADERS.JSON_CHARSET) .expect(statusCode) .end(function (err, res) { return resolve([err, res]); }); }); } export function postProfile(request: any, body: any, token: string, statusCode: number = HTTP_STATUS.OK): Promise { // $FlowFixMe return new Promise((resolve) => { request .post(`/-/npm/v1/user`) .send(body) .set(HEADERS.AUTHORIZATION, `Bearer ${token}`) .expect(HEADER_TYPE.CONTENT_TYPE, HEADERS.JSON_CHARSET) .expect(statusCode) .end(function (err, res) { return resolve([err, res]); }); }); } export async function fetchPackageByVersionAndTag(app, encodedPkgName, pkgName, version, tag = 'latest') { // we retrieve the package to verify const [err, resp] = await getPackage(request(app), '', encodedPkgName); expect(err).toBeNull(); // we check whether the latest version match with the previous published one return getTaggedVersionFromPackage(resp.body, pkgName, tag, version); } export async function isExistPackage(app, packageName) { const [err] = await getPackage(request(app), '', encodeScopedUri(packageName), HTTP_STATUS.OK); return _.isNull(err); } export async function verifyPackageVersionDoesExist(app, packageName, version, token?: string) { const [, res] = await getPackage(request(app), token as string, encodeScopedUri(packageName), HTTP_STATUS.OK); const { versions } = res.body; const versionsKeys = Object.keys(versions); return versionsKeys.includes(version) === false; } export function generateUnPublishURI(pkgName) { return `/${encodeScopedUri(pkgName)}/-rev/${generateRandomHexString(8)}`; }