@drift-labs/sdk-browser
Version:
SDK for Drift Protocol
157 lines (132 loc) • 3.74 kB
text/typescript
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;
}
}