You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
157 lines
4.3 KiB
157 lines
4.3 KiB
2 months ago
|
// A path exclusive reservation system
|
||
|
// reserve([list, of, paths], fn)
|
||
|
// When the fn is first in line for all its paths, it
|
||
|
// is called with a cb that clears the reservation.
|
||
|
//
|
||
|
// Used by async unpack to avoid clobbering paths in use,
|
||
|
// while still allowing maximal safe parallelization.
|
||
|
|
||
|
const assert = require('assert')
|
||
|
const normalize = require('./normalize-unicode.js')
|
||
|
const stripSlashes = require('./strip-trailing-slashes.js')
|
||
|
const { join } = require('path')
|
||
|
|
||
|
const platform = process.env.TESTING_TAR_FAKE_PLATFORM || process.platform
|
||
|
const isWindows = platform === 'win32'
|
||
|
|
||
|
module.exports = () => {
|
||
|
// path => [function or Set]
|
||
|
// A Set object means a directory reservation
|
||
|
// A fn is a direct reservation on that path
|
||
|
const queues = new Map()
|
||
|
|
||
|
// fn => {paths:[path,...], dirs:[path, ...]}
|
||
|
const reservations = new Map()
|
||
|
|
||
|
// return a set of parent dirs for a given path
|
||
|
// '/a/b/c/d' -> ['/', '/a', '/a/b', '/a/b/c', '/a/b/c/d']
|
||
|
const getDirs = path => {
|
||
|
const dirs = path.split('/').slice(0, -1).reduce((set, path) => {
|
||
|
if (set.length) {
|
||
|
path = join(set[set.length - 1], path)
|
||
|
}
|
||
|
set.push(path || '/')
|
||
|
return set
|
||
|
}, [])
|
||
|
return dirs
|
||
|
}
|
||
|
|
||
|
// functions currently running
|
||
|
const running = new Set()
|
||
|
|
||
|
// return the queues for each path the function cares about
|
||
|
// fn => {paths, dirs}
|
||
|
const getQueues = fn => {
|
||
|
const res = reservations.get(fn)
|
||
|
/* istanbul ignore if - unpossible */
|
||
|
if (!res) {
|
||
|
throw new Error('function does not have any path reservations')
|
||
|
}
|
||
|
return {
|
||
|
paths: res.paths.map(path => queues.get(path)),
|
||
|
dirs: [...res.dirs].map(path => queues.get(path)),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// check if fn is first in line for all its paths, and is
|
||
|
// included in the first set for all its dir queues
|
||
|
const check = fn => {
|
||
|
const { paths, dirs } = getQueues(fn)
|
||
|
return paths.every(q => q[0] === fn) &&
|
||
|
dirs.every(q => q[0] instanceof Set && q[0].has(fn))
|
||
|
}
|
||
|
|
||
|
// run the function if it's first in line and not already running
|
||
|
const run = fn => {
|
||
|
if (running.has(fn) || !check(fn)) {
|
||
|
return false
|
||
|
}
|
||
|
running.add(fn)
|
||
|
fn(() => clear(fn))
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
const clear = fn => {
|
||
|
if (!running.has(fn)) {
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
const { paths, dirs } = reservations.get(fn)
|
||
|
const next = new Set()
|
||
|
|
||
|
paths.forEach(path => {
|
||
|
const q = queues.get(path)
|
||
|
assert.equal(q[0], fn)
|
||
|
if (q.length === 1) {
|
||
|
queues.delete(path)
|
||
|
} else {
|
||
|
q.shift()
|
||
|
if (typeof q[0] === 'function') {
|
||
|
next.add(q[0])
|
||
|
} else {
|
||
|
q[0].forEach(fn => next.add(fn))
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
|
||
|
dirs.forEach(dir => {
|
||
|
const q = queues.get(dir)
|
||
|
assert(q[0] instanceof Set)
|
||
|
if (q[0].size === 1 && q.length === 1) {
|
||
|
queues.delete(dir)
|
||
|
} else if (q[0].size === 1) {
|
||
|
q.shift()
|
||
|
|
||
|
// must be a function or else the Set would've been reused
|
||
|
next.add(q[0])
|
||
|
} else {
|
||
|
q[0].delete(fn)
|
||
|
}
|
||
|
})
|
||
|
running.delete(fn)
|
||
|
|
||
|
next.forEach(fn => run(fn))
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
const reserve = (paths, fn) => {
|
||
|
// collide on matches across case and unicode normalization
|
||
|
// On windows, thanks to the magic of 8.3 shortnames, it is fundamentally
|
||
|
// impossible to determine whether two paths refer to the same thing on
|
||
|
// disk, without asking the kernel for a shortname.
|
||
|
// So, we just pretend that every path matches every other path here,
|
||
|
// effectively removing all parallelization on windows.
|
||
|
paths = isWindows ? ['win32 parallelization disabled'] : paths.map(p => {
|
||
|
// don't need normPath, because we skip this entirely for windows
|
||
|
return stripSlashes(join(normalize(p))).toLowerCase()
|
||
|
})
|
||
|
|
||
|
const dirs = new Set(
|
||
|
paths.map(path => getDirs(path)).reduce((a, b) => a.concat(b))
|
||
|
)
|
||
|
reservations.set(fn, { dirs, paths })
|
||
|
paths.forEach(path => {
|
||
|
const q = queues.get(path)
|
||
|
if (!q) {
|
||
|
queues.set(path, [fn])
|
||
|
} else {
|
||
|
q.push(fn)
|
||
|
}
|
||
|
})
|
||
|
dirs.forEach(dir => {
|
||
|
const q = queues.get(dir)
|
||
|
if (!q) {
|
||
|
queues.set(dir, [new Set([fn])])
|
||
|
} else if (q[q.length - 1] instanceof Set) {
|
||
|
q[q.length - 1].add(fn)
|
||
|
} else {
|
||
|
q.push(new Set([fn]))
|
||
|
}
|
||
|
})
|
||
|
|
||
|
return run(fn)
|
||
|
}
|
||
|
|
||
|
return { check, reserve }
|
||
|
}
|