mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-11 18:24:43 +02:00
115 lines
3.1 KiB
TypeScript
115 lines
3.1 KiB
TypeScript
import BaseService from './BaseService';
|
|
const Mutex = require('async-mutex').Mutex;
|
|
|
|
enum ValueType {
|
|
Int = 1,
|
|
Text = 2,
|
|
}
|
|
|
|
export default class KvStore extends BaseService {
|
|
|
|
private incMutex_: any = null;
|
|
private db_: any = null;
|
|
|
|
private static instance_: KvStore = null;
|
|
|
|
public static instance() {
|
|
if (this.instance_) return this.instance_;
|
|
this.instance_ = new KvStore();
|
|
return this.instance_;
|
|
}
|
|
|
|
public static destroyInstance() {
|
|
this.instance_ = null;
|
|
}
|
|
|
|
private constructor() {
|
|
super();
|
|
this.incMutex_ = new Mutex();
|
|
}
|
|
|
|
public setDb(v: any) {
|
|
this.db_ = v;
|
|
}
|
|
|
|
private db() {
|
|
if (!this.db_) throw new Error('Accessing DB before it has been set!');
|
|
return this.db_;
|
|
}
|
|
|
|
private typeFromValue_(value: any) {
|
|
if (typeof value === 'string') return ValueType.Text;
|
|
if (typeof value === 'number') return ValueType.Int;
|
|
throw new Error(`Unsupported value type: ${typeof value}`);
|
|
}
|
|
|
|
private formatValues_(kvs: any[]) {
|
|
const output = [];
|
|
for (const kv of kvs) {
|
|
kv.value = this.formatValue_(kv.value, kv.type);
|
|
output.push(kv);
|
|
}
|
|
return output;
|
|
}
|
|
|
|
private formatValue_(value: any, type: ValueType): string | number {
|
|
if (type === ValueType.Int) return Number(value);
|
|
if (type === ValueType.Text) return `${value}`;
|
|
throw new Error(`Unknown type: ${type}`);
|
|
}
|
|
|
|
public async value<T>(key: string): Promise<T> {
|
|
const r = await this.db().selectOne('SELECT `value`, `type` FROM key_values WHERE `key` = ?', [key]);
|
|
if (!r) return null;
|
|
return this.formatValue_(r.value, r.type) as any;
|
|
}
|
|
|
|
public async setValue(key: string, value: any) {
|
|
const t = Date.now();
|
|
await this.db().exec('INSERT OR REPLACE INTO key_values (`key`, `value`, `type`, `updated_time`) VALUES (?, ?, ?, ?)', [key, value, this.typeFromValue_(value), t]);
|
|
}
|
|
|
|
public async deleteValue(key: string) {
|
|
await this.db().exec('DELETE FROM key_values WHERE `key` = ?', [key]);
|
|
}
|
|
|
|
public async deleteByPrefix(prefix: string) {
|
|
await this.db().exec('DELETE FROM key_values WHERE `key` LIKE ?', [`${prefix}%`]);
|
|
}
|
|
|
|
public async clear() {
|
|
await this.db().exec('DELETE FROM key_values');
|
|
}
|
|
|
|
public async all() {
|
|
return this.formatValues_(await this.db().selectAll('SELECT * FROM key_values'));
|
|
}
|
|
|
|
// Note: atomicity is done at application level so two difference instances
|
|
// accessing the db at the same time could mess up the increment.
|
|
public async incValue(key: string, inc: number = 1) {
|
|
const release = await this.incMutex_.acquire();
|
|
|
|
try {
|
|
const result = await this.db().selectOne('SELECT `value`, `type` FROM key_values WHERE `key` = ?', [key]);
|
|
const newValue = result ? (this.formatValue_(result.value, result.type) as ValueType.Int) + inc : inc;
|
|
await this.setValue(key, newValue);
|
|
release();
|
|
return newValue;
|
|
} catch (error) {
|
|
release();
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
public async searchByPrefix(prefix: string) {
|
|
const results = await this.db().selectAll('SELECT `key`, `value`, `type` FROM key_values WHERE `key` LIKE ?', [`${prefix}%`]);
|
|
return this.formatValues_(results);
|
|
}
|
|
|
|
public async countKeys() {
|
|
const r = await this.db().selectOne('SELECT count(*) as total FROM key_values');
|
|
return r.total ? r.total : 0;
|
|
}
|
|
}
|