|
|
- const cache = new Map()
- const fs = require('fs')
- const { dirname, resolve } = require('path')
-
-
- const lstat = path => new Promise((res, rej) =>
- fs.lstat(path, (er, st) => er ? rej(er) : res(st)))
-
- const inferOwner = path => {
- path = resolve(path)
- if (cache.has(path))
- return Promise.resolve(cache.get(path))
-
- const statThen = st => {
- const { uid, gid } = st
- cache.set(path, { uid, gid })
- return { uid, gid }
- }
- const parent = dirname(path)
- const parentTrap = parent === path ? null : er => {
- return inferOwner(parent).then((owner) => {
- cache.set(path, owner)
- return owner
- })
- }
- return lstat(path).then(statThen, parentTrap)
- }
-
- const inferOwnerSync = path => {
- path = resolve(path)
- if (cache.has(path))
- return cache.get(path)
-
- const parent = dirname(path)
-
- // avoid obscuring call site by re-throwing
- // "catch" the error by returning from a finally,
- // only if we're not at the root, and the parent call works.
- let threw = true
- try {
- const st = fs.lstatSync(path)
- threw = false
- const { uid, gid } = st
- cache.set(path, { uid, gid })
- return { uid, gid }
- } finally {
- if (threw && parent !== path) {
- const owner = inferOwnerSync(parent)
- cache.set(path, owner)
- return owner // eslint-disable-line no-unsafe-finally
- }
- }
- }
-
- const inflight = new Map()
- module.exports = path => {
- path = resolve(path)
- if (inflight.has(path))
- return Promise.resolve(inflight.get(path))
- const p = inferOwner(path).then(owner => {
- inflight.delete(path)
- return owner
- })
- inflight.set(path, p)
- return p
- }
- module.exports.sync = inferOwnerSync
- module.exports.clearCache = () => {
- cache.clear()
- inflight.clear()
- }
|