UNPKG

@bsv/wallet-toolbox

Version:

BRC100 conforming wallet, wallet storage and wallet signer components

401 lines (376 loc) 12.1 kB
import { createSyncMap, EntityBase, EntityCertificate, EntityCertificateField, EntityCommission, EntityOutput, EntityOutputBasket, EntityOutputTag, EntityOutputTagMap, EntityProvenTx, EntityProvenTxReq, EntityStorage, EntitySyncMap, EntityTransaction, EntityTxLabel, EntityTxLabelMap, EntityUser, MergeEntity, SyncError, SyncMap } from '.' import { maxDate, sdk, TableSettings, TableSyncState, verifyId, verifyOne, verifyOneOrNone, verifyTruthy } from '../../../index.client' export class EntitySyncState extends EntityBase<TableSyncState> { constructor(api?: TableSyncState) { const now = new Date() super( api || { syncStateId: 0, created_at: now, updated_at: now, userId: 0, storageIdentityKey: '', storageName: '', init: false, refNum: '', status: 'unknown', when: undefined, errorLocal: undefined, errorOther: undefined, satoshis: undefined, syncMap: JSON.stringify(createSyncMap()) } ) this.errorLocal = this.api.errorLocal ? <SyncError>JSON.parse(this.api.errorLocal) : undefined this.errorOther = this.api.errorOther ? <SyncError>JSON.parse(this.api.errorOther) : undefined this.syncMap = <SyncMap>JSON.parse(this.api.syncMap) this.validateSyncMap(this.syncMap) } validateSyncMap(sm: SyncMap) { for (const key of Object.keys(sm)) { const esm: EntitySyncMap = sm[key] if (typeof esm.maxUpdated_at === 'string') esm.maxUpdated_at = new Date(esm.maxUpdated_at) } } static async fromStorage( storage: sdk.WalletStorageSync, userIdentityKey: string, remoteSettings: TableSettings ): Promise<EntitySyncState> { const { user } = verifyTruthy(await storage.findOrInsertUser(userIdentityKey)) let { syncState: api } = verifyTruthy( await storage.findOrInsertSyncStateAuth( { userId: user.userId, identityKey: userIdentityKey }, remoteSettings.storageIdentityKey, remoteSettings.storageName ) ) if (!api.syncMap || api.syncMap === '{}') api.syncMap = JSON.stringify(createSyncMap()) const ss = new EntitySyncState(api) return ss } /** * Handles both insert and update based on id value: zero indicates insert. * @param storage * @param notSyncMap if not new and true, excludes updating syncMap in storage. * @param trx */ async updateStorage(storage: EntityStorage, notSyncMap?: boolean, trx?: sdk.TrxToken) { this.updated_at = new Date() this.updateApi(notSyncMap && this.id > 0) if (this.id === 0) { await storage.insertSyncState(this.api) } else { const update: Partial<TableSyncState> = { ...this.api } if (notSyncMap) delete update.syncMap delete update.created_at await storage.updateSyncState(verifyId(this.id), update, trx) } } override updateApi(notSyncMap?: boolean): void { this.api.errorLocal = this.apiErrorLocal this.api.errorOther = this.apiErrorOther if (!notSyncMap) this.api.syncMap = this.apiSyncMap } // Pass through api properties set created_at(v: Date) { this.api.created_at = v } get created_at() { return this.api.created_at } set updated_at(v: Date) { this.api.updated_at = v } get updated_at() { return this.api.updated_at } set userId(v: number) { this.api.userId = v } get userId() { return this.api.userId } set storageIdentityKey(v: string) { this.api.storageIdentityKey = v } get storageIdentityKey() { return this.api.storageIdentityKey } set storageName(v: string) { this.api.storageName = v } get storageName() { return this.api.storageName } set init(v: boolean) { this.api.init = v } get init() { return this.api.init } set refNum(v: string) { this.api.refNum = v } get refNum() { return this.api.refNum } set status(v: sdk.SyncStatus) { this.api.status = v } get status(): sdk.SyncStatus { return this.api.status } set when(v: Date | undefined) { this.api.when = v } get when() { return this.api.when } set satoshis(v: number | undefined) { this.api.satoshis = v } get satoshis() { return this.api.satoshis } get apiErrorLocal() { return this.errorToString(this.errorLocal) } get apiErrorOther() { return this.errorToString(this.errorOther) } get apiSyncMap() { return JSON.stringify(this.syncMap) } override get id(): number { return this.api.syncStateId } set id(id: number) { this.api.syncStateId = id } override get entityName(): string { return 'syncState' } override get entityTable(): string { return 'sync_states' } static mergeIdMap(fromMap: Record<number, number>, toMap: Record<number, number>) { for (const [key, value] of Object.entries(fromMap)) { const fromValue = fromMap[key] const toValue = toMap[key] if (toValue !== undefined && toValue !== fromValue) throw new sdk.WERR_INVALID_PARAMETER( 'syncMap', `an unmapped id or the same mapped id. ${key} maps to ${toValue} not equal to ${fromValue}` ) if (toValue === undefined) toMap[key] = value } } /** * Merge additions to the syncMap * @param iSyncMap */ mergeSyncMap(iSyncMap: SyncMap) { EntitySyncState.mergeIdMap(iSyncMap.provenTx.idMap!, this.syncMap.provenTx.idMap!) EntitySyncState.mergeIdMap(iSyncMap.outputBasket.idMap!, this.syncMap.outputBasket.idMap!) EntitySyncState.mergeIdMap(iSyncMap.transaction.idMap!, this.syncMap.transaction.idMap!) EntitySyncState.mergeIdMap(iSyncMap.provenTxReq.idMap!, this.syncMap.provenTxReq.idMap!) EntitySyncState.mergeIdMap(iSyncMap.txLabel.idMap!, this.syncMap.txLabel.idMap!) EntitySyncState.mergeIdMap(iSyncMap.output.idMap!, this.syncMap.output.idMap!) EntitySyncState.mergeIdMap(iSyncMap.outputTag.idMap!, this.syncMap.outputTag.idMap!) EntitySyncState.mergeIdMap(iSyncMap.certificate.idMap!, this.syncMap.certificate.idMap!) EntitySyncState.mergeIdMap(iSyncMap.commission.idMap!, this.syncMap.commission.idMap!) } // stringified api properties errorLocal: SyncError | undefined errorOther: SyncError | undefined syncMap: SyncMap /** * Eliminate any properties besides code and description */ private errorToString(e: SyncError | undefined): string | undefined { if (!e) return undefined const es: SyncError = { code: e.code, description: e.description, stack: e.stack } return JSON.stringify(es) } override equals(ei: TableSyncState, syncMap?: SyncMap | undefined): boolean { return false } override async mergeNew( storage: EntityStorage, userId: number, syncMap: SyncMap, trx?: sdk.TrxToken ): Promise<void> {} override async mergeExisting( storage: EntityStorage, since: Date | undefined, ei: TableSyncState, syncMap: SyncMap, trx?: sdk.TrxToken ): Promise<boolean> { return false } makeRequestSyncChunkArgs( forIdentityKey: string, forStorageIdentityKey: string, maxRoughSize?: number, maxItems?: number ): sdk.RequestSyncChunkArgs { const a: sdk.RequestSyncChunkArgs = { identityKey: forIdentityKey, maxRoughSize: maxRoughSize || 10000000, maxItems: maxItems || 1000, offsets: [], since: this.when, fromStorageIdentityKey: this.storageIdentityKey, toStorageIdentityKey: forStorageIdentityKey } for (const ess of [ this.syncMap.provenTx, this.syncMap.outputBasket, this.syncMap.outputTag, this.syncMap.txLabel, this.syncMap.transaction, this.syncMap.output, this.syncMap.txLabelMap, this.syncMap.outputTagMap, this.syncMap.certificate, this.syncMap.certificateField, this.syncMap.commission, this.syncMap.provenTxReq ]) { if (!ess || !ess.entityName) debugger a.offsets.push({ name: ess.entityName, offset: ess.count }) } return a } static syncChunkSummary(c: sdk.SyncChunk): string { let log = '' log += `SYNC CHUNK SUMMARY from storage: ${c.fromStorageIdentityKey} to storage: ${c.toStorageIdentityKey} for user: ${c.userIdentityKey} ` if (c.user) log += ` USER activeStorage ${c.user.activeStorage}\n` if (!!c.provenTxs) { log += ` PROVEN_TXS\n` for (const r of c.provenTxs) { log += ` ${r.provenTxId} ${r.txid}\n` } } if (!!c.provenTxReqs) { log += ` PROVEN_TX_REQS\n` for (const r of c.provenTxReqs) { log += ` ${r.provenTxReqId} ${r.txid} ${r.status} ${r.provenTxId || ''}\n` } } if (!!c.transactions) { log += ` TRANSACTIONS\n` for (const r of c.transactions) { log += ` ${r.transactionId} ${r.txid} ${r.status} ${r.provenTxId || ''} sats:${r.satoshis}\n` } } if (!!c.outputs) { log += ` OUTPUTS\n` for (const r of c.outputs) { log += ` ${r.outputId} ${r.txid}.${r.vout} ${r.transactionId} ${r.spendable ? 'spendable' : ''} sats:${r.satoshis}\n` } } return log } async processSyncChunk( writer: EntityStorage, args: sdk.RequestSyncChunkArgs, chunk: sdk.SyncChunk ): Promise<{ done: boolean maxUpdated_at: Date | undefined updates: number inserts: number }> { const mes = [ new MergeEntity(chunk.provenTxs, EntityProvenTx.mergeFind, this.syncMap.provenTx), new MergeEntity(chunk.outputBaskets, EntityOutputBasket.mergeFind, this.syncMap.outputBasket), new MergeEntity(chunk.outputTags, EntityOutputTag.mergeFind, this.syncMap.outputTag), new MergeEntity(chunk.txLabels, EntityTxLabel.mergeFind, this.syncMap.txLabel), new MergeEntity(chunk.transactions, EntityTransaction.mergeFind, this.syncMap.transaction), new MergeEntity(chunk.outputs, EntityOutput.mergeFind, this.syncMap.output), new MergeEntity(chunk.txLabelMaps, EntityTxLabelMap.mergeFind, this.syncMap.txLabelMap), new MergeEntity(chunk.outputTagMaps, EntityOutputTagMap.mergeFind, this.syncMap.outputTagMap), new MergeEntity(chunk.certificates, EntityCertificate.mergeFind, this.syncMap.certificate), new MergeEntity(chunk.certificateFields, EntityCertificateField.mergeFind, this.syncMap.certificateField), new MergeEntity(chunk.commissions, EntityCommission.mergeFind, this.syncMap.commission), new MergeEntity(chunk.provenTxReqs, EntityProvenTxReq.mergeFind, this.syncMap.provenTxReq) ] let updates = 0 let inserts = 0 let maxUpdated_at: Date | undefined = undefined let done = true // Merge User if (chunk.user) { const ei = chunk.user const { found, eo } = await EntityUser.mergeFind(writer, this.userId, ei) if (found) { if (await eo.mergeExisting(writer, args.since, ei)) { maxUpdated_at = maxDate(maxUpdated_at, ei.updated_at) updates++ } } } // Merge everything else... for (const me of mes) { const r = await me.merge(args.since, writer, this.userId, this.syncMap) // The counts become the offsets for the next chunk. me.esm.count += me.stateArray?.length || 0 updates += r.updates inserts += r.inserts maxUpdated_at = maxDate(maxUpdated_at, me.esm.maxUpdated_at) // If any entity type either did not report results or if there were at least one, then we aren't done. if (me.stateArray === undefined || me.stateArray.length > 0) done = false //if (me.stateArray !== undefined && me.stateArray.length > 0) // console.log(`merged ${me.stateArray?.length} ${me.esm.entityName} ${r.inserts} inserted, ${r.updates} updated`); } if (done) { // Next batch starts further in the future with offsets of zero. this.when = maxUpdated_at for (const me of mes) me.esm.count = 0 } await this.updateStorage(writer, false) return { done, maxUpdated_at, updates, inserts } } }