verdaccio 6.3.2 → 6.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -67,8 +67,135 @@ Verdaccio can be used as a module to launch a server programmatically. You can f
67
67
  });
68
68
  ```
69
69
 
70
+ ## Get Started
71
+
72
+ To get started, run Verdaccio in your terminal:
73
+
74
+ ```bash
75
+ verdaccio
76
+ ```
77
+
78
+ Optionally, you can set some npm configuration:
79
+
80
+ ```bash
81
+ $ npm set registry http://localhost:4873/
82
+ ```
83
+
84
+ For one-off commands or to avoid setting the registry globally, use:
85
+
86
+ ```bash
87
+ NPM_CONFIG_REGISTRY=http://localhost:4873 npm i
88
+ ```
89
+
90
+ You can now navigate to [http://localhost:4873/](http://localhost:4873/) where your local packages will be listed and searchable.
91
+
92
+ > Warning: Verdaccio [does not currently support PM2's cluster mode](https://github.com/verdaccio/verdaccio/issues/1301#issuecomment-489302298). Running it with cluster mode may lead to unknown behavior.
93
+
70
94
  ## Plugins
71
95
 
96
+ Verdaccio ships with a set of built-in plugins so it works out of the box with zero configuration. You can replace or extend any of them with community plugins.
97
+
98
+ ### Bundled plugins
99
+
100
+ | Plugin | Category | Enabled by default | Description |
101
+ | ---------------------------- | ---------- | ------------------ | -------------------------------------------------------------------------------- |
102
+ | `verdaccio-htpasswd` | auth | Yes | Default authentication backend using an `htpasswd` file to store users. |
103
+ | `verdaccio-audit` | middleware | Yes | Implements the `npm audit` endpoint by proxying requests to a configured uplink. |
104
+ | `@verdaccio/local-storage` | storage | Yes | Default filesystem storage backend for packages and metadata. |
105
+ | `@verdaccio/ui-theme` | theme | Yes | Default web UI theme shipped with Verdaccio. |
106
+ | `@verdaccio/package-filter` | filter | No | Filters package metadata from uplinks (block versions, quarantine, whitelist). |
107
+
108
+ ### Package Filter
109
+
110
+ `@verdaccio/package-filter` is a built-in plugin that intercepts package metadata from uplinks and removes versions matching configurable rules. With no rules configured, it acts as a no-op passthrough.
111
+
112
+ #### Block a compromised package version
113
+
114
+ ```yaml
115
+ filters:
116
+ '@verdaccio/package-filter':
117
+ block:
118
+ - package: 'event-stream'
119
+ versions: '3.3.6'
120
+ ```
121
+
122
+ #### Block an entire malicious scope
123
+
124
+ ```yaml
125
+ filters:
126
+ '@verdaccio/package-filter':
127
+ block:
128
+ - scope: '@malicious'
129
+ ```
130
+
131
+ #### Quarantine recently published versions
132
+
133
+ Hide versions published less than 7 days ago, giving time for review before adoption:
134
+
135
+ ```yaml
136
+ filters:
137
+ '@verdaccio/package-filter':
138
+ minAgeDays: 7
139
+ ```
140
+
141
+ #### Freeze registry to a point in time
142
+
143
+ Only serve versions published before a specific date:
144
+
145
+ ```yaml
146
+ filters:
147
+ '@verdaccio/package-filter':
148
+ dateThreshold: '2025-01-01'
149
+ ```
150
+
151
+ #### Whitelist trusted packages within blocked rules
152
+
153
+ ```yaml
154
+ filters:
155
+ '@verdaccio/package-filter':
156
+ minAgeDays: 30
157
+ allow:
158
+ - scope: '@my-company'
159
+ - package: 'trusted-pkg'
160
+ ```
161
+
162
+ #### Replace instead of remove
163
+
164
+ Substitute a blocked version with the nearest older safe version, useful when removing it would break transitive dependencies:
165
+
166
+ ```yaml
167
+ filters:
168
+ '@verdaccio/package-filter':
169
+ block:
170
+ - package: 'compromised-lib'
171
+ versions: '>=3.0.0'
172
+ strategy: replace
173
+ ```
174
+
175
+ #### Full example
176
+
177
+ ```yaml
178
+ filters:
179
+ '@verdaccio/package-filter':
180
+ minAgeDays: 7
181
+ block:
182
+ - scope: '@malicious'
183
+ - package: 'typosquat-pkg'
184
+ - package: 'compromised-lib'
185
+ versions: '>=3.0.0'
186
+ strategy: replace
187
+ allow:
188
+ - scope: '@my-org'
189
+ - package: 'compromised-lib'
190
+ versions: '3.0.1'
191
+ ```
192
+
193
+ If a filter rejects a package entirely, Verdaccio returns a 404 to the client.
194
+
195
+ Learn more at the [`@verdaccio/package-filter` documentation](https://github.com/verdaccio/verdaccio/blob/8.x/packages/plugins/package-filter/README.md) and the [filter plugins documentation](https://verdaccio.org/docs/plugin-filter).
196
+
197
+ ### Custom plugins
198
+
72
199
  You can develop your own [plugins](https://verdaccio.org/docs/plugins) using the [verdaccio generator](https://github.com/verdaccio/generator-verdaccio-plugin). [Yeoman](https://yeoman.io/) installation is required.
73
200
 
74
201
  Learn more about [how to develop plugins](https://verdaccio.org/docs/dev-plugins) and share them with the community.
@@ -107,39 +234,10 @@ Here are a few examples to get started:
107
234
 
108
235
  - [e2e-ci-example-gh-actions](https://github.com/juanpicado/e2e-ci-example-gh-actions)
109
236
  - [verdaccio-end-to-end-tests](https://github.com/juanpicado/verdaccio-end-to-end-tests)
110
- - [verdaccio-fork](https://github.com/juanpicado/verdaccio-fork)
111
237
 
112
238
  ## Talks
113
239
 
114
- - [**NodeTLV 2022** - Deep Dive into Verdaccio, a Lightweight Node.js Registry - Juan Picado](https://portal.gitnation.org/contents/five-ways-of-taking-advantage-of-verdaccio-your-private-and-proxy-nodejs-registry)
115
- - [Five Ways of Taking Advantage of Verdaccio, Your Private and Proxy Node.js Registry - **Node Congress 2022** - Juan Picado](https://portal.gitnation.org/contents/five-ways-of-taking-advantage-of-verdaccio-your-private-and-proxy-nodejs-registry)
116
- - [Using Docker and Verdaccio to Make Integration Testing Easy - **Docker All Hands #4 December 2021** - Juan Picado](https://www.youtube.com/watch?v=zRI0skF1f8I)
117
-
118
- [View more on the YouTube channel](https://www.youtube.com/channel/UC5i20v6o7lSjXzAHOvatt0w).
119
-
120
- ## Get Started
121
-
122
- To get started, run Verdaccio in your terminal:
123
-
124
- ```bash
125
- verdaccio
126
- ```
127
-
128
- Optionally, you can set some npm configuration:
129
-
130
- ```bash
131
- $ npm set registry http://localhost:4873/
132
- ```
133
-
134
- For one-off commands or to avoid setting the registry globally, use:
135
-
136
- ```bash
137
- NPM_CONFIG_REGISTRY=http://localhost:4873 npm i
138
- ```
139
-
140
- You can now navigate to [http://localhost:4873/](http://localhost:4873/) where your local packages will be listed and searchable.
141
-
142
- > Warning: Verdaccio [does not currently support PM2's cluster mode](https://github.com/verdaccio/verdaccio/issues/1301#issuecomment-489302298). Running it with cluster mode may lead to unknown behavior.
240
+ See the list of Verdaccio talks at [verdaccio.org/talks](https://www.verdaccio.org/talks).
143
241
 
144
242
  ## Publishing
145
243
 
@@ -177,6 +275,8 @@ Available as [tags](https://hub.docker.com/r/verdaccio/verdaccio/tags/).
177
275
  docker pull verdaccio/verdaccio:6.x-next
178
276
  ```
179
277
 
278
+ > The `6.x-next` tag tracks the latest changes from the `6.x` branch. It is useful for previewing upcoming fixes and features, but it is **not recommended for production use**.
279
+
180
280
  ### Running Verdaccio using Docker
181
281
 
182
282
  To run the Docker container:
@@ -191,38 +291,28 @@ Docker examples are available [in this repository](https://github.com/verdaccio/
191
291
 
192
292
  Verdaccio aims to support all relevant features of a standard npm client for private repositories. However, full compatibility isn't always possible.
193
293
 
194
- ### Basic features
195
-
196
- - Installing packages (`npm install`, `npm upgrade`, etc.) - **supported**
197
- - Publishing packages (`npm publish`) - **supported**
198
-
199
- ### Advanced package control
200
-
201
- - Unpublishing packages (`npm unpublish`) - **supported**
202
- - Tagging (`npm tag`) - **supported**
203
- - Deprecation (`npm deprecate`) - **supported**
204
-
205
- ### User management
206
-
207
- - Registering new users (`npm adduser {newuser}`) - **supported**
208
- - Change password (`npm profile set password`) - **supported**
209
- - Transferring ownership (`npm owner add {user} {pkg}`) - not supported, _PRs welcome_
210
- - Token (`npm token`) - **supported** (under flag)
211
-
212
- ### Miscellany
213
-
214
- - Search (`npm search`) - **supported** (cli (`/-/all` and `v1`) / browser)
215
- - Ping (`npm ping`) - **supported**
216
- - Starring (`npm star`, `npm unstar`, `npm stars`) - **supported**
217
-
218
- ### Security
219
-
220
- - npm/yarn audit - **supported**
294
+ | Category | Feature | Command | Status |
295
+ | ------------------------ | ---------------------- | --------------------------------------- | ------------------------- |
296
+ | Basic features | Installing packages | `npm install`, `npm upgrade`, etc. | Supported |
297
+ | Basic features | Publishing packages | `npm publish` | Supported |
298
+ | Advanced package control | Unpublishing packages | `npm unpublish` | Supported |
299
+ | Advanced package control | Tagging | `npm tag` | Supported |
300
+ | Advanced package control | Deprecation | `npm deprecate` | Supported |
301
+ | User management | Registering new users | `npm adduser {newuser}` | Supported |
302
+ | User management | Change password | `npm profile set password` | Supported |
303
+ | User management | Transferring ownership | `npm owner add {user} {pkg}` | Not supported, PRs welcome |
304
+ | User management | Token | `npm token` | Supported (under flag) |
305
+ | Miscellany | Search | `npm search` | Supported (cli `/-/all` and `v1`, browser) |
306
+ | Miscellany | Ping | `npm ping` | Supported |
307
+ | Miscellany | Starring | `npm star`, `npm unstar`, `npm stars` | Supported |
308
+ | Security | Audit | `npm audit`, `yarn audit` | Supported |
221
309
 
222
310
  ## Report a vulnerability
223
311
 
224
312
  To report a security vulnerability, please follow the steps outlined in our [security policy](https://github.com/verdaccio/verdaccio/policy).
225
313
 
314
+ > **Note:** There is currently **no funding available for security research or bounty rewards**.
315
+
226
316
 
227
317
 
228
318
  ## Open Collective Backers
@@ -1,4 +1,4 @@
1
1
  import type { Router } from 'express';
2
- import type Auth from '../../../lib/auth';
2
+ import type { Auth } from '@verdaccio/auth';
3
3
  import type Storage from '../../../lib/storage';
4
4
  export default function (route: Router, auth: Auth, storage: Storage): void;
@@ -74,4 +74,4 @@ function _default(route, auth, storage) {
74
74
  });
75
75
  });
76
76
  }
