0
Fork 0
mirror of https://github.com/verdaccio/verdaccio.git synced 2024-12-30 22:34:10 -05:00

feat: migrate yarn workspaces (#1546)

This commit is contained in:
Juan Picado @jotadeveloper 2020-03-03 23:59:19 +01:00 committed by Juan Picado
parent 73585f0262
commit a70454c7b2
451 changed files with 78127 additions and 7204 deletions

143
.circleci/config.yml Normal file
View file

@ -0,0 +1,143 @@
version: 2.1
executors:
node_latest_browser:
docker:
- image: circleci/node:latest-browsers
node_latest:
docker:
- image: circleci/node:13
node_lts_12:
docker:
- image: circleci/node:12
default_executor: node_latest
aliases:
- &repo_path
~/verdaccio
- &defaults
working_directory: *repo_path
- &yarn_cache_key
yarn-sha-{{ checksum "yarn.lock" }}
- &coverage_key
coverage-{{ .Branch }}-{{ .Revision }}
- &ignore_non_dev_branches
filters:
tags:
only: /.*/
branches:
ignore:
- gh-pages
- /release\/.*/
commands:
restore_repo:
description: Restore repository from workspace
steps:
- attach_workspace:
at: *repo_path
run_test:
description: Run test and functional test
steps:
- run:
name: Test
command: yarn run test
- store_test_results:
path: reports/
jobs:
prepare:
<<: *defaults
executor: default_executor
steps:
- checkout
- restore_cache:
key: *yarn_cache_key
- run:
name: Install dependencies
command: yarn
- run:
name: Install dependencies
command: yarn bootstrap
- run:
name: Prepare CI
command: yarn lint
- run:
name: Clean project
command: yarn clean
- run:
name: Build project
command: yarn build
- save_cache:
key: *yarn_cache_key
paths:
- ~/.yarn
- ~/.cache/yarn
- node_modules
- persist_to_workspace:
root: *repo_path
paths:
- ./*
test_node_latest:
<<: *defaults
executor: node_latest
steps:
- restore_repo
- run_test
- save_cache:
key: *coverage_key
paths:
- coverage
test_node_lts_12:
<<: *defaults
executor: node_lts_12
steps:
- restore_repo
- run_test
# test_e2e_cli:
# <<: *defaults
# executor: node_latest
# steps:
# - restore_repo
# - run:
# name: Test End-to-End ClI
# command: yarn run test:e2e:cli
coverage:
<<: *defaults
executor: default_executor
steps:
- restore_repo
- restore_cache:
key: *coverage_key
- run:
name: Publish coverage
command: yarn run coverage:publish
- store_artifacts:
path: coverage
workflows:
version: 2
workflow:
jobs:
- prepare:
<<: *ignore_non_dev_branches
- test_node_latest:
requires:
- prepare
<<: *ignore_non_dev_branches
- test_node_lts_12:
requires:
- prepare
<<: *ignore_non_dev_branches
# - test_e2e_cli:
# requires:
# - prepare
# - test_node_latest
# - test_node_lts_12
# <<: *ignore_non_dev_branches
# - coverage:
# requires:
# - test_e2e_cli
# <<: *ignore_non_dev_branches

View file

@ -15,8 +15,7 @@ Dockerfile
*.png *.png
*.jpg *.jpg
*.sh *.sh
*.ico **/partials/**
test/unit/partials/ **/fixtures/**
types/custom.d.ts types/custom.d.ts
docker-examples/ **/mock/store/**
LICENSE

View file

@ -9,21 +9,25 @@ jobs:
strategy: strategy:
fail-fast: false fail-fast: false
matrix: matrix:
node_version: [12, 14, 15] node_version: [8, 10, 12, 13]
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2.3.3 - uses: actions/checkout@v2.3.1
- name: Use Node ${{ matrix.node_version }} - name: Use Node ${{ matrix.node_version }}
uses: actions/setup-node@v2.1.5 uses: actions/setup-node@v2.1.1
with: with:
node-version: ${{ matrix.node_version }} node_version: ${{ matrix.node_version }}
- name: Install - name: Install
run: yarn install --immutable run: yarn
- name: Build - name: Bootstrap Lerna
run: yarn code:build run: yarn bootstrap
- name: Lint - name: Lint
run: yarn lint run: yarn lint
- name: Clean
run: yarn clean
- name: Build
run: yarn build
- name: Test - name: Test
run: yarn test run: yarn test

View file

@ -9,7 +9,7 @@ jobs:
steps: steps:
- uses: actions/checkout@v2.3.3 - uses: actions/checkout@v2.3.3
- name: Build - name: Build
run: docker build . run: yarn docker
env: env:
VERDACCIO_BUILD_REGISTRY: https://registry.verdaccio.org VERDACCIO_BUILD_REGISTRY: https://registry.verdaccio.org

29
.github/workflows/release-canary.yml vendored Normal file
View file

@ -0,0 +1,29 @@
name: Canary Release to Verdaccio
on: [push]
jobs:
release:
name: Release
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v1
- name: Use Node (latest)
uses: actions/setup-node@v1
with:
node_version: 13
- name: Install
run: yarn install --no-lockfile
- name: Lint
run: yarn lint
- name: Clean
run: yarn clean
- name: Build
run: yarn build
- name: Test
run: yarn test
- name: Publish
run: |
echo "//registry.verdaccio.org/:_authToken=${{ secrets.VERDACCIO_TOKEN }}" > .npmrc
git update-index --assume-unchanged .npmrc
yarn publish:canary

View file

@ -1,32 +0,0 @@
name: Release
on:
push:
tags:
- '*'
jobs:
release:
name: Release
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2.3.3
- name: Use Node (latest)
uses: actions/setup-node@v2.1.5
with:
node-version: 14
- name: Install
run: yarn install
- name: Build
run: yarn code:build
- name: Lint
run: yarn lint
- name: Publish
run: sh scripts/publish.sh
env:
REGISTRY_AUTH_TOKEN: ${{ secrets.REGISTRY_AUTH_TOKEN }}
REGISTRY_URL: registry.npmjs.org
- name: Create release notes
run: sh scripts/github-release.sh
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}

25
.gitignore vendored
View file

@ -5,41 +5,22 @@ build/
### Test ### Test
test/unit/partials/store/test-*-storage/*
test/unit/partials/store/*-storage/*
test/unit/partials/store/storage_default_storage/*
.verdaccio-db.json .verdaccio-db.json
.sinopia-db.json .sinopia-db.json
### ###
!bin/verdaccio
test-storage* test-storage*
access-storage*
.verdaccio_test_env .verdaccio_test_env
node_modules node_modules
package-lock.json package-lock.json
npm_test-fails-add-tarball*
yarn-error.log yarn-error.log
# Istanbul
# jest
reports/ reports/
coverage/ coverage/
.nyc*
.idea/ .idea/
# React
bundle.js
bundle.js.map
__tests__
# Compiled script # Compiled script
static/* packages/partials
# This is the Yarn build state; it's local to each clone
/.yarn/build-state.yml
# This is the Yarn install state cache, it can be rebuilt anytime
/.yarn/install-state.gz
.history

View file

@ -1,41 +0,0 @@
{
"processors": ["stylelint-processor-styled-components"],
"extends": [
"stylelint-config-recommended"
],
"rules": {
"at-rule-no-unknown": true,
"block-no-empty": true,
"color-named": "always-where-possible",
"comment-no-empty": true,
"declaration-block-no-duplicate-properties": [
true,
{
ignore: ["consecutive-duplicates-with-different-values"]
}
],
"declaration-block-no-shorthand-property-overrides": true,
"font-family-no-duplicate-names": true,
"color-no-invalid-hex": true,
"font-family-no-missing-generic-family-keyword": true,
"function-calc-no-unspaced-operator": true,
"function-linear-gradient-no-nonstandard-direction": true,
"keyframe-declaration-no-important": true,
"property-no-vendor-prefix": true,
"media-feature-name-no-unknown": true,
"no-descending-specificity": [true, { "severity": "warning" }],
"no-duplicate-at-import-rules": true,
"no-duplicate-selectors": true,
"no-empty-source": true,
"no-extra-semicolons": true,
"no-invalid-double-slash-comments": true,
"property-no-unknown": true,
"selector-pseudo-class-no-unknown": true,
"selector-pseudo-element-no-unknown": true,
"selector-type-no-unknown": [true, { "severity": "warning" }],
"string-no-newline": true,
"unit-no-unknown": true
}
}

2
.yarnrc Normal file
View file

@ -0,0 +1,2 @@
save-prefix ""
registry "https://registry.verdaccio.org"

View file

@ -1,4 +1,4 @@
FROM --platform=${BUILDPLATFORM:-linux/amd64} node:14.16.1-alpine as builder FROM node:12.18.3-alpine as builder
ENV NODE_ENV=production \ ENV NODE_ENV=production \
VERDACCIO_BUILD_REGISTRY=https://registry.verdaccio.org VERDACCIO_BUILD_REGISTRY=https://registry.verdaccio.org
@ -12,18 +12,14 @@ RUN apk --no-cache add openssl ca-certificates wget && \
WORKDIR /opt/verdaccio-build WORKDIR /opt/verdaccio-build
COPY . . COPY . .
RUN yarn config set npmRegistryServer $VERDACCIO_BUILD_REGISTRY && \ RUN yarn config set registry $VERDACCIO_BUILD_REGISTRY && \
yarn config set enableProgressBars false && \ yarn install --production=false && \
yarn config set enableTelemetry false && \ yarn build && \
yarn install && \
yarn lint && \ yarn lint && \
yarn code:docker-build && \
yarn cache clean && \ yarn cache clean && \
yarn workspaces focus --production yarn install --production=true
FROM node:12.18.3-alpine
FROM node:14.16.1-alpine
LABEL maintainer="https://github.com/verdaccio/verdaccio" LABEL maintainer="https://github.com/verdaccio/verdaccio"
ENV VERDACCIO_APPDIR=/opt/verdaccio \ ENV VERDACCIO_APPDIR=/opt/verdaccio \
@ -42,10 +38,11 @@ RUN mkdir -p /verdaccio/storage /verdaccio/plugins /verdaccio/conf
COPY --from=builder /opt/verdaccio-build . COPY --from=builder /opt/verdaccio-build .
ADD conf/docker.yaml /verdaccio/conf/config.yaml RUN ls packages/config/src/conf
ADD packages/config/src/conf/docker.yaml /verdaccio/conf/config.yaml
RUN adduser -u $VERDACCIO_USER_UID -S -D -h $VERDACCIO_APPDIR -g "$VERDACCIO_USER_NAME user" -s /sbin/nologin $VERDACCIO_USER_NAME && \ RUN adduser -u $VERDACCIO_USER_UID -S -D -h $VERDACCIO_APPDIR -g "$VERDACCIO_USER_NAME user" -s /sbin/nologin $VERDACCIO_USER_NAME && \
chmod -R +x $VERDACCIO_APPDIR/bin $VERDACCIO_APPDIR/docker-bin && \ chmod -R +x $VERDACCIO_APPDIR/packages/verdaccio/bin $VERDACCIO_APPDIR/docker-bin && \
chown -R $VERDACCIO_USER_UID:root /verdaccio/storage && \ chown -R $VERDACCIO_USER_UID:root /verdaccio/storage && \
chmod -R g=u /verdaccio/storage /etc/passwd chmod -R g=u /verdaccio/storage /etc/passwd
@ -57,4 +54,4 @@ VOLUME /verdaccio/storage
ENTRYPOINT ["uid_entrypoint"] ENTRYPOINT ["uid_entrypoint"]
CMD $VERDACCIO_APPDIR/bin/verdaccio --config /verdaccio/conf/config.yaml --listen $VERDACCIO_PROTOCOL://0.0.0.0:$VERDACCIO_PORT CMD $VERDACCIO_APPDIR/packages/verdaccio/bin/verdaccio --config /verdaccio/conf/config.yaml --listen $VERDACCIO_PROTOCOL://0.0.0.0:$VERDACCIO_PORT

View file

@ -1,3 +0,0 @@
#!/usr/bin/env node
require('../build/lib/cli');

4
debug/.babelrc Normal file
View file

@ -0,0 +1,4 @@
{
"presets": [["@verdaccio"]],
"debug": true
}

2
debug/bootstrap.js vendored
View file

@ -3,4 +3,4 @@
require('@babel/register')({ require('@babel/register')({
extensions: [".ts", ".js"] extensions: [".ts", ".js"]
}); });
require('../src/lib/cli'); require('../packages/cli/src/index');

1
debug/debug.js Normal file
View file

@ -0,0 +1 @@
require('@verdaccio/cli');

9
debug/package.json Normal file
View file

@ -0,0 +1,9 @@
{
"name": "debug",
"private": true,
"version": "1.0.0",
"dependencies": {
"@verdaccio/babel-preset": "^8.2.0",
"@verdaccio/eslint-config": "^8.2.0"
}
}

14
lerna.json Normal file
View file

@ -0,0 +1,14 @@
{
"version": "5.0.0-alpha.0",
"packages": [
"packages/*"
],
"npmClient": "yarn",
"useWorkspaces": true,
"stream": true,
"command": {
"publish": {
"message": "chore: publish release %s"
}
}
}

View file

@ -1,7 +1,4 @@
{ {
"name": "verdaccio",
"version": "5.0.0",
"description": "A lightweight private npm proxy registry",
"author": { "author": {
"name": "Verdaccio Maintainers", "name": "Verdaccio Maintainers",
"email": "verdaccio.npm@gmail.com" "email": "verdaccio.npm@gmail.com"
@ -11,205 +8,80 @@
"url": "git://github.com/verdaccio/verdaccio" "url": "git://github.com/verdaccio/verdaccio"
}, },
"homepage": "https://verdaccio.org", "homepage": "https://verdaccio.org",
"main": "build/index.js", "private": true,
"bin": "./bin/verdaccio", "workspaces": [
"packages/*"
],
"funding": { "funding": {
"type": "opencollective", "type": "opencollective",
"url": "https://opencollective.com/verdaccio" "url": "https://opencollective.com/verdaccio"
}, },
"dependencies": {
"@verdaccio/commons-api": "10.0.0",
"@verdaccio/local-storage": "10.0.1",
"@verdaccio/readme": "10.0.0",
"@verdaccio/streams": "10.0.0",
"@verdaccio/ui-theme": "3.0.1",
"JSONStream": "1.3.5",
"async": "3.2.0",
"body-parser": "1.19.0",
"clipanion": "3.0.0-rc.11",
"compression": "1.7.4",
"cookies": "0.8.0",
"cors": "2.8.5",
"dayjs": "1.10.4",
"debug": "^4.3.1",
"envinfo": "7.7.4",
"express": "4.17.1",
"fast-safe-stringify": "^2.0.7",
"handlebars": "4.7.7",
"http-errors": "1.8.0",
"js-yaml": "4.0.0",
"jsonwebtoken": "8.5.1",
"kleur": "4.1.4",
"lodash": "4.17.21",
"lru-cache": "6.0.0",
"lunr-mutable-indexes": "2.3.2",
"marked": "2.0.1",
"memoizee": "0.4.15",
"mime": "2.5.2",
"minimatch": "3.0.4",
"mkdirp": "1.0.4",
"mv": "2.1.1",
"pino": "6.11.2",
"pkginfo": "0.4.1",
"prettier-bytes": "^1.0.3",
"pretty-ms": "^5.0.0",
"request": "2.88.0",
"semver": "7.3.4",
"validator": "13.5.2",
"verdaccio-audit": "10.0.0",
"verdaccio-htpasswd": "10.0.0"
},
"devDependencies": { "devDependencies": {
"@babel/cli": "7.13.0", "@commitlint/cli": "8.3.5",
"@babel/core": "7.13.8", "@commitlint/config-conventional": "8.2.0",
"@babel/node": "7.13.0", "@octokit/rest": "16.28.9",
"@babel/plugin-proposal-class-properties": "7.13.0", "@types/async": "3.0.3",
"@babel/plugin-proposal-decorators": "7.13.5",
"@babel/plugin-proposal-export-namespace-from": "7.12.13",
"@babel/plugin-proposal-function-sent": "7.12.13",
"@babel/plugin-proposal-json-strings": "7.13.8",
"@babel/plugin-proposal-numeric-separator": "7.12.13",
"@babel/plugin-proposal-object-rest-spread": "7.13.8",
"@babel/plugin-proposal-throw-expressions": "7.12.13",
"@babel/plugin-syntax-dynamic-import": "7.8.3",
"@babel/plugin-syntax-import-meta": "7.10.4",
"@babel/plugin-transform-async-to-generator": "7.13.0",
"@babel/plugin-transform-classes": "7.13.0",
"@babel/plugin-transform-runtime": "7.13.9",
"@babel/polyfill": "^7.12.1",
"@babel/preset-env": "7.13.9",
"@babel/preset-typescript": "7.13.0",
"@babel/register": "7.13.8",
"@babel/runtime": "7.13.9",
"@commitlint/cli": "12.0.1",
"@commitlint/config-conventional": "12.0.1",
"@octokit/rest": "16.43.2",
"@types/async": "3.2.4",
"@types/bunyan": "1.8.6", "@types/bunyan": "1.8.6",
"@types/express": "4.17.6", "@types/express": "4.17.1",
"@types/http-errors": "1.8.0", "@types/http-errors": "1.6.3",
"@types/jest": "26.0.14", "@types/jest": "24.0.25",
"@types/lodash": "4.14.167", "@types/lodash": "4.14.149",
"@types/mime": "2.0.1", "@types/mime": "2.0.1",
"@types/minimatch": "3.0.3", "@types/minimatch": "3.0.3",
"@types/node": "14.14.37", "@types/node": "12.12.21",
"@types/pino": "6.3.6", "@types/request": "2.48.3",
"@types/request": "2.48.5", "@types/semver": "6.2.0",
"@types/semver": "7.3.4", "@types/express-serve-static-core": "4.17.1",
"@typescript-eslint/eslint-plugin": "4.13.0", "@verdaccio/babel-preset": "^8.5.0",
"@typescript-eslint/parser": "4.13.0", "@verdaccio/eslint-config": "^9.0.0",
"@verdaccio/eslint-config": "^8.5.0", "@verdaccio/types": "^9.0.0",
"@verdaccio/types": "^9.7.2", "codecov": "3.6.1",
"all-contributors-cli": "6.20.0", "cross-env": "6.0.3",
"babel-eslint": "10.1.0", "detect-secrets": "1.0.5",
"babel-jest": "26.6.3", "eslint": "6.8.0",
"babel-loader": "^8.2.2", "fs-extra": "8.1.0",
"babel-plugin-dynamic-import-node": "2.3.3", "get-stdin": "7.0.0",
"codecov": "3.8.1", "kleur": "3.0.3",
"cross-env": "7.0.3",
"detect-secrets": "1.0.6",
"eslint": "7.19.0",
"eslint-config-google": "0.14.0",
"eslint-config-prettier": "7.2.0",
"eslint-plugin-babel": "5.3.1",
"eslint-plugin-import": "2.22.1",
"eslint-plugin-jest": "24.1.3",
"eslint-plugin-jsx-a11y": "6.4.1",
"eslint-plugin-react": "7.22.0",
"eslint-plugin-react-hooks": "4.2.0",
"eslint-plugin-simple-import-sort": "7.0.0",
"eslint-plugin-verdaccio": "9.6.1",
"fs-extra": "9.1.0",
"get-stdin": "8.0.0",
"husky": "2.7.0", "husky": "2.7.0",
"in-publish": "2.0.1", "in-publish": "2.0.0",
"jest": "25.5.4", "jest": "^24.9.0",
"jest-environment-node": "25.5.0", "jest-environment-node": "^24.9.0",
"jest-junit": "9.0.0", "jest-junit": "^9.0.0",
"lerna": "^3.18.4",
"lint-staged": "8.2.1", "lint-staged": "8.2.1",
"lockfile-lint": "4.3.7", "nock": "^11.7.2",
"nock": "12.0.3", "prettier": "^1.19.1",
"node-mocks-http": "^1.10.1", "rimraf": "3.0.0",
"prettier": "2.2.1", "selfsigned": "1.10.7",
"puppeteer": "5.5.0", "standard-version": "^7.0.1",
"rimraf": "3.0.2", "supertest": "^4.0.2",
"selfsigned": "1.10.8", "typescript": "^3.7.5",
"standard-version": "9.1.1", "verdaccio-auth-memory": "^8.5.0",
"supertest": "6.1.1", "verdaccio-memory": "^8.5.0",
"typescript": "4.1.3", "verdaccio": "^4.4.0"
"verdaccio-auth-memory": "10.0.0",
"verdaccio-memory": "10.0.0"
}, },
"keywords": [
"private",
"package",
"repository",
"registry",
"enterprise",
"modules",
"proxy",
"server",
"verdaccio"
],
"scripts": { "scripts": {
"release": "standard-version -a -s", "bootstrap": "lerna bootstrap",
"prepublish": "in-publish && yarn run code:build || not-in-publish", "debug": "node debug/bootstrap.js",
"type-check": "tsc --noEmit", "dev": "cross-env BABEL_ENV=registry babel-node --extensions \".ts,.tsx\" packages/cli/src",
"type-check:watch": "yarn run type-check -- --watch", "clean": "lerna run clean",
"pretest": "yarn run code:build", "build": "lerna run build",
"format": "prettier --single-quote --trailing-comma none --write \"{src,test}/**/*.ts\"", "docker": "docker build -t verdaccio/verdaccio:local . --no-cache",
"format:check": "prettier --check \"**/*.{js,jsx,ts,tsx,json,yml,yaml,md}\" --debug-check", "release": "lerna version --conventional-commits",
"test": "yarn run test:unit", "version:canary": "lerna version --conventional-commits --conventional-prerelease --no-commit-hooks --exact --no-changelog --yes --preid alpha",
"test:clean": "npx jest --clearCache", "publish:canary": "lerna publish from-package --canary --yes --no-git-reset --no-git-tag-version --no-push --force-publish --concurrency 1",
"test:unit": "cross-env NODE_ENV=test TZ=UTC FORCE_COLOR=1 jest --config ./jest.config.js --maxWorkers 2 --passWithNoTests", "release:from-prerelease": "lerna version --conventional-commits --conventional-graduate",
"test:functional": "cross-env NODE_ENV=test jest --config ./test/jest.config.functional.js --testPathPattern ./test/functional/index* --passWithNoTests", "release:prerelease": "lerna version --conventional-commits --conventional-prerelease --preid next --registry http://localhost:4873",
"test:e2e:cli": "cross-env NODE_ENV=test jest --config ./test/e2e-cli/jest.config.e2e.cli.js --passWithNoTests", "release:publish": "lerna publish from-git",
"test:e2e": "yarn jest --config ./test/jest.config.e2e.js", "release:publish-prerelease": "lerna publish from-git --pre-dist-tag next",
"test:all": "yarn run test && yarn run test:functional && yarn run test:e2e & yarn run test:e2e:pkg", "lint": "eslint \"packages/**/@(src|tests)/**\"",
"pre:ci": "yarn run lint", "test": "lerna run test --concurrency 1",
"coverage:publish": "codecov", "test:e2e:cli": "cross-env NODE_ENV=test jest --config ./test/e2e-cli/jest.config.e2e.cli.js --passWithNoTests"
"lint": "yarn run type-check && yarn run lint:ts",
"lint:ts": "eslint \"**/*.{js,jsx,ts,tsx}\"",
"lint:lockfile": "lockfile-lint --path yarn.lock --type yarn --validate-https --allowed-hosts verdaccio npm yarn",
"start": "yarn babel-node --extensions \".ts,.tsx\" src/lib/cli",
"start:debug": "yarn node debug/bootstrap.js",
"code:build": "yarn babel src/ --out-dir build/ --copy-files --extensions \".ts,.tsx\" --source-maps inline",
"code:docker-build": "yarn babel src/ --out-dir build/ --copy-files --extensions \".ts,.tsx\"",
"docker": "docker build -t verdaccio/verdaccio:pr-2122 . --no-cache",
"docker:run": "docker run -it --rm -p 4873:4873 verdaccio/verdaccio:local"
},
"engines": {
"node": ">=12",
"npm": ">=6"
},
"preferGlobal": true,
"husky": {
"hooks": {
"pre-commit": "lint-staged",
"commit-msg": "commitlint -e $GIT_PARAMS"
}
},
"lint-staged": {
"relative": true,
"linters": {
"*": [
"eslint .",
"git add"
]
},
"ignore": [
"*.json"
]
}, },
"license": "MIT", "license": "MIT",
"commitlint": { "commitlint": {
"extends": [ "extends": [
"@commitlint/config-conventional" "@commitlint/config-conventional"
] ]
},
"collective": {
"type": "opencollective",
"url": "https://opencollective.com/verdaccio",
"logo": "https://opencollective.com/verdaccio/logo.txt"
} }
} }

