import {Inject, Injectable} from "@extollo/di" import {Collection, Iterable} from "@extollo/util" import {FirebaseUnit, RTDBRef} from "./units/FirebaseUnit" import * as firebase from "firebase-admin" import {Application} from "@extollo/lib" /** * Base interface for an item in a Firebase RTDB collection. */ export interface FirebaseResourceItem { firebaseID: string; seqID: number; } @Injectable() export class FirebaseResource extends Iterable { protected refName!: RTDBRef ref(): firebase.database.Reference { return Application.getApplication().make(FirebaseUnit).ref(this.refName) } /** Get the next sequential ID. */ async getNextID(): Promise { return new Promise((res, rej) => { this.ref().orderByChild('seqID') .on('value', snapshot => { res((this.resolveObject(snapshot.val()).reverse()?.[0]?.seqID ?? -1) + 1) }, rej) }) } /** Get the record at the ith index. */ async at(i: number): Promise { return new Promise((res, rej) => { this.ref().orderByChild('seqID') .startAt(i).endAt(i) .on('value', snapshot => res(this.resolveObject(snapshot.val())[0]), rej) }) } /** Fetch an array of records in a range. */ async range(start: number, end: number): Promise> { return new Promise>((res, rej) => { this.ref().orderByChild('seqID') .startAt(start).endAt(end) .on('value', snapshot => { res(new Collection(this.resolveObject(snapshot.val()))) }, rej) }) } /** Count the items in the collection. */ async count(): Promise { return new Promise((res, rej) => { this.ref().orderByChild('seqID') .on('value', snapshot => { res(this.resolveObject(snapshot.val()).length) }, rej) }) } /** * Push a new item into the collection. * @param item */ async push(item: T): Promise { item.seqID = await this.getNextID() // @ts-ignore delete item.firebaseID await this.ref().push(item) // Look up the firebaseID await new Promise((res, rej) => { this.ref().orderByChild('seqID') .limitToLast(1) .on('value', snapshot => { if ( snapshot.val() ) { item.firebaseID = Object.keys(snapshot.val())[0] } res() }) }) return item } /** * Given the value of a realtime-database snapshot, resolve it to an array of T. * @param snapshot * @protected */ protected resolveObject(snapshot: any | null | undefined) { if ( !snapshot ) snapshot = {} const returns: T[] = [] for ( const key in snapshot ) { if ( !snapshot.hasOwnProperty(key) ) continue snapshot[key].firebaseID = key returns.push(snapshot[key]) } return returns } clone(): Iterable { const inst = new FirebaseResource() inst.refName = this.refName return inst } }