77
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_lodash","_interopRequireDefault","require","_mime","_middleware","_constants","_logger","e","__esModule","default","_default","route","auth","storage","can","allow","beforeAll","params","message","logger","trace","afterAll","tag_package_version","req","res","next","_","isString","body","tags","tag","mergeTags","package","err","status","HTTP_STATUS","CREATED","ok","API_MESSAGE","TAG_ADDED","put","DIST_TAGS_API_ENDPOINTS","tagging","media","mime","getType","post","tagging_package","delete","TAG_REMOVED","get","get_dist_tags","getPackage","name","uplinksLook","callback","info","DIST_TAGS","TAG_UPDATED"],"sources":["../../../../src/api/endpoint/api/dist-tags.ts"],"sourcesContent":["import type { Router } from 'express';\nimport _ from 'lodash';\nimport mime from 'mime';\n\nimport { DIST_TAGS_API_ENDPOINTS, allow, media } from '@verdaccio/middleware';\nimport type { Package } from '@verdaccio/types';\n\nimport type Auth from '../../../lib/auth';\nimport { API_MESSAGE, DIST_TAGS, HTTP_STATUS } from '../../../lib/constants';\nimport { logger } from '../../../lib/logger';\nimport type Storage from '../../../lib/storage';\nimport type { $NextFunctionVer, $RequestExtend, $ResponseExtend } from '../../../types';\n\nexport default function (route: Router, auth: Auth, storage: Storage): void {\n  const can = allow(auth, {\n    beforeAll: (params, message) => logger.trace(params, message),\n    afterAll: (params, message) => logger.trace(params, message),\n  });\n  const tag_package_version = function (\n    req: $RequestExtend,\n    res: $ResponseExtend,\n    next: $NextFunctionVer\n  ): $NextFunctionVer {\n    if (_.isString(req.body) === false) {\n      return next('route');\n    }\n\n    const tags = {};\n    tags[req.params.tag] = req.body;\n    storage.mergeTags(req.params.package, tags, function (err: Error): $NextFunctionVer {\n      if (err) {\n        return next(err);\n      }\n      res.status(HTTP_STATUS.CREATED);\n      return next({ ok: API_MESSAGE.TAG_ADDED });\n    });\n  };\n\n  // tagging a package\n  route.put(\n    DIST_TAGS_API_ENDPOINTS.tagging,\n    can('publish'),\n    media(mime.getType('json')),\n    tag_package_version\n  );\n\n  route.post(\n    DIST_TAGS_API_ENDPOINTS.tagging_package,\n    can('publish'),\n    media(mime.getType('json')),\n    tag_package_version\n  );\n\n  route.put(\n    DIST_TAGS_API_ENDPOINTS.tagging_package,\n    can('publish'),\n    media(mime.getType('json')),\n    tag_package_version\n  );\n\n  route.delete(\n    DIST_TAGS_API_ENDPOINTS.tagging_package,\n    can('publish'),\n    function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n      const tags = {};\n      tags[req.params.tag] = null;\n      storage.mergeTags(req.params.package, tags, function (err: any): $NextFunctionVer {\n        if (err) {\n          return next(err);\n        }\n        res.status(HTTP_STATUS.CREATED);\n        return next({\n          ok: API_MESSAGE.TAG_REMOVED,\n        });\n      });\n    }\n  );\n\n  route.get(\n    DIST_TAGS_API_ENDPOINTS.get_dist_tags,\n    can('access'),\n    function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n      storage.getPackage({\n        name: req.params.package,\n        uplinksLook: true,\n        req,\n        callback: function (err: any, info: Package): $NextFunctionVer {\n          if (err) {\n            return next(err);\n          }\n\n          next(info[DIST_TAGS]);\n        },\n      });\n    }\n  );\n\n  route.post(\n    DIST_TAGS_API_ENDPOINTS.get_dist_tags,\n    can('publish'),\n    function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n      storage.mergeTags(req.params.package, req.body, function (err: any): $NextFunctionVer {\n        if (err) {\n          return next(err);\n        }\n        res.status(HTTP_STATUS.CREATED);\n        return next({\n          ok: API_MESSAGE.TAG_UPDATED,\n        });\n      });\n    }\n  );\n}\n"],"mappings":"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","ignoreList":[]}
77
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_lodash","_interopRequireDefault","require","_mime","_middleware","_constants","_logger","e","__esModule","default","_default","route","auth","storage","can","allow","beforeAll","params","message","logger","trace","afterAll","tag_package_version","req","res","next","_","isString","body","tags","tag","mergeTags","package","err","status","HTTP_STATUS","CREATED","ok","API_MESSAGE","TAG_ADDED","put","DIST_TAGS_API_ENDPOINTS","tagging","media","mime","getType","post","tagging_package","delete","TAG_REMOVED","get","get_dist_tags","getPackage","name","uplinksLook","callback","info","DIST_TAGS","TAG_UPDATED"],"sources":["../../../../src/api/endpoint/api/dist-tags.ts"],"sourcesContent":["import type { Router } from 'express';\nimport _ from 'lodash';\nimport mime from 'mime';\n\nimport type { Auth } from '@verdaccio/auth';\nimport { DIST_TAGS_API_ENDPOINTS, allow, media } from '@verdaccio/middleware';\nimport type { Package } from '@verdaccio/types';\n\nimport { API_MESSAGE, DIST_TAGS, HTTP_STATUS } from '../../../lib/constants';\nimport { logger } from '../../../lib/logger';\nimport type Storage from '../../../lib/storage';\nimport type { $NextFunctionVer, $RequestExtend, $ResponseExtend } from '../../../types';\n\nexport default function (route: Router, auth: Auth, storage: Storage): void {\n  const can = allow(auth, {\n    beforeAll: (params, message) => logger.trace(params, message),\n    afterAll: (params, message) => logger.trace(params, message),\n  });\n  const tag_package_version = function (\n    req: $RequestExtend,\n    res: $ResponseExtend,\n    next: $NextFunctionVer\n  ): $NextFunctionVer {\n    if (_.isString(req.body) === false) {\n      return next('route');\n    }\n\n    const tags = {};\n    tags[req.params.tag] = req.body;\n    storage.mergeTags(req.params.package, tags, function (err: Error): $NextFunctionVer {\n      if (err) {\n        return next(err);\n      }\n      res.status(HTTP_STATUS.CREATED);\n      return next({ ok: API_MESSAGE.TAG_ADDED });\n    });\n  };\n\n  // tagging a package\n  route.put(\n    DIST_TAGS_API_ENDPOINTS.tagging,\n    can('publish'),\n    media(mime.getType('json')),\n    tag_package_version\n  );\n\n  route.post(\n    DIST_TAGS_API_ENDPOINTS.tagging_package,\n    can('publish'),\n    media(mime.getType('json')),\n    tag_package_version\n  );\n\n  route.put(\n    DIST_TAGS_API_ENDPOINTS.tagging_package,\n    can('publish'),\n    media(mime.getType('json')),\n    tag_package_version\n  );\n\n  route.delete(\n    DIST_TAGS_API_ENDPOINTS.tagging_package,\n    can('publish'),\n    function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n      const tags = {};\n      tags[req.params.tag] = null;\n      storage.mergeTags(req.params.package, tags, function (err: any): $NextFunctionVer {\n        if (err) {\n          return next(err);\n        }\n        res.status(HTTP_STATUS.CREATED);\n        return next({\n          ok: API_MESSAGE.TAG_REMOVED,\n        });\n      });\n    }\n  );\n\n  route.get(\n    DIST_TAGS_API_ENDPOINTS.get_dist_tags,\n    can('access'),\n    function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n      storage.getPackage({\n        name: req.params.package,\n        uplinksLook: true,\n        req,\n        callback: function (err: any, info: Package): $NextFunctionVer {\n          if (err) {\n            return next(err);\n          }\n\n          next(info[DIST_TAGS]);\n        },\n      });\n    }\n  );\n\n  route.post(\n    DIST_TAGS_API_ENDPOINTS.get_dist_tags,\n    can('publish'),\n    function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n      storage.mergeTags(req.params.package, req.body, function (err: any): $NextFunctionVer {\n        if (err) {\n          return next(err);\n        }\n        res.status(HTTP_STATUS.CREATED);\n        return next({\n          ok: API_MESSAGE.TAG_UPDATED,\n        });\n      });\n    }\n  );\n}\n"],"mappings":"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","ignoreList":[]}
@@ -1,5 +1,5 @@
1
1
  import type { Router } from 'express';
2
+ import type { Auth } from '@verdaccio/auth';
2
3
  import type { Config } from '@verdaccio/types';
3
- import type Auth from '../../../lib/auth';
4
4
  import type Storage from '../../../lib/storage';
5
5
  export default function (route: Router, auth: Auth, storage: Storage, config: Config): void;
@@ -100,4 +100,4 @@ function _default(route, auth, storage, config) {
100
100
  }
101
101
  });
102
102
  }
