2014-06-06 06:18:15 +08:00
|
|
|
var crypto = require("crypto")
|
2014-11-05 07:08:12 +08:00
|
|
|
var resolve = require("path").resolve
|
2014-06-06 06:18:15 +08:00
|
|
|
|
2014-11-05 07:08:12 +08:00
|
|
|
var lockfile = require("lockfile")
|
2014-06-06 06:18:15 +08:00
|
|
|
var log = require("npmlog")
|
2014-11-05 07:08:12 +08:00
|
|
|
var mkdirp = require("mkdirp")
|
|
|
|
|
|
|
|
var npm = require("../npm.js")
|
|
|
|
var getStat = require("../cache/get-stat.js")
|
|
|
|
|
|
|
|
var installLocks = {}
|
|
|
|
|
|
|
|
function lockFileName (base, name) {
|
|
|
|
var c = name.replace(/[^a-zA-Z0-9]+/g, "-").replace(/^-+|-+$/g, "")
|
|
|
|
, p = resolve(base, name)
|
|
|
|
, h = crypto.createHash("sha1").update(p).digest("hex")
|
|
|
|
, l = resolve(npm.cache, "_locks")
|
|
|
|
|
|
|
|
return resolve(l, c.substr(0, 24)+"-"+h.substr(0, 16)+".lock")
|
2014-06-06 06:18:15 +08:00
|
|
|
}
|
|
|
|
|
2014-11-05 07:08:12 +08:00
|
|
|
function lock (base, name, cb) {
|
|
|
|
getStat(function (er) {
|
|
|
|
var lockDir = resolve(npm.cache, "_locks")
|
|
|
|
mkdirp(lockDir, function () {
|
|
|
|
if (er) return cb(er)
|
|
|
|
|
|
|
|
var opts = { stale: npm.config.get("cache-lock-stale")
|
|
|
|
, retries: npm.config.get("cache-lock-retries")
|
|
|
|
, wait: npm.config.get("cache-lock-wait") }
|
|
|
|
var lf = lockFileName(base, name)
|
|
|
|
lockfile.lock(lf, opts, function (er) {
|
|
|
|
if (er) log.warn("locking", lf, "failed", er)
|
|
|
|
|
|
|
|
if (!er) {
|
|
|
|
log.verbose("lock", "using", lf, "for", resolve(base, name))
|
|
|
|
installLocks[lf] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
cb(er)
|
|
|
|
})
|
2014-06-06 06:18:15 +08:00
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2014-11-05 07:08:12 +08:00
|
|
|
function unlock (base, name, cb) {
|
|
|
|
var lf = lockFileName(base, name)
|
|
|
|
, locked = installLocks[lf]
|
2014-06-06 06:18:15 +08:00
|
|
|
if (locked === false) {
|
|
|
|
return process.nextTick(cb)
|
2014-11-05 07:08:12 +08:00
|
|
|
}
|
|
|
|
else if (locked === true) {
|
|
|
|
lockfile.unlock(lf, function (er) {
|
|
|
|
if (er) {
|
|
|
|
log.warn("unlocking", lf, "failed", er)
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
installLocks[lf] = false
|
|
|
|
log.verbose("unlock", "done using", lf, "for", resolve(base, name))
|
|
|
|
}
|
|
|
|
|
|
|
|
cb(er)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
throw new Error(
|
|
|
|
"Attempt to unlock " + resolve(base, name) + ", which hasn't been locked"
|
|
|
|
)
|
2014-06-06 06:18:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = {
|
2014-11-05 07:08:12 +08:00
|
|
|
lock : lock,
|
|
|
|
unlock : unlock
|
2014-06-06 06:18:15 +08:00
|
|
|
}
|