Skip to content

Commit

Permalink
Experiment with internalizing QHeap dependency (#3451)
Browse files Browse the repository at this point in the history
* Internalize QHeap implementation

* Move over to using lightly modified src of qheap dependency

* Fix typings

* Fix linting issues

* Move qheap to ext folder

* Add index file

* Fix linting issues

* Remove qheap dependency
  • Loading branch information
scorbajio authored Jun 14, 2024
1 parent fe76552 commit 7debfaf
Show file tree
Hide file tree
Showing 6 changed files with 224 additions and 14 deletions.
9 changes: 0 additions & 9 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 0 additions & 1 deletion packages/client/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -87,7 +87,6 @@
"level": "^8.0.0",
"memory-level": "^1.0.0",
"prom-client": "^15.1.0",
"qheap": "^1.4.0",
"winston": "^3.3.3",
"winston-daily-rotate-file": "^4.5.5",
"yargs": "^17.7.1"
Expand Down
3 changes: 3 additions & 0 deletions packages/client/src/ext/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
'use strict'

export * from './qheap.js'
216 changes: 216 additions & 0 deletions packages/client/src/ext/qheap.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,216 @@
/**
* nodejs heap, classic array implementation
*
* Items are stored in a balanced binary tree packed into an array where
* node is at [i], left child is at [2*i], right at [2*i+1]. Root is at [1].
*
* Copyright (C) 2014-2021 Andras Radics
* Licensed under the Apache License, Version 2.0
*/

/**
* QHeap types.
* @types/qheap does not exist, so we define it here.
* https://www.npmjs.com/package/qheap
*/
export type QHeapOptions = {
comparBefore?(a: any, b: any): boolean
compar?(a: any, b: any): number
freeSpace?: number
size?: number
}
export type QHeap<T> = {
// constructor(opts?: QHeapOptions)
insert(item: T): void
push(item: T): void
enqueue(item: T): void
remove(): T | undefined
shift(): T | undefined
dequeue(): T | undefined
peek(): T | undefined
length: number
gc(opts: { minLength: number; maxLength: number }): void
}

export class Heap {
private _list!: any[]
private _isBefore!: (a: any, b: any) => boolean
private _sortBefore!: (a: any, b: any) => number
private _freeSpace!: ((list: any[], len: number) => void) | false
public options!: QHeapOptions
public length!: number

constructor(opts?: QHeapOptions | Function) {
if (!(this instanceof Heap)) return new Heap(opts as QHeapOptions)

if (typeof opts === 'function') opts = { compar: opts as any }

// copy out known options to not bind to caller object
this.options = !opts
? ({} as QHeapOptions)
: {
compar: (opts as QHeapOptions).compar,
comparBefore: (opts as QHeapOptions).comparBefore,
freeSpace: (opts as QHeapOptions).freeSpace,
size: (opts as QHeapOptions).size,
}
opts = this.options

const self = this

this._isBefore = opts.compar
? function (a: any, b: any) {
// @ts-ignore
return opts!.compar!(a, b) < 0
}
: opts.comparBefore ??
function (a: any, b: any): boolean {
return a < b
}

this._sortBefore =
opts.compar ??
function (a: any, b: any) {
return self._isBefore(a, b) ? -1 : 1
}
this._freeSpace = opts.freeSpace === undefined ? this._trimArraySize : false

this._list = new Array(opts.size ?? 20)
this.length = 0
}

/*
* insert new item at end, and bubble up
*/
public insert(item: any): any {
const idx = ++this.length
return this._bubbleup(idx, item)
}
public _bubbleup(idx: number, item: any): void {
const list = this._list
list[idx] = item
if (idx <= 1) return
do {
const pp = idx >>> 1
if (this._isBefore(item, list[pp])) list[idx] = list[pp]
else break
idx = pp
} while (idx > 1)
list[idx] = item
}
public append = this.insert
public push = this.insert
public unshift = this.insert
public enqueue = this.insert

public peek(): any {
return this.length > 0 ? this._list[1] : undefined
}

public size(): number {
return this.length
}

/*
* return the root, and bubble down last item from top root position
* when bubbling down, r: root idx, c: child sub-tree root idx, cv: child root value
* Note that the child at (c == this.length) does not have to be tested in the loop,
* since its value is the one being bubbled down, so can loop `while (c < len)`.
*/
public remove(): any {
const len = this.length
if (len < 1) return undefined
return this._bubbledown(1, len)
}
public _bubbledown(r: number, len: number): any {
const list = this._list,
ret = list[r],
itm = list[len]
let c
const _isBefore = this._isBefore

while ((c = r << 1) < len) {
let cv = list[c]
const cv1 = list[c + 1]
if (_isBefore(cv1, cv)) {
c++
cv = cv1
}
if (!_isBefore(cv, itm)) break
list[r] = cv
r = c
}
list[r] = itm
list[len] = 0
this.length = --len
if (this._freeSpace !== false && this._freeSpace !== undefined)
this._freeSpace(this._list, this.length)

return ret
}

public shift = this.remove
public pop = this.remove
public dequeue = this.remove

// builder, not initializer: appends items, not replaces
// FIXME: more useful to re-initialize from array
public fromArray(array: any[], base?: number, bound?: number): void {
base = (base ?? 0) || 0
bound = (bound ?? 0) || array.length
for (let i = base; i < bound; i++) this.insert(array[i])
}

// FIXME: more useful to return sorted values
public toArray(limit?: number): any[] {
limit = typeof limit === 'number' ? limit + 1 : this.length + 1
return this._list.slice(1, limit)
}

// sort the contents of the storage array
public sort(): void {
if (this.length < 3) return
this._list.splice(this.length + 1)
this._list[0] = this._list[1]
this._list.sort(this._sortBefore)
this._list[0] = 0
}

// Free unused storage slots in the _list.
public gc(options?: { minLength?: number; minFull?: number }): void {
if (!options) options = {}

const minListLength = (options.minLength ?? 0) || 0
const minListFull = (options.minFull ?? 0) || 1.0

if (this._list.length >= minListLength && this.length < this._list.length * minListFull) {
this._list.splice(this.length + 1, this._list.length)
}
}

public _trimArraySize(list: any[], len: number): void {
if (len > 10000 && list.length > 4 * len) {
list.splice(len + 1, list.length)
}
}

public _check(): boolean {
const _compar = this._sortBefore

let i,
p,
fail = 0
for (i = this.length; i > 1; i--) {
// error if parent should go after child, but not if don`t care
p = i >>> 1
// swapping the values must change their ordering, otherwise the
// comparison is a tie. (Ie, consider the ordering func (a <= b)
// that for some values reports both that a < b and b < a.)
if (_compar(this._list[p], this._list[i]) > 0 && _compar(this._list[i], this._list[p]) < 0) {
fail = i
}
}
if (fail) console.log('failed at', fail >>> 1, fail)
return !fail
}
}
5 changes: 3 additions & 2 deletions packages/client/src/service/txpool.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,9 +16,11 @@ import {
equalsBytes,
hexToBytes,
} from '@ethereumjs/util'
import Heap from 'qheap'

import { Heap } from '../ext/qheap.js'

import type { Config } from '../config.js'
import type { QHeap } from '../ext/qheap.js'
import type { Peer } from '../net/peer/peer.js'
import type { PeerPool } from '../net/peerpool.js'
import type { FullEthereumService } from './fullethereumservice.js'
Expand All @@ -29,7 +31,6 @@ import type {
TypedTransaction,
} from '@ethereumjs/tx'
import type { VM } from '@ethereumjs/vm'
import type QHeap from 'qheap'

// Configuration constants
const MIN_GAS_PRICE_BUMP_PERCENT = 10
Expand Down
4 changes: 2 additions & 2 deletions packages/client/src/sync/fetcher/fetcher.ts
Original file line number Diff line number Diff line change
@@ -1,16 +1,16 @@
import debugDefault from 'debug'
import Heap from 'qheap'
import { Readable, Writable } from 'stream'

import { Heap } from '../../ext/qheap.js'
import { Event } from '../../types.js'

import type { Config } from '../../config.js'
import type { QHeap } from '../../ext/qheap.js'
import type { Peer } from '../../net/peer/index.js'
import type { PeerPool } from '../../net/peerpool.js'
import type { JobTask as BlockFetcherJobTask } from './blockfetcherbase.js'
import type { Job } from './types.js'
import type { Debugger } from 'debug'
import type QHeap from 'qheap'

const { debug: createDebugLogger } = debugDefault

Expand Down

0 comments on commit 7debfaf

Please sign in to comment.