103
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_lodash","_interopRequireDefault","require","_core","_middleware","_tarball","_constants","_logger","_utils","e","__esModule","default","downloadStream","packageName","filename","storage","req","res","stream","getTarball","on","content","header","err","locals","report_error","HEADERS","CONTENT_TYPE","OCTET_STREAM","pipe","redirectOrDownloadStream","config","tarballUrlRedirect","_","get","hasLocalTarball","then","context","tarballUrl","template","redirect","catch","_default","route","auth","can","allow","beforeAll","params","message","logger","trace","afterAll","PACKAGE_API_ENDPOINTS","get_package_by_version","next","abbreviated","stringUtils","getByQualityPriorityValue","getPackageMetaCallback","metadata","convertDistRemoteToLocalTarballUrls","protocol","headers","host","hostname","remoteAddress","socket","url_prefix","queryVersion","version","isNil","setHeader","JSON","getVersion","DIST_TAGS","ErrorCode","getNotFound","API_ERROR","VERSION_NOT_EXIST","getPackage","name","package","uplinksLook","callback","get_package_tarball","undefined"],"sources":["../../../../src/api/endpoint/api/package.ts"],"sourcesContent":["import type { Router } from 'express';\nimport _ from 'lodash';\n\nimport { stringUtils } from '@verdaccio/core';\nimport { PACKAGE_API_ENDPOINTS, allow } from '@verdaccio/middleware';\nimport { convertDistRemoteToLocalTarballUrls } from '@verdaccio/tarball';\nimport type { Config, Manifest } from '@verdaccio/types';\n\nimport type Auth from '../../../lib/auth';\nimport { API_ERROR, DIST_TAGS, HEADERS } from '../../../lib/constants';\nimport { logger } from '../../../lib/logger';\nimport type Storage from '../../../lib/storage';\nimport { ErrorCode, getVersion } from '../../../lib/utils';\nimport type { $NextFunctionVer, $RequestExtend, $ResponseExtend } from '../../../types';\n\nconst downloadStream = (\n  packageName: string,\n  filename: string,\n  storage: any,\n  req: $RequestExtend,\n  res: $ResponseExtend\n): void => {\n  const stream = storage.getTarball(packageName, filename);\n\n  stream.on('content-length', function (content): void {\n    res.header('Content-Length', content);\n  });\n\n  stream.on('error', function (err): void {\n    return res.locals.report_error(err);\n  });\n\n  res.header(HEADERS.CONTENT_TYPE, HEADERS.OCTET_STREAM);\n  stream.pipe(res);\n};\n\nconst redirectOrDownloadStream = (\n  packageName: string,\n  filename: string,\n  storage: any,\n  req: $RequestExtend,\n  res: $ResponseExtend,\n  config: Config\n): void => {\n  const tarballUrlRedirect = _.get(config, 'experiments.tarball_url_redirect');\n  storage\n    .hasLocalTarball(packageName, filename)\n    .then((hasLocalTarball) => {\n      if (hasLocalTarball) {\n        const context = { packageName, filename };\n        const tarballUrl =\n          typeof tarballUrlRedirect === 'function'\n            ? tarballUrlRedirect(context)\n            : _.template(tarballUrlRedirect)(context);\n        res.redirect(tarballUrl);\n      } else {\n        downloadStream(packageName, filename, storage, req, res);\n      }\n    })\n    .catch((err) => {\n      res.locals.report_error(err);\n    });\n};\n\nexport default function (route: Router, auth: Auth, storage: Storage, config: Config): void {\n  const can = allow(auth, {\n    beforeAll: (params, message) => logger.trace(params, message),\n    afterAll: (params, message) => logger.trace(params, message),\n  });\n  // TODO: anonymous user?\n  route.get(\n    PACKAGE_API_ENDPOINTS.get_package_by_version,\n    can('access'),\n    function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n      const abbreviated =\n        stringUtils.getByQualityPriorityValue(req.get('Accept')) ===\n        'application/vnd.npm.install-v1+json';\n      const getPackageMetaCallback = function (err, metadata: Manifest): void {\n        if (err) {\n          return next(err);\n        }\n        metadata = convertDistRemoteToLocalTarballUrls(\n          metadata,\n          {\n            protocol: req.protocol,\n            headers: req.headers as any,\n            host: req.hostname,\n            remoteAddress: req.socket.remoteAddress,\n          },\n          config.url_prefix\n        );\n\n        let queryVersion = req.params.version;\n        if (_.isNil(queryVersion)) {\n          if (abbreviated) {\n            res.setHeader(HEADERS.CONTENT_TYPE, 'application/vnd.npm.install-v1+json');\n          } else {\n            res.setHeader(HEADERS.CONTENT_TYPE, HEADERS.JSON);\n          }\n\n          return next(metadata);\n        }\n\n        let version = getVersion(metadata, queryVersion);\n        if (_.isNil(version) === false) {\n          res.setHeader(HEADERS.CONTENT_TYPE, HEADERS.JSON);\n          return next(version);\n        }\n\n        if (_.isNil(metadata[DIST_TAGS]) === false) {\n          if (_.isNil(metadata[DIST_TAGS][queryVersion]) === false) {\n            queryVersion = metadata[DIST_TAGS][queryVersion];\n            version = getVersion(metadata, queryVersion);\n            if (_.isNil(version) === false) {\n              res.setHeader(HEADERS.CONTENT_TYPE, HEADERS.JSON);\n              return next(version);\n            }\n          }\n        }\n        return next(ErrorCode.getNotFound(`${API_ERROR.VERSION_NOT_EXIST}: ${req.params.version}`));\n      };\n\n      storage.getPackage({\n        name: req.params.package,\n        uplinksLook: true,\n        req,\n        abbreviated,\n        callback: getPackageMetaCallback,\n      });\n    }\n  );\n\n  route.get(\n    PACKAGE_API_ENDPOINTS.get_package_tarball,\n    can('access'),\n    function (req: $RequestExtend, res: $ResponseExtend): void {\n      if (_.get(config, 'experiments.tarball_url_redirect') === undefined) {\n        downloadStream(req.params.package, req.params.filename, storage, req, res);\n      } else {\n        redirectOrDownloadStream(\n          req.params.package,\n          req.params.filename,\n          storage,\n          req,\n          res,\n          config\n        );\n      }\n    }\n  );\n}\n"],"mappings":"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","ignoreList":[]}
103
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_lodash","_interopRequireDefault","require","_core","_middleware","_tarball","_constants","_logger","_utils","e","__esModule","default","downloadStream","packageName","filename","storage","req","res","stream","getTarball","on","content","header","err","locals","report_error","HEADERS","CONTENT_TYPE","OCTET_STREAM","pipe","redirectOrDownloadStream","config","tarballUrlRedirect","_","get","hasLocalTarball","then","context","tarballUrl","template","redirect","catch","_default","route","auth","can","allow","beforeAll","params","message","logger","trace","afterAll","PACKAGE_API_ENDPOINTS","get_package_by_version","next","abbreviated","stringUtils","getByQualityPriorityValue","getPackageMetaCallback","metadata","convertDistRemoteToLocalTarballUrls","protocol","headers","host","hostname","remoteAddress","socket","url_prefix","queryVersion","version","isNil","setHeader","JSON","getVersion","DIST_TAGS","ErrorCode","getNotFound","API_ERROR","VERSION_NOT_EXIST","getPackage","name","package","uplinksLook","callback","get_package_tarball","undefined"],"sources":["../../../../src/api/endpoint/api/package.ts"],"sourcesContent":["import type { Router } from 'express';\nimport _ from 'lodash';\n\nimport type { Auth } from '@verdaccio/auth';\nimport { stringUtils } from '@verdaccio/core';\nimport { PACKAGE_API_ENDPOINTS, allow } from '@verdaccio/middleware';\nimport { convertDistRemoteToLocalTarballUrls } from '@verdaccio/tarball';\nimport type { Config, Manifest } from '@verdaccio/types';\n\nimport { API_ERROR, DIST_TAGS, HEADERS } from '../../../lib/constants';\nimport { logger } from '../../../lib/logger';\nimport type Storage from '../../../lib/storage';\nimport { ErrorCode, getVersion } from '../../../lib/utils';\nimport type { $NextFunctionVer, $RequestExtend, $ResponseExtend } from '../../../types';\n\nconst downloadStream = (\n  packageName: string,\n  filename: string,\n  storage: any,\n  req: $RequestExtend,\n  res: $ResponseExtend\n): void => {\n  const stream = storage.getTarball(packageName, filename);\n\n  stream.on('content-length', function (content): void {\n    res.header('Content-Length', content);\n  });\n\n  stream.on('error', function (err): void {\n    return res.locals.report_error(err);\n  });\n\n  res.header(HEADERS.CONTENT_TYPE, HEADERS.OCTET_STREAM);\n  stream.pipe(res);\n};\n\nconst redirectOrDownloadStream = (\n  packageName: string,\n  filename: string,\n  storage: any,\n  req: $RequestExtend,\n  res: $ResponseExtend,\n  config: Config\n): void => {\n  const tarballUrlRedirect = _.get(config, 'experiments.tarball_url_redirect');\n  storage\n    .hasLocalTarball(packageName, filename)\n    .then((hasLocalTarball) => {\n      if (hasLocalTarball) {\n        const context = { packageName, filename };\n        const tarballUrl =\n          typeof tarballUrlRedirect === 'function'\n            ? tarballUrlRedirect(context)\n            : _.template(tarballUrlRedirect)(context);\n        res.redirect(tarballUrl);\n      } else {\n        downloadStream(packageName, filename, storage, req, res);\n      }\n    })\n    .catch((err) => {\n      res.locals.report_error(err);\n    });\n};\n\nexport default function (route: Router, auth: Auth, storage: Storage, config: Config): void {\n  const can = allow(auth, {\n    beforeAll: (params, message) => logger.trace(params, message),\n    afterAll: (params, message) => logger.trace(params, message),\n  });\n  // TODO: anonymous user?\n  route.get(\n    PACKAGE_API_ENDPOINTS.get_package_by_version,\n    can('access'),\n    function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n      const abbreviated =\n        stringUtils.getByQualityPriorityValue(req.get('Accept')) ===\n        'application/vnd.npm.install-v1+json';\n      const getPackageMetaCallback = function (err, metadata: Manifest): void {\n        if (err) {\n          return next(err);\n        }\n        metadata = convertDistRemoteToLocalTarballUrls(\n          metadata,\n          {\n            protocol: req.protocol,\n            headers: req.headers as any,\n            host: req.hostname,\n            remoteAddress: req.socket.remoteAddress,\n          },\n          config.url_prefix\n        );\n\n        let queryVersion = req.params.version;\n        if (_.isNil(queryVersion)) {\n          if (abbreviated) {\n            res.setHeader(HEADERS.CONTENT_TYPE, 'application/vnd.npm.install-v1+json');\n          } else {\n            res.setHeader(HEADERS.CONTENT_TYPE, HEADERS.JSON);\n          }\n\n          return next(metadata);\n        }\n\n        let version = getVersion(metadata, queryVersion);\n        if (_.isNil(version) === false) {\n          res.setHeader(HEADERS.CONTENT_TYPE, HEADERS.JSON);\n          return next(version);\n        }\n\n        if (_.isNil(metadata[DIST_TAGS]) === false) {\n          if (_.isNil(metadata[DIST_TAGS][queryVersion]) === false) {\n            queryVersion = metadata[DIST_TAGS][queryVersion];\n            version = getVersion(metadata, queryVersion);\n            if (_.isNil(version) === false) {\n              res.setHeader(HEADERS.CONTENT_TYPE, HEADERS.JSON);\n              return next(version);\n            }\n          }\n        }\n        return next(ErrorCode.getNotFound(`${API_ERROR.VERSION_NOT_EXIST}: ${req.params.version}`));\n      };\n\n      storage.getPackage({\n        name: req.params.package,\n        uplinksLook: true,\n        req,\n        abbreviated,\n        callback: getPackageMetaCallback,\n      });\n    }\n  );\n\n  route.get(\n    PACKAGE_API_ENDPOINTS.get_package_tarball,\n    can('access'),\n    function (req: $RequestExtend, res: $ResponseExtend): void {\n      if (_.get(config, 'experiments.tarball_url_redirect') === undefined) {\n        downloadStream(req.params.package, req.params.filename, storage, req, res);\n      } else {\n        redirectOrDownloadStream(\n          req.params.package,\n          req.params.filename,\n          storage,\n          req,\n          res,\n          config\n        );\n      }\n    }\n  );\n}\n"],"mappings":";;;;;;AACA,IAAAA,OAAA,GAAAC,sBAAA,CAAAC,OAAA;AAGA,IAAAC,KAAA,GAAAD,OAAA;AACA,IAAAE,WAAA,GAAAF,OAAA;AACA,IAAAG,QAAA,GAAAH,OAAA;AAGA,IAAAI,UAAA,GAAAJ,OAAA;AACA,IAAAK,OAAA,GAAAL,OAAA;AAEA,IAAAM,MAAA,GAAAN,OAAA;AAA2D,SAAAD,uBAAAQ,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAG3D,MAAMG,cAAc,GAAGA,CACrBC,WAAmB,EACnBC,QAAgB,EAChBC,OAAY,EACZC,GAAmB,EACnBC,GAAoB,KACX;EACT,MAAMC,MAAM,GAAGH,OAAO,CAACI,UAAU,CAACN,WAAW,EAAEC,QAAQ,CAAC;EAExDI,MAAM,CAACE,EAAE,CAAC,gBAAgB,EAAE,UAAUC,OAAO,EAAQ;IACnDJ,GAAG,CAACK,MAAM,CAAC,gBAAgB,EAAED,OAAO,CAAC;EACvC,CAAC,CAAC;EAEFH,MAAM,CAACE,EAAE,CAAC,OAAO,EAAE,UAAUG,GAAG,EAAQ;IACtC,OAAON,GAAG,CAACO,MAAM,CAACC,YAAY,CAACF,GAAG,CAAC;EACrC,CAAC,CAAC;EAEFN,GAAG,CAACK,MAAM,CAACI,kBAAO,CAACC,YAAY,EAAED,kBAAO,CAACE,YAAY,CAAC;EACtDV,MAAM,CAACW,IAAI,CAACZ,GAAG,CAAC;AAClB,CAAC;AAED,MAAMa,wBAAwB,GAAGA,CAC/BjB,WAAmB,EACnBC,QAAgB,EAChBC,OAAY,EACZC,GAAmB,EACnBC,GAAoB,EACpBc,MAAc,KACL;EACT,MAAMC,kBAAkB,GAAGC,eAAC,CAACC,GAAG,CAACH,MAAM,EAAE,kCAAkC,CAAC;EAC5EhB,OAAO,CACJoB,eAAe,CAACtB,WAAW,EAAEC,QAAQ,CAAC,CACtCsB,IAAI,CAAED,eAAe,IAAK;IACzB,IAAIA,eAAe,EAAE;MACnB,MAAME,OAAO,GAAG;QAAExB,WAAW;QAAEC;MAAS,CAAC;MACzC,MAAMwB,UAAU,GACd,OAAON,kBAAkB,KAAK,UAAU,GACpCA,kBAAkB,CAACK,OAAO,CAAC,GAC3BJ,eAAC,CAACM,QAAQ,CAACP,kBAAkB,CAAC,CAACK,OAAO,CAAC;MAC7CpB,GAAG,CAACuB,QAAQ,CAACF,UAAU,CAAC;IAC1B,CAAC,MAAM;MACL1B,cAAc,CAACC,WAAW,EAAEC,QAAQ,EAAEC,OAAO,EAAEC,GAAG,EAAEC,GAAG,CAAC;IAC1D;EACF,CAAC,CAAC,CACDwB,KAAK,CAAElB,GAAG,IAAK;IACdN,GAAG,CAACO,MAAM,CAACC,YAAY,CAACF,GAAG,CAAC;EAC9B,CAAC,CAAC;AACN,CAAC;AAEc,SAAAmB,SAAUC,KAAa,EAAEC,IAAU,EAAE7B,OAAgB,EAAEgB,MAAc,EAAQ;EAC1F,MAAMc,GAAG,GAAG,IAAAC,iBAAK,EAACF,IAAI,EAAE;IACtBG,SAAS,EAAEA,CAACC,MAAM,EAAEC,OAAO,KAAKC,cAAM,CAACC,KAAK,CAACH,MAAM,EAAEC,OAAO,CAAC;IAC7DG,QAAQ,EAAEA,CAACJ,MAAM,EAAEC,OAAO,KAAKC,cAAM,CAACC,KAAK,CAACH,MAAM,EAAEC,OAAO;EAC7D,CAAC,CAAC;EACF;EACAN,KAAK,CAACT,GAAG,CACPmB,iCAAqB,CAACC,sBAAsB,EAC5CT,GAAG,CAAC,QAAQ,CAAC,EACb,UAAU7B,GAAmB,EAAEC,GAAoB,EAAEsC,IAAsB,EAAQ;IACjF,MAAMC,WAAW,GACfC,iBAAW,CAACC,yBAAyB,CAAC1C,GAAG,CAACkB,GAAG,CAAC,QAAQ,CAAC,CAAC,KACxD,qCAAqC;IACvC,MAAMyB,sBAAsB,GAAG,SAAAA,CAAUpC,GAAG,EAAEqC,QAAkB,EAAQ;MACtE,IAAIrC,GAAG,EAAE;QACP,OAAOgC,IAAI,CAAChC,GAAG,CAAC;MAClB;MACAqC,QAAQ,GAAG,IAAAC,4CAAmC,EAC5CD,QAAQ,EACR;QACEE,QAAQ,EAAE9C,GAAG,CAAC8C,QAAQ;QACtBC,OAAO,EAAE/C,GAAG,CAAC+C,OAAc;QAC3BC,IAAI,EAAEhD,GAAG,CAACiD,QAAQ;QAClBC,aAAa,EAAElD,GAAG,CAACmD,MAAM,CAACD;MAC5B,CAAC,EACDnC,MAAM,CAACqC,UACT,CAAC;MAED,IAAIC,YAAY,GAAGrD,GAAG,CAACgC,MAAM,CAACsB,OAAO;MACrC,IAAIrC,eAAC,CAACsC,KAAK,CAACF,YAAY,CAAC,EAAE;QACzB,IAAIb,WAAW,EAAE;UACfvC,GAAG,CAACuD,SAAS,CAAC9C,kBAAO,CAACC,YAAY,EAAE,qCAAqC,CAAC;QAC5E,CAAC,MAAM;UACLV,GAAG,CAACuD,SAAS,CAAC9C,kBAAO,CAACC,YAAY,EAAED,kBAAO,CAAC+C,IAAI,CAAC;QACnD;QAEA,OAAOlB,IAAI,CAACK,QAAQ,CAAC;MACvB;MAEA,IAAIU,OAAO,GAAG,IAAAI,iBAAU,EAACd,QAAQ,EAAES,YAAY,CAAC;MAChD,IAAIpC,eAAC,CAACsC,KAAK,CAACD,OAAO,CAAC,KAAK,KAAK,EAAE;QAC9BrD,GAAG,CAACuD,SAAS,CAAC9C,kBAAO,CAACC,YAAY,EAAED,kBAAO,CAAC+C,IAAI,CAAC;QACjD,OAAOlB,IAAI,CAACe,OAAO,CAAC;MACtB;MAEA,IAAIrC,eAAC,CAACsC,KAAK,CAACX,QAAQ,CAACe,oBAAS,CAAC,CAAC,KAAK,KAAK,EAAE;QAC1C,IAAI1C,eAAC,CAACsC,KAAK,CAACX,QAAQ,CAACe,oBAAS,CAAC,CAACN,YAAY,CAAC,CAAC,KAAK,KAAK,EAAE;UACxDA,YAAY,GAAGT,QAAQ,CAACe,oBAAS,CAAC,CAACN,YAAY,CAAC;UAChDC,OAAO,GAAG,IAAAI,iBAAU,EAACd,QAAQ,EAAES,YAAY,CAAC;UAC5C,IAAIpC,eAAC,CAACsC,KAAK,CAACD,OAAO,CAAC,KAAK,KAAK,EAAE;YAC9BrD,GAAG,CAACuD,SAAS,CAAC9C,kBAAO,CAACC,YAAY,EAAED,kBAAO,CAAC+C,IAAI,CAAC;YACjD,OAAOlB,IAAI,CAACe,OAAO,CAAC;UACtB;QACF;MACF;MACA,OAAOf,IAAI,CAACqB,gBAAS,CAACC,WAAW,CAAC,GAAGC,oBAAS,CAACC,iBAAiB,KAAK/D,GAAG,CAACgC,MAAM,CAACsB,OAAO,EAAE,CAAC,CAAC;IAC7F,CAAC;IAEDvD,OAAO,CAACiE,UAAU,CAAC;MACjBC,IAAI,EAAEjE,GAAG,CAACgC,MAAM,CAACkC,OAAO;MACxBC,WAAW,EAAE,IAAI;MACjBnE,GAAG;MACHwC,WAAW;MACX4B,QAAQ,EAAEzB;IACZ,CAAC,CAAC;EACJ,CACF,CAAC;EAEDhB,KAAK,CAACT,GAAG,CACPmB,iCAAqB,CAACgC,mBAAmB,EACzCxC,GAAG,CAAC,QAAQ,CAAC,EACb,UAAU7B,GAAmB,EAAEC,GAAoB,EAAQ;IACzD,IAAIgB,eAAC,CAACC,GAAG,CAACH,MAAM,EAAE,kCAAkC,CAAC,KAAKuD,SAAS,EAAE;MACnE1E,cAAc,CAACI,GAAG,CAACgC,MAAM,CAACkC,OAAO,EAAElE,GAAG,CAACgC,MAAM,CAAClC,QAAQ,EAAEC,OAAO,EAAEC,GAAG,EAAEC,GAAG,CAAC;IAC5E,CAAC,MAAM;MACLa,wBAAwB,CACtBd,GAAG,CAACgC,MAAM,CAACkC,OAAO,EAClBlE,GAAG,CAACgC,MAAM,CAAClC,QAAQ,EACnBC,OAAO,EACPC,GAAG,EACHC,GAAG,EACHc,MACF,CAAC;IACH;EACF,CACF,CAAC;AACH","ignoreList":[]}
@@ -1,6 +1,6 @@
1
1
  import type { Router } from 'express';
2
+ import type { Auth } from '@verdaccio/auth';
2
3
  import type { Config } from '@verdaccio/types';
3
- import type Auth from '../../../lib/auth';
4
4
  import type Storage from '../../../lib/storage';
5
5
  import type { $NextFunctionVer, $RequestExtend, $ResponseExtend } from '../../../types';
6
6
  export default function publish(router: Router, auth: Auth, storage: Storage, config: Config): void;
@@ -356,4 +356,4 @@ function uploadPackageTarball(storage) {
356
356
  });
357
357
  };
