node/deps/npm/lib/adduser.js

139 lines
3.1 KiB
JavaScript

module.exports = adduser
var log = require("npmlog")
, npm = require("./npm.js")
, registry = npm.registry
, read = require("read")
, crypto
try {
crypto = process.binding("crypto") && require("crypto")
} catch (ex) {}
adduser.usage = "npm adduser\nThen enter stuff at the prompts"
function adduser (args, cb) {
if (!crypto) return cb(new Error(
"You must compile node with ssl support to use the adduser feature"))
var c = { u : npm.config.get("username")
, p : npm.config.get("_password")
, e : npm.config.get("email")
}
, changed = false
, u = {}
, fns = [readUsername, readPassword, readEmail, save]
loop()
function loop (er) {
if (er) return cb(er)
var fn = fns.shift()
if (fn) return fn(c, u, loop)
cb()
}
}
function readUsername (c, u, cb) {
read({prompt: "Username: ", default: c.u}, function (er, un) {
if (er) {
return cb(er.message === "cancelled" ? er.message : er)
}
// make sure it's valid. we have to do this here, because
// couchdb will only ever say "bad password" with a 401 when
// you try to PUT a _users record that the validate_doc_update
// rejects for *any* reason.
if (!un) {
return readUsername(c, u, cb)
}
if (un !== un.toLowerCase()) {
log.warn('Username must be lowercase')
return readUsername(c, u, cb)
}
if (un !== encodeURIComponent(un)) {
log.warn('Username may not contain non-url-safe chars')
return readUsername(c, u, cb)
}
if (un.charAt(0) === '.') {
log.warn('Username may not start with "."')
return readUsername(c, u, cb)
}
c.changed = c.u !== un
u.u = un
cb(er)
})
}
function readPassword (c, u, cb) {
if (!c.changed) {
u.p = c.p
return cb()
}
read({prompt: "Password: ", silent: true}, function (er, pw) {
if (er) {
return cb(er.message === "cancelled" ? er.message : er)
}
if (!pw) {
return readPassword(c, u, cb)
}
if (pw.match(/['!:@"]/)) {
log.warn('Sorry, passwords cannot contain these characters: \'!:@"')
return readPassword(c, u, cb)
}
u.p = pw
cb(er)
})
}
function readEmail (c, u, cb) {
read({prompt: "Email: ", default: c.e}, function (er, em) {
if (er) {
return cb(er.message === "cancelled" ? er.message : er)
}
if (!em) {
return readEmail(c, u, cb)
}
if (!em.match(/^.+@.+\..+$/)) {
log.warn('Email must be an email address')
return readEmail(c, u, cb)
}
u.e = em
cb(er)
})
}
function save (c, u, cb) {
if (c.changed) {
delete registry.auth
delete registry.username
delete registry.password
registry.username = u.u
registry.password = u.p
}
// save existing configs, but yank off for this PUT
registry.adduser(u.u, u.p, u.e, function (er) {
if (er) return cb(er)
registry.username = u.u
registry.password = u.p
registry.email = u.e
npm.config.set("username", u.u, "user")
npm.config.set("_password", u.p, "user")
npm.config.set("email", u.e, "user")
log.info("adduser", "Authorized user %s", u.u)
npm.config.save("user", cb)
})
}