Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7834f44901 | ||
|
|
463cf44133 | ||
|
|
e6bb4b76c2 | ||
|
|
b86d8eb110 | ||
|
|
070e72b571 | ||
|
|
6e6f87c147 | ||
|
|
bc9306eda7 | ||
|
|
66ea71756c | ||
|
|
0a1fff70cc | ||
|
|
71da60c16f | ||
|
|
b5d585988f | ||
|
|
758d1d4320 | ||
|
|
7b77f09264 | ||
|
|
7731b44ab4 | ||
|
|
4bc6998040 | ||
|
|
0d28fbde54 | ||
|
|
660035c1d5 | ||
|
|
f21e5a1c07 | ||
|
|
4d20d91965 | ||
|
|
a3e33850fc |
25
History.md
25
History.md
@@ -1,4 +1,29 @@
|
||||
|
||||
1.1.0 / 2014-08-07
|
||||
==================
|
||||
|
||||
* Add user to maintainers when publish. fixed #395
|
||||
* List all npm registry api. close #413
|
||||
* limit list since
|
||||
* change deps by "~"
|
||||
* use cfork to make sure worker fork and restart
|
||||
* handle master uncaughtException. fixed #403
|
||||
|
||||
1.0.6 / 2014-08-02
|
||||
==================
|
||||
|
||||
* WTF moment@2.8.0 missing
|
||||
|
||||
1.0.5 / 2014-08-02
|
||||
==================
|
||||
|
||||
* unpublish pkg@version bug hotfix. fixed #400
|
||||
|
||||
1.0.4 / 2014-08-01
|
||||
==================
|
||||
|
||||
* hotfix #399 use not exists
|
||||
|
||||
1.0.3 / 2014-08-01
|
||||
==================
|
||||
|
||||
|
||||
2
Makefile
2
Makefile
@@ -63,7 +63,7 @@ contributors: install
|
||||
@./node_modules/.bin/contributors -f plain -o AUTHORS
|
||||
|
||||
autod: install
|
||||
@./node_modules/.bin/autod -w -e public,view,docs,backup,coverage -k nodemailer
|
||||
@./node_modules/.bin/autod -w -e public,view,docs,backup,coverage -k nodemailer --prefix "~"
|
||||
@$(MAKE) install
|
||||
|
||||
.PHONY: test
|
||||
|
||||
@@ -143,13 +143,12 @@ exports.show = function* (next) {
|
||||
};
|
||||
return;
|
||||
}
|
||||
var result = yield SyncModuleWorker.sync(name, 'sync-by-install');
|
||||
var result = yield* SyncModuleWorker.sync(name, 'sync-by-install');
|
||||
this.body = result.pkg;
|
||||
this.status = result.ok ? 200 : (result.statusCode || 500);
|
||||
return;
|
||||
}
|
||||
|
||||
var nextMod = null;
|
||||
var latestMod = null;
|
||||
var readme = null;
|
||||
// set tags
|
||||
@@ -166,22 +165,19 @@ exports.show = function* (next) {
|
||||
var createdTime = null;
|
||||
for (var i = 0; i < rows.length; i++) {
|
||||
var row = rows[i];
|
||||
if (row.version === 'next') {
|
||||
nextMod = row;
|
||||
continue;
|
||||
}
|
||||
var pkg = row.package;
|
||||
common.setDownloadURL(pkg, this);
|
||||
pkg._cnpm_publish_time = row.publish_time;
|
||||
versions[pkg.version] = pkg;
|
||||
|
||||
var t = times[pkg.version] = row.publish_time ? new Date(row.publish_time) : row.gmt_modified;
|
||||
if ((!distTags.latest && !latestMod) || distTags.latest === row.version) {
|
||||
if ((!distTags.latest && !latestMod) || distTags.latest === pkg.version) {
|
||||
latestMod = row;
|
||||
readme = pkg.readme;
|
||||
}
|
||||
|
||||
delete pkg.readme;
|
||||
if (maintainers.length > 0) {
|
||||
// TODO: need to use newer maintainers
|
||||
pkg.maintainers = maintainers;
|
||||
}
|
||||
|
||||
@@ -208,21 +204,13 @@ exports.show = function* (next) {
|
||||
}
|
||||
|
||||
if (!latestMod) {
|
||||
latestMod = nextMod || rows[0];
|
||||
}
|
||||
|
||||
if (!nextMod) {
|
||||
nextMod = latestMod;
|
||||
}
|
||||
|
||||
var rev = '';
|
||||
if (nextMod) {
|
||||
rev = String(nextMod.id);
|
||||
latestMod = rows[0];
|
||||
}
|
||||
|
||||
var rev = String(latestMod.id);
|
||||
var pkg = latestMod.package;
|
||||
|
||||
if (tags.length === 0 && pkg.version !== 'next') {
|
||||
if (tags.length === 0) {
|
||||
// some sync error reason, will cause tags missing
|
||||
// set latest tag at least
|
||||
distTags.latest = pkg.version;
|
||||
@@ -438,7 +426,6 @@ exports.addPackageAndDist = function *(next) {
|
||||
// { content_type: 'application/octet-stream',
|
||||
// data: 'H4sIAAAAA
|
||||
// length: 9883
|
||||
|
||||
var pkg = this.request.body;
|
||||
var username = this.user.name;
|
||||
var name = this.params.name || this.params[0];
|
||||
@@ -455,9 +442,10 @@ exports.addPackageAndDist = function *(next) {
|
||||
|
||||
var attachment = pkg._attachments[filename];
|
||||
var versionPackage = pkg.versions[version];
|
||||
var maintainers = versionPackage.maintainers;
|
||||
|
||||
// should never happened in normal request
|
||||
if (!versionPackage.maintainers) {
|
||||
if (!maintainers) {
|
||||
this.status = 400;
|
||||
this.body = {
|
||||
error: 'maintainers error',
|
||||
@@ -471,7 +459,7 @@ exports.addPackageAndDist = function *(next) {
|
||||
|
||||
// make sure user in auth is in maintainers
|
||||
// should never happened in normal request
|
||||
var m = versionPackage.maintainers.filter(function (maintainer) {
|
||||
var m = maintainers.filter(function (maintainer) {
|
||||
return maintainer.name === username;
|
||||
});
|
||||
if (!m.length) {
|
||||
@@ -592,6 +580,11 @@ exports.addPackageAndDist = function *(next) {
|
||||
});
|
||||
}
|
||||
|
||||
// ensure maintainers exists
|
||||
yield* packageService.addMaintainers(name, maintainers.map(function (item) {
|
||||
return item.name;
|
||||
}));
|
||||
|
||||
this.status = 201;
|
||||
this.body = {
|
||||
ok: true,
|
||||
@@ -747,7 +740,7 @@ exports.removeWithVersions = function* (next) {
|
||||
debug('remove versions: %j, remain versions: %j', removeVersions, remainVersions);
|
||||
|
||||
// step 4: remove all the versions which need to remove
|
||||
yield Module.removeByNameAndVersions(name, removeVersions);
|
||||
// let removeTar do remove versions from module table
|
||||
var tags = yield Module.listTags(name);
|
||||
|
||||
var removeTags = [];
|
||||
@@ -785,20 +778,24 @@ exports.removeTar = function* (next) {
|
||||
var name = this.params.name || this.params[0];
|
||||
var filename = this.params.filename || this.params[1];
|
||||
var id = Number(this.params.rev || this.params[2]);
|
||||
debug('remove tarball with filename: %s, id: %s', filename, id);
|
||||
// cnpmjs.org-2.0.0.tgz
|
||||
var version = filename.split(name + '-')[1];
|
||||
if (version) {
|
||||
// 2.0.0.tgz
|
||||
version = version.substring(0, version.lastIndexOf('.tgz'));
|
||||
}
|
||||
if (!version) {
|
||||
return yield* next;
|
||||
}
|
||||
|
||||
debug('remove tarball with filename: %s, version: %s, revert to => rev id: %s', filename, version, id);
|
||||
|
||||
var username = this.user.name;
|
||||
if (isNaN(id)) {
|
||||
return yield* next;
|
||||
}
|
||||
|
||||
var mod = yield Module.getById(id);
|
||||
if (!mod || mod.name !== name) {
|
||||
return yield* next;
|
||||
}
|
||||
|
||||
var isMaintainer = yield* packageService.isMaintainer(name, username);
|
||||
|
||||
if (!isMaintainer && !this.user.isAdmin) {
|
||||
this.status = 403;
|
||||
this.body = {
|
||||
@@ -807,9 +804,33 @@ exports.removeTar = function* (next) {
|
||||
};
|
||||
return;
|
||||
}
|
||||
var key = mod.package.dist && mod.package.dist.key;
|
||||
|
||||
var rs = yield [
|
||||
Module.getById(id),
|
||||
Module.get(name, version),
|
||||
];
|
||||
var revertTo = rs[0];
|
||||
var mod = rs[1]; // module need to delete
|
||||
if (!mod || mod.name !== name) {
|
||||
return yield* next;
|
||||
}
|
||||
|
||||
var key = mod.package && mod.package.dist && mod.package.dist.key;
|
||||
key = key || common.getCDNKey(mod.name, filename);
|
||||
yield nfs.remove(key);
|
||||
|
||||
if (revertTo && revertTo.package) {
|
||||
debug('removing key: %s from nfs, revert to %s@%s', key, revertTo.name, revertTo.package.version);
|
||||
} else {
|
||||
debug('removing key: %s from nfs, no revert mod', key);
|
||||
}
|
||||
try {
|
||||
yield nfs.remove(key);
|
||||
} catch (err) {
|
||||
logger.error(err);
|
||||
}
|
||||
// remove version from table
|
||||
yield Module.removeByNameAndVersions(name, [version]);
|
||||
debug('removed %s@%s', name, version);
|
||||
this.body = { ok: true };
|
||||
};
|
||||
|
||||
@@ -858,6 +879,9 @@ exports.removeAll = function* (next) {
|
||||
}
|
||||
}
|
||||
|
||||
// remove the maintainers
|
||||
yield* packageService.removeAllMaintainers(name);
|
||||
|
||||
this.body = { ok: true };
|
||||
};
|
||||
|
||||
@@ -894,6 +918,8 @@ exports.listAllModules = function *() {
|
||||
this.body = result;
|
||||
};
|
||||
|
||||
var A_WEEK_MS = 3600000 * 24 * 7;
|
||||
|
||||
exports.listAllModulesSince = function *() {
|
||||
var query = this.query || {};
|
||||
if (query.stale !== 'update_after') {
|
||||
@@ -908,6 +934,11 @@ exports.listAllModulesSince = function *() {
|
||||
debug('list all modules from %s', query.startkey);
|
||||
var startkey = Number(query.startkey) || 0;
|
||||
var updated = Date.now();
|
||||
if (updated - startkey > A_WEEK_MS) {
|
||||
startkey = updated - A_WEEK_MS;
|
||||
console.warn('[%s] list modules since time out of range: query: %j, ip: %s',
|
||||
Date(), query, this.ip);
|
||||
}
|
||||
var mods = yield Module.listSince(startkey);
|
||||
var result = { _updated: updated };
|
||||
mods.forEach(function (mod) {
|
||||
|
||||
@@ -214,6 +214,22 @@ exports.add = function* () {
|
||||
};
|
||||
|
||||
// logined before update, no need to auth user again
|
||||
// { name: 'admin',
|
||||
// password: '123123',
|
||||
// email: 'fengmk2@gmail.com',
|
||||
// _id: 'org.couchdb.user:admin',
|
||||
// type: 'user',
|
||||
// roles: [],
|
||||
// date: '2014-08-05T16:08:22.645Z',
|
||||
// _rev: '1-1a18c3d73ba42e863523a399ff3304d8',
|
||||
// _cnpm_meta:
|
||||
// { id: 14,
|
||||
// npm_user: false,
|
||||
// custom_user: false,
|
||||
// gmt_create: '2014-08-05T15:46:58.000Z',
|
||||
// gmt_modified: '2014-08-05T15:46:58.000Z',
|
||||
// admin: true,
|
||||
// scopes: [ '@cnpm', '@cnpmtest' ] } }
|
||||
exports.update = function *(next) {
|
||||
var name = this.params.name;
|
||||
var rev = this.params.rev;
|
||||
|
||||
@@ -44,10 +44,12 @@ exports.display = function* (next) {
|
||||
return yield* next;
|
||||
}
|
||||
|
||||
user = user || {};
|
||||
|
||||
var data = {
|
||||
name: name,
|
||||
email: user && user.email,
|
||||
json: user && user.json || {}
|
||||
email: user.email,
|
||||
json: user.json || {}
|
||||
};
|
||||
|
||||
if (data.json.login) {
|
||||
@@ -72,7 +74,7 @@ exports.display = function* (next) {
|
||||
title: 'User - ' + name,
|
||||
packages: packages,
|
||||
user: data,
|
||||
lastModified: user.gmt_modified,
|
||||
lastModified: user && user.gmt_modified,
|
||||
isAdmin: isAdmin,
|
||||
scopes: scopes
|
||||
});
|
||||
|
||||
28
dispatch.js
28
dispatch.js
@@ -18,6 +18,7 @@
|
||||
var path = require('path');
|
||||
var util = require('util');
|
||||
var cluster = require('cluster');
|
||||
var cfork = require('cfork');
|
||||
var config = require('./config');
|
||||
var workerPath = path.join(__dirname, 'worker.js');
|
||||
var childProcess = require('child_process');
|
||||
@@ -36,32 +37,21 @@ if (config.enableCluster) {
|
||||
}
|
||||
|
||||
function forkWorker() {
|
||||
cluster.setupMaster({
|
||||
exec: workerPath
|
||||
});
|
||||
|
||||
cluster.on('fork', function (worker) {
|
||||
cfork({
|
||||
exec: workerPath,
|
||||
count: config.numCPUs,
|
||||
}).on('fork', function (worker) {
|
||||
console.log('[%s] [worker:%d] new worker start', Date(), worker.process.pid);
|
||||
});
|
||||
|
||||
cluster.on('disconnect', function (worker) {
|
||||
var w = cluster.fork();
|
||||
console.error('[%s] [master:%s] wroker:%s disconnect, suicide: %s, state: %s. New worker:%s fork',
|
||||
Date(), process.pid, worker.process.pid, worker.suicide, worker.state, w.process.pid);
|
||||
});
|
||||
|
||||
cluster.on('exit', function (worker, code, signal) {
|
||||
}).on('disconnect', function (worker) {
|
||||
console.error('[%s] [master:%s] wroker:%s disconnect, suicide: %s, state: %s.',
|
||||
Date(), process.pid, worker.process.pid, worker.suicide, worker.state);
|
||||
}).on('exit', function (worker, code, signal) {
|
||||
var exitCode = worker.process.exitCode;
|
||||
var err = new Error(util.format('worker %s died (code: %s, signal: %s, suicide: %s, state: %s)',
|
||||
worker.process.pid, exitCode, signal, worker.suicide, worker.state));
|
||||
err.name = 'WorkerDiedError';
|
||||
console.error('[%s] [master:%s] wroker exit: %s', Date(), process.pid, err.stack);
|
||||
});
|
||||
|
||||
// Fork workers.
|
||||
for (var i = 0; i < config.numCPUs; i++) {
|
||||
cluster.fork();
|
||||
}
|
||||
}
|
||||
|
||||
function forkSyncer() {
|
||||
|
||||
833
docs/registry-api.md
Normal file
833
docs/registry-api.md
Normal file
@@ -0,0 +1,833 @@
|
||||
# NPM Registry API
|
||||
|
||||
## Overview
|
||||
|
||||
* [Schema](/docs/registry-api.md#schema)
|
||||
* [Client Errors](/docs/registry-api.md#client-errors)
|
||||
* [Authentication](/docs/registry-api.md#authentication)
|
||||
* [Package](/docs/registry-api.md#package)
|
||||
* [User](/docs/registry-api.md#user)
|
||||
* [Search](/docs/registry-api.md#search)
|
||||
|
||||
## Schema
|
||||
|
||||
All API access is over HTTPS or HTTP,
|
||||
and accessed from the `registry.npmjs.org` domain.
|
||||
All data is sent and received as JSON.
|
||||
|
||||
```bash
|
||||
$ curl -i https://registry.npmjs.org
|
||||
|
||||
HTTP/1.1 200 OK
|
||||
Date: Tue, 05 Aug 2014 10:53:24 GMT
|
||||
Server: CouchDB/1.5.0 (Erlang OTP/R16B03)
|
||||
Content-Type: text/plain; charset=utf-8
|
||||
Cache-Control: max-age=60
|
||||
Content-Length: 258
|
||||
Accept-Ranges: bytes
|
||||
Via: 1.1 varnish
|
||||
Age: 11
|
||||
X-Served-By: cache-ty67-TYO
|
||||
X-Cache: HIT
|
||||
X-Cache-Hits: 1
|
||||
X-Timer: S1407236004.867906,VS0,VE0
|
||||
|
||||
{"db_name":"registry","doc_count":90789,"doc_del_count":381,"update_seq":137250,"purge_seq":0,
|
||||
"compact_running":false,"disk_size":436228219,"data_size":332875061,
|
||||
"instance_start_time":"1405721973718703","disk_format_version":6,"committed_update_seq":137250}
|
||||
```
|
||||
|
||||
## Client Errors
|
||||
|
||||
```json
|
||||
Status: 4xx
|
||||
|
||||
{
|
||||
"error": "error_name",
|
||||
"reason": "error reason string"
|
||||
}
|
||||
```
|
||||
|
||||
## Authentication
|
||||
|
||||
There is only one way to authenticate through the API.
|
||||
|
||||
## Basic Authentication
|
||||
|
||||
```bash
|
||||
$ curl -u "username:password" https://registry.npmjs.org
|
||||
```
|
||||
|
||||
## Failed login limit
|
||||
|
||||
```bash
|
||||
$ curl -i -X PUT -u foo:pwd \
|
||||
-d '{"name":"foo","email":"foo@bar.com","type":"user","roles":[]}' \
|
||||
https://registry.npmjs.org/-/user/org.couchdb.user:foo/-rev/11-d226c6afa9286ab5b9eb858c429bdabf
|
||||
|
||||
HTTP/1.1 401 Unauthorized
|
||||
Date: Tue, 05 Aug 2014 15:33:25 GMT
|
||||
Server: CouchDB/1.5.0 (Erlang OTP/R14B04)
|
||||
Content-Type: text/plain; charset=utf-8
|
||||
Cache-Control: max-age=60
|
||||
Content-Length: 67
|
||||
Accept-Ranges: bytes
|
||||
Via: 1.1 varnish
|
||||
X-Served-By: cache-ty66-TYO
|
||||
X-Cache: MISS
|
||||
X-Cache-Hits: 0
|
||||
X-Timer: S1407252805.261390,VS0,VE434
|
||||
|
||||
{"error":"unauthorized","reason":"Name or password is incorrect."}
|
||||
```
|
||||
|
||||
## Package
|
||||
|
||||
* Read
|
||||
* [Get a single package](/docs/registry-api.md#get-a-single-package)
|
||||
* [Get a special version or tag package](/docs/registry-api.md#get-a-special-version-or-tag-package)
|
||||
* [List packages since from a update time](/docs/registry-api.md#list-packages-since-from-a-update-time)
|
||||
* Write
|
||||
* [Publish a new package](/docs/registry-api.md#publish-a-new-package)
|
||||
* [Update a package's tag](/docs/registry-api.md#update-a-packages-tag)
|
||||
* [Update a package's maintainers](/docs/registry-api.md#update-a-packages-maintainers)
|
||||
* [Remove one version from package](/docs/registry-api.md#remove-one-version-from-package)
|
||||
* [Remove a tgz file from package](/docs/registry-api.md#remove-a-tgz-file-from-package)
|
||||
|
||||
### Get a single package
|
||||
|
||||
```
|
||||
GET /:package
|
||||
```
|
||||
|
||||
#### Response
|
||||
|
||||
```json
|
||||
HTTP/1.1 200 OK
|
||||
Etag: "8UDCP753LFXOG42NMX88JAN40"
|
||||
Content-Type: application/json
|
||||
Cache-Control: max-age=60
|
||||
Content-Length: 2243
|
||||
|
||||
{
|
||||
"_id": "pedding",
|
||||
"_rev": "11-e6d1e6e96eaf72433fef6aaabe843af8",
|
||||
"name": "pedding",
|
||||
"description": "Just pedding for callback.",
|
||||
"dist-tags": {
|
||||
"latest": "1.0.0"
|
||||
},
|
||||
"versions": {
|
||||
"1.0.0": {
|
||||
"name": "pedding",
|
||||
"version": "1.0.0",
|
||||
"description": "Just pedding for callback.",
|
||||
"main": "index.js",
|
||||
"scripts": {
|
||||
"test": "make test-all"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git://github.com/fengmk2/pedding.git"
|
||||
},
|
||||
"keywords": [
|
||||
"pedding",
|
||||
"callback"
|
||||
],
|
||||
"devDependencies": {
|
||||
"contributors": "*",
|
||||
"mocha": "*",
|
||||
"mocha-phantomjs": "*",
|
||||
"component": "*",
|
||||
"chai": "*"
|
||||
},
|
||||
"author": {
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com"
|
||||
},
|
||||
"license": "MIT",
|
||||
"contributors": [
|
||||
{
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com",
|
||||
"url": "https://github.com/fengmk2"
|
||||
},
|
||||
{
|
||||
"name": "dead-horse",
|
||||
"email": "dead_horse@qq.com",
|
||||
"url": "https://github.com/dead-horse"
|
||||
}
|
||||
],
|
||||
"gitHead": "b42a708414a704336e9dee570a963e2dbe43e529",
|
||||
"bugs": {
|
||||
"url": "https://github.com/fengmk2/pedding/issues"
|
||||
},
|
||||
"homepage": "https://github.com/fengmk2/pedding",
|
||||
"_id": "pedding@1.0.0",
|
||||
"_shasum": "7f5098d60307b4ef7240c3d693cb20a9473c6074",
|
||||
"_from": ".",
|
||||
"_npmVersion": "1.4.13",
|
||||
"_npmUser": {
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com"
|
||||
},
|
||||
"maintainers": [
|
||||
{
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com"
|
||||
},
|
||||
{
|
||||
"name": "dead-horse",
|
||||
"email": "dead_horse@qq.com"
|
||||
}
|
||||
],
|
||||
"dist": {
|
||||
"shasum": "7f5098d60307b4ef7240c3d693cb20a9473c6074",
|
||||
"tarball": "http://registry.npmjs.org/pedding/-/pedding-1.0.0.tgz"
|
||||
},
|
||||
"directories": {}
|
||||
}
|
||||
},
|
||||
"readme": "# pedding\n readme...",
|
||||
"maintainers": [
|
||||
{
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com"
|
||||
},
|
||||
{
|
||||
"name": "dead-horse",
|
||||
"email": "dead_horse@qq.com"
|
||||
},
|
||||
{
|
||||
"name": "dead_horse",
|
||||
"email": "dead_horse@qq.com"
|
||||
}
|
||||
],
|
||||
"time": {
|
||||
"modified": "2014-07-05T14:22:53.849Z",
|
||||
"created": "2012-09-18T14:46:08.346Z",
|
||||
"0.0.1": "2012-09-18T14:46:21.321Z",
|
||||
"0.0.2": "2013-06-22T08:26:45.125Z",
|
||||
"0.0.3": "2013-07-02T15:20:34.707Z",
|
||||
"1.0.0": "2014-07-05T11:08:51.614Z"
|
||||
},
|
||||
"author": {
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git://github.com/fengmk2/pedding.git"
|
||||
},
|
||||
"keywords": [
|
||||
"pedding",
|
||||
"callback"
|
||||
],
|
||||
"bugs": {
|
||||
"url": "https://github.com/fengmk2/pedding/issues"
|
||||
},
|
||||
"license": "MIT",
|
||||
"readmeFilename": "README.md",
|
||||
"homepage": "https://github.com/fengmk2/pedding",
|
||||
"contributors": [
|
||||
{
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com",
|
||||
"url": "https://github.com/fengmk2"
|
||||
},
|
||||
{
|
||||
"name": "dead-horse",
|
||||
"email": "dead_horse@qq.com",
|
||||
"url": "https://github.com/dead-horse"
|
||||
}
|
||||
],
|
||||
"_attachments": {}
|
||||
}
|
||||
```
|
||||
|
||||
### Get a special version or tag package
|
||||
|
||||
```
|
||||
GET /:package/:tag_or_version
|
||||
```
|
||||
|
||||
#### Reponse
|
||||
|
||||
```json
|
||||
HTTP/1.1 200 OK
|
||||
Etag: "1WJ4JF535RO3BDZR2BARXSGLY"
|
||||
Content-Type: application/json
|
||||
Cache-Control: max-age=60
|
||||
Content-Length: 1183
|
||||
|
||||
{
|
||||
"name": "pedding",
|
||||
"version": "1.0.0",
|
||||
"description": "Just pedding for callback.",
|
||||
"main": "index.js",
|
||||
"scripts": {
|
||||
"test": "make test-all"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git://github.com/fengmk2/pedding.git"
|
||||
},
|
||||
"keywords": [
|
||||
"pedding",
|
||||
"callback"
|
||||
],
|
||||
"devDependencies": {
|
||||
"contributors": "*",
|
||||
"mocha": "*",
|
||||
"mocha-phantomjs": "*",
|
||||
"component": "*",
|
||||
"chai": "*"
|
||||
},
|
||||
"author": {
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com"
|
||||
},
|
||||
"license": "MIT",
|
||||
"contributors": [
|
||||
{
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com",
|
||||
"url": "https://github.com/fengmk2"
|
||||
},
|
||||
{
|
||||
"name": "dead-horse",
|
||||
"email": "dead_horse@qq.com",
|
||||
"url": "https://github.com/dead-horse"
|
||||
}
|
||||
],
|
||||
"gitHead": "b42a708414a704336e9dee570a963e2dbe43e529",
|
||||
"bugs": {
|
||||
"url": "https://github.com/fengmk2/pedding/issues"
|
||||
},
|
||||
"homepage": "https://github.com/fengmk2/pedding",
|
||||
"_id": "pedding@1.0.0",
|
||||
"_shasum": "7f5098d60307b4ef7240c3d693cb20a9473c6074",
|
||||
"_from": ".",
|
||||
"_npmVersion": "1.4.13",
|
||||
"_npmUser": {
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com"
|
||||
},
|
||||
"maintainers": [
|
||||
{
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com"
|
||||
},
|
||||
{
|
||||
"name": "dead-horse",
|
||||
"email": "dead_horse@qq.com"
|
||||
}
|
||||
],
|
||||
"dist": {
|
||||
"shasum": "7f5098d60307b4ef7240c3d693cb20a9473c6074",
|
||||
"tarball": "http://registry.npmjs.org/pedding/-/pedding-1.0.0.tgz"
|
||||
},
|
||||
"directories": {}
|
||||
}
|
||||
```
|
||||
|
||||
### Publish a new package
|
||||
|
||||
* Authentication required.
|
||||
|
||||
```
|
||||
PUT /:package
|
||||
```
|
||||
|
||||
#### Input
|
||||
|
||||
```json
|
||||
{
|
||||
"_id": "pedding",
|
||||
"name": "pedding",
|
||||
"description": "Just pedding for callback.",
|
||||
"dist-tags": {
|
||||
"latest": "1.0.0"
|
||||
},
|
||||
"versions": {
|
||||
"1.0.0": {
|
||||
"name": "pedding",
|
||||
"version": "1.0.0",
|
||||
"description": "Just pedding for callback.",
|
||||
"main": "index.js",
|
||||
"scripts": {
|
||||
"test": "make test-all"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git://github.com/fengmk2/pedding.git"
|
||||
},
|
||||
"keywords": [ "pedding","callback" ],
|
||||
"devDependencies": {
|
||||
"contributors": "*",
|
||||
"mocha": "*",
|
||||
"mocha-phantomjs": "*",
|
||||
"component": "*",
|
||||
"chai": "*"
|
||||
},
|
||||
"dependencies": {},
|
||||
"author": {
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com"
|
||||
},
|
||||
"license": "MIT",
|
||||
"contributors": [
|
||||
{
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com",
|
||||
"url": "https://github.com/fengmk2"
|
||||
},
|
||||
{
|
||||
"name": "dead-horse",
|
||||
"email": "dead_horse@qq.com",
|
||||
"url": "https://github.com/dead-horse"
|
||||
}
|
||||
],
|
||||
"readme": "# pedding ...",
|
||||
"readmeFilename": "README.md",
|
||||
"gitHead": "b42a708414a704336e9dee570a963e2dbe43e529",
|
||||
"bugs": {
|
||||
"url": "https://github.com/fengmk2/pedding/issues"
|
||||
},
|
||||
"homepage": "https://github.com/fengmk2/pedding",
|
||||
"_id": "pedding@1.0.0",
|
||||
"_shasum": "7f5098d60307b4ef7240c3d693cb20a9473c6074",
|
||||
"_from": ".",
|
||||
"_npmVersion": "1.5.0-alpha-4",
|
||||
"_npmUser": {
|
||||
"name": "admin",
|
||||
"email": "fengmk2@gmail.com"
|
||||
},
|
||||
"maintainers": [
|
||||
{
|
||||
"name": "admin",
|
||||
"email": "fengmk2@gmail.com"
|
||||
}
|
||||
],
|
||||
"dist": {
|
||||
"shasum": "7f5098d60307b4ef7240c3d693cb20a9473c6074",
|
||||
"tarball": "https://registry.npmjs.org/pedding/-/pedding-1.0.0.tgz"
|
||||
}
|
||||
}
|
||||
},
|
||||
"readme": "# pedding ...",
|
||||
"maintainers": [
|
||||
{
|
||||
"name": "admin",
|
||||
"email": "fengmk2@gmail.com"
|
||||
}
|
||||
],
|
||||
"_attachments": {
|
||||
"pedding-1.0.0.tgz":{
|
||||
"content_type": "application/octet-stream",
|
||||
"data": "H4sIAAAAAAAAA+0aa3PbNjKf8Su...",
|
||||
"length": 2107
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
#### Response
|
||||
|
||||
```json
|
||||
Status: 201 Created
|
||||
|
||||
{
|
||||
"ok": true,
|
||||
"rev": "11-e6d1e6e96eaf72433fef6aaabe843af8"
|
||||
}
|
||||
```
|
||||
|
||||
### Update a package's tag
|
||||
|
||||
* Authentication required.
|
||||
|
||||
```
|
||||
PUT /:package/:tag
|
||||
```
|
||||
|
||||
#### Input
|
||||
|
||||
The total input body is the `version` string which's setting to the tag.
|
||||
|
||||
```json
|
||||
"1.0.0"
|
||||
```
|
||||
|
||||
#### Response
|
||||
|
||||
```json
|
||||
Status: 201 Created
|
||||
|
||||
{
|
||||
"ok": true
|
||||
}
|
||||
```
|
||||
|
||||
### Update a package's maintainers
|
||||
|
||||
* Authentication required.
|
||||
|
||||
```
|
||||
PUT /:package/-rev/:rev
|
||||
```
|
||||
|
||||
#### Input
|
||||
|
||||
```json
|
||||
{
|
||||
"_id": "pedding",
|
||||
"_rev": "11-e6d1e6e96eaf72433fef6aaabe843af8",
|
||||
"maintainers":[
|
||||
{ "name": "fengmk2", "email": "fengmk2@gmail.com" },
|
||||
{ "name": "dead-horse", "email": "dead_horse@qq.com" }
|
||||
]
|
||||
}
|
||||
```
|
||||
|
||||
#### Response
|
||||
|
||||
```json
|
||||
Status: 201 Created
|
||||
|
||||
{
|
||||
"ok": true,
|
||||
"id": "pedding",
|
||||
"rev": "12-bb300a90c9aeb779748b83ec1b744039"
|
||||
}
|
||||
```
|
||||
|
||||
### Remove one version from package
|
||||
|
||||
* Authentication required.
|
||||
|
||||
```
|
||||
PUT /:package/-rev/:rev
|
||||
```
|
||||
|
||||
#### Input
|
||||
|
||||
Example for removing `0.0.1` version:
|
||||
|
||||
```json
|
||||
{
|
||||
"_id": "pedding",
|
||||
"_rev": "12-bb300a90c9aeb779748b83ec1b744039",
|
||||
"name": "pedding",
|
||||
"description": "desc",
|
||||
"dist-tags": { "latest": "1.0.0" },
|
||||
"maintainers":
|
||||
[ ... ],
|
||||
"time":
|
||||
{ ... },
|
||||
"users": {},
|
||||
"author": { ... },
|
||||
"repository": { ... },
|
||||
"versions":
|
||||
{ "1.0.0":
|
||||
{ ... },
|
||||
"0.0.3":
|
||||
{ ... },
|
||||
"0.0.2":
|
||||
{ ... } },
|
||||
"readme": "...",
|
||||
"homepage": "https://github.com/fengmk2/pedding",
|
||||
"bugs": { ... },
|
||||
"license": "MIT" }
|
||||
```
|
||||
|
||||
#### Response
|
||||
|
||||
```json
|
||||
Status: 201 Created
|
||||
|
||||
{
|
||||
"ok": true
|
||||
}
|
||||
```
|
||||
|
||||
### Remove a tgz file from package
|
||||
|
||||
* Authentication required.
|
||||
|
||||
```
|
||||
DELETE /:tgzfilepath/-rev/:rev
|
||||
```
|
||||
|
||||
Exmaple for removing `https://registry.npmjs.org/pedding/-/pedding-0.0.1.tgz` file:
|
||||
|
||||
```
|
||||
DELETE /pedding/-/pedding-0.0.1.tgz/-rev/12-bb300a90c9aeb779748b83ec1b744039
|
||||
```
|
||||
|
||||
#### Response
|
||||
|
||||
```json
|
||||
Status: 200 OK
|
||||
|
||||
{
|
||||
"ok": true
|
||||
}
|
||||
```
|
||||
|
||||
### List packages since from a update time
|
||||
|
||||
```
|
||||
GET /-/all/since?stale=update_after&startkey=:startkey
|
||||
```
|
||||
|
||||
* `startkey` is a ms timestamp
|
||||
|
||||
#### Response
|
||||
|
||||
```bash
|
||||
$ curl -i "https://registry.npmjs.org/-/all/since?stale=update_after&startkey=1407255748643"
|
||||
```
|
||||
|
||||
```json
|
||||
HTTP/1.1 200 OK
|
||||
|
||||
{
|
||||
"_updated": 1407255883282,
|
||||
"bacon-and-eggs": {
|
||||
"name": "bacon-and-eggs",
|
||||
"description": "A functional reactive Twitter API client in node",
|
||||
"dist-tags": {
|
||||
"latest": "0.0.4"
|
||||
},
|
||||
"maintainers": [
|
||||
{
|
||||
"name": "mikegroseclose",
|
||||
"email": "mike.groseclose@gmail.com"
|
||||
}
|
||||
],
|
||||
"homepage": "http://github.com/mikegroseclose/bacon-and-eggs",
|
||||
"keywords": [
|
||||
"twitter",
|
||||
"api",
|
||||
"frp",
|
||||
"functional",
|
||||
"reactive",
|
||||
"bacon",
|
||||
"eggs",
|
||||
"oauth",
|
||||
"stream",
|
||||
"streams"
|
||||
],
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git://github.com/mikegroseclose/gulp-regex-replace.git"
|
||||
},
|
||||
"author": {
|
||||
"name": "Mike Groseclose",
|
||||
"email": "mike.groseclose@gmail.com",
|
||||
"url": "http://mikegroseclose.com"
|
||||
},
|
||||
"bugs": {
|
||||
"url": "https://github.com/mikegroseclose/gulp-regex-replace/issues"
|
||||
},
|
||||
"readmeFilename": "README.md",
|
||||
"time": {
|
||||
"modified": "2014-08-05T16:21:17.041Z"
|
||||
},
|
||||
"versions": {
|
||||
"0.0.4": "latest"
|
||||
}
|
||||
},
|
||||
"git-perm-rm": {
|
||||
"name": "git-perm-rm",
|
||||
"description": "Permanently remove a file or directory from a git repo including all related commit records.",
|
||||
"dist-tags": {
|
||||
"latest": "1.0.1"
|
||||
},
|
||||
"maintainers": [
|
||||
{
|
||||
"name": "kael",
|
||||
"email": "i@kael.me"
|
||||
}
|
||||
],
|
||||
"homepage": "https://github.com/kaelzhang/git-perm-rm",
|
||||
"keywords": [
|
||||
"git",
|
||||
"rm",
|
||||
"git-perm-rm",
|
||||
"remove",
|
||||
"permanently"
|
||||
],
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git://github.com/kaelzhang/git-perm-rm.git"
|
||||
},
|
||||
"author": {
|
||||
"name": "Kael"
|
||||
},
|
||||
"bugs": {
|
||||
"url": "https://github.com/kaelzhang/git-perm-rm/issues"
|
||||
},
|
||||
"license": "MIT",
|
||||
"readmeFilename": "README.md",
|
||||
"time": {
|
||||
"modified": "2014-08-05T16:22:41.253Z"
|
||||
},
|
||||
"versions": {
|
||||
"1.0.1": "latest"
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
## User
|
||||
|
||||
* [Get a single user](/docs/registry-api.md#get-a-single-user)
|
||||
* [Add a new user](/docs/registry-api.md#add-a-new-user)
|
||||
* [Update a exists user](/docs/registry-api.md#update-a-exists-user)
|
||||
|
||||
### Get a single user
|
||||
|
||||
```
|
||||
GET /-/user/org.couchdb.user::username
|
||||
```
|
||||
|
||||
#### Response
|
||||
|
||||
```json
|
||||
HTTP/1.1 200 OK
|
||||
ETag: "32-984ee97e01aea166dcab6d1517c730e3"
|
||||
|
||||
{
|
||||
"_id": "org.couchdb.user:fengmk2",
|
||||
"_rev": "32-984ee97e01aea166dcab6d1517c730e3",
|
||||
"name": "fengmk2",
|
||||
"email": "fengmk2@gmail.com",
|
||||
"type": "user",
|
||||
"roles": [],
|
||||
"date": "2014-08-04T10:43:07.063Z",
|
||||
"fullname": "fengmk2",
|
||||
"avatar": "https://secure.gravatar.com/avatar/95b9d41231617a05ced5604d242c9670?s=50&d=retro",
|
||||
"freenode": "",
|
||||
"github": "fengmk2",
|
||||
"homepage": "http://fengmk2.github.com",
|
||||
"twitter": "fengmk2",
|
||||
"avatarMedium": "https://secure.gravatar.com/avatar/95b9d41231617a05ced5604d242c9670?s=100&d=retro",
|
||||
"avatarLarge": "https://secure.gravatar.com/avatar/95b9d41231617a05ced5604d242c9670?s=496&d=retro",
|
||||
"fields": [
|
||||
{
|
||||
"name": "fullname",
|
||||
"value": "fengmk2",
|
||||
"title": "Full Name",
|
||||
"show": "fengmk2"
|
||||
},
|
||||
{
|
||||
"name": "email",
|
||||
"value": "fengmk2@gmail.com",
|
||||
"title": "Email",
|
||||
"show": "<a href=\"mailto:fengmk2@gmail.com\">fengmk2@gmail.com</a>"
|
||||
},
|
||||
{
|
||||
"name": "github",
|
||||
"value": "fengmk2",
|
||||
"title": "Github",
|
||||
"show": "<a rel=\"me\" href=\"https://github.com/fengmk2\">fengmk2</a>"
|
||||
},
|
||||
{
|
||||
"name": "twitter",
|
||||
"value": "fengmk2",
|
||||
"title": "Twitter",
|
||||
"show": "<a rel=\"me\" href=\"https://twitter.com/fengmk2\">@fengmk2</a>"
|
||||
},
|
||||
{
|
||||
"name": "appdotnet",
|
||||
"value": "",
|
||||
"title": "App.net",
|
||||
"show": ""
|
||||
},
|
||||
{
|
||||
"name": "homepage",
|
||||
"value": "http://fengmk2.github.com",
|
||||
"title": "Homepage",
|
||||
"show": "<a rel=\"me\" href=\"http://fengmk2.github.com/\">http://fengmk2.github.com</a>"
|
||||
},
|
||||
{
|
||||
"name": "freenode",
|
||||
"value": "",
|
||||
"title": "IRC Handle",
|
||||
"show": ""
|
||||
}
|
||||
],
|
||||
"appdotnet": "fengmk2"
|
||||
}
|
||||
```
|
||||
|
||||
### Add a new user
|
||||
|
||||
```
|
||||
PUT /-/user/org.couchdb.user::username
|
||||
```
|
||||
|
||||
#### Input
|
||||
|
||||
```json
|
||||
{
|
||||
"name": "admin",
|
||||
"password": "123",
|
||||
"email": "fengmk2@gmail.com",
|
||||
"_id": "org.couchdb.user:admin",
|
||||
"type": "user",
|
||||
"roles": [],
|
||||
"date": "2014-08-05T16:05:17.792Z"
|
||||
}
|
||||
```
|
||||
|
||||
#### Response
|
||||
|
||||
```json
|
||||
Status: 201 Created
|
||||
|
||||
{
|
||||
"ok": true,
|
||||
"id": "org.couchdb.user:fengmk2",
|
||||
"rev": "32-984ee97e01aea166dcab6d1517c730e3"
|
||||
}
|
||||
```
|
||||
|
||||
### Update a exists user
|
||||
|
||||
* Authentication required.
|
||||
|
||||
```
|
||||
PUT /-/user/org.couchdb.user::username/-rev/:rev
|
||||
```
|
||||
|
||||
#### Input
|
||||
|
||||
```json
|
||||
{
|
||||
"name": "admin",
|
||||
"password": "123",
|
||||
"email": "fengmk2@gmail.com",
|
||||
"_id": "org.couchdb.user:admin",
|
||||
"type": "user",
|
||||
"roles": [],
|
||||
"date": "2014-08-05T16:05:17.792Z",
|
||||
"_rev": "2-1a18c3d73ba42e863523a399ff3304d8"
|
||||
}
|
||||
```
|
||||
|
||||
#### Response
|
||||
|
||||
```json
|
||||
Status: 201 Created
|
||||
|
||||
{
|
||||
"ok": true,
|
||||
"id": "org.couchdb.user:fengmk2",
|
||||
"rev": "3-bb300a90c9aeb779748b83ec1b744039"
|
||||
}
|
||||
```
|
||||
|
||||
## Search
|
||||
83
package.json
83
package.json
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "cnpmjs.org",
|
||||
"version": "1.0.3",
|
||||
"version": "1.1.0",
|
||||
"description": "Private npm registry and web for Enterprise, base on MySQL and Simple Store Service",
|
||||
"main": "index.js",
|
||||
"scripts": {
|
||||
@@ -10,57 +10,58 @@
|
||||
"stop": "./bin/nodejsctl stop"
|
||||
},
|
||||
"dependencies": {
|
||||
"bytes": "1.0.0",
|
||||
"cheerio": "0.17.0",
|
||||
"co": "3.1.0",
|
||||
"co-defer": "0.1.0",
|
||||
"co-gather": "0.0.1",
|
||||
"co-read": "0.1.0",
|
||||
"co-redis": "1.1.0",
|
||||
"co-urllib": "0.2.3",
|
||||
"co-write": "0.3.0",
|
||||
"copy-to": "1.0.1",
|
||||
"debug": "1.0.4",
|
||||
"error-formater": "1.0.3",
|
||||
"eventproxy": "0.3.1",
|
||||
"giturl": "0.0.3",
|
||||
"graceful": "0.1.0",
|
||||
"gravatar": "1.0.6",
|
||||
"humanize-number": "0.0.2",
|
||||
"koa": "0.8.2",
|
||||
"koa-limit": "1.0.2",
|
||||
"koa-markdown": "0.0.3",
|
||||
"koa-middlewares": "1.2.0",
|
||||
"marked": "0.3.2",
|
||||
"mime": "1.2.11",
|
||||
"mini-logger": "0.3.0",
|
||||
"mkdirp": "0.5.0",
|
||||
"moment": "2.8.0",
|
||||
"ms": "0.6.2",
|
||||
"multiline": "0.3.4",
|
||||
"mysql": "2.4.1",
|
||||
"bytes": "~1.0.0",
|
||||
"cfork": "~1.0.1",
|
||||
"cheerio": "~0.17.0",
|
||||
"co": "~3.1.0",
|
||||
"co-defer": "~0.1.0",
|
||||
"co-gather": "~0.0.1",
|
||||
"co-read": "~0.1.0",
|
||||
"co-redis": "~1.1.0",
|
||||
"co-urllib": "~0.2.3",
|
||||
"co-write": "~0.3.0",
|
||||
"copy-to": "~1.0.1",
|
||||
"debug": "~1.0.4",
|
||||
"error-formater": "~1.0.3",
|
||||
"eventproxy": "~0.3.1",
|
||||
"giturl": "~0.0.3",
|
||||
"graceful": "~0.1.0",
|
||||
"gravatar": "~1.0.6",
|
||||
"humanize-number": "~0.0.2",
|
||||
"koa": "~0.8.2",
|
||||
"koa-limit": "~1.0.2",
|
||||
"koa-markdown": "~0.0.3",
|
||||
"koa-middlewares": "~1.2.0",
|
||||
"marked": "~0.3.2",
|
||||
"mime": "~1.2.11",
|
||||
"mini-logger": "~0.3.0",
|
||||
"mkdirp": "~0.5.0",
|
||||
"moment": "~2.8.1",
|
||||
"ms": "~0.6.2",
|
||||
"multiline": "~0.3.4",
|
||||
"mysql": "~2.4.2",
|
||||
"nodemailer": "0.7.1",
|
||||
"qn": "0.2.2",
|
||||
"ready": "0.1.1",
|
||||
"redis": "0.11.0",
|
||||
"semver": "3.0.1",
|
||||
"thunkify-wrap": "1.0.2",
|
||||
"utility": "0.1.16"
|
||||
"qn": "~0.2.2",
|
||||
"ready": "~0.1.1",
|
||||
"redis": "~0.11.0",
|
||||
"semver": "~3.0.1",
|
||||
"thunkify-wrap": "~1.0.2",
|
||||
"utility": "~1.0.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"autod": "~0.2.0",
|
||||
"chunkstream": "0.0.1",
|
||||
"chunkstream": "~0.0.1",
|
||||
"co-mocha": "0.0.2",
|
||||
"contributors": "*",
|
||||
"cov": "*",
|
||||
"istanbul-harmony": "*",
|
||||
"jshint": "*",
|
||||
"mm": "0.2.1",
|
||||
"mm": "~0.2.1",
|
||||
"mocha": "*",
|
||||
"pedding": "1.0.0",
|
||||
"should": "4.0.4",
|
||||
"pedding": "~1.0.0",
|
||||
"should": "~4.0.4",
|
||||
"should-http": "0.0.1",
|
||||
"supertest": "0.13.0"
|
||||
"supertest": "~0.13.0"
|
||||
},
|
||||
"homepage": "https://github.com/cnpm/cnpmjs.org",
|
||||
"repository": {
|
||||
|
||||
@@ -44,6 +44,20 @@ function* remove(name, usernames) {
|
||||
return yield mysql.query(REMOVE_SQL, [name, usernames]);
|
||||
}
|
||||
|
||||
var REMOVE_ALL_SQL = 'DELETE FROM module_maintainer WHERE name = ?';
|
||||
|
||||
exports.removeAll = function* (name) {
|
||||
return yield mysql.query(REMOVE_ALL_SQL, [name]);
|
||||
};
|
||||
|
||||
exports.addMulti = function* (name, usernames) {
|
||||
var tasks = [];
|
||||
for (var i = 0; i < usernames.length; i++) {
|
||||
tasks.push(add(name, usernames[i]));
|
||||
}
|
||||
return yield tasks;
|
||||
};
|
||||
|
||||
exports.update = function* (name, maintainers) {
|
||||
// maintainers should be [name1, name2, ...] format
|
||||
// find out the exists maintainers then remove the deletes and add the left
|
||||
@@ -64,11 +78,8 @@ exports.update = function* (name, maintainers) {
|
||||
}
|
||||
}
|
||||
}
|
||||
var tasks = [];
|
||||
for (var i = 0; i < addUsers.length; i++) {
|
||||
tasks.push(add(name, addUsers[i]));
|
||||
}
|
||||
yield tasks;
|
||||
|
||||
yield* exports.addMulti(name, addUsers);
|
||||
// make sure all add users success then remove users
|
||||
if (removeUsers.length > 0) {
|
||||
yield* remove(name, removeUsers);
|
||||
|
||||
@@ -76,12 +76,14 @@ SyncModuleWorker.prototype.finish = function () {
|
||||
if (this._finished || Object.keys(this.syncingNames).length > 0) {
|
||||
return;
|
||||
}
|
||||
this._finished = true;
|
||||
this.log('[done] Sync %s module finished, %d success, %d fail\nSuccess: [ %s ]\nFail: [ %s ]',
|
||||
this.startName,
|
||||
this.successes.length, this.fails.length,
|
||||
this.successes.join(', '), this.fails.join(', '));
|
||||
this.emit('end');
|
||||
this._finished = true;
|
||||
// make sure all event listeners release
|
||||
this.removeAllListeners();
|
||||
};
|
||||
|
||||
SyncModuleWorker.prototype.log = function (format, arg1, arg2) {
|
||||
|
||||
@@ -36,6 +36,10 @@ exports.listMaintainerNamesOnly = function* (name) {
|
||||
return yield* ModuleMaintainer.get(name);
|
||||
};
|
||||
|
||||
exports.addMaintainers = function* (name, usernames) {
|
||||
return yield* ModuleMaintainer.addMulti(name, usernames);
|
||||
};
|
||||
|
||||
exports.updateMaintainers = function* (name, usernames) {
|
||||
var rs = yield [
|
||||
ModuleMaintainer.update(name, usernames),
|
||||
@@ -44,6 +48,10 @@ exports.updateMaintainers = function* (name, usernames) {
|
||||
return rs[0];
|
||||
};
|
||||
|
||||
exports.removeAllMaintainers = function* (name) {
|
||||
return yield* ModuleMaintainer.removeAll(name);
|
||||
};
|
||||
|
||||
exports.isMaintainer = function* (name, username) {
|
||||
var rs = yield [
|
||||
ModuleMaintainer.get(name),
|
||||
|
||||
@@ -30,6 +30,7 @@ var controller = require('../../../controllers/registry/module');
|
||||
var ModuleDeps = require('../../../proxy/module_deps');
|
||||
var SyncModuleWorker = require('../../../proxy/sync_module_worker');
|
||||
var utils = require('../../utils');
|
||||
var mysql = require('../../../common/mysql');
|
||||
|
||||
var fixtures = path.join(path.dirname(path.dirname(__dirname)), 'fixtures');
|
||||
|
||||
@@ -532,7 +533,8 @@ describe('controllers/registry/module.test.js', function () {
|
||||
|
||||
describe('PUT /:name publish new flow addPackageAndDist()', function () {
|
||||
it('should publish with tgz base64, addPackageAndDist()', function (done) {
|
||||
var pkg = utils.getPackage('testpublishmodule', '0.0.2');
|
||||
done = pedding(2, done);
|
||||
var pkg = utils.getPackage('testpublishmodule-new-add', '0.0.2');
|
||||
request(app)
|
||||
.put('/' + pkg.name)
|
||||
.set('authorization', utils.adminAuth)
|
||||
@@ -555,6 +557,15 @@ describe('controllers/registry/module.test.js', function () {
|
||||
});
|
||||
done();
|
||||
});
|
||||
|
||||
// maintainers should exists
|
||||
mysql.query('SELECT user FROM module_maintainer WHERE name=?', ['testpublishmodule-new-add'],
|
||||
function (err, rows) {
|
||||
should.not.exist(err);
|
||||
rows.length.should.above(0);
|
||||
rows.should.eql([ { user: 'cnpmjstest10' } ]);
|
||||
done();
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
@@ -922,7 +933,7 @@ describe('controllers/registry/module.test.js', function () {
|
||||
.expect(201, done);
|
||||
});
|
||||
|
||||
it('shold fail when user not maintainer', function (done) {
|
||||
it('should fail when user not maintainer', function (done) {
|
||||
request(app)
|
||||
.del('/remove-all-module/-rev/1')
|
||||
.set('authorization', utils.otherUserAuth)
|
||||
@@ -936,14 +947,19 @@ describe('controllers/registry/module.test.js', function () {
|
||||
});
|
||||
});
|
||||
|
||||
it('shold ok', function (done) {
|
||||
it('should remove all versions ok', function (done) {
|
||||
request(app)
|
||||
.del('/remove-all-module/-rev/1')
|
||||
.set('authorization', utils.adminAuth)
|
||||
.expect(200, function (err, res) {
|
||||
should.not.exist(err);
|
||||
should.not.exist(res.headers['set-cookie']);
|
||||
done();
|
||||
mysql.query('SELECT * FROM module_maintainer WHERE name=?', ['remove-all-module'],
|
||||
function (err, rows) {
|
||||
should.not.exist(err);
|
||||
rows.should.length(0);
|
||||
done();
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
@@ -560,7 +560,7 @@ describe('controllers/registry/module/public_module.test.js', function () {
|
||||
mm(config, 'enablePrivate', false);
|
||||
mm(config, 'forcePublishWithScope', false);
|
||||
});
|
||||
before(function (done) {
|
||||
beforeEach(function (done) {
|
||||
mm(config, 'enablePrivate', false);
|
||||
mm(config, 'forcePublishWithScope', false);
|
||||
var pkg = utils.getPackage('public-test-delete-download-module', '0.1.9', utils.otherUser);
|
||||
@@ -569,9 +569,11 @@ describe('controllers/registry/module/public_module.test.js', function () {
|
||||
.set('content-type', 'application/json')
|
||||
.set('authorization', utils.otherUserAuth)
|
||||
.send(pkg)
|
||||
.expect(201, function (err, res) {
|
||||
.end(function (err, res) {
|
||||
should.not.exist(err);
|
||||
withoutScopeRev = res.body.rev;
|
||||
if (res.body.rev) {
|
||||
withoutScopeRev = res.body.rev;
|
||||
}
|
||||
done();
|
||||
});
|
||||
});
|
||||
@@ -603,7 +605,7 @@ describe('controllers/registry/module/public_module.test.js', function () {
|
||||
mm(config, 'enablePrivate', false);
|
||||
mm(config, 'forcePublishWithScope', true);
|
||||
});
|
||||
before(function (done) {
|
||||
beforeEach(function (done) {
|
||||
mm(config, 'enablePrivate', false);
|
||||
mm(config, 'forcePublishWithScope', true);
|
||||
var pkg = utils.getPackage('@cnpm/public-test-delete-download-module', '0.1.9', utils.otherUser);
|
||||
@@ -612,9 +614,11 @@ describe('controllers/registry/module/public_module.test.js', function () {
|
||||
.set('content-type', 'application/json')
|
||||
.set('authorization', utils.otherUserAuth)
|
||||
.send(pkg)
|
||||
.expect(201, function (err, res) {
|
||||
.end(function (err, res) {
|
||||
should.not.exist(err);
|
||||
withScopeRev = res.body.rev;
|
||||
if (res.body.rev) {
|
||||
withScopeRev = res.body.rev;
|
||||
}
|
||||
done();
|
||||
});
|
||||
});
|
||||
|
||||
@@ -218,7 +218,7 @@ describe('controllers/web/package.test.js', function () {
|
||||
describe('unpublished package', function () {
|
||||
before(function (done) {
|
||||
var worker = new SyncModuleWorker({
|
||||
name: ['browserjs'],
|
||||
name: ['tnpm'],
|
||||
username: 'fengmk2'
|
||||
});
|
||||
|
||||
@@ -226,14 +226,14 @@ describe('controllers/web/package.test.js', function () {
|
||||
worker.on('end', function () {
|
||||
var names = worker.successes.concat(worker.fails);
|
||||
names.sort();
|
||||
names.should.eql(['browserjs']);
|
||||
names.should.eql(['tnpm']);
|
||||
done();
|
||||
});
|
||||
});
|
||||
|
||||
it('should display unpublished info', function (done) {
|
||||
request(app)
|
||||
.get('/package/browserjs')
|
||||
.get('/package/tnpm')
|
||||
.expect(200)
|
||||
.expect(/This package has been unpublished\./, done);
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user