358
358
  }
359
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_debug","_interopRequireDefault","require","_lodash","_mime","_path","_core","_hooks","_middleware","_constants","_logger","_storageUtils","_utils","_star","e","__esModule","default","debug","buildDebug","publish","router","auth","storage","config","can","allow","beforeAll","params","message","logger","trace","afterAll","put","media","mime","getType","expectJson","publishPackage","delete","unPublishPackage","removeTarball","HEADERS","OCTET_STREAM","uploadPackageTarball","addVersion","starApi","star","req","res","next","packageName","package","createTarball","filename","data","cb","stream","addTarball","on","err","end","Buffer","from","done","createVersion","version","metadata","addTags","tags","mergeTags","afterChange","error","okMessage","metadataCopy","_attachments","versions","_","isNil","JSON","stringify","status","HTTP_STATUS","CREATED","ok","success","isInvalidBodyFormat","isObject","hasDiffOneKey","info","ErrorCode","getBadRequest","API_ERROR","UNSUPORTED_REGISTRY_CALL","CONFLICT","firstAttachmentKey","Object","keys","Path","basename","versionToPublish","versionMetadataToPublish","readme","String","DIST_TAGS","notify","remote_user","name","isPublishablePackage","body","users","validationUtils","normalizeMetadata","_rev","isRelatedToDeprecation","isEmpty","remote","allow_unpublish","changePackage","revision","API_MESSAGE","PKG_CHANGED","addPackage","PKG_CREATED","getBadData","BAD_PACKAGE_DATA","removePackage","PKG_REMOVED","TARBALL_REMOVED","tag","PKG_PUBLISHED","pipe","complete","abort","locals","report_error","TARBALL_UPLOADED"],"sources":["../../../../src/api/endpoint/api/publish.ts"],"sourcesContent":["import buildDebug from 'debug';\nimport type { Router } from 'express';\nimport _ from 'lodash';\nimport mime from 'mime';\nimport Path from 'path';\n\nimport { validationUtils } from '@verdaccio/core';\nimport { notify } from '@verdaccio/hooks';\nimport { allow, expectJson, media } from '@verdaccio/middleware';\nimport type { Callback, Config, MergeTags, Package, Version } from '@verdaccio/types';\n\nimport type Auth from '../../../lib/auth';\nimport { API_ERROR, API_MESSAGE, DIST_TAGS, HEADERS, HTTP_STATUS } from '../../../lib/constants';\nimport { logger } from '../../../lib/logger';\nimport type Storage from '../../../lib/storage';\nimport { isPublishablePackage } from '../../../lib/storage-utils';\nimport { ErrorCode, hasDiffOneKey, isObject, isRelatedToDeprecation } from '../../../lib/utils';\nimport type { $NextFunctionVer, $RequestExtend, $ResponseExtend } from '../../../types';\nimport star from './star';\n\nconst debug = buildDebug('verdaccio:publish');\n\nexport default function publish(\n  router: Router,\n  auth: Auth,\n  storage: Storage,\n  config: Config\n): void {\n  const can = allow(auth, {\n    beforeAll: (params, message) => logger.trace(params, message),\n    afterAll: (params, message) => logger.trace(params, message),\n  });\n\n  /**\n   * Publish a package / update package / un/start a package\n   *\n   * There are multiples scenarios here to be considered:\n   *\n   * 1. Publish scenario\n   *\n   * Publish a package consist of at least 1 step (PUT) with a metadata payload.\n   * When a package is published, an _attachment property is present that contains the data\n   * of the tarball.\n   *\n   * Example flow of publish.\n   *\n   *  npm http fetch PUT 201 http://localhost:4873/@scope%2ftest1 9627ms\n      npm info lifecycle @scope/test1@1.0.1~publish: @scope/test1@1.0.1\n      npm info lifecycle @scope/test1@1.0.1~postpublish: @scope/test1@1.0.1\n      + @scope/test1@1.0.1\n      npm verb exit [ 0, true ]\n   *\n   *\n   * 2. Unpublish scenario\n   *\n   * Unpublish consist in 3 steps.\n   *  1. Try to fetch  metadata -> if it fails, return 404\n   *  2. Compute metadata locally (client side) and send a mutate payload excluding the version to be unpublished\n   *    eg: if metadata reflects 1.0.1, 1.0.2 and 1.0.3, the computed metadata won't include 1.0.3.\n   *  3. Once the second step has been successfully finished, delete the tarball.\n   *\n   *  All these steps are consecutive and required, there is no transacions here, if step 3 fails, metadata might\n   *  get corrupted.\n   *\n   *  Note the unpublish call will suffix in the url a /-rev/14-5d500cfce92f90fd revision number, this not\n   *  used internally.\n   *\n   *\n   * Example flow of unpublish.\n   *\n   * npm http fetch GET 200 http://localhost:4873/@scope%2ftest1?write=true 1680ms\n     npm http fetch PUT 201 http://localhost:4873/@scope%2ftest1/-rev/14-5d500cfce92f90fd 956606ms attempt #2\n     npm http fetch GET 200 http://localhost:4873/@scope%2ftest1?write=true 1601ms\n     npm http fetch DELETE 201 http://localhost:4873/@scope%2ftest1/-/test1-1.0.3.tgz/-rev/16-e11c8db282b2d992 19ms\n   *\n   * 3. Star a package\n   *\n   * Permissions: start a package depends of the publish and unpublish permissions, there is no specific flag for star or un start.\n   * The URL for star is similar to the unpublish (change package format)\n   *\n   * npm has no enpoint for star a package, rather mutate the metadata and acts as, the difference is the\n   * users property which is part of the payload and the body only includes\n   *\n   * {\n\t\t  \"_id\": pkgName,\n\t  \t\"_rev\": \"3-b0cdaefc9bdb77c8\",\n\t\t  \"users\": {\n\t\t    [username]: boolean value (true, false)\n\t\t  }\n\t}\n   *\n   */\n  router.put(\n    '/:package/:_rev?/:revision?',\n    can('publish'),\n    media(mime.getType('json')),\n    expectJson,\n    publishPackage(storage, config, auth)\n  );\n\n  /**\n   * Un-publishing an entire package.\n   *\n   * This scenario happens when the first call detect there is only one version remaining\n   * in the metadata, then the client decides to DELETE the resource\n   * npm http fetch GET 304 http://localhost:4873/@scope%2ftest1?write=true 1076ms (from cache)\n     npm http fetch DELETE 201 http://localhost:4873/@scope%2ftest1/-rev/18-d8ebe3020bd4ac9c 22ms\n   */\n  router.delete('/:package/-rev/*', can('unpublish'), unPublishPackage(storage));\n\n  // removing a tarball\n  router.delete(\n    '/:package/-/:filename/-rev/:revision',\n    can('unpublish'),\n    can('publish'),\n    removeTarball(storage)\n  );\n\n  // uploading package tarball\n  router.put(\n    '/:package/-/:filename/*',\n    can('publish'),\n    media(HEADERS.OCTET_STREAM),\n    uploadPackageTarball(storage)\n  );\n\n  // only used for development\n  if (config._debug) {\n    // adding a version\n    router.put(\n      '/:package/:version/-tag/:tag',\n      can('publish'),\n      media(mime.getType('json')),\n      expectJson,\n      addVersion(storage)\n    );\n  }\n}\n\n/**\n * Publish a package\n */\nexport function publishPackage(storage: Storage, config: Config, auth: Auth): any {\n  const starApi = star(storage);\n  return function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    const packageName = req.params.package;\n    debug('publishing or updating a new version for %o', packageName);\n    /**\n     * Write tarball of stream data from package clients.\n     */\n    const createTarball = function (filename: string, data, cb: Callback): void {\n      const stream = storage.addTarball(packageName, filename);\n      stream.on('error', function (err) {\n        cb(err);\n      });\n      stream.on('success', function () {\n        cb();\n      });\n      // this is dumb and memory-consuming, but what choices do we have?\n      // flow: we need first refactor this file before decides which type use here\n      stream.end(Buffer.from(data.data, 'base64'));\n      stream.done();\n    };\n\n    /**\n     * Add new package version in storage\n     */\n    const createVersion = function (version: string, metadata: Version, cb: Callback): void {\n      storage.addVersion(packageName, version, metadata, null, cb);\n    };\n\n    /**\n     * Add new tags in storage\n     */\n    const addTags = function (tags: MergeTags, cb: Callback): void {\n      storage.mergeTags(packageName, tags, cb);\n    };\n\n    const afterChange = function (error, okMessage, metadata): void {\n      const metadataCopy: Package = { ...metadata };\n\n      const { _attachments, versions } = metadataCopy;\n\n      // `npm star` wouldn't have attachments\n      // and `npm deprecate` would have attachments as a empty object, i.e {}\n      if (_.isNil(_attachments) || JSON.stringify(_attachments) === '{}') {\n        if (error) {\n          return next(error);\n        }\n        res.status(HTTP_STATUS.CREATED);\n        return next({\n          ok: okMessage,\n          success: true,\n        });\n      }\n\n      // npm-registry-client 0.3+ embeds tarball into the json upload\n      // https://github.com/isaacs/npm-registry-client/commit/e9fbeb8b67f249394f735c74ef11fe4720d46ca0\n      // issue https://github.com/rlidwka/sinopia/issues/31, dealing with it here:\n      const isInvalidBodyFormat =\n        isObject(_attachments) === false ||\n        hasDiffOneKey(_attachments) ||\n        isObject(versions) === false ||\n        hasDiffOneKey(versions);\n\n      if (isInvalidBodyFormat) {\n        // npm is doing something strange again\n        // if this happens in normal circumstances, report it as a bug\n        logger.info({ packageName }, `wrong package format on publish a package @{packageName}`);\n        return next(ErrorCode.getBadRequest(API_ERROR.UNSUPORTED_REGISTRY_CALL));\n      }\n\n      if (error && error.status !== HTTP_STATUS.CONFLICT) {\n        return next(error);\n      }\n\n      // at this point document is either created or existed before\n      const [firstAttachmentKey] = Object.keys(_attachments);\n\n      createTarball(\n        Path.basename(firstAttachmentKey),\n        _attachments[firstAttachmentKey],\n        function (error) {\n          if (error) {\n            return next(error);\n          }\n\n          const versionToPublish = Object.keys(versions)[0];\n          const versionMetadataToPublish = versions[versionToPublish];\n\n          versionMetadataToPublish.readme =\n            _.isNil(versionMetadataToPublish.readme) === false\n              ? String(versionMetadataToPublish.readme)\n              : '';\n\n          createVersion(versionToPublish, versionMetadataToPublish, function (error) {\n            if (error) {\n              return next(error);\n            }\n\n            addTags(metadataCopy[DIST_TAGS], async function (error) {\n              if (error) {\n                return next(error);\n              }\n\n              try {\n                await notify(\n                  metadataCopy,\n                  config,\n                  req.remote_user,\n                  `${metadataCopy.name}@${versionToPublish}`\n                );\n              } catch (error) {\n                logger.error({ error }, 'notify batch service has failed: @{error}');\n              }\n\n              res.status(HTTP_STATUS.CREATED);\n              return next({ ok: okMessage, success: true });\n            });\n          });\n        }\n      );\n    };\n\n    if (isPublishablePackage(req.body) === false && isObject(req.body.users)) {\n      return starApi(req, res, next);\n    }\n\n    try {\n      const metadata = validationUtils.normalizeMetadata(req.body, packageName);\n      // check _attachments to distinguish publish and deprecate\n      if (\n        req.params._rev ||\n        (isRelatedToDeprecation(req.body) && _.isEmpty(req.body._attachments))\n      ) {\n        debug('updating a new version for %o', packageName);\n        // we check unpublish permissions, an update is basically remove versions\n        const remote = req.remote_user;\n        auth.allow_unpublish({ packageName }, remote, (error) => {\n          if (error) {\n            logger.error({ packageName }, `not allowed to unpublish a version for @{packageName}`);\n            return next(error);\n          }\n          storage.changePackage(packageName, metadata, req.params.revision, function (error) {\n            afterChange(error, API_MESSAGE.PKG_CHANGED, metadata);\n          });\n        });\n      } else {\n        debug('adding a new version for %o', packageName);\n        storage.addPackage(packageName, metadata, function (error) {\n          afterChange(error, API_MESSAGE.PKG_CREATED, metadata);\n        });\n      }\n    } catch (error: any) {\n      debug('error on publish: %s', error.message);\n      logger.error({ packageName }, 'error on publish, bad package data for @{packageName}');\n      return next(ErrorCode.getBadData(API_ERROR.BAD_PACKAGE_DATA));\n    }\n  };\n}\n\n/**\n * un-publish a package\n */\nexport function unPublishPackage(storage: Storage) {\n  return function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    const packageName = req.params.package;\n    debug('unpublishing %o', packageName);\n    storage.removePackage(packageName, function (err) {\n      if (err) {\n        return next(err);\n      }\n      res.status(HTTP_STATUS.CREATED);\n      return next({ ok: API_MESSAGE.PKG_REMOVED });\n    });\n  };\n}\n\n/**\n * Delete tarball\n */\nexport function removeTarball(storage: Storage) {\n  return function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    const packageName = req.params.package;\n    const { filename, revision } = req.params;\n    debug('removing a tarball for %o-%o-%o', packageName, filename, revision);\n    storage.removeTarball(packageName, filename, revision, function (err) {\n      if (err) {\n        return next(err);\n      }\n      res.status(HTTP_STATUS.CREATED);\n      debug('success remove tarball for %o-%o-%o', packageName, filename, revision);\n      return next({ ok: API_MESSAGE.TARBALL_REMOVED });\n    });\n  };\n}\n/**\n * Adds a new version\n */\nexport function addVersion(storage: Storage) {\n  return function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    const { version, tag } = req.params;\n    const packageName = req.params.package;\n\n    storage.addVersion(packageName, version, req.body, tag, function (error) {\n      if (error) {\n        return next(error);\n      }\n\n      res.status(HTTP_STATUS.CREATED);\n      return next({\n        ok: API_MESSAGE.PKG_PUBLISHED,\n      });\n    });\n  };\n}\n\n/**\n * uploadPackageTarball\n */\nexport function uploadPackageTarball(storage: Storage) {\n  return function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    const packageName = req.params.package;\n    const stream = storage.addTarball(packageName, req.params.filename);\n    req.pipe(stream);\n\n    // checking if end event came before closing\n    let complete = false;\n    req.on('end', function () {\n      complete = true;\n      stream.done();\n    });\n\n    req.on('close', function () {\n      if (!complete) {\n        stream.abort();\n      }\n    });\n\n    stream.on('error', function (err) {\n      return res.locals.report_error(err);\n    });\n\n    stream.on('success', function () {\n      res.status(HTTP_STATUS.CREATED);\n      return next({\n        ok: API_MESSAGE.TARBALL_UPLOADED,\n      });\n    });\n  };\n}\n"],"mappings":";;;;;;;;;;;AAAA,IAAAA,MAAA,GAAAC,sBAAA,CAAAC,OAAA;AAEA,IAAAC,OAAA,GAAAF,sBAAA,CAAAC,OAAA;AACA,IAAAE,KAAA,GAAAH,sBAAA,CAAAC,OAAA;AACA,IAAAG,KAAA,GAAAJ,sBAAA,CAAAC,OAAA;AAEA,IAAAI,KAAA,GAAAJ,OAAA;AACA,IAAAK,MAAA,GAAAL,OAAA;AACA,IAAAM,WAAA,GAAAN,OAAA;AAIA,IAAAO,UAAA,GAAAP,OAAA;AACA,IAAAQ,OAAA,GAAAR,OAAA;AAEA,IAAAS,aAAA,GAAAT,OAAA;AACA,IAAAU,MAAA,GAAAV,OAAA;AAEA,IAAAW,KAAA,GAAAZ,sBAAA,CAAAC,OAAA;AAA0B,SAAAD,uBAAAa,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAE1B,MAAMG,KAAK,GAAG,IAAAC,cAAU,EAAC,mBAAmB,CAAC;AAE9B,SAASC,OAAOA,CAC7BC,MAAc,EACdC,IAAU,EACVC,OAAgB,EAChBC,MAAc,EACR;EACN,MAAMC,GAAG,GAAG,IAAAC,iBAAK,EAACJ,IAAI,EAAE;IACtBK,SAAS,EAAEA,CAACC,MAAM,EAAEC,OAAO,KAAKC,cAAM,CAACC,KAAK,CAACH,MAAM,EAAEC,OAAO,CAAC;IAC7DG,QAAQ,EAAEA,CAACJ,MAAM,EAAEC,OAAO,KAAKC,cAAM,CAACC,KAAK,CAACH,MAAM,EAAEC,OAAO;EAC7D,CAAC,CAAC;;EAEF;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACER,MAAM,CAACY,GAAG,CACR,6BAA6B,EAC7BR,GAAG,CAAC,SAAS,CAAC,EACd,IAAAS,iBAAK,EAACC,aAAI,CAACC,OAAO,CAAC,MAAM,CAAC,CAAC,EAC3BC,sBAAU,EACVC,cAAc,CAACf,OAAO,EAAEC,MAAM,EAAEF,IAAI,CACtC,CAAC;;EAED;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACED,MAAM,CAACkB,MAAM,CAAC,kBAAkB,EAAEd,GAAG,CAAC,WAAW,CAAC,EAAEe,gBAAgB,CAACjB,OAAO,CAAC,CAAC;;EAE9E;EACAF,MAAM,CAACkB,MAAM,CACX,sCAAsC,EACtCd,GAAG,CAAC,WAAW,CAAC,EAChBA,GAAG,CAAC,SAAS,CAAC,EACdgB,aAAa,CAAClB,OAAO,CACvB,CAAC;;EAED;EACAF,MAAM,CAACY,GAAG,CACR,yBAAyB,EACzBR,GAAG,CAAC,SAAS,CAAC,EACd,IAAAS,iBAAK,EAACQ,kBAAO,CAACC,YAAY,CAAC,EAC3BC,oBAAoB,CAACrB,OAAO,CAC9B,CAAC;;EAED;EACA,IAAIC,MAAM,CAACvB,MAAM,EAAE;IACjB;IACAoB,MAAM,CAACY,GAAG,CACR,8BAA8B,EAC9BR,GAAG,CAAC,SAAS,CAAC,EACd,IAAAS,iBAAK,EAACC,aAAI,CAACC,OAAO,CAAC,MAAM,CAAC,CAAC,EAC3BC,sBAAU,EACVQ,UAAU,CAACtB,OAAO,CACpB,CAAC;EACH;AACF;;AAEA;AACA;AACA;AACO,SAASe,cAAcA,CAACf,OAAgB,EAAEC,MAAc,EAAEF,IAAU,EAAO;EAChF,MAAMwB,OAAO,GAAG,IAAAC,aAAI,EAACxB,OAAO,CAAC;EAC7B,OAAO,UAAUyB,GAAmB,EAAEC,GAAoB,EAAEC,IAAsB,EAAQ;IACxF,MAAMC,WAAW,GAAGH,GAAG,CAACpB,MAAM,CAACwB,OAAO;IACtClC,KAAK,CAAC,6CAA6C,EAAEiC,WAAW,CAAC;IACjE;AACJ;AACA;IACI,MAAME,aAAa,GAAG,SAAAA,CAAUC,QAAgB,EAAEC,IAAI,EAAEC,EAAY,EAAQ;MAC1E,MAAMC,MAAM,GAAGlC,OAAO,CAACmC,UAAU,CAACP,WAAW,EAAEG,QAAQ,CAAC;MACxDG,MAAM,CAACE,EAAE,CAAC,OAAO,EAAE,UAAUC,GAAG,EAAE;QAChCJ,EAAE,CAACI,GAAG,CAAC;MACT,CAAC,CAAC;MACFH,MAAM,CAACE,EAAE,CAAC,SAAS,EAAE,YAAY;QAC/BH,EAAE,CAAC,CAAC;MACN,CAAC,CAAC;MACF;MACA;MACAC,MAAM,CAACI,GAAG,CAACC,MAAM,CAACC,IAAI,CAACR,IAAI,CAACA,IAAI,EAAE,QAAQ,CAAC,CAAC;MAC5CE,MAAM,CAACO,IAAI,CAAC,CAAC;IACf,CAAC;;IAED;AACJ;AACA;IACI,MAAMC,aAAa,GAAG,SAAAA,CAAUC,OAAe,EAAEC,QAAiB,EAAEX,EAAY,EAAQ;MACtFjC,OAAO,CAACsB,UAAU,CAACM,WAAW,EAAEe,OAAO,EAAEC,QAAQ,EAAE,IAAI,EAAEX,EAAE,CAAC;IAC9D,CAAC;;IAED;AACJ;AACA;IACI,MAAMY,OAAO,GAAG,SAAAA,CAAUC,IAAe,EAAEb,EAAY,EAAQ;MAC7DjC,OAAO,CAAC+C,SAAS,CAACnB,WAAW,EAAEkB,IAAI,EAAEb,EAAE,CAAC;IAC1C,CAAC;IAED,MAAMe,WAAW,GAAG,SAAAA,CAAUC,KAAK,EAAEC,SAAS,EAAEN,QAAQ,EAAQ;MAC9D,MAAMO,YAAqB,GAAG;QAAE,GAAGP;MAAS,CAAC;MAE7C,MAAM;QAAEQ,YAAY;QAAEC;MAAS,CAAC,GAAGF,YAAY;;MAE/C;MACA;MACA,IAAIG,eAAC,CAACC,KAAK,CAACH,YAAY,CAAC,IAAII,IAAI,CAACC,SAAS,CAACL,YAAY,CAAC,KAAK,IAAI,EAAE;QAClE,IAAIH,KAAK,EAAE;UACT,OAAOtB,IAAI,CAACsB,KAAK,CAAC;QACpB;QACAvB,GAAG,CAACgC,MAAM,CAACC,sBAAW,CAACC,OAAO,CAAC;QAC/B,OAAOjC,IAAI,CAAC;UACVkC,EAAE,EAAEX,SAAS;UACbY,OAAO,EAAE;QACX,CAAC,CAAC;MACJ;;MAEA;MACA;MACA;MACA,MAAMC,mBAAmB,GACvB,IAAAC,eAAQ,EAACZ,YAAY,CAAC,KAAK,KAAK,IAChC,IAAAa,oBAAa,EAACb,YAAY,CAAC,IAC3B,IAAAY,eAAQ,EAACX,QAAQ,CAAC,KAAK,KAAK,IAC5B,IAAAY,oBAAa,EAACZ,QAAQ,CAAC;MAEzB,IAAIU,mBAAmB,EAAE;QACvB;QACA;QACAxD,cAAM,CAAC2D,IAAI,CAAC;UAAEtC;QAAY,CAAC,EAAE,0DAA0D,CAAC;QACxF,OAAOD,IAAI,CAACwC,gBAAS,CAACC,aAAa,CAACC,oBAAS,CAACC,wBAAwB,CAAC,CAAC;MAC1E;MAEA,IAAIrB,KAAK,IAAIA,KAAK,CAACS,MAAM,KAAKC,sBAAW,CAACY,QAAQ,EAAE;QAClD,OAAO5C,IAAI,CAACsB,KAAK,CAAC;MACpB;;MAEA;MACA,MAAM,CAACuB,kBAAkB,CAAC,GAAGC,MAAM,CAACC,IAAI,CAACtB,YAAY,CAAC;MAEtDtB,aAAa,CACX6C,aAAI,CAACC,QAAQ,CAACJ,kBAAkB,CAAC,EACjCpB,YAAY,CAACoB,kBAAkB,CAAC,EAChC,UAAUvB,KAAK,EAAE;QACf,IAAIA,KAAK,EAAE;UACT,OAAOtB,IAAI,CAACsB,KAAK,CAAC;QACpB;QAEA,MAAM4B,gBAAgB,GAAGJ,MAAM,CAACC,IAAI,CAACrB,QAAQ,CAAC,CAAC,CAAC,CAAC;QACjD,MAAMyB,wBAAwB,GAAGzB,QAAQ,CAACwB,gBAAgB,CAAC;QAE3DC,wBAAwB,CAACC,MAAM,GAC7BzB,eAAC,CAACC,KAAK,CAACuB,wBAAwB,CAACC,MAAM,CAAC,KAAK,KAAK,GAC9CC,MAAM,CAACF,wBAAwB,CAACC,MAAM,CAAC,GACvC,EAAE;QAERrC,aAAa,CAACmC,gBAAgB,EAAEC,wBAAwB,EAAE,UAAU7B,KAAK,EAAE;UACzE,IAAIA,KAAK,EAAE;YACT,OAAOtB,IAAI,CAACsB,KAAK,CAAC;UACpB;UAEAJ,OAAO,CAACM,YAAY,CAAC8B,oBAAS,CAAC,EAAE,gBAAgBhC,KAAK,EAAE;YACtD,IAAIA,KAAK,EAAE;cACT,OAAOtB,IAAI,CAACsB,KAAK,CAAC;YACpB;YAEA,IAAI;cACF,MAAM,IAAAiC,aAAM,EACV/B,YAAY,EACZlD,MAAM,EACNwB,GAAG,CAAC0D,WAAW,EACf,GAAGhC,YAAY,CAACiC,IAAI,IAAIP,gBAAgB,EAC1C,CAAC;YACH,CAAC,CAAC,OAAO5B,KAAK,EAAE;cACd1C,cAAM,CAAC0C,KAAK,CAAC;gBAAEA;cAAM,CAAC,EAAE,2CAA2C,CAAC;YACtE;YAEAvB,GAAG,CAACgC,MAAM,CAACC,sBAAW,CAACC,OAAO,CAAC;YAC/B,OAAOjC,IAAI,CAAC;cAAEkC,EAAE,EAAEX,SAAS;cAAEY,OAAO,EAAE;YAAK,CAAC,CAAC;UAC/C,CAAC,CAAC;QACJ,CAAC,CAAC;MACJ,CACF,CAAC;IACH,CAAC;IAED,IAAI,IAAAuB,kCAAoB,EAAC5D,GAAG,CAAC6D,IAAI,CAAC,KAAK,KAAK,IAAI,IAAAtB,eAAQ,EAACvC,GAAG,CAAC6D,IAAI,CAACC,KAAK,CAAC,EAAE;MACxE,OAAOhE,OAAO,CAACE,GAAG,EAAEC,GAAG,EAAEC,IAAI,CAAC;IAChC;IAEA,IAAI;MACF,MAAMiB,QAAQ,GAAG4C,qBAAe,CAACC,iBAAiB,CAAChE,GAAG,CAAC6D,IAAI,EAAE1D,WAAW,CAAC;MACzE;MACA,IACEH,GAAG,CAACpB,MAAM,CAACqF,IAAI,IACd,IAAAC,6BAAsB,EAAClE,GAAG,CAAC6D,IAAI,CAAC,IAAIhC,eAAC,CAACsC,OAAO,CAACnE,GAAG,CAAC6D,IAAI,CAAClC,YAAY,CAAE,EACtE;QACAzD,KAAK,CAAC,+BAA+B,EAAEiC,WAAW,CAAC;QACnD;QACA,MAAMiE,MAAM,GAAGpE,GAAG,CAAC0D,WAAW;QAC9BpF,IAAI,CAAC+F,eAAe,CAAC;UAAElE;QAAY,CAAC,EAAEiE,MAAM,EAAG5C,KAAK,IAAK;UACvD,IAAIA,KAAK,EAAE;YACT1C,cAAM,CAAC0C,KAAK,CAAC;cAAErB;YAAY,CAAC,EAAE,uDAAuD,CAAC;YACtF,OAAOD,IAAI,CAACsB,KAAK,CAAC;UACpB;UACAjD,OAAO,CAAC+F,aAAa,CAACnE,WAAW,EAAEgB,QAAQ,EAAEnB,GAAG,CAACpB,MAAM,CAAC2F,QAAQ,EAAE,UAAU/C,KAAK,EAAE;YACjFD,WAAW,CAACC,KAAK,EAAEgD,sBAAW,CAACC,WAAW,EAAEtD,QAAQ,CAAC;UACvD,CAAC,CAAC;QACJ,CAAC,CAAC;MACJ,CAAC,MAAM;QACLjD,KAAK,CAAC,6BAA6B,EAAEiC,WAAW,CAAC;QACjD5B,OAAO,CAACmG,UAAU,CAACvE,WAAW,EAAEgB,QAAQ,EAAE,UAAUK,KAAK,EAAE;UACzDD,WAAW,CAACC,KAAK,EAAEgD,sBAAW,CAACG,WAAW,EAAExD,QAAQ,CAAC;QACvD,CAAC,CAAC;MACJ;IACF,CAAC,CAAC,OAAOK,KAAU,EAAE;MACnBtD,KAAK,CAAC,sBAAsB,EAAEsD,KAAK,CAAC3C,OAAO,CAAC;MAC5CC,cAAM,CAAC0C,KAAK,CAAC;QAAErB;MAAY,CAAC,EAAE,uDAAuD,CAAC;MACtF,OAAOD,IAAI,CAACwC,gBAAS,CAACkC,UAAU,CAAChC,oBAAS,CAACiC,gBAAgB,CAAC,CAAC;IAC/D;EACF,CAAC;AACH;;AAEA;AACA;AACA;AACO,SAASrF,gBAAgBA,CAACjB,OAAgB,EAAE;EACjD,OAAO,UAAUyB,GAAmB,EAAEC,GAAoB,EAAEC,IAAsB,EAAQ;IACxF,MAAMC,WAAW,GAAGH,GAAG,CAACpB,MAAM,CAACwB,OAAO;IACtClC,KAAK,CAAC,iBAAiB,EAAEiC,WAAW,CAAC;IACrC5B,OAAO,CAACuG,aAAa,CAAC3E,WAAW,EAAE,UAAUS,GAAG,EAAE;MAChD,IAAIA,GAAG,EAAE;QACP,OAAOV,IAAI,CAACU,GAAG,CAAC;MAClB;MACAX,GAAG,CAACgC,MAAM,CAACC,sBAAW,CAACC,OAAO,CAAC;MAC/B,OAAOjC,IAAI,CAAC;QAAEkC,EAAE,EAAEoC,sBAAW,CAACO;MAAY,CAAC,CAAC;IAC9C,CAAC,CAAC;EACJ,CAAC;AACH;;AAEA;AACA;AACA;AACO,SAAStF,aAAaA,CAAClB,OAAgB,EAAE;EAC9C,OAAO,UAAUyB,GAAmB,EAAEC,GAAoB,EAAEC,IAAsB,EAAQ;IACxF,MAAMC,WAAW,GAAGH,GAAG,CAACpB,MAAM,CAACwB,OAAO;IACtC,MAAM;MAAEE,QAAQ;MAAEiE;IAAS,CAAC,GAAGvE,GAAG,CAACpB,MAAM;IACzCV,KAAK,CAAC,iCAAiC,EAAEiC,WAAW,EAAEG,QAAQ,EAAEiE,QAAQ,CAAC;IACzEhG,OAAO,CAACkB,aAAa,CAACU,WAAW,EAAEG,QAAQ,EAAEiE,QAAQ,EAAE,UAAU3D,GAAG,EAAE;MACpE,IAAIA,GAAG,EAAE;QACP,OAAOV,IAAI,CAACU,GAAG,CAAC;MAClB;MACAX,GAAG,CAACgC,MAAM,CAACC,sBAAW,CAACC,OAAO,CAAC;MAC/BjE,KAAK,CAAC,qCAAqC,EAAEiC,WAAW,EAAEG,QAAQ,EAAEiE,QAAQ,CAAC;MAC7E,OAAOrE,IAAI,CAAC;QAAEkC,EAAE,EAAEoC,sBAAW,CAACQ;MAAgB,CAAC,CAAC;IAClD,CAAC,CAAC;EACJ,CAAC;AACH;AACA;AACA;AACA;AACO,SAASnF,UAAUA,CAACtB,OAAgB,EAAE;EAC3C,OAAO,UAAUyB,GAAmB,EAAEC,GAAoB,EAAEC,IAAsB,EAAQ;IACxF,MAAM;MAAEgB,OAAO;MAAE+D;IAAI,CAAC,GAAGjF,GAAG,CAACpB,MAAM;IACnC,MAAMuB,WAAW,GAAGH,GAAG,CAACpB,MAAM,CAACwB,OAAO;IAEtC7B,OAAO,CAACsB,UAAU,CAACM,WAAW,EAAEe,OAAO,EAAElB,GAAG,CAAC6D,IAAI,EAAEoB,GAAG,EAAE,UAAUzD,KAAK,EAAE;MACvE,IAAIA,KAAK,EAAE;QACT,OAAOtB,IAAI,CAACsB,KAAK,CAAC;MACpB;MAEAvB,GAAG,CAACgC,MAAM,CAACC,sBAAW,CAACC,OAAO,CAAC;MAC/B,OAAOjC,IAAI,CAAC;QACVkC,EAAE,EAAEoC,sBAAW,CAACU;MAClB,CAAC,CAAC;IACJ,CAAC,CAAC;EACJ,CAAC;AACH;;AAEA;AACA;AACA;AACO,SAAStF,oBAAoBA,CAACrB,OAAgB,EAAE;EACrD,OAAO,UAAUyB,GAAmB,EAAEC,GAAoB,EAAEC,IAAsB,EAAQ;IACxF,MAAMC,WAAW,GAAGH,GAAG,CAACpB,MAAM,CAACwB,OAAO;IACtC,MAAMK,MAAM,GAAGlC,OAAO,CAACmC,UAAU,CAACP,WAAW,EAAEH,GAAG,CAACpB,MAAM,CAAC0B,QAAQ,CAAC;IACnEN,GAAG,CAACmF,IAAI,CAAC1E,MAAM,CAAC;;IAEhB;IACA,IAAI2E,QAAQ,GAAG,KAAK;IACpBpF,GAAG,CAACW,EAAE,CAAC,KAAK,EAAE,YAAY;MACxByE,QAAQ,GAAG,IAAI;MACf3E,MAAM,CAACO,IAAI,CAAC,CAAC;IACf,CAAC,CAAC;IAEFhB,GAAG,CAACW,EAAE,CAAC,OAAO,EAAE,YAAY;MAC1B,IAAI,CAACyE,QAAQ,EAAE;QACb3E,MAAM,CAAC4E,KAAK,CAAC,CAAC;MAChB;IACF,CAAC,CAAC;IAEF5E,MAAM,CAACE,EAAE,CAAC,OAAO,EAAE,UAAUC,GAAG,EAAE;MAChC,OAAOX,GAAG,CAACqF,MAAM,CAACC,YAAY,CAAC3E,GAAG,CAAC;IACrC,CAAC,CAAC;IAEFH,MAAM,CAACE,EAAE,CAAC,SAAS,EAAE,YAAY;MAC/BV,GAAG,CAACgC,MAAM,CAACC,sBAAW,CAACC,OAAO,CAAC;MAC/B,OAAOjC,IAAI,CAAC;QACVkC,EAAE,EAAEoC,sBAAW,CAACgB;MAClB,CAAC,CAAC;IACJ,CAAC,CAAC;EACJ,CAAC;AACH","ignoreList":[]}
359
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_debug","_interopRequireDefault","require","_lodash","_mime","_path","_core","_hooks","_middleware","_constants","_logger","_storageUtils","_utils","_star","e","__esModule","default","debug","buildDebug","publish","router","auth","storage","config","can","allow","beforeAll","params","message","logger","trace","afterAll","put","media","mime","getType","expectJson","publishPackage","delete","unPublishPackage","removeTarball","HEADERS","OCTET_STREAM","uploadPackageTarball","addVersion","starApi","star","req","res","next","packageName","package","createTarball","filename","data","cb","stream","addTarball","on","err","end","Buffer","from","done","createVersion","version","metadata","addTags","tags","mergeTags","afterChange","error","okMessage","metadataCopy","_attachments","versions","_","isNil","JSON","stringify","status","HTTP_STATUS","CREATED","ok","success","isInvalidBodyFormat","isObject","hasDiffOneKey","info","ErrorCode","getBadRequest","API_ERROR","UNSUPORTED_REGISTRY_CALL","CONFLICT","firstAttachmentKey","Object","keys","Path","basename","versionToPublish","versionMetadataToPublish","readme","String","DIST_TAGS","notify","remote_user","name","isPublishablePackage","body","users","validationUtils","normalizeMetadata","_rev","isRelatedToDeprecation","isEmpty","remote","allow_unpublish","changePackage","revision","API_MESSAGE","PKG_CHANGED","addPackage","PKG_CREATED","getBadData","BAD_PACKAGE_DATA","removePackage","PKG_REMOVED","TARBALL_REMOVED","tag","PKG_PUBLISHED","pipe","complete","abort","locals","report_error","TARBALL_UPLOADED"],"sources":["../../../../src/api/endpoint/api/publish.ts"],"sourcesContent":["import buildDebug from 'debug';\nimport type { Router } from 'express';\nimport _ from 'lodash';\nimport mime from 'mime';\nimport Path from 'path';\n\nimport type { Auth } from '@verdaccio/auth';\nimport { validationUtils } from '@verdaccio/core';\nimport { notify } from '@verdaccio/hooks';\nimport { allow, expectJson, media } from '@verdaccio/middleware';\nimport type { Callback, Config, MergeTags, Package, Version } from '@verdaccio/types';\n\nimport { API_ERROR, API_MESSAGE, DIST_TAGS, HEADERS, HTTP_STATUS } from '../../../lib/constants';\nimport { logger } from '../../../lib/logger';\nimport type Storage from '../../../lib/storage';\nimport { isPublishablePackage } from '../../../lib/storage-utils';\nimport { ErrorCode, hasDiffOneKey, isObject, isRelatedToDeprecation } from '../../../lib/utils';\nimport type { $NextFunctionVer, $RequestExtend, $ResponseExtend } from '../../../types';\nimport star from './star';\n\nconst debug = buildDebug('verdaccio:publish');\n\nexport default function publish(\n  router: Router,\n  auth: Auth,\n  storage: Storage,\n  config: Config\n): void {\n  const can = allow(auth, {\n    beforeAll: (params, message) => logger.trace(params, message),\n    afterAll: (params, message) => logger.trace(params, message),\n  });\n\n  /**\n   * Publish a package / update package / un/start a package\n   *\n   * There are multiples scenarios here to be considered:\n   *\n   * 1. Publish scenario\n   *\n   * Publish a package consist of at least 1 step (PUT) with a metadata payload.\n   * When a package is published, an _attachment property is present that contains the data\n   * of the tarball.\n   *\n   * Example flow of publish.\n   *\n   *  npm http fetch PUT 201 http://localhost:4873/@scope%2ftest1 9627ms\n      npm info lifecycle @scope/test1@1.0.1~publish: @scope/test1@1.0.1\n      npm info lifecycle @scope/test1@1.0.1~postpublish: @scope/test1@1.0.1\n      + @scope/test1@1.0.1\n      npm verb exit [ 0, true ]\n   *\n   *\n   * 2. Unpublish scenario\n   *\n   * Unpublish consist in 3 steps.\n   *  1. Try to fetch  metadata -> if it fails, return 404\n   *  2. Compute metadata locally (client side) and send a mutate payload excluding the version to be unpublished\n   *    eg: if metadata reflects 1.0.1, 1.0.2 and 1.0.3, the computed metadata won't include 1.0.3.\n   *  3. Once the second step has been successfully finished, delete the tarball.\n   *\n   *  All these steps are consecutive and required, there is no transacions here, if step 3 fails, metadata might\n   *  get corrupted.\n   *\n   *  Note the unpublish call will suffix in the url a /-rev/14-5d500cfce92f90fd revision number, this not\n   *  used internally.\n   *\n   *\n   * Example flow of unpublish.\n   *\n   * npm http fetch GET 200 http://localhost:4873/@scope%2ftest1?write=true 1680ms\n     npm http fetch PUT 201 http://localhost:4873/@scope%2ftest1/-rev/14-5d500cfce92f90fd 956606ms attempt #2\n     npm http fetch GET 200 http://localhost:4873/@scope%2ftest1?write=true 1601ms\n     npm http fetch DELETE 201 http://localhost:4873/@scope%2ftest1/-/test1-1.0.3.tgz/-rev/16-e11c8db282b2d992 19ms\n   *\n   * 3. Star a package\n   *\n   * Permissions: start a package depends of the publish and unpublish permissions, there is no specific flag for star or un start.\n   * The URL for star is similar to the unpublish (change package format)\n   *\n   * npm has no enpoint for star a package, rather mutate the metadata and acts as, the difference is the\n   * users property which is part of the payload and the body only includes\n   *\n   * {\n\t\t  \"_id\": pkgName,\n\t  \t\"_rev\": \"3-b0cdaefc9bdb77c8\",\n\t\t  \"users\": {\n\t\t    [username]: boolean value (true, false)\n\t\t  }\n\t}\n   *\n   */\n  router.put(\n    '/:package/:_rev?/:revision?',\n    can('publish'),\n    media(mime.getType('json')),\n    expectJson,\n    publishPackage(storage, config, auth)\n  );\n\n  /**\n   * Un-publishing an entire package.\n   *\n   * This scenario happens when the first call detect there is only one version remaining\n   * in the metadata, then the client decides to DELETE the resource\n   * npm http fetch GET 304 http://localhost:4873/@scope%2ftest1?write=true 1076ms (from cache)\n     npm http fetch DELETE 201 http://localhost:4873/@scope%2ftest1/-rev/18-d8ebe3020bd4ac9c 22ms\n   */\n  router.delete('/:package/-rev/*', can('unpublish'), unPublishPackage(storage));\n\n  // removing a tarball\n  router.delete(\n    '/:package/-/:filename/-rev/:revision',\n    can('unpublish'),\n    can('publish'),\n    removeTarball(storage)\n  );\n\n  // uploading package tarball\n  router.put(\n    '/:package/-/:filename/*',\n    can('publish'),\n    media(HEADERS.OCTET_STREAM),\n    uploadPackageTarball(storage)\n  );\n\n  // only used for development\n  if (config._debug) {\n    // adding a version\n    router.put(\n      '/:package/:version/-tag/:tag',\n      can('publish'),\n      media(mime.getType('json')),\n      expectJson,\n      addVersion(storage)\n    );\n  }\n}\n\n/**\n * Publish a package\n */\nexport function publishPackage(storage: Storage, config: Config, auth: Auth): any {\n  const starApi = star(storage);\n  return function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    const packageName = req.params.package;\n    debug('publishing or updating a new version for %o', packageName);\n    /**\n     * Write tarball of stream data from package clients.\n     */\n    const createTarball = function (filename: string, data, cb: Callback): void {\n      const stream = storage.addTarball(packageName, filename);\n      stream.on('error', function (err) {\n        cb(err);\n      });\n      stream.on('success', function () {\n        cb();\n      });\n      // this is dumb and memory-consuming, but what choices do we have?\n      // flow: we need first refactor this file before decides which type use here\n      stream.end(Buffer.from(data.data, 'base64'));\n      stream.done();\n    };\n\n    /**\n     * Add new package version in storage\n     */\n    const createVersion = function (version: string, metadata: Version, cb: Callback): void {\n      storage.addVersion(packageName, version, metadata, null, cb);\n    };\n\n    /**\n     * Add new tags in storage\n     */\n    const addTags = function (tags: MergeTags, cb: Callback): void {\n      storage.mergeTags(packageName, tags, cb);\n    };\n\n    const afterChange = function (error, okMessage, metadata): void {\n      const metadataCopy: Package = { ...metadata };\n\n      const { _attachments, versions } = metadataCopy;\n\n      // `npm star` wouldn't have attachments\n      // and `npm deprecate` would have attachments as a empty object, i.e {}\n      if (_.isNil(_attachments) || JSON.stringify(_attachments) === '{}') {\n        if (error) {\n          return next(error);\n        }\n        res.status(HTTP_STATUS.CREATED);\n        return next({\n          ok: okMessage,\n          success: true,\n        });\n      }\n\n      // npm-registry-client 0.3+ embeds tarball into the json upload\n      // https://github.com/isaacs/npm-registry-client/commit/e9fbeb8b67f249394f735c74ef11fe4720d46ca0\n      // issue https://github.com/rlidwka/sinopia/issues/31, dealing with it here:\n      const isInvalidBodyFormat =\n        isObject(_attachments) === false ||\n        hasDiffOneKey(_attachments) ||\n        isObject(versions) === false ||\n        hasDiffOneKey(versions);\n\n      if (isInvalidBodyFormat) {\n        // npm is doing something strange again\n        // if this happens in normal circumstances, report it as a bug\n        logger.info({ packageName }, `wrong package format on publish a package @{packageName}`);\n        return next(ErrorCode.getBadRequest(API_ERROR.UNSUPORTED_REGISTRY_CALL));\n      }\n\n      if (error && error.status !== HTTP_STATUS.CONFLICT) {\n        return next(error);\n      }\n\n      // at this point document is either created or existed before\n      const [firstAttachmentKey] = Object.keys(_attachments);\n\n      createTarball(\n        Path.basename(firstAttachmentKey),\n        _attachments[firstAttachmentKey],\n        function (error) {\n          if (error) {\n            return next(error);\n          }\n\n          const versionToPublish = Object.keys(versions)[0];\n          const versionMetadataToPublish = versions[versionToPublish];\n\n          versionMetadataToPublish.readme =\n            _.isNil(versionMetadataToPublish.readme) === false\n              ? String(versionMetadataToPublish.readme)\n              : '';\n\n          createVersion(versionToPublish, versionMetadataToPublish, function (error) {\n            if (error) {\n              return next(error);\n            }\n\n            addTags(metadataCopy[DIST_TAGS], async function (error) {\n              if (error) {\n                return next(error);\n              }\n\n              try {\n                await notify(\n                  metadataCopy,\n                  config,\n                  req.remote_user,\n                  `${metadataCopy.name}@${versionToPublish}`\n                );\n              } catch (error) {\n                logger.error({ error }, 'notify batch service has failed: @{error}');\n              }\n\n              res.status(HTTP_STATUS.CREATED);\n              return next({ ok: okMessage, success: true });\n            });\n          });\n        }\n      );\n    };\n\n    if (isPublishablePackage(req.body) === false && isObject(req.body.users)) {\n      return starApi(req, res, next);\n    }\n\n    try {\n      const metadata = validationUtils.normalizeMetadata(req.body, packageName);\n      // check _attachments to distinguish publish and deprecate\n      if (\n        req.params._rev ||\n        (isRelatedToDeprecation(req.body) && _.isEmpty(req.body._attachments))\n      ) {\n        debug('updating a new version for %o', packageName);\n        // we check unpublish permissions, an update is basically remove versions\n        const remote = req.remote_user;\n        auth.allow_unpublish({ packageName }, remote, (error) => {\n          if (error) {\n            logger.error({ packageName }, `not allowed to unpublish a version for @{packageName}`);\n            return next(error);\n          }\n          storage.changePackage(packageName, metadata, req.params.revision, function (error) {\n            afterChange(error, API_MESSAGE.PKG_CHANGED, metadata);\n          });\n        });\n      } else {\n        debug('adding a new version for %o', packageName);\n        storage.addPackage(packageName, metadata, function (error) {\n          afterChange(error, API_MESSAGE.PKG_CREATED, metadata);\n        });\n      }\n    } catch (error: any) {\n      debug('error on publish: %s', error.message);\n      logger.error({ packageName }, 'error on publish, bad package data for @{packageName}');\n      return next(ErrorCode.getBadData(API_ERROR.BAD_PACKAGE_DATA));\n    }\n  };\n}\n\n/**\n * un-publish a package\n */\nexport function unPublishPackage(storage: Storage) {\n  return function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    const packageName = req.params.package;\n    debug('unpublishing %o', packageName);\n    storage.removePackage(packageName, function (err) {\n      if (err) {\n        return next(err);\n      }\n      res.status(HTTP_STATUS.CREATED);\n      return next({ ok: API_MESSAGE.PKG_REMOVED });\n    });\n  };\n}\n\n/**\n * Delete tarball\n */\nexport function removeTarball(storage: Storage) {\n  return function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    const packageName = req.params.package;\n    const { filename, revision } = req.params;\n    debug('removing a tarball for %o-%o-%o', packageName, filename, revision);\n    storage.removeTarball(packageName, filename, revision, function (err) {\n      if (err) {\n        return next(err);\n      }\n      res.status(HTTP_STATUS.CREATED);\n      debug('success remove tarball for %o-%o-%o', packageName, filename, revision);\n      return next({ ok: API_MESSAGE.TARBALL_REMOVED });\n    });\n  };\n}\n/**\n * Adds a new version\n */\nexport function addVersion(storage: Storage) {\n  return function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    const { version, tag } = req.params;\n    const packageName = req.params.package;\n\n    storage.addVersion(packageName, version, req.body, tag, function (error) {\n      if (error) {\n        return next(error);\n      }\n\n      res.status(HTTP_STATUS.CREATED);\n      return next({\n        ok: API_MESSAGE.PKG_PUBLISHED,\n      });\n    });\n  };\n}\n\n/**\n * uploadPackageTarball\n */\nexport function uploadPackageTarball(storage: Storage) {\n  return function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    const packageName = req.params.package;\n    const stream = storage.addTarball(packageName, req.params.filename);\n    req.pipe(stream);\n\n    // checking if end event came before closing\n    let complete = false;\n    req.on('end', function () {\n      complete = true;\n      stream.done();\n    });\n\n    req.on('close', function () {\n      if (!complete) {\n        stream.abort();\n      }\n    });\n\n    stream.on('error', function (err) {\n      return res.locals.report_error(err);\n    });\n\n    stream.on('success', function () {\n      res.status(HTTP_STATUS.CREATED);\n      return next({\n        ok: API_MESSAGE.TARBALL_UPLOADED,\n      });\n    });\n  };\n}\n"],"mappings":"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","ignoreList":[]}
@@ -1,4 +1,4 @@
1
1
  import type { Router } from 'express';
2
+ import type { Auth } from '@verdaccio/auth';
2
3
  import type { Config } from '@verdaccio/types';
3
- import type Auth from '../../../lib/auth';
4
4
  export default function (route: Router, auth: Auth, config: Config): void;
@@ -81,4 +81,4 @@ function _default(route, auth, config) {
81
81
  });
82
82
  route.use(userRouter);
83
83
  }
