Skip to content

Commit

Permalink
/blocks & /transactions improvement (#385)
Browse files Browse the repository at this point in the history
* Fixed tx ordering, added block reward and vout total

* Tests passing

* Refactor tests and rename vout

* PR feedback,. simplified db structure, fixed test

* Fix api test

* Correct test

* Fix test

* Apply suggestions from code review

Co-authored-by: Fuxing Loh <[email protected]>
  • Loading branch information
monstrobishi and fuxingloh authored Sep 26, 2021
1 parent e75cc5c commit 83c1fd9
Show file tree
Hide file tree
Showing 9 changed files with 60 additions and 22 deletions.
29 changes: 27 additions & 2 deletions apps/whale-api/src/module.api/block.controller.e2e.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,12 @@ import { BlockController, parseHeight } from '@src/module.api/block.controller'
import { MasterNodeRegTestContainer } from '@defichain/testcontainers'
import { NestFastifyApplication } from '@nestjs/platform-fastify'
import { createTestingApp, stopTestingApp, waitForIndexedHeight } from '@src/e2e.module'
import { JsonRpcClient } from '@defichain/jellyfish-api-jsonrpc'

const container = new MasterNodeRegTestContainer()
let app: NestFastifyApplication
let controller: BlockController
let client: JsonRpcClient

beforeAll(async () => {
await container.start()
Expand All @@ -15,6 +17,15 @@ beforeAll(async () => {

await waitForIndexedHeight(app, 100)
controller = app.get(BlockController)
client = new JsonRpcClient(await container.getCachedRpcUrl())

const address = await container.getNewAddress()
for (let i = 0; i < 4; i += 1) {
await container.call('sendtoaddress', [address, 0.1])
}

await container.generate(3)
await waitForIndexedHeight(app, 103)
})

afterAll(async () => {
Expand Down Expand Up @@ -84,14 +95,14 @@ describe('list', () => {
describe('getTransactions', () => {
it('should get transactions from a block by hash', async () => {
const blockHash = await container.call('getblockhash', [100])
const paginatedTransactions = await controller.getTransactions(blockHash, { size: 30, next: '10' })
const paginatedTransactions = await controller.getTransactions(blockHash, { size: 30 })

expect(paginatedTransactions.data.length).toBeGreaterThanOrEqual(1)
expect(paginatedTransactions.data[0].block.height).toStrictEqual(100)
})

it('getTransactions should not get transactions by height', async () => {
const paginatedTransactions = await controller.getTransactions('0', { size: 30, next: '10' })
const paginatedTransactions = await controller.getTransactions('0', { size: 30 })

expect(paginatedTransactions.data.length).toStrictEqual(0)
})
Expand All @@ -107,6 +118,20 @@ describe('getTransactions', () => {

expect(paginatedTransactions.data.length).toStrictEqual(0)
})

it('should list transactions in the right order', async () => {
const blockHash = await container.call('getblockhash', [103])
const paginatedTransactions = await controller.getTransactions(blockHash, { size: 30 })

expect(paginatedTransactions.data.length).toBeGreaterThanOrEqual(4)
expect(paginatedTransactions.data[0].block.height).toStrictEqual(103)

const rpcBlock = await client.blockchain.getBlock(blockHash, 2)
expect(paginatedTransactions.data[0].hash).toStrictEqual(rpcBlock.tx[0].hash)
expect(paginatedTransactions.data[1].hash).toStrictEqual(rpcBlock.tx[1].hash)
expect(paginatedTransactions.data[2].hash).toStrictEqual(rpcBlock.tx[2].hash)
expect(paginatedTransactions.data[3].hash).toStrictEqual(rpcBlock.tx[3].hash)
})
})

describe('parseHeight', () => {
Expand Down
6 changes: 3 additions & 3 deletions apps/whale-api/src/module.api/block.controller.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { Controller, Get, Param, Query } from '@nestjs/common'
import { Block, BlockMapper } from '@src/module.model/block'
import { BlockMapper, Block } from '@src/module.model/block'
import { Transaction, TransactionMapper } from '@src/module.model/transaction'
import { ApiPagedResponse } from '@src/module.api/_core/api.paged.response'
import { PaginationQuery } from '@src/module.api/_core/api.query'
Expand Down Expand Up @@ -51,9 +51,9 @@ export class BlockController {
return ApiPagedResponse.empty()
}

const transactions = await this.transactionMapper.queryByBlockHash(hash, query.size, query.next)
const transactions = await this.transactionMapper.queryByBlockHash(hash, query.size, parseHeight(query.next))
return ApiPagedResponse.of(transactions, query.size, transaction => {
return transaction.id
return transaction.order.toString()
})
}
}
4 changes: 3 additions & 1 deletion apps/whale-api/src/module.api/transaction.controller.e2e.ts
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@ describe('get', () => {
const transaction = await controller.get(txid)
expect(transaction).toStrictEqual({
id: txid,
order: expect.any(Number),
block: {
hash: expect.any(String),
height: expect.any(Number),
Expand All @@ -73,7 +74,8 @@ describe('get', () => {
weight: expect.any(Number),
lockTime: expect.any(Number),
vinCount: expect.any(Number),
voutCount: expect.any(Number)
voutCount: expect.any(Number),
totalVoutValue: expect.any(String)
})
})

Expand Down
3 changes: 1 addition & 2 deletions apps/whale-api/src/module.api/transaction.controller.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import { Controller, Get, NotFoundException, Param, Query } from '@nestjs/common'
import { TransactionMapper } from '@src/module.model/transaction'
import { Transaction } from '@whale-api-client/api/transactions'
import { TransactionMapper, Transaction } from '@src/module.model/transaction'
import { PaginationQuery } from '@src/module.api/_core/api.query'
import { ApiPagedResponse } from '@src/module.api/_core/api.paged.response'
import { TransactionVin, TransactionVinMapper } from '@src/module.model/transaction.vin'
Expand Down
8 changes: 5 additions & 3 deletions apps/whale-api/src/module.indexer/model/block.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,14 +9,15 @@ export class BlockIndexer extends Indexer {
}

async index (block: RawBlock): Promise<void> {
await this.mapper.put(this.map(block))
const reward = block.tx[0].vout[0].value.toFixed(8)
await this.mapper.put(this.map(block, reward))
}

async invalidate (block: RawBlock): Promise<void> {
await this.mapper.delete(block.hash)
}

map (block: RawBlock): Block {
map (block: RawBlock, reward: string): Block {
return {
id: block.hash,
hash: block.hash,
Expand All @@ -34,7 +35,8 @@ export class BlockIndexer extends Indexer {
merkleroot: block.merkleroot,
size: block.size,
sizeStripped: block.strippedsize,
weight: block.weight
weight: block.weight,
reward: reward
}
}
}
14 changes: 10 additions & 4 deletions apps/whale-api/src/module.indexer/model/transaction.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { Injectable } from '@nestjs/common'
import { defid, Indexer, RawBlock } from '@src/module.indexer/model/_abstract'
import { Transaction, TransactionMapper } from '@src/module.model/transaction'
import BigNumber from 'bignumber.js'

@Injectable()
export class TransactionIndexer extends Indexer {
Expand All @@ -9,8 +10,11 @@ export class TransactionIndexer extends Indexer {
}

async index (block: RawBlock): Promise<void> {
for (const txn of block.tx) {
await this.mapper.put(this.map(block, txn))
for (const [order, txn] of block.tx.entries()) {
const totalVOut = txn.vout.reduce<BigNumber>((prev, vout) => {
return prev.plus(vout.value)
}, new BigNumber(0))
await this.mapper.put(this.map(block, txn, order, totalVOut.toFixed(8)))
}
}

Expand All @@ -20,9 +24,10 @@ export class TransactionIndexer extends Indexer {
}
}

map (block: RawBlock, txn: defid.Transaction): Transaction {
map (block: RawBlock, txn: defid.Transaction, order: number, totalVOut: string): Transaction {
return {
id: txn.txid,
order: order,
block: {
hash: block.hash,
height: block.height,
Expand All @@ -37,7 +42,8 @@ export class TransactionIndexer extends Indexer {
weight: txn.weight,
lockTime: txn.locktime,
vinCount: txn.vin.length,
voutCount: txn.vout.length
voutCount: txn.vout.length,
totalVoutValue: totalVOut
}
}
}
3 changes: 2 additions & 1 deletion apps/whale-api/src/module.model/block.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,8 @@ beforeEach(async () => {
time: 0,
transactionCount: 0,
version: 0,
weight: 0
weight: 0,
reward: ''
})
}

Expand Down
1 change: 1 addition & 0 deletions apps/whale-api/src/module.model/block.ts
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@ export interface Block extends Model {
masternode: string
minter: string
minterBlockCount: number
reward: string // ------------| reward string as decimal

stakeModifier: string
merkleroot: string
Expand Down
14 changes: 8 additions & 6 deletions apps/whale-api/src/module.model/transaction.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,15 +5,15 @@ import { Database, SortOrder } from '@src/module.database/database'
const TransactionMapping: ModelMapping<Transaction> = {
type: 'transaction',
index: {
block_txid: {
name: 'transaction_block_txid',
block_order: {
name: 'transaction_block_order',
partition: {
type: 'string',
key: (b: Transaction) => b.block.hash
},
sort: {
type: 'string',
key: (b: Transaction) => b.txid
type: 'number',
key: (b: Transaction) => b.order
}
}
}
Expand All @@ -24,8 +24,8 @@ export class TransactionMapper {
public constructor (protected readonly database: Database) {
}

async queryByBlockHash (hash: string, limit: number, gt?: string): Promise<Transaction[]> {
return await this.database.query(TransactionMapping.index.block_txid, {
async queryByBlockHash (hash: string, limit: number, gt?: number): Promise<Transaction[]> {
return await this.database.query(TransactionMapping.index.block_order, {
partitionKey: hash,
limit: limit,
order: SortOrder.ASC,
Expand All @@ -51,6 +51,7 @@ export class TransactionMapper {
*/
export interface Transaction extends Model {
id: string // ----------------| unique id of the transaction, same as the txid
order: number // --------------| tx order

block: {
hash: string
Expand All @@ -66,6 +67,7 @@ export interface Transaction extends Model {
size: number
vSize: number
weight: number
totalVoutValue: string

lockTime: number

Expand Down

0 comments on commit 83c1fd9

Please sign in to comment.