mirror of
https://github.com/notion-enhancer/notion-enhancer.git
synced 2025-04-05 13:19:03 +00:00
reduce multi-process file op interference
This commit is contained in:
parent
50a77d05a1
commit
e900bdb681
@ -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);
|
||||||
|
Loading…
Reference in New Issue
Block a user