mirror of
https://github.com/notion-enhancer/notion-enhancer.git
synced 2025-04-06 05:29:02 +00:00
137 lines
3.9 KiB
JavaScript
137 lines
3.9 KiB
JavaScript
/*
|
|
* notion-enhancer core: api
|
|
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
|
|
* (https://notion-enhancer.github.io/) under the MIT license
|
|
*/
|
|
|
|
'use strict';
|
|
module.exports = {};
|
|
|
|
/**
|
|
* environment-specific data persistence
|
|
* @module notion-enhancer/api/storage
|
|
*/
|
|
|
|
const _queue = [],
|
|
_onChangeListeners = [];
|
|
|
|
const os = require('os'),
|
|
path = require('path'),
|
|
fs = require('fs'),
|
|
_cacheFile = path.resolve(`${os.homedir()}/.notion-enhancer`);
|
|
|
|
// handle leftover cache from prev versions
|
|
if (fs.existsSync(_cacheFile) && fs.lstatSync(_cacheFile).isDirectory()) {
|
|
fs.rmdirSync(_cacheFile);
|
|
}
|
|
if (!fs.existsSync(_cacheFile)) fs.writeFileSync(_cacheFile, '{}', 'utf8');
|
|
|
|
const getData = () => {
|
|
try {
|
|
return JSON.parse(fs.readFileSync(_cacheFile));
|
|
} catch (err) {
|
|
return {};
|
|
}
|
|
},
|
|
saveData = (data) => fs.writeFileSync(_cacheFile, JSON.stringify(data));
|
|
|
|
/**
|
|
* get persisted data
|
|
* @param {array<string>} path - the path of keys to the value being fetched
|
|
* @param {*} [fallback] - a default value if the path is not matched
|
|
* @returns {Promise} value ?? fallback
|
|
*/
|
|
module.exports.get = (path, fallback = undefined) => {
|
|
if (!path.length) return fallback;
|
|
const values = getData();
|
|
let value = values;
|
|
while (path.length) {
|
|
if (value === undefined) {
|
|
value = fallback;
|
|
break;
|
|
}
|
|
value = value[path.shift()];
|
|
}
|
|
return Promise.resolve(value ?? fallback);
|
|
};
|
|
|
|
/**
|
|
* persist data
|
|
* @param {array<string>} path - the path of keys to the value being set
|
|
* @param {*} value - the data to save
|
|
* @returns {Promise} resolves when data has been saved
|
|
*/
|
|
module.exports.set = (path, value) => {
|
|
if (!path.length) return undefined;
|
|
const precursor = _queue[_queue.length - 1] || undefined,
|
|
interaction = new Promise(async (res, rej) => {
|
|
if (precursor !== undefined) {
|
|
await precursor;
|
|
_queue.shift();
|
|
}
|
|
const pathClone = [...path],
|
|
values = getData();
|
|
let pointer = values,
|
|
old;
|
|
while (path.length) {
|
|
const key = path.shift();
|
|
if (!path.length) {
|
|
old = pointer[key];
|
|
pointer[key] = value;
|
|
break;
|
|
}
|
|
pointer[key] = pointer[key] ?? {};
|
|
pointer = pointer[key];
|
|
}
|
|
saveData(values);
|
|
_onChangeListeners.forEach((listener) => listener({ path: pathClone, new: value, old }));
|
|
res(value);
|
|
});
|
|
_queue.push(interaction);
|
|
return interaction;
|
|
};
|
|
|
|
/**
|
|
* create a wrapper for accessing a partition of the storage
|
|
* @param {array<string>} namespace - the path of keys to prefix all storage requests with
|
|
* @param {function} [get] - the storage get function to be wrapped
|
|
* @param {function} [set] - the storage set function to be wrapped
|
|
* @returns {object} an object with the wrapped get/set functions
|
|
*/
|
|
module.exports.db = (
|
|
namespace,
|
|
getFunc = module.exports.get,
|
|
setFunc = module.exports.set
|
|
) => {
|
|
if (typeof namespace === 'string') namespace = [namespace];
|
|
return {
|
|
get: (path = [], fallback = undefined) => getFunc([...namespace, ...path], fallback),
|
|
set: (path, value) => setFunc([...namespace, ...path], value),
|
|
};
|
|
};
|
|
|
|
/**
|
|
* add an event listener for changes in storage
|
|
* @param {onStorageChangeCallback} callback - called whenever a change in
|
|
* storage is initiated from the current process
|
|
*/
|
|
module.exports.addChangeListener = (callback) => {
|
|
_onChangeListeners.push(callback);
|
|
};
|
|
|
|
/**
|
|
* remove a listener added with storage.addChangeListener
|
|
* @param {onStorageChangeCallback} callback
|
|
*/
|
|
module.exports.removeChangeListener = (callback) => {
|
|
_onChangeListeners = _onChangeListeners.filter((listener) => listener !== callback);
|
|
};
|
|
|
|
/**
|
|
* @callback onStorageChangeCallback
|
|
* @param {object} event
|
|
* @param {string} event.path- the path of keys to the changed value
|
|
* @param {string} [event.new] - the new value being persisted to the store
|
|
* @param {string} [event.old] - the previous value associated with the key
|
|
*/
|