@xyo-network/chain-services 1.3.17 → 1.3.18

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -8,6 +8,7 @@ import { analyzeChain, BalanceAnalyzer, isChainSummaryBalances } from "@xyo-netw
8
8
  import { toPositiveBigInt } from "@xyo-network/chain-protocol";
9
9
 
10
10
  // src/BaseService.ts
11
+ import { trace } from "@opentelemetry/api";
11
12
  import { span, spanAsync } from "@xyo-network/chain-utils";
12
13
  import { BaseEmitter } from "@xyo-network/module-event-emitter";
13
14
  import { Mutex } from "async-mutex";
@@ -26,7 +27,7 @@ var BaseService = class extends BaseEmitter {
26
27
  return this.constructor.name;
27
28
  }
28
29
  get tracer() {
29
- return this.params.traceProvider?.getTracer(this.name);
30
+ return this.params.traceProvider?.getTracer(this.name) || trace.getTracer(this.name);
30
31
  }
31
32
  static async create(params) {
32
33
  const result = new this(params);
@@ -46,8 +47,8 @@ var BaseService = class extends BaseEmitter {
46
47
  span(name, fn) {
47
48
  return span(`${this.name}:${name}`, fn, this.tracer);
48
49
  }
49
- async spanAsync(name, fn, timeout) {
50
- return await spanAsync(`${this.name}:${name}`, fn, this.tracer, timeout);
50
+ async spanAsync(name, fn) {
51
+ return await spanAsync(`${this.name}:${name}`, fn, this.tracer);
51
52
  }
52
53
  };
53
54
  var BaseAccountableService = class extends BaseService {
@@ -651,11 +652,12 @@ XyoElectionService = _ts_decorate6([
651
652
  ], XyoElectionService);
652
653
 
653
654
  // src/PendingTransactions/PendingTransactions.ts
655
+ import { ValueType } from "@opentelemetry/api";
654
656
  import { filterAs, filterAsync } from "@xylabs/array";
655
657
  import { assertEx as assertEx7 } from "@xylabs/assert";
656
658
  import { exists } from "@xylabs/exists";
659
+ import { forget } from "@xylabs/forget";
657
660
  import { MemoryArchivist } from "@xyo-network/archivist-memory";
658
- import { hydrateBlock } from "@xyo-network/chain-protocol";
659
661
  import { validateTransaction } from "@xyo-network/chain-validation";
660
662
  import { asOptionalTransactionBoundWitnessWithStorageMeta } from "@xyo-network/xl1-protocol";
661
663
  import { flattenHydratedTransactions, tryHydrateTransaction } from "@xyo-network/xl1-protocol-sdk";
@@ -689,12 +691,25 @@ var XyoPendingTransactionsService = class _XyoPendingTransactionsService extends
689
691
  */
690
692
  RemoveFinalizedTransactions: 1
691
693
  };
692
- // used to remember the last block's transactions that was removed due to it being in the chain
693
- _cleanedToBlock;
694
- // a local archivist that has only the transactions in it (used as queue)
695
- _pendingTransactionsLocalArchivist;
696
- _pendingTransactionsLocalArchivistCursor;
697
- _updatePendingTransactionsLocalArchivistMutex = new Mutex2();
694
+ /**
695
+ * A mutex to ensure that the counting the number of pending transactions is
696
+ * not called concurrently
697
+ */
698
+ _countPendingTransactionsMutex = new Mutex2();
699
+ /**
700
+ * A local Archivist optimized for fast retrieval that stores only validated
701
+ * pending transactions
702
+ */
703
+ _curatedPendingTransactionsArchivist;
704
+ /**
705
+ * The last count of total pending transactions
706
+ */
707
+ _pendingTransactionsCount = 0;
708
+ /**
709
+ * A mutex to ensure that the curated pending transactions archivist is
710
+ * updated in a thread-safe manner
711
+ */
712
+ _updateCuratedPendingTransactionsArchivistMutex = new Mutex2();
698
713
  get chainArchivist() {
699
714
  return assertEx7(this.params.chainArchivist, () => "No completed blocks with data archivist");
700
715
  }
@@ -704,46 +719,41 @@ var XyoPendingTransactionsService = class _XyoPendingTransactionsService extends
704
719
  get pendingTransactionsArchivist() {
705
720
  return assertEx7(this.params.pendingTransactionsArchivist, () => "No pending transactions archivist");
706
721
  }
722
+ get pendingTransactionsCount() {
723
+ forget(this.countPendingTransactions());
724
+ return this._pendingTransactionsCount;
725
+ }
707
726
  get pendingTransactionsLocalArchivist() {
708
- return assertEx7(this._pendingTransactionsLocalArchivist, () => "No pending transactions curate archivist");
727
+ return assertEx7(this._curatedPendingTransactionsArchivist, () => "No pending transactions curated archivist");
709
728
  }
710
729
  get rejectedTransactionsArchivist() {
711
730
  return assertEx7(this.params.rejectedTransactionsArchivist, () => "No rejected transactions archivist");
712
731
  }
713
732
  async createHandler() {
714
733
  await super.createHandler();
715
- this._pendingTransactionsLocalArchivist = await MemoryArchivist.create({
734
+ this._curatedPendingTransactionsArchivist = await MemoryArchivist.create({
716
735
  account: "random"
717
736
  });
718
737
  this.pendingTransactionsArchivist.on("inserted", async ({ payloads }) => {
719
- await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {
720
- const unprocessedTransactions = await this.filterAlreadyFinalizedTransactions(payloads);
721
- const hydratedUnprocessedTransactions = (await Promise.all(unprocessedTransactions.map((tx) => {
722
- return tryHydrateTransaction(this.pendingTransactionsArchivist, tx._hash);
723
- }))).filter(exists);
724
- const validTransactions = await filterAsync(hydratedUnprocessedTransactions, async (tx) => {
725
- const errors = await validateTransaction(tx, this.chainIdentification.id);
726
- return errors.length > 0 ? false : true;
727
- });
728
- await this.pendingTransactionsLocalArchivist.insert(flattenHydratedTransactions(validTransactions));
729
- }, _XyoPendingTransactionsService.MutexPriority.InsertNewTransactions);
738
+ await this.insertNewTransactions(payloads);
730
739
  });
731
740
  this.chainArchivist.on("inserted", async ({ payloads }) => {
732
- await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {
733
- const transactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta);
734
- await this.pendingTransactionsLocalArchivist.delete(transactions.map((tx) => tx._hash));
735
- }, _XyoPendingTransactionsService.MutexPriority.RemoveFinalizedTransactions);
741
+ await this.removeFinalizedTransactions(payloads);
736
742
  });
737
743
  this.rejectedTransactionsArchivist.on("inserted", async ({ payloads }) => {
738
- await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {
739
- const transactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta);
740
- await this.pendingTransactionsLocalArchivist.delete(transactions.map((tx) => tx._hash));
741
- }, _XyoPendingTransactionsService.MutexPriority.RemoveRejectedTransactions);
744
+ await this.removeRejectedTransactions(payloads);
745
+ });
746
+ const pendingTransactionsGauge = this.meter?.createObservableGauge("xyo_pending_transactions_count", {
747
+ description: "The current number of pending transactions",
748
+ valueType: ValueType.INT
749
+ });
750
+ pendingTransactionsGauge?.addCallback((observer) => {
751
+ observer.observe(this.pendingTransactionsCount);
742
752
  });
743
753
  }
744
- async getPendingTransactions(head, limit, timeout = 1e6) {
754
+ async getPendingTransactions(head, limit) {
745
755
  return await this.spanAsync("getPendingTransactions", async () => {
746
- return await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {
756
+ return await this._updateCuratedPendingTransactionsArchivistMutex.runExclusive(async () => {
747
757
  const foundPendingTransactions = [];
748
758
  let cursor;
749
759
  while (foundPendingTransactions.length < limit) {
@@ -760,7 +770,15 @@ var XyoPendingTransactionsService = class _XyoPendingTransactionsService extends
760
770
  const hydratedTransactions = await Promise.all(foundPendingTransactions.map((tx) => tryHydrateTransaction(this.pendingTransactionsLocalArchivist, tx._hash)));
761
771
  return hydratedTransactions.filter(exists);
762
772
  }, _XyoPendingTransactionsService.MutexPriority.ReadTransactions);
763
- }, timeout);
773
+ });
774
+ }
775
+ async countPendingTransactions() {
776
+ if (this._countPendingTransactionsMutex.isLocked()) return;
777
+ await this._countPendingTransactionsMutex.runExclusive(async () => {
778
+ const payloads = await this._curatedPendingTransactionsArchivist?.all() ?? [];
779
+ const pendingTransactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta);
780
+ this._pendingTransactionsCount = pendingTransactions.length;
781
+ });
764
782
  }
765
783
  async filterAlreadyFinalizedTransactions(payloads) {
766
784
  const incomingTransactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta);
@@ -770,28 +788,34 @@ var XyoPendingTransactionsService = class _XyoPendingTransactionsService extends
770
788
  const nonFinalizedTransactions = incomingTransactions.filter((item) => !finalizedTransactionHashes.has(item._hash));
771
789
  return nonFinalizedTransactions;
772
790
  }
773
- async removeAlreadyIncludedTransactions(head, timeout = 1e6) {
774
- return await this.spanAsync("removeAlreadyIncludedTransactions", async () => {
775
- let headPtr = head ?? null;
776
- while (headPtr !== null) {
777
- const [block, payloads] = assertEx7(await hydrateBlock(this.chainArchivist, headPtr), () => `Failed to find head [${head}]`);
778
- if (block._hash === this._cleanedToBlock) break;
779
- await this.pendingTransactionsLocalArchivist.delete(payloads.map((payload) => asOptionalTransactionBoundWitnessWithStorageMeta(payload)?._hash).filter(exists));
780
- this._cleanedToBlock = block._hash;
781
- headPtr = block?.previous;
782
- }
783
- }, timeout);
791
+ async insertNewTransactions(payloads) {
792
+ return await this.spanAsync("InsertNewTransactions", async () => {
793
+ return await this._updateCuratedPendingTransactionsArchivistMutex.runExclusive(async () => {
794
+ const unprocessedTransactions = await this.filterAlreadyFinalizedTransactions(payloads);
795
+ const hydratedUnprocessedTransactions = (await Promise.all(unprocessedTransactions.map((tx) => {
796
+ return tryHydrateTransaction(this.pendingTransactionsArchivist, tx._hash);
797
+ }))).filter(exists);
798
+ const validTransactions = await filterAsync(hydratedUnprocessedTransactions, async (tx) => {
799
+ const errors = await validateTransaction(tx, this.chainIdentification.id);
800
+ return errors.length > 0 ? false : true;
801
+ });
802
+ await this.pendingTransactionsLocalArchivist.insert(flattenHydratedTransactions(validTransactions));
803
+ }, _XyoPendingTransactionsService.MutexPriority.InsertNewTransactions);
804
+ });
784
805
  }
785
- async updatePendingTransactionsLocalArchivist(timeout = 1e6) {
786
- return await this.spanAsync("updatePendingTransactionsLocalArchivist", async () => {
787
- if (this._updatePendingTransactionsLocalArchivistMutex.isLocked()) {
788
- console.warn("updatePendingTransactionsLocalArchivist busy - skipping");
789
- return;
790
- }
791
- return await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {
792
- await Promise.resolve();
793
- });
794
- }, timeout);
806
+ async removeFinalizedTransactions(payloads) {
807
+ return await this.removeTransactions(payloads, "RemoveFinalizedTransactions");
808
+ }
809
+ async removeRejectedTransactions(payloads) {
810
+ return await this.removeTransactions(payloads, "RemoveRejectedTransactions");
811
+ }
812
+ async removeTransactions(payloads, priority) {
813
+ return await this.spanAsync(priority, async () => {
814
+ return await this._updateCuratedPendingTransactionsArchivistMutex.runExclusive(async () => {
815
+ const transactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta);
816
+ await this.pendingTransactionsLocalArchivist.delete(transactions.map((tx) => tx._hash));
817
+ }, _XyoPendingTransactionsService.MutexPriority[priority]);
818
+ });
795
819
  }
796
820
  };
