Skip to content

Commit

Permalink
feat: browser compatible BlockfrostChainHistoryProvider
Browse files Browse the repository at this point in the history
  • Loading branch information
rhyslbw committed Nov 19, 2024
1 parent cde3d51 commit 4f6dced
Show file tree
Hide file tree
Showing 7 changed files with 113 additions and 117 deletions.
Original file line number Diff line number Diff line change
@@ -1,14 +1,15 @@
// eslint-disable-next-line jsdoc/check-param-names
import * as Crypto from '@cardano-sdk/crypto';
import { BlockfrostProvider, BlockfrostProviderDependencies } from '../../util/BlockfrostProvider/BlockfrostProvider';

import {
BlockfrostClient,
BlockfrostProvider,
BlockfrostToCore,
BlockfrostTransactionContent,
blockfrostMetadataToTxMetadata,
blockfrostToProviderError,
fetchByAddressSequentially,
isBlockfrostNotFoundError
} from '@cardano-sdk/cardano-services-client';
} from '../blockfrost';
import {
BlocksByIdsArgs,
Cardano,
Expand All @@ -22,22 +23,19 @@ import {
TransactionsByIdsArgs,
createSlotEpochCalc
} from '@cardano-sdk/core';
import { DB_MAX_SAFE_INTEGER } from '../DbSyncChainHistory/queries';
import { Logger } from 'ts-log';
import { Responses } from '@blockfrost/blockfrost-js';
import { Schemas } from '@blockfrost/blockfrost-js/lib/types/open-api';
import omit from 'lodash/omit.js';

type WithCertIndex<T> = T & { cert_index: number };

export interface BlockfrostChainHistoryProviderDependencies extends BlockfrostProviderDependencies {
networkInfoProvider: NetworkInfoProvider;
}
export const DB_MAX_SAFE_INTEGER = 2_147_483_647;

export class BlockfrostChainHistoryProvider extends BlockfrostProvider implements ChainHistoryProvider {
private networkInfoProvider: NetworkInfoProvider;

constructor({ logger, blockfrost, networkInfoProvider }: BlockfrostChainHistoryProviderDependencies) {
super({ blockfrost, logger });
constructor(client: BlockfrostClient, networkInfoProvider: NetworkInfoProvider, logger: Logger) {
super(client, logger);
this.networkInfoProvider = networkInfoProvider;
}

Expand All @@ -46,7 +44,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
redeemer_count
}: Responses['tx_content']): Promise<Cardano.Redeemer[] | undefined> {
if (!redeemer_count) return;
return this.blockfrost.txsRedeemers(hash).then((response) =>
return this.request<Responses['tx_content_redeemers']>(`txs/${hash}/redeemers`).then((response) =>
response.map(
({ purpose, script_hash, unit_mem, unit_steps, tx_index }): Cardano.Redeemer => ({
data: Buffer.from(script_hash),
Expand Down Expand Up @@ -79,7 +77,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
hash
}: Responses['tx_content']): Promise<Cardano.Withdrawal[] | undefined> {
if (!withdrawal_count) return;
return this.blockfrost.txsWithdrawals(hash).then((response) =>
return this.request<Responses['tx_content_withdrawals']>(`txs/${hash}/withdrawals`).then((response) =>
response.map(
({ address, amount }): Cardano.Withdrawal => ({
quantity: BigInt(amount),
Expand Down Expand Up @@ -107,7 +105,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
}

protected async fetchPoolRetireCerts(hash: string): Promise<WithCertIndex<Cardano.PoolRetirementCertificate>[]> {
return this.blockfrost.txsPoolRetires(hash).then((response) =>
return this.request<Responses['tx_content_pool_retires']>(`txs/${hash}/pool_retires`).then((response) =>
response.map(({ pool_id, retiring_epoch, cert_index }) => ({
__typename: Cardano.CertificateType.PoolRetirement,
cert_index,
Expand All @@ -118,7 +116,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
}

protected async fetchPoolUpdateCerts(hash: string): Promise<WithCertIndex<Cardano.PoolRegistrationCertificate>[]> {
return this.blockfrost.txsPoolUpdates(hash).then((response) =>
return this.request<Responses['tx_content_pool_certs']>(`txs/${hash}/pool_updates`).then((response) =>
response.map(({ pool_id, cert_index, fixed_cost, margin_cost, pledge, reward_account, vrf_key }) => ({
__typename: Cardano.CertificateType.PoolRegistration,
cert_index,
Expand All @@ -138,10 +136,9 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
}

async fetchCBOR(hash: string): Promise<string> {
return this.blockfrost
.instance<Schemas['script_cbor']>(`txs/${hash}/cbor`)
return this.request<Responses['tx_content_cbor']>(`txs/${hash}/cbor`)
.then((response) => {
if (response.body.cbor) return response.body.cbor;
if (response) return response.cbor;
throw new Error('CBOR is null');
})
.catch((_error) => {
Expand All @@ -163,7 +160,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
}

protected async fetchMirCerts(hash: string): Promise<WithCertIndex<Cardano.MirCertificate>[]> {
return this.blockfrost.txsMirs(hash).then((response) =>
return this.request<Responses['tx_content_mirs']>(`txs/${hash}/mirs`).then((response) =>
response.map(({ address, amount, cert_index, pot }) => ({
__typename: Cardano.CertificateType.MIR,
cert_index,
Expand All @@ -176,7 +173,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
}

protected async fetchStakeCerts(hash: string): Promise<WithCertIndex<Cardano.StakeAddressCertificate>[]> {
return this.blockfrost.txsStakes(hash).then((response) =>
return this.request<Responses['tx_content_stake_addr']>(`txs/${hash}/stakes`).then((response) =>
response.map(({ address, cert_index, registration }) => ({
__typename: registration
? Cardano.CertificateType.StakeRegistration
Expand All @@ -191,7 +188,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
}

protected async fetchDelegationCerts(hash: string): Promise<WithCertIndex<Cardano.StakeDelegationCertificate>[]> {
return this.blockfrost.txsDelegations(hash).then((response) =>
return this.request<Responses['tx_content_delegations']>(`txs/${hash}/delegations`).then((response) =>
response.map(({ address, pool_id, cert_index }) => ({
__typename: Cardano.CertificateType.StakeDelegation,
cert_index,
Expand Down Expand Up @@ -230,8 +227,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement

protected async fetchJsonMetadataAsAuxiliaryData(txHash: string): Promise<Cardano.AuxiliaryData | undefined> {
const UNDEFINED = undefined;
return this.blockfrost
.txsMetadata(txHash)
return this.request<Responses['tx_content_metadata']>(`txs/${txHash}/metadata`)
.then((m) => {
const metadata = blockfrostMetadataToTxMetadata(m);
return metadata && metadata.size > 0
Expand All @@ -257,7 +253,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
}

protected async fetchEpochParameters(epochNo: Cardano.EpochNo): Promise<Schemas['epoch_param_content']> {
return await this.blockfrost.epochsParameters(epochNo);
return await this.request<Responses['epoch_param_content']>(`epochs/${epochNo}/parameters`);
}

protected async processCertificates(
Expand Down Expand Up @@ -375,7 +371,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
const txFromCBOR = await this.fetchDetailsFromCBOR(id);
if (!txFromCBOR) return;

const utxos: Schemas['tx_content_utxo'] = (await this.blockfrost.txsUtxos(id)) as Schemas['tx_content_utxo'];
const utxos = await this.request<Responses['tx_content_utxo']>(`txs/${id}/utxos`);

// We can't use txFromCBOR.body.inputs since it misses HydratedTxIn.address
const { inputs, outputs, collaterals } = this.transactionUtxos(utxos, txFromCBOR);
Expand Down Expand Up @@ -418,11 +414,11 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement

protected async fetchTransaction(txId: Cardano.TransactionId): Promise<Cardano.HydratedTx> {
try {
const txContent = await this.blockfrost.txs(txId.toString());
const txContent = await this.request<Responses['tx_content']>(`txs/${txId.toString()}`);

return (await this.transactionDetailsUsingCBOR(txContent)) ?? (await this.transactionDetailsUsingAPIs(txContent));
} catch (error) {
throw blockfrostToProviderError(error);
throw this.toProviderError(error);
}
}

Expand All @@ -444,7 +440,9 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement

public async blocksByHashes({ ids }: BlocksByIdsArgs): Promise<Cardano.ExtendedBlockInfo[]> {
try {
const responses = await Promise.all(ids.map((id) => this.blockfrost.blocks(id.toString())));
const responses = await Promise.all(
ids.map((id) => this.request<Responses['block_content']>(`blocks/${id.toString()}`))
);
return responses.map((response) => {
if (!response.epoch || !response.epoch_slot || !response.height || !response.slot || !response.block_vrf) {
throw new ProviderError(ProviderFailure.Unknown, null, 'Queried unsupported block');
Expand All @@ -470,15 +468,15 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
};
});
} catch (error) {
throw blockfrostToProviderError(error);
throw this.toProviderError(error);
}
}

public async transactionsByHashes({ ids }: TransactionsByIdsArgs): Promise<Cardano.HydratedTx[]> {
try {
return Promise.all(ids.map((id) => this.fetchTransaction(id)));
} catch (error) {
throw blockfrostToProviderError(error);
throw this.toProviderError(error);
}
}

Expand All @@ -505,11 +503,15 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
transactions.length > 0 &&
transactions[transactions.length - 1].block_height < blockRange!.lowerBound!
: undefined,
request: (addr: Cardano.PaymentAddress, paginationOptions) =>
this.blockfrost.addressesTransactions(addr.toString(), paginationOptions, {
from: blockRange?.lowerBound ? blockRange?.lowerBound.toString() : undefined,
to: blockRange?.upperBound ? blockRange?.upperBound.toString() : undefined
})
request: (addr: Cardano.PaymentAddress, { page, count, order }) => {
let queryString = `addresses/${addr.toString()}/transactions`;
if (page) queryString += `?page=${page}`;
if (count) queryString += `?count=${count}`;
if (order) queryString += `?order=${order}`;
if (blockRange?.lowerBound) queryString += `?from=${blockRange.lowerBound.toString()}`;
if (blockRange?.upperBound) queryString += `?to=${blockRange.upperBound.toString()}`;
return this.request<Responses['address_transactions_content']>(queryString);
}
})
)
);
Expand All @@ -526,7 +528,7 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement

return { pageResults, totalResultCount: allTransactions.length };
} catch (error) {
throw blockfrostToProviderError(error);
throw this.toProviderError(error);
}
}

Expand Down Expand Up @@ -581,6 +583,6 @@ export class BlockfrostChainHistoryProvider extends BlockfrostProvider implement
}

private fetchUtxos(id: Cardano.TransactionId): Promise<Schemas['tx_content_utxo']> {
return this.blockfrost.txsUtxos(id);
return this.request<Responses['tx_content_utxo']>(`txs/${id}/utxos`);
}
}
Original file line number Diff line number Diff line change
@@ -1 +1,2 @@
export * from './BlockfrostChainHistoryProvider';
export * from './chainHistoryHttpProvider';
Loading

0 comments on commit 4f6dced

Please sign in to comment.