84
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_express","_interopRequireDefault","require","_lodash","_auth","_config","_core","_middleware","_utils","_constants","_logger","_utils2","e","__esModule","default","_default","route","auth","config","userRouter","express","Router","get","rateLimit","userRateLimit","req","res","next","status","HTTP_STATUS","OK","ok","getAuthenticatedMessage","remote_user","name","put","password","body","remoteName","_","isNil","authenticate","callbackAuthenticate","err","user","logger","error","ErrorCode","getCode","UNAUTHORIZED","API_ERROR","BAD_USERNAME_PASSWORD","restoredRemoteUser","createRemoteUser","groups","token","getApiToken","CREATED","set","HEADERS","CACHE_CONTROL","validationUtils","validatePassword","BAD_REQUEST","PASSWORD_SHORT","add_user","INTERNAL_ERROR","message","getConflict","undefined","delete","API_MESSAGE","LOGGED_OUT","use"],"sources":["../../../../src/api/endpoint/api/user.ts"],"sourcesContent":["import type { Response, Router } from 'express';\nimport express from 'express';\nimport _ from 'lodash';\n\nimport { getApiToken } from '@verdaccio/auth';\nimport { createRemoteUser } from '@verdaccio/config';\nimport { validationUtils } from '@verdaccio/core';\nimport { rateLimit } from '@verdaccio/middleware';\nimport type { Config, RemoteUser } from '@verdaccio/types';\nimport { getAuthenticatedMessage } from '@verdaccio/utils';\n\nimport type Auth from '../../../lib/auth';\nimport { API_ERROR, API_MESSAGE, HEADERS, HTTP_STATUS } from '../../../lib/constants';\nimport { logger } from '../../../lib/logger';\nimport { ErrorCode } from '../../../lib/utils';\nimport type { $NextFunctionVer, $RequestExtend } from '../../../types';\n\nexport default function (route: Router, auth: Auth, config: Config): void {\n  /* eslint new-cap:off */\n  const userRouter = express.Router();\n\n  userRouter.get(\n    '/-/user/:org_couchdb_user',\n    rateLimit(config?.userRateLimit),\n    function (req: $RequestExtend, res: Response, next: $NextFunctionVer): void {\n      res.status(HTTP_STATUS.OK);\n      next({\n        ok: getAuthenticatedMessage(req.remote_user.name),\n      });\n    }\n  );\n\n  userRouter.put(\n    '/-/user/:org_couchdb_user/:_rev?/:revision?',\n    rateLimit(config?.userRateLimit),\n    function (req: $RequestExtend, res: Response, next: $NextFunctionVer): void {\n      const { name, password } = req.body;\n      const remoteName = req.remote_user.name;\n\n      if (_.isNil(remoteName) === false && _.isNil(name) === false && remoteName === name) {\n        auth.authenticate(\n          name,\n          password,\n          async function callbackAuthenticate(err, user): Promise<void> {\n            if (err) {\n              logger.error(\n                { name, err },\n                'authenticating for user @{username} failed. Error: @{err.message}'\n              );\n              return next(\n                ErrorCode.getCode(HTTP_STATUS.UNAUTHORIZED, API_ERROR.BAD_USERNAME_PASSWORD)\n              );\n            }\n\n            const restoredRemoteUser: RemoteUser = createRemoteUser(name, user?.groups ?? []);\n            const token = await getApiToken(auth, config, restoredRemoteUser, password);\n\n            res.status(HTTP_STATUS.CREATED);\n            res.set(HEADERS.CACHE_CONTROL, 'no-cache, no-store');\n            return next({\n              ok: getAuthenticatedMessage(req.remote_user.name),\n              token,\n            });\n          }\n        );\n      } else {\n        if (validationUtils.validatePassword(password) === false) {\n          return next(ErrorCode.getCode(HTTP_STATUS.BAD_REQUEST, API_ERROR.PASSWORD_SHORT));\n        }\n\n        auth.add_user(name, password, async function (err, user): Promise<void> {\n          if (err) {\n            if (err.status >= HTTP_STATUS.BAD_REQUEST && err.status < HTTP_STATUS.INTERNAL_ERROR) {\n              // With npm registering is the same as logging in,\n              // and npm accepts only an 409 error.\n              // So, changing status code here.\n              return next(\n                ErrorCode.getCode(err.status, err.message) || ErrorCode.getConflict(err.message)\n              );\n            }\n            return next(err);\n          }\n\n          const token =\n            name && password\n              ? await getApiToken(auth, config, user as RemoteUser, password)\n              : undefined;\n\n          req.remote_user = user;\n          res.status(HTTP_STATUS.CREATED);\n          res.set(HEADERS.CACHE_CONTROL, 'no-cache, no-store');\n          return next({\n            ok: `user '${req.body.name}' created`,\n            token,\n          });\n        });\n      }\n    }\n  );\n\n  userRouter.delete(\n    '/-/user/token/*',\n    rateLimit(config?.userRateLimit),\n    function (req: $RequestExtend, res: Response, next: $NextFunctionVer): void {\n      res.status(HTTP_STATUS.OK);\n      next({\n        ok: API_MESSAGE.LOGGED_OUT,\n      });\n    }\n  );\n\n  route.use(userRouter);\n}\n"],"mappings":"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","ignoreList":[]}
84
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_express","_interopRequireDefault","require","_lodash","_auth","_config","_core","_middleware","_utils","_constants","_logger","_utils2","e","__esModule","default","_default","route","auth","config","userRouter","express","Router","get","rateLimit","userRateLimit","req","res","next","status","HTTP_STATUS","OK","ok","getAuthenticatedMessage","remote_user","name","put","password","body","remoteName","_","isNil","authenticate","callbackAuthenticate","err","user","logger","error","ErrorCode","getCode","UNAUTHORIZED","API_ERROR","BAD_USERNAME_PASSWORD","restoredRemoteUser","createRemoteUser","groups","token","getApiToken","CREATED","set","HEADERS","CACHE_CONTROL","validationUtils","validatePassword","BAD_REQUEST","PASSWORD_SHORT","add_user","INTERNAL_ERROR","message","getConflict","undefined","delete","API_MESSAGE","LOGGED_OUT","use"],"sources":["../../../../src/api/endpoint/api/user.ts"],"sourcesContent":["import type { Response, Router } from 'express';\nimport express from 'express';\nimport _ from 'lodash';\n\nimport { getApiToken } from '@verdaccio/auth';\nimport type { Auth } from '@verdaccio/auth';\nimport { createRemoteUser } from '@verdaccio/config';\nimport { validationUtils } from '@verdaccio/core';\nimport { rateLimit } from '@verdaccio/middleware';\nimport type { Config, RemoteUser } from '@verdaccio/types';\nimport { getAuthenticatedMessage } from '@verdaccio/utils';\n\nimport { API_ERROR, API_MESSAGE, HEADERS, HTTP_STATUS } from '../../../lib/constants';\nimport { logger } from '../../../lib/logger';\nimport { ErrorCode } from '../../../lib/utils';\nimport type { $NextFunctionVer, $RequestExtend } from '../../../types';\n\nexport default function (route: Router, auth: Auth, config: Config): void {\n  /* eslint new-cap:off */\n  const userRouter = express.Router();\n\n  userRouter.get(\n    '/-/user/:org_couchdb_user',\n    rateLimit(config?.userRateLimit),\n    function (req: $RequestExtend, res: Response, next: $NextFunctionVer): void {\n      res.status(HTTP_STATUS.OK);\n      next({\n        ok: getAuthenticatedMessage(req.remote_user.name),\n      });\n    }\n  );\n\n  userRouter.put(\n    '/-/user/:org_couchdb_user/:_rev?/:revision?',\n    rateLimit(config?.userRateLimit),\n    function (req: $RequestExtend, res: Response, next: $NextFunctionVer): void {\n      const { name, password } = req.body;\n      const remoteName = req.remote_user.name;\n\n      if (_.isNil(remoteName) === false && _.isNil(name) === false && remoteName === name) {\n        auth.authenticate(\n          name,\n          password,\n          async function callbackAuthenticate(err, user): Promise<void> {\n            if (err) {\n              logger.error(\n                { name, err },\n                'authenticating for user @{username} failed. Error: @{err.message}'\n              );\n              return next(\n                ErrorCode.getCode(HTTP_STATUS.UNAUTHORIZED, API_ERROR.BAD_USERNAME_PASSWORD)\n              );\n            }\n\n            const restoredRemoteUser: RemoteUser = createRemoteUser(name, user?.groups ?? []);\n            const token = await getApiToken(auth, config, restoredRemoteUser, password);\n\n            res.status(HTTP_STATUS.CREATED);\n            res.set(HEADERS.CACHE_CONTROL, 'no-cache, no-store');\n            return next({\n              ok: getAuthenticatedMessage(req.remote_user.name),\n              token,\n            });\n          }\n        );\n      } else {\n        if (validationUtils.validatePassword(password) === false) {\n          return next(ErrorCode.getCode(HTTP_STATUS.BAD_REQUEST, API_ERROR.PASSWORD_SHORT));\n        }\n\n        auth.add_user(name, password, async function (err, user): Promise<void> {\n          if (err) {\n            if (err.status >= HTTP_STATUS.BAD_REQUEST && err.status < HTTP_STATUS.INTERNAL_ERROR) {\n              // With npm registering is the same as logging in,\n              // and npm accepts only an 409 error.\n              // So, changing status code here.\n              return next(\n                ErrorCode.getCode(err.status, err.message) || ErrorCode.getConflict(err.message)\n              );\n            }\n            return next(err);\n          }\n\n          const token =\n            name && password\n              ? await getApiToken(auth, config, user as RemoteUser, password)\n              : undefined;\n\n          req.remote_user = user;\n          res.status(HTTP_STATUS.CREATED);\n          res.set(HEADERS.CACHE_CONTROL, 'no-cache, no-store');\n          return next({\n            ok: `user '${req.body.name}' created`,\n            token,\n          });\n        });\n      }\n    }\n  );\n\n  userRouter.delete(\n    '/-/user/token/*',\n    rateLimit(config?.userRateLimit),\n    function (req: $RequestExtend, res: Response, next: $NextFunctionVer): void {\n      res.status(HTTP_STATUS.OK);\n      next({\n        ok: API_MESSAGE.LOGGED_OUT,\n      });\n    }\n  );\n\n  route.use(userRouter);\n}\n"],"mappings":"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","ignoreList":[]}
@@ -1,6 +1,6 @@
1
1
  import type { Router } from 'express';
2
+ import type { Auth } from '@verdaccio/auth';
2
3
  import type { ConfigYaml } from '@verdaccio/types';
3
- import type Auth from '../../../../lib/auth';
4
4
  export interface Profile {
5
5
  tfa: boolean;
6
6
  name: string;