Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

blockchain.getBlock, getBlockHeight, getBlockHash #91

Merged
merged 8 commits into from
Apr 6, 2021
108 changes: 93 additions & 15 deletions packages/jellyfish-api-core/__tests__/category/blockchain.test.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { RegTestContainer, MasterNodeRegTestContainer } from '@defichain/testcontainers'
import { ContainerAdapterClient } from '../container_adapter_client'
import waitForExpect from 'wait-for-expect'
import { Block, RawTx } from '../../src/category/blockchain'

describe('non masternode', () => {
const container = new RegTestContainer()
Expand Down Expand Up @@ -55,26 +56,103 @@ describe('masternode', () => {
await container.stop()
})

it('should getBlockchainInfo', async () => {
await waitForExpect(async () => {
describe('getBlockchainInfo', () => {
it('should getBlockchainInfo', async () => {
await waitForExpect(async () => {
const info = await client.blockchain.getBlockchainInfo()
await expect(info.blocks).toBeGreaterThan(1)
})

const info = await client.blockchain.getBlockchainInfo()
await expect(info.blocks).toBeGreaterThan(1)

expect(info.chain).toBe('regtest')
expect(info.blocks).toBeGreaterThan(0)
expect(info.headers).toBeGreaterThan(0)

expect(info.bestblockhash.length).toBe(64)
expect(info.difficulty).toBeGreaterThan(0)
expect(info.mediantime).toBeGreaterThan(1550000000)

expect(info.verificationprogress).toBe(1)
expect(info.initialblockdownload).toBe(false)
expect(info.chainwork.length).toBe(64)
expect(info.size_on_disk).toBeGreaterThan(0)
expect(info.pruned).toBe(false)
})
})

const info = await client.blockchain.getBlockchainInfo()
describe('getBlock', () => {
let blockHash: string = ''

expect(info.chain).toBe('regtest')
expect(info.blocks).toBeGreaterThan(0)
expect(info.headers).toBeGreaterThan(0)
beforeAll(async () => {
canonbrother marked this conversation as resolved.
Show resolved Hide resolved
blockHash = await client.blockchain.getBlockHash(1)
})

expect(info.bestblockhash.length).toBe(64)
expect(info.difficulty).toBeGreaterThan(0)
expect(info.mediantime).toBeGreaterThan(1550000000)
it('test getblock with verbo 0, should return hash', async () => {
const data: string | Block<unknown> = await client.blockchain.getBlock(blockHash, 0)
expect(data).not.toBeNull()
})

expect(info.verificationprogress).toBe(1)
expect(info.initialblockdownload).toBe(false)
expect(info.chainwork.length).toBe(64)
expect(info.size_on_disk).toBeGreaterThan(0)
expect(info.pruned).toBe(false)
it('test getblock with verbo 1', async () => {
const data: string | Block<string> = await client.blockchain.getBlock<string>(blockHash, 1)
const block = data as Block<string>
expect(block).toHaveProperty('hash')
expect(block.confirmations).toStrictEqual(2)
expect(block.strippedsize).toStrictEqual(360)
expect(block.size).toStrictEqual(396)
expect(block.weight).toStrictEqual(1476)
expect(block.height).toStrictEqual(1)
canonbrother marked this conversation as resolved.
Show resolved Hide resolved
expect(block.masternode).toStrictEqual('e86c027861cc0af423313f4152a44a83296a388eb51bf1a6dde9bd75bed55fb4')
expect(block.minter).toStrictEqual('mswsMVsyGMj1FzDMbbxw2QW3KvQAv2FKiy')
expect(block.mintedBlocks).toStrictEqual(1)
expect(block.stakeModifier).toStrictEqual('fdd82eafa32300653d3b2d9b98a6650b4b15fe2eb32cdd847d3bf2272514cfbf')
expect(block.version).toStrictEqual(536870912)
expect(block.versionHex).toStrictEqual('20000000')
expect(block.merkleroot).toStrictEqual('00eed320c213f506038fa29f77d4d2535232fa97b7789ff6fb516c63201c5e44')
expect(block.tx.length).toStrictEqual(1)
expect(block.tx[0]).toStrictEqual('00eed320c213f506038fa29f77d4d2535232fa97b7789ff6fb516c63201c5e44')
expect(block).toHaveProperty('time')
expect(block).toHaveProperty('mediantime')
expect(block.bits).toStrictEqual('207fffff')
expect(block.difficulty).toStrictEqual(4.656542373906925e-10)
expect(block.chainwork).toStrictEqual('0000000000000000000000000000000000000000000000000000000000000004')
expect(block.nTx).toStrictEqual(1)
expect(block.previousblockhash).toStrictEqual('0091f00915b263d08eba2091ba70ba40cea75242b3f51ea29f4a1b8d7814cd01')

// NOTE(canonbrother): Get block without verbo, the verbo default should be 1
const dataWithoutVerbo: string | Block<string> = await client.blockchain.getBlock<string>(blockHash)
const blockWithoutVerbo = dataWithoutVerbo as Block<string>
expect(blockWithoutVerbo).toStrictEqual(dataWithoutVerbo)
})

it('test getblock with verbo2', async () => {
const data: string | Block<RawTx> = await client.blockchain.getBlock<RawTx>(blockHash, 2)
const block = data as Block<RawTx>
// NOTE(canonbrother): The only diff between verbo1 and verbo2 is "tx" format
expect(block.tx.length).toStrictEqual(1)
expect(block.tx[0].vin[0].coinbase).toStrictEqual('5100')
expect(block.tx[0].vin[0].sequence).toStrictEqual(4294967295)
expect(block.tx[0].vout[0].value).toStrictEqual(38)
expect(block.tx[0].vout[0].n).toStrictEqual(0)
expect(block.tx[0].vout[0].scriptPubKey.asm).toStrictEqual('OP_DUP OP_HASH160 b36814fd26190b321aa985809293a41273cfe15e OP_EQUALVERIFY OP_CHECKSIG')
expect(block.tx[0].vout[0].scriptPubKey).toHaveProperty('hex')
expect(block.tx[0].vout[0].scriptPubKey.reqSigs).toStrictEqual(1)
expect(block.tx[0].vout[0].scriptPubKey.type).toStrictEqual('pubkeyhash')
expect(block.tx[0].vout[0].scriptPubKey.addresses[0]).toStrictEqual('mwsZw8nF7pKxWH8eoKL9tPxTpaFkz7QeLU')
})
})

describe('getBlockHash', () => {
it('should getBlockHash', async () => {
const blockHash: string = await client.blockchain.getBlockHash(1)
expect(blockHash).not.toBeNull()
})
})

describe('getBlockCount', () => {
it('should getBlockCount', async () => {
const blockCount: number = await client.blockchain.getBlockCount()
expect(blockCount).toStrictEqual(2)
})
})
})
90 changes: 90 additions & 0 deletions packages/jellyfish-api-core/src/category/blockchain.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,33 @@ export class Blockchain {
async getBlockchainInfo (): Promise<BlockchainInfo> {
return await this.client.call('getblockchaininfo', [], 'number')
}

/**
* Get a hash of block in best-block-chain at height provided.
* @param height
* @return Promise<string>
*/
async getBlockHash (height: number): Promise<string> {
return await this.client.call('getblockhash', [height], 'number')
}

/**
* Get the height of the most-work fully-validated chain.
* @return Promise<number>
*/
async getBlockCount (): Promise<number> {
return await this.client.call('getblockcount', [], 'number')
}
canonbrother marked this conversation as resolved.
Show resolved Hide resolved

/**
* Get block data with particular header hash.
* @param blockHash
* @param verbosity optional, default is 1, 0 for hex encoded
* @return Promise<string | Block>
*/
async getBlock<T> (blockHash: string, verbosity?: 0 | 1 | 2): Promise<string | Block<T>> {
return await this.client.call('getblock', [blockHash, verbosity], 'number')
}
canonbrother marked this conversation as resolved.
Show resolved Hide resolved
}

/**
Expand All @@ -43,3 +70,66 @@ export interface BlockchainInfo {
}
warnings: string
}

export interface Block<T> {
hash: string
confirmations: number
strippedsize: number
size: number
weight: number
height: number
masternode: string
minter: string
mintedBlocks: number
stakeModifier: string
version: number
versionHex: string
merkleroot: string
time: number
mediantime: number
bits: string
difficulty: number
chainwork: string
tx: T[]
nTx: number
previousblockhash: string
nextblockhash: string
}

export interface RawTx {
txid: string
hash: string
version: number
size: number
vsize: number
weight: number
locktime: number
vin: Vin[]
vout: Vout[]
hex: string
}

export interface Vin {
coinbase: string
txid: string
vout: number
scriptSig: {
asm: string
hex: string
}
txinwitness: string[]
sequence: string
}

export interface Vout {
value: number
n: number
scriptPubKey: {
asm: string
hex: string
type: string
reqSigs: number
addresses: string[]
tokenId: string
}
}
56 changes: 56 additions & 0 deletions website/docs/jellyfish/api/blockchain.md
Original file line number Diff line number Diff line change
Expand Up @@ -44,3 +44,59 @@ interface BlockchainInfo {
warnings: string
}
```

## getBlock

Get block data with particular header hash.

```ts title="client.blockchain.getBlock()"
interface blockchain {
async getBlock<T> (blockHash: string, verbosity?: 0 | 1 | 2): Promise<string | Block<T>>
}

export interface Block<T> {
hash: string
confirmations: number
strippedsize: number
size: number
weight: number
height: number
masternode: string
minter: string
mintedBlocks: number
stakeModifier: string
version: number
versionHex: string
merkleroot: string
time: number
mediantime: number
bits: string
difficulty: number
chainwork: string
tx: T[]
nTx: number
previousblockhash: string
nextblockhash: string
}
```

## getBlockHash

Get a hash of block in best-block-chain at height provided.

```ts title="client.blockchain.getBlockHash()"
interface blockchain {
getBlockHash(height: number): Promise<string>
}
```

## getBlockCount

Get the height of the most-work fully-validated chain.

```ts title="client.blockchain.getBlockCount()"
interface blockchain {
getBlockCount (): Promise<number>
}
```