reduce multi-process file op interference

This commit is contained in:
dragonwocky 2021-12-14 11:44:49 +11:00
parent 50a77d05a1
commit e900bdb681
Signed by: dragonwocky
GPG Key ID: 86DFC3C312A56010

View File

@ -20,26 +20,49 @@ if (fs.existsSync(_cacheFile) && fs.lstatSync(_cacheFile).isDirectory()) {
const isRenderer = process && process.type === 'renderer'; const isRenderer = process && process.type === 'renderer';
const getCache = async () => { // things are a little weird here:
try { // multiple processes performing file ops at once
return fs.readFileSync(_cacheFile); // (e.g. when too many windows/tabs are open)
} catch (err) { // = an empty string is returned the cache contents
await new Promise((res, rej) => setTimeout(res, 50)); // and the db is reset. this loop roughly addresses that.
return getCache();
} // a "real" db might be better, but sql or query-based
}, // would be incompatible with the chrome ext.
getData = async () => { // -- lowdb might have been a nice flat/json db,
// but unfortunately it is esm only
const getData = async () => {
if (!fs.existsSync(_cacheFile)) { if (!fs.existsSync(_cacheFile)) {
fs.writeFileSync(_cacheFile, '{}', 'utf8'); fs.writeFileSync(_cacheFile, '{}', 'utf8');
return {}; return {};
} }
try {
return JSON.parse(await getCache()); let cacheBuffer = '',
} catch (err) { jsonData = {},
return {}; attemptsRemaining = 3;
while (attemptsRemaining) {
cacheBuffer = fs.readFileSync(_cacheFile);
if (cacheBuffer) {
try {
jsonData = JSON.parse(cacheBuffer);
break;
} catch {
jsonData = {};
}
}
--attemptsRemaining || (await new Promise((res, rej) => setTimeout(res, 50)));
} }
return jsonData;
}, },
saveData = (data) => fs.writeFileSync(_cacheFile, JSON.stringify(data)); saveData = (data) => fs.writeFileSync(_cacheFile, JSON.stringify(data)),
performFsOperation = async (callback) => {
while (_fsQueue.size) await new Promise(requestIdleCallback);
const op = Symbol();
_fsQueue.add(op);
const res = await callback();
_fsQueue.delete(op);
return res;
};
const db = { const db = {
get: async (path, fallback = undefined) => { get: async (path, fallback = undefined) => {
@ -58,27 +81,27 @@ const db = {
}, },
set: async (path, value) => { set: async (path, value) => {
if (!path.length) return undefined; if (!path.length) return undefined;
while (_fsQueue.size) await new Promise(requestIdleCallback); return await performFsOperation(async () => {
const op = Symbol(); const pathClone = [...path],
_fsQueue.add(op); values = await getData();
const pathClone = [...path], let pointer = values,
values = await getData(); old;
let pointer = values, while (path.length) {
old; const key = path.shift();
while (path.length) { if (!path.length) {
const key = path.shift(); old = pointer[key];
if (!path.length) { pointer[key] = value;
old = pointer[key]; break;
pointer[key] = value; }
break; pointer[key] = pointer[key] ?? {};
pointer = pointer[key];
} }
pointer[key] = pointer[key] ?? {}; saveData(values);
pointer = pointer[key]; _onDbChangeListeners.forEach((listener) =>
} listener({ path: pathClone, new: value, old })
saveData(values); );
_onDbChangeListeners.forEach((listener) => listener({ path: pathClone, new: value, old })); return value;
_fsQueue.delete(op); });
return value;
}, },
addChangeListener: (callback) => { addChangeListener: (callback) => {
_onDbChangeListeners.push(callback); _onDbChangeListeners.push(callback);