797
821
  XyoPendingTransactionsService = _ts_decorate7([
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/AccountBalance/XyoChainAccountBalanceService.ts","../../src/BaseService.ts","../../src/BaseEthProvider.ts","../../src/BlockProducer/XyoBlockProducer.ts","../../src/BlockProducer/generateTransactionTransfers.ts","../../src/BlockReward/EvmBlockRewardService.ts","../../src/BlockReward/XyoBlockRewardService.ts","../../src/ChainValidator/XyoValidator.ts","../../src/Election/XyoElectionService.ts","../../src/PendingTransactions/PendingTransactions.ts","../../src/StakeIntent/XyoStakeIntentService.ts","../../src/Staker/Evm/Evm.ts","../../src/XyoChainBlockNumberIterator.ts"],"sourcesContent":["import { assertEx } from '@xylabs/assert'\nimport {\n Address, Hash, Hex,\n toHex,\n} from '@xylabs/hex'\nimport {\n analyzeChain, BalanceAnalyzer, isChainSummaryBalances,\n} from '@xyo-network/chain-analyze'\nimport { SignedBigInt, toPositiveBigInt } from '@xyo-network/chain-protocol'\nimport {\n AccountBalanceService, BaseServiceParams,\n ChainServiceCollection,\n} from '@xyo-network/xl1-protocol'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport type ChainAccountBalanceServiceParams = BaseServiceParams<Partial<Pick<ChainServiceCollection, 'chainArchivist'>>>\n\n@creatableService()\nexport class XyoChainAccountBalanceService extends BaseService<ChainAccountBalanceServiceParams> implements AccountBalanceService {\n private _balances: Record<Address, SignedBigInt> = {}\n private _previousHead: Hash | null = null\n\n protected get chainArchivist() {\n return assertEx(this.params.chainArchivist, () => 'chainArchivist is required')\n }\n\n getBalance(address: Address): Hex {\n return toPositiveBigInt(this._balances[address] ?? { positive: toHex(0n) }).positive\n }\n\n getBalances(): Record<Address, Hex> {\n const result: Record<Address, Hex> = {}\n const entries = Object.entries(this._balances) as [Address, SignedBigInt][]\n for (const [address, balance] of entries) {\n result[address] = toPositiveBigInt(balance).positive\n }\n return result\n }\n\n async sync(head: Hash): Promise<void> {\n const analysis = await analyzeChain(\n this.chainArchivist,\n [new BalanceAnalyzer()],\n head,\n this._previousHead,\n -1n,\n this.tracer,\n )\n this._balances = assertEx(analysis.find(isChainSummaryBalances)?.balances, () => 'Failed to sync balances')\n }\n}\n","import type { Promisable } from '@xylabs/promise'\nimport { span, spanAsync } from '@xyo-network/chain-utils'\nimport { BaseEmitter } from '@xyo-network/module-event-emitter'\nimport type {\n BaseAccountableServiceParams, BaseServiceParams, Service,\n} from '@xyo-network/xl1-protocol'\nimport { Mutex } from 'async-mutex'\n\ndeclare global {\n var xyoServiceSingletons: Record<string, unknown>\n}\n\nexport class BaseService<TParams extends BaseServiceParams = BaseServiceParams> extends BaseEmitter<TParams> {\n private static singletonInitMutex = new Mutex()\n\n static get singletons() {\n return globalThis['xyoServiceSingletons'] ?? (globalThis['xyoServiceSingletons'] = {})\n }\n\n get meter() {\n return this.params.meterProvider?.getMeter(this.name)\n }\n\n get name() {\n return this.constructor.name\n }\n\n get tracer() {\n return this.params.traceProvider?.getTracer(this.name)\n }\n\n static async create<TService extends BaseService<TParams>, TParams extends BaseServiceParams = TService['params']>(\n this: CreatableService<TService>,\n params: TParams,\n ): Promise<TService> {\n const result = new this(params)\n if (result.name === 'BaseService') throw new Error('Cannot create BaseService')\n await result.createHandler()\n return result\n }\n\n static initSingleton<TService extends BaseService<TParams>, TParams extends BaseServiceParams>(params: TParams) {\n if (this.singletons[this.name]) throw new Error(`Singleton already initialized for ${this.name}`)\n return this.singletonInitMutex.runExclusive(async () => {\n return await this.create(params) as TService\n })\n }\n\n createHandler(): Promisable<void> {\n return\n }\n\n span<T>(name: string, fn: () => T): T {\n return span(`${this.name}:${name}`, fn, this.tracer)\n }\n\n async spanAsync<T>(name: string, fn: () => Promise<T>, timeout?: number): Promise<T> {\n return await spanAsync(`${this.name}:${name}`, fn, this.tracer, timeout)\n }\n}\n\nexport abstract class BaseAccountableService<\n TParams extends BaseAccountableServiceParams = BaseAccountableServiceParams,\n> extends BaseService<TParams> {}\n\nexport interface CreatableService<T extends BaseService = BaseService> extends Service {\n new (params: T['params']): T\n create<T extends BaseService>(this: CreatableService<T>, params: T['params']): Promisable<T>\n}\n\nexport function creatableService<T extends BaseService = BaseService>() {\n return <U extends CreatableService<T>>(constructor: U) => {\n // eslint-disable-next-line @typescript-eslint/no-unused-expressions\n constructor\n }\n}\n","import type {\n Address, BlockAPI,\n BlockNumberOrTag,\n CompileResultAPI,\n EthExecutionAPI,\n FeeHistoryResultAPI,\n Filter,\n FilterResultsAPI,\n HexString, HexString8Bytes,\n HexString32Bytes,\n HexString256Bytes,\n HexStringBytes,\n SignedTransactionInfoAPI,\n SyncingStatusAPI,\n TransactionCallAPI,\n TransactionInfoAPI,\n TransactionReceiptAPI,\n TransactionWithSenderAPI,\n Uint,\n Uint256,\n} from 'web3-types'\n\nexport class BaseEthProvider implements EthExecutionAPI {\n eth_accounts(): Address[] {\n throw new Error('Method not implemented.')\n }\n\n eth_blockNumber(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_call(_transaction: TransactionCallAPI, _blockNumber: BlockNumberOrTag): HexStringBytes {\n throw new Error('Method not implemented.')\n }\n\n eth_clearSubscriptions(_keepSyncing?: boolean): void {\n throw new Error('Method not implemented.')\n }\n\n eth_coinbase(): Address {\n throw new Error('Method not implemented.')\n }\n\n eth_compileLLL(_code: string): HexStringBytes {\n throw new Error('Method not implemented.')\n }\n\n eth_compileSerpent(_code: string): HexStringBytes {\n throw new Error('Method not implemented.')\n }\n\n eth_compileSolidity(_code: string): CompileResultAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_estimateGas(_transaction: Partial<TransactionWithSenderAPI>, _blockNumber: BlockNumberOrTag): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_feeHistory(_blockCount: Uint, _newestBlock: BlockNumberOrTag, _rewardPercentiles: number[]): FeeHistoryResultAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_gasPrice(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getBalance(_address: Address, _blockNumber: BlockNumberOrTag): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getBlockByHash(_blockHash: HexString32Bytes, _hydrated: boolean): BlockAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getBlockByNumber(_blockNumber: BlockNumberOrTag, _hydrated: boolean): BlockAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getBlockTransactionCountByHash(_blockHash: HexString32Bytes): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getBlockTransactionCountByNumber(_blockNumber: BlockNumberOrTag): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getCode(_address: Address, _blockNumber: BlockNumberOrTag): HexStringBytes {\n throw new Error('Method not implemented.')\n }\n\n eth_getCompilers(): string[] {\n throw new Error('Method not implemented.')\n }\n\n eth_getFilterChanges(_filterIdentifier: Uint): FilterResultsAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getFilterLogs(_filterIdentifier: Uint): FilterResultsAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getLogs(_filter: Filter): FilterResultsAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getStorageAt(_address: Address, _storageSlot: Uint256, _blockNumber: BlockNumberOrTag): HexStringBytes {\n throw new Error('Method not implemented.')\n }\n\n eth_getTransactionByBlockHashAndIndex(_blockHash: HexString32Bytes, _transactionIndex: Uint): TransactionInfoAPI | undefined {\n throw new Error('Method not implemented.')\n }\n\n eth_getTransactionByBlockNumberAndIndex(_blockNumber: BlockNumberOrTag, _transactionIndex: Uint): TransactionInfoAPI | undefined {\n throw new Error('Method not implemented.')\n }\n\n eth_getTransactionByHash(_transactionHash: HexString32Bytes): TransactionInfoAPI | undefined {\n throw new Error('Method not implemented.')\n }\n\n eth_getTransactionCount(_address: Address, _blockNumber: BlockNumberOrTag): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getTransactionReceipt(_transactionHash: HexString32Bytes): TransactionReceiptAPI | undefined {\n throw new Error('Method not implemented.')\n }\n\n eth_getUncleByBlockHashAndIndex(_blockHash: HexString32Bytes, _uncleIndex: Uint): BlockAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getUncleByBlockNumberAndIndex(_blockNumber: BlockNumberOrTag, _uncleIndex: Uint): BlockAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getUncleCountByBlockHash(_blockHash: HexString32Bytes): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getUncleCountByBlockNumber(_blockNumber: BlockNumberOrTag): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getWork(): [HexString32Bytes, HexString32Bytes, HexString32Bytes] {\n throw new Error('Method not implemented.')\n }\n\n eth_hashrate(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_maxPriorityFeePerGas(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_mining(): boolean {\n throw new Error('Method not implemented.')\n }\n\n eth_newBlockFilter(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_newFilter(_filter: Filter): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_newPendingTransactionFilter(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_protocolVersion(): string {\n throw new Error('Method not implemented.')\n }\n\n eth_sendRawTransaction(_transaction: HexStringBytes): HexString32Bytes {\n throw new Error('Method not implemented.')\n }\n\n eth_sendTransaction(_transaction: TransactionWithSenderAPI | Partial<TransactionWithSenderAPI>): HexString32Bytes {\n throw new Error('Method not implemented.')\n }\n\n eth_sign(_address: Address, _message: HexStringBytes): HexString256Bytes {\n throw new Error('Method not implemented.')\n }\n\n eth_signTransaction(_transaction: TransactionWithSenderAPI | Partial<TransactionWithSenderAPI>): HexStringBytes | SignedTransactionInfoAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_submitHashrate(_hashRate: HexString32Bytes, _id: HexString32Bytes): boolean {\n throw new Error('Method not implemented.')\n }\n\n eth_submitWork(_nonce: HexString8Bytes, _hash: HexString32Bytes, _digest: HexString32Bytes): boolean {\n throw new Error('Method not implemented.')\n }\n\n eth_subscribe(..._params: ['newHeads'] | ['newPendingTransactions'] | ['syncing'] | ['logs', { address?: HexString; topics?: HexString[] }]): HexString {\n throw new Error('Method not implemented.')\n }\n\n eth_syncing(): SyncingStatusAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_uninstallFilter(_filterIdentifier: Uint): boolean {\n throw new Error('Method not implemented.')\n }\n\n eth_unsubscribe(_subscriptionId: HexString): HexString {\n throw new Error('Method not implemented.')\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { Address, toHex } from '@xylabs/hex'\nimport { ArchivistInstance, ReadArchivist } from '@xyo-network/archivist-model'\nimport {\n BlockRewardDiviner, FixedPercentageBlockRewardDiviner, FixedPercentageBlockRewardDivinerConfigSchema,\n} from '@xyo-network/chain-modules'\nimport { buildBlock } from '@xyo-network/chain-protocol'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { WithStorageMeta } from '@xyo-network/payload-model'\nimport {\n AllowedBlockPayload, asBlockBoundWitness, BlockBoundWitness, BlockNumber, BlockNumberSchema, BlockProducer, ChainStakeIntent, ChainStakeIntentSchema,\n HydratedBlock, PendingTransactionsService, StakeIntentService, Transfer,\n} from '@xyo-network/xl1-protocol'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\nimport { XyoValidatorParams } from '../ChainValidator/index.ts'\nimport { generateTransactionTransfers } from './generateTransactionTransfers.ts'\n\n/**\n * The default block size for a block\n */\nexport const DEFAULT_BLOCK_SIZE = 10\n\n// /**\n// * The amount of time for which a producer will restake their intent\n// */\nexport const XYO_PRODUCER_RESTAKE_DURATION = 10_000\n\n/**\n * The number of blocks within which a producer will redeclare\n * their intent to produce blocks\n */\nexport const XYO_PRODUCER_RESTAKE_WINDOW = 500n\n\nexport interface XyoBlockProducerParams extends XyoValidatorParams {\n pendingTransactionsArchivist?: ReadArchivist\n pendingTransactionsService?: PendingTransactionsService\n rejectedTransactionsArchivist?: ArchivistInstance\n rewardAddress?: Address\n stakeIntentService?: StakeIntentService\n}\n\n@creatableService()\nexport class XyoBlockProducer extends BaseService<XyoBlockProducerParams> implements BlockProducer {\n protected _blockRewardDiviner: BlockRewardDiviner | undefined\n\n static get DefaultBlockSize(): number {\n return DEFAULT_BLOCK_SIZE\n }\n\n get address() {\n return this.account.address\n }\n\n protected get account() {\n return assertEx(this.params.account, () => 'account is required')\n }\n\n protected get chainFinalizedArchivist() {\n return assertEx(this.params.chainFinalizedArchivist, () => 'chainFinalizedArchivist is required')\n }\n\n protected get chainInfo() {\n return assertEx(this.params.chainInformation, () => 'chainInfo is required')\n }\n\n protected get electionService() {\n return assertEx(this.params.electionService, () => 'electionService is required')\n }\n\n protected get pendingTransactions() {\n return assertEx(this.params.pendingTransactions, () => 'pendingTransactions is required')\n }\n\n protected get pendingTransactionsService() {\n return assertEx(this.params.pendingTransactionsService, () => 'Missing pendingTransactionsService')\n }\n\n protected get rejectedTransactionsArchivist() {\n return assertEx(this.params.rejectedTransactionsArchivist, () => 'No rejected transactions archivist')\n }\n\n protected get rewardAddress(): Address {\n return assertEx(this.params.rewardAddress, () => 'No reward address provided')\n }\n\n protected get rewardService() {\n return assertEx(this.params.rewardService, () => 'rewardService is required')\n }\n\n protected get stakeIntentService(): StakeIntentService {\n return assertEx(this.params.stakeIntentService, () => 'No StakeIntentService provided')\n }\n\n protected get validateHydratedBlockState() {\n return assertEx(this.params.validateHydratedBlockState, () => 'validateHydratedBlockState is required')\n }\n\n async next(head: WithStorageMeta<BlockBoundWitness>): Promise<HydratedBlock | undefined> {\n // If the block is for another chain, ignore\n if (head.chain !== this.chainInfo.id) return\n const leaders = await this.electionService.getCreatorCommitteeForNextBlock(head)\n // TODO: Should we propose block if creator committee is empty?\n // TODO: Handle the case where we're not the 1st leader but they're not responding\n // at a higher level than here as that's a network issue\n if (leaders?.[0] !== this.address) return\n return this.proposeNextValidBlock(head)\n }\n\n protected async getBlockRewardTransfer(block: number): Promise<Transfer | undefined> {\n if (!this._blockRewardDiviner) {\n // TODO: Adjust to allow for genesis block reward vs. normal block reward\n this._blockRewardDiviner = await FixedPercentageBlockRewardDiviner.create({\n account: 'random',\n blockRewardService: this.rewardService,\n config: {\n rewardAddress: this.rewardAddress,\n rewardPercentageRatio: 0,\n schema: FixedPercentageBlockRewardDivinerConfigSchema,\n },\n })\n }\n const blockHex = assertEx(toHex(block), () => 'Failed to convert block to hex')\n const blockId = new PayloadBuilder<BlockNumber>({ schema: BlockNumberSchema }).fields({ block: blockHex }).build()\n const rewards = await this._blockRewardDiviner.divine([blockId])\n const [reward] = rewards\n return reward\n }\n\n /**\n * Handles the producer redeclaration logic\n * @param head The current head block\n * @returns\n */\n protected async getProducerRedeclaration(head: WithStorageMeta<BlockBoundWitness>): Promise<ChainStakeIntent | undefined> {\n const redeclareIntent = process.env.XYO_PRODUCER_REDECLARE_INTENT\n if (!redeclareIntent) return\n // Decide if we need to redeclare\n const ranges = await this.stakeIntentService.getDeclaredCandidateRanges(this.address, 'producer')\n // TODO: This doesn't handle the case where the producer had declared a range for the future\n // but we're in a range that's not the future\n // Sort in ascending order based on ending range to get range with highest ending block\n const lastRange = ranges.toSorted((a, b) => a[1] > b[1] ? 1 : -1).at(-1)\n if (!lastRange) return\n const [, currentDeclarationEnd] = lastRange\n const currentBlock = head.block\n const timeToProducerExpiration = currentDeclarationEnd - currentBlock\n if (timeToProducerExpiration > XYO_PRODUCER_RESTAKE_WINDOW) return\n // Create redeclaration\n const intent = new PayloadBuilder<ChainStakeIntent>({ schema: ChainStakeIntentSchema }).fields({\n from: this.address,\n intent: 'producer',\n nbf: currentBlock,\n exp: currentBlock + XYO_PRODUCER_RESTAKE_DURATION,\n }).build()\n return intent\n }\n\n protected async proposeNextValidBlock(head: WithStorageMeta<BlockBoundWitness>): Promise<HydratedBlock | undefined> {\n return await this.spanAsync('proposeNextValidBlock', async () => {\n // Calculate the next block components\n const { block: previousBlock } = assertEx(asBlockBoundWitness(head), () => 'Invalid head block')\n const nextBlock = previousBlock + 1\n const nextBlockTransactions = await this.pendingTransactionsService.getPendingTransactions(head._hash, XyoBlockProducer.DefaultBlockSize)\n\n const blockPayloads: AllowedBlockPayload[] = []\n\n // Calculate the optional producer redeclaration and add it if necessary\n const producerRedeclarationPayload = await this.getProducerRedeclaration(head)\n if (producerRedeclarationPayload) blockPayloads.push(producerRedeclarationPayload)\n\n // If there are no transactions and no payloads, we don't need to create a block\n if (blockPayloads.length === 0 && nextBlockTransactions.length === 0) return\n\n // Calculate the optional block reward transfer and add if necessary\n const rewardTransferPayload = await this.getBlockRewardTransfer(nextBlock)\n if (rewardTransferPayload) blockPayloads.push(rewardTransferPayload)\n\n const transactionTransfers = await generateTransactionTransfers(this.address, nextBlockTransactions)\n blockPayloads.push(...transactionTransfers)\n\n // Build the block\n const block = await buildBlock(head, nextBlockTransactions, blockPayloads, [this.account])\n console.log('buildBlock', block)\n const errors = await this.validateHydratedBlockState(block, this.chainInfo.id, this.chainFinalizedArchivist)\n if (errors.length > 0) {\n this.logger?.error('Validation of produced block failed', errors)\n const rejectedTransactions = block[1]\n await this.rejectedTransactionsArchivist.insert(rejectedTransactions)\n } else {\n return block\n }\n })\n }\n}\n","import { type Address, hexFromBigInt } from '@xylabs/hex'\nimport { HydratedTransactionWrapper } from '@xyo-network/chain-wrappers'\nimport {\n type HydratedTransaction, type Transfer, TransferSchema,\n} from '@xyo-network/xl1-protocol'\n\nexport async function generateTransactionTransfers(address: Address, transactions: HydratedTransaction[]): Promise<Transfer[]> {\n const txs = await Promise.all(transactions.map(tx => HydratedTransactionWrapper.parse(tx)))\n\n // merge transactions with the same from address\n const txCosts: [Address, bigint][] = []\n for (const tx of txs) {\n const existingItem = txCosts.find(([addr]) => addr === tx.boundWitness.from)\n if (existingItem) {\n existingItem[1] += tx.fees.base\n } else {\n txCosts.push([tx.boundWitness.from, tx.fees.base])\n }\n }\n\n // generate actual Transfer Payloads\n const payloads = txCosts.map(([from, amount]) => {\n const payload: Transfer = {\n schema: TransferSchema,\n epoch: Date.now(),\n from,\n // TODO: add function to Wrapper to calculate entire cost of transaction\n transfers: { [address]: hexFromBigInt(amount) },\n }\n return payload\n })\n return payloads\n}\n","import { assertEx } from '@xylabs/assert'\nimport { XyoChainRewards__factory as XyoChainRewardsFactory } from '@xyo-network/typechain'\nimport {\n BaseAccountableServiceParams, BlockRewardService, ChainService,\n toEthAddress,\n} from '@xyo-network/xl1-protocol'\nimport { Provider } from 'ethers/providers'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport interface EvmBlockRewardServiceParams extends BaseAccountableServiceParams {\n chainService?: ChainService\n provider?: Provider\n}\n\n@creatableService<EvmBlockRewardService>()\nexport class EvmBlockRewardService extends BaseService<EvmBlockRewardServiceParams> implements BlockRewardService {\n protected _contractAddress: string | undefined\n\n protected get chainService() {\n return assertEx(this.params.chainService, () => 'chainService is required')\n }\n\n protected get contractAddress() {\n return assertEx(this._contractAddress, () => 'contractAddress is required')\n }\n\n protected get provider() {\n return assertEx(this.params.provider, () => 'provider is required')\n }\n\n override async createHandler() {\n await super.createHandler()\n this._contractAddress = await this.chainService.rewardsContract()\n }\n\n async getRewardForBlock(blockNumber: bigint): Promise<bigint> {\n const contract = XyoChainRewardsFactory.connect(toEthAddress(this.contractAddress), this.provider)\n return await contract.calcBlockReward(blockNumber)\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { toFixedPoint } from '@xylabs/decimal-precision'\nimport { Promisable } from '@xylabs/promise'\nimport { rewardFromBlockNumber } from '@xyo-network/chain-protocol'\nimport { BaseServiceParams, BlockRewardService } from '@xyo-network/xl1-protocol'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport interface XyoBlockRewardServiceParams extends BaseServiceParams {\n creatorReward?: bigint\n initialStepReward?: bigint\n minRewardPerBlock?: bigint\n stepFactorDenominator?: bigint\n stepFactorNumerator?: bigint\n stepSize?: bigint\n}\n\n@creatableService()\nexport class XyoBlockRewardService extends BaseService<XyoBlockRewardServiceParams> implements BlockRewardService {\n protected readonly rewardFromBlockNumber = rewardFromBlockNumber(18)\n\n constructor(\n {\n creatorReward = toFixedPoint(20_000_000_000n),\n initialStepReward = toFixedPoint(3000n),\n minRewardPerBlock = toFixedPoint(30n),\n stepFactorDenominator = 100n,\n stepFactorNumerator = 90n,\n stepSize = 1_000_000n,\n }: Partial<XyoBlockRewardServiceParams> = {},\n ) {\n super({\n initialStepReward, minRewardPerBlock, stepSize, stepFactorDenominator, stepFactorNumerator, creatorReward,\n })\n }\n\n get creatorReward() {\n return assertEx(this.params.creatorReward, () => 'creatorReward is required')\n }\n\n get initialReward() {\n return assertEx(this.params.initialStepReward, () => 'initialStepReward is required')\n }\n\n get minRewardPerBlock() {\n return assertEx(this.params.minRewardPerBlock, () => 'minRewardPerBlock is required')\n }\n\n get stepFactorDenominator() {\n return assertEx(this.params.stepFactorDenominator, () => 'stepFactorDenominator is required')\n }\n\n get stepFactorNumerator() {\n return assertEx(this.params.stepFactorNumerator, () => 'stepFactorNumerator is required')\n }\n\n get stepSize() {\n return assertEx(this.params.stepSize, () => 'stepSize is required')\n }\n\n getRewardForBlock(blockNumber: bigint): Promisable<bigint> {\n return this.rewardFromBlockNumber(\n blockNumber,\n this.initialReward,\n this.stepSize,\n this.stepFactorNumerator,\n this.stepFactorDenominator,\n this.minRewardPerBlock,\n this.creatorReward,\n )\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { Promisable } from '@xylabs/promise'\nimport { AccountInstance } from '@xyo-network/account-model'\nimport { ArchivistInstance, ReadArchivist } from '@xyo-network/archivist-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport {\n BaseServiceParams,\n BlockBoundWitness,\n BlockRewardService,\n ChainInformation,\n ElectionService,\n HydratedBlockStateValidationFunction,\n HydratedTransaction,\n HydratedTransactionWithStorageMeta,\n StakeIntentService,\n} from '@xyo-network/xl1-protocol'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\nimport { Validator } from './model/index.ts'\n\nconst addStorageMeta = async ([tx, payloads]: HydratedTransaction): Promise<HydratedTransactionWithStorageMeta> => {\n return [await PayloadBuilder.addStorageMeta(tx), await PayloadBuilder.addStorageMeta(payloads)]\n}\n\nexport interface XyoValidatorParams extends BaseServiceParams {\n account?: AccountInstance\n chainFinalizedArchivist?: ReadArchivist\n chainInformation?: ChainInformation\n electionService?: ElectionService\n pendingTransactions?: ArchivistInstance\n rewardService?: BlockRewardService\n stakeIntentService?: StakeIntentService\n validateHydratedBlockState?: HydratedBlockStateValidationFunction\n}\n\n@creatableService()\nexport class XyoValidator<TParams extends XyoValidatorParams = XyoValidatorParams> extends BaseService<TParams> implements Validator {\n get address() {\n return this.account.address\n }\n\n protected get account() {\n return assertEx(this.params.account, () => 'account is required')\n }\n\n protected get chainFinalizedArchivist() {\n return assertEx(this.params.chainFinalizedArchivist, () => 'chainFinalizedArchivist is required')\n }\n\n protected get chainInfo() {\n return assertEx(this.params.chainInformation, () => 'chainInfo is required')\n }\n\n protected get electionService() {\n return assertEx(this.params.electionService, () => 'electionService is required')\n }\n\n protected get pendingTransactions() {\n return assertEx(this.params.pendingTransactions, () => 'pendingTransactions is required')\n }\n\n protected get rewardService() {\n return assertEx(this.params.rewardService, () => 'rewardService is required')\n }\n\n validatePendingBlock(_block: BlockBoundWitness): Promisable<Error[]> {\n return [] // await validateBlockProtocol(block, this.chainInfo)\n }\n\n // TODO: Move to validator and inherit this class from validator\n async validatePendingTransaction(hydratedTransaction: HydratedTransaction): Promise<boolean> {\n const [tx] = await addStorageMeta(hydratedTransaction)\n // Ensure not confirmed already (replay attack)\n if ((await this.chainFinalizedArchivist.get([tx._hash])).length > 0) return false\n // TODO: Ensure transaction is valid (double spend, has voucher, has required stake, etc.)\n // TODO: Ensure validator stake is valid\n return await Promise.resolve(true)\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { Address, Hash } from '@xylabs/hex'\nimport { hexToLast4BytesInt, shuffleWithSeed } from '@xyo-network/chain-utils'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport type {\n BaseServiceParams, ChainBlockNumberIterator, ChainStakeViewer, StakeIntentService,\n} from '@xyo-network/xl1-protocol'\nimport { BlockBoundWitness, ElectionService } from '@xyo-network/xl1-protocol'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport interface XyoElectionServicesParams extends BaseServiceParams {\n chainIterator?: ChainBlockNumberIterator\n chainStakeViewer?: ChainStakeViewer\n stakeIntentService?: StakeIntentService\n}\n\n@creatableService()\nexport class XyoElectionService extends BaseService<XyoElectionServicesParams> implements ElectionService {\n constructor(params: XyoElectionServicesParams) {\n super(params)\n }\n\n get chainIterator() {\n return assertEx(this.params.chainIterator, () => 'No chain iterator')\n }\n\n get chainStakeViewer() {\n return assertEx(this.params.chainStakeViewer, () => 'No chain stake viewer')\n }\n\n get stakeIntentService() {\n return assertEx(this.params.stakeIntentService, () => 'No staked intent service')\n }\n\n async getCreatorCommitteeForNextBlock(current: BlockBoundWitness): Promise<Address[]> {\n return await this.spanAsync('getCreatorCommitteeForNextBlock', async () => {\n const nextBlock = current.block + 1\n const candidates = await this.stakeIntentService.getDeclaredCandidatesForBlock(nextBlock, 'producer')\n const previousBlockHash = await PayloadBuilder.hash(current)\n return this.generateCreatorCommittee(candidates, previousBlockHash)\n })\n }\n\n protected generateCreatorCommittee(candidates: Address[], previousBlockHash: Hash, maxSize = 3): Address[] {\n const creators = new Set<Address>(candidates)\n const seed = hexToLast4BytesInt(previousBlockHash)\n const creatorArray = shuffleWithSeed(creators, seed)\n return creatorArray.slice(0, maxSize)\n }\n}\n","import { filterAs, filterAsync } from '@xylabs/array'\nimport { assertEx } from '@xylabs/assert'\nimport { exists } from '@xylabs/exists'\nimport { Hash } from '@xylabs/hex'\nimport { MemoryArchivist } from '@xyo-network/archivist-memory'\nimport { ArchivistInstance } from '@xyo-network/archivist-model'\nimport { hydrateBlock } from '@xyo-network/chain-protocol'\nimport { validateTransaction } from '@xyo-network/chain-validation'\nimport {\n Payload, Sequence, WithStorageMeta,\n} from '@xyo-network/payload-model'\nimport {\n asOptionalTransactionBoundWitnessWithStorageMeta, BaseServiceParams, ChainIdentification, HydratedTransactionWithStorageMeta,\n PendingTransactionsService, TransactionBoundWitness,\n} from '@xyo-network/xl1-protocol'\nimport { flattenHydratedTransactions, tryHydrateTransaction } from '@xyo-network/xl1-protocol-sdk'\nimport { Mutex } from 'async-mutex'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport interface XyoPendingTransactionsServiceParams extends BaseServiceParams {\n chainArchivist?: ArchivistInstance\n chainIdentification?: ChainIdentification\n pendingTransactionsArchivist?: ArchivistInstance\n rejectedTransactionsArchivist?: ArchivistInstance\n}\n\n@creatableService()\nexport class XyoPendingTransactionsService extends BaseService<XyoPendingTransactionsServiceParams> implements PendingTransactionsService {\n private static readonly MutexPriority = {\n /**\n * Priority for inserting new transactions\n */\n InsertNewTransactions: 5,\n /**\n * Priority for reading pending transactions\n */\n ReadTransactions: 3,\n /**\n * Priority for removing rejected transactions\n */\n RemoveRejectedTransactions: 2,\n /**\n * Priority for removing finalized transactions\n */\n RemoveFinalizedTransactions: 1,\n }\n\n // used to remember the last block's transactions that was removed due to it being in the chain\n private _cleanedToBlock: Hash | undefined\n\n // a local archivist that has only the transactions in it (used as queue)\n private _pendingTransactionsLocalArchivist: MemoryArchivist | undefined\n private _pendingTransactionsLocalArchivistCursor: Sequence | undefined\n\n private _updatePendingTransactionsLocalArchivistMutex = new Mutex()\n\n private get chainArchivist() {\n return assertEx(this.params.chainArchivist, () => 'No completed blocks with data archivist')\n }\n\n private get chainIdentification() {\n return assertEx(this.params.chainIdentification, () => 'No chain id')\n }\n\n private get pendingTransactionsArchivist() {\n return assertEx(this.params.pendingTransactionsArchivist, () => 'No pending transactions archivist')\n }\n\n private get pendingTransactionsLocalArchivist() {\n return assertEx(this._pendingTransactionsLocalArchivist, () => 'No pending transactions curate archivist')\n }\n\n private get rejectedTransactionsArchivist() {\n return assertEx(this.params.rejectedTransactionsArchivist, () => 'No rejected transactions archivist')\n }\n\n override async createHandler() {\n await super.createHandler()\n this._pendingTransactionsLocalArchivist = await MemoryArchivist.create({ account: 'random' })\n // On new pending transactions, insert them into the curated archivist\n this.pendingTransactionsArchivist.on('inserted', async ({ payloads }) => {\n await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {\n // Check incoming transactions against finalized transactions\n const unprocessedTransactions = await this.filterAlreadyFinalizedTransactions(payloads)\n // Hydrate all unprocessed transactions\n const hydratedUnprocessedTransactions = (await Promise.all(unprocessedTransactions.map((tx) => {\n return tryHydrateTransaction(this.pendingTransactionsArchivist, tx._hash)\n }))).filter(exists)\n // Filter to only valid transactions\n const validTransactions = await filterAsync(hydratedUnprocessedTransactions, async (tx) => {\n const errors = await validateTransaction(tx, this.chainIdentification.id)\n return errors.length > 0 ? false : true\n })\n await this.pendingTransactionsLocalArchivist.insert(flattenHydratedTransactions(validTransactions))\n }, XyoPendingTransactionsService.MutexPriority.InsertNewTransactions)\n })\n // On new finalized blocks, remove the transactions from the curated archivist\n this.chainArchivist.on('inserted', async ({ payloads }) => {\n await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {\n const transactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta)\n await this.pendingTransactionsLocalArchivist.delete(transactions.map(tx => tx._hash))\n }, XyoPendingTransactionsService.MutexPriority.RemoveFinalizedTransactions)\n })\n // On new rejected blocks, remove the transactions from the curated archivist\n this.rejectedTransactionsArchivist.on('inserted', async ({ payloads }) => {\n await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {\n const transactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta)\n await this.pendingTransactionsLocalArchivist.delete(transactions.map(tx => tx._hash))\n }, XyoPendingTransactionsService.MutexPriority.RemoveRejectedTransactions)\n })\n }\n\n async getPendingTransactions(head: Hash, limit: number, timeout = 1_000_000): Promise<HydratedTransactionWithStorageMeta[]> {\n return await this.spanAsync('getPendingTransactions', async () => {\n // await this.updatePendingTransactionsLocalArchivist()\n // await this.removeAlreadyIncludedTransactions(head)\n return await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {\n const foundPendingTransactions: WithStorageMeta<TransactionBoundWitness>[] = []\n let cursor: Sequence | undefined\n while (foundPendingTransactions.length < limit) {\n const payloads = await this.pendingTransactionsLocalArchivist.next({\n limit: 100, order: 'asc', cursor,\n })\n if (payloads.length === 0) break\n cursor = payloads.at(-1)?._sequence\n const transactions = payloads.map(payload => asOptionalTransactionBoundWitnessWithStorageMeta(payload)).filter(exists)\n foundPendingTransactions.push(...transactions)\n }\n const hydratedTransactions = await Promise.all(\n foundPendingTransactions.map(tx => tryHydrateTransaction(this.pendingTransactionsLocalArchivist, tx._hash)),\n )\n return hydratedTransactions.filter(exists)\n }, XyoPendingTransactionsService.MutexPriority.ReadTransactions)\n }, timeout)\n }\n\n private async filterAlreadyFinalizedTransactions(payloads: WithStorageMeta<Payload>[]): Promise<WithStorageMeta<TransactionBoundWitness>[]> {\n const incomingTransactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta)\n const incomingTransactionHashes = incomingTransactions.map(payload => payload._hash)\n const finalizedTransactions = await this.chainArchivist.get(incomingTransactionHashes)\n const finalizedTransactionHashes = new Set(finalizedTransactions.map(item => item._hash))\n const nonFinalizedTransactions = incomingTransactions.filter(item => !finalizedTransactionHashes.has(item._hash))\n return nonFinalizedTransactions\n }\n\n private async removeAlreadyIncludedTransactions(head?: Hash, timeout = 1_000_000) {\n return await this.spanAsync('removeAlreadyIncludedTransactions', async () => {\n let headPtr: Hash | null = head ?? null\n while (headPtr !== null) {\n const [block, payloads] = assertEx(await hydrateBlock(this.chainArchivist, headPtr), () => `Failed to find head [${head}]`)\n if (block._hash === this._cleanedToBlock) break\n await this.pendingTransactionsLocalArchivist.delete(\n payloads.map(payload => asOptionalTransactionBoundWitnessWithStorageMeta(payload)?._hash).filter(exists),\n )\n this._cleanedToBlock = block._hash\n headPtr = block?.previous\n }\n }, timeout)\n }\n\n private async updatePendingTransactionsLocalArchivist(timeout = 1_000_000) {\n return await this.spanAsync('updatePendingTransactionsLocalArchivist', async () => {\n if (this._updatePendingTransactionsLocalArchivistMutex.isLocked()) {\n console.warn('updatePendingTransactionsLocalArchivist busy - skipping')\n return\n }\n return await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {\n // const localWithDataArchivist = this.pendingTransactionsArchivist\n // sync from remote\n // await this.spanAsync('sync from remote [pending]', async () => {\n // this._pendingTransactionsLocalAllDataArchivistCursor = await synchronizeArchivists(\n // this.externalPendingTransactionsWithDataArchivist,\n // localWithDataArchivist,\n // this._pendingTransactionsLocalAllDataArchivistCursor,\n // )\n // }, timeout / 2)\n // // extract transactions only [from localWithDataArchivist]\n // await this.spanAsync('sync from local [pending]', async () => {\n // this._pendingTransactionsLocalArchivistCursor = await synchronizeArchivists(\n // localWithDataArchivist,\n // this.pendingTransactionsLocalArchivist,\n // this._pendingTransactionsLocalArchivistCursor,\n // isTransactionBoundWitness,\n // isTransactionBoundWitness,\n // )\n // }, timeout / 10)\n await Promise.resolve()\n })\n }, timeout)\n }\n}\n","import { filterAs } from '@xylabs/array'\nimport { assertEx } from '@xylabs/assert'\nimport {\n Address, asAddress, Hash,\n} from '@xylabs/hex'\nimport { ArchivistInstance, ArchivistNextOptions } from '@xyo-network/archivist-model'\nimport {\n analyzeChain, ChainStakeIntentAnalyzer,\n isChainSummaryStakeIntent,\n} from '@xyo-network/chain-analyze'\nimport {\n DEFAULT_FIND_FIRST_MATCHING_NEXT_OPTIONS,\n findFirstMatching, IntervalMap,\n SerializedIntervalMap,\n} from '@xyo-network/chain-utils'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { Payload, WithStorageMeta } from '@xyo-network/payload-model'\nimport type {\n BaseServiceParams, EventingChainBlockNumberIterator, Intent,\n} from '@xyo-network/xl1-protocol'\nimport {\n asOptionalBlockBoundWitness, asOptionalBlockBoundWitnessWithStorageMeta,\n asOptionalChainIndexingServiceStateWithStorageMeta, asOptionalChainStakeIntent, ChainIndexingServiceState, ChainIndexingServiceStateSchema, ChainStakeViewer,\n isChainIndexingServiceState,\n StakeIntentService,\n} from '@xyo-network/xl1-protocol'\nimport { Mutex } from 'async-mutex'\nimport { LRUCache } from 'lru-cache'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport interface XyoStakeIntentServiceParams extends BaseServiceParams {\n chainArchivist?: ArchivistInstance\n chainIterator?: EventingChainBlockNumberIterator\n chainStakeViewer?: ChainStakeViewer\n stakeIntentStateArchivist?: ArchivistInstance\n}\n\n/**\n * The number of blocks to periodically persist state\n */\nconst STATE_PERSISTENCE_INTERVAL = 60n\n\nconst ACTIVE_STAKE_TTL = 1000 * 60 * 60 * 2 // 2 hours in milliseconds\nconst NO_ACTIVE_STAKE_TTL = 1000 * 2 // 2 seconds in milliseconds\nconst STAKE_CACHE_MAX_ENTRIES = 10_000\n\n@creatableService()\nexport class XyoStakeIntentService extends BaseService<XyoStakeIntentServiceParams> implements StakeIntentService {\n // TODO: Use hash instead of block number to handle chain reorgs\n protected _lastIndexedBlockHash: Hash | undefined = undefined\n // TODO: Interval tree per declaration (bank, validator, etc.)\n // NOTE: Ideally move to DataIntervalTree to handle declared\n // ranges as it enables range queries in O(min(n, k * log n)) time,\n // where k is the number of intervals in the output list time\n // Currently using set based because it's simpler, equivalent\n // in performance for small sets, and (most importantly) easily\n // persisted so we can recover state on restart.\n protected _producers: IntervalMap<Address> = new IntervalMap()\n protected _stakeCache = new LRUCache<Address, bigint>({ max: STAKE_CACHE_MAX_ENTRIES })\n protected _updateMutex = new Mutex()\n\n constructor(params: XyoStakeIntentServiceParams) {\n super(params)\n this.chainIterator.on('headUpdated', async () => {\n await this.updateIndex()\n })\n }\n\n protected get chainArchivist() {\n return assertEx(this.params.chainArchivist, () => 'chainArchivist not set')\n }\n\n protected get chainIterator() {\n return assertEx(this.params.chainIterator, () => 'chainIterator not set')\n }\n\n protected get chainStakeViewer() {\n return assertEx(this.params.chainStakeViewer, () => 'chainStakeViewer not set')\n }\n\n protected get stakeIntentStateArchivist() {\n return assertEx(this.params.stakeIntentStateArchivist, () => 'stakeIntentStateArchivist not set')\n }\n\n override async createHandler(): Promise<void> {\n const head = await this.chainIterator.head()\n const headHash = await PayloadBuilder.hash(head)\n if (head?.block === undefined) return\n await this.recoverState(headHash)\n await this.updateIndex(true)\n }\n\n async getDeclaredCandidateRanges(address: Address, intent: Intent): Promise<Readonly<Readonly<[number, number]>[]>> {\n await Promise.resolve()\n assertEx(intent === 'producer', () => `Error: Support not yet added for intent ${intent}`)\n const results = this._producers.get(address)\n return results ?? []\n }\n\n async getDeclaredCandidatesForBlock(block: number, intent: Intent): Promise<Address[]> {\n return await this.spanAsync('getDeclaredCandidatesForBlock', async () => {\n assertEx(intent === 'producer', () => `Error: Support not yet added for intent ${intent}`)\n const results = this._producers.findAllContaining(block)\n const candidates = [...results]\n const validCandidates = await this.filterToValidStake(candidates, this.chainStakeViewer)\n return validCandidates\n })\n }\n\n async isStakedForBlock(block: number, intent: Intent, address: Address): Promise<boolean> {\n const candidates = await this.getDeclaredCandidatesForBlock(block, intent)\n return candidates.includes(address)\n }\n\n private async filterToValidStake(\n candidates: Address[],\n chainStakeViewer: ChainStakeViewer,\n requiredMinimumStake: bigint = 1n,\n ): Promise<Address[]> {\n type CandidateStake = { candidate: Address; stake: bigint }\n\n // Find the stake for each candidate\n const candidatesWithStake: CandidateStake[] = await Promise.all(\n candidates.map(async (candidate) => {\n // Check if the stake is already cached\n const stake = this._stakeCache.get(candidate)\n if (stake === undefined) {\n // Fetch from chainStakeViewer if not cached\n const activeStake = await chainStakeViewer.activeByAddressStaked(`${candidate}`)\n if (activeStake > 0n) {\n // Store result in cache\n this._stakeCache.set(candidate, activeStake, { ttl: ACTIVE_STAKE_TTL })\n } else {\n this._stakeCache.set(candidate, activeStake, { ttl: NO_ACTIVE_STAKE_TTL })\n }\n return { candidate, stake: activeStake }\n } else {\n return { candidate, stake }\n }\n }),\n )\n\n // Filter out candidates whose stake is greater than or equal to the required minimum\n return candidatesWithStake\n .filter(({ stake }) => stake >= requiredMinimumStake)\n .map(({ candidate }) => candidate)\n }\n\n private async persistState(current: Hash): Promise<void> {\n const state = this._producers.serialize()\n const payload = new PayloadBuilder<ChainIndexingServiceState<SerializedIntervalMap>>({ schema: ChainIndexingServiceStateSchema })\n .fields({ endBlockHash: current, state })\n .build()\n await this.stakeIntentStateArchivist.insert([payload])\n }\n\n private async recoverState(current: Hash): Promise<void> {\n const currentBlock = assertEx(asOptionalBlockBoundWitness((await this.chainArchivist.get([current]))?.[0]), () => `Block ${current} not found`)\n const currentBlockNum = currentBlock.block\n // Find last state before current head (in case of rollback, we indexed past it on an uncle chain, etc.)\n const opts: ArchivistNextOptions = { ...DEFAULT_FIND_FIRST_MATCHING_NEXT_OPTIONS }\n while (true) {\n const predicate = (p: WithStorageMeta<Payload>) => {\n const state = asOptionalChainIndexingServiceStateWithStorageMeta(p)\n return state ? true : false\n }\n const state = await findFirstMatching(this.stakeIntentStateArchivist, predicate, opts)\n if (isChainIndexingServiceState<SerializedIntervalMap>(state)) {\n const indexed = (await this.chainArchivist.get([state.endBlockHash]))?.[0]\n const indexedBlock = asOptionalBlockBoundWitnessWithStorageMeta(indexed)\n if (indexedBlock) {\n const indexedBlockNum = indexedBlock.block\n if (indexedBlockNum <= currentBlockNum) {\n const data = state.state as SerializedIntervalMap\n this._producers = new IntervalMap(data)\n this._lastIndexedBlockHash = indexedBlock._hash\n break\n }\n }\n } else {\n // No state found, start from genesis\n break\n }\n opts.open = true\n }\n }\n\n private async updateIndex(displayProgress = false): Promise<void> {\n if (this._updateMutex.isLocked()) {\n return\n }\n await this._updateMutex.runExclusive(async () => {\n return await this.spanAsync('updateIndex', async () => {\n const currentHead = await this.chainIterator.head()\n const currentHeadHash = await PayloadBuilder.hash(currentHead)\n const result = await analyzeChain(this.chainArchivist, [new ChainStakeIntentAnalyzer('producer')], currentHeadHash, this._lastIndexedBlockHash)\n const signedDeclarations = filterAs(result.find(isChainSummaryStakeIntent)?.intents ?? [], asOptionalChainStakeIntent)\n if (currentHead.block === undefined) return\n const currentHeadBlockNum = currentHead.block\n if (displayProgress) this.logger?.info(`Updating index through 0x${currentHeadBlockNum}`)\n for (const signedDeclaration of signedDeclarations) {\n const { exp, nbf } = signedDeclaration\n const start = nbf\n const stop = exp\n const address = asAddress(signedDeclaration?.from)\n if (start !== undefined && stop !== undefined && address !== undefined) {\n this._producers.insert(address, start, stop)\n }\n }\n /*\n if (index % STATE_PERSISTENCE_INTERVAL === 0n) {\n if (displayProgress) this.logger?.info(`Persisting state at block ${index}`)\n await this.persistState(await PayloadBuilder.hash(block))\n }\n */\n this._lastIndexedBlockHash = currentHeadHash\n })\n })\n }\n}\n","import { Base } from '@xylabs/base'\nimport type { Address } from '@xylabs/hex'\nimport { toAddress } from '@xylabs/hex'\nimport type { Promisable } from '@xylabs/promise'\nimport type { StakedXyoChain } from '@xyo-network/typechain'\nimport { StakedXyoChain__factory as StakedXyoChainFactory } from '@xyo-network/typechain'\nimport type { BaseServiceParams, ChainService } from '@xyo-network/xl1-protocol'\nimport { getAddress } from 'ethers/address'\nimport type { ContractRunner } from 'ethers/providers'\n\nexport interface EvmChainServiceParams extends BaseServiceParams {\n contract: StakedXyoChain\n id: Address\n runner: ContractRunner\n}\n\n/**\n * A class that represents a chain stake as backed by an EVM smart contract\n */\nexport class EvmChainService extends Base<EvmChainServiceParams> implements ChainService {\n private _contract: StakedXyoChain\n\n protected constructor(params: EvmChainServiceParams) {\n super(params)\n this._contract = params.contract\n }\n\n get id(): Address {\n return this.params.id\n }\n\n get runner(): ContractRunner {\n return this.params.runner\n }\n\n static create({\n id, runner, logger, traceProvider, meterProvider,\n }: Omit<EvmChainServiceParams, 'contract'>): Promisable<EvmChainService> {\n const contractAddress = getAddress(id)\n const contract: StakedXyoChain = StakedXyoChainFactory.connect(contractAddress, runner)\n return new this({\n id, contract, runner, logger, traceProvider, meterProvider,\n })\n }\n\n async active(): Promise<bigint> {\n return await this._contract.active()\n }\n\n async activeByAddressStaked(address: string): Promise<bigint> {\n return await this._contract.activeByAddressStaked(getAddress(address))\n }\n\n async activeByStaker(address: string): Promise<bigint> {\n return await this._contract.activeByStaker(getAddress(address))\n }\n\n async addStake(staked: string, amount: bigint): Promise<boolean> {\n const result = await this._contract.addStake(getAddress(staked), amount)\n await result.wait()\n return true\n }\n\n async chainId(): Promise<Address> {\n return toAddress(await this._contract.chainId())\n }\n\n async forkedAtBlockNumber(): Promise<bigint> {\n return await this._contract.forkedAtBlockNumber()\n }\n\n async forkedAtHash(): Promise<bigint> {\n return await this._contract.forkedAtHash()\n }\n\n async forkedChainId(): Promise<Address> {\n return toAddress(await this._contract.forkedChainId())\n }\n\n async minWithdrawalBlocks(): Promise<bigint> {\n return await this._contract.minWithdrawalBlocks()\n }\n\n async pending(): Promise<bigint> {\n return await this._contract.pending()\n }\n\n async pendingByStaker(staker: string): Promise<bigint> {\n return await this._contract.pendingByStaker(getAddress(staker))\n }\n\n async removeStake(slot: bigint): Promise<boolean> {\n const result = await this._contract.removeStake(slot)\n await result.wait()\n return true\n }\n\n async rewardsContract(): Promise<string> {\n return await this._contract.rewardsContract()\n }\n\n async stakingTokenAddress(): Promise<string> {\n return await this._contract.stakingTokenAddress()\n }\n\n async withdrawStake(slot: bigint): Promise<boolean> {\n const result = await this._contract.withdrawStake(slot)\n await result.wait()\n return true\n }\n\n async withdrawn(): Promise<bigint> {\n return await this._contract.withdrawn()\n }\n\n async withdrawnByStaker(staker: string): Promise<bigint> {\n return await this._contract.withdrawnByStaker(getAddress(staker))\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport type { BaseParams } from '@xylabs/base'\nimport type { ArchivistInstance } from '@xyo-network/archivist-model'\nimport { BaseEmitter } from '@xyo-network/module-event-emitter'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport type {\n BlockBoundWitness, ChainIdentification,\n ChainIteratorEventData,\n EventingChainBlockNumberIterator,\n XyoChainIteratorParams,\n} from '@xyo-network/xl1-protocol'\nimport {\n asBlockBoundWitness, asOptionalBlockBoundWitness,\n isBlockBoundWitness,\n} from '@xyo-network/xl1-protocol'\nimport { LRUCache } from 'lru-cache'\n\nexport class XyoChainBlockNumberIterator extends BaseEmitter<BaseParams, ChainIteratorEventData> implements EventingChainBlockNumberIterator {\n protected _blocksByBlockNumber = new LRUCache<number, BlockBoundWitness>({ max: 10_000 })\n protected _chainArchivist: ArchivistInstance\n protected _chainInformation: ChainIdentification\n protected _head: BlockBoundWitness\n\n protected constructor(params: XyoChainIteratorParams) {\n super({})\n const { chainArchivist: archivist, head } = params\n this._chainArchivist = archivist\n this._head = head\n this._chainInformation = { id: head.chain }\n }\n\n get archivist(): ArchivistInstance { return this._chainArchivist }\n\n get chainIdentification(): ChainIdentification { return this._chainInformation }\n\n static async create(params: XyoChainIteratorParams): Promise<XyoChainBlockNumberIterator> {\n await Promise.resolve()\n return new XyoChainBlockNumberIterator(params)\n }\n\n async get(block: number): Promise<BlockBoundWitness> {\n // Bail early if the block requested is newer than the current head\n assertEx(this._head.block >= block, () => `Block requested is newer than the current head [${block}]`)\n const cached = this._blocksByBlockNumber.get(block)\n if (cached) return cached\n // Start at the current head and traverse backwards until the requested block is found\n const startingBlock = this._head\n const currentBlockHash = await PayloadBuilder.hash(startingBlock)\n let [currentBlock] = await this._chainArchivist.get([currentBlockHash])\n while (currentBlock) {\n assertEx(asBlockBoundWitness(currentBlock), () => `Expected hash to be a block bound witness [${currentBlock._hash}]`)\n if (isBlockBoundWitness(currentBlock)) {\n this._blocksByBlockNumber.set(currentBlock.block, currentBlock)\n if (currentBlock.block === block) {\n return currentBlock\n }\n const { previous } = currentBlock\n if (!previous) break\n [currentBlock] = await this._chainArchivist.get([previous])\n }\n }\n throw new Error(`Block not found: ${block}`)\n }\n\n async head(): Promise<BlockBoundWitness> {\n return await Promise.resolve(this._head)\n }\n\n async next(block: number): Promise<BlockBoundWitness | undefined> {\n const currentBlock = block\n const nextBlockNumber = currentBlock + 1\n return await this.get(nextBlockNumber)\n }\n\n // TODO: Decide on inclusive/exclusive (probably need inclusive to account for chain head)\n // and then communicate via method name and documentation\n async previous(block: number | undefined = undefined, count: number = 1): Promise<BlockBoundWitness[]> {\n const results: BlockBoundWitness[] = []\n let currentBlock: BlockBoundWitness | undefined = block ? (await this.get(block)) : this._head\n while (currentBlock && results.length < count) {\n if (isBlockBoundWitness(currentBlock)) {\n results.push(currentBlock)\n const { previous } = currentBlock\n if (!previous) break\n const nextBlock = await this._chainArchivist.get([previous])\n currentBlock = asOptionalBlockBoundWitness(nextBlock[0])\n } else {\n const hash = PayloadBuilder.hash(currentBlock)\n assertEx(asBlockBoundWitness(currentBlock), () => `Expected hash to be a block bound witness [${hash}]`)\n }\n }\n return results\n }\n\n async updateHead(head: BlockBoundWitness): Promise<void> {\n // Async to allow for re-indexing, etc.\n await Promise.resolve()\n this._head = head\n void this.emit('headUpdated', { blocks: [head] })\n }\n}\n"],"mappings":"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","names":["assertEx","toHex","analyzeChain","BalanceAnalyzer","isChainSummaryBalances","toPositiveBigInt","span","spanAsync","BaseEmitter","Mutex","BaseService","BaseEmitter","singletonInitMutex","Mutex","singletons","globalThis","meter","params","meterProvider","getMeter","name","constructor","tracer","traceProvider","getTracer","create","result","Error","createHandler","initSingleton","runExclusive","span","fn","spanAsync","timeout","BaseAccountableService","creatableService","XyoChainAccountBalanceService","BaseService","_balances","_previousHead","chainArchivist","assertEx","params","getBalance","address","toPositiveBigInt","positive","toHex","getBalances","result","entries","Object","balance","sync","head","analysis","analyzeChain","BalanceAnalyzer","tracer","find","isChainSummaryBalances","balances","BaseEthProvider","eth_accounts","Error","eth_blockNumber","eth_call","_transaction","_blockNumber","eth_clearSubscriptions","_keepSyncing","eth_coinbase","eth_compileLLL","_code","eth_compileSerpent","eth_compileSolidity","eth_estimateGas","eth_feeHistory","_blockCount","_newestBlock","_rewardPercentiles","eth_gasPrice","eth_getBalance","_address","eth_getBlockByHash","_blockHash","_hydrated","eth_getBlockByNumber","eth_getBlockTransactionCountByHash","eth_getBlockTransactionCountByNumber","eth_getCode","eth_getCompilers","eth_getFilterChanges","_filterIdentifier","eth_getFilterLogs","eth_getLogs","_filter","eth_getStorageAt","_storageSlot","eth_getTransactionByBlockHashAndIndex","_transactionIndex","eth_getTransactionByBlockNumberAndIndex","eth_getTransactionByHash","_transactionHash","eth_getTransactionCount","eth_getTransactionReceipt","eth_getUncleByBlockHashAndIndex","_uncleIndex","eth_getUncleByBlockNumberAndIndex","eth_getUncleCountByBlockHash","eth_getUncleCountByBlockNumber","eth_getWork","eth_hashrate","eth_maxPriorityFeePerGas","eth_mining","eth_newBlockFilter","eth_newFilter","eth_newPendingTransactionFilter","eth_protocolVersion","eth_sendRawTransaction","eth_sendTransaction","eth_sign","_message","eth_signTransaction","eth_submitHashrate","_hashRate","_id","eth_submitWork","_nonce","_hash","_digest","eth_subscribe","_params","eth_syncing","eth_uninstallFilter","eth_unsubscribe","_subscriptionId","assertEx","toHex","FixedPercentageBlockRewardDiviner","FixedPercentageBlockRewardDivinerConfigSchema","buildBlock","PayloadBuilder","asBlockBoundWitness","BlockNumberSchema","ChainStakeIntentSchema","hexFromBigInt","HydratedTransactionWrapper","TransferSchema","generateTransactionTransfers","address","transactions","txs","Promise","all","map","tx","HydratedTransactionWrapper","parse","txCosts","existingItem","find","addr","boundWitness","from","fees","base","push","payloads","amount","payload","schema","TransferSchema","epoch","Date","now","transfers","hexFromBigInt","DEFAULT_BLOCK_SIZE","XYO_PRODUCER_RESTAKE_DURATION","XYO_PRODUCER_RESTAKE_WINDOW","XyoBlockProducer","BaseService","_blockRewardDiviner","DefaultBlockSize","address","account","assertEx","params","chainFinalizedArchivist","chainInfo","chainInformation","electionService","pendingTransactions","pendingTransactionsService","rejectedTransactionsArchivist","rewardAddress","rewardService","stakeIntentService","validateHydratedBlockState","next","head","chain","id","leaders","getCreatorCommitteeForNextBlock","proposeNextValidBlock","getBlockRewardTransfer","block","FixedPercentageBlockRewardDiviner","create","blockRewardService","config","rewardPercentageRatio","schema","FixedPercentageBlockRewardDivinerConfigSchema","blockHex","toHex","blockId","PayloadBuilder","BlockNumberSchema","fields","build","rewards","divine","reward","getProducerRedeclaration","redeclareIntent","process","env","XYO_PRODUCER_REDECLARE_INTENT","ranges","getDeclaredCandidateRanges","lastRange","toSorted","a","b","at","currentDeclarationEnd","currentBlock","timeToProducerExpiration","intent","ChainStakeIntentSchema","from","nbf","exp","spanAsync","previousBlock","asBlockBoundWitness","nextBlock","nextBlockTransactions","getPendingTransactions","_hash","blockPayloads","producerRedeclarationPayload","push","length","rewardTransferPayload","transactionTransfers","generateTransactionTransfers","buildBlock","console","log","errors","logger","error","rejectedTransactions","insert","assertEx","XyoChainRewards__factory","XyoChainRewardsFactory","toEthAddress","EvmBlockRewardService","BaseService","_contractAddress","chainService","assertEx","params","contractAddress","provider","createHandler","rewardsContract","getRewardForBlock","blockNumber","contract","XyoChainRewardsFactory","connect","toEthAddress","calcBlockReward","assertEx","toFixedPoint","rewardFromBlockNumber","XyoBlockRewardService","BaseService","rewardFromBlockNumber","constructor","creatorReward","toFixedPoint","initialStepReward","minRewardPerBlock","stepFactorDenominator","stepFactorNumerator","stepSize","assertEx","params","initialReward","getRewardForBlock","blockNumber","assertEx","PayloadBuilder","addStorageMeta","tx","payloads","PayloadBuilder","XyoValidator","BaseService","address","account","assertEx","params","chainFinalizedArchivist","chainInfo","chainInformation","electionService","pendingTransactions","rewardService","validatePendingBlock","_block","validatePendingTransaction","hydratedTransaction","get","_hash","length","Promise","resolve","assertEx","hexToLast4BytesInt","shuffleWithSeed","PayloadBuilder","XyoElectionService","BaseService","constructor","params","chainIterator","assertEx","chainStakeViewer","stakeIntentService","getCreatorCommitteeForNextBlock","current","spanAsync","nextBlock","block","candidates","getDeclaredCandidatesForBlock","previousBlockHash","PayloadBuilder","hash","generateCreatorCommittee","maxSize","creators","Set","seed","hexToLast4BytesInt","creatorArray","shuffleWithSeed","slice","filterAs","filterAsync","assertEx","exists","MemoryArchivist","hydrateBlock","validateTransaction","asOptionalTransactionBoundWitnessWithStorageMeta","flattenHydratedTransactions","tryHydrateTransaction","Mutex","XyoPendingTransactionsService","BaseService","MutexPriority","InsertNewTransactions","ReadTransactions","RemoveRejectedTransactions","RemoveFinalizedTransactions","_cleanedToBlock","_pendingTransactionsLocalArchivist","_pendingTransactionsLocalArchivistCursor","_updatePendingTransactionsLocalArchivistMutex","Mutex","chainArchivist","assertEx","params","chainIdentification","pendingTransactionsArchivist","pendingTransactionsLocalArchivist","rejectedTransactionsArchivist","createHandler","MemoryArchivist","create","account","on","payloads","runExclusive","unprocessedTransactions","filterAlreadyFinalizedTransactions","hydratedUnprocessedTransactions","Promise","all","map","tx","tryHydrateTransaction","_hash","filter","exists","validTransactions","filterAsync","errors","validateTransaction","id","length","insert","flattenHydratedTransactions","transactions","filterAs","asOptionalTransactionBoundWitnessWithStorageMeta","delete","getPendingTransactions","head","limit","timeout","spanAsync","foundPendingTransactions","cursor","next","order","at","_sequence","payload","push","hydratedTransactions","incomingTransactions","incomingTransactionHashes","finalizedTransactions","get","finalizedTransactionHashes","Set","item","nonFinalizedTransactions","has","removeAlreadyIncludedTransactions","headPtr","block","hydrateBlock","previous","updatePendingTransactionsLocalArchivist","isLocked","console","warn","resolve","filterAs","assertEx","asAddress","analyzeChain","ChainStakeIntentAnalyzer","isChainSummaryStakeIntent","DEFAULT_FIND_FIRST_MATCHING_NEXT_OPTIONS","findFirstMatching","IntervalMap","PayloadBuilder","asOptionalBlockBoundWitness","asOptionalBlockBoundWitnessWithStorageMeta","asOptionalChainIndexingServiceStateWithStorageMeta","asOptionalChainStakeIntent","ChainIndexingServiceStateSchema","isChainIndexingServiceState","Mutex","LRUCache","ACTIVE_STAKE_TTL","NO_ACTIVE_STAKE_TTL","STAKE_CACHE_MAX_ENTRIES","XyoStakeIntentService","BaseService","_lastIndexedBlockHash","undefined","_producers","IntervalMap","_stakeCache","LRUCache","max","_updateMutex","Mutex","constructor","params","chainIterator","on","updateIndex","chainArchivist","assertEx","chainStakeViewer","stakeIntentStateArchivist","createHandler","head","headHash","PayloadBuilder","hash","block","recoverState","getDeclaredCandidateRanges","address","intent","Promise","resolve","results","get","getDeclaredCandidatesForBlock","spanAsync","findAllContaining","candidates","validCandidates","filterToValidStake","isStakedForBlock","includes","requiredMinimumStake","candidatesWithStake","all","map","candidate","stake","activeStake","activeByAddressStaked","set","ttl","filter","persistState","current","state","serialize","payload","schema","ChainIndexingServiceStateSchema","fields","endBlockHash","build","insert","currentBlock","asOptionalBlockBoundWitness","currentBlockNum","opts","DEFAULT_FIND_FIRST_MATCHING_NEXT_OPTIONS","predicate","p","asOptionalChainIndexingServiceStateWithStorageMeta","findFirstMatching","isChainIndexingServiceState","indexed","indexedBlock","asOptionalBlockBoundWitnessWithStorageMeta","indexedBlockNum","data","_hash","open","displayProgress","isLocked","runExclusive","currentHead","currentHeadHash","result","analyzeChain","ChainStakeIntentAnalyzer","signedDeclarations","filterAs","find","isChainSummaryStakeIntent","intents","asOptionalChainStakeIntent","currentHeadBlockNum","logger","info","signedDeclaration","exp","nbf","start","stop","asAddress","from","Base","toAddress","StakedXyoChain__factory","StakedXyoChainFactory","getAddress","EvmChainService","Base","_contract","params","contract","id","runner","create","logger","traceProvider","meterProvider","contractAddress","getAddress","StakedXyoChainFactory","connect","active","activeByAddressStaked","address","activeByStaker","addStake","staked","amount","result","wait","chainId","toAddress","forkedAtBlockNumber","forkedAtHash","forkedChainId","minWithdrawalBlocks","pending","pendingByStaker","staker","removeStake","slot","rewardsContract","stakingTokenAddress","withdrawStake","withdrawn","withdrawnByStaker","assertEx","BaseEmitter","PayloadBuilder","asBlockBoundWitness","asOptionalBlockBoundWitness","isBlockBoundWitness","LRUCache","XyoChainBlockNumberIterator","BaseEmitter","_blocksByBlockNumber","LRUCache","max","_chainArchivist","_chainInformation","_head","params","chainArchivist","archivist","head","id","chain","chainIdentification","create","Promise","resolve","get","block","assertEx","cached","startingBlock","currentBlockHash","PayloadBuilder","hash","currentBlock","asBlockBoundWitness","_hash","isBlockBoundWitness","set","previous","Error","next","nextBlockNumber","undefined","count","results","length","push","nextBlock","asOptionalBlockBoundWitness","updateHead","emit","blocks"]}
1
+ {"version":3,"sources":["../../src/AccountBalance/XyoChainAccountBalanceService.ts","../../src/BaseService.ts","../../src/BaseEthProvider.ts","../../src/BlockProducer/XyoBlockProducer.ts","../../src/BlockProducer/generateTransactionTransfers.ts","../../src/BlockReward/EvmBlockRewardService.ts","../../src/BlockReward/XyoBlockRewardService.ts","../../src/ChainValidator/XyoValidator.ts","../../src/Election/XyoElectionService.ts","../../src/PendingTransactions/PendingTransactions.ts","../../src/StakeIntent/XyoStakeIntentService.ts","../../src/Staker/Evm/Evm.ts","../../src/XyoChainBlockNumberIterator.ts"],"sourcesContent":["import { assertEx } from '@xylabs/assert'\nimport {\n Address, Hash, Hex,\n toHex,\n} from '@xylabs/hex'\nimport {\n analyzeChain, BalanceAnalyzer, isChainSummaryBalances,\n} from '@xyo-network/chain-analyze'\nimport { SignedBigInt, toPositiveBigInt } from '@xyo-network/chain-protocol'\nimport {\n AccountBalanceService, BaseServiceParams,\n ChainServiceCollection,\n} from '@xyo-network/xl1-protocol'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport type ChainAccountBalanceServiceParams = BaseServiceParams<Partial<Pick<ChainServiceCollection, 'chainArchivist'>>>\n\n@creatableService()\nexport class XyoChainAccountBalanceService extends BaseService<ChainAccountBalanceServiceParams> implements AccountBalanceService {\n private _balances: Record<Address, SignedBigInt> = {}\n private _previousHead: Hash | null = null\n\n protected get chainArchivist() {\n return assertEx(this.params.chainArchivist, () => 'chainArchivist is required')\n }\n\n getBalance(address: Address): Hex {\n return toPositiveBigInt(this._balances[address] ?? { positive: toHex(0n) }).positive\n }\n\n getBalances(): Record<Address, Hex> {\n const result: Record<Address, Hex> = {}\n const entries = Object.entries(this._balances) as [Address, SignedBigInt][]\n for (const [address, balance] of entries) {\n result[address] = toPositiveBigInt(balance).positive\n }\n return result\n }\n\n async sync(head: Hash): Promise<void> {\n const analysis = await analyzeChain(\n this.chainArchivist,\n [new BalanceAnalyzer()],\n head,\n this._previousHead,\n -1n,\n this.tracer,\n )\n this._balances = assertEx(analysis.find(isChainSummaryBalances)?.balances, () => 'Failed to sync balances')\n }\n}\n","import type { Tracer } from '@opentelemetry/api'\nimport { trace } from '@opentelemetry/api'\nimport type { Promisable } from '@xylabs/promise'\nimport { span, spanAsync } from '@xyo-network/chain-utils'\nimport { BaseEmitter } from '@xyo-network/module-event-emitter'\nimport type {\n BaseAccountableServiceParams, BaseServiceParams, Service,\n} from '@xyo-network/xl1-protocol'\nimport { Mutex } from 'async-mutex'\n\ndeclare global {\n var xyoServiceSingletons: Record<string, unknown>\n}\n\nexport class BaseService<TParams extends BaseServiceParams = BaseServiceParams> extends BaseEmitter<TParams> {\n private static singletonInitMutex = new Mutex()\n\n static get singletons() {\n return globalThis['xyoServiceSingletons'] ?? (globalThis['xyoServiceSingletons'] = {})\n }\n\n get meter() {\n return this.params.meterProvider?.getMeter(this.name)\n }\n\n get name() {\n return this.constructor.name\n }\n\n get tracer(): Tracer {\n return this.params.traceProvider?.getTracer(this.name) || trace.getTracer(this.name)\n }\n\n static async create<TService extends BaseService<TParams>, TParams extends BaseServiceParams = TService['params']>(\n this: CreatableService<TService>,\n params: TParams,\n ): Promise<TService> {\n const result = new this(params)\n if (result.name === 'BaseService') throw new Error('Cannot create BaseService')\n await result.createHandler()\n return result\n }\n\n static initSingleton<TService extends BaseService<TParams>, TParams extends BaseServiceParams>(params: TParams) {\n if (this.singletons[this.name]) throw new Error(`Singleton already initialized for ${this.name}`)\n return this.singletonInitMutex.runExclusive(async () => {\n return await this.create(params) as TService\n })\n }\n\n createHandler(): Promisable<void> {\n return\n }\n\n span<T>(name: string, fn: () => T): T {\n return span(`${this.name}:${name}`, fn, this.tracer)\n }\n\n async spanAsync<T>(name: string, fn: () => Promise<T>): Promise<T> {\n return await spanAsync(`${this.name}:${name}`, fn, this.tracer)\n }\n}\n\nexport abstract class BaseAccountableService<\n TParams extends BaseAccountableServiceParams = BaseAccountableServiceParams,\n> extends BaseService<TParams> {\n // Base class for services that have an account\n}\n\nexport interface CreatableService<T extends BaseService = BaseService> extends Service {\n new(params: T['params']): T\n create<T extends BaseService>(this: CreatableService<T>, params: T['params']): Promisable<T>\n}\n\nexport function creatableService<T extends BaseService = BaseService>() {\n return <U extends CreatableService<T>>(constructor: U) => {\n // eslint-disable-next-line @typescript-eslint/no-unused-expressions\n constructor\n }\n}\n","import type {\n Address, BlockAPI,\n BlockNumberOrTag,\n CompileResultAPI,\n EthExecutionAPI,\n FeeHistoryResultAPI,\n Filter,\n FilterResultsAPI,\n HexString, HexString8Bytes,\n HexString32Bytes,\n HexString256Bytes,\n HexStringBytes,\n SignedTransactionInfoAPI,\n SyncingStatusAPI,\n TransactionCallAPI,\n TransactionInfoAPI,\n TransactionReceiptAPI,\n TransactionWithSenderAPI,\n Uint,\n Uint256,\n} from 'web3-types'\n\nexport class BaseEthProvider implements EthExecutionAPI {\n eth_accounts(): Address[] {\n throw new Error('Method not implemented.')\n }\n\n eth_blockNumber(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_call(_transaction: TransactionCallAPI, _blockNumber: BlockNumberOrTag): HexStringBytes {\n throw new Error('Method not implemented.')\n }\n\n eth_clearSubscriptions(_keepSyncing?: boolean): void {\n throw new Error('Method not implemented.')\n }\n\n eth_coinbase(): Address {\n throw new Error('Method not implemented.')\n }\n\n eth_compileLLL(_code: string): HexStringBytes {\n throw new Error('Method not implemented.')\n }\n\n eth_compileSerpent(_code: string): HexStringBytes {\n throw new Error('Method not implemented.')\n }\n\n eth_compileSolidity(_code: string): CompileResultAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_estimateGas(_transaction: Partial<TransactionWithSenderAPI>, _blockNumber: BlockNumberOrTag): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_feeHistory(_blockCount: Uint, _newestBlock: BlockNumberOrTag, _rewardPercentiles: number[]): FeeHistoryResultAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_gasPrice(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getBalance(_address: Address, _blockNumber: BlockNumberOrTag): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getBlockByHash(_blockHash: HexString32Bytes, _hydrated: boolean): BlockAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getBlockByNumber(_blockNumber: BlockNumberOrTag, _hydrated: boolean): BlockAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getBlockTransactionCountByHash(_blockHash: HexString32Bytes): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getBlockTransactionCountByNumber(_blockNumber: BlockNumberOrTag): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getCode(_address: Address, _blockNumber: BlockNumberOrTag): HexStringBytes {\n throw new Error('Method not implemented.')\n }\n\n eth_getCompilers(): string[] {\n throw new Error('Method not implemented.')\n }\n\n eth_getFilterChanges(_filterIdentifier: Uint): FilterResultsAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getFilterLogs(_filterIdentifier: Uint): FilterResultsAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getLogs(_filter: Filter): FilterResultsAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getStorageAt(_address: Address, _storageSlot: Uint256, _blockNumber: BlockNumberOrTag): HexStringBytes {\n throw new Error('Method not implemented.')\n }\n\n eth_getTransactionByBlockHashAndIndex(_blockHash: HexString32Bytes, _transactionIndex: Uint): TransactionInfoAPI | undefined {\n throw new Error('Method not implemented.')\n }\n\n eth_getTransactionByBlockNumberAndIndex(_blockNumber: BlockNumberOrTag, _transactionIndex: Uint): TransactionInfoAPI | undefined {\n throw new Error('Method not implemented.')\n }\n\n eth_getTransactionByHash(_transactionHash: HexString32Bytes): TransactionInfoAPI | undefined {\n throw new Error('Method not implemented.')\n }\n\n eth_getTransactionCount(_address: Address, _blockNumber: BlockNumberOrTag): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getTransactionReceipt(_transactionHash: HexString32Bytes): TransactionReceiptAPI | undefined {\n throw new Error('Method not implemented.')\n }\n\n eth_getUncleByBlockHashAndIndex(_blockHash: HexString32Bytes, _uncleIndex: Uint): BlockAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getUncleByBlockNumberAndIndex(_blockNumber: BlockNumberOrTag, _uncleIndex: Uint): BlockAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_getUncleCountByBlockHash(_blockHash: HexString32Bytes): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getUncleCountByBlockNumber(_blockNumber: BlockNumberOrTag): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_getWork(): [HexString32Bytes, HexString32Bytes, HexString32Bytes] {\n throw new Error('Method not implemented.')\n }\n\n eth_hashrate(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_maxPriorityFeePerGas(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_mining(): boolean {\n throw new Error('Method not implemented.')\n }\n\n eth_newBlockFilter(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_newFilter(_filter: Filter): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_newPendingTransactionFilter(): Uint {\n throw new Error('Method not implemented.')\n }\n\n eth_protocolVersion(): string {\n throw new Error('Method not implemented.')\n }\n\n eth_sendRawTransaction(_transaction: HexStringBytes): HexString32Bytes {\n throw new Error('Method not implemented.')\n }\n\n eth_sendTransaction(_transaction: TransactionWithSenderAPI | Partial<TransactionWithSenderAPI>): HexString32Bytes {\n throw new Error('Method not implemented.')\n }\n\n eth_sign(_address: Address, _message: HexStringBytes): HexString256Bytes {\n throw new Error('Method not implemented.')\n }\n\n eth_signTransaction(_transaction: TransactionWithSenderAPI | Partial<TransactionWithSenderAPI>): HexStringBytes | SignedTransactionInfoAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_submitHashrate(_hashRate: HexString32Bytes, _id: HexString32Bytes): boolean {\n throw new Error('Method not implemented.')\n }\n\n eth_submitWork(_nonce: HexString8Bytes, _hash: HexString32Bytes, _digest: HexString32Bytes): boolean {\n throw new Error('Method not implemented.')\n }\n\n eth_subscribe(..._params: ['newHeads'] | ['newPendingTransactions'] | ['syncing'] | ['logs', { address?: HexString; topics?: HexString[] }]): HexString {\n throw new Error('Method not implemented.')\n }\n\n eth_syncing(): SyncingStatusAPI {\n throw new Error('Method not implemented.')\n }\n\n eth_uninstallFilter(_filterIdentifier: Uint): boolean {\n throw new Error('Method not implemented.')\n }\n\n eth_unsubscribe(_subscriptionId: HexString): HexString {\n throw new Error('Method not implemented.')\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { Address, toHex } from '@xylabs/hex'\nimport { ArchivistInstance, ReadArchivist } from '@xyo-network/archivist-model'\nimport {\n BlockRewardDiviner, FixedPercentageBlockRewardDiviner, FixedPercentageBlockRewardDivinerConfigSchema,\n} from '@xyo-network/chain-modules'\nimport { buildBlock } from '@xyo-network/chain-protocol'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { WithStorageMeta } from '@xyo-network/payload-model'\nimport {\n AllowedBlockPayload, asBlockBoundWitness, BlockBoundWitness, BlockNumber, BlockNumberSchema, BlockProducer, ChainStakeIntent, ChainStakeIntentSchema,\n HydratedBlock, PendingTransactionsService, StakeIntentService, Transfer,\n} from '@xyo-network/xl1-protocol'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\nimport { XyoValidatorParams } from '../ChainValidator/index.ts'\nimport { generateTransactionTransfers } from './generateTransactionTransfers.ts'\n\n/**\n * The default block size for a block\n */\nexport const DEFAULT_BLOCK_SIZE = 10\n\n// /**\n// * The amount of time for which a producer will restake their intent\n// */\nexport const XYO_PRODUCER_RESTAKE_DURATION = 10_000\n\n/**\n * The number of blocks within which a producer will redeclare\n * their intent to produce blocks\n */\nexport const XYO_PRODUCER_RESTAKE_WINDOW = 500n\n\nexport interface XyoBlockProducerParams extends XyoValidatorParams {\n pendingTransactionsArchivist?: ReadArchivist\n pendingTransactionsService?: PendingTransactionsService\n rejectedTransactionsArchivist?: ArchivistInstance\n rewardAddress?: Address\n stakeIntentService?: StakeIntentService\n}\n\n@creatableService()\nexport class XyoBlockProducer extends BaseService<XyoBlockProducerParams> implements BlockProducer {\n protected _blockRewardDiviner: BlockRewardDiviner | undefined\n\n static get DefaultBlockSize(): number {\n return DEFAULT_BLOCK_SIZE\n }\n\n get address() {\n return this.account.address\n }\n\n protected get account() {\n return assertEx(this.params.account, () => 'account is required')\n }\n\n protected get chainFinalizedArchivist() {\n return assertEx(this.params.chainFinalizedArchivist, () => 'chainFinalizedArchivist is required')\n }\n\n protected get chainInfo() {\n return assertEx(this.params.chainInformation, () => 'chainInfo is required')\n }\n\n protected get electionService() {\n return assertEx(this.params.electionService, () => 'electionService is required')\n }\n\n protected get pendingTransactions() {\n return assertEx(this.params.pendingTransactions, () => 'pendingTransactions is required')\n }\n\n protected get pendingTransactionsService() {\n return assertEx(this.params.pendingTransactionsService, () => 'Missing pendingTransactionsService')\n }\n\n protected get rejectedTransactionsArchivist() {\n return assertEx(this.params.rejectedTransactionsArchivist, () => 'No rejected transactions archivist')\n }\n\n protected get rewardAddress(): Address {\n return assertEx(this.params.rewardAddress, () => 'No reward address provided')\n }\n\n protected get rewardService() {\n return assertEx(this.params.rewardService, () => 'rewardService is required')\n }\n\n protected get stakeIntentService(): StakeIntentService {\n return assertEx(this.params.stakeIntentService, () => 'No StakeIntentService provided')\n }\n\n protected get validateHydratedBlockState() {\n return assertEx(this.params.validateHydratedBlockState, () => 'validateHydratedBlockState is required')\n }\n\n async next(head: WithStorageMeta<BlockBoundWitness>): Promise<HydratedBlock | undefined> {\n // If the block is for another chain, ignore\n if (head.chain !== this.chainInfo.id) return\n const leaders = await this.electionService.getCreatorCommitteeForNextBlock(head)\n // TODO: Should we propose block if creator committee is empty?\n // TODO: Handle the case where we're not the 1st leader but they're not responding\n // at a higher level than here as that's a network issue\n if (leaders?.[0] !== this.address) return\n return this.proposeNextValidBlock(head)\n }\n\n protected async getBlockRewardTransfer(block: number): Promise<Transfer | undefined> {\n if (!this._blockRewardDiviner) {\n // TODO: Adjust to allow for genesis block reward vs. normal block reward\n this._blockRewardDiviner = await FixedPercentageBlockRewardDiviner.create({\n account: 'random',\n blockRewardService: this.rewardService,\n config: {\n rewardAddress: this.rewardAddress,\n rewardPercentageRatio: 0,\n schema: FixedPercentageBlockRewardDivinerConfigSchema,\n },\n })\n }\n const blockHex = assertEx(toHex(block), () => 'Failed to convert block to hex')\n const blockId = new PayloadBuilder<BlockNumber>({ schema: BlockNumberSchema }).fields({ block: blockHex }).build()\n const rewards = await this._blockRewardDiviner.divine([blockId])\n const [reward] = rewards\n return reward\n }\n\n /**\n * Handles the producer redeclaration logic\n * @param head The current head block\n * @returns\n */\n protected async getProducerRedeclaration(head: WithStorageMeta<BlockBoundWitness>): Promise<ChainStakeIntent | undefined> {\n const redeclareIntent = process.env.XYO_PRODUCER_REDECLARE_INTENT\n if (!redeclareIntent) return\n // Decide if we need to redeclare\n const ranges = await this.stakeIntentService.getDeclaredCandidateRanges(this.address, 'producer')\n // TODO: This doesn't handle the case where the producer had declared a range for the future\n // but we're in a range that's not the future\n // Sort in ascending order based on ending range to get range with highest ending block\n const lastRange = ranges.toSorted((a, b) => a[1] > b[1] ? 1 : -1).at(-1)\n if (!lastRange) return\n const [, currentDeclarationEnd] = lastRange\n const currentBlock = head.block\n const timeToProducerExpiration = currentDeclarationEnd - currentBlock\n if (timeToProducerExpiration > XYO_PRODUCER_RESTAKE_WINDOW) return\n // Create redeclaration\n const intent = new PayloadBuilder<ChainStakeIntent>({ schema: ChainStakeIntentSchema }).fields({\n from: this.address,\n intent: 'producer',\n nbf: currentBlock,\n exp: currentBlock + XYO_PRODUCER_RESTAKE_DURATION,\n }).build()\n return intent\n }\n\n protected async proposeNextValidBlock(head: WithStorageMeta<BlockBoundWitness>): Promise<HydratedBlock | undefined> {\n return await this.spanAsync('proposeNextValidBlock', async () => {\n // Calculate the next block components\n const { block: previousBlock } = assertEx(asBlockBoundWitness(head), () => 'Invalid head block')\n const nextBlock = previousBlock + 1\n const nextBlockTransactions = await this.pendingTransactionsService.getPendingTransactions(head._hash, XyoBlockProducer.DefaultBlockSize)\n\n const blockPayloads: AllowedBlockPayload[] = []\n\n // Calculate the optional producer redeclaration and add it if necessary\n const producerRedeclarationPayload = await this.getProducerRedeclaration(head)\n if (producerRedeclarationPayload) blockPayloads.push(producerRedeclarationPayload)\n\n // If there are no transactions and no payloads, we don't need to create a block\n if (blockPayloads.length === 0 && nextBlockTransactions.length === 0) return\n\n // Calculate the optional block reward transfer and add if necessary\n const rewardTransferPayload = await this.getBlockRewardTransfer(nextBlock)\n if (rewardTransferPayload) blockPayloads.push(rewardTransferPayload)\n\n const transactionTransfers = await generateTransactionTransfers(this.address, nextBlockTransactions)\n blockPayloads.push(...transactionTransfers)\n\n // Build the block\n const block = await buildBlock(head, nextBlockTransactions, blockPayloads, [this.account])\n console.log('buildBlock', block)\n const errors = await this.validateHydratedBlockState(block, this.chainInfo.id, this.chainFinalizedArchivist)\n if (errors.length > 0) {\n this.logger?.error('Validation of produced block failed', errors)\n const rejectedTransactions = block[1]\n await this.rejectedTransactionsArchivist.insert(rejectedTransactions)\n } else {\n return block\n }\n })\n }\n}\n","import { type Address, hexFromBigInt } from '@xylabs/hex'\nimport { HydratedTransactionWrapper } from '@xyo-network/chain-wrappers'\nimport {\n type HydratedTransaction, type Transfer, TransferSchema,\n} from '@xyo-network/xl1-protocol'\n\nexport async function generateTransactionTransfers(address: Address, transactions: HydratedTransaction[]): Promise<Transfer[]> {\n const txs = await Promise.all(transactions.map(tx => HydratedTransactionWrapper.parse(tx)))\n\n // merge transactions with the same from address\n const txCosts: [Address, bigint][] = []\n for (const tx of txs) {\n const existingItem = txCosts.find(([addr]) => addr === tx.boundWitness.from)\n if (existingItem) {\n existingItem[1] += tx.fees.base\n } else {\n txCosts.push([tx.boundWitness.from, tx.fees.base])\n }\n }\n\n // generate actual Transfer Payloads\n const payloads = txCosts.map(([from, amount]) => {\n const payload: Transfer = {\n schema: TransferSchema,\n epoch: Date.now(),\n from,\n // TODO: add function to Wrapper to calculate entire cost of transaction\n transfers: { [address]: hexFromBigInt(amount) },\n }\n return payload\n })\n return payloads\n}\n","import { assertEx } from '@xylabs/assert'\nimport { XyoChainRewards__factory as XyoChainRewardsFactory } from '@xyo-network/typechain'\nimport {\n BaseAccountableServiceParams, BlockRewardService, ChainService,\n toEthAddress,\n} from '@xyo-network/xl1-protocol'\nimport { Provider } from 'ethers/providers'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport interface EvmBlockRewardServiceParams extends BaseAccountableServiceParams {\n chainService?: ChainService\n provider?: Provider\n}\n\n@creatableService<EvmBlockRewardService>()\nexport class EvmBlockRewardService extends BaseService<EvmBlockRewardServiceParams> implements BlockRewardService {\n protected _contractAddress: string | undefined\n\n protected get chainService() {\n return assertEx(this.params.chainService, () => 'chainService is required')\n }\n\n protected get contractAddress() {\n return assertEx(this._contractAddress, () => 'contractAddress is required')\n }\n\n protected get provider() {\n return assertEx(this.params.provider, () => 'provider is required')\n }\n\n override async createHandler() {\n await super.createHandler()\n this._contractAddress = await this.chainService.rewardsContract()\n }\n\n async getRewardForBlock(blockNumber: bigint): Promise<bigint> {\n const contract = XyoChainRewardsFactory.connect(toEthAddress(this.contractAddress), this.provider)\n return await contract.calcBlockReward(blockNumber)\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { toFixedPoint } from '@xylabs/decimal-precision'\nimport { Promisable } from '@xylabs/promise'\nimport { rewardFromBlockNumber } from '@xyo-network/chain-protocol'\nimport { BaseServiceParams, BlockRewardService } from '@xyo-network/xl1-protocol'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport interface XyoBlockRewardServiceParams extends BaseServiceParams {\n creatorReward?: bigint\n initialStepReward?: bigint\n minRewardPerBlock?: bigint\n stepFactorDenominator?: bigint\n stepFactorNumerator?: bigint\n stepSize?: bigint\n}\n\n@creatableService()\nexport class XyoBlockRewardService extends BaseService<XyoBlockRewardServiceParams> implements BlockRewardService {\n protected readonly rewardFromBlockNumber = rewardFromBlockNumber(18)\n\n constructor(\n {\n creatorReward = toFixedPoint(20_000_000_000n),\n initialStepReward = toFixedPoint(3000n),\n minRewardPerBlock = toFixedPoint(30n),\n stepFactorDenominator = 100n,\n stepFactorNumerator = 90n,\n stepSize = 1_000_000n,\n }: Partial<XyoBlockRewardServiceParams> = {},\n ) {\n super({\n initialStepReward, minRewardPerBlock, stepSize, stepFactorDenominator, stepFactorNumerator, creatorReward,\n })\n }\n\n get creatorReward() {\n return assertEx(this.params.creatorReward, () => 'creatorReward is required')\n }\n\n get initialReward() {\n return assertEx(this.params.initialStepReward, () => 'initialStepReward is required')\n }\n\n get minRewardPerBlock() {\n return assertEx(this.params.minRewardPerBlock, () => 'minRewardPerBlock is required')\n }\n\n get stepFactorDenominator() {\n return assertEx(this.params.stepFactorDenominator, () => 'stepFactorDenominator is required')\n }\n\n get stepFactorNumerator() {\n return assertEx(this.params.stepFactorNumerator, () => 'stepFactorNumerator is required')\n }\n\n get stepSize() {\n return assertEx(this.params.stepSize, () => 'stepSize is required')\n }\n\n getRewardForBlock(blockNumber: bigint): Promisable<bigint> {\n return this.rewardFromBlockNumber(\n blockNumber,\n this.initialReward,\n this.stepSize,\n this.stepFactorNumerator,\n this.stepFactorDenominator,\n this.minRewardPerBlock,\n this.creatorReward,\n )\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { Promisable } from '@xylabs/promise'\nimport { AccountInstance } from '@xyo-network/account-model'\nimport { ArchivistInstance, ReadArchivist } from '@xyo-network/archivist-model'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport {\n BaseServiceParams,\n BlockBoundWitness,\n BlockRewardService,\n ChainInformation,\n ElectionService,\n HydratedBlockStateValidationFunction,\n HydratedTransaction,\n HydratedTransactionWithStorageMeta,\n StakeIntentService,\n} from '@xyo-network/xl1-protocol'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\nimport { Validator } from './model/index.ts'\n\nconst addStorageMeta = async ([tx, payloads]: HydratedTransaction): Promise<HydratedTransactionWithStorageMeta> => {\n return [await PayloadBuilder.addStorageMeta(tx), await PayloadBuilder.addStorageMeta(payloads)]\n}\n\nexport interface XyoValidatorParams extends BaseServiceParams {\n account?: AccountInstance\n chainFinalizedArchivist?: ReadArchivist\n chainInformation?: ChainInformation\n electionService?: ElectionService\n pendingTransactions?: ArchivistInstance\n rewardService?: BlockRewardService\n stakeIntentService?: StakeIntentService\n validateHydratedBlockState?: HydratedBlockStateValidationFunction\n}\n\n@creatableService()\nexport class XyoValidator<TParams extends XyoValidatorParams = XyoValidatorParams> extends BaseService<TParams> implements Validator {\n get address() {\n return this.account.address\n }\n\n protected get account() {\n return assertEx(this.params.account, () => 'account is required')\n }\n\n protected get chainFinalizedArchivist() {\n return assertEx(this.params.chainFinalizedArchivist, () => 'chainFinalizedArchivist is required')\n }\n\n protected get chainInfo() {\n return assertEx(this.params.chainInformation, () => 'chainInfo is required')\n }\n\n protected get electionService() {\n return assertEx(this.params.electionService, () => 'electionService is required')\n }\n\n protected get pendingTransactions() {\n return assertEx(this.params.pendingTransactions, () => 'pendingTransactions is required')\n }\n\n protected get rewardService() {\n return assertEx(this.params.rewardService, () => 'rewardService is required')\n }\n\n validatePendingBlock(_block: BlockBoundWitness): Promisable<Error[]> {\n return [] // await validateBlockProtocol(block, this.chainInfo)\n }\n\n // TODO: Move to validator and inherit this class from validator\n async validatePendingTransaction(hydratedTransaction: HydratedTransaction): Promise<boolean> {\n const [tx] = await addStorageMeta(hydratedTransaction)\n // Ensure not confirmed already (replay attack)\n if ((await this.chainFinalizedArchivist.get([tx._hash])).length > 0) return false\n // TODO: Ensure transaction is valid (double spend, has voucher, has required stake, etc.)\n // TODO: Ensure validator stake is valid\n return await Promise.resolve(true)\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { Address, Hash } from '@xylabs/hex'\nimport { hexToLast4BytesInt, shuffleWithSeed } from '@xyo-network/chain-utils'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport type {\n BaseServiceParams, ChainBlockNumberIterator, ChainStakeViewer, StakeIntentService,\n} from '@xyo-network/xl1-protocol'\nimport { BlockBoundWitness, ElectionService } from '@xyo-network/xl1-protocol'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport interface XyoElectionServicesParams extends BaseServiceParams {\n chainIterator?: ChainBlockNumberIterator\n chainStakeViewer?: ChainStakeViewer\n stakeIntentService?: StakeIntentService\n}\n\n@creatableService()\nexport class XyoElectionService extends BaseService<XyoElectionServicesParams> implements ElectionService {\n constructor(params: XyoElectionServicesParams) {\n super(params)\n }\n\n get chainIterator() {\n return assertEx(this.params.chainIterator, () => 'No chain iterator')\n }\n\n get chainStakeViewer() {\n return assertEx(this.params.chainStakeViewer, () => 'No chain stake viewer')\n }\n\n get stakeIntentService() {\n return assertEx(this.params.stakeIntentService, () => 'No staked intent service')\n }\n\n async getCreatorCommitteeForNextBlock(current: BlockBoundWitness): Promise<Address[]> {\n return await this.spanAsync('getCreatorCommitteeForNextBlock', async () => {\n const nextBlock = current.block + 1\n const candidates = await this.stakeIntentService.getDeclaredCandidatesForBlock(nextBlock, 'producer')\n const previousBlockHash = await PayloadBuilder.hash(current)\n return this.generateCreatorCommittee(candidates, previousBlockHash)\n })\n }\n\n protected generateCreatorCommittee(candidates: Address[], previousBlockHash: Hash, maxSize = 3): Address[] {\n const creators = new Set<Address>(candidates)\n const seed = hexToLast4BytesInt(previousBlockHash)\n const creatorArray = shuffleWithSeed(creators, seed)\n return creatorArray.slice(0, maxSize)\n }\n}\n","import { ValueType } from '@opentelemetry/api'\nimport { filterAs, filterAsync } from '@xylabs/array'\nimport { assertEx } from '@xylabs/assert'\nimport { exists } from '@xylabs/exists'\nimport { forget } from '@xylabs/forget'\nimport { Hash } from '@xylabs/hex'\nimport { MemoryArchivist } from '@xyo-network/archivist-memory'\nimport { ArchivistInstance } from '@xyo-network/archivist-model'\nimport { validateTransaction } from '@xyo-network/chain-validation'\nimport {\n Payload, Sequence, WithStorageMeta,\n} from '@xyo-network/payload-model'\nimport {\n asOptionalTransactionBoundWitnessWithStorageMeta, BaseServiceParams, ChainIdentification, HydratedTransactionWithStorageMeta,\n PendingTransactionsService, TransactionBoundWitness,\n} from '@xyo-network/xl1-protocol'\nimport { flattenHydratedTransactions, tryHydrateTransaction } from '@xyo-network/xl1-protocol-sdk'\nimport { Mutex } from 'async-mutex'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport interface XyoPendingTransactionsServiceParams extends BaseServiceParams {\n chainArchivist?: ArchivistInstance\n chainIdentification?: ChainIdentification\n pendingTransactionsArchivist?: ArchivistInstance\n rejectedTransactionsArchivist?: ArchivistInstance\n}\n\n@creatableService()\nexport class XyoPendingTransactionsService extends BaseService<XyoPendingTransactionsServiceParams> implements PendingTransactionsService {\n private static readonly MutexPriority = {\n /**\n * Priority for inserting new transactions\n */\n InsertNewTransactions: 5,\n /**\n * Priority for reading pending transactions\n */\n ReadTransactions: 3,\n /**\n * Priority for removing rejected transactions\n */\n RemoveRejectedTransactions: 2,\n /**\n * Priority for removing finalized transactions\n */\n RemoveFinalizedTransactions: 1,\n } as const\n\n /**\n * A mutex to ensure that the counting the number of pending transactions is\n * not called concurrently\n */\n private _countPendingTransactionsMutex = new Mutex()\n\n /**\n * A local Archivist optimized for fast retrieval that stores only validated\n * pending transactions\n */\n private _curatedPendingTransactionsArchivist: MemoryArchivist | undefined\n\n /**\n * The last count of total pending transactions\n */\n private _pendingTransactionsCount: number = 0\n\n /**\n * A mutex to ensure that the curated pending transactions archivist is\n * updated in a thread-safe manner\n */\n private _updateCuratedPendingTransactionsArchivistMutex = new Mutex()\n\n private get chainArchivist() {\n return assertEx(this.params.chainArchivist, () => 'No completed blocks with data archivist')\n }\n\n private get chainIdentification() {\n return assertEx(this.params.chainIdentification, () => 'No chain id')\n }\n\n private get pendingTransactionsArchivist() {\n return assertEx(this.params.pendingTransactionsArchivist, () => 'No pending transactions archivist')\n }\n\n private get pendingTransactionsCount() {\n forget(this.countPendingTransactions())\n return this._pendingTransactionsCount\n }\n\n private get pendingTransactionsLocalArchivist() {\n return assertEx(this._curatedPendingTransactionsArchivist, () => 'No pending transactions curated archivist')\n }\n\n private get rejectedTransactionsArchivist() {\n return assertEx(this.params.rejectedTransactionsArchivist, () => 'No rejected transactions archivist')\n }\n\n override async createHandler() {\n await super.createHandler()\n this._curatedPendingTransactionsArchivist = await MemoryArchivist.create({ account: 'random' })\n\n // On new pending transactions, insert them into the curated archivist\n this.pendingTransactionsArchivist.on('inserted', async ({ payloads }) => {\n await this.insertNewTransactions(payloads)\n })\n\n // On new finalized blocks, remove the transactions from the curated archivist\n this.chainArchivist.on('inserted', async ({ payloads }) => {\n await this.removeFinalizedTransactions(payloads)\n })\n\n // On new rejected blocks, remove the transactions from the curated archivist\n this.rejectedTransactionsArchivist.on('inserted', async ({ payloads }) => {\n await this.removeRejectedTransactions(payloads)\n })\n\n const pendingTransactionsGauge = this.meter?.createObservableGauge(\n 'xyo_pending_transactions_count',\n { description: 'The current number of pending transactions', valueType: ValueType.INT },\n )\n pendingTransactionsGauge?.addCallback((observer) => {\n observer.observe(this.pendingTransactionsCount)\n })\n }\n\n async getPendingTransactions(head: Hash, limit: number): Promise<HydratedTransactionWithStorageMeta[]> {\n return await this.spanAsync('getPendingTransactions', async () => {\n return await this._updateCuratedPendingTransactionsArchivistMutex.runExclusive(async () => {\n const foundPendingTransactions: WithStorageMeta<TransactionBoundWitness>[] = []\n let cursor: Sequence | undefined\n while (foundPendingTransactions.length < limit) {\n const payloads = await this.pendingTransactionsLocalArchivist.next({\n limit: 100, order: 'asc', cursor,\n })\n if (payloads.length === 0) break\n cursor = payloads.at(-1)?._sequence\n const transactions = payloads.map(payload => asOptionalTransactionBoundWitnessWithStorageMeta(payload)).filter(exists)\n foundPendingTransactions.push(...transactions)\n }\n const hydratedTransactions = await Promise.all(\n foundPendingTransactions.map(tx => tryHydrateTransaction(this.pendingTransactionsLocalArchivist, tx._hash)),\n )\n return hydratedTransactions.filter(exists)\n }, XyoPendingTransactionsService.MutexPriority.ReadTransactions)\n })\n }\n\n private async countPendingTransactions() {\n if (this._countPendingTransactionsMutex.isLocked()) return\n await this._countPendingTransactionsMutex.runExclusive(async () => {\n const payloads = (await this._curatedPendingTransactionsArchivist?.all()) ?? []\n const pendingTransactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta)\n this._pendingTransactionsCount = pendingTransactions.length\n })\n }\n\n private async filterAlreadyFinalizedTransactions(payloads: WithStorageMeta<Payload>[]): Promise<WithStorageMeta<TransactionBoundWitness>[]> {\n const incomingTransactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta)\n const incomingTransactionHashes = incomingTransactions.map(payload => payload._hash)\n const finalizedTransactions = await this.chainArchivist.get(incomingTransactionHashes)\n const finalizedTransactionHashes = new Set(finalizedTransactions.map(item => item._hash))\n const nonFinalizedTransactions = incomingTransactions.filter(item => !finalizedTransactionHashes.has(item._hash))\n return nonFinalizedTransactions\n }\n\n private async insertNewTransactions(payloads: WithStorageMeta<Payload>[]) {\n return await this.spanAsync('InsertNewTransactions', async () => {\n return await this._updateCuratedPendingTransactionsArchivistMutex.runExclusive(async () => {\n // Check incoming transactions against finalized transactions\n const unprocessedTransactions = await this.filterAlreadyFinalizedTransactions(payloads)\n // Hydrate all unprocessed transactions\n const hydratedUnprocessedTransactions = (await Promise.all(unprocessedTransactions.map((tx) => {\n return tryHydrateTransaction(this.pendingTransactionsArchivist, tx._hash)\n }))).filter(exists)\n // Filter to only valid transactions\n const validTransactions = await filterAsync(hydratedUnprocessedTransactions, async (tx) => {\n const errors = await validateTransaction(tx, this.chainIdentification.id)\n return errors.length > 0 ? false : true\n })\n await this.pendingTransactionsLocalArchivist.insert(flattenHydratedTransactions(validTransactions))\n }, XyoPendingTransactionsService.MutexPriority.InsertNewTransactions)\n })\n }\n\n private async removeFinalizedTransactions(payloads: WithStorageMeta<Payload>[]) {\n return await this.removeTransactions(payloads, 'RemoveFinalizedTransactions')\n }\n\n private async removeRejectedTransactions(payloads: WithStorageMeta<Payload>[]) {\n return await this.removeTransactions(payloads, 'RemoveRejectedTransactions')\n }\n\n private async removeTransactions(\n payloads: WithStorageMeta<Payload>[],\n priority: keyof typeof XyoPendingTransactionsService.MutexPriority,\n ) {\n return await this.spanAsync(priority, async () => {\n return await this._updateCuratedPendingTransactionsArchivistMutex.runExclusive(async () => {\n const transactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta)\n await this.pendingTransactionsLocalArchivist.delete(transactions.map(tx => tx._hash))\n }, XyoPendingTransactionsService.MutexPriority[priority])\n })\n }\n}\n","import { filterAs } from '@xylabs/array'\nimport { assertEx } from '@xylabs/assert'\nimport {\n Address, asAddress, Hash,\n} from '@xylabs/hex'\nimport { ArchivistInstance, ArchivistNextOptions } from '@xyo-network/archivist-model'\nimport {\n analyzeChain, ChainStakeIntentAnalyzer,\n isChainSummaryStakeIntent,\n} from '@xyo-network/chain-analyze'\nimport {\n DEFAULT_FIND_FIRST_MATCHING_NEXT_OPTIONS,\n findFirstMatching, IntervalMap,\n SerializedIntervalMap,\n} from '@xyo-network/chain-utils'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport { Payload, WithStorageMeta } from '@xyo-network/payload-model'\nimport type {\n BaseServiceParams, EventingChainBlockNumberIterator, Intent,\n} from '@xyo-network/xl1-protocol'\nimport {\n asOptionalBlockBoundWitness, asOptionalBlockBoundWitnessWithStorageMeta,\n asOptionalChainIndexingServiceStateWithStorageMeta, asOptionalChainStakeIntent, ChainIndexingServiceState, ChainIndexingServiceStateSchema, ChainStakeViewer,\n isChainIndexingServiceState,\n StakeIntentService,\n} from '@xyo-network/xl1-protocol'\nimport { Mutex } from 'async-mutex'\nimport { LRUCache } from 'lru-cache'\n\nimport { BaseService, creatableService } from '../BaseService.ts'\n\nexport interface XyoStakeIntentServiceParams extends BaseServiceParams {\n chainArchivist?: ArchivistInstance\n chainIterator?: EventingChainBlockNumberIterator\n chainStakeViewer?: ChainStakeViewer\n stakeIntentStateArchivist?: ArchivistInstance\n}\n\n/**\n * The number of blocks to periodically persist state\n */\nconst STATE_PERSISTENCE_INTERVAL = 60n\n\nconst ACTIVE_STAKE_TTL = 1000 * 60 * 60 * 2 // 2 hours in milliseconds\nconst NO_ACTIVE_STAKE_TTL = 1000 * 2 // 2 seconds in milliseconds\nconst STAKE_CACHE_MAX_ENTRIES = 10_000\n\n@creatableService()\nexport class XyoStakeIntentService extends BaseService<XyoStakeIntentServiceParams> implements StakeIntentService {\n // TODO: Use hash instead of block number to handle chain reorgs\n protected _lastIndexedBlockHash: Hash | undefined = undefined\n // TODO: Interval tree per declaration (bank, validator, etc.)\n // NOTE: Ideally move to DataIntervalTree to handle declared\n // ranges as it enables range queries in O(min(n, k * log n)) time,\n // where k is the number of intervals in the output list time\n // Currently using set based because it's simpler, equivalent\n // in performance for small sets, and (most importantly) easily\n // persisted so we can recover state on restart.\n protected _producers: IntervalMap<Address> = new IntervalMap()\n protected _stakeCache = new LRUCache<Address, bigint>({ max: STAKE_CACHE_MAX_ENTRIES })\n protected _updateMutex = new Mutex()\n\n constructor(params: XyoStakeIntentServiceParams) {\n super(params)\n this.chainIterator.on('headUpdated', async () => {\n await this.updateIndex()\n })\n }\n\n protected get chainArchivist() {\n return assertEx(this.params.chainArchivist, () => 'chainArchivist not set')\n }\n\n protected get chainIterator() {\n return assertEx(this.params.chainIterator, () => 'chainIterator not set')\n }\n\n protected get chainStakeViewer() {\n return assertEx(this.params.chainStakeViewer, () => 'chainStakeViewer not set')\n }\n\n protected get stakeIntentStateArchivist() {\n return assertEx(this.params.stakeIntentStateArchivist, () => 'stakeIntentStateArchivist not set')\n }\n\n override async createHandler(): Promise<void> {\n const head = await this.chainIterator.head()\n const headHash = await PayloadBuilder.hash(head)\n if (head?.block === undefined) return\n await this.recoverState(headHash)\n await this.updateIndex(true)\n }\n\n async getDeclaredCandidateRanges(address: Address, intent: Intent): Promise<Readonly<Readonly<[number, number]>[]>> {\n await Promise.resolve()\n assertEx(intent === 'producer', () => `Error: Support not yet added for intent ${intent}`)\n const results = this._producers.get(address)\n return results ?? []\n }\n\n async getDeclaredCandidatesForBlock(block: number, intent: Intent): Promise<Address[]> {\n return await this.spanAsync('getDeclaredCandidatesForBlock', async () => {\n assertEx(intent === 'producer', () => `Error: Support not yet added for intent ${intent}`)\n const results = this._producers.findAllContaining(block)\n const candidates = [...results]\n const validCandidates = await this.filterToValidStake(candidates, this.chainStakeViewer)\n return validCandidates\n })\n }\n\n async isStakedForBlock(block: number, intent: Intent, address: Address): Promise<boolean> {\n const candidates = await this.getDeclaredCandidatesForBlock(block, intent)\n return candidates.includes(address)\n }\n\n private async filterToValidStake(\n candidates: Address[],\n chainStakeViewer: ChainStakeViewer,\n requiredMinimumStake: bigint = 1n,\n ): Promise<Address[]> {\n type CandidateStake = { candidate: Address; stake: bigint }\n\n // Find the stake for each candidate\n const candidatesWithStake: CandidateStake[] = await Promise.all(\n candidates.map(async (candidate) => {\n // Check if the stake is already cached\n const stake = this._stakeCache.get(candidate)\n if (stake === undefined) {\n // Fetch from chainStakeViewer if not cached\n const activeStake = await chainStakeViewer.activeByAddressStaked(`${candidate}`)\n if (activeStake > 0n) {\n // Store result in cache\n this._stakeCache.set(candidate, activeStake, { ttl: ACTIVE_STAKE_TTL })\n } else {\n this._stakeCache.set(candidate, activeStake, { ttl: NO_ACTIVE_STAKE_TTL })\n }\n return { candidate, stake: activeStake }\n } else {\n return { candidate, stake }\n }\n }),\n )\n\n // Filter out candidates whose stake is greater than or equal to the required minimum\n return candidatesWithStake\n .filter(({ stake }) => stake >= requiredMinimumStake)\n .map(({ candidate }) => candidate)\n }\n\n private async persistState(current: Hash): Promise<void> {\n const state = this._producers.serialize()\n const payload = new PayloadBuilder<ChainIndexingServiceState<SerializedIntervalMap>>({ schema: ChainIndexingServiceStateSchema })\n .fields({ endBlockHash: current, state })\n .build()\n await this.stakeIntentStateArchivist.insert([payload])\n }\n\n private async recoverState(current: Hash): Promise<void> {\n const currentBlock = assertEx(asOptionalBlockBoundWitness((await this.chainArchivist.get([current]))?.[0]), () => `Block ${current} not found`)\n const currentBlockNum = currentBlock.block\n // Find last state before current head (in case of rollback, we indexed past it on an uncle chain, etc.)\n const opts: ArchivistNextOptions = { ...DEFAULT_FIND_FIRST_MATCHING_NEXT_OPTIONS }\n while (true) {\n const predicate = (p: WithStorageMeta<Payload>) => {\n const state = asOptionalChainIndexingServiceStateWithStorageMeta(p)\n return state ? true : false\n }\n const state = await findFirstMatching(this.stakeIntentStateArchivist, predicate, opts)\n if (isChainIndexingServiceState<SerializedIntervalMap>(state)) {\n const indexed = (await this.chainArchivist.get([state.endBlockHash]))?.[0]\n const indexedBlock = asOptionalBlockBoundWitnessWithStorageMeta(indexed)\n if (indexedBlock) {\n const indexedBlockNum = indexedBlock.block\n if (indexedBlockNum <= currentBlockNum) {\n const data = state.state as SerializedIntervalMap\n this._producers = new IntervalMap(data)\n this._lastIndexedBlockHash = indexedBlock._hash\n break\n }\n }\n } else {\n // No state found, start from genesis\n break\n }\n opts.open = true\n }\n }\n\n private async updateIndex(displayProgress = false): Promise<void> {\n if (this._updateMutex.isLocked()) {\n return\n }\n await this._updateMutex.runExclusive(async () => {\n return await this.spanAsync('updateIndex', async () => {\n const currentHead = await this.chainIterator.head()\n const currentHeadHash = await PayloadBuilder.hash(currentHead)\n const result = await analyzeChain(this.chainArchivist, [new ChainStakeIntentAnalyzer('producer')], currentHeadHash, this._lastIndexedBlockHash)\n const signedDeclarations = filterAs(result.find(isChainSummaryStakeIntent)?.intents ?? [], asOptionalChainStakeIntent)\n if (currentHead.block === undefined) return\n const currentHeadBlockNum = currentHead.block\n if (displayProgress) this.logger?.info(`Updating index through 0x${currentHeadBlockNum}`)\n for (const signedDeclaration of signedDeclarations) {\n const { exp, nbf } = signedDeclaration\n const start = nbf\n const stop = exp\n const address = asAddress(signedDeclaration?.from)\n if (start !== undefined && stop !== undefined && address !== undefined) {\n this._producers.insert(address, start, stop)\n }\n }\n /*\n if (index % STATE_PERSISTENCE_INTERVAL === 0n) {\n if (displayProgress) this.logger?.info(`Persisting state at block ${index}`)\n await this.persistState(await PayloadBuilder.hash(block))\n }\n */\n this._lastIndexedBlockHash = currentHeadHash\n })\n })\n }\n}\n","import { Base } from '@xylabs/base'\nimport type { Address } from '@xylabs/hex'\nimport { toAddress } from '@xylabs/hex'\nimport type { Promisable } from '@xylabs/promise'\nimport type { StakedXyoChain } from '@xyo-network/typechain'\nimport { StakedXyoChain__factory as StakedXyoChainFactory } from '@xyo-network/typechain'\nimport type { BaseServiceParams, ChainService } from '@xyo-network/xl1-protocol'\nimport { getAddress } from 'ethers/address'\nimport type { ContractRunner } from 'ethers/providers'\n\nexport interface EvmChainServiceParams extends BaseServiceParams {\n contract: StakedXyoChain\n id: Address\n runner: ContractRunner\n}\n\n/**\n * A class that represents a chain stake as backed by an EVM smart contract\n */\nexport class EvmChainService extends Base<EvmChainServiceParams> implements ChainService {\n private _contract: StakedXyoChain\n\n protected constructor(params: EvmChainServiceParams) {\n super(params)\n this._contract = params.contract\n }\n\n get id(): Address {\n return this.params.id\n }\n\n get runner(): ContractRunner {\n return this.params.runner\n }\n\n static create({\n id, runner, logger, traceProvider, meterProvider,\n }: Omit<EvmChainServiceParams, 'contract'>): Promisable<EvmChainService> {\n const contractAddress = getAddress(id)\n const contract: StakedXyoChain = StakedXyoChainFactory.connect(contractAddress, runner)\n return new this({\n id, contract, runner, logger, traceProvider, meterProvider,\n })\n }\n\n async active(): Promise<bigint> {\n return await this._contract.active()\n }\n\n async activeByAddressStaked(address: string): Promise<bigint> {\n return await this._contract.activeByAddressStaked(getAddress(address))\n }\n\n async activeByStaker(address: string): Promise<bigint> {\n return await this._contract.activeByStaker(getAddress(address))\n }\n\n async addStake(staked: string, amount: bigint): Promise<boolean> {\n const result = await this._contract.addStake(getAddress(staked), amount)\n await result.wait()\n return true\n }\n\n async chainId(): Promise<Address> {\n return toAddress(await this._contract.chainId())\n }\n\n async forkedAtBlockNumber(): Promise<bigint> {\n return await this._contract.forkedAtBlockNumber()\n }\n\n async forkedAtHash(): Promise<bigint> {\n return await this._contract.forkedAtHash()\n }\n\n async forkedChainId(): Promise<Address> {\n return toAddress(await this._contract.forkedChainId())\n }\n\n async minWithdrawalBlocks(): Promise<bigint> {\n return await this._contract.minWithdrawalBlocks()\n }\n\n async pending(): Promise<bigint> {\n return await this._contract.pending()\n }\n\n async pendingByStaker(staker: string): Promise<bigint> {\n return await this._contract.pendingByStaker(getAddress(staker))\n }\n\n async removeStake(slot: bigint): Promise<boolean> {\n const result = await this._contract.removeStake(slot)\n await result.wait()\n return true\n }\n\n async rewardsContract(): Promise<string> {\n return await this._contract.rewardsContract()\n }\n\n async stakingTokenAddress(): Promise<string> {\n return await this._contract.stakingTokenAddress()\n }\n\n async withdrawStake(slot: bigint): Promise<boolean> {\n const result = await this._contract.withdrawStake(slot)\n await result.wait()\n return true\n }\n\n async withdrawn(): Promise<bigint> {\n return await this._contract.withdrawn()\n }\n\n async withdrawnByStaker(staker: string): Promise<bigint> {\n return await this._contract.withdrawnByStaker(getAddress(staker))\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport type { BaseParams } from '@xylabs/base'\nimport type { ArchivistInstance } from '@xyo-network/archivist-model'\nimport { BaseEmitter } from '@xyo-network/module-event-emitter'\nimport { PayloadBuilder } from '@xyo-network/payload-builder'\nimport type {\n BlockBoundWitness, ChainIdentification,\n ChainIteratorEventData,\n EventingChainBlockNumberIterator,\n XyoChainIteratorParams,\n} from '@xyo-network/xl1-protocol'\nimport {\n asBlockBoundWitness, asOptionalBlockBoundWitness,\n isBlockBoundWitness,\n} from '@xyo-network/xl1-protocol'\nimport { LRUCache } from 'lru-cache'\n\nexport class XyoChainBlockNumberIterator extends BaseEmitter<BaseParams, ChainIteratorEventData> implements EventingChainBlockNumberIterator {\n protected _blocksByBlockNumber = new LRUCache<number, BlockBoundWitness>({ max: 10_000 })\n protected _chainArchivist: ArchivistInstance\n protected _chainInformation: ChainIdentification\n protected _head: BlockBoundWitness\n\n protected constructor(params: XyoChainIteratorParams) {\n super({})\n const { chainArchivist: archivist, head } = params\n this._chainArchivist = archivist\n this._head = head\n this._chainInformation = { id: head.chain }\n }\n\n get archivist(): ArchivistInstance { return this._chainArchivist }\n\n get chainIdentification(): ChainIdentification { return this._chainInformation }\n\n static async create(params: XyoChainIteratorParams): Promise<XyoChainBlockNumberIterator> {\n await Promise.resolve()\n return new XyoChainBlockNumberIterator(params)\n }\n\n async get(block: number): Promise<BlockBoundWitness> {\n // Bail early if the block requested is newer than the current head\n assertEx(this._head.block >= block, () => `Block requested is newer than the current head [${block}]`)\n const cached = this._blocksByBlockNumber.get(block)\n if (cached) return cached\n // Start at the current head and traverse backwards until the requested block is found\n const startingBlock = this._head\n const currentBlockHash = await PayloadBuilder.hash(startingBlock)\n let [currentBlock] = await this._chainArchivist.get([currentBlockHash])\n while (currentBlock) {\n assertEx(asBlockBoundWitness(currentBlock), () => `Expected hash to be a block bound witness [${currentBlock._hash}]`)\n if (isBlockBoundWitness(currentBlock)) {\n this._blocksByBlockNumber.set(currentBlock.block, currentBlock)\n if (currentBlock.block === block) {\n return currentBlock\n }\n const { previous } = currentBlock\n if (!previous) break\n [currentBlock] = await this._chainArchivist.get([previous])\n }\n }\n throw new Error(`Block not found: ${block}`)\n }\n\n async head(): Promise<BlockBoundWitness> {\n return await Promise.resolve(this._head)\n }\n\n async next(block: number): Promise<BlockBoundWitness | undefined> {\n const currentBlock = block\n const nextBlockNumber = currentBlock + 1\n return await this.get(nextBlockNumber)\n }\n\n // TODO: Decide on inclusive/exclusive (probably need inclusive to account for chain head)\n // and then communicate via method name and documentation\n async previous(block: number | undefined = undefined, count: number = 1): Promise<BlockBoundWitness[]> {\n const results: BlockBoundWitness[] = []\n let currentBlock: BlockBoundWitness | undefined = block ? (await this.get(block)) : this._head\n while (currentBlock && results.length < count) {\n if (isBlockBoundWitness(currentBlock)) {\n results.push(currentBlock)\n const { previous } = currentBlock\n if (!previous) break\n const nextBlock = await this._chainArchivist.get([previous])\n currentBlock = asOptionalBlockBoundWitness(nextBlock[0])\n } else {\n const hash = PayloadBuilder.hash(currentBlock)\n assertEx(asBlockBoundWitness(currentBlock), () => `Expected hash to be a block bound witness [${hash}]`)\n }\n }\n return results\n }\n\n async updateHead(head: BlockBoundWitness): Promise<void> {\n // Async to allow for re-indexing, etc.\n await Promise.resolve()\n this._head = head\n void this.emit('headUpdated', { blocks: [head] })\n }\n}\n"],"mappings":"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","names":["assertEx","toHex","analyzeChain","BalanceAnalyzer","isChainSummaryBalances","toPositiveBigInt","trace","span","spanAsync","BaseEmitter","Mutex","BaseService","BaseEmitter","singletonInitMutex","Mutex","singletons","globalThis","meter","params","meterProvider","getMeter","name","constructor","tracer","traceProvider","getTracer","trace","create","result","Error","createHandler","initSingleton","runExclusive","span","fn","spanAsync","BaseAccountableService","creatableService","XyoChainAccountBalanceService","BaseService","_balances","_previousHead","chainArchivist","assertEx","params","getBalance","address","toPositiveBigInt","positive","toHex","getBalances","result","entries","Object","balance","sync","head","analysis","analyzeChain","BalanceAnalyzer","tracer","find","isChainSummaryBalances","balances","BaseEthProvider","eth_accounts","Error","eth_blockNumber","eth_call","_transaction","_blockNumber","eth_clearSubscriptions","_keepSyncing","eth_coinbase","eth_compileLLL","_code","eth_compileSerpent","eth_compileSolidity","eth_estimateGas","eth_feeHistory","_blockCount","_newestBlock","_rewardPercentiles","eth_gasPrice","eth_getBalance","_address","eth_getBlockByHash","_blockHash","_hydrated","eth_getBlockByNumber","eth_getBlockTransactionCountByHash","eth_getBlockTransactionCountByNumber","eth_getCode","eth_getCompilers","eth_getFilterChanges","_filterIdentifier","eth_getFilterLogs","eth_getLogs","_filter","eth_getStorageAt","_storageSlot","eth_getTransactionByBlockHashAndIndex","_transactionIndex","eth_getTransactionByBlockNumberAndIndex","eth_getTransactionByHash","_transactionHash","eth_getTransactionCount","eth_getTransactionReceipt","eth_getUncleByBlockHashAndIndex","_uncleIndex","eth_getUncleByBlockNumberAndIndex","eth_getUncleCountByBlockHash","eth_getUncleCountByBlockNumber","eth_getWork","eth_hashrate","eth_maxPriorityFeePerGas","eth_mining","eth_newBlockFilter","eth_newFilter","eth_newPendingTransactionFilter","eth_protocolVersion","eth_sendRawTransaction","eth_sendTransaction","eth_sign","_message","eth_signTransaction","eth_submitHashrate","_hashRate","_id","eth_submitWork","_nonce","_hash","_digest","eth_subscribe","_params","eth_syncing","eth_uninstallFilter","eth_unsubscribe","_subscriptionId","assertEx","toHex","FixedPercentageBlockRewardDiviner","FixedPercentageBlockRewardDivinerConfigSchema","buildBlock","PayloadBuilder","asBlockBoundWitness","BlockNumberSchema","ChainStakeIntentSchema","hexFromBigInt","HydratedTransactionWrapper","TransferSchema","generateTransactionTransfers","address","transactions","txs","Promise","all","map","tx","HydratedTransactionWrapper","parse","txCosts","existingItem","find","addr","boundWitness","from","fees","base","push","payloads","amount","payload","schema","TransferSchema","epoch","Date","now","transfers","hexFromBigInt","DEFAULT_BLOCK_SIZE","XYO_PRODUCER_RESTAKE_DURATION","XYO_PRODUCER_RESTAKE_WINDOW","XyoBlockProducer","BaseService","_blockRewardDiviner","DefaultBlockSize","address","account","assertEx","params","chainFinalizedArchivist","chainInfo","chainInformation","electionService","pendingTransactions","pendingTransactionsService","rejectedTransactionsArchivist","rewardAddress","rewardService","stakeIntentService","validateHydratedBlockState","next","head","chain","id","leaders","getCreatorCommitteeForNextBlock","proposeNextValidBlock","getBlockRewardTransfer","block","FixedPercentageBlockRewardDiviner","create","blockRewardService","config","rewardPercentageRatio","schema","FixedPercentageBlockRewardDivinerConfigSchema","blockHex","toHex","blockId","PayloadBuilder","BlockNumberSchema","fields","build","rewards","divine","reward","getProducerRedeclaration","redeclareIntent","process","env","XYO_PRODUCER_REDECLARE_INTENT","ranges","getDeclaredCandidateRanges","lastRange","toSorted","a","b","at","currentDeclarationEnd","currentBlock","timeToProducerExpiration","intent","ChainStakeIntentSchema","from","nbf","exp","spanAsync","previousBlock","asBlockBoundWitness","nextBlock","nextBlockTransactions","getPendingTransactions","_hash","blockPayloads","producerRedeclarationPayload","push","length","rewardTransferPayload","transactionTransfers","generateTransactionTransfers","buildBlock","console","log","errors","logger","error","rejectedTransactions","insert","assertEx","XyoChainRewards__factory","XyoChainRewardsFactory","toEthAddress","EvmBlockRewardService","BaseService","_contractAddress","chainService","assertEx","params","contractAddress","provider","createHandler","rewardsContract","getRewardForBlock","blockNumber","contract","XyoChainRewardsFactory","connect","toEthAddress","calcBlockReward","assertEx","toFixedPoint","rewardFromBlockNumber","XyoBlockRewardService","BaseService","rewardFromBlockNumber","constructor","creatorReward","toFixedPoint","initialStepReward","minRewardPerBlock","stepFactorDenominator","stepFactorNumerator","stepSize","assertEx","params","initialReward","getRewardForBlock","blockNumber","assertEx","PayloadBuilder","addStorageMeta","tx","payloads","PayloadBuilder","XyoValidator","BaseService","address","account","assertEx","params","chainFinalizedArchivist","chainInfo","chainInformation","electionService","pendingTransactions","rewardService","validatePendingBlock","_block","validatePendingTransaction","hydratedTransaction","get","_hash","length","Promise","resolve","assertEx","hexToLast4BytesInt","shuffleWithSeed","PayloadBuilder","XyoElectionService","BaseService","constructor","params","chainIterator","assertEx","chainStakeViewer","stakeIntentService","getCreatorCommitteeForNextBlock","current","spanAsync","nextBlock","block","candidates","getDeclaredCandidatesForBlock","previousBlockHash","PayloadBuilder","hash","generateCreatorCommittee","maxSize","creators","Set","seed","hexToLast4BytesInt","creatorArray","shuffleWithSeed","slice","ValueType","filterAs","filterAsync","assertEx","exists","forget","MemoryArchivist","validateTransaction","asOptionalTransactionBoundWitnessWithStorageMeta","flattenHydratedTransactions","tryHydrateTransaction","Mutex","XyoPendingTransactionsService","BaseService","MutexPriority","InsertNewTransactions","ReadTransactions","RemoveRejectedTransactions","RemoveFinalizedTransactions","_countPendingTransactionsMutex","Mutex","_curatedPendingTransactionsArchivist","_pendingTransactionsCount","_updateCuratedPendingTransactionsArchivistMutex","chainArchivist","assertEx","params","chainIdentification","pendingTransactionsArchivist","pendingTransactionsCount","forget","countPendingTransactions","pendingTransactionsLocalArchivist","rejectedTransactionsArchivist","createHandler","MemoryArchivist","create","account","on","payloads","insertNewTransactions","removeFinalizedTransactions","removeRejectedTransactions","pendingTransactionsGauge","meter","createObservableGauge","description","valueType","ValueType","INT","addCallback","observer","observe","getPendingTransactions","head","limit","spanAsync","runExclusive","foundPendingTransactions","cursor","length","next","order","at","_sequence","transactions","map","payload","asOptionalTransactionBoundWitnessWithStorageMeta","filter","exists","push","hydratedTransactions","Promise","all","tx","tryHydrateTransaction","_hash","isLocked","pendingTransactions","filterAs","filterAlreadyFinalizedTransactions","incomingTransactions","incomingTransactionHashes","finalizedTransactions","get","finalizedTransactionHashes","Set","item","nonFinalizedTransactions","has","unprocessedTransactions","hydratedUnprocessedTransactions","validTransactions","filterAsync","errors","validateTransaction","id","insert","flattenHydratedTransactions","removeTransactions","priority","delete","filterAs","assertEx","asAddress","analyzeChain","ChainStakeIntentAnalyzer","isChainSummaryStakeIntent","DEFAULT_FIND_FIRST_MATCHING_NEXT_OPTIONS","findFirstMatching","IntervalMap","PayloadBuilder","asOptionalBlockBoundWitness","asOptionalBlockBoundWitnessWithStorageMeta","asOptionalChainIndexingServiceStateWithStorageMeta","asOptionalChainStakeIntent","ChainIndexingServiceStateSchema","isChainIndexingServiceState","Mutex","LRUCache","ACTIVE_STAKE_TTL","NO_ACTIVE_STAKE_TTL","STAKE_CACHE_MAX_ENTRIES","XyoStakeIntentService","BaseService","_lastIndexedBlockHash","undefined","_producers","IntervalMap","_stakeCache","LRUCache","max","_updateMutex","Mutex","constructor","params","chainIterator","on","updateIndex","chainArchivist","assertEx","chainStakeViewer","stakeIntentStateArchivist","createHandler","head","headHash","PayloadBuilder","hash","block","recoverState","getDeclaredCandidateRanges","address","intent","Promise","resolve","results","get","getDeclaredCandidatesForBlock","spanAsync","findAllContaining","candidates","validCandidates","filterToValidStake","isStakedForBlock","includes","requiredMinimumStake","candidatesWithStake","all","map","candidate","stake","activeStake","activeByAddressStaked","set","ttl","filter","persistState","current","state","serialize","payload","schema","ChainIndexingServiceStateSchema","fields","endBlockHash","build","insert","currentBlock","asOptionalBlockBoundWitness","currentBlockNum","opts","DEFAULT_FIND_FIRST_MATCHING_NEXT_OPTIONS","predicate","p","asOptionalChainIndexingServiceStateWithStorageMeta","findFirstMatching","isChainIndexingServiceState","indexed","indexedBlock","asOptionalBlockBoundWitnessWithStorageMeta","indexedBlockNum","data","_hash","open","displayProgress","isLocked","runExclusive","currentHead","currentHeadHash","result","analyzeChain","ChainStakeIntentAnalyzer","signedDeclarations","filterAs","find","isChainSummaryStakeIntent","intents","asOptionalChainStakeIntent","currentHeadBlockNum","logger","info","signedDeclaration","exp","nbf","start","stop","asAddress","from","Base","toAddress","StakedXyoChain__factory","StakedXyoChainFactory","getAddress","EvmChainService","Base","_contract","params","contract","id","runner","create","logger","traceProvider","meterProvider","contractAddress","getAddress","StakedXyoChainFactory","connect","active","activeByAddressStaked","address","activeByStaker","addStake","staked","amount","result","wait","chainId","toAddress","forkedAtBlockNumber","forkedAtHash","forkedChainId","minWithdrawalBlocks","pending","pendingByStaker","staker","removeStake","slot","rewardsContract","stakingTokenAddress","withdrawStake","withdrawn","withdrawnByStaker","assertEx","BaseEmitter","PayloadBuilder","asBlockBoundWitness","asOptionalBlockBoundWitness","isBlockBoundWitness","LRUCache","XyoChainBlockNumberIterator","BaseEmitter","_blocksByBlockNumber","LRUCache","max","_chainArchivist","_chainInformation","_head","params","chainArchivist","archivist","head","id","chain","chainIdentification","create","Promise","resolve","get","block","assertEx","cached","startingBlock","currentBlockHash","PayloadBuilder","hash","currentBlock","asBlockBoundWitness","_hash","isBlockBoundWitness","set","previous","Error","next","nextBlockNumber","undefined","count","results","length","push","nextBlock","asOptionalBlockBoundWitness","updateHead","emit","blocks"]}
@@ -1,3 +1,4 @@
1
+ import type { Tracer } from '@opentelemetry/api';
1
2
  import type { Promisable } from '@xylabs/promise';
2
3
  import { BaseEmitter } from '@xyo-network/module-event-emitter';
3
4
  import type { BaseAccountableServiceParams, BaseServiceParams, Service } from '@xyo-network/xl1-protocol';
@@ -9,12 +10,12 @@ export declare class BaseService<TParams extends BaseServiceParams = BaseService
9
10
  static get singletons(): Record<string, unknown>;
10
11
  get meter(): import("@opentelemetry/api").Meter | undefined;
11
12
  get name(): string;
12
- get tracer(): import("@opentelemetry/api").Tracer | undefined;
13
+ get tracer(): Tracer;
13
14
  static create<TService extends BaseService<TParams>, TParams extends BaseServiceParams = TService['params']>(this: CreatableService<TService>, params: TParams): Promise<TService>;
14
15
  static initSingleton<TService extends BaseService<TParams>, TParams extends BaseServiceParams>(params: TParams): Promise<TService>;
15
16
  createHandler(): Promisable<void>;
16
17
  span<T>(name: string, fn: () => T): T;
17
- spanAsync<T>(name: string, fn: () => Promise<T>, timeout?: number): Promise<T>;
18
+ spanAsync<T>(name: string, fn: () => Promise<T>): Promise<T>;
18
19
  }
19
20
  export declare abstract class BaseAccountableService<TParams extends BaseAccountableServiceParams = BaseAccountableServiceParams> extends BaseService<TParams> {
20
21
  }
@@ -1 +1 @@
1
- {"version":3,"file":"BaseService.d.ts","sourceRoot":"","sources":["../../src/BaseService.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAA;AAEjD,OAAO,EAAE,WAAW,EAAE,MAAM,mCAAmC,CAAA;AAC/D,OAAO,KAAK,EACV,4BAA4B,EAAE,iBAAiB,EAAE,OAAO,EACzD,MAAM,2BAA2B,CAAA;AAGlC,OAAO,CAAC,MAAM,CAAC;IACb,IAAI,oBAAoB,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAClD;AAED,qBAAa,WAAW,CAAC,OAAO,SAAS,iBAAiB,GAAG,iBAAiB,CAAE,SAAQ,WAAW,CAAC,OAAO,CAAC;IAC1G,OAAO,CAAC,MAAM,CAAC,kBAAkB,CAAc;IAE/C,MAAM,KAAK,UAAU,4BAEpB;IAED,IAAI,KAAK,mDAER;IAED,IAAI,IAAI,WAEP;IAED,IAAI,MAAM,oDAET;WAEY,MAAM,CAAC,QAAQ,SAAS,WAAW,CAAC,OAAO,CAAC,EAAE,OAAO,SAAS,iBAAiB,GAAG,QAAQ,CAAC,QAAQ,CAAC,EAC/G,IAAI,EAAE,gBAAgB,CAAC,QAAQ,CAAC,EAChC,MAAM,EAAE,OAAO,GACd,OAAO,CAAC,QAAQ,CAAC;IAOpB,MAAM,CAAC,aAAa,CAAC,QAAQ,SAAS,WAAW,CAAC,OAAO,CAAC,EAAE,OAAO,SAAS,iBAAiB,EAAE,MAAM,EAAE,OAAO;IAO9G,aAAa,IAAI,UAAU,CAAC,IAAI,CAAC;IAIjC,IAAI,CAAC,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,CAAC,GAAG,CAAC;IAI/B,SAAS,CAAC,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,OAAO,CAAC,CAAC,CAAC,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,CAAC,CAAC;CAGrF;AAED,8BAAsB,sBAAsB,CAC1C,OAAO,SAAS,4BAA4B,GAAG,4BAA4B,CAC3E,SAAQ,WAAW,CAAC,OAAO,CAAC;CAAG;AAEjC,MAAM,WAAW,gBAAgB,CAAC,CAAC,SAAS,WAAW,GAAG,WAAW,CAAE,SAAQ,OAAO;IACpF,KAAK,MAAM,EAAE,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAA;IAC5B,MAAM,CAAC,CAAC,SAAS,WAAW,EAAE,IAAI,EAAE,gBAAgB,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC,QAAQ,CAAC,GAAG,UAAU,CAAC,CAAC,CAAC,CAAA;CAC7F;AAED,wBAAgB,gBAAgB,CAAC,CAAC,SAAS,WAAW,GAAG,WAAW,MAC1D,CAAC,SAAS,gBAAgB,CAAC,CAAC,CAAC,EAAE,aAAa,CAAC,UAItD"}
1
+ {"version":3,"file":"BaseService.d.ts","sourceRoot":"","sources":["../../src/BaseService.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAA;AAEhD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAA;AAEjD,OAAO,EAAE,WAAW,EAAE,MAAM,mCAAmC,CAAA;AAC/D,OAAO,KAAK,EACV,4BAA4B,EAAE,iBAAiB,EAAE,OAAO,EACzD,MAAM,2BAA2B,CAAA;AAGlC,OAAO,CAAC,MAAM,CAAC;IACb,IAAI,oBAAoB,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAClD;AAED,qBAAa,WAAW,CAAC,OAAO,SAAS,iBAAiB,GAAG,iBAAiB,CAAE,SAAQ,WAAW,CAAC,OAAO,CAAC;IAC1G,OAAO,CAAC,MAAM,CAAC,kBAAkB,CAAc;IAE/C,MAAM,KAAK,UAAU,4BAEpB;IAED,IAAI,KAAK,mDAER;IAED,IAAI,IAAI,WAEP;IAED,IAAI,MAAM,IAAI,MAAM,CAEnB;WAEY,MAAM,CAAC,QAAQ,SAAS,WAAW,CAAC,OAAO,CAAC,EAAE,OAAO,SAAS,iBAAiB,GAAG,QAAQ,CAAC,QAAQ,CAAC,EAC/G,IAAI,EAAE,gBAAgB,CAAC,QAAQ,CAAC,EAChC,MAAM,EAAE,OAAO,GACd,OAAO,CAAC,QAAQ,CAAC;IAOpB,MAAM,CAAC,aAAa,CAAC,QAAQ,SAAS,WAAW,CAAC,OAAO,CAAC,EAAE,OAAO,SAAS,iBAAiB,EAAE,MAAM,EAAE,OAAO;IAO9G,aAAa,IAAI,UAAU,CAAC,IAAI,CAAC;IAIjC,IAAI,CAAC,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,CAAC,GAAG,CAAC;IAI/B,SAAS,CAAC,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;CAGnE;AAED,8BAAsB,sBAAsB,CAC1C,OAAO,SAAS,4BAA4B,GAAG,4BAA4B,CAC3E,SAAQ,WAAW,CAAC,OAAO,CAAC;CAE7B;AAED,MAAM,WAAW,gBAAgB,CAAC,CAAC,SAAS,WAAW,GAAG,WAAW,CAAE,SAAQ,OAAO;IACpF,KAAI,MAAM,EAAE,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAA;IAC3B,MAAM,CAAC,CAAC,SAAS,WAAW,EAAE,IAAI,EAAE,gBAAgB,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC,QAAQ,CAAC,GAAG,UAAU,CAAC,CAAC,CAAC,CAAA;CAC7F;AAED,wBAAgB,gBAAgB,CAAC,CAAC,SAAS,WAAW,GAAG,WAAW,MAC1D,CAAC,SAAS,gBAAgB,CAAC,CAAC,CAAC,EAAE,aAAa,CAAC,UAItD"}
@@ -10,19 +10,38 @@ export interface XyoPendingTransactionsServiceParams extends BaseServiceParams {
10
10
  }
11
11
  export declare class XyoPendingTransactionsService extends BaseService<XyoPendingTransactionsServiceParams> implements PendingTransactionsService {
12
12
  private static readonly MutexPriority;
13
- private _cleanedToBlock;
14
- private _pendingTransactionsLocalArchivist;
15
- private _pendingTransactionsLocalArchivistCursor;
16
- private _updatePendingTransactionsLocalArchivistMutex;
13
+ /**
14
+ * A mutex to ensure that the counting the number of pending transactions is
15
+ * not called concurrently
16
+ */
17
+ private _countPendingTransactionsMutex;
18
+ /**
19
+ * A local Archivist optimized for fast retrieval that stores only validated
20
+ * pending transactions
21
+ */
22
+ private _curatedPendingTransactionsArchivist;
23
+ /**
24
+ * The last count of total pending transactions
25
+ */
26
+ private _pendingTransactionsCount;
27
+ /**
28
+ * A mutex to ensure that the curated pending transactions archivist is
29
+ * updated in a thread-safe manner
30
+ */
31
+ private _updateCuratedPendingTransactionsArchivistMutex;
17
32
  private get chainArchivist();
18
33
  private get chainIdentification();
19
34
  private get pendingTransactionsArchivist();
35
+ private get pendingTransactionsCount();
20
36
  private get pendingTransactionsLocalArchivist();
21
37
  private get rejectedTransactionsArchivist();
22
38
  createHandler(): Promise<void>;
23
- getPendingTransactions(head: Hash, limit: number, timeout?: number): Promise<HydratedTransactionWithStorageMeta[]>;
39
+ getPendingTransactions(head: Hash, limit: number): Promise<HydratedTransactionWithStorageMeta[]>;
40
+ private countPendingTransactions;
24
41
  private filterAlreadyFinalizedTransactions;
25
- private removeAlreadyIncludedTransactions;
26
- private updatePendingTransactionsLocalArchivist;
42
+ private insertNewTransactions;
43
+ private removeFinalizedTransactions;
44
+ private removeRejectedTransactions;
45
+ private removeTransactions;
27
46
  }
28
47
  //# sourceMappingURL=PendingTransactions.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"PendingTransactions.d.ts","sourceRoot":"","sources":["../../../src/PendingTransactions/PendingTransactions.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,IAAI,EAAE,MAAM,aAAa,CAAA;AAElC,OAAO,EAAE,iBAAiB,EAAE,MAAM,8BAA8B,CAAA;AAMhE,OAAO,EAC6C,iBAAiB,EAAE,mBAAmB,EAAE,kCAAkC,EAC5H,0BAA0B,EAC3B,MAAM,2BAA2B,CAAA;AAIlC,OAAO,EAAE,WAAW,EAAoB,MAAM,mBAAmB,CAAA;AAEjE,MAAM,WAAW,mCAAoC,SAAQ,iBAAiB;IAC5E,cAAc,CAAC,EAAE,iBAAiB,CAAA;IAClC,mBAAmB,CAAC,EAAE,mBAAmB,CAAA;IACzC,4BAA4B,CAAC,EAAE,iBAAiB,CAAA;IAChD,6BAA6B,CAAC,EAAE,iBAAiB,CAAA;CAClD;AAED,qBACa,6BAA8B,SAAQ,WAAW,CAAC,mCAAmC,CAAE,YAAW,0BAA0B;IACvI,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,aAAa,CAiBpC;IAGD,OAAO,CAAC,eAAe,CAAkB;IAGzC,OAAO,CAAC,kCAAkC,CAA6B;IACvE,OAAO,CAAC,wCAAwC,CAAsB;IAEtE,OAAO,CAAC,6CAA6C,CAAc;IAEnE,OAAO,KAAK,cAAc,GAEzB;IAED,OAAO,KAAK,mBAAmB,GAE9B;IAED,OAAO,KAAK,4BAA4B,GAEvC;IAED,OAAO,KAAK,iCAAiC,GAE5C;IAED,OAAO,KAAK,6BAA6B,GAExC;IAEc,aAAa;IAoCtB,sBAAsB,CAAC,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,SAAY,GAAG,OAAO,CAAC,kCAAkC,EAAE,CAAC;YAwB7G,kCAAkC;YASlC,iCAAiC;YAejC,uCAAuC;CA8BtD"}
1
+ {"version":3,"file":"PendingTransactions.d.ts","sourceRoot":"","sources":["../../../src/PendingTransactions/PendingTransactions.ts"],"names":[],"mappings":"AAKA,OAAO,EAAE,IAAI,EAAE,MAAM,aAAa,CAAA;AAElC,OAAO,EAAE,iBAAiB,EAAE,MAAM,8BAA8B,CAAA;AAKhE,OAAO,EAC6C,iBAAiB,EAAE,mBAAmB,EAAE,kCAAkC,EAC5H,0BAA0B,EAC3B,MAAM,2BAA2B,CAAA;AAIlC,OAAO,EAAE,WAAW,EAAoB,MAAM,mBAAmB,CAAA;AAEjE,MAAM,WAAW,mCAAoC,SAAQ,iBAAiB;IAC5E,cAAc,CAAC,EAAE,iBAAiB,CAAA;IAClC,mBAAmB,CAAC,EAAE,mBAAmB,CAAA;IACzC,4BAA4B,CAAC,EAAE,iBAAiB,CAAA;IAChD,6BAA6B,CAAC,EAAE,iBAAiB,CAAA;CAClD;AAED,qBACa,6BAA8B,SAAQ,WAAW,CAAC,mCAAmC,CAAE,YAAW,0BAA0B;IACvI,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,aAAa,CAiB3B;IAEV;;;OAGG;IACH,OAAO,CAAC,8BAA8B,CAAc;IAEpD;;;OAGG;IACH,OAAO,CAAC,oCAAoC,CAA6B;IAEzE;;OAEG;IACH,OAAO,CAAC,yBAAyB,CAAY;IAE7C;;;OAGG;IACH,OAAO,CAAC,+CAA+C,CAAc;IAErE,OAAO,KAAK,cAAc,GAEzB;IAED,OAAO,KAAK,mBAAmB,GAE9B;IAED,OAAO,KAAK,4BAA4B,GAEvC;IAED,OAAO,KAAK,wBAAwB,GAGnC;IAED,OAAO,KAAK,iCAAiC,GAE5C;IAED,OAAO,KAAK,6BAA6B,GAExC;IAEc,aAAa;IA4BtB,sBAAsB,CAAC,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,kCAAkC,EAAE,CAAC;YAsBxF,wBAAwB;YASxB,kCAAkC;YASlC,qBAAqB;YAmBrB,2BAA2B;YAI3B,0BAA0B;YAI1B,kBAAkB;CAWjC"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "$schema": "http://json.schemastore.org/package.json",
3
3
  "name": "@xyo-network/chain-services",
4
- "version": "1.3.17",
4
+ "version": "1.3.18",
5
5
  "description": "XYO Layer One SDK Services",
6
6
  "homepage": "https://xylabs.com",
7
7
  "bugs": {
@@ -34,48 +34,48 @@
34
34
  "deploy3": "echo Deploy3 not allowed!"
35
35
  },
36
36
  "dependencies": {
37
- "@xylabs/array": "^4.9.3",
38
- "@xylabs/assert": "^4.9.3",
39
- "@xylabs/base": "^4.9.3",
40
- "@xylabs/decimal-precision": "^4.9.3",
41
- "@xylabs/exists": "^4.9.3",
42
- "@xylabs/hex": "^4.9.3",
43
- "@xylabs/promise": "^4.9.3",
44
- "@xyo-network/account-model": "^3.14.17",
45
- "@xyo-network/archivist-memory": "^3.14.17",
46
- "@xyo-network/archivist-model": "^3.14.17",
47
- "@xyo-network/boundwitness-model": "^3.14.17",
48
- "@xyo-network/boundwitness-validator": "^3.14.17",
49
- "@xyo-network/boundwitness-wrapper": "^3.14.17",
50
- "@xyo-network/chain-analyze": "^1.3.17",
51
- "@xyo-network/chain-modules": "^1.3.17",
52
- "@xyo-network/chain-protocol": "^1.3.17",
53
- "@xyo-network/chain-utils": "^1.3.17",
54
- "@xyo-network/chain-validation": "^1.3.17",
55
- "@xyo-network/chain-wrappers": "^1.3.17",
56
- "@xyo-network/module-event-emitter": "^3.14.17",
57
- "@xyo-network/payload-builder": "^3.14.17",
58
- "@xyo-network/payload-model": "^3.14.17",
37
+ "@xylabs/array": "^4.9.4",
38
+ "@xylabs/assert": "^4.9.4",
39
+ "@xylabs/base": "^4.9.4",
40
+ "@xylabs/decimal-precision": "^4.9.4",
41
+ "@xylabs/exists": "^4.9.4",
42
+ "@xylabs/hex": "^4.9.4",
43
+ "@xylabs/promise": "^4.9.4",
44
+ "@xyo-network/account-model": "^3.14.18",
45
+ "@xyo-network/archivist-memory": "^3.14.18",
46
+ "@xyo-network/archivist-model": "^3.14.18",
47
+ "@xyo-network/boundwitness-model": "^3.14.18",
48
+ "@xyo-network/boundwitness-validator": "^3.14.18",
49
+ "@xyo-network/boundwitness-wrapper": "^3.14.18",
50
+ "@xyo-network/chain-analyze": "^1.3.18",
51
+ "@xyo-network/chain-modules": "^1.3.18",
52
+ "@xyo-network/chain-protocol": "^1.3.18",
53
+ "@xyo-network/chain-utils": "^1.3.18",
54
+ "@xyo-network/chain-validation": "^1.3.18",
55
+ "@xyo-network/chain-wrappers": "^1.3.18",
56
+ "@xyo-network/module-event-emitter": "^3.14.18",
57
+ "@xyo-network/payload-builder": "^3.14.18",
58
+ "@xyo-network/payload-model": "^3.14.18",
59
59
  "@xyo-network/typechain": "^3.5.2",
60
60
  "@xyo-network/xl1-protocol": "^1.3.12",
61
- "@xyo-network/xl1-protocol-sdk": "^1.3.17",
61
+ "@xyo-network/xl1-protocol-sdk": "^1.3.18",
62
62
  "async-mutex": "^0.5.0",
63
- "ethers": "6.13.7",
63
+ "ethers": "6.14.0",
64
64
  "lru-cache": "^11.1.0",
65
65
  "web3-types": "^1.10.0"
66
66
  },
67
67
  "devDependencies": {
68
- "@types/node": "^22.15.14",
69
- "@xylabs/delay": "^4.9.3",
68
+ "@types/node": "^22.15.17",
69
+ "@xylabs/delay": "^4.9.4",
70
70
  "@xylabs/ts-scripts-yarn3": "^6.5.5",
71
71
  "@xylabs/tsconfig": "^6.5.5",
72
- "@xylabs/vitest-extended": "^4.9.3",
73
- "@xyo-network/account": "^3.14.17",
74
- "@xyo-network/account-model": "^3.14.17",
75
- "@xyo-network/chain-ethereum": "^1.3.17",
76
- "@xyo-network/chain-validation": "^1.3.17",
77
- "@xyo-network/wallet": "^3.14.17",
78
- "knip": "^5.54.1",
72
+ "@xylabs/vitest-extended": "^4.9.4",
73
+ "@xyo-network/account": "^3.14.18",
74
+ "@xyo-network/account-model": "^3.14.18",
75
+ "@xyo-network/chain-ethereum": "^1.3.18",
76
+ "@xyo-network/chain-validation": "^1.3.18",
77
+ "@xyo-network/wallet": "^3.14.18",
78
+ "knip": "^5.55.1",
79
79
  "typescript": "^5.8.3",
80
80
  "vitest": "^3.1.3",
81
81
  "vitest-mock-extended": "^3.1.0"
@@ -1,3 +1,5 @@
1
+ import type { Tracer } from '@opentelemetry/api'
2
+ import { trace } from '@opentelemetry/api'
1
3
  import type { Promisable } from '@xylabs/promise'
2
4
  import { span, spanAsync } from '@xyo-network/chain-utils'
3
5
  import { BaseEmitter } from '@xyo-network/module-event-emitter'
@@ -25,8 +27,8 @@ export class BaseService<TParams extends BaseServiceParams = BaseServiceParams>
25
27
  return this.constructor.name
26
28
  }
27
29
 
28
- get tracer() {
29
- return this.params.traceProvider?.getTracer(this.name)
30
+ get tracer(): Tracer {
31
+ return this.params.traceProvider?.getTracer(this.name) || trace.getTracer(this.name)
30
32
  }
31
33
 
32
34
  static async create<TService extends BaseService<TParams>, TParams extends BaseServiceParams = TService['params']>(
@@ -54,17 +56,19 @@ export class BaseService<TParams extends BaseServiceParams = BaseServiceParams>
54
56
  return span(`${this.name}:${name}`, fn, this.tracer)
55
57
  }
56
58
 
57
- async spanAsync<T>(name: string, fn: () => Promise<T>, timeout?: number): Promise<T> {
58
- return await spanAsync(`${this.name}:${name}`, fn, this.tracer, timeout)
59
+ async spanAsync<T>(name: string, fn: () => Promise<T>): Promise<T> {
60
+ return await spanAsync(`${this.name}:${name}`, fn, this.tracer)
59
61
  }
60
62
  }
61
63
 
62
64
  export abstract class BaseAccountableService<
63
65
  TParams extends BaseAccountableServiceParams = BaseAccountableServiceParams,
64
- > extends BaseService<TParams> {}
66
+ > extends BaseService<TParams> {
67
+ // Base class for services that have an account
68
+ }
65
69
 
66
70
  export interface CreatableService<T extends BaseService = BaseService> extends Service {
67
- new (params: T['params']): T
71
+ new(params: T['params']): T
68
72
  create<T extends BaseService>(this: CreatableService<T>, params: T['params']): Promisable<T>
69
73
  }
70
74
 
@@ -1,10 +1,11 @@
1
+ import { ValueType } from '@opentelemetry/api'
1
2
  import { filterAs, filterAsync } from '@xylabs/array'
2
3
  import { assertEx } from '@xylabs/assert'
3
4
  import { exists } from '@xylabs/exists'
5
+ import { forget } from '@xylabs/forget'
4
6
  import { Hash } from '@xylabs/hex'
5
7
  import { MemoryArchivist } from '@xyo-network/archivist-memory'
6
8
  import { ArchivistInstance } from '@xyo-network/archivist-model'
7
- import { hydrateBlock } from '@xyo-network/chain-protocol'
8
9
  import { validateTransaction } from '@xyo-network/chain-validation'
9
10
  import {
10
11
  Payload, Sequence, WithStorageMeta,
@@ -44,16 +45,30 @@ export class XyoPendingTransactionsService extends BaseService<XyoPendingTransac
44
45
  * Priority for removing finalized transactions
45
46
  */
46
47
  RemoveFinalizedTransactions: 1,
47
- }
48
-
49
- // used to remember the last block's transactions that was removed due to it being in the chain
50
- private _cleanedToBlock: Hash | undefined
51
-
52
- // a local archivist that has only the transactions in it (used as queue)
53
- private _pendingTransactionsLocalArchivist: MemoryArchivist | undefined
54
- private _pendingTransactionsLocalArchivistCursor: Sequence | undefined
55
-
56
- private _updatePendingTransactionsLocalArchivistMutex = new Mutex()
48
+ } as const
49
+
50
+ /**
51
+ * A mutex to ensure that the counting the number of pending transactions is
52
+ * not called concurrently
53
+ */
54
+ private _countPendingTransactionsMutex = new Mutex()
55
+
56
+ /**
57
+ * A local Archivist optimized for fast retrieval that stores only validated
58
+ * pending transactions
59
+ */
60
+ private _curatedPendingTransactionsArchivist: MemoryArchivist | undefined
61
+
62
+ /**
63
+ * The last count of total pending transactions
64
+ */
65
+ private _pendingTransactionsCount: number = 0
66
+
67
+ /**
68
+ * A mutex to ensure that the curated pending transactions archivist is
69
+ * updated in a thread-safe manner
70
+ */
71
+ private _updateCuratedPendingTransactionsArchivistMutex = new Mutex()
57
72
 
58
73
  private get chainArchivist() {
59
74
  return assertEx(this.params.chainArchivist, () => 'No completed blocks with data archivist')
@@ -67,8 +82,13 @@ export class XyoPendingTransactionsService extends BaseService<XyoPendingTransac
67
82
  return assertEx(this.params.pendingTransactionsArchivist, () => 'No pending transactions archivist')
68
83
  }
69
84
 
85
+ private get pendingTransactionsCount() {
86
+ forget(this.countPendingTransactions())
87
+ return this._pendingTransactionsCount
88
+ }
89
+
70
90
  private get pendingTransactionsLocalArchivist() {
71
- return assertEx(this._pendingTransactionsLocalArchivist, () => 'No pending transactions curate archivist')
91
+ return assertEx(this._curatedPendingTransactionsArchivist, () => 'No pending transactions curated archivist')
72
92
  }
73
93
 
74
94
  private get rejectedTransactionsArchivist() {
@@ -77,45 +97,35 @@ export class XyoPendingTransactionsService extends BaseService<XyoPendingTransac
77
97
 
78
98
  override async createHandler() {
79
99
  await super.createHandler()
80
- this._pendingTransactionsLocalArchivist = await MemoryArchivist.create({ account: 'random' })
100
+ this._curatedPendingTransactionsArchivist = await MemoryArchivist.create({ account: 'random' })
101
+
81
102
  // On new pending transactions, insert them into the curated archivist
82
103
  this.pendingTransactionsArchivist.on('inserted', async ({ payloads }) => {
83
- await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {
84
- // Check incoming transactions against finalized transactions
85
- const unprocessedTransactions = await this.filterAlreadyFinalizedTransactions(payloads)
86
- // Hydrate all unprocessed transactions
87
- const hydratedUnprocessedTransactions = (await Promise.all(unprocessedTransactions.map((tx) => {
88
- return tryHydrateTransaction(this.pendingTransactionsArchivist, tx._hash)
89
- }))).filter(exists)
90
- // Filter to only valid transactions
91
- const validTransactions = await filterAsync(hydratedUnprocessedTransactions, async (tx) => {
92
- const errors = await validateTransaction(tx, this.chainIdentification.id)
93
- return errors.length > 0 ? false : true
94
- })
95
- await this.pendingTransactionsLocalArchivist.insert(flattenHydratedTransactions(validTransactions))
96
- }, XyoPendingTransactionsService.MutexPriority.InsertNewTransactions)
104
+ await this.insertNewTransactions(payloads)
97
105
  })
106
+
98
107
  // On new finalized blocks, remove the transactions from the curated archivist
99
108
  this.chainArchivist.on('inserted', async ({ payloads }) => {
100
- await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {
101
- const transactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta)
102
- await this.pendingTransactionsLocalArchivist.delete(transactions.map(tx => tx._hash))
103
- }, XyoPendingTransactionsService.MutexPriority.RemoveFinalizedTransactions)
109
+ await this.removeFinalizedTransactions(payloads)
104
110
  })
111
+
105
112
  // On new rejected blocks, remove the transactions from the curated archivist
106
113
  this.rejectedTransactionsArchivist.on('inserted', async ({ payloads }) => {
107
- await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {
108
- const transactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta)
109
- await this.pendingTransactionsLocalArchivist.delete(transactions.map(tx => tx._hash))
110
- }, XyoPendingTransactionsService.MutexPriority.RemoveRejectedTransactions)
114
+ await this.removeRejectedTransactions(payloads)
115
+ })
116
+
117
+ const pendingTransactionsGauge = this.meter?.createObservableGauge(
118
+ 'xyo_pending_transactions_count',
119
+ { description: 'The current number of pending transactions', valueType: ValueType.INT },
120
+ )
121
+ pendingTransactionsGauge?.addCallback((observer) => {
122
+ observer.observe(this.pendingTransactionsCount)
111
123
  })
112
124
  }
113
125
 
114
- async getPendingTransactions(head: Hash, limit: number, timeout = 1_000_000): Promise<HydratedTransactionWithStorageMeta[]> {
126
+ async getPendingTransactions(head: Hash, limit: number): Promise<HydratedTransactionWithStorageMeta[]> {
115
127
  return await this.spanAsync('getPendingTransactions', async () => {
116
- // await this.updatePendingTransactionsLocalArchivist()
117
- // await this.removeAlreadyIncludedTransactions(head)
118
- return await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {
128
+ return await this._updateCuratedPendingTransactionsArchivistMutex.runExclusive(async () => {
119
129
  const foundPendingTransactions: WithStorageMeta<TransactionBoundWitness>[] = []
120
130
  let cursor: Sequence | undefined
121
131
  while (foundPendingTransactions.length < limit) {
@@ -132,7 +142,16 @@ export class XyoPendingTransactionsService extends BaseService<XyoPendingTransac
132
142
  )
133
143
  return hydratedTransactions.filter(exists)
134
144
  }, XyoPendingTransactionsService.MutexPriority.ReadTransactions)
135
- }, timeout)
145
+ })
146
+ }
147
+
148
+ private async countPendingTransactions() {
149
+ if (this._countPendingTransactionsMutex.isLocked()) return
150
+ await this._countPendingTransactionsMutex.runExclusive(async () => {
151
+ const payloads = (await this._curatedPendingTransactionsArchivist?.all()) ?? []
152
+ const pendingTransactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta)
153
+ this._pendingTransactionsCount = pendingTransactions.length
154
+ })
136
155
  }
137
156
 
138
157
  private async filterAlreadyFinalizedTransactions(payloads: WithStorageMeta<Payload>[]): Promise<WithStorageMeta<TransactionBoundWitness>[]> {
@@ -144,49 +163,42 @@ export class XyoPendingTransactionsService extends BaseService<XyoPendingTransac
144
163
  return nonFinalizedTransactions
145
164
  }
146
165
 
147
- private async removeAlreadyIncludedTransactions(head?: Hash, timeout = 1_000_000) {
148
- return await this.spanAsync('removeAlreadyIncludedTransactions', async () => {
149
- let headPtr: Hash | null = head ?? null
150
- while (headPtr !== null) {
151
- const [block, payloads] = assertEx(await hydrateBlock(this.chainArchivist, headPtr), () => `Failed to find head [${head}]`)
152
- if (block._hash === this._cleanedToBlock) break
153
- await this.pendingTransactionsLocalArchivist.delete(
154
- payloads.map(payload => asOptionalTransactionBoundWitnessWithStorageMeta(payload)?._hash).filter(exists),
155
- )
156
- this._cleanedToBlock = block._hash
157
- headPtr = block?.previous
158
- }
159
- }, timeout)
166
+ private async insertNewTransactions(payloads: WithStorageMeta<Payload>[]) {
167
+ return await this.spanAsync('InsertNewTransactions', async () => {
168
+ return await this._updateCuratedPendingTransactionsArchivistMutex.runExclusive(async () => {
169
+ // Check incoming transactions against finalized transactions
170
+ const unprocessedTransactions = await this.filterAlreadyFinalizedTransactions(payloads)
171
+ // Hydrate all unprocessed transactions
172
+ const hydratedUnprocessedTransactions = (await Promise.all(unprocessedTransactions.map((tx) => {
173
+ return tryHydrateTransaction(this.pendingTransactionsArchivist, tx._hash)
174
+ }))).filter(exists)
175
+ // Filter to only valid transactions
176
+ const validTransactions = await filterAsync(hydratedUnprocessedTransactions, async (tx) => {
177
+ const errors = await validateTransaction(tx, this.chainIdentification.id)
178
+ return errors.length > 0 ? false : true
179
+ })
180
+ await this.pendingTransactionsLocalArchivist.insert(flattenHydratedTransactions(validTransactions))
181
+ }, XyoPendingTransactionsService.MutexPriority.InsertNewTransactions)
182
+ })
160
183
  }
161
184
 
162
- private async updatePendingTransactionsLocalArchivist(timeout = 1_000_000) {
163
- return await this.spanAsync('updatePendingTransactionsLocalArchivist', async () => {
164
- if (this._updatePendingTransactionsLocalArchivistMutex.isLocked()) {
165
- console.warn('updatePendingTransactionsLocalArchivist busy - skipping')
166
- return
167
- }
168
- return await this._updatePendingTransactionsLocalArchivistMutex.runExclusive(async () => {
169
- // const localWithDataArchivist = this.pendingTransactionsArchivist
170
- // sync from remote
171
- // await this.spanAsync('sync from remote [pending]', async () => {
172
- // this._pendingTransactionsLocalAllDataArchivistCursor = await synchronizeArchivists(
173
- // this.externalPendingTransactionsWithDataArchivist,
174
- // localWithDataArchivist,
175
- // this._pendingTransactionsLocalAllDataArchivistCursor,
176
- // )
177
- // }, timeout / 2)
178
- // // extract transactions only [from localWithDataArchivist]
179
- // await this.spanAsync('sync from local [pending]', async () => {
180
- // this._pendingTransactionsLocalArchivistCursor = await synchronizeArchivists(
181
- // localWithDataArchivist,
182
- // this.pendingTransactionsLocalArchivist,
183
- // this._pendingTransactionsLocalArchivistCursor,
184
- // isTransactionBoundWitness,
185
- // isTransactionBoundWitness,
186
- // )
187
- // }, timeout / 10)
188
- await Promise.resolve()
189
- })
190
- }, timeout)
185
+ private async removeFinalizedTransactions(payloads: WithStorageMeta<Payload>[]) {
186
+ return await this.removeTransactions(payloads, 'RemoveFinalizedTransactions')
187
+ }
188
+
189
+ private async removeRejectedTransactions(payloads: WithStorageMeta<Payload>[]) {
190
+ return await this.removeTransactions(payloads, 'RemoveRejectedTransactions')
191
+ }
192
+
193
+ private async removeTransactions(
194
+ payloads: WithStorageMeta<Payload>[],
195
+ priority: keyof typeof XyoPendingTransactionsService.MutexPriority,
196
+ ) {
197
+ return await this.spanAsync(priority, async () => {
198
+ return await this._updateCuratedPendingTransactionsArchivistMutex.runExclusive(async () => {
199
+ const transactions = filterAs(payloads, asOptionalTransactionBoundWitnessWithStorageMeta)
200
+ await this.pendingTransactionsLocalArchivist.delete(transactions.map(tx => tx._hash))
201
+ }, XyoPendingTransactionsService.MutexPriority[priority])
202
+ })
191
203
  }
192
204
  }