feature: emit events

This commit is contained in:
Andy Burke 2025-07-01 19:14:18 -07:00
parent 0d0f399cc2
commit 3214d17b80
7 changed files with 460 additions and 18 deletions

View file

@ -11,6 +11,8 @@ optimization to the filesystem layer.
`collection.delete(T)` - removes an object from the system and the disk
`collection.all([options])` - iterate over all objects
`collection.find(criteria[, options])` - find all objects that match the criteria *that have an index*
`collection.on(event, callback)` - set a callback for the given event (create,update,get,delete,write,index,all,find)
`collection.off(event, callback)` - remove a callback for the given event
### Example
@ -157,6 +159,27 @@ for browsing the data as a human.
TODO: index everything into a sqlite setup as well? would give a way to run
SQL against data still stored on disk in a nicely human browsable format.
## Environment Variables
| variable | description |
| --------------------------- | ----------------------------------------------- |
| FSDB_ROOT | controls the root directory, default: ./.fsdb |
| FSDB_PERF | set to true for performance tracking |
| FSDB_LOG_EVENTS | set to true to log the events system |
## TODO
- [ ] make all()/find() return something like { file_info, entry: { private data = undefined; load() => { data = data ?? await Deno.readTextFile(this.file_info.path); return data; } } }
- [ ] make all()/find() return something like
```
{
file_info,
entry: {
private data = undefined;
load() => {
data = data ?? await Deno.readTextFile(this.file_info.path);
return data;
}
}
}
```

8
cli.ts
View file