3
packages/api/.babelrc Normal file
View file

@ -0,0 +1,3 @@
{
"presets": [["@verdaccio"]]
}

View file

@ -0,0 +1,9 @@
module.exports = {
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx'],
transform: {
'^.+\\.(js|jsx|ts|tsx)$': 'babel-jest',
},
verbose: true,
collectCoverage: true,
coveragePathIgnorePatterns: ['node_modules', 'fixtures'],
};

43
packages/api/package.json Normal file
View file

@ -0,0 +1,43 @@
{
"name": "@verdaccio/api",
"version": "5.0.0-alpha.0",
"description": "loaders logic",
"main": "./build/index.js",
"types": "build/index.d.ts",
"author": {
"name": "Juan Picado",
"email": "juanpicado19@gmail.com"
},
"repository": {
"type": "git",
"url": "git://github.com/verdaccio/verdaccio"
},
"homepage": "https://verdaccio.org",
"scripts": {
"clean": "rimraf ./build",
"test": "cross-env NODE_ENV=test BABEL_ENV=test jest",
"type-check": "tsc --noEmit",
"build:types": "tsc --emitDeclarationOnly --declaration true",
"build:js": "cross-env BABEL_ENV=registry babel src/ --out-dir build/ --copy-files --extensions \".ts,.tsx\" --source-maps",
"build": "npm run build:js && npm run build:types"
},
"license": "MIT",
"dependencies": {
"@verdaccio/commons-api": "^9.0.0",
"@verdaccio/dev-commons": "5.0.0-alpha.0",
"@verdaccio/hooks": "5.0.0-alpha.0",
"@verdaccio/logger": "5.0.0-alpha.0",
"@verdaccio/middleware": "5.0.0-alpha.0",
"@verdaccio/utils": "5.0.0-alpha.0",
"body-parser": "^1.19.0",
"cookies": "^0.8.0",
"express": "4.17.1",
"lodash": "^4.17.15",
"mime": "^2.4.4"
},
"devDependencies": {
"@verdaccio/dev-types": "5.0.0-alpha.0",
"@verdaccio/types": "^9.0.0"
},
"gitHead": "7c246ede52ff717707fcae66dd63fc4abd536982"
}

View file

@ -0,0 +1,78 @@
import mime from 'mime';
import _ from 'lodash';
import{ Router } from 'express';
import { media, allow } from '@verdaccio/middleware';
import { API_MESSAGE, HTTP_STATUS, DIST_TAGS } from '@verdaccio/dev-commons';
import { VerdaccioError } from '@verdaccio/commons-api';
import { Package } from '@verdaccio/types';
// @ts-ignore
import{ IAuth, $ResponseExtend, $RequestExtend, $NextFunctionVer, IStorageHandler } from '@verdaccio/dev-types';
export default function(route: Router, auth: IAuth, storage: IStorageHandler): void {
const can = allow(auth);
const tag_package_version = function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): $NextFunctionVer {
if (_.isString(req.body) === false) {
return next('route');
}
const tags = {};
tags[req.params.tag] = req.body;
storage.mergeTags(req.params.package, tags, function(err: Error): $NextFunctionVer {
if (err) {
return next(err);
}
res.status(HTTP_STATUS.CREATED);
return next({ ok: API_MESSAGE.TAG_ADDED });
});
};
// tagging a package
route.put('/:package/:tag', can('publish'), media(mime.getType('json')), tag_package_version);
route.post('/-/package/:package/dist-tags/:tag', can('publish'), media(mime.getType('json')), tag_package_version);
route.put('/-/package/:package/dist-tags/:tag', can('publish'), media(mime.getType('json')), tag_package_version);
route.delete('/-/package/:package/dist-tags/:tag', can('publish'), function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {
const tags = {};
tags[req.params.tag] = null;
storage.mergeTags(req.params.package, tags, function(err: VerdaccioError): $NextFunctionVer {
if (err) {
return next(err);
}
res.status(HTTP_STATUS.CREATED);
return next({
ok: API_MESSAGE.TAG_REMOVED,
});
});
});
route.get('/-/package/:package/dist-tags', can('access'), function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {
storage.getPackage({
name: req.params.package,
uplinksLook: true,
req,
callback: function(err: VerdaccioError, info: Package): $NextFunctionVer {
if (err) {
return next(err);
}
next(info[DIST_TAGS]);
},
});
});
route.post('/-/package/:package/dist-tags', can('publish'), function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {
storage.mergeTags(req.params.package, req.body, function(err: VerdaccioError): $NextFunctionVer {
if (err) {
return next(err);
}
res.status(HTTP_STATUS.CREATED);
return next({
ok: API_MESSAGE.TAG_UPDATED,
});
});
});
}

View file

