UNPKG

@drift-labs/sdk-browser

Version:
157 lines (132 loc) 3.74 kB
import { DataAndSlot, NotSubscribedError, UserStatsAccountSubscriber, UserStatsAccountEvents, } from './types'; import { Program } from '@coral-xyz/anchor'; import StrictEventEmitter from 'strict-event-emitter-types'; import { EventEmitter } from 'events'; import { PublicKey } from '@solana/web3.js'; import { UserStatsAccount } from '../types'; import { BulkAccountLoader } from './bulkAccountLoader'; export class PollingUserStatsAccountSubscriber implements UserStatsAccountSubscriber { isSubscribed: boolean; program: Program; eventEmitter: StrictEventEmitter<EventEmitter, UserStatsAccountEvents>; userStatsAccountPublicKey: PublicKey; accountLoader: BulkAccountLoader; callbackId?: string; errorCallbackId?: string; userStats?: DataAndSlot<UserStatsAccount>; public constructor( program: Program, userStatsAccountPublicKey: PublicKey, accountLoader: BulkAccountLoader ) { this.isSubscribed = false; this.program = program; this.accountLoader = accountLoader; this.eventEmitter = new EventEmitter(); this.userStatsAccountPublicKey = userStatsAccountPublicKey; } async subscribe(userStatsAccount?: UserStatsAccount): Promise<boolean> { if (this.isSubscribed) { return true; } if (userStatsAccount) { this.userStats = { data: userStatsAccount, slot: undefined }; } await this.addToAccountLoader(); await this.fetchIfUnloaded(); if (this.doesAccountExist()) { this.eventEmitter.emit('update'); } this.isSubscribed = true; return true; } async addToAccountLoader(): Promise<void> { if (this.callbackId !== undefined) { return; } this.callbackId = await this.accountLoader.addAccount( this.userStatsAccountPublicKey, (buffer, slot: number) => { if (!buffer) { return; } if (this.userStats && this.userStats.slot > slot) { return; } const account = this.program.account.userStats.coder.accounts.decodeUnchecked( 'UserStats', buffer ); this.userStats = { data: account, slot }; this.eventEmitter.emit('userStatsAccountUpdate', account); this.eventEmitter.emit('update'); } ); this.errorCallbackId = this.accountLoader.addErrorCallbacks((error) => { this.eventEmitter.emit('error', error); }); } async fetchIfUnloaded(): Promise<void> { if (this.userStats === undefined) { await this.fetch(); } } async fetch(): Promise<void> { try { const dataAndContext = await this.program.account.userStats.fetchAndContext( this.userStatsAccountPublicKey, this.accountLoader.commitment ); if (dataAndContext.context.slot > (this.userStats?.slot ?? 0)) { this.userStats = { data: dataAndContext.data as UserStatsAccount, slot: dataAndContext.context.slot, }; } } catch (e) { console.log( `PollingUserStatsAccountSubscriber.fetch() UserStatsAccount does not exist: ${e.message}` ); } } doesAccountExist(): boolean { return this.userStats !== undefined; } async unsubscribe(): Promise<void> { if (!this.isSubscribed) { return; } this.accountLoader.removeAccount( this.userStatsAccountPublicKey, this.callbackId ); this.callbackId = undefined; this.accountLoader.removeErrorCallbacks(this.errorCallbackId); this.errorCallbackId = undefined; this.isSubscribed = false; } assertIsSubscribed(): void { if (!this.isSubscribed) { throw new NotSubscribedError( 'You must call `subscribe` before using this function' ); } } public getUserStatsAccountAndSlot(): DataAndSlot<UserStatsAccount> { if (!this.doesAccountExist()) { throw new NotSubscribedError( 'You must call `subscribe` or `fetch` before using this function' ); } return this.userStats; } }