@ -91,7 +91,7 @@ switch (command) {
Deno.exit(1);
}
console.log(JSON.stringify(item, null, 4));
console.log(JSON.stringify(item, null, '\t'));
break;
}
case 'create': {
@ -109,7 +109,7 @@ switch (command) {
const item: any = JSON.parse(item_json);
const created_item: any = await collection.create(item);
console.log('created: ' + JSON.stringify(created_item, null, 4));
console.log('created: ' + JSON.stringify(created_item, null, '\t'));
break;
}
case 'update': {
@ -127,7 +127,7 @@ switch (command) {
const item: any = JSON.parse(item_json);
const updated_item: any = await collection.update(item);
console.log('updated: ' + JSON.stringify(updated_item, null, 4));
console.log('updated: ' + JSON.stringify(updated_item, null, '\t'));
break;
}
case 'delete': {
@ -145,7 +145,7 @@ switch (command) {
const item: any = JSON.parse(item_json);
const deleted_item: any = await collection.delete(item);
console.log('deleted: ' + JSON.stringify(deleted_item, null, 4));
console.log('deleted: ' + JSON.stringify(deleted_item, null, '\t'));
break;
}
case 'find':

View file

@ -1,6 +1,6 @@
{
"name": "@andyburke/fsdb",
"version": "0.6.1",
"version": "0.7.0",
"license": "MIT",
"exports": {
".": "./fsdb.ts",
@ -12,7 +12,7 @@
"tasks": {
"lint": "deno lint",
"fmt": "deno fmt",
"test": "cd tests && DENO_ENV=test TEST_DATA_STORAGE_ROOT=./data/$(date --iso-8601=seconds) deno test --allow-env --allow-read --allow-write --fail-fast --trace-leaks ./",
"test": "cd tests && DENO_ENV=test FSDB_TEST_DATA_STORAGE_ROOT=./data/$(date --iso-8601=seconds) deno test --allow-env --allow-read --allow-write --fail-fast --trace-leaks ./",
"fsdb": "deno run --allow-env --allow-read --allow-write cli.ts"
},

116
fsdb.ts
View file

@ -34,6 +34,7 @@ export interface FSDB_INDEXER<T> {
export class FSDB_COLLECTION<T extends Record<string, any>> {
private config: FSDB_COLLECTION_CONFIG;
public INDEX: Record<string, FSDB_INDEXER<any>>;
private event_listeners: Record<string, []>;
constructor(input_config: FSDB_COLLECTION_CONFIG_INPUT) {
this.config = {
@ -45,6 +46,8 @@ export class FSDB_COLLECTION<T extends Record<string, any>> {
...(input_config ?? {})
};
this.event_listeners = {};
this.INDEX = this.config.indexers ?? {};
for (const indexer of Object.values(this.INDEX)) {
indexer.set_fsdb_root(this.config.root);
@ -84,7 +87,7 @@ export class FSDB_COLLECTION<T extends Record<string, any>> {
}
const collection_info_file_path: string = path.resolve(path.join(this.config.root, '.fsdb.collection.json'));
const collection_info_json: string = JSON.stringify(this.config, null, 4);
const collection_info_json: string = JSON.stringify(this.config, null, '\t');
Deno.mkdirSync(path.dirname(collection_info_file_path), {
recursive: true
});
@ -114,26 +117,43 @@ export class FSDB_COLLECTION<T extends Record<string, any>> {
private async write_item(item: T, override_path?: string): Promise<void> {
const item_path: string = override_path ?? await this.ensure_item_path(item, this.config.id_field);
Deno.writeTextFileSync(item_path, JSON.stringify(item, null, 1));
Deno.writeTextFileSync(item_path, JSON.stringify(item, null, '\t'));
this.emit('write', {
item,
item_path
});
if (this.config.indexers) {
for (const indexer of Object.values(this.config.indexers)) {
await (indexer as FSDB_INDEXER<T>).index(item, item_path);
this.emit('index', {
item,
item_path,
indexer
});
}
}
}
/** Get an item from the collection given its id. */
async get(id: string): Promise<T | null> {
const id_path: string = this.get_organized_id_path(id);
const item_exists: boolean = await fs.exists(id_path);
const item_path: string = this.get_organized_id_path(id);
const item_exists: boolean = await fs.exists(item_path);
if (!item_exists) {
return null;
}
const content: string = await Deno.readTextFile(id_path);
return JSON.parse(content);
const content: string = await Deno.readTextFile(item_path);
const item: T = JSON.parse(content);
this.emit('get', {
item,
item_path
});
return item;
}
/** Create an item in the collection. */
@ -149,15 +169,21 @@ export class FSDB_COLLECTION<T extends Record<string, any>> {
await this.write_item(item);
this.emit('create', {
item,
item_path
});
return item;
}
/** Update the given item in the collection, requiring the id to be stable. */
async update(item: T): Promise<T> {
const item_path: string = this.get_organized_item_path(item);
const item_exists: boolean = await fs.exists(item_path);
const id: string = item[this.config.id_field];
const previous: T | null = await this.get(id);
if (!item_exists) {
if (!previous) {
throw new Error('item does not exist', {
cause: 'item_does_not_exist'
});
@ -165,6 +191,12 @@ export class FSDB_COLLECTION<T extends Record<string, any>> {
await this.write_item(item, item_path);
this.emit('update', {
item,
previous,
item_path
});
return item;
}
@ -202,6 +234,11 @@ export class FSDB_COLLECTION<T extends Record<string, any>> {
await Deno.remove(dir);
dir = path.dirname(dir);
} while (dir.length);
this.emit('delete', {
item
});
return item;
}
@ -302,6 +339,11 @@ export class FSDB_COLLECTION<T extends Record<string, any>> {
if (Deno.env.get('FSDB_PERF')) performance.mark('fsdb_all_end');
if (Deno.env.get('FSDB_PERF')) console.dir(performance.measure('fsdb all items time', 'fsdb_all_begin', 'fsdb_all_end'));
this.emit('all', {
options,
results
});
return results;
}
@ -350,6 +392,64 @@ export class FSDB_COLLECTION<T extends Record<string, any>> {
if (Deno.env.get('FSDB_PERF')) performance.mark('fsdb_find_end');
if (Deno.env.get('FSDB_PERF')) console.dir(performance.measure('fsdb find time', 'fsdb_find_begin', 'fsdb_find_end'));
this.emit('find', {
criteria,
options,
results
});
return results;
}
public on(event: string, handler: (event_data: any) => void) {
const listeners: ((event: any) => void)[] = this.event_listeners[event] = this.event_listeners[event] ?? [];
if (!listeners.includes(handler)) {
listeners.push(handler);
}
if (Deno.env.get('FSDB_LOG_EVENTS')) {
console.dir({
on: {
event,
handler
},
listeners
});
}
}
public off(event: string, handler: (event_data: any) => void) {
const listeners: ((event: any) => void)[] = this.event_listeners[event] = this.event_listeners[event] ?? [];
if (listeners.includes(handler)) {
listeners.splice(listeners.indexOf(handler), 1);
}
if (Deno.env.get('FSDB_LOG_EVENTS')) {
console.dir({
off: {
event: event,
handler
},
listeners
});
}
}
private emit(event_name: string, event_data: any) {
const listeners: ((event: any) => void)[] = this.event_listeners[event_name] = this.event_listeners[event_name] ?? [];
if (Deno.env.get('FSDB_LOG_EVENTS')) {
console.dir({
emitting: {
event_name,
event_data,
listeners
}
});
}
for (const listener of listeners) {
listener(event_data);
}
}
}

View file

@ -0,0 +1,319 @@
import * as asserts from '@std/assert';
import * as fsdb from '../fsdb.ts';
import { get_data_dir } from './helpers.ts';
import lurid from '@andyburke/lurid';
import { FSDB_INDEXER_SYMLINKS } from '../indexers/symlinks.ts';
type ITEM = {
id: string;
value: string;
created: string;
};
const item_collection: fsdb.FSDB_COLLECTION<ITEM> = new fsdb.FSDB_COLLECTION<ITEM>({
name: 'test-05-items',
root: get_data_dir() + '/test-05-items',
indexers: {
email: new FSDB_INDEXER_SYMLINKS<ITEM>({
name: 'created',
field: 'created',
organize: (value) => {
const date = new Date(value);
return [`${date.getFullYear()}`, `${date.getMonth()}`, `${date.getDate()}`, `${value}.json`];
}
})
}
});
Deno.test({
name: 'events - create',
permissions: {
env: true,
// https://github.com/denoland/deno/discussions/17258
read: true,
write: true
},
fn: async () => {
asserts.assert(item_collection);
let create_event: any = null;
item_collection.on('create', (event) => {
create_event = event;
});
const now = new Date().toISOString();
const item = {
id: lurid(),
value: 'test',
created: now
};
await item_collection.create(item);
asserts.assert(create_event);
asserts.assertEquals(create_event?.item, item);
asserts.assert(create_event?.item_path);
}
});
Deno.test({
name: 'events - delete',
permissions: {
env: true,
// https://github.com/denoland/deno/discussions/17258
read: true,
write: true
},
fn: async () => {
asserts.assert(item_collection);
let delete_event: any = null;
item_collection.on('delete', (event) => {
delete_event = event;
});
const now = new Date().toISOString();
const item = {
id: lurid(),
value: 'test',
created: now
};
await item_collection.create(item);
await item_collection.delete(item);
asserts.assert(delete_event);
asserts.assertEquals(delete_event?.item, item);
}
});
Deno.test({
name: 'events - write',
permissions: {
env: true,
// https://github.com/denoland/deno/discussions/17258
read: true,
write: true
},
fn: async () => {
asserts.assert(item_collection);
let create_write_event: any = null;
function set_create_event(event: any): void {
create_write_event = event;
}
item_collection.on('write', set_create_event);
const now = new Date().toISOString();
const item = {
id: lurid(),
value: 'test',
created: now
};
await item_collection.create(item);
item_collection.off('write', set_create_event);
asserts.assert(create_write_event);
asserts.assertEquals(create_write_event?.item, item);
asserts.assert(create_write_event.item_path);
const updated_item = { ...item };
updated_item.value = 'different';
let update_write_event: any = null;
function set_update_event(event: any): void {
update_write_event = event;
}
item_collection.on('write', set_update_event);
await item_collection.update(updated_item);
item_collection.off('write', set_update_event);
asserts.assert(update_write_event);
asserts.assertEquals(update_write_event?.item, updated_item);
asserts.assert(update_write_event.item_path);
}
});
Deno.test({
name: 'events - get',
permissions: {
env: true,
// https://github.com/denoland/deno/discussions/17258
read: true,
write: true
},
fn: async () => {
asserts.assert(item_collection);
let get_event: any = null;
item_collection.on('get', (event) => {
get_event = event;
});
const now = new Date().toISOString();
const item = {
id: lurid(),
value: 'test',
created: now
};
await item_collection.create(item);
const fetched_item: ITEM | null = await item_collection.get(item.id);
asserts.assertEquals(fetched_item, item);
asserts.assert(get_event);
asserts.assertEquals(get_event?.item, fetched_item);
asserts.assert(get_event?.item_path);
}
});
Deno.test({
name: 'events - index',
permissions: {
env: true,
// https://github.com/denoland/deno/discussions/17258
read: true,
write: true
},
fn: async () => {
asserts.assert(item_collection);
let index_event: any = null;
item_collection.on('index', (event) => {
index_event = event;
});
const now = new Date().toISOString();
const item = {
id: lurid(),
value: 'test',
created: now
};
await item_collection.create(item);
asserts.assert(index_event);
asserts.assertEquals(index_event?.item, item);
asserts.assert(index_event?.indexer);
}
});
Deno.test({
name: 'events - update',
permissions: {
env: true,
// https://github.com/denoland/deno/discussions/17258
read: true,
write: true
},
fn: async () => {
asserts.assert(item_collection);
let update_event: any = null;
item_collection.on('update', (event) => {
update_event = event;
});
const now = new Date().toISOString();
const item = {
id: lurid(),
value: 'test',
created: now
};
await item_collection.create(item);
const updated_item = { ...item };
updated_item.created = new Date().toISOString();
await item_collection.update(updated_item);
asserts.assert(update_event);
asserts.assertEquals(update_event?.item, updated_item);
asserts.assertEquals(update_event.previous, item);
}
});
Deno.test({
name: 'events - find',
permissions: {
env: true,
// https://github.com/denoland/deno/discussions/17258
read: true,
write: true
},
fn: async () => {
asserts.assert(item_collection);
let find_event: any = null;
item_collection.on('find', (event) => {
find_event = event;
});
const now = new Date().toISOString();
const item = {
id: lurid(),
value: 'test',
created: now
};
await item_collection.create(item);
const criteria = {
created: now
};
const options = {
limit: 3
};
const results = await item_collection.find(criteria, options);
asserts.assert(find_event);
asserts.assertEquals(find_event.criteria, criteria);
asserts.assertEquals(find_event.options?.limit, options.limit);
asserts.assertEquals(find_event.results, results);
}
});
Deno.test({
name: 'events - all',
permissions: {
env: true,
// https://github.com/denoland/deno/discussions/17258
read: true,
write: true
},
fn: async () => {
asserts.assert(item_collection);
let all_event: any = null;
item_collection.on('all', (event) => {
all_event = event;
});
for (let i = 0; i < 5; ++i) {
const item = {
id: lurid(),
value: 'test ' + i,
created: new Date().toISOString()
};
await item_collection.create(item);
}
const options = {
limit: 2
};
const results = await item_collection.all(options);
asserts.assert(all_event);
asserts.assertEquals(results.length, options.limit);
asserts.assertEquals(all_event.options?.limit, options.limit);
asserts.assertEquals(all_event.results, results);
}
});

View file

@ -22,8 +22,8 @@ Deno.test({
};
const item_collection: fsdb.FSDB_COLLECTION<ITEM> = new fsdb.FSDB_COLLECTION<ITEM>({
name: 'test-05-items',
root: get_data_dir() + '/test-05-items'
name: 'test-06-items',
root: get_data_dir() + '/test-06-items'
});
asserts.assert(item_collection);

View file

@ -49,5 +49,5 @@ export function random_phone_number(): string {
const DATA_DIR = lurid();
export function get_data_dir(): string {
return Deno.env.get('TEST_DATA_STORAGE_ROOT') ?? DATA_DIR;
return Deno.env.get('FSDB_TEST_DATA_STORAGE_ROOT') ?? DATA_DIR;
}