2023-07-27 16:05:56 +01:00
|
|
|
import Logger from '@joplin/utils/Logger';
|
2021-02-09 17:54:29 +00:00
|
|
|
import shim from '../../shim';
|
|
|
|
import Setting from '../Setting';
|
|
|
|
|
|
|
|
const logger = Logger.create('Settings');
|
|
|
|
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2021-02-09 17:54:29 +00:00
|
|
|
export type SettingValues = Record<string, any>;
|
|
|
|
|
|
|
|
export default class FileHandler {
|
|
|
|
|
|
|
|
private filePath_: string;
|
|
|
|
private valueJsonCache_: string = null;
|
2024-05-28 05:42:52 -07:00
|
|
|
private parsedJsonCache_: SettingValues = null;
|
2021-02-09 17:54:29 +00:00
|
|
|
|
|
|
|
public constructor(filePath: string) {
|
|
|
|
this.filePath_ = filePath;
|
|
|
|
}
|
|
|
|
|
|
|
|
public async load(): Promise<SettingValues> {
|
2021-02-09 20:58:17 +00:00
|
|
|
if (!this.valueJsonCache_) {
|
|
|
|
if (!(await shim.fsDriver().exists(this.filePath_))) {
|
|
|
|
this.valueJsonCache_ = '{}';
|
|
|
|
} else {
|
|
|
|
this.valueJsonCache_ = await shim.fsDriver().readFile(this.filePath_, 'utf8');
|
|
|
|
}
|
2024-05-28 05:42:52 -07:00
|
|
|
this.parsedJsonCache_ = null;
|
2021-02-09 17:54:29 +00:00
|
|
|
}
|
|
|
|
|
2024-05-28 05:42:52 -07:00
|
|
|
if (this.parsedJsonCache_) return this.parsedJsonCache_;
|
|
|
|
|
|
|
|
let result: SettingValues;
|
2021-02-09 17:54:29 +00:00
|
|
|
try {
|
|
|
|
const values = JSON.parse(this.valueJsonCache_);
|
|
|
|
delete values['$id'];
|
|
|
|
delete values['$schema'];
|
2024-05-28 05:42:52 -07:00
|
|
|
result = values;
|
2021-02-09 17:54:29 +00:00
|
|
|
} catch (error) {
|
|
|
|
// Most likely the user entered invalid JSON - in this case we move
|
|
|
|
// the broken file to a new name (otherwise it would be overwritten
|
|
|
|
// by valid JSON and user will lose all their settings).
|
|
|
|
logger.error(`Could not parse JSON file: ${this.filePath_}`, error);
|
|
|
|
await shim.fsDriver().move(this.filePath_, `${this.filePath_}-${Date.now()}-invalid.bak`);
|
2024-05-28 05:42:52 -07:00
|
|
|
result = {};
|
2021-02-09 17:54:29 +00:00
|
|
|
}
|
2024-05-28 05:42:52 -07:00
|
|
|
|
|
|
|
this.parsedJsonCache_ = result;
|
|
|
|
|
|
|
|
return result;
|
2021-02-09 17:54:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public async save(values: SettingValues) {
|
2024-05-28 05:42:52 -07:00
|
|
|
values = { ...values };
|
|
|
|
|
|
|
|
// Merge with existing settings. This prevents settings stored by disabled or not-yet-loaded
|
|
|
|
// plugins from being deleted.
|
|
|
|
for (const key in this.parsedJsonCache_) {
|
|
|
|
const includesSetting = Object.prototype.hasOwnProperty.call(values, key);
|
|
|
|
if (!includesSetting) {
|
|
|
|
values[key] = this.parsedJsonCache_[key];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-09 17:54:29 +00:00
|
|
|
const json = `${JSON.stringify({
|
|
|
|
'$schema': Setting.schemaUrl,
|
|
|
|
...values,
|
|
|
|
}, null, '\t')}\n`;
|
|
|
|
|
|
|
|
if (json === this.valueJsonCache_) return;
|
|
|
|
|
|
|
|
await shim.fsDriver().writeFile(this.filePath_, json, 'utf8');
|
|
|
|
this.valueJsonCache_ = json;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|