Skip to content

Commit a44e65f

Browse files
feat: introduce persistent cache for providers
- define a contract of the cache - enable chain history and utxo providers to use injected cache - create persistent cache storage - utilizing web extension storage - implementing the cache contract - having data envicting capability
1 parent 24061e8 commit a44e65f

File tree

6 files changed

+147
-12
lines changed

6 files changed

+147
-12
lines changed

packages/cardano-services-client/src/ChainHistoryProvider/BlockfrostChainHistoryProvider.ts

+15-6
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ import {
2424
import { Logger } from 'ts-log';
2525
import omit from 'lodash/omit.js';
2626
import uniq from 'lodash/uniq.js';
27+
import type { Cache } from '@cardano-sdk/util';
2728
import type { Responses } from '@blockfrost/blockfrost-js';
2829
import type { Schemas } from '@blockfrost/blockfrost-js/lib/types/open-api';
2930

@@ -33,12 +34,20 @@ export const DB_MAX_SAFE_INTEGER = 2_147_483_647;
3334
type BlockfrostTx = Pick<Responses['address_transactions_content'][0], 'block_height' | 'tx_index'>;
3435
const compareTx = (a: BlockfrostTx, b: BlockfrostTx) => a.block_height - b.block_height || a.tx_index - b.tx_index;
3536

37+
type BlockfrostChainHistoryProviderDependencies = {
38+
cache: Cache<Cardano.HydratedTx>;
39+
client: BlockfrostClient;
40+
networkInfoProvider: NetworkInfoProvider;
41+
logger: Logger;
42+
};
43+
3644
export class BlockfrostChainHistoryProvider extends BlockfrostProvider implements ChainHistoryProvider {
37-
private readonly cache: Map<string, Cardano.HydratedTx> = new Map();
45+
private readonly cache: Cache<Cardano.HydratedTx>;
3846
private networkInfoProvider: NetworkInfoProvider;
3947

40-
constructor(client: BlockfrostClient, networkInfoProvider: NetworkInfoProvider, logger: Logger) {
48+
constructor({ cache, client, networkInfoProvider, logger }: BlockfrostChainHistoryProviderDependencies) {
4149
super(client, logger);
50+
this.cache = cache;
4251
this.networkInfoProvider = networkInfoProvider;
4352
}
4453

@@ -478,12 +487,12 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
478487
public async transactionsByHashes({ ids }: TransactionsByIdsArgs): Promise<Cardano.HydratedTx[]> {
479488
return Promise.all(
480489
ids.map(async (id) => {
481-
if (this.cache.has(id)) {
482-
return this.cache.get(id)!;
483-
}
490+
const cached = await this.cache.get(id);
491+
if (cached) return cached;
492+
484493
try {
485494
const fetchedTransaction = await this.fetchTransaction(id);
486-
this.cache.set(id, fetchedTransaction);
495+
void this.cache.set(id, fetchedTransaction);
487496
return fetchedTransaction;
488497
} catch (error) {
489498
throw this.toProviderError(error);

packages/cardano-services-client/src/UtxoProvider/BlockfrostUtxoProvider.ts

+18-6
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,22 @@
1-
import { BlockfrostProvider, BlockfrostToCore, fetchSequentially } from '../blockfrost';
1+
import { BlockfrostClient, BlockfrostProvider, BlockfrostToCore, fetchSequentially } from '../blockfrost';
22
import { Cardano, Serialization, UtxoByAddressesArgs, UtxoProvider } from '@cardano-sdk/core';
3+
import { Logger } from 'ts-log';
4+
import type { Cache } from '@cardano-sdk/util';
35
import type { Responses } from '@blockfrost/blockfrost-js';
46

7+
type BlockfrostUtxoProviderDependencies = {
8+
client: BlockfrostClient;
9+
cache: Cache<Cardano.Tx>;
10+
logger: Logger;
11+
};
12+
513
export class BlockfrostUtxoProvider extends BlockfrostProvider implements UtxoProvider {
6-
private readonly cache: Map<string, Cardano.Tx> = new Map();
14+
private readonly cache: Cache<Cardano.Tx>;
15+
16+
constructor({ cache, client, logger }: BlockfrostUtxoProviderDependencies) {
17+
super(client, logger);
18+
this.cache = cache;
19+
}
720

821
protected async fetchUtxos(addr: Cardano.PaymentAddress, paginationQueryString: string): Promise<Cardano.Utxo[]> {
922
const queryString = `addresses/${addr.toString()}/utxos?${paginationQueryString}`;
@@ -29,9 +42,8 @@ export class BlockfrostUtxoProvider extends BlockfrostProvider implements UtxoPr
2942
});
3043
}
3144
protected async fetchDetailsFromCBOR(hash: string) {
32-
if (this.cache.has(hash)) {
33-
return this.cache.get(hash);
34-
}
45+
const cached = await this.cache.get(hash);
46+
if (cached) return cached;
3547

3648
const result = await this.fetchCBOR(hash)
3749
.then((cbor) => {
@@ -48,7 +60,7 @@ export class BlockfrostUtxoProvider extends BlockfrostProvider implements UtxoPr
4860
return null;
4961
}
5062

51-
this.cache.set(hash, result);
63+
void this.cache.set(hash, result);
5264
return result;
5365
}
5466

packages/util/src/types.ts

+5
Original file line numberDiff line numberDiff line change
@@ -28,3 +28,8 @@ type Impossible<K extends keyof any> = {
2828
[P in K]: never;
2929
};
3030
export type NoExtraProperties<T, U> = U & Impossible<Exclude<keyof U, keyof T>>;
31+
32+
export type Cache<T> = {
33+
get(key: string): Promise<T | undefined>;
34+
set(key: string, value: T): Promise<void>;
35+
};

packages/web-extension/src/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -4,3 +4,4 @@ export * from './supplyDistributionTracker';
44
export * from './keyAgent';
55
export * from './walletManager';
66
export * as cip30 from './cip30';
7+
export * from './storage';
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export * from './persistentCacheStorage';
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,107 @@
1+
import { Storage, storage } from 'webextension-polyfill';
2+
import StorageArea = Storage.StorageArea;
3+
4+
type MetadataItem = {
5+
accessTime: number;
6+
};
7+
8+
type Metadata = Record<string, MetadataItem>;
9+
10+
const sizeOfChunkToBePurged = 0.1;
11+
12+
const isGetBytesInUsePresent = (
13+
storageLocal: StorageArea
14+
): storageLocal is StorageArea & { getBytesInUse: (keys?: string | string[]) => Promise<number> } =>
15+
'getBytesInUse' in storageLocal;
16+
17+
export const createPersistentCacheStorage = <T>({
18+
fallbackMaxCollectionItemsGuard,
19+
resourceName,
20+
quotaInBytes
21+
}: {
22+
fallbackMaxCollectionItemsGuard: number;
23+
resourceName: string;
24+
quotaInBytes: number;
25+
}) => {
26+
const loaded = new Map<string, T>();
27+
const metadataKey = `${resourceName}-metadata`;
28+
const getItemKey = (key: string) => `${resourceName}-item-${key}`;
29+
30+
const getMetadata = async () => {
31+
const result = await storage.local.get(metadataKey);
32+
return result[metadataKey] as Metadata;
33+
};
34+
35+
const updateAccessTime = async (key: string) => {
36+
const metadata = await getMetadata();
37+
const nextMetadata: Metadata = {
38+
...metadata,
39+
[key]: {
40+
accessTime: Date.now()
41+
}
42+
};
43+
await storage.local.set({ [metadataKey]: nextMetadata });
44+
};
45+
46+
const isQuotaExceeded = async () => {
47+
const metadata = await getMetadata();
48+
const allCollectionKeys = [metadataKey, ...Object.keys(metadata)];
49+
50+
// Polyfill we use does not list the getBytesInUse method but that method exists in chrome API
51+
if (isGetBytesInUsePresent(storage.local)) {
52+
const bytesInUse = await storage.local.getBytesInUse(allCollectionKeys);
53+
return bytesInUse > quotaInBytes;
54+
}
55+
56+
return allCollectionKeys.length > fallbackMaxCollectionItemsGuard;
57+
};
58+
59+
const evict = async () => {
60+
let metadata = await getMetadata();
61+
const mostDatedKeysToPurge = Object.entries(metadata)
62+
.map(([key, { accessTime }]) => ({ accessTime, key }))
63+
.sort((a, b) => a.accessTime - b.accessTime)
64+
.filter((_, index, self) => {
65+
const numberOfItemsToPurge = Math.abs(self.length * sizeOfChunkToBePurged);
66+
return index < numberOfItemsToPurge;
67+
})
68+
.map((i) => i.key);
69+
70+
await storage.local.remove(mostDatedKeysToPurge);
71+
metadata = await getMetadata();
72+
for (const key of mostDatedKeysToPurge) {
73+
delete metadata[key];
74+
}
75+
await storage.local.set({ [metadataKey]: metadata });
76+
};
77+
78+
return {
79+
async get(key: string) {
80+
const itemKey = getItemKey(key);
81+
82+
let value = loaded.get(key);
83+
if (!value) {
84+
const result = await storage.local.get(itemKey);
85+
value = result[itemKey];
86+
}
87+
88+
if (value) {
89+
void updateAccessTime(itemKey);
90+
}
91+
92+
return value;
93+
},
94+
async set(key: string, value: T) {
95+
const itemKey = getItemKey(key);
96+
loaded.set(itemKey, value);
97+
await storage.local.set({ [itemKey]: value });
98+
99+
void (async () => {
100+
await updateAccessTime(itemKey);
101+
if (await isQuotaExceeded()) await evict();
102+
})();
103+
}
104+
};
105+
};
106+
107+
export type CreatePersistentCacheStorage = typeof createPersistentCacheStorage;

0 commit comments

Comments
 (0)