Files
cnpmjs.org/controllers/registry/module.js

288 lines
7.4 KiB
JavaScript

/*!
* cnpmjs.org - controllers/registry/module.js
*
* Copyright(c) cnpmjs.org and other contributors.
* MIT Licensed
*
* Authors:
* dead_horse <dead_horse@qq.com>
* fengmk2 <fengmk2@gmail.com> (http://fengmk2.github.com)
*/
'use strict';
/**
* Module dependencies.
*/
var debug = require('debug')('cnpmjs.org:controllers:registry:module');
var path = require('path');
var fs = require('fs');
var crypto = require('crypto');
var eventproxy = require('eventproxy');
var config = require('../../config');
var Module = require('../../proxy/module');
exports.show = function (req, res, next) {
var name = req.params.name;
Module.listByName(name, function (err, rows) {
if (err || rows.length === 0) {
return next(err);
}
var nextMod = rows[0];
var latest = rows[1];
var startIndex = 1;
if (nextMod.version !== 'next') {
// next create fail
latest = nextMod;
startIndex = 0;
nextMod = null;
}
if (!latest) {
latest = nextMod;
}
var distTags = {};
var versions = {};
var times = {};
var attachments = {};
for (var i = startIndex; i < rows.length; i++) {
var row = rows[i];
var pkg = row.package;
versions[pkg.version] = pkg;
times[pkg.version] = row.gmt_modified;
}
if (latest.package.version && latest.package.version !== 'next') {
distTags.latest = latest.package.version;
}
var rev = '';
if (nextMod) {
rev = String(nextMod.id);
}
var info = {
_id: latest.name,
_rev: rev,
name: latest.name,
description: latest.package.description,
versions: versions,
"dist-tags": distTags,
readme: latest.package.readme,
maintainers: latest.package.maintainers,
time: times,
author: latest.package.author,
repository: latest.package.repository,
_attachments: attachments
};
res.json(info);
});
};
exports.upload = function (req, res, next) {
var length = Number(req.headers['content-length']) || 0;
if (!length || req.headers['content-type'] !== 'application/octet-stream') {
return next();
}
var username = req.session.name;
var name = req.params.name;
var id = Number(req.params.rev);
var filename = req.params.filename;
var version = filename.substring(filename.indexOf('-') + 1);
version = version.replace(/\.tgz$/, '');
// save version on pkg upload
debug('%s: upload %s, file size: %d', username, req.url, length);
Module.getById(id, function (err, mod) {
if (err || !mod) {
return next(err);
}
var match = mod.package.maintainers.filter(function (item) {
return item.name === username;
});
if (match.length === 0 || mod.name !== name) {
return res.json(403, {
error: 'no_perms',
reason: 'Current user can not publish this module'
});
}
if (mod.version !== 'next') {
// rev wrong
return res.json(403, {
error: 'rev_wrong',
reason: 'rev not match next module'
});
}
var filepath = path.join(config.uploadDir, filename);
var ws = fs.createWriteStream(filepath);
var shasum = crypto.createHash('sha1');
req.pipe(ws);
var dataSize = 0;
req.on('data', function (data) {
shasum.update(data);
dataSize += data.length;
});
ws.on('finish', function () {
if (dataSize !== length) {
return res.json(403, {
error: 'size_wrong',
reason: 'Header size ' + length + ' not match download size ' + dataSize,
});
}
shasum = shasum.digest('hex');
var dist = {
tarball: 'http://' + req.headers.host + '/dist/' + filename,
shasum: shasum,
size: length
};
mod.package.dist = dist;
mod.package.version = version;
debug('%s module: save file to %s, size: %d, sha1: %s, dist: %j, version: %s',
id, filepath, length, shasum, dist, version);
Module.update(mod, function (err, result) {
if (err) {
return next(err);
}
res.json(201, {ok: true, rev: String(result.id)});
});
});
});
};
exports.updateLatest = function (req, res, next) {
var username = req.session.name;
var name = req.params.name;
var version = req.params.version;
Module.get(name, 'next', function (err, nextMod) {
if (err) {
return next(err);
}
if (!nextMod) {
return next();
}
var match = nextMod.package.maintainers.filter(function (item) {
return item.name === username;
});
if (match.length === 0) {
return res.json(401, {
error: 'noperms',
reason: 'Current user can not publish this module'
});
}
// check version if not match pkg upload
if (nextMod.package.version !== version) {
return res.json(403, {
error: 'version_wrong',
reason: 'version not match'
});
}
var body = req.body;
nextMod.version = version;
nextMod.author = username;
body.dist = nextMod.package.dist;
body.maintainers = nextMod.package.maintainers;
if (!body.author) {
body.author = {
name: username,
email: req.session.email,
};
}
nextMod.package = body;
debug('update %s:%s %j', nextMod.package.name, nextMod.package.version, nextMod.package.dist);
// change latest to version
Module.update(nextMod, function (err) {
if (err) {
return next(err);
}
// add a new latest version
nextMod.version = 'next';
Module.add(nextMod, function (err, result) {
if (err) {
return next(err);
}
res.json(201, {ok: true, rev: String(result.id)});
});
});
});
};
exports.add = function (req, res, next) {
var username = req.session.name;
var name = req.params.name;
var pkg = req.body;
var maintainers = pkg.maintainers || [];
var match = maintainers.filter(function (item) {
return item.name === username;
});
if (match.length === 0) {
return res.json(403, {
error: 'no_perms',
reason: 'Current user can not publish this module'
});
}
var ep = eventproxy.create();
ep.fail(next);
Module.getLatest(name, ep.doneLater('latest'));
Module.get(name, 'next', ep.done(function (nextMod) {
if (nextMod) {
nextMod.exists = true;
return ep.emit('next', nextMod);
}
// ensure next module exits
// because updateLatest will create next module fail
nextMod = {
name: name,
version: 'next',
author: username,
package: {
name: name,
version: 'next',
description: pkg.description,
readme: pkg.readme,
maintainers: pkg.maintainers,
},
};
Module.add(nextMod, ep.done(function (result) {
nextMod.id = result.id;
ep.emit('next', nextMod);
}));
}));
ep.all('latest', 'next', function (latestMod, nextMod) {
var maintainers = latestMod ? latestMod.package.maintainers : nextMod.package.maintainers;
var match = maintainers.filter(function (item) {
return item.name === username;
});
if (match.length === 0) {
return res.json(403, {
error: 'no_perms',
reason: 'Current user can not publish this module'
});
}
if (latestMod || nextMod.exists) {
return res.json(409, {
error: 'conflict',
reason: 'Document update conflict.'
});
}
res.json(201, {
ok: true,
id: name,
rev: String(nextMod.id),
});
});
};