verdaccio 5.2.3 → 5.4.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/CHANGELOG.md +41 -2
- package/README.md +6 -5
- package/build/api/endpoint/api/publish.js +3 -3
- package/build/api/endpoint/api/user.js +15 -5
- package/build/api/endpoint/api/v1/index.js +26 -0
- package/build/api/endpoint/api/v1/profile.js +12 -4
- package/build/api/endpoint/api/v1/token.js +14 -5
- package/build/api/endpoint/index.js +4 -7
- package/build/api/index.js +11 -5
- package/build/api/rate-limiter.js +24 -0
- package/build/api/web/api.js +4 -16
- package/build/api/web/endpoint/index.js +45 -0
- package/build/api/web/endpoint/package.js +15 -6
- package/build/api/web/endpoint/search.js +8 -2
- package/build/api/web/endpoint/user.js +9 -3
- package/build/api/web/index.js +5 -2
- package/build/lib/auth-utils.js +11 -4
- package/build/lib/config.js +15 -6
- package/build/lib/constants.js +5 -4
- package/build/lib/up-storage.js +6 -3
- package/build/lib/utils.js +14 -4
- package/conf/default.yaml +3 -1
- package/conf/docker.yaml +3 -1
- package/package.json +23 -17
package/CHANGELOG.md
CHANGED
|
@@ -2,12 +2,51 @@
|
|
|
2
2
|
|
|
3
3
|
All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
|
|
4
4
|
|
|
5
|
-
|
|
5
|
+
## [5.4.0](https://github.com/verdaccio/verdaccio/compare/v5.3.2...v5.4.0) (2021-12-24)
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
### Features
|
|
9
|
+
|
|
10
|
+
* add cache-control header to endpoints ([#2791](https://github.com/verdaccio/verdaccio/issues/2791)) ([fd3ad1e](https://github.com/verdaccio/verdaccio/commit/fd3ad1e546f5293e862d767f23b3714e6dd5dc8c))
|
|
11
|
+
* add rate limit to user api endpoints ([#2800](https://github.com/verdaccio/verdaccio/issues/2800)) ([#2799](https://github.com/verdaccio/verdaccio/issues/2799)) ([f64e403](https://github.com/verdaccio/verdaccio/commit/f64e403f0a1d710a0d866ca2db480199c83a3609))
|
|
12
|
+
* add rate limit to web endpoints ([#2799](https://github.com/verdaccio/verdaccio/issues/2799)) ([c91d6be](https://github.com/verdaccio/verdaccio/commit/c91d6beb8bde1c5f275d7f570209cf723dce47c5))
|
|
13
|
+
* allow override user rate limit and user agent ([#2803](https://github.com/verdaccio/verdaccio/issues/2803)) ([5b1264c](https://github.com/verdaccio/verdaccio/commit/5b1264c733ea249efc38e963f0e99fb093d61406))
|
|
14
|
+
* jwt default time new set ([#2787](https://github.com/verdaccio/verdaccio/issues/2787)) ([1c17d09](https://github.com/verdaccio/verdaccio/commit/1c17d0986ea551b427994f8392a9d8ef29ba88e9))
|
|
15
|
+
* user agent is the client by default ([#2793](https://github.com/verdaccio/verdaccio/issues/2793)) ([e5d79ce](https://github.com/verdaccio/verdaccio/commit/e5d79ce8f00a18f0c0646b761f22dc608b5baf87))
|
|
6
16
|
|
|
7
17
|
|
|
8
18
|
### Bug Fixes
|
|
9
19
|
|
|
10
|
-
* update
|
|
20
|
+
* **deps:** update dependency @verdaccio/commons-api to v10.1.0 ([#2789](https://github.com/verdaccio/verdaccio/issues/2789)) ([c5c0ef8](https://github.com/verdaccio/verdaccio/commit/c5c0ef8750abeb50d933c639a2041d25a889820b))
|
|
21
|
+
* **deps:** update dependency body-parser to v1.19.1 ([#2788](https://github.com/verdaccio/verdaccio/issues/2788)) ([46e583f](https://github.com/verdaccio/verdaccio/commit/46e583f683520b3a69dbd14ed43146492c0e80be))
|
|
22
|
+
|
|
23
|
+
### [5.3.2](https://github.com/verdaccio/verdaccio/compare/v5.3.1...v5.3.2) (2021-12-16)
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
### Bug Fixes
|
|
27
|
+
|
|
28
|
+
* publish with deprecated field ([#2771](https://github.com/verdaccio/verdaccio/issues/2771)) ([b5ac3c0](https://github.com/verdaccio/verdaccio/commit/b5ac3c0f287b8d9ae9121a47d6ac9d27e2bd4ce6)), closes [#2766](https://github.com/verdaccio/verdaccio/issues/2766)
|
|
29
|
+
|
|
30
|
+
### [5.3.1](https://github.com/verdaccio/verdaccio/compare/v5.3.0...v5.3.1) (2021-12-11)
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
### Bug Fixes
|
|
34
|
+
|
|
35
|
+
* conflict core-js dependency ([#2754](https://github.com/verdaccio/verdaccio/issues/2754)) ([29d4551](https://github.com/verdaccio/verdaccio/commit/29d4551e5bfc066a5ff7109786a73ec8e782a65f))
|
|
36
|
+
|
|
37
|
+
## [5.3.0](https://github.com/verdaccio/verdaccio/compare/v5.2.2...v5.3.0) (2021-12-08)
|
|
38
|
+
|
|
39
|
+
|
|
40
|
+
### Bug Fixes
|
|
41
|
+
|
|
42
|
+
* markdown anchor links in package readme.md broken [#2712](https://github.com/verdaccio/verdaccio/issues/2712) ([820c9d5](https://github.com/verdaccio/verdaccio/commit/820c9d5db2fe7e99cab3c75c431b8df7dfd096d0))
|
|
43
|
+
* update pnpm and bum up core deps ([#2716](https://github.com/verdaccio/verdaccio/issues/2716)) ([60cd25c](https://github.com/verdaccio/verdaccio/commit/60cd25ce4fe107f7d25780447289b6fb0e21d4b6))
|
|
44
|
+
* **deps:** update dependency @verdaccio/commons-api to v10.0.2 ([#2706](https://github.com/verdaccio/verdaccio/issues/2706)) ([73d565f](https://github.com/verdaccio/verdaccio/commit/73d565f5f558b7746f538d8fbbb15ea24dcedb23))
|
|
45
|
+
* **deps:** update dependency @verdaccio/readme to v10.0.1 ([#2708](https://github.com/verdaccio/verdaccio/issues/2708)) ([41cadcc](https://github.com/verdaccio/verdaccio/commit/41cadccaef358bb11f4b2f015b0885f32dffa91a))
|
|
46
|
+
* **deps:** update dependency @verdaccio/streams to v10.0.1 ([#2709](https://github.com/verdaccio/verdaccio/issues/2709)) ([74d7d02](https://github.com/verdaccio/verdaccio/commit/74d7d02c396185f76767f234592eb14f497d7929))
|
|
47
|
+
* **deps:** update dependency http-errors to v1.8.1 ([#2705](https://github.com/verdaccio/verdaccio/issues/2705)) ([eee19a5](https://github.com/verdaccio/verdaccio/commit/eee19a5f791dd93f4ee078243ed310fa88aa5ae7))
|
|
48
|
+
* **deps:** update dependency verdaccio-audit to v10.0.3 ([#2710](https://github.com/verdaccio/verdaccio/issues/2710)) ([8c074e8](https://github.com/verdaccio/verdaccio/commit/8c074e8f24e61e9c399bf4f2e6dbaa0d5d5da9fd))
|
|
49
|
+
* **deps:** update dependency verdaccio-htpasswd to v10.0.1 ([#2711](https://github.com/verdaccio/verdaccio/issues/2711)) ([8e28848](https://github.com/verdaccio/verdaccio/commit/8e2884836b4d8a659ca1cec80794c7e30ef6c208))
|
|
11
50
|
|
|
12
51
|
### [5.2.2](https://github.com/verdaccio/verdaccio/compare/v5.2.1...v5.2.2) (2021-11-08)
|
|
13
52
|
|
package/README.md
CHANGED
|
@@ -69,15 +69,16 @@ booted in a couple of seconds, fast enough for any CI. Many open source projects
|
|
|
69
69
|
|
|
70
70
|
## Talks
|
|
71
71
|
|
|
72
|
-
### **
|
|
72
|
+
### **Using Docker and Verdaccio to make Integration Testing Easy - Docker All Hands #4 December - 2021**.
|
|
73
73
|
|
|
74
|
-
[](https://www.youtube.com/watch?v=zRI0skF1f8I)
|
|
75
75
|
|
|
76
76
|
You might want to check out as well our previous talks:
|
|
77
77
|
|
|
78
|
-
- [
|
|
79
|
-
- [
|
|
80
|
-
- [
|
|
78
|
+
- [**Juan Picado** – Testing the integrity of React components by publishing in a private registry](https://www.youtube.com/watch?v=bRKZbrlQqLY&t=16s&ab_channel=ReactFinland)
|
|
79
|
+
- [BeerJS Cba Meetup No. 53 May 2021 - **Juan Picado** React Finland - 2021](https://www.youtube.com/watch?v=6SyjqBmS49Y&ab_channel=BeerJSCba)
|
|
80
|
+
- [Node.js Dependency Confusion Attacks - April 2021 - **Juan Picado**](https://www.youtube.com/watch?v=qTRADSp3Hpo)
|
|
81
|
+
- [**OpenJS World 2020** about \*Cover your Projects with a Multi purpose Lightweight Node.js Registry - **Juan Picado**](https://www.youtube.com/watch?v=oVCjDWeehAQ)
|
|
81
82
|
- [ViennaJS Meetup - Introduction to Verdaccio by **Priscila Olivera** and **Juan Picado**](https://www.youtube.com/watch?v=hDIFKzmoCa)
|
|
82
83
|
- [Open Source? trivago - Verdaccio (**Ayush** and **Juan Picado**) January 2020](https://www.youtube.com/watch?v=A5CWxJC9xzc)
|
|
83
84
|
- [GitNation Open Source Stage - How we have built a Node.js Registry with React - **Juan Picado** December 2019](https://www.youtube.com/watch?v=gpjC8Qp9B9A)
|
|
@@ -249,9 +249,9 @@ function publishPackage(storage, config, auth) {
|
|
|
249
249
|
}
|
|
250
250
|
|
|
251
251
|
try {
|
|
252
|
-
const metadata = (0, _utils.validateMetadata)(req.body, packageName); //
|
|
252
|
+
const metadata = (0, _utils.validateMetadata)(req.body, packageName); // check _attachments to distinguish publish and deprecate
|
|
253
253
|
|
|
254
|
-
if (req.params._rev || (0, _utils.isRelatedToDeprecation)(req.body)) {
|
|
254
|
+
if (req.params._rev || (0, _utils.isRelatedToDeprecation)(req.body) && _lodash.default.isEmpty(req.body._attachments)) {
|
|
255
255
|
debug('updating a new version for %o', packageName); // we check unpublish permissions, an update is basically remove versions
|
|
256
256
|
|
|
257
257
|
const remote = req.remote_user;
|
|
@@ -388,4 +388,4 @@ function uploadPackageTarball(storage) {
|
|
|
388
388
|
});
|
|
389
389
|
};
|
|
390
390
|
}
|
|
391
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/api/endpoint/api/publish.ts"],"names":["debug","publish","router","auth","storage","config","can","put","mime","getType","expectJson","publishPackage","delete","unPublishPackage","removeTarball","HEADERS","OCTET_STREAM","uploadPackageTarball","addVersion","starApi","req","res","next","packageName","params","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","logger","info","ErrorCode","getBadRequest","API_ERROR","UNSUPORTED_REGISTRY_CALL","CONFLICT","firstAttachmentKey","Object","keys","Path","basename","versionToPublish","versionMetadataToPublish","readme","String","DIST_TAGS","remote_user","name","body","users","_rev","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"],"mappings":";;;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AAIA;;AACA;;AACA;;AACA;;AAGA;;AACA;;AACA;;;;;;;;;;AAEA,MAAMA,KAAK,GAAG,oBAAW,mBAAX,CAAd;;AAEe,SAASC,OAAT,CAAiBC,MAAjB,EAAiCC,IAAjC,EAA8CC,OAA9C,EAAwEC,MAAxE,EAA8F;AAC3G,QAAMC,GAAG,GAAG,uBAAMH,IAAN,CAAZ;AAEA;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;;AACED,EAAAA,MAAM,CAACK,GAAP,CAAW,6BAAX,EAA0CD,GAAG,CAAC,SAAD,CAA7C,EAA0D,uBAAME,cAAKC,OAAL,CAAa,MAAb,CAAN,CAA1D,EAAuFC,sBAAvF,EAAmGC,cAAc,CAACP,OAAD,EAAUC,MAAV,EAAkBF,IAAlB,CAAjH;AAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;;AACED,EAAAA,MAAM,CAACU,MAAP,CAAc,kBAAd,EAAkCN,GAAG,CAAC,WAAD,CAArC,EAAoDO,gBAAgB,CAACT,OAAD,CAApE,EAxE2G,CA0E3G;;AACAF,EAAAA,MAAM,CAACU,MAAP,CAAc,sCAAd,EAAsDN,GAAG,CAAC,WAAD,CAAzD,EAAwEA,GAAG,CAAC,SAAD,CAA3E,EAAwFQ,aAAa,CAACV,OAAD,CAArG,EA3E2G,CA6E3G;;AACAF,EAAAA,MAAM,CAACK,GAAP,CAAW,yBAAX,EAAsCD,GAAG,CAAC,SAAD,CAAzC,EAAsD,uBAAMS,mBAAQC,YAAd,CAAtD,EAAmFC,oBAAoB,CAACb,OAAD,CAAvG,EA9E2G,CAgF3G;;AACAF,EAAAA,MAAM,CAACK,GAAP,CAAW,8BAAX,EAA2CD,GAAG,CAAC,SAAD,CAA9C,EAA2D,uBAAME,cAAKC,OAAL,CAAa,MAAb,CAAN,CAA3D,EAAwFC,sBAAxF,EAAoGQ,UAAU,CAACd,OAAD,CAA9G;AACD;AAED;AACA;AACA;;;AACO,SAASO,cAAT,CAAwBP,OAAxB,EAAkDC,MAAlD,EAAkEF,IAAlE,EAAoF;AACzF,QAAMgB,OAAO,GAAG,mBAAKf,OAAL,CAAhB;AACA,SAAO,UAAUgB,GAAV,EAA+BC,GAA/B,EAAqDC,IAArD,EAAmF;AACxF,UAAMC,WAAW,GAAGH,GAAG,CAACI,MAAJ,CAAWC,OAA/B;AACAzB,IAAAA,KAAK,CAAC,6CAAD,EAAgDuB,WAAhD,CAAL;AACA;AACJ;AACA;;AACI,UAAMG,aAAa,GAAG,UAAUC,QAAV,EAA4BC,IAA5B,EAAkCC,EAAlC,EAAsD;AAC1E,YAAMC,MAAM,GAAG1B,OAAO,CAAC2B,UAAR,CAAmBR,WAAnB,EAAgCI,QAAhC,CAAf;AACAG,MAAAA,MAAM,CAACE,EAAP,CAAU,OAAV,EAAmB,UAAUC,GAAV,EAAe;AAChCJ,QAAAA,EAAE,CAACI,GAAD,CAAF;AACD,OAFD;AAGAH,MAAAA,MAAM,CAACE,EAAP,CAAU,SAAV,EAAqB,YAAY;AAC/BH,QAAAA,EAAE;AACH,OAFD,EAL0E,CAQ1E;AACA;;AACAC,MAAAA,MAAM,CAACI,GAAP,CAAWC,MAAM,CAACC,IAAP,CAAYR,IAAI,CAACA,IAAjB,EAAuB,QAAvB,CAAX;AACAE,MAAAA,MAAM,CAACO,IAAP;AACD,KAZD;AAcA;AACJ;AACA;;;AACI,UAAMC,aAAa,GAAG,UAAUC,OAAV,EAA2BC,QAA3B,EAA8CX,EAA9C,EAAkE;AACtFzB,MAAAA,OAAO,CAACc,UAAR,CAAmBK,WAAnB,EAAgCgB,OAAhC,EAAyCC,QAAzC,EAAmD,IAAnD,EAAyDX,EAAzD;AACD,KAFD;AAIA;AACJ;AACA;;;AACI,UAAMY,OAAO,GAAG,UAAUC,IAAV,EAA2Bb,EAA3B,EAA+C;AAC7DzB,MAAAA,OAAO,CAACuC,SAAR,CAAkBpB,WAAlB,EAA+BmB,IAA/B,EAAqCb,EAArC;AACD,KAFD;;AAIA,UAAMe,WAAW,GAAG,UAAUC,KAAV,EAAiBC,SAAjB,EAA4BN,QAA5B,EAA4C;AAC9D,YAAMO,YAAqB,qBAAQP,QAAR,CAA3B;;AAEA,YAAM;AAAEQ,QAAAA,YAAF;AAAgBC,QAAAA;AAAhB,UAA6BF,YAAnC,CAH8D,CAK9D;AACA;;AACA,UAAIG,gBAAEC,KAAF,CAAQH,YAAR,KAAyBI,IAAI,CAACC,SAAL,CAAeL,YAAf,MAAiC,IAA9D,EAAoE;AAClE,YAAIH,KAAJ,EAAW;AACT,iBAAOvB,IAAI,CAACuB,KAAD,CAAX;AACD;;AACDxB,QAAAA,GAAG,CAACiC,MAAJ,CAAWC,uBAAYC,OAAvB;AACA,eAAOlC,IAAI,CAAC;AACVmC,UAAAA,EAAE,EAAEX,SADM;AAEVY,UAAAA,OAAO,EAAE;AAFC,SAAD,CAAX;AAID,OAhB6D,CAkB9D;AACA;AACA;;;AACA,YAAMC,mBAAmB,GAAG,qBAASX,YAAT,MAA2B,KAA3B,IAAoC,0BAAcA,YAAd,CAApC,IAAmE,qBAASC,QAAT,MAAuB,KAA1F,IAAmG,0BAAcA,QAAd,CAA/H;;AAEA,UAAIU,mBAAJ,EAAyB;AACvB;AACA;AACAC,uBAAOC,IAAP,CAAY;AAAEtC,UAAAA;AAAF,SAAZ,EAA8B,0DAA9B;;AACA,eAAOD,IAAI,CAACwC,iBAAUC,aAAV,CAAwBC,qBAAUC,wBAAlC,CAAD,CAAX;AACD;;AAED,UAAIpB,KAAK,IAAIA,KAAK,CAACS,MAAN,KAAiBC,uBAAYW,QAA1C,EAAoD;AAClD,eAAO5C,IAAI,CAACuB,KAAD,CAAX;AACD,OAhC6D,CAkC9D;;;AACA,YAAM,CAACsB,kBAAD,IAAuBC,MAAM,CAACC,IAAP,CAAYrB,YAAZ,CAA7B;AAEAtB,MAAAA,aAAa,CAAC4C,cAAKC,QAAL,CAAcJ,kBAAd,CAAD,EAAoCnB,YAAY,CAACmB,kBAAD,CAAhD,EAAsE,UAAUtB,KAAV,EAAiB;AAClG,YAAIA,KAAJ,EAAW;AACT,iBAAOvB,IAAI,CAACuB,KAAD,CAAX;AACD;;AAED,cAAM2B,gBAAgB,GAAGJ,MAAM,CAACC,IAAP,CAAYpB,QAAZ,EAAsB,CAAtB,CAAzB;AACA,cAAMwB,wBAAwB,GAAGxB,QAAQ,CAACuB,gBAAD,CAAzC;AAEAC,QAAAA,wBAAwB,CAACC,MAAzB,GAAkCxB,gBAAEC,KAAF,CAAQsB,wBAAwB,CAACC,MAAjC,MAA6C,KAA7C,GAAqDC,MAAM,CAACF,wBAAwB,CAACC,MAA1B,CAA3D,GAA+F,EAAjI;AAEApC,QAAAA,aAAa,CAACkC,gBAAD,EAAmBC,wBAAnB,EAA6C,UAAU5B,KAAV,EAAiB;AACzE,cAAIA,KAAJ,EAAW;AACT,mBAAOvB,IAAI,CAACuB,KAAD,CAAX;AACD;;AAEDJ,UAAAA,OAAO,CAACM,YAAY,CAAC6B,oBAAD,CAAb,EAA0B,gBAAgB/B,KAAhB,EAAuB;AACtD,gBAAIA,KAAJ,EAAW;AACT,qBAAOvB,IAAI,CAACuB,KAAD,CAAX;AACD;;AAED,gBAAI;AACF,oBAAM,oBAAOE,YAAP,EAAqB1C,MAArB,EAA6Be,GAAG,CAACyD,WAAjC,EAA+C,GAAE9B,YAAY,CAAC+B,IAAK,IAAGN,gBAAiB,EAAvF,CAAN;AACD,aAFD,CAEE,OAAO3B,KAAP,EAAc;AACde,6BAAOf,KAAP,CAAa;AAAEA,gBAAAA;AAAF,eAAb,EAAwB,2CAAxB;AACD;;AAEDxB,YAAAA,GAAG,CAACiC,MAAJ,CAAWC,uBAAYC,OAAvB;AACA,mBAAOlC,IAAI,CAAC;AAAEmC,cAAAA,EAAE,EAAEX,SAAN;AAAiBY,cAAAA,OAAO,EAAE;AAA1B,aAAD,CAAX;AACD,WAbM,CAAP;AAcD,SAnBY,CAAb;AAoBD,OA9BY,CAAb;AA+BD,KApED;;AAsEA,QAAI,wCAAqBtC,GAAG,CAAC2D,IAAzB,MAAmC,KAAnC,IAA4C,qBAAS3D,GAAG,CAAC2D,IAAJ,CAASC,KAAlB,CAAhD,EAA0E;AACxE,aAAO7D,OAAO,CAACC,GAAD,EAAMC,GAAN,EAAWC,IAAX,CAAd;AACD;;AAED,QAAI;AACF,YAAMkB,QAAQ,GAAG,6BAAiBpB,GAAG,CAAC2D,IAArB,EAA2BxD,WAA3B,CAAjB,CADE,CAEF;;AACA,UAAIH,GAAG,CAACI,MAAJ,CAAWyD,IAAX,IAAmB,mCAAuB7D,GAAG,CAAC2D,IAA3B,CAAvB,EAAyD;AACvD/E,QAAAA,KAAK,CAAC,+BAAD,EAAkCuB,WAAlC,CAAL,CADuD,CAEvD;;AACA,cAAM2D,MAAM,GAAG9D,GAAG,CAACyD,WAAnB;AACA1E,QAAAA,IAAI,CAACgF,eAAL,CAAqB;AAAE5D,UAAAA;AAAF,SAArB,EAAsC2D,MAAtC,EAA+CrC,KAAD,IAAW;AACvD,cAAIA,KAAJ,EAAW;AACTe,2BAAOf,KAAP,CAAa;AAAEtB,cAAAA;AAAF,aAAb,EAA+B,uDAA/B;;AACA,mBAAOD,IAAI,CAACuB,KAAD,CAAX;AACD;;AACDzC,UAAAA,OAAO,CAACgF,aAAR,CAAsB7D,WAAtB,EAAmCiB,QAAnC,EAA6CpB,GAAG,CAACI,MAAJ,CAAW6D,QAAxD,EAAkE,UAAUxC,KAAV,EAAiB;AACjFD,YAAAA,WAAW,CAACC,KAAD,EAAQyC,uBAAYC,WAApB,EAAiC/C,QAAjC,CAAX;AACD,WAFD;AAGD,SARD;AASD,OAbD,MAaO;AACLxC,QAAAA,KAAK,CAAC,6BAAD,EAAgCuB,WAAhC,CAAL;AACAnB,QAAAA,OAAO,CAACoF,UAAR,CAAmBjE,WAAnB,EAAgCiB,QAAhC,EAA0C,UAAUK,KAAV,EAAiB;AACzDD,UAAAA,WAAW,CAACC,KAAD,EAAQyC,uBAAYG,WAApB,EAAiCjD,QAAjC,CAAX;AACD,SAFD;AAGD;AACF,KAtBD,CAsBE,OAAOK,KAAP,EAAc;AACde,qBAAOf,KAAP,CAAa;AAAEtB,QAAAA;AAAF,OAAb,EAA8B,uDAA9B;;AACA,aAAOD,IAAI,CAACwC,iBAAU4B,UAAV,CAAqB1B,qBAAU2B,gBAA/B,CAAD,CAAX;AACD;AACF,GAtID;AAuID;AAED;AACA;AACA;;;AACO,SAAS9E,gBAAT,CAA0BT,OAA1B,EAAoD;AACzD,SAAO,UAAUgB,GAAV,EAA+BC,GAA/B,EAAqDC,IAArD,EAAmF;AACxF,UAAMC,WAAW,GAAGH,GAAG,CAACI,MAAJ,CAAWC,OAA/B;AACAzB,IAAAA,KAAK,CAAC,iBAAD,EAAoBuB,WAApB,CAAL;AACAnB,IAAAA,OAAO,CAACwF,aAAR,CAAsBrE,WAAtB,EAAmC,UAAUU,GAAV,EAAe;AAChD,UAAIA,GAAJ,EAAS;AACP,eAAOX,IAAI,CAACW,GAAD,CAAX;AACD;;AACDZ,MAAAA,GAAG,CAACiC,MAAJ,CAAWC,uBAAYC,OAAvB;AACA,aAAOlC,IAAI,CAAC;AAAEmC,QAAAA,EAAE,EAAE6B,uBAAYO;AAAlB,OAAD,CAAX;AACD,KAND;AAOD,GAVD;AAWD;AAED;AACA;AACA;;;AACO,SAAS/E,aAAT,CAAuBV,OAAvB,EAAiD;AACtD,SAAO,UAAUgB,GAAV,EAA+BC,GAA/B,EAAqDC,IAArD,EAAmF;AACxF,UAAMC,WAAW,GAAGH,GAAG,CAACI,MAAJ,CAAWC,OAA/B;AACA,UAAM;AAAEE,MAAAA,QAAF;AAAY0D,MAAAA;AAAZ,QAAyBjE,GAAG,CAACI,MAAnC;AACAxB,IAAAA,KAAK,CAAC,iCAAD,EAAoCuB,WAApC,EAAiDI,QAAjD,EAA2D0D,QAA3D,CAAL;AACAjF,IAAAA,OAAO,CAACU,aAAR,CAAsBS,WAAtB,EAAmCI,QAAnC,EAA6C0D,QAA7C,EAAuD,UAAUpD,GAAV,EAAe;AACpE,UAAIA,GAAJ,EAAS;AACP,eAAOX,IAAI,CAACW,GAAD,CAAX;AACD;;AACDZ,MAAAA,GAAG,CAACiC,MAAJ,CAAWC,uBAAYC,OAAvB;AACAxD,MAAAA,KAAK,CAAC,qCAAD,EAAwCuB,WAAxC,EAAqDI,QAArD,EAA+D0D,QAA/D,CAAL;AACA,aAAO/D,IAAI,CAAC;AAAEmC,QAAAA,EAAE,EAAE6B,uBAAYQ;AAAlB,OAAD,CAAX;AACD,KAPD;AAQD,GAZD;AAaD;AACD;AACA;AACA;;;AACO,SAAS5E,UAAT,CAAoBd,OAApB,EAA8C;AACnD,SAAO,UAAUgB,GAAV,EAA+BC,GAA/B,EAAqDC,IAArD,EAAmF;AACxF,UAAM;AAAEiB,MAAAA,OAAF;AAAWwD,MAAAA;AAAX,QAAmB3E,GAAG,CAACI,MAA7B;AACA,UAAMD,WAAW,GAAGH,GAAG,CAACI,MAAJ,CAAWC,OAA/B;AAEArB,IAAAA,OAAO,CAACc,UAAR,CAAmBK,WAAnB,EAAgCgB,OAAhC,EAAyCnB,GAAG,CAAC2D,IAA7C,EAAmDgB,GAAnD,EAAwD,UAAUlD,KAAV,EAAiB;AACvE,UAAIA,KAAJ,EAAW;AACT,eAAOvB,IAAI,CAACuB,KAAD,CAAX;AACD;;AAEDxB,MAAAA,GAAG,CAACiC,MAAJ,CAAWC,uBAAYC,OAAvB;AACA,aAAOlC,IAAI,CAAC;AACVmC,QAAAA,EAAE,EAAE6B,uBAAYU;AADN,OAAD,CAAX;AAGD,KATD;AAUD,GAdD;AAeD;AAED;AACA;AACA;;;AACO,SAAS/E,oBAAT,CAA8Bb,OAA9B,EAAwD;AAC7D,SAAO,UAAUgB,GAAV,EAA+BC,GAA/B,EAAqDC,IAArD,EAAmF;AACxF,UAAMC,WAAW,GAAGH,GAAG,CAACI,MAAJ,CAAWC,OAA/B;AACA,UAAMK,MAAM,GAAG1B,OAAO,CAAC2B,UAAR,CAAmBR,WAAnB,EAAgCH,GAAG,CAACI,MAAJ,CAAWG,QAA3C,CAAf;AACAP,IAAAA,GAAG,CAAC6E,IAAJ,CAASnE,MAAT,EAHwF,CAKxF;;AACA,QAAIoE,QAAQ,GAAG,KAAf;AACA9E,IAAAA,GAAG,CAACY,EAAJ,CAAO,KAAP,EAAc,YAAY;AACxBkE,MAAAA,QAAQ,GAAG,IAAX;AACApE,MAAAA,MAAM,CAACO,IAAP;AACD,KAHD;AAKAjB,IAAAA,GAAG,CAACY,EAAJ,CAAO,OAAP,EAAgB,YAAY;AAC1B,UAAI,CAACkE,QAAL,EAAe;AACbpE,QAAAA,MAAM,CAACqE,KAAP;AACD;AACF,KAJD;AAMArE,IAAAA,MAAM,CAACE,EAAP,CAAU,OAAV,EAAmB,UAAUC,GAAV,EAAe;AAChC,aAAOZ,GAAG,CAAC+E,MAAJ,CAAWC,YAAX,CAAwBpE,GAAxB,CAAP;AACD,KAFD;AAIAH,IAAAA,MAAM,CAACE,EAAP,CAAU,SAAV,EAAqB,YAAY;AAC/BX,MAAAA,GAAG,CAACiC,MAAJ,CAAWC,uBAAYC,OAAvB;AACA,aAAOlC,IAAI,CAAC;AACVmC,QAAAA,EAAE,EAAE6B,uBAAYgB;AADN,OAAD,CAAX;AAGD,KALD;AAMD,GA5BD;AA6BD","sourcesContent":["import Path from 'path';\nimport _ from 'lodash';\nimport buildDebug from 'debug';\nimport mime from 'mime';\n\nimport { Router } from 'express';\nimport { Config, Callback, MergeTags, Version, Package } from '@verdaccio/types';\nimport { API_MESSAGE, HEADERS, DIST_TAGS, API_ERROR, HTTP_STATUS } from '../../../lib/constants';\nimport { validateMetadata, isObject, ErrorCode, hasDiffOneKey, isRelatedToDeprecation } from '../../../lib/utils';\nimport { media, expectJson, allow } from '../../middleware';\nimport { notify } from '../../../lib/notify';\n\nimport { IAuth, $ResponseExtend, $RequestExtend, $NextFunctionVer, IStorageHandler } from '../../../../types';\nimport { logger } from '../../../lib/logger';\nimport { isPublishablePackage } from '../../../lib/storage-utils';\nimport star from './star';\n\nconst debug = buildDebug('verdaccio:publish');\n\nexport default function publish(router: Router, auth: IAuth, storage: IStorageHandler, config: Config): void {\n  const can = allow(auth);\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('/:package/:_rev?/:revision?', can('publish'), media(mime.getType('json')), expectJson, publishPackage(storage, config, auth));\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('/:package/-/:filename/-rev/:revision', can('unpublish'), can('publish'), removeTarball(storage));\n\n  // uploading package tarball\n  router.put('/:package/-/:filename/*', can('publish'), media(HEADERS.OCTET_STREAM), uploadPackageTarball(storage));\n\n  // adding a version\n  router.put('/:package/:version/-tag/:tag', can('publish'), media(mime.getType('json')), expectJson, addVersion(storage));\n}\n\n/**\n * Publish a package\n */\nexport function publishPackage(storage: IStorageHandler, config: Config, auth: IAuth): 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 = isObject(_attachments) === false || hasDiffOneKey(_attachments) || isObject(versions) === false || 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(Path.basename(firstAttachmentKey), _attachments[firstAttachmentKey], 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 = _.isNil(versionMetadataToPublish.readme) === false ? String(versionMetadataToPublish.readme) : '';\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(metadataCopy, config, req.remote_user, `${metadataCopy.name}@${versionToPublish}`);\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    if (isPublishablePackage(req.body) === false && isObject(req.body.users)) {\n      return starApi(req, res, next);\n    }\n\n    try {\n      const metadata = validateMetadata(req.body, packageName);\n      // treating deprecation as updating a package\n      if (req.params._rev || isRelatedToDeprecation(req.body)) {\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) {\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: IStorageHandler) {\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: IStorageHandler) {\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: IStorageHandler) {\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: IStorageHandler) {\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"]}
|
|
391
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/api/endpoint/api/publish.ts"],"names":["debug","publish","router","auth","storage","config","can","put","mime","getType","expectJson","publishPackage","delete","unPublishPackage","removeTarball","HEADERS","OCTET_STREAM","uploadPackageTarball","addVersion","starApi","req","res","next","packageName","params","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","logger","info","ErrorCode","getBadRequest","API_ERROR","UNSUPORTED_REGISTRY_CALL","CONFLICT","firstAttachmentKey","Object","keys","Path","basename","versionToPublish","versionMetadataToPublish","readme","String","DIST_TAGS","remote_user","name","body","users","_rev","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"],"mappings":"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","sourcesContent":["import Path from 'path';\nimport _ from 'lodash';\nimport buildDebug from 'debug';\nimport mime from 'mime';\n\nimport { Router } from 'express';\nimport { Config, Callback, MergeTags, Version, Package } from '@verdaccio/types';\nimport { API_MESSAGE, HEADERS, DIST_TAGS, API_ERROR, HTTP_STATUS } from '../../../lib/constants';\nimport { validateMetadata, isObject, ErrorCode, hasDiffOneKey, isRelatedToDeprecation } from '../../../lib/utils';\nimport { media, expectJson, allow } from '../../middleware';\nimport { notify } from '../../../lib/notify';\n\nimport { IAuth, $ResponseExtend, $RequestExtend, $NextFunctionVer, IStorageHandler } from '../../../../types';\nimport { logger } from '../../../lib/logger';\nimport { isPublishablePackage } from '../../../lib/storage-utils';\nimport star from './star';\n\nconst debug = buildDebug('verdaccio:publish');\n\nexport default function publish(router: Router, auth: IAuth, storage: IStorageHandler, config: Config): void {\n  const can = allow(auth);\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('/:package/:_rev?/:revision?', can('publish'), media(mime.getType('json')), expectJson, publishPackage(storage, config, auth));\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('/:package/-/:filename/-rev/:revision', can('unpublish'), can('publish'), removeTarball(storage));\n\n  // uploading package tarball\n  router.put('/:package/-/:filename/*', can('publish'), media(HEADERS.OCTET_STREAM), uploadPackageTarball(storage));\n\n  // adding a version\n  router.put('/:package/:version/-tag/:tag', can('publish'), media(mime.getType('json')), expectJson, addVersion(storage));\n}\n\n/**\n * Publish a package\n */\nexport function publishPackage(storage: IStorageHandler, config: Config, auth: IAuth): 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 = isObject(_attachments) === false || hasDiffOneKey(_attachments) || isObject(versions) === false || 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(Path.basename(firstAttachmentKey), _attachments[firstAttachmentKey], 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 = _.isNil(versionMetadataToPublish.readme) === false ? String(versionMetadataToPublish.readme) : '';\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(metadataCopy, config, req.remote_user, `${metadataCopy.name}@${versionToPublish}`);\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    if (isPublishablePackage(req.body) === false && isObject(req.body.users)) {\n      return starApi(req, res, next);\n    }\n\n    try {\n      const metadata = validateMetadata(req.body, packageName);\n      // check _attachments to distinguish publish and deprecate\n      if (req.params._rev || (isRelatedToDeprecation(req.body) && _.isEmpty(req.body._attachments))) {\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) {\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: IStorageHandler) {\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: IStorageHandler) {\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: IStorageHandler) {\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: IStorageHandler) {\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"]}
|
|
@@ -9,6 +9,8 @@ var _lodash = _interopRequireDefault(require("lodash"));
|
|
|
9
9
|
|
|
10
10
|
var _cookies = _interopRequireDefault(require("cookies"));
|
|
11
11
|
|
|
12
|
+
var _express = _interopRequireDefault(require("express"));
|
|
13
|
+
|
|
12
14
|
var _utils = require("../../../lib/utils");
|
|
13
15
|
|
|
14
16
|
var _constants = require("../../../lib/constants");
|
|
@@ -17,16 +19,21 @@ var _authUtils = require("../../../lib/auth-utils");
|
|
|
17
19
|
|
|
18
20
|
var _logger = require("../../../lib/logger");
|
|
19
21
|
|
|
22
|
+
var _rateLimiter = require("../../rate-limiter");
|
|
23
|
+
|
|
20
24
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
21
25
|
|
|
22
26
|
function _default(route, auth, config) {
|
|
23
|
-
|
|
27
|
+
/* eslint new-cap:off */
|
|
28
|
+
const userRouter = _express.default.Router();
|
|
29
|
+
|
|
30
|
+
userRouter.get('/-/user/:org_couchdb_user', (0, _rateLimiter.limiter)(config === null || config === void 0 ? void 0 : config.userRateLimit), function (req, res, next) {
|
|
24
31
|
res.status(_constants.HTTP_STATUS.OK);
|
|
25
32
|
next({
|
|
26
33
|
ok: (0, _authUtils.getAuthenticatedMessage)(req.remote_user.name)
|
|
27
34
|
});
|
|
28
35
|
});
|
|
29
|
-
|
|
36
|
+
userRouter.put('/-/user/:org_couchdb_user/:_rev?/:revision?', (0, _rateLimiter.limiter)(config === null || config === void 0 ? void 0 : config.userRateLimit), function (req, res, next) {
|
|
30
37
|
const {
|
|
31
38
|
name,
|
|
32
39
|
password
|
|
@@ -47,6 +54,7 @@ function _default(route, auth, config) {
|
|
|
47
54
|
const restoredRemoteUser = (0, _authUtils.createRemoteUser)(name, user.groups || []);
|
|
48
55
|
const token = await (0, _authUtils.getApiToken)(auth, config, restoredRemoteUser, password);
|
|
49
56
|
res.status(_constants.HTTP_STATUS.CREATED);
|
|
57
|
+
res.set(_constants.HEADERS.CACHE_CONTROL, 'no-cache, no-store');
|
|
50
58
|
return next({
|
|
51
59
|
ok: (0, _authUtils.getAuthenticatedMessage)(req.remote_user.name),
|
|
52
60
|
token
|
|
@@ -73,6 +81,7 @@ function _default(route, auth, config) {
|
|
|
73
81
|
const token = name && password ? await (0, _authUtils.getApiToken)(auth, config, user, password) : undefined;
|
|
74
82
|
req.remote_user = user;
|
|
75
83
|
res.status(_constants.HTTP_STATUS.CREATED);
|
|
84
|
+
res.set(_constants.HEADERS.CACHE_CONTROL, 'no-cache, no-store');
|
|
76
85
|
return next({
|
|
77
86
|
ok: `user '${req.body.name}' created`,
|
|
78
87
|
token
|
|
@@ -80,7 +89,7 @@ function _default(route, auth, config) {
|
|
|
80
89
|
});
|
|
81
90
|
}
|
|
82
91
|
});
|
|
83
|
-
|
|
92
|
+
userRouter.delete('/-/user/token/*', (0, _rateLimiter.limiter)(config === null || config === void 0 ? void 0 : config.userRateLimit), function (req, res, next) {
|
|
84
93
|
res.status(_constants.HTTP_STATUS.OK);
|
|
85
94
|
next({
|
|
86
95
|
ok: _constants.API_MESSAGE.LOGGED_OUT
|
|
@@ -88,7 +97,7 @@ function _default(route, auth, config) {
|
|
|
88
97
|
}); // placeholder 'cause npm require to be authenticated to publish
|
|
89
98
|
// we do not do any real authentication yet
|
|
90
99
|
|
|
91
|
-
|
|
100
|
+
userRouter.post('/_session', _cookies.default.express(), function (req, res, next) {
|
|
92
101
|
res.cookies.set('AuthSession', String(Math.random()), (0, _authUtils.createSessionToken)());
|
|
93
102
|
next({
|
|
94
103
|
ok: true,
|
|
@@ -96,5 +105,6 @@ function _default(route, auth, config) {
|
|
|
96
105
|
roles: []
|
|
97
106
|
});
|
|
98
107
|
});
|
|
108
|
+
route.use(userRouter);
|
|
99
109
|
}
|
|
100
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/api/endpoint/api/user.ts"],"names":["route","auth","config","get","req","res","next","status","HTTP_STATUS","OK","ok","remote_user","name","put","password","body","remoteName","_","isNil","authenticate","callbackAuthenticate","err","user","logger","error","ErrorCode","getCode","UNAUTHORIZED","API_ERROR","BAD_USERNAME_PASSWORD","restoredRemoteUser","groups","token","CREATED","BAD_REQUEST","PASSWORD_SHORT","add_user","INTERNAL_ERROR","message","getConflict","undefined","delete","API_MESSAGE","LOGGED_OUT","post","Cookies","express","cookies","set","String","Math","random","roles"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport Cookies from 'cookies';\n\nimport { Config, RemoteUser } from '@verdaccio/types';\nimport { Response, Router } from 'express';\nimport { ErrorCode } from '../../../lib/utils';\nimport { API_ERROR, API_MESSAGE, HTTP_STATUS } from '../../../lib/constants';\nimport { createRemoteUser, createSessionToken, getApiToken, getAuthenticatedMessage, validatePassword } from '../../../lib/auth-utils';\nimport { logger } from '../../../lib/logger';\n\nimport { $RequestExtend, $ResponseExtend, $NextFunctionVer, IAuth } from '../../../../types';\n\nexport default function (route: Router, auth: IAuth, config: Config): void {\n  route.get('/-/user/:org_couchdb_user', 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  route.put('/-/user/:org_couchdb_user/:_rev?/:revision?', 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(name, password, async function callbackAuthenticate(err, user): Promise<void> {\n        if (err) {\n          logger.error({ name, err }, 'authenticating for user @{username} failed. Error: @{err.message}');\n          return next(ErrorCode.getCode(HTTP_STATUS.UNAUTHORIZED, API_ERROR.BAD_USERNAME_PASSWORD));\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\n        return next({\n          ok: getAuthenticatedMessage(req.remote_user.name),\n          token,\n        });\n      });\n    } else {\n      if (validatePassword(password) === false) {\n        // eslint-disable-next-line new-cap\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(ErrorCode.getCode(err.status, err.message) || ErrorCode.getConflict(err.message));\n          }\n          return next(err);\n        }\n\n        const token = name && password ? await getApiToken(auth, config, user, password) : undefined;\n\n        req.remote_user = user;\n        res.status(HTTP_STATUS.CREATED);\n        return next({\n          ok: `user '${req.body.name}' created`,\n          token,\n        });\n      });\n    }\n  });\n\n  route.delete('/-/user/token/*', 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  // placeholder 'cause npm require to be authenticated to publish\n  // we do not do any real authentication yet\n  route.post('/_session', Cookies.express(), function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    res.cookies.set('AuthSession', String(Math.random()), createSessionToken());\n\n    next({\n      ok: true,\n      name: 'somebody',\n      roles: [],\n    });\n  });\n}\n"]}
|
|
110
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/api/endpoint/api/user.ts"],"names":["route","auth","config","userRouter","express","Router","get","userRateLimit","req","res","next","status","HTTP_STATUS","OK","ok","remote_user","name","put","password","body","remoteName","_","isNil","authenticate","callbackAuthenticate","err","user","logger","error","ErrorCode","getCode","UNAUTHORIZED","API_ERROR","BAD_USERNAME_PASSWORD","restoredRemoteUser","groups","token","CREATED","set","HEADERS","CACHE_CONTROL","BAD_REQUEST","PASSWORD_SHORT","add_user","INTERNAL_ERROR","message","getConflict","undefined","delete","API_MESSAGE","LOGGED_OUT","post","Cookies","cookies","String","Math","random","roles","use"],"mappings":";;;;;;;AAAA;;AACA;;AAGA;;AACA;;AACA;;AACA;;AACA;;AAGA;;;;AAEe,kBAAUA,KAAV,EAAyBC,IAAzB,EAAsCC,MAAtC,EAA4D;AACzE;AACA,QAAMC,UAAU,GAAGC,iBAAQC,MAAR,EAAnB;;AAEAF,EAAAA,UAAU,CAACG,GAAX,CAAe,2BAAf,EAA4C,0BAAQJ,MAAR,aAAQA,MAAR,uBAAQA,MAAM,CAAEK,aAAhB,CAA5C,EAA4E,UAAUC,GAAV,EAA+BC,GAA/B,EAA8CC,IAA9C,EAA4E;AACtJD,IAAAA,GAAG,CAACE,MAAJ,CAAWC,uBAAYC,EAAvB;AACAH,IAAAA,IAAI,CAAC;AACHI,MAAAA,EAAE,EAAE,wCAAwBN,GAAG,CAACO,WAAJ,CAAgBC,IAAxC;AADD,KAAD,CAAJ;AAGD,GALD;AAOAb,EAAAA,UAAU,CAACc,GAAX,CAAe,6CAAf,EAA8D,0BAAQf,MAAR,aAAQA,MAAR,uBAAQA,MAAM,CAAEK,aAAhB,CAA9D,EAA8F,UAAUC,GAAV,EAA+BC,GAA/B,EAA8CC,IAA9C,EAA4E;AACxK,UAAM;AAAEM,MAAAA,IAAF;AAAQE,MAAAA;AAAR,QAAqBV,GAAG,CAACW,IAA/B;AACA,UAAMC,UAAU,GAAGZ,GAAG,CAACO,WAAJ,CAAgBC,IAAnC;;AAEA,QAAIK,gBAAEC,KAAF,CAAQF,UAAR,MAAwB,KAAxB,IAAiCC,gBAAEC,KAAF,CAAQN,IAAR,MAAkB,KAAnD,IAA4DI,UAAU,KAAKJ,IAA/E,EAAqF;AACnFf,MAAAA,IAAI,CAACsB,YAAL,CAAkBP,IAAlB,EAAwBE,QAAxB,EAAkC,eAAeM,oBAAf,CAAoCC,GAApC,EAAyCC,IAAzC,EAA8D;AAC9F,YAAID,GAAJ,EAAS;AACPE,yBAAOC,KAAP,CAAa;AAAEZ,YAAAA,IAAF;AAAQS,YAAAA;AAAR,WAAb,EAA4B,mEAA5B;;AACA,iBAAOf,IAAI,CAACmB,iBAAUC,OAAV,CAAkBlB,uBAAYmB,YAA9B,EAA4CC,qBAAUC,qBAAtD,CAAD,CAAX;AACD;;AAED,cAAMC,kBAA8B,GAAG,iCAAiBlB,IAAjB,EAAuBU,IAAI,CAACS,MAAL,IAAe,EAAtC,CAAvC;AACA,cAAMC,KAAK,GAAG,MAAM,4BAAYnC,IAAZ,EAAkBC,MAAlB,EAA0BgC,kBAA1B,EAA8ChB,QAA9C,CAApB;AAEAT,QAAAA,GAAG,CAACE,MAAJ,CAAWC,uBAAYyB,OAAvB;AACA5B,QAAAA,GAAG,CAAC6B,GAAJ,CAAQC,mBAAQC,aAAhB,EAA+B,oBAA/B;AACA,eAAO9B,IAAI,CAAC;AACVI,UAAAA,EAAE,EAAE,wCAAwBN,GAAG,CAACO,WAAJ,CAAgBC,IAAxC,CADM;AAEVoB,UAAAA;AAFU,SAAD,CAAX;AAID,OAfD;AAgBD,KAjBD,MAiBO;AACL,UAAI,iCAAiBlB,QAAjB,MAA+B,KAAnC,EAA0C;AACxC;AACA,eAAOR,IAAI,CAACmB,iBAAUC,OAAV,CAAkBlB,uBAAY6B,WAA9B,EAA2CT,qBAAUU,cAAV,EAA3C,CAAD,CAAX;AACD;;AAEDzC,MAAAA,IAAI,CAAC0C,QAAL,CAAc3B,IAAd,EAAoBE,QAApB,EAA8B,gBAAgBO,GAAhB,EAAqBC,IAArB,EAA0C;AACtE,YAAID,GAAJ,EAAS;AACP,cAAIA,GAAG,CAACd,MAAJ,IAAcC,uBAAY6B,WAA1B,IAAyChB,GAAG,CAACd,MAAJ,GAAaC,uBAAYgC,cAAtE,EAAsF;AACpF;AACA;AACA;AACA,mBAAOlC,IAAI,CAACmB,iBAAUC,OAAV,CAAkBL,GAAG,CAACd,MAAtB,EAA8Bc,GAAG,CAACoB,OAAlC,KAA8ChB,iBAAUiB,WAAV,CAAsBrB,GAAG,CAACoB,OAA1B,CAA/C,CAAX;AACD;;AACD,iBAAOnC,IAAI,CAACe,GAAD,CAAX;AACD;;AAED,cAAMW,KAAK,GAAGpB,IAAI,IAAIE,QAAR,GAAmB,MAAM,4BAAYjB,IAAZ,EAAkBC,MAAlB,EAA0BwB,IAA1B,EAAgCR,QAAhC,CAAzB,GAAqE6B,SAAnF;AAEAvC,QAAAA,GAAG,CAACO,WAAJ,GAAkBW,IAAlB;AACAjB,QAAAA,GAAG,CAACE,MAAJ,CAAWC,uBAAYyB,OAAvB;AACA5B,QAAAA,GAAG,CAAC6B,GAAJ,CAAQC,mBAAQC,aAAhB,EAA+B,oBAA/B;AACA,eAAO9B,IAAI,CAAC;AACVI,UAAAA,EAAE,EAAG,SAAQN,GAAG,CAACW,IAAJ,CAASH,IAAK,WADjB;AAEVoB,UAAAA;AAFU,SAAD,CAAX;AAID,OApBD;AAqBD;AACF,GAjDD;AAmDAjC,EAAAA,UAAU,CAAC6C,MAAX,CAAkB,iBAAlB,EAAqC,0BAAQ9C,MAAR,aAAQA,MAAR,uBAAQA,MAAM,CAAEK,aAAhB,CAArC,EAAqE,UAAUC,GAAV,EAA+BC,GAA/B,EAA8CC,IAA9C,EAA4E;AAC/ID,IAAAA,GAAG,CAACE,MAAJ,CAAWC,uBAAYC,EAAvB;AACAH,IAAAA,IAAI,CAAC;AACHI,MAAAA,EAAE,EAAEmC,uBAAYC;AADb,KAAD,CAAJ;AAGD,GALD,EA9DyE,CAqEzE;AACA;;AACA/C,EAAAA,UAAU,CAACgD,IAAX,CAAgB,WAAhB,EAA6BC,iBAAQhD,OAAR,EAA7B,EAAgD,UAAUI,GAAV,EAA+BC,GAA/B,EAAqDC,IAArD,EAAmF;AACjID,IAAAA,GAAG,CAAC4C,OAAJ,CAAYf,GAAZ,CAAgB,aAAhB,EAA+BgB,MAAM,CAACC,IAAI,CAACC,MAAL,EAAD,CAArC,EAAsD,oCAAtD;AAEA9C,IAAAA,IAAI,CAAC;AACHI,MAAAA,EAAE,EAAE,IADD;AAEHE,MAAAA,IAAI,EAAE,UAFH;AAGHyC,MAAAA,KAAK,EAAE;AAHJ,KAAD,CAAJ;AAKD,GARD;AAUAzD,EAAAA,KAAK,CAAC0D,GAAN,CAAUvD,UAAV;AACD","sourcesContent":["import _ from 'lodash';\nimport Cookies from 'cookies';\n\nimport { Config, RemoteUser } from '@verdaccio/types';\nimport express, { Response, Router } from 'express';\nimport { ErrorCode } from '../../../lib/utils';\nimport { API_ERROR, API_MESSAGE, HEADERS, HTTP_STATUS } from '../../../lib/constants';\nimport { createRemoteUser, createSessionToken, getApiToken, getAuthenticatedMessage, validatePassword } from '../../../lib/auth-utils';\nimport { logger } from '../../../lib/logger';\n\nimport { $RequestExtend, $ResponseExtend, $NextFunctionVer, IAuth } from '../../../../types';\nimport { limiter } from '../../rate-limiter';\n\nexport default function (route: Router, auth: IAuth, config: Config): void {\n  /* eslint new-cap:off */\n  const userRouter = express.Router();\n\n  userRouter.get('/-/user/:org_couchdb_user', limiter(config?.userRateLimit), 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  userRouter.put('/-/user/:org_couchdb_user/:_rev?/:revision?', limiter(config?.userRateLimit), 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(name, password, async function callbackAuthenticate(err, user): Promise<void> {\n        if (err) {\n          logger.error({ name, err }, 'authenticating for user @{username} failed. Error: @{err.message}');\n          return next(ErrorCode.getCode(HTTP_STATUS.UNAUTHORIZED, API_ERROR.BAD_USERNAME_PASSWORD));\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    } else {\n      if (validatePassword(password) === false) {\n        // eslint-disable-next-line new-cap\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(ErrorCode.getCode(err.status, err.message) || ErrorCode.getConflict(err.message));\n          }\n          return next(err);\n        }\n\n        const token = name && password ? await getApiToken(auth, config, user, password) : 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  userRouter.delete('/-/user/token/*', limiter(config?.userRateLimit), 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  // placeholder 'cause npm require to be authenticated to publish\n  // we do not do any real authentication yet\n  userRouter.post('/_session', Cookies.express(), function (req: $RequestExtend, res: $ResponseExtend, next: $NextFunctionVer): void {\n    res.cookies.set('AuthSession', String(Math.random()), createSessionToken());\n\n    next({\n      ok: true,\n      name: 'somebody',\n      roles: [],\n    });\n  });\n\n  route.use(userRouter);\n}\n"]}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.default = void 0;
|
|
7
|
+
|
|
8
|
+
var _express = require("express");
|
|
9
|
+
|
|
10
|
+
var _profile = _interopRequireDefault(require("./profile"));
|
|
11
|
+
|
|
12
|
+
var _token = _interopRequireDefault(require("./token"));
|
|
13
|
+
|
|
14
|
+
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
15
|
+
|
|
16
|
+
var _default = (auth, storage, config) => {
|
|
17
|
+
const route = (0, _express.Router)();
|
|
18
|
+
/* eslint new-cap: 0 */
|
|
19
|
+
|
|
20
|
+
route.use('/-/npm/v1/', (0, _profile.default)(auth, config));
|
|
21
|
+
route.use('/-/npm/v1/', (0, _token.default)(auth, storage, config));
|
|
22
|
+
return route;
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
exports.default = _default;
|
|
26
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9hcGkvZW5kcG9pbnQvYXBpL3YxL2luZGV4LnRzIl0sIm5hbWVzIjpbImF1dGgiLCJzdG9yYWdlIiwiY29uZmlnIiwicm91dGUiLCJ1c2UiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFDQTs7OztlQUVlLENBQUNBLElBQUQsRUFBT0MsT0FBUCxFQUFnQkMsTUFBaEIsS0FBMkI7QUFDeEMsUUFBTUMsS0FBSyxHQUFHLHNCQUFkO0FBQXdCOztBQUN4QkEsRUFBQUEsS0FBSyxDQUFDQyxHQUFOLENBQVUsWUFBVixFQUF3QixzQkFBUUosSUFBUixFQUFjRSxNQUFkLENBQXhCO0FBQ0FDLEVBQUFBLEtBQUssQ0FBQ0MsR0FBTixDQUFVLFlBQVYsRUFBd0Isb0JBQU1KLElBQU4sRUFBWUMsT0FBWixFQUFxQkMsTUFBckIsQ0FBeEI7QUFDQSxTQUFPQyxLQUFQO0FBQ0QsQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJlc3BvbnNlLCBSb3V0ZXIgfSBmcm9tICdleHByZXNzJztcbmltcG9ydCBwcm9maWxlIGZyb20gJy4vcHJvZmlsZSc7XG5pbXBvcnQgdG9rZW4gZnJvbSAnLi90b2tlbic7XG5cbmV4cG9ydCBkZWZhdWx0IChhdXRoLCBzdG9yYWdlLCBjb25maWcpID0+IHtcbiAgY29uc3Qgcm91dGUgPSBSb3V0ZXIoKTsgLyogZXNsaW50IG5ldy1jYXA6IDAgKi9cbiAgcm91dGUudXNlKCcvLS9ucG0vdjEvJywgcHJvZmlsZShhdXRoLCBjb25maWcpKTtcbiAgcm91dGUudXNlKCcvLS9ucG0vdjEvJywgdG9rZW4oYXV0aCwgc3RvcmFnZSwgY29uZmlnKSk7XG4gIHJldHVybiByb3V0ZTtcbn07XG4iXX0=
|
|
@@ -7,15 +7,22 @@ exports.default = _default;
|
|
|
7
7
|
|
|
8
8
|
var _lodash = _interopRequireDefault(require("lodash"));
|
|
9
9
|
|
|
10
|
+
var _express = require("express");
|
|
11
|
+
|
|
10
12
|
var _constants = require("../../../../lib/constants");
|
|
11
13
|
|
|
12
14
|
var _utils = require("../../../../lib/utils");
|
|
13
15
|
|
|
14
16
|
var _authUtils = require("../../../../lib/auth-utils");
|
|
15
17
|
|
|
18
|
+
var _rateLimiter = require("../../../rate-limiter");
|
|
19
|
+
|
|
16
20
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
17
21
|
|
|
18
|
-
function _default(
|
|
22
|
+
function _default(auth, config) {
|
|
23
|
+
const profileRoute = (0, _express.Router)();
|
|
24
|
+
/* eslint new-cap: 0 */
|
|
25
|
+
|
|
19
26
|
function buildProfile(name) {
|
|
20
27
|
return {
|
|
21
28
|
tfa: false,
|
|
@@ -29,7 +36,7 @@ function _default(route, auth) {
|
|
|
29
36
|
};
|
|
30
37
|
}
|
|
31
38
|
|
|
32
|
-
|
|
39
|
+
profileRoute.get('/user', (0, _rateLimiter.limiter)(config === null || config === void 0 ? void 0 : config.userRateLimit), function (req, res, next) {
|
|
33
40
|
if (_lodash.default.isNil(req.remote_user.name) === false) {
|
|
34
41
|
return next(buildProfile(req.remote_user.name));
|
|
35
42
|
}
|
|
@@ -39,7 +46,7 @@ function _default(route, auth) {
|
|
|
39
46
|
message: _constants.API_ERROR.MUST_BE_LOGGED
|
|
40
47
|
});
|
|
41
48
|
});
|
|
42
|
-
|
|
49
|
+
profileRoute.post('/user', (0, _rateLimiter.limiter)(config === null || config === void 0 ? void 0 : config.userRateLimit), function (req, res, next) {
|
|
43
50
|
if (_lodash.default.isNil(req.remote_user.name)) {
|
|
44
51
|
res.status(_constants.HTTP_STATUS.UNAUTHORIZED);
|
|
45
52
|
return next({
|
|
@@ -79,5 +86,6 @@ function _default(route, auth) {
|
|
|
79
86
|
return next(_utils.ErrorCode.getCode(_constants.HTTP_STATUS.INTERNAL_ERROR, _constants.APP_ERROR.PROFILE_ERROR));
|
|
80
87
|
}
|
|
81
88
|
});
|
|
89
|
+
return profileRoute;
|
|
82
90
|
}
|
|
83
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
91
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9hcGkvZW5kcG9pbnQvYXBpL3YxL3Byb2ZpbGUudHMiXSwibmFtZXMiOlsiYXV0aCIsImNvbmZpZyIsInByb2ZpbGVSb3V0ZSIsImJ1aWxkUHJvZmlsZSIsIm5hbWUiLCJ0ZmEiLCJlbWFpbCIsImVtYWlsX3ZlcmlmaWVkIiwiY3JlYXRlZCIsInVwZGF0ZWQiLCJjaWRyX3doaXRlbGlzdCIsImZ1bGxuYW1lIiwiZ2V0IiwidXNlclJhdGVMaW1pdCIsInJlcSIsInJlcyIsIm5leHQiLCJfIiwiaXNOaWwiLCJyZW1vdGVfdXNlciIsInN0YXR1cyIsIkhUVFBfU1RBVFVTIiwiVU5BVVRIT1JJWkVEIiwibWVzc2FnZSIsIkFQSV9FUlJPUiIsIk1VU1RfQkVfTE9HR0VEIiwicG9zdCIsInBhc3N3b3JkIiwiYm9keSIsIm5ldyIsIkVycm9yQ29kZSIsImdldENvZGUiLCJQQVNTV09SRF9TSE9SVCIsImNoYW5nZVBhc3N3b3JkIiwib2xkIiwiZXJyIiwiaXNVcGRhdGVkIiwiaXNOdWxsIiwiZ2V0Q29uZmxpY3QiLCJnZXRJbnRlcm5hbEVycm9yIiwiSU5URVJOQUxfU0VSVkVSX0VSUk9SIiwiU0VSVklDRV9VTkFWQUlMQUJMRSIsIlNVUFBPUlRfRVJST1JTIiwiVEZBX0RJU0FCTEVEIiwiSU5URVJOQUxfRVJST1IiLCJBUFBfRVJST1IiLCJQUk9GSUxFX0VSUk9SIl0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBQUE7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBR0E7Ozs7QUFhZSxrQkFBVUEsSUFBVixFQUF1QkMsTUFBdkIsRUFBdUM7QUFDcEQsUUFBTUMsWUFBWSxHQUFHLHNCQUFyQjtBQUErQjs7QUFDL0IsV0FBU0MsWUFBVCxDQUFzQkMsSUFBdEIsRUFBNkM7QUFDM0MsV0FBTztBQUNMQyxNQUFBQSxHQUFHLEVBQUUsS0FEQTtBQUVMRCxNQUFBQSxJQUZLO0FBR0xFLE1BQUFBLEtBQUssRUFBRSxFQUhGO0FBSUxDLE1BQUFBLGNB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NEI7QUFDakMsYUFBT1csSUFBSSxDQUFDYyxpQkFBVUMsT0FBVixDQUFrQlYsdUJBQVlvQixtQkFBOUIsRUFBbURDLDBCQUFlQyxZQUFsRSxDQUFELENBQVg7QUFDRCxLQUZNLE1BRUE7QUFDTCxhQUFPM0IsSUFBSSxDQUFDYyxpQkFBVUMsT0FBVixDQUFrQlYsdUJBQVl1QixjQUE5QixFQUE4Q0MscUJBQVVDLGFBQXhELENBQUQsQ0FBWDtBQUNEO0FBQ0YsR0FqQ0Q7QUFtQ0EsU0FBTzVDLFlBQVA7QUFDRCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5pbXBvcnQgeyBSZXNwb25zZSwgUm91dGVyIH0gZnJvbSAnZXhwcmVzcyc7XG5pbXBvcnQgeyBBUElfRVJST1IsIEFQUF9FUlJPUiwgSFRUUF9TVEFUVVMsIFNVUFBPUlRfRVJST1JTIH0gZnJvbSAnLi4vLi4vLi4vLi4vbGliL2NvbnN0YW50cyc7XG5pbXBvcnQgeyBFcnJvckNvZGUgfSBmcm9tICcuLi8uLi8uLi8uLi9saWIvdXRpbHMnO1xuaW1wb3J0IHsgdmFsaWRhdGVQYXNzd29yZCB9IGZyb20gJy4uLy4uLy4uLy4uL2xpYi9hdXRoLXV0aWxzJztcblxuaW1wb3J0IHsgJE5leHRGdW5jdGlvblZlciwgJFJlcXVlc3RFeHRlbmQsIElBdXRoIH0gZnJvbSAnLi4vLi4vLi4vLi4vLi4vdHlwZXMnO1xuaW1wb3J0IHsgbGltaXRlciB9IGZyb20gJy4uLy4uLy4uL3JhdGUtbGltaXRlcic7XG5cbmV4cG9ydCBpbnRlcmZhY2UgUHJvZmlsZSB7XG4gIHRmYTogYm9vbGVhbjtcbiAgbmFtZTogc3RyaW5nO1xuICBlbWFpbDogc3RyaW5nO1xuICBlbWFpbF92ZXJpZmllZDogYm9vbGVhbjtcbiAgY3JlYXRlZDogc3RyaW5nO1xuICB1cGRhdGVkOiBzdHJpbmc7XG4gIGNpZHJfd2hpdGVsaXN0OiBzdHJpbmdbXSB8IG51bGw7XG4gIGZ1bGxuYW1lOiBzdHJpbmc7XG59XG5cbmV4cG9ydCBkZWZhdWx0IGZ1bmN0aW9uIChhdXRoOiBJQXV0aCwgY29uZmlnKTogUm91dGVyIHtcbiAgY29uc3QgcHJvZmlsZVJvdXRlID0gUm91dGVyKCk7IC8qIGVzbGludCBuZXctY2FwOiAwICovXG4gIGZ1bmN0aW9uIGJ1aWxkUHJvZmlsZShuYW1lOiBzdHJpbmcpOiBQcm9maWxlIHtcbiAgICByZXR1cm4ge1xuICAgICAgdGZhOiBmYWxzZSxcbiAgICAgIG5hbWUsXG4gICAgICBlbWFpbDogJycsXG4gICAgICBlbWFpbF92ZXJpZmllZDogZmFsc2UsXG4gICAgICBjcmVhdGVkOiAnJyxcbiAgICAgIHVwZGF0ZWQ6ICcnLFxuICAgICAgY2lkcl93aGl0ZWxpc3Q6IG51bGwsXG4gICAgICBmdWxsbmFtZTogJycsXG4gICAgfTtcbiAgfVxuXG4gIHByb2ZpbGVSb3V0ZS5nZXQoJy91c2VyJywgbGltaXRlcihjb25maWc/LnVzZXJSYXRlTGltaXQpLCBmdW5jdGlvbiAocmVxOiAkUmVxdWVzdEV4dGVuZCwgcmVzOiBSZXNwb25zZSwgbmV4dDogJE5leHRGdW5jdGlvblZlcik6IHZvaWQge1xuICAgIGlmIChfLmlzTmlsKHJlcS5yZW1vdGVfdXNlci5uYW1lKSA9PT0gZmFsc2UpIHtcbiAgICAgIHJldHVybiBuZXh0KGJ1aWxkUHJvZmlsZShyZXEucmVtb3RlX3VzZXIubmFtZSkpO1xuICAgIH1cblxuICAgIHJlcy5zdGF0dXMoSFRUUF9TVEFUVVMuVU5BVVRIT1JJWkVEKTtcbiAgICByZXR1cm4gbmV4dCh7XG4gICAgICBtZXNzYWdlOiBBUElfRVJST1IuTVVTVF9CRV9MT0dHRUQsXG4gICAgfSk7XG4gIH0pO1xuXG4gIHByb2ZpbGVSb3V0ZS5wb3N0KCcvdXNlcicsIGxpbWl0ZXIoY29uZmlnPy51c2VyUmF0ZUxpbWl0KSwgZnVuY3Rpb24gKHJlcTogJFJlcXVlc3RFeHRlbmQsIHJlczogUmVzcG9uc2UsIG5leHQ6ICROZXh0RnVuY3Rpb25WZXIpOiB2b2lkIHtcbiAgICBpZiAoXy5pc05pbChyZXEucmVtb3RlX3VzZXIubmFtZSkpIHtcbiAgICAgIHJlcy5zdGF0dXMoSFRUUF9TVEFUVVMuVU5BVVRIT1JJWkVEKTtcbiAgICAgIHJldHVybiBuZXh0KHtcbiAgICAgICAgbWVzc2FnZTogQVBJX0VSUk9SLk1VU1RfQkVfTE9HR0VELFxuICAgICAgfSk7XG4gICAgfVxuXG4gICAgY29uc3QgeyBwYXNzd29yZCwgdGZhIH0gPSByZXEuYm9keTtcbiAgICBjb25zdCB7IG5hbWUgfSA9IHJlcS5yZW1vdGVfdXNlcjtcblxuICAgIGlmIChfLmlzTmlsKHBhc3N3b3JkKSA9PT0gZmFsc2UpIHtcbiAgICAgIGlmICh2YWxpZGF0ZVBhc3N3b3JkKHBhc3N3b3JkLm5ldykgPT09IGZhbHNlKSB7XG4gICAgICAgIC8qIGVzbGludCBuZXctY2FwOm9mZiAqL1xuICAgICAgICByZXR1cm4gbmV4dChFcnJvckNvZGUuZ2V0Q29kZShIVFRQX1NUQVRVUy5VTkFVVEhPUklaRUQsIEFQSV9FUlJPUi5QQVNTV09SRF9TSE9SVCgpKSk7XG4gICAgICAgIC8qIGVzbGludCBuZXctY2FwOm9mZiAqL1xuICAgICAgfVxuXG4gICAgICBhdXRoLmNoYW5nZVBhc3N3b3JkKG5hbWUsIHBhc3N3b3JkLm9sZCwgcGFzc3dvcmQubmV3LCAoZXJyLCBpc1VwZGF0ZWQpOiAkTmV4dEZ1bmN0aW9uVmVyID0+IHtcbiAgICAgICAgaWYgKF8uaXNOdWxsKGVycikgPT09IGZhbHNlKSB7XG4gICAgICAgICAgcmV0dXJuIG5leHQoRXJyb3JDb2RlLmdldENvZGUoZXJyLnN0YXR1cywgZXJyLm1lc3NhZ2UpIHx8IEVycm9yQ29kZS5nZXRDb25mbGljdChlcnIubWVzc2FnZSkpO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKGlzVXBkYXRlZCkge1xuICAgICAgICAgIHJldHVybiBuZXh0KGJ1aWxkUHJvZmlsZShyZXEucmVtb3RlX3VzZXIubmFtZSkpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBuZXh0KEVycm9yQ29kZS5nZXRJbnRlcm5hbEVycm9yKEFQSV9FUlJPUi5JTlRFUk5BTF9TRVJWRVJfRVJST1IpKTtcbiAgICAgIH0pO1xuICAgIH0gZWxzZSBpZiAoXy5pc05pbCh0ZmEpID09PSBmYWxzZSkge1xuICAgICAgcmV0dXJuIG5leHQoRXJyb3JDb2RlLmdldENvZGUoSFRUUF9TVEFUVVMuU0VSVklDRV9VTkFWQUlMQUJMRSwgU1VQUE9SVF9FUlJPUlMuVEZBX0RJU0FCTEVEKSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHJldHVybiBuZXh0KEVycm9yQ29kZS5nZXRDb2RlKEhUVFBfU1RBVFVTLklOVEVSTkFMX0VSUk9SLCBBUFBfRVJST1IuUFJPRklMRV9FUlJPUikpO1xuICAgIH1cbiAgfSk7XG5cbiAgcmV0dXJuIHByb2ZpbGVSb3V0ZTtcbn1cbiJdfQ==
|
|
@@ -9,6 +9,8 @@ var _lodash = _interopRequireDefault(require("lodash"));
|
|
|
9
9
|
|
|
10
10
|
var _debug = _interopRequireDefault(require("debug"));
|
|
11
11
|
|
|
12
|
+
var _express = require("express");
|
|
13
|
+
|
|
12
14
|
var _constants = require("../../../../lib/constants");
|
|
13
15
|
|
|
14
16
|
var _utils = require("../../../../lib/utils");
|
|
@@ -19,6 +21,8 @@ var _cryptoUtils = require("../../../../lib/crypto-utils");
|
|
|
19
21
|
|
|
20
22
|
var _logger = require("../../../../lib/logger");
|
|
21
23
|
|
|
24
|
+
var _rateLimiter = require("../../../rate-limiter");
|
|
25
|
+
|
|
22
26
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
23
27
|
|
|
24
28
|
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) { symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); } keys.push.apply(keys, symbols); } return keys; }
|
|
@@ -36,8 +40,11 @@ function normalizeToken(token) {
|
|
|
36
40
|
} // https://github.com/npm/npm-profile/blob/latest/lib/index.js
|
|
37
41
|
|
|
38
42
|
|
|
39
|
-
function _default(
|
|
40
|
-
|
|
43
|
+
function _default(auth, storage, config) {
|
|
44
|
+
const tokenRoute = (0, _express.Router)();
|
|
45
|
+
/* eslint new-cap: 0 */
|
|
46
|
+
|
|
47
|
+
tokenRoute.get('/tokens', (0, _rateLimiter.limiter)(config === null || config === void 0 ? void 0 : config.userRateLimit), async function (req, res, next) {
|
|
41
48
|
const {
|
|
42
49
|
name
|
|
43
50
|
} = req.remote_user;
|
|
@@ -68,7 +75,7 @@ function _default(route, auth, storage, config) {
|
|
|
68
75
|
|
|
69
76
|
return next(_utils.ErrorCode.getUnauthorized());
|
|
70
77
|
});
|
|
71
|
-
|
|
78
|
+
tokenRoute.post('/tokens', (0, _rateLimiter.limiter)(config === null || config === void 0 ? void 0 : config.userRateLimit), function (req, res, next) {
|
|
72
79
|
const {
|
|
73
80
|
password,
|
|
74
81
|
readonly,
|
|
@@ -115,6 +122,7 @@ function _default(route, auth, storage, config) {
|
|
|
115
122
|
};
|
|
116
123
|
await storage.saveToken(saveToken);
|
|
117
124
|
debug('token %o was created for user %o', key, name);
|
|
125
|
+
res.set(_constants.HEADERS.CACHE_CONTROL, 'no-cache, no-store');
|
|
118
126
|
return next(normalizeToken({
|
|
119
127
|
token,
|
|
120
128
|
user: name,
|
|
@@ -132,7 +140,7 @@ function _default(route, auth, storage, config) {
|
|
|
132
140
|
}
|
|
133
141
|
});
|
|
134
142
|
});
|
|
135
|
-
|
|
143
|
+
tokenRoute.delete('/tokens/token/:tokenKey', (0, _rateLimiter.limiter)(config === null || config === void 0 ? void 0 : config.userRateLimit), async (req, res, next) => {
|
|
136
144
|
const {
|
|
137
145
|
params: {
|
|
138
146
|
tokenKey
|
|
@@ -165,5 +173,6 @@ function _default(route, auth, storage, config) {
|
|
|
165
173
|
|
|
166
174
|
return next(_utils.ErrorCode.getUnauthorized());
|
|
167
175
|
});
|
|
176
|
+
return tokenRoute;
|
|
168
177
|
}
|
|
169
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/api/endpoint/api/v1/token.ts"],"names":["debug","normalizeToken","token","created","Date","toISOString","route","auth","storage","config","get","req","res","next","name","remote_user","_","isNil","tokens","readTokens","user","totalTokens","length","status","HTTP_STATUS","OK","objects","map","urls","error","logger","msg","ErrorCode","getCode","INTERNAL_ERROR","message","getUnauthorized","post","password","readonly","cidr_whitelist","body","isBoolean","isArray","BAD_DATA","SUPPORT_ERRORS","PARAMETERS_NOT_VALID","authenticate","err","errorCode","UNAUTHORIZED","isFunction","saveToken","NOT_IMPLEMENTED","STORAGE_NOT_IMPLEMENT","key","maskedToken","getTime","cidr","delete","params","tokenKey","deleteToken","info"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport buildDebug from 'debug';\nimport { Response, Router } from 'express';\nimport { Config, RemoteUser, Token } from '@verdaccio/types';\nimport { HTTP_STATUS, SUPPORT_ERRORS } from '../../../../lib/constants';\nimport { ErrorCode, mask } from '../../../../lib/utils';\nimport { getApiToken } from '../../../../lib/auth-utils';\nimport { stringToMD5 } from '../../../../lib/crypto-utils';\nimport { logger } from '../../../../lib/logger';\n\nimport { $NextFunctionVer, $RequestExtend, IAuth, IStorageHandler } from '../../../../../types';\n\nconst debug = buildDebug('verdaccio:token');\nexport type NormalizeToken = Token & {\n  created: string;\n};\n\nfunction normalizeToken(token: Token): NormalizeToken {\n  return {\n    ...token,\n    created: new Date(token.created).toISOString(),\n  };\n}\n\n// https://github.com/npm/npm-profile/blob/latest/lib/index.js\nexport default function (route: Router, auth: IAuth, storage: IStorageHandler, config: Config): void {\n  route.get('/-/npm/v1/tokens', async function (req: $RequestExtend, res: Response, next: $NextFunctionVer) {\n    const { name } = req.remote_user;\n\n    if (_.isNil(name) === false) {\n      try {\n        const tokens = await storage.readTokens({ user: name });\n        const totalTokens = tokens.length;\n        debug('token list retrieved: %o', totalTokens);\n        res.status(HTTP_STATUS.OK);\n        return next({\n          objects: tokens.map(normalizeToken),\n          urls: {\n            next: '', // TODO: pagination?\n          },\n        });\n      } catch (error) {\n        logger.error({ error: error.msg }, 'token list has failed: @{error}');\n        return next(ErrorCode.getCode(HTTP_STATUS.INTERNAL_ERROR, error.message));\n      }\n    }\n    return next(ErrorCode.getUnauthorized());\n  });\n\n  route.post('/-/npm/v1/tokens', function (req: $RequestExtend, res: Response, next: $NextFunctionVer) {\n    const { password, readonly, cidr_whitelist } = req.body;\n    const { name } = req.remote_user;\n\n    if (!_.isBoolean(readonly) || !_.isArray(cidr_whitelist)) {\n      return next(ErrorCode.getCode(HTTP_STATUS.BAD_DATA, SUPPORT_ERRORS.PARAMETERS_NOT_VALID));\n    }\n\n    auth.authenticate(name, password, async (err, user: RemoteUser) => {\n      if (err) {\n        const errorCode = err.message ? HTTP_STATUS.UNAUTHORIZED : HTTP_STATUS.INTERNAL_ERROR;\n        return next(ErrorCode.getCode(errorCode, err.message));\n      }\n\n      req.remote_user = user;\n\n      if (!_.isFunction(storage.saveToken)) {\n        return next(ErrorCode.getCode(HTTP_STATUS.NOT_IMPLEMENTED, SUPPORT_ERRORS.STORAGE_NOT_IMPLEMENT));\n      }\n\n      try {\n        const token = await getApiToken(auth, config, user, password);\n        const key = stringToMD5(token);\n        // TODO: use a utility here\n        const maskedToken = mask(token, 5);\n        const created = new Date().getTime();\n\n        /**\n         * cidr_whitelist: is not being used, we pass it through\n         * token: we do not store the real token (it is generated once and retrieved to the user), just a mask of it.\n         */\n        const saveToken: Token = {\n          user: name,\n          token: maskedToken,\n          key,\n          cidr: cidr_whitelist,\n          readonly,\n          created,\n        };\n\n        await storage.saveToken(saveToken);\n        debug('token %o was created for user %o', key, name);\n        return next(\n          normalizeToken({\n            token,\n            user: name,\n            key: saveToken.key,\n            cidr: cidr_whitelist,\n            readonly,\n            created: saveToken.created,\n          })\n        );\n      } catch (error) {\n        logger.error({ error: error.msg }, 'token creation has failed: @{error}');\n        return next(ErrorCode.getCode(HTTP_STATUS.INTERNAL_ERROR, error.message));\n      }\n    });\n  });\n\n  route.delete('/-/npm/v1/tokens/token/:tokenKey', async (req: $RequestExtend, res: Response, next: $NextFunctionVer) => {\n    const {\n      params: { tokenKey },\n    } = req;\n    const { name } = req.remote_user;\n\n    if (_.isNil(name) === false) {\n      debug('%o has requested remove a token', name);\n      try {\n        await storage.deleteToken(name, tokenKey);\n        logger.info({ tokenKey, name }, 'token id @{tokenKey} was revoked for user @{name}');\n        return next({});\n      } catch (error) {\n        logger.error({ error: error.msg }, 'token creation has failed: @{error}');\n        return next(ErrorCode.getCode(HTTP_STATUS.INTERNAL_ERROR, error.message));\n      }\n    }\n    return next(ErrorCode.getUnauthorized());\n  });\n}\n"]}
|
|
178
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/api/endpoint/api/v1/token.ts"],"names":["debug","normalizeToken","token","created","Date","toISOString","auth","storage","config","tokenRoute","get","userRateLimit","req","res","next","name","remote_user","_","isNil","tokens","readTokens","user","totalTokens","length","status","HTTP_STATUS","OK","objects","map","urls","error","logger","msg","ErrorCode","getCode","INTERNAL_ERROR","message","getUnauthorized","post","password","readonly","cidr_whitelist","body","isBoolean","isArray","BAD_DATA","SUPPORT_ERRORS","PARAMETERS_NOT_VALID","authenticate","err","errorCode","UNAUTHORIZED","isFunction","saveToken","NOT_IMPLEMENTED","STORAGE_NOT_IMPLEMENT","key","maskedToken","getTime","cidr","set","HEADERS","CACHE_CONTROL","delete","params","tokenKey","deleteToken","info"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport buildDebug from 'debug';\nimport { Response, Router } from 'express';\nimport { Config, RemoteUser, Token } from '@verdaccio/types';\nimport { HEADERS, HTTP_STATUS, SUPPORT_ERRORS } from '../../../../lib/constants';\nimport { ErrorCode, mask } from '../../../../lib/utils';\nimport { getApiToken } from '../../../../lib/auth-utils';\nimport { stringToMD5 } from '../../../../lib/crypto-utils';\nimport { logger } from '../../../../lib/logger';\n\nimport { $NextFunctionVer, $RequestExtend, IAuth, IStorageHandler } from '../../../../../types';\nimport { limiter } from '../../../rate-limiter';\n\nconst debug = buildDebug('verdaccio:token');\nexport type NormalizeToken = Token & {\n  created: string;\n};\n\nfunction normalizeToken(token: Token): NormalizeToken {\n  return {\n    ...token,\n    created: new Date(token.created).toISOString(),\n  };\n}\n\n// https://github.com/npm/npm-profile/blob/latest/lib/index.js\nexport default function (auth: IAuth, storage: IStorageHandler, config: Config): Router {\n  const tokenRoute = Router(); /* eslint new-cap: 0 */\n  tokenRoute.get('/tokens', limiter(config?.userRateLimit), async function (req: $RequestExtend, res: Response, next: $NextFunctionVer) {\n    const { name } = req.remote_user;\n\n    if (_.isNil(name) === false) {\n      try {\n        const tokens = await storage.readTokens({ user: name });\n        const totalTokens = tokens.length;\n        debug('token list retrieved: %o', totalTokens);\n        res.status(HTTP_STATUS.OK);\n        return next({\n          objects: tokens.map(normalizeToken),\n          urls: {\n            next: '', // TODO: pagination?\n          },\n        });\n      } catch (error) {\n        logger.error({ error: error.msg }, 'token list has failed: @{error}');\n        return next(ErrorCode.getCode(HTTP_STATUS.INTERNAL_ERROR, error.message));\n      }\n    }\n    return next(ErrorCode.getUnauthorized());\n  });\n\n  tokenRoute.post('/tokens', limiter(config?.userRateLimit), function (req: $RequestExtend, res: Response, next: $NextFunctionVer) {\n    const { password, readonly, cidr_whitelist } = req.body;\n    const { name } = req.remote_user;\n\n    if (!_.isBoolean(readonly) || !_.isArray(cidr_whitelist)) {\n      return next(ErrorCode.getCode(HTTP_STATUS.BAD_DATA, SUPPORT_ERRORS.PARAMETERS_NOT_VALID));\n    }\n\n    auth.authenticate(name, password, async (err, user: RemoteUser) => {\n      if (err) {\n        const errorCode = err.message ? HTTP_STATUS.UNAUTHORIZED : HTTP_STATUS.INTERNAL_ERROR;\n        return next(ErrorCode.getCode(errorCode, err.message));\n      }\n\n      req.remote_user = user;\n\n      if (!_.isFunction(storage.saveToken)) {\n        return next(ErrorCode.getCode(HTTP_STATUS.NOT_IMPLEMENTED, SUPPORT_ERRORS.STORAGE_NOT_IMPLEMENT));\n      }\n\n      try {\n        const token = await getApiToken(auth, config, user, password);\n        const key = stringToMD5(token);\n        // TODO: use a utility here\n        const maskedToken = mask(token, 5);\n        const created = new Date().getTime();\n\n        /**\n         * cidr_whitelist: is not being used, we pass it through\n         * token: we do not store the real token (it is generated once and retrieved to the user), just a mask of it.\n         */\n        const saveToken: Token = {\n          user: name,\n          token: maskedToken,\n          key,\n          cidr: cidr_whitelist,\n          readonly,\n          created,\n        };\n\n        await storage.saveToken(saveToken);\n        debug('token %o was created for user %o', key, name);\n        res.set(HEADERS.CACHE_CONTROL, 'no-cache, no-store');\n        return next(\n          normalizeToken({\n            token,\n            user: name,\n            key: saveToken.key,\n            cidr: cidr_whitelist,\n            readonly,\n            created: saveToken.created,\n          })\n        );\n      } catch (error) {\n        logger.error({ error: error.msg }, 'token creation has failed: @{error}');\n        return next(ErrorCode.getCode(HTTP_STATUS.INTERNAL_ERROR, error.message));\n      }\n    });\n  });\n\n  tokenRoute.delete('/tokens/token/:tokenKey', limiter(config?.userRateLimit), async (req: $RequestExtend, res: Response, next: $NextFunctionVer) => {\n    const {\n      params: { tokenKey },\n    } = req;\n    const { name } = req.remote_user;\n\n    if (_.isNil(name) === false) {\n      debug('%o has requested remove a token', name);\n      try {\n        await storage.deleteToken(name, tokenKey);\n        logger.info({ tokenKey, name }, 'token id @{tokenKey} was revoked for user @{name}');\n        return next({});\n      } catch (error) {\n        logger.error({ error: error.msg }, 'token creation has failed: @{error}');\n        return next(ErrorCode.getCode(HTTP_STATUS.INTERNAL_ERROR, error.message));\n      }\n    }\n    return next(ErrorCode.getUnauthorized());\n  });\n\n  return tokenRoute;\n}\n"]}
|