@ -1,29 +1,24 @@
import { Config } from '@verdaccio/types';
import _ from 'lodash'; import _ from 'lodash';
import express from 'express'; import express from 'express';
import { match, validateName, validatePackage, encodeScopePackage, antiLoop } from '@verdaccio/middleware';
import { IAuth, IStorageHandler } from '@verdaccio/dev-types';
import { Config } from '@verdaccio/types';
import bodyParser from 'body-parser'; import bodyParser from 'body-parser';
import { IAuth, IStorageHandler } from '../../../types';
import whoami from './api/whoami';
import ping from './api/ping';
import user from './api/user';
import distTags from './api/dist-tags';
import publish from './api/publish';
import search from './api/search';
import pkg from './api/package';
import stars from './api/stars';
import profile from './api/v1/profile';
import token from './api/v1/token';
import v1Search from './api/v1/search'; import whoami from './whoami';
import ping from './ping';
import user from './user';
import distTags from './dist-tags';
import publish from './publish';
import search from './search';
import pkg from './package';
import stars from './stars';
import profile from './v1/profile';
import token from './v1/token';
import v1Search from './api/v1/search'
const { const { match, validateName, validatePackage, encodeScopePackage, antiLoop } = require('../middleware');
match,
validateName,
validatePackage,
encodeScopePackage,
antiLoop
} = require('../middleware');
export default function(config: Config, auth: IAuth, storage: IStorageHandler) { export default function(config: Config, auth: IAuth, storage: IStorageHandler) {
/* eslint new-cap:off */ /* eslint new-cap:off */
@ -49,6 +44,7 @@ export default function (config: Config, auth: IAuth, storage: IStorageHandler)
app.use(auth.apiJWTmiddleware()); app.use(auth.apiJWTmiddleware());
app.use(bodyParser.json({ strict: false, limit: config.max_body_size || '10mb' })); app.use(bodyParser.json({ strict: false, limit: config.max_body_size || '10mb' }));
// @ts-ignore
app.use(antiLoop(config)); app.use(antiLoop(config));
// encode / in a scoped package name to be matched as a single parameter in routes // encode / in a scoped package name to be matched as a single parameter in routes
app.use(encodeScopePackage); app.use(encodeScopePackage);
@ -64,7 +60,7 @@ export default function (config: Config, auth: IAuth, storage: IStorageHandler)
stars(app, storage); stars(app, storage);
if (_.get(config, 'experiments.search') === true) { if (_.get(config, 'experiments.search') === true) {
v1Search(app, auth, storage); v1Search(app, auth, storage)
} }
if (_.get(config, 'experiments.token') === true) { if (_.get(config, 'experiments.token') === true) {

View file

@ -0,0 +1,74 @@
import _ from 'lodash';
import { Router } from 'express';
import { allow } from '@verdaccio/middleware';
import { convertDistRemoteToLocalTarballUrls, getVersion, ErrorCode } from '@verdaccio/utils';
import { HEADERS, DIST_TAGS, API_ERROR } from '@verdaccio/dev-commons';
import { Config, Package } from '@verdaccio/types';
import { IAuth, $ResponseExtend, $RequestExtend, $NextFunctionVer, IStorageHandler } from '@verdaccio/dev-types';
const downloadStream = (packageName: string, filename: string, storage: any, req: $RequestExtend, res: $ResponseExtend): void => {
const stream = storage.getTarball(packageName, filename);
stream.on('content-length', function(content): void {
res.header('Content-Length', content);
});
stream.on('error', function(err): void {
return res.report_error(err);
});
res.header(HEADERS.CONTENT_TYPE, HEADERS.OCTET_STREAM);
stream.pipe(res);
};
export default function(route: Router, auth: IAuth, storage: IStorageHandler, config: Config): void {
const can = allow(auth);
// TODO: anonymous user?
route.get('/:package/:version?', can('access'), function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {
const getPackageMetaCallback = function(err, metadata: Package): void {
if (err) {
return next(err);
}
metadata = convertDistRemoteToLocalTarballUrls(metadata, req, config.url_prefix);
let queryVersion = req.params.version;
if (_.isNil(queryVersion)) {
return next(metadata);
}
let version = getVersion(metadata, queryVersion);
if (_.isNil(version) === false) {
return next(version);
}
if (_.isNil(metadata[DIST_TAGS]) === false) {
if (_.isNil(metadata[DIST_TAGS][queryVersion]) === false) {
queryVersion = metadata[DIST_TAGS][queryVersion];
version = getVersion(metadata, queryVersion);
if (_.isNil(version) === false) {
return next(version);
}
}
}
return next(ErrorCode.getNotFound(`${API_ERROR.VERSION_NOT_EXIST}: ${req.params.version}`));
};
storage.getPackage({
name: req.params.package,
uplinksLook: true,
req,
callback: getPackageMetaCallback,
});
});
route.get('/:scopedPackage/-/:scope/:filename', can('access'), function(req: $RequestExtend, res: $ResponseExtend): void {
const { scopedPackage, filename } = req.params;
downloadStream(scopedPackage, filename, storage, req, res);
});
route.get('/:package/-/:filename', can('access'), function(req: $RequestExtend, res: $ResponseExtend): void {
downloadStream(req.params.package, req.params.filename, storage, req, res);
});
}

View file

@ -1,10 +1,5 @@
/**
* @prettier
* @flow
*/
import { Router } from 'express'; import { Router } from 'express';
import { $RequestExtend, $ResponseExtend, $NextFunctionVer } from '../../../../types'; import { $RequestExtend, $ResponseExtend, $NextFunctionVer } from '@verdaccio/dev-types';
export default function (route: Router): void { export default function (route: Router): void {
route.get( route.get(

View file

@ -1,21 +1,18 @@
import Path from 'path';
import _ from 'lodash'; import _ from 'lodash';
import buildDebug from 'debug'; import Path from 'path';
import mime from 'mime'; import mime from 'mime';
import { Router } from 'express'; import { Router } from 'express';
import { IAuth, $ResponseExtend, $RequestExtend, $NextFunctionVer, IStorageHandler } from '@verdaccio/dev-types';
import { API_MESSAGE, HEADERS, DIST_TAGS, API_ERROR, HTTP_STATUS } from '@verdaccio/dev-commons';
import {validateMetadata, isObject, ErrorCode, hasDiffOneKey, isRelatedToDeprecation} from '@verdaccio/utils';
import { media, expectJson, allow } from '@verdaccio/middleware';
import { notify } from '@verdaccio/hooks';
import { Config, Callback, MergeTags, Version, Package } from '@verdaccio/types'; import { Config, Callback, MergeTags, Version, Package } from '@verdaccio/types';
import { API_MESSAGE, HEADERS, DIST_TAGS, API_ERROR, HTTP_STATUS } from '../../../lib/constants'; import { logger } from '@verdaccio/logger';
import { validateMetadata, isObject, ErrorCode, hasDiffOneKey, isRelatedToDeprecation } from '../../../lib/utils';
import { media, expectJson, allow } from '../../middleware';
import { notify } from '../../../lib/notify';
import { IAuth, $ResponseExtend, $RequestExtend, $NextFunctionVer, IStorageHandler } from '../../../../types';
import { logger } from '../../../lib/logger';
import { isPublishablePackage } from '../../../lib/storage-utils';
import star from './star'; import star from './star';
import {isPublishablePackage} from "./utils";
const debug = buildDebug('verdaccio:publish');
export default function publish(router: Router, auth: IAuth, storage: IStorageHandler, config: Config): void { export default function publish(router: Router, auth: IAuth, storage: IStorageHandler, config: Config): void {
const can = allow(auth); const can = allow(auth);
@ -108,7 +105,9 @@ export function publishPackage(storage: IStorageHandler, config: Config, auth: I
const starApi = star(storage); const starApi = star(storage);
return function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void { return function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {
const packageName = req.params.package; const packageName = req.params.package;
debug('publishing or updating a new version for %o', packageName);
logger.debug({packageName} , `publishing or updating a new version for @{packageName}`);
/** /**
* Write tarball of stream data from package clients. * Write tarball of stream data from package clients.
*/ */
@ -161,7 +160,8 @@ export function publishPackage(storage: IStorageHandler, config: Config, auth: I
// npm-registry-client 0.3+ embeds tarball into the json upload // npm-registry-client 0.3+ embeds tarball into the json upload
// https://github.com/isaacs/npm-registry-client/commit/e9fbeb8b67f249394f735c74ef11fe4720d46ca0 // https://github.com/isaacs/npm-registry-client/commit/e9fbeb8b67f249394f735c74ef11fe4720d46ca0
// issue https://github.com/rlidwka/sinopia/issues/31, dealing with it here: // issue https://github.com/rlidwka/sinopia/issues/31, dealing with it here:
const isInvalidBodyFormat = isObject(_attachments) === false || hasDiffOneKey(_attachments) || isObject(versions) === false || hasDiffOneKey(versions); const isInvalidBodyFormat = isObject(_attachments) === false || hasDiffOneKey(_attachments) ||
isObject(versions) === false || hasDiffOneKey(versions);
if (isInvalidBodyFormat) { if (isInvalidBodyFormat) {
// npm is doing something strange again // npm is doing something strange again
@ -218,20 +218,21 @@ export function publishPackage(storage: IStorageHandler, config: Config, auth: I
const metadata = validateMetadata(req.body, packageName); const metadata = validateMetadata(req.body, packageName);
// treating deprecation as updating a package // treating deprecation as updating a package
if (req.params._rev || isRelatedToDeprecation(req.body)) { if (req.params._rev || isRelatedToDeprecation(req.body)) {
debug('updating a new version for %o', packageName); logger.debug({packageName} , `updating a new version for @{packageName}`);
// we check unpublish permissions, an update is basically remove versions // we check unpublish permissions, an update is basically remove versions
const remote = req.remote_user; const remote = req.remote_user;
auth.allow_unpublish({packageName}, remote, (error) => { auth.allow_unpublish({packageName}, remote, (error) => {
if (error) { if (error) {
logger.error({ packageName }, `not allowed to unpublish a version for @{packageName}`); logger.debug({packageName} , `not allowed to unpublish a version for @{packageName}`);
return next(error); return next(error);
} }
storage.changePackage(packageName, metadata, req.params.revision, function(error) { storage.changePackage(packageName, metadata, req.params.revision, function(error) {
afterChange(error, API_MESSAGE.PKG_CHANGED, metadata); afterChange(error, API_MESSAGE.PKG_CHANGED, metadata);
}); });
}); });
} else { } else {
debug('adding a new version for %o', packageName); logger.debug({packageName} , `adding a new version for @{packageName}`);
storage.addPackage(packageName, metadata, function(error) { storage.addPackage(packageName, metadata, function(error) {
afterChange(error, API_MESSAGE.PKG_CREATED, metadata); afterChange(error, API_MESSAGE.PKG_CREATED, metadata);
}); });
@ -249,7 +250,8 @@ export function publishPackage(storage: IStorageHandler, config: Config, auth: I
export function unPublishPackage(storage: IStorageHandler) { export function unPublishPackage(storage: IStorageHandler) {
return function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void { return function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {
const packageName = req.params.package; const packageName = req.params.package;
debug('unpublishing %o', packageName);
logger.debug({packageName} , `unpublishing @{packageName}`);
storage.removePackage(packageName, function(err) { storage.removePackage(packageName, function(err) {
if (err) { if (err) {
return next(err); return next(err);
@ -267,13 +269,15 @@ export function removeTarball(storage: IStorageHandler) {
return function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void { return function(req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {
const packageName = req.params.package; const packageName = req.params.package;
const {filename, revision} = req.params; const {filename, revision} = req.params;
debug('removing a tarball for %o-%o-%o', packageName, filename, revision);
logger.debug({packageName, filename, revision} , `removing a tarball for @{packageName}-@{tarballName}-@{revision}`);
storage.removeTarball(packageName, filename, revision, function(err) { storage.removeTarball(packageName, filename, revision, function(err) {
if (err) { if (err) {
return next(err); return next(err);
} }
res.status(HTTP_STATUS.CREATED); res.status(HTTP_STATUS.CREATED);
debug('success remove tarball for %o-%o-%o', packageName, filename, revision);
logger.debug({packageName, filename, revision} , `success remove tarball for @{packageName}-@{tarballName}-@{revision}`);
return next({ ok: API_MESSAGE.TARBALL_REMOVED }); return next({ ok: API_MESSAGE.TARBALL_REMOVED });
}); });
}; };
@ -322,7 +326,7 @@ export function uploadPackageTarball(storage: IStorageHandler) {
}); });
stream.on('error', function(err) { stream.on('error', function(err) {
return res.locals.report_error(err); return res.report_error(err);
}); });
stream.on('success', function() { stream.on('success', function() {

View file

@ -1,8 +1,4 @@
import { HEADERS } from '../../../lib/constants'; import { HEADERS } from '@verdaccio/dev-commons';
/**
* @prettier
*/
export default function(route, auth, storage): void { export default function(route, auth, storage): void {
// searching packages // searching packages
@ -13,7 +9,7 @@ export default function (route, auth, storage): void {
let firstPackage = true; let firstPackage = true;
res.status(200); res.status(200);
res.set(HEADERS.CONTENT_TYPE, HEADERS.JSON_CHARSET); res.set(HEADERS.CONTENT_TYPE, HEADERS.JSON_CHARSET)
/* /*
* Offical NPM registry (registry.npmjs.org) no longer return whole database, * Offical NPM registry (registry.npmjs.org) no longer return whole database,

View file

@ -1,13 +1,10 @@
// @flow import { USERS, HTTP_STATUS } from '@verdaccio/dev-commons';
import {Response} from 'express'; import {Response} from 'express';
import _ from 'lodash'; import _ from 'lodash';
import buildDebug from 'debug'; import { logger } from '@verdaccio/logger';
import { USERS, HTTP_STATUS } from '../../../lib/constants';
import { $RequestExtend, $NextFunctionVer, IStorageHandler } from '../../../../types'; import {$RequestExtend, $NextFunctionVer, IStorageHandler} from '@verdaccio/dev-types';
import { logger } from '../../../lib/logger';
const debug = buildDebug('verdaccio:star');
export default function(storage: IStorageHandler): (req: $RequestExtend, res: Response, next: $NextFunctionVer) => void { export default function(storage: IStorageHandler): (req: $RequestExtend, res: Response, next: $NextFunctionVer) => void {
const validateInputs = (newUsers, localUsers, username, isStar): boolean => { const validateInputs = (newUsers, localUsers, username, isStar): boolean => {
const isExistlocalUsers = _.isNil(localUsers[username]) === false; const isExistlocalUsers = _.isNil(localUsers[username]) === false;
@ -23,7 +20,7 @@ export default function (storage: IStorageHandler): (req: $RequestExtend, res: R
return (req: $RequestExtend, res: Response, next: $NextFunctionVer): void => { return (req: $RequestExtend, res: Response, next: $NextFunctionVer): void => {
const name = req.params.package; const name = req.params.package;
debug('starring a package for %o', name); logger.debug({name}, 'starring a package for @{name}');
const afterChangePackage = function(err?: Error) { const afterChangePackage = function(err?: Error) {
if (err) { if (err) {
return next(err); return next(err);
@ -49,21 +46,15 @@ export default function (storage: IStorageHandler): (req: $RequestExtend, res: R
if (_.isNil(localStarUsers) === false && validateInputs(newStarUser, localStarUsers, remoteUsername, isStar)) { if (_.isNil(localStarUsers) === false && validateInputs(newStarUser, localStarUsers, remoteUsername, isStar)) {
return afterChangePackage(); return afterChangePackage();
} }
const users = isStar const users = isStar ? {
? {
...localStarUsers, ...localStarUsers,
[remoteUsername]: true, [remoteUsername]: true,
} } : _.reduce(localStarUsers, (users, value, key) => {
: _.reduce(
localStarUsers,
(users, value, key) => {
if (key !== remoteUsername) { if (key !== remoteUsername) {
users[key] = value; users[key] = value;
} }
return users; return users;
}, }, {});
{}
);
storage.changePackage(name, { ...info, users}, req.body._rev, function(err) { storage.changePackage(name, { ...info, users}, req.body._rev, function(err) {
afterChangePackage(err); afterChangePackage(err);
}); });

32
packages/api/src/stars.ts Normal file
View file

@ -0,0 +1,32 @@
import _ from 'lodash';
import { Response, Router } from 'express';
import { USERS, HTTP_STATUS } from '@verdaccio/dev-commons';
import { Package } from '@verdaccio/types';
import { $RequestExtend, $NextFunctionVer, IStorageHandler } from '@verdaccio/dev-types';
type Packages = Package[];
export default function(route: Router, storage: IStorageHandler): void {
route.get('/-/_view/starredByUser', (req: $RequestExtend, res: Response, next: $NextFunctionVer): void => {
const remoteUsername = req.remote_user.name;
storage.getLocalDatabase((err, localPackages: Packages) => {
if (err) {
return next(err);
}
const filteredPackages: Packages = localPackages.filter((localPackage: Package) =>
_.keys(localPackage[USERS]).includes(remoteUsername)
);
res.status(HTTP_STATUS.OK);
next({
rows: filteredPackages.map((filteredPackage: Package) => ({
value: filteredPackage.name,
})),
});
});
});
}

View file

@ -1,14 +1,13 @@
import _ from 'lodash'; import _ from 'lodash';
import Cookies from 'cookies'; import Cookies from 'cookies';
import { Response, Router } from 'express';
import { createRemoteUser, createSessionToken, getApiToken, getAuthenticatedMessage, validatePassword, ErrorCode } from '@verdaccio/utils';
import { logger } from '@verdaccio/logger';
import { Config, RemoteUser } from '@verdaccio/types'; import { Config, RemoteUser } from '@verdaccio/types';
import { Response, Router } from 'express'; import { $RequestExtend, $ResponseExtend, $NextFunctionVer, IAuth } from '@verdaccio/dev-types';
import { ErrorCode } from '../../../lib/utils'; import { API_ERROR, API_MESSAGE, HTTP_STATUS } from '@verdaccio/dev-commons';
import { API_ERROR, API_MESSAGE, HTTP_STATUS } from '../../../lib/constants';
import { createRemoteUser, createSessionToken, getApiToken, getAuthenticatedMessage, validatePassword } from '../../../lib/auth-utils';
import { logger } from '../../../lib/logger';
import { $RequestExtend, $ResponseExtend, $NextFunctionVer, IAuth } from '../../../../types';
export default function(route: Router, auth: IAuth, config: Config): void { export default function(route: Router, auth: IAuth, config: Config): void {
route.get('/-/user/:org_couchdb_user', function(req: $RequestExtend, res: Response, next: $NextFunctionVer): void { route.get('/-/user/:org_couchdb_user', function(req: $RequestExtend, res: Response, next: $NextFunctionVer): void {
@ -25,7 +24,7 @@ export default function (route: Router, auth: IAuth, config: Config): void {
if (_.isNil(remoteName) === false && _.isNil(name) === false && remoteName === name) { if (_.isNil(remoteName) === false && _.isNil(name) === false && remoteName === name) {
auth.authenticate(name, password, async function callbackAuthenticate(err, user): Promise<void> { auth.authenticate(name, password, async function callbackAuthenticate(err, user): Promise<void> {
if (err) { if (err) {
logger.error({ name, err }, 'authenticating for user @{username} failed. Error: @{err.message}'); logger.trace({ name, err }, 'authenticating for user @{username} failed. Error: @{err.message}');
return next(ErrorCode.getCode(HTTP_STATUS.UNAUTHORIZED, API_ERROR.BAD_USERNAME_PASSWORD)); return next(ErrorCode.getCode(HTTP_STATUS.UNAUTHORIZED, API_ERROR.BAD_USERNAME_PASSWORD));
} }

13
packages/api/src/utils.ts Normal file
View file

@ -0,0 +1,13 @@
import {Package} from '@verdaccio/types';
import _ from 'lodash';
/**
* Check whether the package metadta has enough data to be published
* @param pkg metadata
*/
export function isPublishablePackage(pkg: Package): boolean {
const keys: string[] = Object.keys(pkg);
return _.includes(keys, 'versions');
}

View file

@ -0,0 +1,83 @@
import _ from 'lodash';
import { Response, Router } from 'express';
import { API_ERROR, APP_ERROR, HTTP_STATUS, SUPPORT_ERRORS } from '@verdaccio/dev-commons';
import { ErrorCode, validatePassword } from '@verdaccio/utils';
import { $NextFunctionVer, $RequestExtend, IAuth } from '@verdaccio/dev-types';
export interface Profile {
tfa: boolean;
name: string;
email: string;
email_verified: boolean;
created: string;
updated: string;
cidr_whitelist: string[] | null;
fullname: string;
}
export default function(route: Router, auth: IAuth): void {
function buildProfile(name: string): Profile {
return {
tfa: false,
name,
email: '',
email_verified: false,
created: '',
updated: '',
cidr_whitelist: null,
fullname: '',
};
}
route.get('/-/npm/v1/user', function(req: $RequestExtend, res: Response, next: $NextFunctionVer): void {
if (_.isNil(req.remote_user.name) === false) {
return next(buildProfile(req.remote_user.name));
}
res.status(HTTP_STATUS.UNAUTHORIZED);
return next({
message: API_ERROR.MUST_BE_LOGGED,
});
});
route.post('/-/npm/v1/user', function(req: $RequestExtend, res: Response, next: $NextFunctionVer): void {
if (_.isNil(req.remote_user.name)) {
res.status(HTTP_STATUS.UNAUTHORIZED);
return next({
message: API_ERROR.MUST_BE_LOGGED,
});
}
const { password, tfa } = req.body;
const { name } = req.remote_user;
if (_.isNil(password) === false) {
if (validatePassword(password.new) === false) {
/* eslint new-cap:off */
return next(ErrorCode.getCode(HTTP_STATUS.UNAUTHORIZED, API_ERROR.PASSWORD_SHORT()));
/* eslint new-cap:off */
}
auth.changePassword(
name,
password.old,
password.new,
(err, isUpdated): $NextFunctionVer => {
if (_.isNull(err) === false) {
return next(ErrorCode.getCode(err.status, err.message) || ErrorCode.getConflict(err.message));
}
if (isUpdated) {
return next(buildProfile(req.remote_user.name));
}
return next(ErrorCode.getInternalError(API_ERROR.INTERNAL_SERVER_ERROR));
}
);
} else if (_.isNil(tfa) === false) {
return next(ErrorCode.getCode(HTTP_STATUS.SERVICE_UNAVAILABLE, SUPPORT_ERRORS.TFA_DISABLED));
} else {
return next(ErrorCode.getCode(HTTP_STATUS.INTERNAL_ERROR, APP_ERROR.PROFILE_ERROR));
}
});
}

View file

@ -0,0 +1,121 @@
import _ from 'lodash';
import { HTTP_STATUS, SUPPORT_ERRORS } from '@verdaccio/dev-commons';
import {ErrorCode, stringToMD5, mask, getApiToken } from '@verdaccio/utils';
import { logger } from '@verdaccio/logger';
import { Response, Router } from 'express';
import {$NextFunctionVer, $RequestExtend, IAuth, IStorageHandler} from '../../../types';
import { Config, RemoteUser, Token } from '@verdaccio/types';
export type NormalizeToken = Token & {
created: string;
};
function normalizeToken(token: Token): NormalizeToken {
return {
...token,
created: new Date(token.created).toISOString(),
};
};
// https://github.com/npm/npm-profile/blob/latest/lib/index.js
export default function(route: Router, auth: IAuth, storage: IStorageHandler, config: Config): void {
route.get('/-/npm/v1/tokens', async function(req: $RequestExtend, res: Response, next: $NextFunctionVer) {
const { name } = req.remote_user;
if (_.isNil(name) === false) {
try {
const tokens = await storage.readTokens({user: name});
const totalTokens = tokens.length;
logger.debug({totalTokens}, 'token list retrieved: @{totalTokens}');
res.status(HTTP_STATUS.OK);
return next({
objects: tokens.map(normalizeToken),
urls: {
next: '', // TODO: pagination?
},
});
} catch (error) {
logger.error({ error: error.msg }, 'token list has failed: @{error}');
return next(ErrorCode.getCode(HTTP_STATUS.INTERNAL_ERROR, error.message));
}
}
return next(ErrorCode.getUnauthorized());
});
route.post('/-/npm/v1/tokens', function(req: $RequestExtend, res: Response, next: $NextFunctionVer) {
const { password, readonly, cidr_whitelist } = req.body;
const { name } = req.remote_user;
if (!_.isBoolean(readonly) || !_.isArray(cidr_whitelist)) {
return next(ErrorCode.getCode(HTTP_STATUS.BAD_DATA, SUPPORT_ERRORS.PARAMETERS_NOT_VALID));
}
auth.authenticate(name, password, async (err, user: RemoteUser) => {
if (err) {
const errorCode = err.message ? HTTP_STATUS.UNAUTHORIZED : HTTP_STATUS.INTERNAL_ERROR;
return next(ErrorCode.getCode(errorCode, err.message));
}
req.remote_user = user;
if (!_.isFunction(storage.saveToken)) {
return next(ErrorCode.getCode(HTTP_STATUS.NOT_IMPLEMENTED, SUPPORT_ERRORS.STORAGE_NOT_IMPLEMENT));
}
try {
const token = await getApiToken(auth, config, user, password);
const key = stringToMD5(token);
// TODO: use a utility here
const maskedToken = mask(token, 5);
const created = new Date().getTime();
/**
* cidr_whitelist: is not being used, we pass it through
* token: we do not store the real token (it is generated once and retrieved to the user), just a mask of it.
*/
const saveToken: Token = {
user: name,
token: maskedToken,
key,
cidr: cidr_whitelist,
readonly,
created,
};
await storage.saveToken(saveToken);
logger.debug({ key, name }, 'token @{key} was created for user @{name}');
return next(normalizeToken({
token,
user: name,
key: saveToken.key,
cidr: cidr_whitelist,
readonly,
created: saveToken.created,
}));
} catch (error) {
logger.error({ error: error.msg }, 'token creation has failed: @{error}');
return next(ErrorCode.getCode(HTTP_STATUS.INTERNAL_ERROR, error.message));
}
});
});
route.delete('/-/npm/v1/tokens/token/:tokenKey', async (req: $RequestExtend, res: Response, next: $NextFunctionVer) => {
const { params: { tokenKey }} = req;
const { name } = req.remote_user;
if (_.isNil(name) === false) {
logger.debug({name}, '@{name} has requested remove a token');
try {
await storage.deleteToken(name, tokenKey);
logger.info({ tokenKey, name }, 'token id @{tokenKey} was revoked for user @{name}');
return next({});
} catch(error) {
logger.error({ error: error.msg }, 'token creation has failed: @{error}');
return next(ErrorCode.getCode(HTTP_STATUS.INTERNAL_ERROR, error.message));
}
}
return next(ErrorCode.getUnauthorized());
});
}

View file

@ -1,5 +1,5 @@
import { Response, Router } from 'express'; import { Response, Router } from 'express';
import { $RequestExtend, $NextFunctionVer } from '../../../../types'; import { $RequestExtend, $NextFunctionVer } from '@verdaccio/dev-types';
export default function (route: Router): void { export default function (route: Router): void {
route.get('/whoami', (req: $RequestExtend, res: Response, next: $NextFunctionVer): void => { route.get('/whoami', (req: $RequestExtend, res: Response, next: $NextFunctionVer): void => {

View file

@ -1,15 +1,11 @@
import { import { addVersion, uploadPackageTarball, removeTarball, unPublishPackage, publishPackage } from '../src/publish';
addVersion, import { HTTP_STATUS, API_ERROR } from '@verdaccio/dev-commons';
uploadPackageTarball,
removeTarball,
unPublishPackage,
publishPackage
} from '../../../../src/api/endpoint/api/publish';
import { HTTP_STATUS, API_ERROR } from '../../../../src/lib/constants';
const REVISION_MOCK = '15-e53a77096b0ee33e'; const REVISION_MOCK = '15-e53a77096b0ee33e';
require('../../../../src/lib/logger').setup([{ type: 'stdout', format: 'pretty', level: 'info' }]); require('@verdaccio/logger').setup([
{ type: 'stdout', format: 'pretty', level: 'info' }
]);
describe('Publish endpoints - add a tag', () => { describe('Publish endpoints - add a tag', () => {
let req; let req;
@ -21,18 +17,18 @@ describe('Publish endpoints - add a tag', () => {
params: { params: {
version: '1.0.0', version: '1.0.0',
tag: 'tag', tag: 'tag',
package: 'verdaccio' package: 'verdaccio',
}, },
body: '' body: '',
}; };
res = { res = {
status: jest.fn() status: jest.fn(),
}; };
next = jest.fn(); next = jest.fn();
}); });
test('should add a version', (done) => { test('should add a version', done => {
const storage = { const storage = {
addVersion: (packageName, version, body, tag, cb) => { addVersion: (packageName, version, body, tag, cb) => {
expect(packageName).toEqual(req.params.package); expect(packageName).toEqual(req.params.package);
@ -41,7 +37,7 @@ describe('Publish endpoints - add a tag', () => {
expect(tag).toEqual(req.params.tag); expect(tag).toEqual(req.params.tag);
cb(); cb();
done(); done();
} },
}; };
// @ts-ignore // @ts-ignore
@ -51,15 +47,15 @@ describe('Publish endpoints - add a tag', () => {
expect(next).toHaveBeenLastCalledWith({ ok: 'package published' }); expect(next).toHaveBeenLastCalledWith({ ok: 'package published' });
}); });
test('when failed to add a version', (done) => { test('when failed to add a version', done => {
const storage = { const storage = {
addVersion: (packageName, version, body, tag, cb) => { addVersion: (packageName, version, body, tag, cb) => {
const error = { const error = {
message: 'failure' message: 'failure',
}; };
cb(error); cb(error);
done(); done();
} },
}; };
// @ts-ignore // @ts-ignore
@ -81,10 +77,10 @@ describe('Publish endpoints - upload package tarball', () => {
req = { req = {
params: { params: {
filename: 'verdaccio.gzip', filename: 'verdaccio.gzip',
package: 'verdaccio' package: 'verdaccio',
}, },
pipe: jest.fn(), pipe: jest.fn(),
on: jest.fn() on: jest.fn(),
}; };
res = { status: jest.fn(), report_error: jest.fn() }; res = { status: jest.fn(), report_error: jest.fn() };
next = jest.fn(); next = jest.fn();
@ -94,14 +90,14 @@ describe('Publish endpoints - upload package tarball', () => {
const stream = { const stream = {
done: jest.fn(), done: jest.fn(),
abort: jest.fn(), abort: jest.fn(),
on: jest.fn(() => (status, cb) => cb()) on: jest.fn(() => (status, cb) => cb()),
}; };
const storage = { const storage = {
addTarball(packageName, filename) { addTarball(packageName, filename) {
expect(packageName).toEqual(req.params.package); expect(packageName).toEqual(req.params.package);
expect(filename).toEqual(req.params.filename); expect(filename).toEqual(req.params.filename);
return stream; return stream;
} },
}; };
// @ts-ignore // @ts-ignore
@ -124,14 +120,14 @@ describe('Publish endpoints - delete tarball', () => {
params: { params: {
filename: 'verdaccio.gzip', filename: 'verdaccio.gzip',
package: 'verdaccio', package: 'verdaccio',
revision: REVISION_MOCK revision: REVISION_MOCK,
} },
}; };
res = { status: jest.fn() }; res = { status: jest.fn() };
next = jest.fn(); next = jest.fn();
}); });
test('should delete tarball successfully', (done) => { test('should delete tarball successfully', done => {
const storage = { const storage = {
removeTarball(packageName, filename, revision, cb) { removeTarball(packageName, filename, revision, cb) {
expect(packageName).toEqual(req.params.package); expect(packageName).toEqual(req.params.package);
@ -139,7 +135,7 @@ describe('Publish endpoints - delete tarball', () => {
expect(revision).toEqual(req.params.revision); expect(revision).toEqual(req.params.revision);
cb(); cb();
done(); done();
} },
}; };
// @ts-ignore // @ts-ignore
@ -148,15 +144,15 @@ describe('Publish endpoints - delete tarball', () => {
expect(next).toHaveBeenCalledWith({ ok: 'tarball removed' }); expect(next).toHaveBeenCalledWith({ ok: 'tarball removed' });
}); });
test('failed while deleting the tarball', (done) => { test('failed while deleting the tarball', done => {
const error = { const error = {
message: 'deletion failed' message: 'deletion failed',
}; };
const storage = { const storage = {
removeTarball(packageName, filename, revision, cb) { removeTarball(packageName, filename, revision, cb) {
cb(error); cb(error);
done(); done();
} },
}; };
// @ts-ignore // @ts-ignore
@ -176,20 +172,20 @@ describe('Publish endpoints - un-publish package', () => {
beforeEach(() => { beforeEach(() => {
req = { req = {
params: { params: {
package: 'verdaccio' package: 'verdaccio',
} },
}; };
res = { status: jest.fn() }; res = { status: jest.fn() };
next = jest.fn(); next = jest.fn();
}); });
test('should un-publish package successfully', (done) => { test('should un-publish package successfully', done => {
const storage = { const storage = {
removePackage(packageName, cb) { removePackage(packageName, cb) {
expect(packageName).toEqual(req.params.package); expect(packageName).toEqual(req.params.package);
cb(); cb();
done(); done();
} },
}; };
// @ts-ignore // @ts-ignore
@ -198,15 +194,15 @@ describe('Publish endpoints - un-publish package', () => {
expect(next).toHaveBeenCalledWith({ ok: 'package removed' }); expect(next).toHaveBeenCalledWith({ ok: 'package removed' });
}); });
test('un-publish failed', (done) => { test('un-publish failed', done => {
const error = { const error = {
message: 'un-publish failed' message: 'un-publish failed',
}; };
const storage = { const storage = {
removePackage(packageName, cb) { removePackage(packageName, cb) {
cb(error); cb(error);
done(); done();
} },
}; };
// @ts-ignore // @ts-ignore
@ -226,11 +222,11 @@ describe('Publish endpoints - publish package', () => {
beforeEach(() => { beforeEach(() => {
req = { req = {
body: { body: {
name: 'verdaccio' name: 'verdaccio',
}, },
params: { params: {
package: 'verdaccio' package: 'verdaccio',
} },
}; };
res = { status: jest.fn() }; res = { status: jest.fn() };
next = jest.fn(); next = jest.fn();
@ -238,7 +234,7 @@ describe('Publish endpoints - publish package', () => {
test('should change the existing package', () => { test('should change the existing package', () => {
const storage = { const storage = {
changePackage: jest.fn() changePackage: jest.fn(),
}; };
req.params._rev = REVISION_MOCK; req.params._rev = REVISION_MOCK;
@ -250,7 +246,7 @@ describe('Publish endpoints - publish package', () => {
test('should publish a new a new package', () => { test('should publish a new a new package', () => {
const storage = { const storage = {
addPackage: jest.fn() addPackage: jest.fn(),
}; };
// @ts-ignore // @ts-ignore
@ -262,7 +258,7 @@ describe('Publish endpoints - publish package', () => {
const storage = { const storage = {
addPackage() { addPackage() {
throw new Error(); throw new Error();
} },
}; };
// @ts-ignore // @ts-ignore
@ -276,23 +272,23 @@ describe('Publish endpoints - publish package', () => {
changePackage: jest.fn(), changePackage: jest.fn(),
getPackage({ callback }) { getPackage({ callback }) {
callback(null, { callback(null, {
users: {} users: {},
}); });
} },
}; };
req = { req = {
params: { params: {
package: 'verdaccio' package: 'verdaccio',
}, },
body: { body: {
_rev: REVISION_MOCK, _rev: REVISION_MOCK,
users: { users: {
verdaccio: true verdaccio: true,
} },
}, },
remote_user: { remote_user: {
name: 'verdaccio' name: 'verdaccio',
} },
}; };
// @ts-ignore // @ts-ignore

View file

@ -12,9 +12,8 @@ import fs from 'fs';
app.param('token', validate_name); app.param('token', validate_name);
*/ */
describe('api endpoint app.param()', () => { describe('api endpoint app.param()', () => {
const file = '../endpoint/index.ts';
let m; let m;
const requirePath = path.normalize(path.join(__dirname + '/../../../../src/api/web/', file)); const requirePath = path.normalize(path.join(__dirname, '../src/index.ts'));
const source = fs.readFileSync(requirePath, 'utf8'); const source = fs.readFileSync(requirePath, 'utf8');
const very_scary_regexp = /\n\s*app\.(\w+)\s*\(\s*(("[^"]*")|('[^']*'))\s*,/g; const very_scary_regexp = /\n\s*app\.(\w+)\s*\(\s*(("[^"]*")|('[^']*'))\s*,/g;
const appParams = {}; const appParams = {};

View file

@ -0,0 +1,9 @@
{
"extends": "../../tsconfig",
"compilerOptions": {
"rootDir": "./src",
"outDir": "./build"
},
"include": ["src/**/*", "types/*.d.ts"],
"exclude": ["src/**/*.test.ts"]
}

3
packages/auth/.babelrc Normal file
View file

@ -0,0 +1,3 @@
{
"presets": [["@verdaccio"]]
}

View file

@ -0,0 +1,9 @@
module.exports = {
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx'],
transform: {
'^.+\\.(js|jsx|ts|tsx)$': 'babel-jest',
},
verbose: true,
collectCoverage: true,
coveragePathIgnorePatterns: ['node_modules', 'fixtures'],
};

View file

@ -0,0 +1,39 @@
{
"name": "@verdaccio/auth",
"version": "5.0.0-alpha.0",
"description": "logger",
"main": "./build/index.js",
"types": "build/index.d.ts",
"author": {
"name": "Juan Picado",
"email": "juanpicado19@gmail.com"
},
"repository": {
"type": "git",
"url": "git://github.com/verdaccio/verdaccio"
},
"homepage": "https://verdaccio.org",
"scripts": {
"clean": "rimraf ./build",
"test": "cross-env NODE_ENV=test BABEL_ENV=test jest",
"type-check": "tsc --noEmit",
"build:types": "tsc --emitDeclarationOnly --declaration true",
"build:js": "cross-env BABEL_ENV=registry babel src/ --out-dir build/ --copy-files --extensions \".ts,.tsx\" --source-maps",
"build": "npm run build:js && npm run build:types"
},
"license": "MIT",
"dependencies": {
"@verdaccio/commons-api": "^9.0.0",
"@verdaccio/dev-commons": "5.0.0-alpha.0",
"@verdaccio/loaders": "5.0.0-alpha.0",
"@verdaccio/utils": "5.0.0-alpha.0",
"bunyan": "1.8.12",
"express": "^4.17.1",
"lodash": "^4.17.15"
},
"devDependencies": {
"@types/bunyan": "1.8.6",
"@verdaccio/dev-types": "5.0.0-alpha.0"
},
"gitHead": "7c246ede52ff717707fcae66dd63fc4abd536982"
}

View file

@ -1,13 +1,10 @@
import _ from 'lodash'; import _ from 'lodash';
import { VerdaccioError } from '@verdaccio/commons-api';
import buildDebug from 'debug';
import { Config, Logger, Callback, IPluginAuth, RemoteUser, JWTSignOptions, Security, AuthPluginPackage, AllowAccess, PackageAccess } from '@verdaccio/types';
import { NextFunction } from 'express'; import { NextFunction } from 'express';
import loadPlugin from '../lib/plugin-loader';
import { $RequestExtend, $ResponseExtend, IAuth, AESPayload } from '../../types'; import { VerdaccioError } from '@verdaccio/commons-api';
import { API_ERROR, SUPPORT_ERRORS, TOKEN_BASIC, TOKEN_BEARER } from './constants'; import {API_ERROR, SUPPORT_ERRORS, TOKEN_BASIC, TOKEN_BEARER} from '@verdaccio/dev-commons';
import { aesEncrypt, signPayload } from './crypto-utils'; import { loadPlugin } from '@verdaccio/loaders';
import { logger } from './logger'; import { aesEncrypt, signPayload } from '@verdaccio/utils';
import { import {
getDefaultPlugins, getDefaultPlugins,
getMiddlewareCredentials, getMiddlewareCredentials,
@ -16,24 +13,29 @@ import {
isAuthHeaderValid, isAuthHeaderValid,
getSecurity, getSecurity,
isAESLegacy, isAESLegacy,
convertPayloadToBase64,
ErrorCode,
parseAuthTokenHeader, parseAuthTokenHeader,
parseBasicPayload, parseBasicPayload,
createRemoteUser, createRemoteUser,
} from './auth-utils'; } from '@verdaccio/utils';
import { convertPayloadToBase64, ErrorCode } from './utils';
import { getMatchedPackagesSpec } from './config-utils';
const debug = buildDebug('verdaccio:auth'); import { getMatchedPackagesSpec } from '@verdaccio/utils';
import { Config, Logger, Callback, IPluginAuth, RemoteUser, JWTSignOptions, Security, AuthPluginPackage, AllowAccess, PackageAccess } from '@verdaccio/types';
import { $RequestExtend, $ResponseExtend, IAuth, AESPayload } from '@verdaccio/dev-types';
/* eslint-disable @typescript-eslint/no-var-requires */
const LoggerApi = require('@verdaccio/logger');
class Auth implements IAuth { class Auth implements IAuth {
public config: Config; public config: Config;
public logger: Logger; public logger: Logger;
public secret: string; // pragma: allowlist secret public secret: string;
public plugins: IPluginAuth<Config>[]; public plugins: IPluginAuth<Config>[];
public constructor(config: Config) { public constructor(config: Config) {
this.config = config; this.config = config;
this.logger = logger; this.logger = LoggerApi.logger.child({ sub: 'auth' });
this.secret = config.secret; this.secret = config.secret;
this.plugins = this._loadPlugin(config); this.plugins = this._loadPlugin(config);
this._applyDefaultPlugins(); this._applyDefaultPlugins();
@ -45,24 +47,25 @@ class Auth implements IAuth {
logger: this.logger, logger: this.logger,
}; };
return loadPlugin<IPluginAuth<Config>>(config, config.auth, pluginOptions, (plugin: IPluginAuth<Config>): boolean => { return loadPlugin<IPluginAuth<Config>>(
config,
config.auth,
pluginOptions,
(plugin: IPluginAuth<Config>): boolean => {
const { authenticate, allow_access, allow_publish } = plugin; const { authenticate, allow_access, allow_publish } = plugin;
// @ts-ignore // @ts-ignore
return authenticate || allow_access || allow_publish; return authenticate || allow_access || allow_publish;
}); }
);
} }
private _applyDefaultPlugins(): void { private _applyDefaultPlugins(): void {
this.plugins.push(getDefaultPlugins(this.logger)); this.plugins.push(getDefaultPlugins());
} }
public changePassword( public changePassword(username: string, password: string, newPassword: string, cb: Callback): void {
username: string, const validPlugins = _.filter(this.plugins, plugin => _.isFunction(plugin.changePassword));
password: string, // pragma: allowlist secret
newPassword: string, // pragma: allowlist secret
cb: Callback
): void {
const validPlugins = _.filter(this.plugins, (plugin) => _.isFunction(plugin.changePassword));
if (_.isEmpty(validPlugins)) { if (_.isEmpty(validPlugins)) {
return cb(ErrorCode.getInternalError(SUPPORT_ERRORS.PLUGIN_MISSING_INTERFACE)); return cb(ErrorCode.getInternalError(SUPPORT_ERRORS.PLUGIN_MISSING_INTERFACE));
@ -70,11 +73,15 @@ class Auth implements IAuth {
for (const plugin of validPlugins) { for (const plugin of validPlugins) {
if (_.isNil(plugin) || _.isFunction(plugin.changePassword) === false) { if (_.isNil(plugin) || _.isFunction(plugin.changePassword) === false) {
debug('auth plugin does not implement changePassword, trying next one'); this.logger.trace('auth plugin does not implement changePassword, trying next one');
continue; continue;
} else { } else {
debug('updating password for %o', username); this.logger.trace({username}, 'updating password for @{username}');
plugin.changePassword!(username, password, newPassword, (err, profile): void => { plugin.changePassword!(
username,
password,
newPassword,
(err, profile): void => {
if (err) { if (err) {
this.logger.error( this.logger.error(
{username, err}, {username, err},
@ -83,9 +90,11 @@ class Auth implements IAuth {
); );
return cb(err); return cb(err);
} }
this.logger.info({ username }, 'updated password for @{username} was successful');
this.logger.trace({username}, 'updated password for @{username} was successful');
return cb(null, profile); return cb(null, profile);
}); }
);
} }
} }
} }
@ -95,13 +104,15 @@ class Auth implements IAuth {
const self = this; const self = this;
(function next(): void { (function next(): void {
const plugin = plugins.shift() as IPluginAuth<Config>; const plugin = plugins.shift() as IPluginAuth<Config>;
if (_.isFunction(plugin.authenticate) === false) { if (_.isFunction(plugin.authenticate) === false) {
return next(); return next();
} }
debug('authenticating %o', username);
self.logger.trace({ username }, 'authenticating @{username}');
plugin.authenticate(username, password, function(err, groups): void { plugin.authenticate(username, password, function(err, groups): void {
if (err) { if (err) {
self.logger.error({ username, err }, 'authenticating for user @{username} failed. Error: @{err.message}'); self.logger.trace({ username, err }, 'authenticating for user @{username} failed. Error: @{err.message}');
return cb(err); return cb(err);
} }
@ -121,7 +132,8 @@ class Auth implements IAuth {
if (!isGroupValid) { if (!isGroupValid) {
throw new TypeError(API_ERROR.BAD_FORMAT_USER_GROUP); throw new TypeError(API_ERROR.BAD_FORMAT_USER_GROUP);
} }
debug('authentication for user %o was successfully. Groups: %o', username, groups);
self.logger.trace({ username, groups }, 'authentication for user @{username} was successfully. Groups: @{groups}');
return cb(err, createRemoteUser(username, groups)); return cb(err, createRemoteUser(username, groups));
} }
next(); next();
@ -132,7 +144,8 @@ class Auth implements IAuth {
public add_user(user: string, password: string, cb: Callback): void { public add_user(user: string, password: string, cb: Callback): void {
const self = this; const self = this;
const plugins = this.plugins.slice(0); const plugins = this.plugins.slice(0);
debug('add user %o', user); this.logger.trace({ user }, 'add user @{user}');
(function next(): void { (function next(): void {
const plugin = plugins.shift() as IPluginAuth<Config>; const plugin = plugins.shift() as IPluginAuth<Config>;
let method = 'adduser'; let method = 'adduser';
@ -147,11 +160,11 @@ class Auth implements IAuth {
// p.add_user() execution // p.add_user() execution
plugin[method](user, password, function(err, ok): void { plugin[method](user, password, function(err, ok): void {
if (err) { if (err) {
self.logger.error({ user, err: err.message }, 'the user @{user} could not being added. Error: @{err}'); self.logger.trace({ user, err: err.message }, 'the user @{user} could not being added. Error: @{err}');
return cb(err); return cb(err);
} }
if (ok) { if (ok) {
self.logger.info({ user }, 'the user @{user} has been added'); self.logger.trace({ user }, 'the user @{user} has been added');
return self.authenticate(user, password, cb); return self.authenticate(user, password, cb);
} }
next(); next();
@ -165,10 +178,10 @@ class Auth implements IAuth {
*/ */
public allow_access({ packageName, packageVersion }: AuthPluginPackage, user: RemoteUser, callback: Callback): void { public allow_access({ packageName, packageVersion }: AuthPluginPackage, user: RemoteUser, callback: Callback): void {
const plugins = this.plugins.slice(0); const plugins = this.plugins.slice(0);
const self = this;
const pkgAllowAcces: AllowAccess = { name: packageName, version: packageVersion }; const pkgAllowAcces: AllowAccess = { name: packageName, version: packageVersion };
const pkg = Object.assign({}, pkgAllowAcces, getMatchedPackagesSpec(packageName, this.config.packages)) as AllowAccess & PackageAccess; const pkg = Object.assign({}, pkgAllowAcces, getMatchedPackagesSpec(packageName, this.config.packages)) as AllowAccess & PackageAccess;
debug('allow access for %o', packageName); const self = this;
this.logger.trace({ packageName }, 'allow access for @{packageName}');
(function next(): void { (function next(): void {
const plugin: IPluginAuth<Config> = plugins.shift() as IPluginAuth<Config>; const plugin: IPluginAuth<Config> = plugins.shift() as IPluginAuth<Config>;
@ -179,12 +192,12 @@ class Auth implements IAuth {
plugin.allow_access!(user, pkg, function(err, ok: boolean): void { plugin.allow_access!(user, pkg, function(err, ok: boolean): void {
if (err) { if (err) {
self.logger.error({ packageName, err }, 'forbidden access for @{packageName}. Error: @{err.message}'); self.logger.trace({ packageName, err }, 'forbidden access for @{packageName}. Error: @{err.message}');
return callback(err); return callback(err);
} }
if (ok) { if (ok) {
self.logger.info({ packageName }, 'allowed access for @{packageName}'); self.logger.trace({ packageName }, 'allowed access for @{packageName}');
return callback(null, ok); return callback(null, ok);
} }
@ -195,30 +208,35 @@ class Auth implements IAuth {
public allow_unpublish({ packageName, packageVersion }: AuthPluginPackage, user: RemoteUser, callback: Callback): void { public allow_unpublish({ packageName, packageVersion }: AuthPluginPackage, user: RemoteUser, callback: Callback): void {
const pkg = Object.assign({ name: packageName, version: packageVersion }, getMatchedPackagesSpec(packageName, this.config.packages)); const pkg = Object.assign({ name: packageName, version: packageVersion }, getMatchedPackagesSpec(packageName, this.config.packages));
debug('allow unpublish for %o', packageName); this.logger.trace({ packageName }, 'allow unpublish for @{packageName}');
for (const plugin of this.plugins) { for (const plugin of this.plugins) {
if (_.isNil(plugin) || _.isFunction(plugin.allow_unpublish) === false) { if (_.isNil(plugin) || _.isFunction(plugin.allow_unpublish) === false) {
debug('allow unpublish for %o plugin does not implement allow_unpublish', packageName); this.logger.trace({ packageName }, 'allow unpublish for @{packageName} plugin does not implement allow_unpublish');
continue; continue;
} else { } else {
plugin.allow_unpublish!(user, pkg, (err, ok: boolean): void => { plugin.allow_unpublish!(
user,
pkg,
(err, ok: boolean): void => {
if (err) { if (err) {
this.logger.error({ packageName, user: user?.name }, '@{user} forbidden publish for @{packageName}, it will fallback on unpublish permissions'); this.logger.trace({ packageName }, 'forbidden publish for @{packageName}, it will fallback on unpublish permissions');
return callback(err); return callback(err);
} }
if (_.isNil(ok) === true) { if (_.isNil(ok) === true) {
debug('we bypass unpublish for %o, publish will handle the access', packageName); this.logger.trace({ packageName }, 'we bypass unpublish for @{packageName}, publish will handle the access');
// @ts-ignore // @ts-ignore
// eslint-disable-next-line // eslint-disable-next-line
return this.allow_publish(...arguments); return this.allow_publish(...arguments);
} }
if (ok) { if (ok) {
this.logger.info({ packageName, user: user?.name }, '@{user} allowed unpublish for @{packageName}'); this.logger.trace({ packageName }, 'allowed unpublish for @{packageName}');
return callback(null, ok); return callback(null, ok);
} }
}); }
);
} }
} }
} }
@ -230,29 +248,36 @@ class Auth implements IAuth {
const plugins = this.plugins.slice(0); const plugins = this.plugins.slice(0);
const self = this; const self = this;
const pkg = Object.assign({ name: packageName, version: packageVersion }, getMatchedPackagesSpec(packageName, this.config.packages)); const pkg = Object.assign({ name: packageName, version: packageVersion }, getMatchedPackagesSpec(packageName, this.config.packages));
debug('allow publish for %o init | plugins: %o', packageName, plugins); this.logger.trace({ packageName, plugins: this.plugins.length }, 'allow publish for @{packageName} init | plugins: @{plugins}');
(function next(): void { (function next(): void {
const plugin = plugins.shift(); const plugin = plugins.shift();
if (_.isNil(plugin) || _.isFunction(plugin.allow_publish) === false) { if (_.isNil(plugin) || _.isFunction(plugin.allow_publish) === false) {
debug('allow publish for %o plugin does not implement allow_publish', packageName); self.logger.trace({ packageName }, 'allow publish for @{packageName} plugin does not implement allow_publish');
return next(); return next();
} }
// @ts-ignore // @ts-ignore
plugin.allow_publish(user, pkg, (err: VerdaccioError, ok: boolean): void => { plugin.allow_publish(
user,
pkg,
// @ts-ignore
(err: VerdaccioError, ok: boolean): void => {
if (_.isNil(err) === false && _.isError(err)) { if (_.isNil(err) === false && _.isError(err)) {
self.logger.error({ packageName, user: user?.name }, '@{user} is forbidden publish for @{packageName}'); self.logger.trace({ packageName }, 'forbidden publish for @{packageName}');
return callback(err); return callback(err);
} }
if (ok) { if (ok) {
self.logger.info({ packageName, user: user?.name }, '@{user} is allowed publish for @{packageName}'); self.logger.trace({ packageName }, 'allowed publish for @{packageName}');
return callback(null, ok); return callback(null, ok);
} }
debug('allow publish skip validation for %o', packageName);
self.logger.trace({ packageName }, 'allow publish skip validation for @{packageName}');
next(); // cb(null, false) causes next plugin to roll next(); // cb(null, false) causes next plugin to roll
}); }
);
})(); })();
} }
@ -293,7 +318,7 @@ class Auth implements IAuth {
} }
if (!isAuthHeaderValid(authorization)) { if (!isAuthHeaderValid(authorization)) {
debug('api middleware auth heather is not valid'); this.logger.trace('api middleware auth heather is not valid');
return next(ErrorCode.getBadRequest(API_ERROR.BAD_AUTH_HEADER)); return next(ErrorCode.getBadRequest(API_ERROR.BAD_AUTH_HEADER));
} }
@ -301,10 +326,10 @@ class Auth implements IAuth {
const { secret } = this.config; const { secret } = this.config;
if (isAESLegacy(security)) { if (isAESLegacy(security)) {
debug('api middleware using legacy auth token'); this.logger.trace('api middleware using legacy auth token');
this._handleAESMiddleware(req, security, secret, authorization, next); this._handleAESMiddleware(req, security, secret, authorization, next);
} else { } else {
debug('api middleware using JWT auth token'); this.logger.trace('api middleware using JWT auth token');
this._handleJWTAPIMiddleware(req, security, secret, authorization, next); this._handleJWTAPIMiddleware(req, security, secret, authorization, next);
} }
}; };
@ -316,7 +341,10 @@ class Auth implements IAuth {
// this should happen when client tries to login with an existing user // this should happen when client tries to login with an existing user
const credentials = convertPayloadToBase64(token).toString(); const credentials = convertPayloadToBase64(token).toString();
const { user, password } = parseBasicPayload(credentials) as AESPayload; const { user, password } = parseBasicPayload(credentials) as AESPayload;
this.authenticate(user, password, (err, user): void => { this.authenticate(
user,
password,
(err, user): void => {
if (!err) { if (!err) {
req.remote_user = user; req.remote_user = user;
next(); next();
@ -324,7 +352,8 @@ class Auth implements IAuth {
req.remote_user = createAnonymousRemoteUser(); req.remote_user = createAnonymousRemoteUser();
next(err); next(err);
} }
}); }
);
} else { } else {
// jwt handler // jwt handler
const credentials: any = getMiddlewareCredentials(security, secret, authorization); const credentials: any = getMiddlewareCredentials(security, secret, authorization);
@ -343,7 +372,10 @@ class Auth implements IAuth {
const credentials: any = getMiddlewareCredentials(security, secret, authorization); const credentials: any = getMiddlewareCredentials(security, secret, authorization);
if (credentials) { if (credentials) {
const { user, password } = credentials; const { user, password } = credentials;
this.authenticate(user, password, (err, user): void => { this.authenticate(
user,
password,
(err, user): void => {
if (!err) { if (!err) {
req.remote_user = user; req.remote_user = user;
next(); next();
@ -351,7 +383,8 @@ class Auth implements IAuth {
req.remote_user = createAnonymousRemoteUser(); req.remote_user = createAnonymousRemoteUser();
next(err); next(err);
} }
}); }
);
} else { } else {
// we force npm client to ask again with basic authentication // we force npm client to ask again with basic authentication
return next(ErrorCode.getBadRequest(API_ERROR.BAD_AUTH_HEADER)); return next(ErrorCode.getBadRequest(API_ERROR.BAD_AUTH_HEADER));
@ -438,4 +471,4 @@ class Auth implements IAuth {
} }
} }
export default Auth; export { Auth };

View file

@ -0,0 +1 @@
export { Auth } from './auth'

View file

@ -1,27 +1,35 @@
import _ from 'lodash'; import _ from 'lodash';
import Auth from '../../../../src/lib/auth'; import { Auth } from '@verdaccio/auth';
import { CHARACTER_ENCODING, TOKEN_BEARER } from '../../../../src/lib/constants'; import {CHARACTER_ENCODING, TOKEN_BEARER} from '@verdaccio/dev-commons';
// $FlowFixMe
import configExample from '../../partials/config'; import { configExample } from '@verdaccio/mock';
import AppConfig from '../../../../src/lib/config'; import {Config as AppConfig } from '@verdaccio/config';
import { setup } from '../../../../src/lib/logger'; import {setup} from '@verdaccio/logger';
import { buildToken, convertPayloadToBase64, parseConfigFile } from '../../../../src/lib/utils';
import { import {
buildUserBuffer, buildUserBuffer,
getApiToken, getApiToken,
getAuthenticatedMessage, getAuthenticatedMessage,
getMiddlewareCredentials, getMiddlewareCredentials,
getSecurity getSecurity,
} from '../../../../src/lib/auth-utils'; aesDecrypt, verifyPayload,
import { aesDecrypt, verifyPayload } from '../../../../src/lib/crypto-utils'; buildToken, convertPayloadToBase64, parseConfigFile
import { parseConfigurationFile } from '../../__helper'; } from '@verdaccio/utils';
import { IAuth } from '../../../../types'; import { IAuth } from '@verdaccio/dev-types';
import {Config, Security, RemoteUser} from '@verdaccio/types'; import {Config, Security, RemoteUser} from '@verdaccio/types';
import path from "path";
setup([]); setup([]);
const parseConfigurationFile = (conf) => {
const { name, ext } = path.parse(conf);
const format = ext.startsWith('.') ? ext.substring(1) : 'yaml';
return path.join(__dirname, `./partials/config/${format}/security/${name}.${format}`);
};
describe('Auth utilities', () => { describe('Auth utilities', () => {
jest.setTimeout(20000); jest.setTimeout(20000);
@ -44,8 +52,7 @@ describe('Auth utilities', () => {
password: string, password: string,
secret = '12345', secret = '12345',
methodToSpy: string, methodToSpy: string,
methodNotBeenCalled: string methodNotBeenCalled: string): Promise<string> {
): Promise<string> {
const config: Config = getConfig(configFileName, secret); const config: Config = getConfig(configFileName, secret);
const auth: IAuth = new Auth(config); const auth: IAuth = new Auth(config);
// @ts-ignore // @ts-ignore
@ -74,9 +81,7 @@ describe('Auth utilities', () => {
}; };
const verifyAES = (token: string, user: string, password: string, secret: string) => { const verifyAES = (token: string, user: string, password: string, secret: string) => {
const payload = aesDecrypt(convertPayloadToBase64(token), secret).toString( const payload = aesDecrypt(convertPayloadToBase64(token), secret).toString(CHARACTER_ENCODING.UTF8);
CHARACTER_ENCODING.UTF8
);
const content = payload.split(':'); const content = payload.split(':');
expect(content[0]).toBe(user); expect(content[0]).toBe(user);
@ -85,98 +90,56 @@ describe('Auth utilities', () => {
describe('getApiToken test', () => { describe('getApiToken test', () => {
test('should sign token with aes and security missing', async () => { test('should sign token with aes and security missing', async () => {
const token = await signCredentials( const token = await signCredentials('security-missing',
'security-missing', 'test', 'test', '1234567', 'aesEncrypt', 'jwtEncrypt');
'test',
'test',
'1234567',
'aesEncrypt',
'jwtEncrypt'
);
verifyAES(token, 'test', 'test', '1234567'); verifyAES(token, 'test', 'test', '1234567');
expect(_.isString(token)).toBeTruthy(); expect(_.isString(token)).toBeTruthy();
}); });
test('should sign token with aes and security empty', async () => { test('should sign token with aes and security empty', async () => {
const token = await signCredentials( const token = await signCredentials('security-empty',
'security-empty', 'test', 'test', '123456', 'aesEncrypt', 'jwtEncrypt');
'test',
'test',
'123456',
'aesEncrypt',
'jwtEncrypt'
);
verifyAES(token, 'test', 'test', '123456'); verifyAES(token, 'test', 'test', '123456');
expect(_.isString(token)).toBeTruthy(); expect(_.isString(token)).toBeTruthy();
}); });
test('should sign token with aes', async () => { test('should sign token with aes', async () => {
const token = await signCredentials( const token = await signCredentials('security-basic',
'security-basic', 'test', 'test', '123456', 'aesEncrypt', 'jwtEncrypt');
'test',
'test',
'123456',
'aesEncrypt',
'jwtEncrypt'
);
verifyAES(token, 'test', 'test', '123456'); verifyAES(token, 'test', 'test', '123456');
expect(_.isString(token)).toBeTruthy(); expect(_.isString(token)).toBeTruthy();
}); });
test('should sign token with legacy and jwt disabled', async () => { test('should sign token with legacy and jwt disabled', async () => {
const token = await signCredentials( const token = await signCredentials('security-no-legacy',
'security-no-legacy', 'test', 'test', 'x8T#ZCx=2t', 'aesEncrypt', 'jwtEncrypt');
'test',
'test',
'x8T#ZCx=2t',
'aesEncrypt',
'jwtEncrypt'
);
expect(_.isString(token)).toBeTruthy(); expect(_.isString(token)).toBeTruthy();
verifyAES(token, 'test', 'test', 'x8T#ZCx=2t'); verifyAES(token, 'test', 'test', 'x8T#ZCx=2t');
}); });
test('should sign token with legacy enabled and jwt enabled', async () => { test('should sign token with legacy enabled and jwt enabled', async () => {
const token = await signCredentials( const token = await signCredentials('security-jwt-legacy-enabled',
'security-jwt-legacy-enabled', 'test', 'test', 'secret', 'jwtEncrypt', 'aesEncrypt');
'test',
'test',
'secret',
'jwtEncrypt',
'aesEncrypt'
);
verifyJWT(token, 'test', 'test', 'secret'); verifyJWT(token, 'test', 'test', 'secret');
expect(_.isString(token)).toBeTruthy(); expect(_.isString(token)).toBeTruthy();
}); });
test('should sign token with jwt enabled', async () => { test('should sign token with jwt enabled', async () => {
const token = await signCredentials( const token = await signCredentials('security-jwt',
'security-jwt', 'test', 'test', 'secret', 'jwtEncrypt', 'aesEncrypt');
'test',
'test',
'secret',
'jwtEncrypt',
'aesEncrypt'
);
expect(_.isString(token)).toBeTruthy(); expect(_.isString(token)).toBeTruthy();
verifyJWT(token, 'test', 'test', 'secret'); verifyJWT(token, 'test', 'test', 'secret');
}); });
test('should sign with jwt whether legacy is disabled', async () => { test('should sign with jwt whether legacy is disabled', async () => {
const token = await signCredentials( const token = await signCredentials('security-legacy-disabled',
'security-legacy-disabled', 'test', 'test', 'secret', 'jwtEncrypt', 'aesEncrypt');
'test',
'test',
'secret',
'jwtEncrypt',
'aesEncrypt'
);
expect(_.isString(token)).toBeTruthy(); expect(_.isString(token)).toBeTruthy();
verifyJWT(token, 'test', 'test', 'secret'); verifyJWT(token, 'test', 'test', 'secret');
@ -185,7 +148,7 @@ describe('Auth utilities', () => {
describe('getAuthenticatedMessage test', () => { describe('getAuthenticatedMessage test', () => {
test('should sign token with jwt enabled', () => { test('should sign token with jwt enabled', () => {
expect(getAuthenticatedMessage('test')).toBe("you are authenticated as 'test'"); expect(getAuthenticatedMessage('test')).toBe('you are authenticated as \'test\'');
}); });
}); });
@ -195,14 +158,8 @@ describe('Auth utilities', () => {
const secret = 'secret'; const secret = 'secret';
const user = 'test'; const user = 'test';
const pass = 'test'; const pass = 'test';
const token = await signCredentials( const token = await signCredentials('security-legacy',
'security-legacy', user, pass, secret, 'aesEncrypt', 'jwtEncrypt');
user,
pass,
secret,
'aesEncrypt',
'jwtEncrypt'
);
const config: Config = getConfig('security-legacy', secret); const config: Config = getConfig('security-legacy', secret);
const security: Security = getSecurity(config); const security: Security = getSecurity(config);
const credentials = getMiddlewareCredentials(security, secret, `Bearer ${token}`); const credentials = getMiddlewareCredentials(security, secret, `Bearer ${token}`);
@ -230,41 +187,21 @@ describe('Auth utilities', () => {
test.concurrent('should return empty credential wrong secret key', async () => { test.concurrent('should return empty credential wrong secret key', async () => {
const secret = 'secret'; const secret = 'secret';
const token = await signCredentials( const token = await signCredentials('security-legacy',
'security-legacy', 'test', 'test', secret, 'aesEncrypt', 'jwtEncrypt');
'test',
'test',
secret,
'aesEncrypt',
'jwtEncrypt'
);
const config: Config = getConfig('security-legacy', secret); const config: Config = getConfig('security-legacy', secret);
const security: Security = getSecurity(config); const security: Security = getSecurity(config);
const credentials = getMiddlewareCredentials( const credentials = getMiddlewareCredentials(security, 'BAD_SECRET', buildToken(TOKEN_BEARER, token));
security,
'BAD_SECRET',
buildToken(TOKEN_BEARER, token)
);
expect(credentials).not.toBeDefined(); expect(credentials).not.toBeDefined();
}); });
test.concurrent('should return empty credential wrong scheme', async () => { test.concurrent('should return empty credential wrong scheme', async () => {
const secret = 'secret'; const secret = 'secret';
const token = await signCredentials( const token = await signCredentials('security-legacy',
'security-legacy', 'test', 'test', secret, 'aesEncrypt', 'jwtEncrypt');
'test',
'test',
secret,
'aesEncrypt',
'jwtEncrypt'
);
const config: Config = getConfig('security-legacy', secret); const config: Config = getConfig('security-legacy', secret);
const security: Security = getSecurity(config); const security: Security = getSecurity(config);
const credentials = getMiddlewareCredentials( const credentials = getMiddlewareCredentials(security, secret, buildToken('BAD_SCHEME', token));
security,
secret,
buildToken('BAD_SCHEME', token)
);
expect(credentials).not.toBeDefined(); expect(credentials).not.toBeDefined();
}); });
@ -274,11 +211,7 @@ describe('Auth utilities', () => {
const auth: IAuth = new Auth(config); const auth: IAuth = new Auth(config);
const token = auth.aesEncrypt(Buffer.from(`corruptedBuffer`)).toString('base64'); const token = auth.aesEncrypt(Buffer.from(`corruptedBuffer`)).toString('base64');
const security: Security = getSecurity(config); const security: Security = getSecurity(config);
const credentials = getMiddlewareCredentials( const credentials = getMiddlewareCredentials(security, secret, buildToken(TOKEN_BEARER, token));
security,
secret,
buildToken(TOKEN_BEARER, token)
);
expect(credentials).not.toBeDefined(); expect(credentials).not.toBeDefined();
}); });
}); });
@ -287,11 +220,7 @@ describe('Auth utilities', () => {
test('should return anonymous whether token is corrupted', () => { test('should return anonymous whether token is corrupted', () => {
const config: Config = getConfig('security-jwt', '12345'); const config: Config = getConfig('security-jwt', '12345');
const security: Security = getSecurity(config); const security: Security = getSecurity(config);
const credentials = getMiddlewareCredentials( const credentials = getMiddlewareCredentials(security, '12345', buildToken(TOKEN_BEARER, 'fakeToken'));
security,
'12345',
buildToken(TOKEN_BEARER, 'fakeToken')
);
expect(credentials).toBeDefined(); expect(credentials).toBeDefined();
// @ts-ignore // @ts-ignore
@ -305,11 +234,7 @@ describe('Auth utilities', () => {
test('should return anonymous whether token and scheme are corrupted', () => { test('should return anonymous whether token and scheme are corrupted', () => {
const config: Config = getConfig('security-jwt', '12345'); const config: Config = getConfig('security-jwt', '12345');
const security: Security = getSecurity(config); const security: Security = getSecurity(config);
const credentials = getMiddlewareCredentials( const credentials = getMiddlewareCredentials(security, '12345', buildToken('FakeScheme', 'fakeToken'));
security,
'12345',
buildToken('FakeScheme', 'fakeToken')
);
expect(credentials).not.toBeDefined(); expect(credentials).not.toBeDefined();
}); });
@ -318,20 +243,10 @@ describe('Auth utilities', () => {
const secret = 'secret'; const secret = 'secret';
const user = 'test'; const user = 'test';
const config: Config = getConfig('security-jwt', secret); const config: Config = getConfig('security-jwt', secret);
const token = await signCredentials( const token = await signCredentials('security-jwt',
'security-jwt', user, 'secretTest', secret, 'jwtEncrypt', 'aesEncrypt');
user,
'secretTest',
secret,
'jwtEncrypt',
'aesEncrypt'
);
const security: Security = getSecurity(config); const security: Security = getSecurity(config);
const credentials = getMiddlewareCredentials( const credentials = getMiddlewareCredentials(security, secret, buildToken(TOKEN_BEARER, token));
security,
secret,
buildToken(TOKEN_BEARER, token)
);
expect(credentials).toBeDefined(); expect(credentials).toBeDefined();
// @ts-ignore // @ts-ignore
expect(credentials.name).toEqual(user); expect(credentials.name).toEqual(user);

View file

@ -0,0 +1,14 @@
import {aesDecrypt, aesEncrypt, convertPayloadToBase64} from "@verdaccio/utils";
describe('test crypto utils', () => {
describe('default encryption', () => {
test('decrypt payload flow', () => {
const payload = 'juan';
const token = aesEncrypt(Buffer.from(payload), '12345').toString('base64');
const data = aesDecrypt(convertPayloadToBase64(token), '12345').toString('utf8');
expect(payload).toEqual(data);
});
});
});

View file

@ -0,0 +1,9 @@
{
"extends": "../../tsconfig",
"compilerOptions": {
"rootDir": "./src",
"outDir": "./build"
},
"include": ["src/**/*"],
"exclude": ["src/**/*.test.ts"]
}

3
packages/cli/.babelrc Normal file
View file

@ -0,0 +1,3 @@
{
"presets": [["@verdaccio"]]
}

3
packages/cli/bin/verdaccio Executable file
View file

@ -0,0 +1,3 @@
#!/usr/bin/env node
require('../build');

View file

@ -0,0 +1,9 @@
module.exports = {
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx'],
transform: {
'^.+\\.(js|jsx|ts|tsx)$': 'babel-jest',
},
verbose: true,
collectCoverage: true,
coveragePathIgnorePatterns: ['node_modules', 'fixtures'],
};

40
packages/cli/package.json Normal file
View file

@ -0,0 +1,40 @@
{
"name": "@verdaccio/cli",
"version": "5.0.0-alpha.0",
"author": {
"name": "Juan Picado",
"email": "juanpicado19@gmail.com"
},
"bin": {
"verdaccio": "./bin/verdaccio",
"verdaccio-cli": "./bin/verdaccio"
},
"repository": {
"type": "git",
"url": "git://github.com/verdaccio/verdaccio"
},
"homepage": "https://verdaccio.org",
"description": "verdaccio CLI",
"license": "MIT",
"main": "./build/index.js",
"types": "build/index.d.ts",
"scripts": {
"clean": "rimraf ./build",
"test": "cross-env NODE_ENV=test BABEL_ENV=test jest",
"type-check": "tsc --noEmit",
"build:types": "tsc --emitDeclarationOnly --declaration true",
"build:js": "cross-env BABEL_ENV=registry babel src/ --out-dir build/ --copy-files --extensions \".ts,.tsx\" --source-maps",
"build": "npm run build:js && npm run build:types",
"bundle": "cross-env BABEL_ENV=registry webpack --config scripts/bundle.js --profile --json > compilation-stats.json"
},
"dependencies": {
"@verdaccio/config": "5.0.0-alpha.0",
"@verdaccio/node-api": "5.0.0-alpha.0",
"@verdaccio/utils": "5.0.0-alpha.0",
"commander": "3.0.2",
"envinfo": "7.4.0",
"kleur": "3.0.3",
"semver": "7.1.2"
},
"gitHead": "7c246ede52ff717707fcae66dd63fc4abd536982"
}

View file

@ -0,0 +1,5 @@
{
"rules": {
"no-console": 0
}
}

60
packages/cli/src/cli.ts Normal file
View file

@ -0,0 +1,60 @@
/* eslint no-sync:0 */
/* eslint no-empty:0 */
import commander from 'commander';
import { bgYellow, bgRed } from 'kleur';
import { setup, logger } from '@verdaccio/logger';
import infoCommand from "./commands/info";
import initProgram from "./commands/init";
import {isVersionValid, MIN_NODE_VERSION} from "./utils";
const isRootUser = process.getuid && process.getuid() === 0;
if (isRootUser) {
global.console.warn(bgYellow().red('*** WARNING: Verdaccio doesn\'t need superuser privileges. Don\'t run it under root! ***'));
}
if (isVersionValid()) {
global.console.error(bgRed(`Verdaccio requires at least Node.js ${MIN_NODE_VERSION} or higher, please upgrade your Node.js distribution`));
process.exit(1);
}
process.title = 'verdaccio';
// default setup
setup(null, {logStart: false});
const pkgVersion = '5.0.0';
const pkgName = 'verdaccio';
commander
.option('-i, --info', 'prints debugging information about the local environment')
.option('-l, --listen <[host:]port>', 'host:port number to listen on (default: localhost:4873)')
.option('-c, --config <config.yaml>', 'use this configuration file (default: ./config.yaml)')
.version(pkgVersion)
.parse(process.argv);
const fallbackConfig = commander.args.length == 1 && !commander.config;
const isHelp = commander.args.length !== 0;
if (commander.info) {
infoCommand();
} else if (fallbackConfig) {
// handling "verdaccio [config]" case if "-c" is missing in command line
commander.config = commander.args.pop();
initProgram(commander, pkgVersion, pkgName);
} else if (isHelp) {
commander.help();
} else {
initProgram(commander, pkgVersion, pkgName);
}
process.on('uncaughtException', function(err) {
logger.fatal( {
err: err,
},
'uncaught exception, please report (https://github.com/verdaccio/verdaccio/issues) this: \n@{err.stack}' );
process.exit(255);
});

View file

@ -0,0 +1,18 @@
import envinfo from 'envinfo';
export default function infoCommand() {
// eslint-disable-next-line no-console
console.log('\nEnvironment Info:');
(async () => {
const data = await envinfo.run({
System: ['OS', 'CPU'],
Binaries: ['Node', 'Yarn', 'npm'],
Virtualization: ['Docker'],
Browsers: ['Chrome', 'Edge', 'Firefox', 'Safari'],
npmGlobalPackages: ['verdaccio'],
});
// eslint-disable-next-line no-console
console.log(data);
process.exit(0);
})();
}

View file

@ -0,0 +1,42 @@
import path from "path";
import _ from 'lodash';
import { parseConfigFile} from "@verdaccio/utils";
import { findConfigFile } from "@verdaccio/config";
import { logger } from '@verdaccio/logger';
import {startVerdaccio, listenDefaultCallback} from "@verdaccio/node-api";
export const DEFAULT_PROCESS_NAME: string = 'verdaccio';
export default function initProgram(commander, pkgVersion, pkgName) {
const cliListener = commander.listen;
let configPathLocation;
let verdaccioConfiguration;
try {
configPathLocation = findConfigFile(commander.config);
verdaccioConfiguration = parseConfigFile(configPathLocation);
const { web, https, self_path } = verdaccioConfiguration;
process.title = web && web.title || DEFAULT_PROCESS_NAME;
// note: self_path is only being used by @verdaccio/storage , not really useful and migth be removed soon
if (!self_path) {
verdaccioConfiguration = _.assign({}, verdaccioConfiguration, {
self_path: path.resolve(configPathLocation)
});
}
if (!https) {
verdaccioConfiguration = _.assign({}, verdaccioConfiguration, {
https: {enable: false}
});
}
logger.warn({file: configPathLocation}, 'config file - @{file}');
startVerdaccio(verdaccioConfiguration, cliListener, configPathLocation, pkgVersion, pkgName, listenDefaultCallback);
} catch (err) {
logger.fatal({file: configPathLocation, err: err}, 'cannot open config file @{file}: @{!err.message}');
process.exit(1);
}
}

View file

@ -0,0 +1 @@
require('./cli');

View file

@ -0,0 +1,5 @@
import semver from "semver";
export const MIN_NODE_VERSION = '6.9.0';
export const isVersionValid = () => semver.satisfies(process.version, `>=${MIN_NODE_VERSION}`) === false

View file

@ -0,0 +1,3 @@
describe('cli test', () => {
test.todo('write some test for this module');
});

View file

@ -0,0 +1,9 @@
{
"extends": "../../tsconfig",
"compilerOptions": {
"rootDir": "./src",
"outDir": "./build"
},
"include": ["src/**/*"],
"exclude": ["src/**/*.test.ts"]
}

View file

@ -0,0 +1,3 @@
{
"presets": [["@verdaccio"]]
}

View file

@ -0,0 +1,25 @@
{
"name": "@verdaccio/dev-commons",
"version": "5.0.0-alpha.0",
"description": "loaders logic",
"main": "./build/index.js",
"types": "build/index.d.ts",
"author": {
"name": "Juan Picado",
"email": "juanpicado19@gmail.com"
},
"repository": {
"type": "git",
"url": "git://github.com/verdaccio/verdaccio"
},
"homepage": "https://verdaccio.org",
"license": "MIT",
"scripts": {
"clean": "rimraf ./build",
"type-check": "tsc --noEmit",
"build:types": "tsc --emitDeclarationOnly --declaration true",
"build:js": "cross-env BABEL_ENV=registry babel src/ --out-dir build/ --copy-files --extensions \".ts,.tsx\" --source-maps",
"build": "npm run build:js && npm run build:types"
},
"gitHead": "7c246ede52ff717707fcae66dd63fc4abd536982"
}

View file

@ -1,9 +1,3 @@
/**
* @prettier
*/
// @flow
export const DEFAULT_PORT = '4873'; export const DEFAULT_PORT = '4873';
export const DEFAULT_PROTOCOL = 'http'; export const DEFAULT_PROTOCOL = 'http';
export const DEFAULT_DOMAIN = 'localhost'; export const DEFAULT_DOMAIN = 'localhost';

View file

@ -0,0 +1 @@
export * from './constants';

View file

@ -0,0 +1,9 @@
{
"extends": "../../tsconfig",
"compilerOptions": {
"rootDir": "./src",
"outDir": "./build"
},
"include": ["src/**/*"],
"exclude": ["src/**/*.test.ts"]
}

3
packages/config/.babelrc Normal file
View file

@ -0,0 +1,3 @@
{
"presets": [["@verdaccio"]]
}

View file

@ -0,0 +1,9 @@
module.exports = {
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx'],
transform: {
'^.+\\.(js|jsx|ts|tsx)$': 'babel-jest',
},
verbose: true,
collectCoverage: true,
coveragePathIgnorePatterns: ['node_modules', 'fixtures'],
};

View file

@ -0,0 +1,32 @@
{
"name": "@verdaccio/config",
"version": "5.0.0-alpha.0",
"description": "logger",
"main": "./build/index.js",
"types": "build/index.d.ts",
"author": {
"name": "Juan Picado",
"email": "juanpicado19@gmail.com"
},
"repository": {
"type": "git",
"url": "git://github.com/verdaccio/verdaccio"
},
"license": "MIT",
"homepage": "https://verdaccio.org",
"scripts": {
"clean": "rimraf ./build",
"test": "cross-env NODE_ENV=test BABEL_ENV=test jest",
"type-check": "tsc --noEmit",
"build:types": "tsc --emitDeclarationOnly --declaration true",
"build:js": "cross-env BABEL_ENV=registry babel src/ --out-dir build/ --copy-files --extensions \".ts,.tsx\" --source-maps",
"build": "npm run build:js && npm run build:types"
},
"dependencies": {
"@verdaccio/utils": "5.0.0-alpha.0"
},
"devDependencies": {
"@types/bunyan": "1.8.6"
},
"gitHead": "7c246ede52ff717707fcae66dd63fc4abd536982"
}

View file

@ -1,22 +1,23 @@
import fs from 'fs'; import fs from 'fs';
import Path from 'path'; import path from 'path';
import _ from 'lodash'; import _ from 'lodash';
import Path from 'path';
import { logger } from '@verdaccio/logger';
import mkdirp from 'mkdirp'; import mkdirp from 'mkdirp';
import { logger } from './logger';
import { folderExists, fileExists } from './utils'; import { folderExists, fileExists } from '@verdaccio/utils';
import { CHARACTER_ENCODING } from './constants'; import { CHARACTER_ENCODING } from '@verdaccio/dev-commons';
const CONFIG_FILE = 'config.yaml'; const CONFIG_FILE = 'config.yaml';
const XDG = 'xdg'; const XDG = 'xdg';
const WIN = 'win'; const WIN = 'win';
const WIN32 = 'win32'; const WIN32 = 'win32';
// eslint-disable-next-line // eslint-disable-next-line
const pkgJSON = require('../../package.json'); const pkgJSON = require('../package.json');
export type SetupDirectory = { export type SetupDirectory = {
path: string; path: string;
type: string; type: string
}; };
/** /**
@ -34,9 +35,7 @@ function findConfigFile(configPath: string): string {
throw new Error('no configuration files can be processed'); throw new Error('no configuration files can be processed');
} }
const primaryConf: any = _.find(configPaths, (configLocation: any) => const primaryConf: any = _.find(configPaths, (configLocation: any) => fileExists(configLocation.path));
fileExists(configLocation.path)
);
if (_.isNil(primaryConf) === false) { if (_.isNil(primaryConf) === false) {
return primaryConf.path; return primaryConf.path;
} }
@ -54,8 +53,11 @@ function createConfigFile(configLocation: any): SetupDirectory {
return configLocation; return configLocation;
} }
function readDefaultConfig(): string { export function readDefaultConfig(): Buffer {
return fs.readFileSync(require.resolve('../../conf/default.yaml'), 'utf-8'); const pathDefaultConf: string = path.resolve(__dirname, 'conf/default.yaml');
// @ts-ignore
return fs.readFileSync(pathDefaultConf, CHARACTER_ENCODING.UTF8);
} }
function createConfigFolder(configLocation): void { function createConfigFolder(configLocation): void {
@ -71,8 +73,7 @@ function updateStorageLinks(configLocation, defaultConfig): string {
// $XDG_DATA_HOME defines the base directory relative to which user specific data files should be stored, // $XDG_DATA_HOME defines the base directory relative to which user specific data files should be stored,
// If $XDG_DATA_HOME is either not set or empty, a default equal to $HOME/.local/share should be used. // If $XDG_DATA_HOME is either not set or empty, a default equal to $HOME/.local/share should be used.
// $FlowFixMe // $FlowFixMe
let dataDir = let dataDir = process.env.XDG_DATA_HOME || Path.join(process.env.HOME as string, '.local', 'share');
process.env.XDG_DATA_HOME || Path.join(process.env.HOME as string, '.local', 'share');
if (folderExists(dataDir)) { if (folderExists(dataDir)) {
dataDir = Path.resolve(Path.join(dataDir, pkgJSON.name, 'storage')); dataDir = Path.resolve(Path.join(dataDir, pkgJSON.name, 'storage'));
return defaultConfig.replace(/^storage: .\/storage$/m, `storage: ${dataDir}`); return defaultConfig.replace(/^storage: .\/storage$/m, `storage: ${dataDir}`);
@ -81,12 +82,8 @@ function updateStorageLinks(configLocation, defaultConfig): string {
} }
function getConfigPaths(): SetupDirectory[] { function getConfigPaths(): SetupDirectory[] {
const listPaths: SetupDirectory[] = [ const listPaths: SetupDirectory[] = [getXDGDirectory(), getWindowsDirectory(), getRelativeDefaultDirectory(), getOldDirectory()].reduce(
getXDGDirectory(), function(acc, currentValue: any): SetupDirectory[] {
getWindowsDirectory(),
getRelativeDefaultDirectory(),
getOldDirectory()
].reduce(function (acc, currentValue: any): SetupDirectory[] {
if (_.isUndefined(currentValue) === false) { if (_.isUndefined(currentValue) === false) {
acc.push(currentValue); acc.push(currentValue);
} }
@ -102,7 +99,7 @@ const getXDGDirectory = (): SetupDirectory | void => {
if (XDGConfig && folderExists(XDGConfig)) { if (XDGConfig && folderExists(XDGConfig)) {
return { return {
path: Path.join(XDGConfig, pkgJSON.name, CONFIG_FILE), path: Path.join(XDGConfig, pkgJSON.name, CONFIG_FILE),
type: XDG type: XDG,
}; };
} }
}; };
@ -113,7 +110,7 @@ const getWindowsDirectory = (): SetupDirectory | void => {
if (process.platform === WIN32 && process.env.APPDATA && folderExists(process.env.APPDATA)) { if (process.platform === WIN32 && process.env.APPDATA && folderExists(process.env.APPDATA)) {
return { return {
path: Path.resolve(Path.join(process.env.APPDATA, pkgJSON.name, CONFIG_FILE)), path: Path.resolve(Path.join(process.env.APPDATA, pkgJSON.name, CONFIG_FILE)),
type: WIN type: WIN,
}; };
} }
}; };
@ -121,15 +118,15 @@ const getWindowsDirectory = (): SetupDirectory | void => {
const getRelativeDefaultDirectory = (): SetupDirectory => { const getRelativeDefaultDirectory = (): SetupDirectory => {
return { return {
path: Path.resolve(Path.join('.', pkgJSON.name, CONFIG_FILE)), path: Path.resolve(Path.join('.', pkgJSON.name, CONFIG_FILE)),
type: 'def' type: 'def',
}; };
}; };
const getOldDirectory = (): SetupDirectory => { const getOldDirectory = (): SetupDirectory => {
return { return {
path: Path.resolve(Path.join('.', CONFIG_FILE)), path: Path.resolve(Path.join('.', CONFIG_FILE)),
type: 'old' type: 'old',
}; };
}; };
export default findConfigFile; export { findConfigFile };

View file

@ -1,19 +1,23 @@
import assert from 'assert';
import _ from 'lodash'; import _ from 'lodash';
import assert from 'assert';
import { PackageList, Config as AppConfig, Security, Logger } from '@verdaccio/types';
import { MatchedPackage, StartUpConfig } from '../../types';
import { generateRandomHexString } from './crypto-utils';
import { import {
getMatchedPackagesSpec, getMatchedPackagesSpec,
normalisePackageAccess, normalisePackageAccess,
sanityCheckUplinksProps, sanityCheckUplinksProps,
uplinkSanityCheck uplinkSanityCheck,
} from './config-utils'; generateRandomHexString,
import { getUserAgent, isObject } from './utils'; getUserAgent,
import { APP_ERROR } from './constants'; isObject
} from '@verdaccio/utils';
import { APP_ERROR } from '@verdaccio/dev-commons';
import { PackageList, Config as AppConfig, Security, Logger } from '@verdaccio/types';
import { MatchedPackage, StartUpConfig } from '@verdaccio/dev-types';
const LoggerApi = require('@verdaccio/logger');
const LoggerApi = require('./logger');
const strategicConfigProps = ['uplinks', 'packages']; const strategicConfigProps = ['uplinks', 'packages'];
const allowedEnvConfig = ['http_proxy', 'https_proxy', 'no_proxy']; const allowedEnvConfig = ['http_proxy', 'https_proxy', 'no_proxy'];
@ -75,11 +79,13 @@ class Config implements AppConfig {
this.packages = normalisePackageAccess(self.packages); this.packages = normalisePackageAccess(self.packages);
// loading these from ENV if aren't in config // loading these from ENV if aren't in config
allowedEnvConfig.forEach((envConf): void => { allowedEnvConfig.forEach(
(envConf): void => {
if (!(envConf in self)) { if (!(envConf in self)) {
self[envConf] = process.env[envConf] || process.env[envConf.toUpperCase()]; self[envConf] = process.env[envConf] || process.env[envConf.toUpperCase()];
} }
}); }
);
// unique identifier of self server (or a cluster), used to avoid loops // unique identifier of self server (or a cluster), used to avoid loops
// @ts-ignore // @ts-ignore
@ -110,4 +116,4 @@ class Config implements AppConfig {
} }
} }
export default Config; export { Config };

View file

@ -0,0 +1,2 @@
export * from './config'
export * from './config-path'

View file

@ -1,18 +1,20 @@
import path from 'path'; import path from 'path';
import _ from 'lodash'; import _ from 'lodash';
import Config from '../../../../src/lib/config'; import { Config, readDefaultConfig } from '@verdaccio/config';
import { parseConfigFile } from '../../../../src/lib/utils'; import { setup } from '@verdaccio/logger';
import { DEFAULT_REGISTRY, DEFAULT_UPLINK, ROLES, WEB_TITLE } from '../../../../src/lib/constants'; import {DEFAULT_REGISTRY, DEFAULT_UPLINK, ROLES, WEB_TITLE} from '@verdaccio/dev-commons';
import { setup } from '../../../../src/lib/logger';
import {parseConfigFile} from '@verdaccio/utils';
setup([]); setup([]);
const resolveConf = (conf) => { const resolveConf = (conf) => {
const { name, ext } = path.parse(conf); const { name, ext } = path.parse(conf);
return path.join(__dirname, `../../../../conf/${name}${ext.startsWith('.') ? ext : '.yaml'}`); return path.join(__dirname, `../src/conf/${name}${ext.startsWith('.') ? ext : '.yaml'}`);
}; };
const checkDefaultUplink = (config) => { const checkDefaultUplink = (config) => {
expect(_.isObject(config.uplinks[DEFAULT_UPLINK])).toBeTruthy(); expect(_.isObject(config.uplinks[DEFAULT_UPLINK])).toBeTruthy();
expect(config.uplinks[DEFAULT_UPLINK].url).toMatch(DEFAULT_REGISTRY); expect(config.uplinks[DEFAULT_UPLINK].url).toMatch(DEFAULT_REGISTRY);
@ -43,7 +45,7 @@ const checkDefaultConfPackages = (config) => {
expect(config.packages['**'].publish).toBeDefined(); expect(config.packages['**'].publish).toBeDefined();
expect(config.packages['**'].publish).toContainEqual(ROLES.$AUTH); expect(config.packages['**'].publish).toContainEqual(ROLES.$AUTH);
expect(config.packages['**'].proxy).toBeDefined(); expect(config.packages['**'].proxy).toBeDefined();
expect(config.packages['**'].proxy).toContainEqual(DEFAULT_UPLINK); expect(config.packages['**'].proxy,).toContainEqual(DEFAULT_UPLINK);
// uplinks // uplinks
expect(config.uplinks[DEFAULT_UPLINK]).toBeDefined(); expect(config.uplinks[DEFAULT_UPLINK]).toBeDefined();
expect(config.uplinks[DEFAULT_UPLINK].url).toEqual(DEFAULT_REGISTRY); expect(config.uplinks[DEFAULT_UPLINK].url).toEqual(DEFAULT_REGISTRY);
@ -93,5 +95,9 @@ describe('Config file', () => {
}); });
}); });
describe('Config file', () => {}); describe('Config file', () => {
}); });
});

View file

@ -0,0 +1,9 @@
{
"extends": "../../tsconfig",
"compilerOptions": {
"rootDir": "./src",
"outDir": "./build"
},
"include": ["src/**/*"],
"exclude": ["src/**/*.test.ts"]
}

3
packages/hooks/.babelrc Normal file
View file

@ -0,0 +1,3 @@
{
"presets": [["@verdaccio"]]
}

View file

@ -0,0 +1,9 @@
module.exports = {
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx'],
transform: {
'^.+\\.(js|jsx|ts|tsx)$': 'babel-jest',
},
verbose: true,
collectCoverage: true,
coveragePathIgnorePatterns: ['node_modules', 'fixtures'],
};

View file

@ -0,0 +1,36 @@
{
"name": "@verdaccio/hooks",
"version": "5.0.0-alpha.0",
"description": "loaders logic",
"main": "./build/index.js",
"types": "build/index.d.ts",
"author": {
"name": "Juan Picado",
"email": "juanpicado19@gmail.com"
},
"repository": {
"type": "git",
"url": "git://github.com/verdaccio/verdaccio"
},
"license": "MIT",
"homepage": "https://verdaccio.org",
"dependencies": {
"@verdaccio/commons-api": "^9.0.0",
"@verdaccio/logger": "5.0.0-alpha.0",
"handlebars": "^4.5.3",
"request": "2.87.0"
},
"devDependencies": {
"@verdaccio/dev-commons": "5.0.0-alpha.0",
"@verdaccio/types": "^8.5.2"
},
"scripts": {
"clean": "rimraf ./build",
"test": "cross-env NODE_ENV=test BABEL_ENV=test jest",
"type-check": "tsc --noEmit",
"build:types": "tsc --emitDeclarationOnly --declaration true",
"build:js": "cross-env BABEL_ENV=registry babel src/ --out-dir build/ --copy-files --extensions \".ts,.tsx\" --source-maps",
"build": "npm run build:js && npm run build:types"
},
"gitHead": "7c246ede52ff717707fcae66dd63fc4abd536982"
}

View file

@ -0,0 +1 @@
export {handleNotify, notify, sendNotification } from './notify';

View file

@ -0,0 +1,25 @@
import isNil from 'lodash/isNil';
import request, { RequiredUriUrl } from 'request';
import { logger } from '@verdaccio/logger';
import { HTTP_STATUS } from '@verdaccio/commons-api';
export function notifyRequest(options: RequiredUriUrl, content): Promise<any | Error> {
return new Promise(
(resolve, reject): void => {
request(options, function(err, response, body): void {
if (err || response.statusCode >= HTTP_STATUS.BAD_REQUEST) {
const errorMessage = isNil(err) ? response.body : err.message;
logger.error({ errorMessage }, 'notify service has thrown an error: @{errorMessage}');
reject(errorMessage);
}
logger.info({ content }, 'A notification has been shipped: @{content}');
if (isNil(body) === false) {
logger.debug({ body }, ' body: @{body}');
resolve(body);
}
reject(Error('body is missing'));
});
}
);
}

View file

@ -1,14 +1,13 @@
import { parseConfigurationFile } from '../../__helper'; import {parseConfigFile} from '@verdaccio/utils';
import { parseConfigFile } from '../../../../src/lib/utils'; import { setup } from '@verdaccio/logger';
import { notify } from '../../../../src/lib/notify';
import { notifyRequest } from '../../../../src/lib/notify/notify-request'; import {notify} from '../src';
import {notifyRequest} from '../src/notify-request';
import { setup } from '../../../../src/lib/logger'; import {parseConfigurationFile} from './__helper';
setup([]); setup([]);
jest.mock('./../../../../src/lib/notify/notify-request', () => ({ jest.mock('../src/notify-request', () => ({
notifyRequest: jest.fn((options, content) => Promise.resolve([options, content])) notifyRequest: jest.fn((options, content) => Promise.resolve([options, content]))
})); }));
@ -16,29 +15,26 @@ const parseConfigurationNotifyFile = (name) => {
return parseConfigurationFile(`notify/${name}`); return parseConfigurationFile(`notify/${name}`);
}; };
const singleNotificationConfig = parseConfigFile(parseConfigurationNotifyFile('single.notify')); const singleNotificationConfig = parseConfigFile(parseConfigurationNotifyFile('single.notify'));
const singleHeaderNotificationConfig = parseConfigFile( const singleHeaderNotificationConfig = parseConfigFile(parseConfigurationNotifyFile('single.header.notify'));
parseConfigurationNotifyFile('single.header.notify') const packagePatternNotificationConfig = parseConfigFile(parseConfigurationNotifyFile('single.packagePattern.notify'));
);
const packagePatternNotificationConfig = parseConfigFile(
parseConfigurationNotifyFile('single.packagePattern.notify')
);
const multiNotificationConfig = parseConfigFile(parseConfigurationNotifyFile('multiple.notify')); const multiNotificationConfig = parseConfigFile(parseConfigurationNotifyFile('multiple.notify'));
describe('Notifications:: Notify', () => { describe('Notifications:: Notify', () => {
beforeEach(() => { beforeEach(() => {
jest.clearAllMocks(); jest.clearAllMocks();
}); });
// FUTURE: we should add some sort of health check of all props, (not implemented yet) // FUTURE: we should add some sort of health check of all props, (not implemented yet)
test('should not fails if config is not provided', async () => { test("should not fails if config is not provided", async () => {
// @ts-ignore // @ts-ignore
await notify({}, {}); await notify({}, {});
expect(notifyRequest).toHaveBeenCalledTimes(0); expect(notifyRequest).toHaveBeenCalledTimes(0);
}); });
test('should send notification', async () => { test("should send notification", async () => {
const name = 'package'; const name = 'package';
// @ts-ignore // @ts-ignore
const response = await notify({name}, singleNotificationConfig, { name: 'foo'}, 'bar'); const response = await notify({name}, singleNotificationConfig, { name: 'foo'}, 'bar');
@ -52,14 +48,14 @@ describe('Notifications:: Notify', () => {
expect(notifyRequest).toHaveBeenCalledTimes(1); expect(notifyRequest).toHaveBeenCalledTimes(1);
}); });
test('should send single header notification', async () => { test("should send single header notification", async () => {
// @ts-ignore // @ts-ignore
await notify({}, singleHeaderNotificationConfig, { name: 'foo'}, 'bar'); await notify({}, singleHeaderNotificationConfig, { name: 'foo'}, 'bar');
expect(notifyRequest).toHaveBeenCalledTimes(1); expect(notifyRequest).toHaveBeenCalledTimes(1);
}); });
test('should send multiple notification', async () => { test("should send multiple notification", async () => {
// @ts-ignore // @ts-ignore
await notify({name}, multiNotificationConfig, { name: 'foo'}, 'bar'); await notify({name}, multiNotificationConfig, { name: 'foo'}, 'bar');
@ -68,20 +64,23 @@ describe('Notifications:: Notify', () => {
}); });
describe('packagePatternFlags', () => { describe('packagePatternFlags', () => {
test('should send single notification with packagePatternFlags', async () => { test("should send single notification with packagePatternFlags", async () => {
const name = 'package'; const name = 'package';
// @ts-ignore // @ts-ignore
await notify({name}, packagePatternNotificationConfig, { name: 'foo'}, 'bar'); await notify({name}, packagePatternNotificationConfig, { name: 'foo'}, 'bar');
expect(notifyRequest).toHaveBeenCalledTimes(1); expect(notifyRequest).toHaveBeenCalledTimes(1);
}); });
test('should not match on send single notification with packagePatternFlags', async () => { test("should not match on send single notification with packagePatternFlags", async () => {
const name = 'no-mach-name'; const name = 'no-mach-name';
// @ts-ignore // @ts-ignore
await notify({name}, packagePatternNotificationConfig, { name: 'foo'}, 'bar'); await notify({name}, packagePatternNotificationConfig, { name: 'foo'}, 'bar');
expect(notifyRequest).toHaveBeenCalledTimes(0); expect(notifyRequest).toHaveBeenCalledTimes(0);
}); });
}); })
}); });

View file

@ -1,4 +1,4 @@
import { HTTP_STATUS, API_ERROR } from '../../../../src/lib/constants'; import { HTTP_STATUS, API_ERROR } from '@verdaccio/dev-commons';
/* eslint-disable @typescript-eslint/no-var-requires */ /* eslint-disable @typescript-eslint/no-var-requires */
/** /**
@ -8,16 +8,16 @@ const logger = {
logger: { logger: {
error: jest.fn(), error: jest.fn(),
debug: jest.fn(), debug: jest.fn(),
info: jest.fn() info: jest.fn(),
} },
}; };
jest.doMock('../../../../src/lib/logger', () => logger); jest.doMock('@verdaccio/logger', () => logger);
/** /**
* Test Data * Test Data
*/ */
const options = { const options = {
url: 'http://slack-service' url: 'http://slack-service',
}; };
const content = 'Verdaccio@x.x.x successfully published'; const content = 'Verdaccio@x.x.x successfully published';
@ -29,19 +29,16 @@ describe('Notifications:: notifyRequest', () => {
test('when notification service throws error', async () => { test('when notification service throws error', async () => {
jest.doMock('request', () => (options, resolver) => { jest.doMock('request', () => (options, resolver) => {
const response = { const response = {
statusCode: HTTP_STATUS.BAD_REQUEST statusCode: HTTP_STATUS.BAD_REQUEST,
}; };
const error = { const error = {
message: API_ERROR.BAD_DATA message: API_ERROR.BAD_DATA,
}; };
resolver(error, response); resolver(error, response);
}); });
const notification = require('../../../../src/lib/notify/notify-request'); const notification = require('../src/notify-request');
const args = [ const args = [{ errorMessage: 'bad data' }, 'notify service has thrown an error: @{errorMessage}'];
{ errorMessage: 'bad data' },
'notify service has thrown an error: @{errorMessage}'
];
await expect(notification.notifyRequest(options, content)).rejects.toEqual(API_ERROR.BAD_DATA); await expect(notification.notifyRequest(options, content)).rejects.toEqual(API_ERROR.BAD_DATA);
expect(logger.logger.error).toHaveBeenCalledWith(...args); expect(logger.logger.error).toHaveBeenCalledWith(...args);
@ -51,17 +48,14 @@ describe('Notifications:: notifyRequest', () => {
jest.doMock('request', () => (options, resolver) => { jest.doMock('request', () => (options, resolver) => {
const response = { const response = {
statusCode: HTTP_STATUS.BAD_REQUEST, statusCode: HTTP_STATUS.BAD_REQUEST,
body: API_ERROR.BAD_DATA body: API_ERROR.BAD_DATA,
}; };
resolver(null, response); resolver(null, response);
}); });
const notification = require('../../../../src/lib/notify/notify-request'); const notification = require('../src/notify-request');
const args = [ const args = [{ errorMessage: 'bad data' }, 'notify service has thrown an error: @{errorMessage}'];
{ errorMessage: 'bad data' },
'notify service has thrown an error: @{errorMessage}'
];
await expect(notification.notifyRequest(options, content)).rejects.toEqual(API_ERROR.BAD_DATA); await expect(notification.notifyRequest(options, content)).rejects.toEqual(API_ERROR.BAD_DATA);
expect(logger.logger.error).toHaveBeenCalledWith(...args); expect(logger.logger.error).toHaveBeenCalledWith(...args);
@ -71,19 +65,17 @@ describe('Notifications:: notifyRequest', () => {
jest.doMock('request', () => (options, resolver) => { jest.doMock('request', () => (options, resolver) => {
const response = { const response = {
statusCode: HTTP_STATUS.OK, statusCode: HTTP_STATUS.OK,
body: 'Successfully delivered' body: 'Successfully delivered',
}; };
resolver(null, response, response.body); resolver(null, response, response.body);
}); });
const notification = require('../../../../src/lib/notify/notify-request'); const notification = require('../src/notify-request');
const infoArgs = [{ content }, 'A notification has been shipped: @{content}']; const infoArgs = [{ content }, 'A notification has been shipped: @{content}'];
const debugArgs = [{ body: 'Successfully delivered' }, ' body: @{body}']; const debugArgs = [{ body: 'Successfully delivered' }, ' body: @{body}'];
await expect(notification.notifyRequest(options, content)).resolves.toEqual( await expect(notification.notifyRequest(options, content)).resolves.toEqual('Successfully delivered');
'Successfully delivered'
);
expect(logger.logger.info).toHaveBeenCalledWith(...infoArgs); expect(logger.logger.info).toHaveBeenCalledWith(...infoArgs);
expect(logger.logger.debug).toHaveBeenCalledWith(...debugArgs); expect(logger.logger.debug).toHaveBeenCalledWith(...debugArgs);
}); });
@ -91,13 +83,13 @@ describe('Notifications:: notifyRequest', () => {
test('when notification is successfully delivered but body is undefined/null', async () => { test('when notification is successfully delivered but body is undefined/null', async () => {
jest.doMock('request', () => (options, resolver) => { jest.doMock('request', () => (options, resolver) => {
const response = { const response = {
statusCode: HTTP_STATUS.OK statusCode: HTTP_STATUS.OK,
}; };
resolver(null, response); resolver(null, response);
}); });
const notification = require('../../../../src/lib/notify/notify-request'); const notification = require('../src/notify-request');
const infoArgs = [{ content }, 'A notification has been shipped: @{content}']; const infoArgs = [{ content }, 'A notification has been shipped: @{content}'];
await expect(notification.notifyRequest(options, content)).rejects.toThrow('body is missing'); await expect(notification.notifyRequest(options, content)).rejects.toThrow('body is missing');

View file

@ -0,0 +1,9 @@
{
"extends": "../../tsconfig",
"compilerOptions": {
"rootDir": "./src",
"outDir": "./build"
},
"include": ["src/**/*"],
"exclude": ["src/**/*.test.ts"]
}

Some files were not shown because too many files have changed in this diff Show more