-
Notifications
You must be signed in to change notification settings - Fork 17
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #4 from nimiq/retrieve-size-from-ended-epochs
Retrieve size from ended epochs
Showing
59 changed files
with
1,023 additions
and
14,779 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
File renamed without changes.
4 changes: 2 additions & 2 deletions
4
packages/nimiq-vts/README.md → packages/nimiq-validators-score/README.md
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
File renamed without changes.
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,139 @@ | ||
import { type ElectionMacroBlock, InherentType, type NimiqRPCClient } from 'nimiq-rpc-client-ts' | ||
import { getPolicyConstants } from './utils' | ||
import type { EpochActivity, EpochsActivities } from './types' | ||
|
||
// TODO remove Console log | ||
|
||
/** | ||
* For a given block number, fetches the validator slots assignation. | ||
* The block number MUST be an election block otherwise it will throw an error. | ||
*/ | ||
export async function fetchActivity(client: NimiqRPCClient, epochIndex: number) { | ||
const { batchesPerEpoch, genesisBlockNumber, blocksPerBatch, slots: slotsCount, blocksPerEpoch } = await getPolicyConstants(client) | ||
|
||
const electionBlock = genesisBlockNumber + (epochIndex * blocksPerEpoch) | ||
const { data: block, error } = await client.blockchain.getBlockByNumber(electionBlock, { includeTransactions: true }) | ||
if (error || !block) | ||
throw new Error(JSON.stringify({ epochIndex, error, block })) | ||
if (!('isElectionBlock' in block)) | ||
throw new Error(JSON.stringify({ message: 'Block is not election block', epochIndex, block })) | ||
|
||
const { data: currentEpoch, error: errorCurrentEpoch } = await client.blockchain.getEpochNumber() | ||
if (errorCurrentEpoch || !currentEpoch) | ||
throw new Error(`There was an error fetching current epoch: ${JSON.stringify({ epochIndex, errorCurrentEpoch, currentEpoch })}`) | ||
if (epochIndex >= currentEpoch) | ||
throw new Error(`You tried to fetch an epoch that is not finished yet: ${JSON.stringify({ epochIndex, currentEpoch })}`) | ||
|
||
// The election block will be the first block of the epoch, since we only fetch finished epochs, we can assume that all the batches in this epoch can be fetched | ||
// First, we need to know in which batch this block is. Batches start at 0 | ||
const firstBatchIndex = (electionBlock - genesisBlockNumber) / blocksPerBatch | ||
if (firstBatchIndex % 1 !== 0) | ||
// It should be an exact division since we are fetching election blocks | ||
throw new Error(JSON.stringify({ message: 'Something happened calculating batchIndex', firstBatchIndex, electionBlock, block })) | ||
|
||
// Initialize the list of validators and their activity in the epoch | ||
const epochActivity: EpochActivity = {} | ||
for (const { numSlots: likelihood, validator } of (block as ElectionMacroBlock).slots) { | ||
epochActivity[validator] = { likelihood, missed: 0, rewarded: 0, sizeRatio: likelihood / slotsCount, sizeRatioViaSlots: true } | ||
} | ||
|
||
const maxBatchSize = 120 | ||
const minBatchSize = 10 | ||
let batchSize = maxBatchSize | ||
for (let i = 0; i < batchesPerEpoch; i += batchSize) { | ||
const batchPromises = Array.from({ length: Math.min(batchSize, batchesPerEpoch - i) }, (_, j) => createPromise(i + j)) | ||
|
||
let results = await Promise.allSettled(batchPromises) | ||
|
||
let rejectedIndexes: number[] = results.reduce((acc: number[], result, index) => { | ||
if (result.status === 'rejected') { | ||
acc.push(index) | ||
} | ||
return acc | ||
}, []) | ||
|
||
if (rejectedIndexes.length > 0) { | ||
// Lowering the batch size to prevent more rejections | ||
batchSize = Math.max(minBatchSize, Math.floor(batchSize / 2)) | ||
} | ||
else { | ||
// Increasing the batch size to speed up the process | ||
batchSize = Math.min(maxBatchSize, Math.floor(batchSize + batchSize / 2)) | ||
} | ||
|
||
while (rejectedIndexes.length > 0) { | ||
const retryPromises = rejectedIndexes.map(index => createPromise(i + index)) | ||
results = await Promise.allSettled(retryPromises) | ||
|
||
rejectedIndexes = results.reduce((acc: number[], result, index) => { | ||
if (result.status === 'rejected') { | ||
acc.push(rejectedIndexes[index]) | ||
} | ||
return acc | ||
}, []) | ||
} | ||
} | ||
|
||
async function createPromise(index: number) { | ||
const { data: inherents, error: errorBatch } = await client.blockchain.getInherentsByBatchNumber(firstBatchIndex + index) | ||
return new Promise<void>((resolve, reject) => { | ||
if (errorBatch || !inherents) { | ||
reject(JSON.stringify({ epochIndex, blockNumber: electionBlock, errorBatch, index, firstBatchIndex, currentIndex: firstBatchIndex + index })) | ||
} | ||
else { | ||
for (const { type, validatorAddress } of inherents) { | ||
if (validatorAddress === 'NQ07 0000 0000 0000 0000 0000 0000 0000 0000') | ||
continue | ||
if (!epochActivity[validatorAddress]) | ||
continue | ||
epochActivity[validatorAddress].rewarded += type === InherentType.Reward ? 1 : 0 | ||
epochActivity[validatorAddress].missed += [InherentType.Penalize, InherentType.Jail].includes(type) ? 1 : 0 | ||
} | ||
resolve() | ||
} | ||
}) | ||
} | ||
|
||
return epochActivity | ||
} | ||
|
||
/** | ||
* Fetches the activity for the given block numbers. | ||
* This function is an asynchronous generator. It yields each activity one by one, | ||
* allowing the caller to decide when to fetch the next activity. | ||
* | ||
* @param client - The client instance to use for fetching validator activities. | ||
* @param epochsIndexes - An array of epoch block numbers to fetch the activities for. | ||
* @returns An asynchronous generator yielding objects containing the address, epoch block number, and activity. | ||
* | ||
* Usage: | ||
* const activitiesGenerator = fetchActivities(client, epochBlockNumbers); | ||
* for await (const { key, activity } of activitiesGenerator) { | ||
* console.log(`Address: ${key.address}, Epoch: ${key.epochBlockNumber}, Activity: ${activity}`); | ||
* } | ||
*/ | ||
export async function* fetchEpochs(client: NimiqRPCClient, epochsIndexes: number[]) { | ||
for (const epochIndex of epochsIndexes) { | ||
const validatorActivities = await fetchActivity(client, epochIndex) | ||
for (const [address, activity] of Object.entries(validatorActivities)) { | ||
yield { address, epochIndex, activity } | ||
} | ||
} | ||
} | ||
|
||
export async function fetchCurrentEpoch(client: NimiqRPCClient) { | ||
const { data: currentEpoch, error } = await client.blockchain.getEpochNumber() | ||
if (error || !currentEpoch) | ||
throw new Error(JSON.stringify({ error, currentEpoch })) | ||
const { data: activeValidators, error: errorValidators } = await client.blockchain.getActiveValidators() | ||
if (errorValidators || !activeValidators) | ||
throw new Error(JSON.stringify({ errorValidators, activeValidators })) | ||
const totalBalance = Object.values(activeValidators).reduce((acc, { balance }) => acc + balance, 0) | ||
const epochActivity: EpochsActivities = { | ||
[currentEpoch]: Object.entries(activeValidators).reduce((acc, [, { address, balance }]) => { | ||
acc[address] = { likelihood: -1, missed: -1, rewarded: -1, sizeRatio: balance / totalBalance, sizeRatioViaSlots: false } | ||
return acc | ||
}, {} as EpochActivity), | ||
} | ||
return epochActivity | ||
} |
File renamed without changes.
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
File renamed without changes.
This file was deleted.
Oops, something went wrong.
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,7 +1,7 @@ | ||
packages: | ||
- packages/* | ||
catalog: | ||
'@antfu/eslint-config': ^2.26.1 | ||
eslint: ^9.9.0 | ||
'@antfu/eslint-config': ^3.0.0 | ||
eslint: ^9.9.1 | ||
lint-staged: ^15.2.9 | ||
simple-git-hooks: ^2.11.1 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"name": "Optional. The name of the validator, by default it is the address", | ||
"address": "NQXX XXXX XXXX XXXX XXXX XXXX XXXX XXXX XXXX", | ||
"fee": -1, | ||
"payoutType": "restake | direct", | ||
"tag": "Optional field. Valid values are: 'community' | 'unkwown'" | ||
} |
7 changes: 7 additions & 0 deletions
7
public/validators/NQ08 N4RH FQDL TE7S 8C66 65LT KYDU Q382 YG7U.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"name": "Not-A-Pool", | ||
"address": "NQ08 N4RH FQDL TE7S 8C66 65LT KYDU Q382 YG7U", | ||
"fee": 0.11, | ||
"payoutType": "restake", | ||
"tag": "Nimiq" | ||
} |
7 changes: 7 additions & 0 deletions
7
public/validators/NQ24 DJE3 KX3U HG5X 1BXP 8XQ3 SK7S X364 N7G7.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"name": "Pooly McPoolface", | ||
"address": "NQ24 DJE3 KX3U HG5X 1BXP 8XQ3 SK7S X364 N7G7", | ||
"fee": 0.09, | ||
"payoutType": "restake", | ||
"tag": "Nimiq" | ||
} |
7 changes: 7 additions & 0 deletions
7
public/validators/NQ38 YX2J GTMX 5XAU LKFU H0GS A4AA U26L MDA3.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"name": "Swimming Pool", | ||
"address": "NQ38 YX2J GTMX 5XAU LKFU H0GS A4AA U26L MDA3", | ||
"fee": 0.1, | ||
"payoutType": "restake", | ||
"tag": "Nimiq" | ||
} |
9 changes: 9 additions & 0 deletions
9
public/validators/NQ49 E4LQ FN9M B9BP 0FRE BCL5 MHFY TGQE D4XX.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
{ | ||
"name": "Helvetia Staking", | ||
"address": "NQ49 E4LQ FN9M B9BP 0FRE BCL5 MHFY TGQE D4XX", | ||
"fee": 0.1, | ||
"payoutType": "restake", | ||
"description": "The Swiss-standard of NIMIQ Staking", | ||
"icon": " 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", | ||
"tag": "community" | ||
} |
7 changes: 7 additions & 0 deletions
7
public/validators/NQ57 UQJL 5A3H N45M 1FHS 2454 C7L5 BTE6 KEU1.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"name": "Kiddie Pool", | ||
"address": "NQ57 UQJL 5A3H N45M 1FHS 2454 C7L5 BTE6 KEU1", | ||
"fee": 0.1, | ||
"payoutType": "restake", | ||
"tag": "Nimiq" | ||
} |
8 changes: 8 additions & 0 deletions
8
public/validators/NQ65 DHN8 4BSR 5YSX FC3V BB5J GKM2 GB2L H17C.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
{ | ||
"name": "AceStaking", | ||
"address": "NQ65 DHN8 4BSR 5YSX FC3V BB5J GKM2 GB2L H17C", | ||
"fee": 0.1, | ||
"payoutType": "direct", | ||
"description": "The Ace in staking", | ||
"tag": "Community" | ||
} |
7 changes: 7 additions & 0 deletions
7
public/validators/NQ71 CK94 3V7U H62Y 4L0F GUUK DPA4 6SA6 DKKM.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"name": "Monopooly", | ||
"address": "NQ71 CK94 3V7U H62Y 4L0F GUUK DPA4 6SA6 DKKM", | ||
"fee": 0.11, | ||
"payoutType": "direct", | ||
"tag": "Nimiq" | ||
} |
7 changes: 7 additions & 0 deletions
7
public/validators/NQ82 BHPS UR9K 07X1 X6QH 3DY3 J325 UCSP UHV3.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"name": "Puddle", | ||
"address": "NQ82 BHPS UR9K 07X1 X6QH 3DY3 J325 UCSP UHV3", | ||
"fee": 0.095, | ||
"payoutType": "direct", | ||
"tag": "Nimiq" | ||
} |
7 changes: 7 additions & 0 deletions
7
public/validators/NQ87 FEGQ 01TF M29N T03J 3YCB JB5M X5VM XP8Q.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"name": "Cool Pool", | ||
"address": "NQ87 FEGQ 01TF M29N T03J 3YCB JB5M X5VM XP8Q", | ||
"fee": 0.09, | ||
"payoutType": "direct", | ||
"tag": "Nimiq" | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
import { poolQuerySchema } from '../utils/schemas' | ||
import { fetchValidators } from '../utils/validators' | ||
|
||
export default defineEventHandler(async (event) => { | ||
const { onlyPools } = await getValidatedQuery(event, poolQuerySchema.parse) | ||
const validators = await fetchValidators({ onlyPools }) | ||
return { validators } | ||
}) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,32 +1,44 @@ | ||
import { integer, primaryKey, real, sqliteTable, text } from 'drizzle-orm/sqlite-core' | ||
import { index, integer, primaryKey, real, sqliteTable, text, uniqueIndex } from 'drizzle-orm/sqlite-core' | ||
|
||
// TODO | ||
// Is delete on cascade ok? | ||
|
||
export const validators = sqliteTable('validators', { | ||
id: integer('id').primaryKey({ autoIncrement: true, onConflict: 'replace' }), | ||
name: text('name').default('Unknown validator'), | ||
name: text('name').default('Unknown validator').notNull(), | ||
address: text('address').notNull().unique(), | ||
fee: real('fee').default(-1), | ||
payoutType: text('payout_type').default('unknown'), | ||
description: text('description'), | ||
icon: text('icon').notNull(), | ||
tag: text('tag').default('unknown'), | ||
website: text('website'), | ||
}) | ||
}, table => ({ | ||
uniqueAddress: uniqueIndex('validators_address_unique').on(table.address), | ||
})) | ||
|
||
// TODO The score is calculated based on a window of epochs (default 9 months) | ||
// Maybe we could add those two parameters (fromEpochNumber and toEpochNumber) to the scores table to have more context | ||
export const scores = sqliteTable('scores', { | ||
id: integer('score_id').notNull().primaryKey(), | ||
validatorId: integer('validator_id').notNull().references(() => validators.id).unique(), | ||
validatorId: integer('validator_id').notNull().references(() => validators.id), | ||
fromEpoch: integer('from_epoch').notNull(), | ||
toEpoch: integer('to_epoch').notNull(), | ||
total: real('total').notNull(), | ||
liveness: real('liveness').notNull(), | ||
size: real('size').notNull(), | ||
reliability: real('reliability').notNull(), | ||
}) | ||
}, table => ({ | ||
idxValidatorId: index('idx_validator_id').on(table.validatorId), | ||
compositePrimaryKey: primaryKey({ columns: [table.validatorId, table.fromEpoch, table.toEpoch] }), | ||
})) | ||
|
||
export const activity = sqliteTable('activity', { | ||
validatorId: integer('validator_id').notNull().references(() => validators.id), | ||
epochBlockNumber: integer('epoch_block_number').notNull(), | ||
epochNumber: integer('epoch_number').notNull(), | ||
likelihood: integer('likelihood').notNull(), | ||
rewarded: integer('rewarded').notNull(), | ||
missed: integer('missed').notNull(), | ||
}, ({ epochBlockNumber, validatorId }) => ({ pk: primaryKey({ columns: [validatorId, epochBlockNumber] }) })) | ||
sizeRatio: integer('size_ratio').notNull(), | ||
sizeRatioViaSlots: integer('size_ratio_via_slots').notNull(), | ||
}, table => ({ | ||
idxElectionBlock: index('idx_election_block').on(table.epochNumber), | ||
compositePrimaryKey: primaryKey({ columns: [table.validatorId, table.epochNumber] }), | ||
})) |
This file was deleted.
Oops, something went wrong.
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file was deleted.
Oops, something went wrong.
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,78 @@ | ||
import { gte, lte } from 'drizzle-orm' | ||
import type { Activity, EpochsActivities, Range } from 'nimiq-validators-score' | ||
import type { NewActivity } from './drizzle' | ||
import { storeValidator } from './validators' | ||
|
||
/** | ||
* Given a range, it returns the epochs that are missing in the database. | ||
*/ | ||
export async function findMissingEpochs(range: Range) { | ||
const existingEpochs = await useDrizzle() | ||
.selectDistinct({ epochBlockNumber: tables.activity.epochNumber }) | ||
.from(tables.activity) | ||
.where(and( | ||
gte(tables.activity.epochNumber, range.fromEpoch), | ||
lte(tables.activity.epochNumber, range.toEpoch), | ||
)) | ||
.execute().then(r => r.map(r => r.epochBlockNumber)) | ||
|
||
const missingEpochs = [] | ||
for (let i = range.fromEpoch; i <= range.toEpoch; i++) { | ||
if (!existingEpochs.includes(i)) | ||
missingEpochs.push(i) | ||
} | ||
return missingEpochs | ||
} | ||
|
||
/** | ||
* We loop over all the pairs activities/epochBlockNumber and store the validator activities. | ||
*/ | ||
export async function storeActivities(epochs: EpochsActivities) { | ||
const promises = Object.entries(epochs).map(async ([_epochNumber, activities]) => { | ||
const epochNumber = Number(_epochNumber) | ||
const activityPromises = Object.entries(activities).map(async ([address, activity]) => storeSingleActivity({ address, activity, epochNumber })) | ||
return await Promise.all(activityPromises) | ||
}) | ||
await Promise.all(promises) | ||
} | ||
|
||
interface StoreActivityParams { | ||
address: string | ||
activity: Activity | ||
epochNumber: number | ||
} | ||
|
||
async function storeSingleActivity({ address, activity, epochNumber }: StoreActivityParams) { | ||
const validatorId = await storeValidator(address) | ||
if (!validatorId) | ||
return | ||
// If we ever move out of cloudflare we could use transactions to avoid inconsistencies and improve performance | ||
// Cloudflare D1 does not support transactions: https://github.com/cloudflare/workerd/blob/e78561270004797ff008f17790dae7cfe4a39629/src/workerd/api/sql-test.js#L252-L253 | ||
const existingActivity = await useDrizzle() | ||
.select({ sizeRatioViaSlots: tables.activity.sizeRatioViaSlots, sizeRatio: tables.activity.sizeRatio }) | ||
.from(tables.activity) | ||
.where(and( | ||
eq(tables.activity.epochNumber, epochNumber), | ||
eq(tables.activity.validatorId, validatorId), | ||
)) | ||
|
||
const { likelihood, rewarded, missed, sizeRatio: _sizeRatio, sizeRatioViaSlots: _sizeRatioViaSlots } = activity | ||
|
||
// We always want to update db except the columns `sizeRatio` and `sizeRatioViaSlots`. | ||
// If we have `sizeRatioViaSlots` as false and `sizeRatio` > 0, then we won't update only those columns | ||
// As we want to keep the values from the first time we inserted the activity as they are more accurate | ||
const viaSlotsDb = Boolean(existingActivity.at(0)?.sizeRatioViaSlots) | ||
const sizeRatioDb = existingActivity.at(0)?.sizeRatio || 0 | ||
const updateSizeColumns = viaSlotsDb !== false || sizeRatioDb <= 0 | ||
const sizeRatio = updateSizeColumns ? _sizeRatio : sizeRatioDb | ||
const sizeRatioViaSlotsBool = updateSizeColumns ? _sizeRatioViaSlots : viaSlotsDb | ||
const sizeRatioViaSlots = sizeRatioViaSlotsBool ? 1 : 0 | ||
|
||
await useDrizzle().delete(tables.activity) | ||
.where(and( | ||
eq(tables.activity.epochNumber, epochNumber), | ||
eq(tables.activity.validatorId, validatorId), | ||
)) | ||
const activityDb: NewActivity = { likelihood, rewarded, missed, epochNumber, validatorId, sizeRatio, sizeRatioViaSlots } | ||
await useDrizzle().insert(tables.activity).values(activityDb) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
import type { Range } from 'nimiq-validators-score' | ||
import { getRange } from 'nimiq-validators-score' | ||
import type { NimiqRPCClient } from 'nimiq-rpc-client-ts' | ||
import type { EventHandlerRequest, H3Event } from 'h3' | ||
import { consola } from 'consola' | ||
import type { Result } from './types' | ||
import { rangeQuerySchema } from './schemas' | ||
|
||
/** | ||
* To compute the score for a validator for a given range, it is mandatory that we have the activity for that validator | ||
* in the last epoch of the range. If we don't have the activity for that epoch, we can't compute the score. | ||
* Instead of throwing an error, we will modify the range so the last epoch is the last epoch where we have activity. | ||
*/ | ||
// export async function adjustRangeForAvailableData(expectedRange: Range): Result<Range> { | ||
// const highestScoreEpoch = await useDrizzle() | ||
// .select({ toEpoch: max(tables.scores.toEpoch) }) | ||
// .from(tables.scores) | ||
// .where(and( | ||
// gte(tables.scores.fromEpoch, expectedRange.fromEpoch), | ||
// lte(tables.scores.toEpoch, expectedRange.toEpoch), | ||
// )) | ||
// .get() | ||
// .then(r => r?.toEpoch) | ||
// consola.info({ highestScoreEpoch }) | ||
// if (!highestScoreEpoch) | ||
// return { error: `No scores found between epochs ${expectedRange.fromEpoch} and ${expectedRange.toEpoch}. Run the fetch task first.`, data: undefined } | ||
|
||
// const toEpoch = Math.min(highestScoreEpoch, expectedRange.toEpoch) | ||
// const toBlockNumber = expectedRange.epochIndexToBlockNumber(toEpoch) | ||
// const range: Range = { ...expectedRange, toEpoch, toBlockNumber } | ||
// return { data: range, error: undefined } | ||
// } | ||
|
||
export async function extractRangeFromRequest(rpcClient: NimiqRPCClient, event: H3Event<EventHandlerRequest>): Result<Range> { | ||
const { data: currentEpoch, error: currentEpochError } = await rpcClient.blockchain.getEpochNumber() | ||
if (currentEpochError || !currentEpoch) | ||
return { error: JSON.stringify(currentEpochError), data: undefined } | ||
const { epoch: userEpoch } = await getValidatedQuery(event, rangeQuerySchema.parse) | ||
|
||
let epoch | ||
if (userEpoch === 'latest') | ||
epoch = currentEpoch - 1 | ||
else if (currentEpoch <= userEpoch) | ||
return { error: `Epoch ${epoch} is in the future or it didn't finished yet. The newest epoch you can fetch is ${currentEpoch - 1}.`, data: undefined } | ||
else | ||
epoch = userEpoch | ||
consola.info(`Fetching data for epoch ${epoch}`) | ||
let range: Range | ||
consola.info(`Fetching data for epoch ${epoch}`) | ||
try { | ||
range = await getRange(rpcClient, { toEpochIndex: epoch }) | ||
} | ||
catch (error: unknown) { return { error: JSON.stringify(error), data: undefined } } | ||
return { data: range, error: undefined } | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,27 @@ | ||
import { z } from 'zod' | ||
import { DEFAULT_WINDOW_IN_DAYS, DEFAULT_WINDOW_IN_MS } from '~~/packages/nimiq-validators-score/src' | ||
|
||
export const rangeQuerySchema = z.object({ | ||
epoch: z.literal('latest').or(z.number().min(1)).default('latest'), | ||
epochsCount: z.number().min(1).default(DEFAULT_WINDOW_IN_DAYS), | ||
durationWindowMs: z.number().min(1).default(DEFAULT_WINDOW_IN_MS), | ||
}).refine(({ epochsCount, durationWindowMs }) => { | ||
const defaultCounts = epochsCount === DEFAULT_WINDOW_IN_DAYS | ||
const defaultWindow = durationWindowMs === DEFAULT_WINDOW_IN_MS | ||
return (!epochsCount || !durationWindowMs) || (defaultCounts && defaultWindow) || (!defaultCounts && !defaultWindow) | ||
}) | ||
|
||
export const validatorSchema = z.object({ | ||
name: z.string().optional(), | ||
address: z.string().regex(/^NQ\d{2}(\s\w{4}){8}$/, 'Invalid Nimiq address format'), | ||
fee: z.number().min(0).max(1), | ||
payoutType: z.nativeEnum(PayoutType), | ||
tag: z.nativeEnum(ValidatorTag), | ||
description: z.string().optional(), | ||
website: z.string().url().optional(), | ||
icon: z.string().optional(), | ||
}) | ||
|
||
export const poolQuerySchema = z.object({ | ||
onlyPools: z.literal('true').or(z.literal('false')).default('false').transform(v => v === 'true'), | ||
}) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,123 @@ | ||
import { gte, inArray, lte } from 'drizzle-orm' | ||
import type { Range, ScoreParams } from 'nimiq-validators-score' | ||
import { computeScore } from 'nimiq-validators-score' | ||
import type { NewScore } from './drizzle' | ||
import type { Result, ValidatorScore } from './types' | ||
import { fetchValidatorsScoreByIds } from './validators' | ||
import { findMissingEpochs } from './activities' | ||
|
||
interface GetScoresResult { | ||
validators: ValidatorScore[] | ||
range: Range | ||
} | ||
|
||
/** | ||
* Given a range of epochs, it returns the scores for the validators in that range. | ||
*/ | ||
export async function calculateScores(range: Range): Result<GetScoresResult> { | ||
const missingEpochs = await findMissingEpochs(range) | ||
if (missingEpochs.length > 0) | ||
throw new Error(`Missing epochs in database: ${missingEpochs.join(', ')}. Run the fetch task first.`) | ||
|
||
// TODO Decide how we want to handle the case of missing activity | ||
// const { data: range, error: rangeError } = await adjustRangeForAvailableData(expectedRange) | ||
// consola.info({ range, rangeError }) | ||
// if (rangeError || !range) | ||
// return { error: rangeError, data: undefined } | ||
|
||
// TODO Check if we already have scores for the given range | ||
|
||
const sizeLastEpoch = await useDrizzle() | ||
.select({ | ||
sizeRatio: tables.activity.sizeRatio, | ||
sizeRatioViaSlots: tables.activity.sizeRatioViaSlots, | ||
validatorId: tables.activity.validatorId, | ||
}) | ||
.from(tables.activity) | ||
.where(and( | ||
eq(tables.activity.epochNumber, range.toEpoch), | ||
)) | ||
|
||
const sizeLastEpochByValidator = new Map<number, { sizeRatio: number, sizeRatioViaSlots: boolean }>() | ||
sizeLastEpoch.forEach(({ validatorId, sizeRatio, sizeRatioViaSlots }) => | ||
sizeLastEpochByValidator.set(validatorId, { sizeRatio, sizeRatioViaSlots: Boolean(sizeRatioViaSlots) })) | ||
const validatorsIds = Array.from(sizeLastEpochByValidator.keys()) | ||
|
||
const _activities = await useDrizzle() | ||
.select({ | ||
epoch: tables.activity.epochNumber, | ||
validatorId: tables.validators.id, | ||
rewarded: tables.activity.rewarded, | ||
missed: tables.activity.missed, | ||
}) | ||
.from(tables.activity) | ||
.innerJoin(tables.validators, eq(tables.activity.validatorId, tables.validators.id)) | ||
.where(and( | ||
gte(tables.activity.epochNumber, range.fromEpoch), | ||
lte(tables.activity.epochNumber, range.toEpoch), | ||
inArray(tables.activity.validatorId, validatorsIds), | ||
)) | ||
.orderBy(tables.activity.epochNumber) | ||
.execute() | ||
|
||
type Activity = Map<number /* validatorId */, { inherentsPerEpoch: Map<number /* epoch */, { rewarded: number, missed: number }>, sizeRatio: number, sizeRatioViaSlots: boolean }> | ||
|
||
const validatorsParams: Activity = new Map() | ||
|
||
for (const { epoch, missed, rewarded, validatorId } of _activities) { | ||
if (!validatorsParams.has(validatorId)) { | ||
const { sizeRatio, sizeRatioViaSlots } = sizeLastEpochByValidator.get(validatorId) ?? { sizeRatio: -1, sizeRatioViaSlots: false } | ||
if (sizeRatio === -1) | ||
return { error: `Missing size ratio for validator ${validatorId}. Range: ${range.fromEpoch}-${range.toEpoch}`, data: undefined } | ||
validatorsParams.set(validatorId, { sizeRatio, sizeRatioViaSlots, inherentsPerEpoch: new Map() }) | ||
} | ||
const validatorInherents = validatorsParams.get(validatorId)!.inherentsPerEpoch | ||
if (!validatorInherents.has(epoch)) | ||
validatorInherents.set(epoch, { rewarded: 0, missed: 0 }) | ||
const { missed: accMissed, rewarded: accRewarded } = validatorInherents.get(epoch)! | ||
validatorInherents.set(epoch, { rewarded: accRewarded + rewarded, missed: accMissed + missed }) | ||
} | ||
const scores = Array.from(validatorsParams.entries()).map(([validatorId, { inherentsPerEpoch }]) => { | ||
const activeEpochStates = Array.from({ length: range.toEpoch - range.fromEpoch + 1 }, (_, i) => inherentsPerEpoch.has(range.fromEpoch + i) ? 1 : 0) | ||
const size: ScoreParams['size'] = { sizeRatio: sizeLastEpochByValidator.get(validatorId)?.sizeRatio ?? -1 } | ||
const liveness: ScoreParams['liveness'] = { activeEpochStates } | ||
const reliability: ScoreParams['reliability'] = { inherentsPerEpoch } | ||
const score = computeScore({ liveness, size, reliability }) | ||
const newScore: NewScore = { validatorId: Number(validatorId), fromEpoch: range.fromEpoch, toEpoch: range.toEpoch, ...score } | ||
return newScore | ||
}) | ||
|
||
await persistScores(scores) | ||
const { data: validators, error: errorValidators } = await fetchValidatorsScoreByIds(scores.map(s => s.validatorId)) | ||
if (errorValidators || !validators) | ||
return { error: errorValidators, data: undefined } | ||
return { data: { validators, range }, error: undefined } | ||
} | ||
|
||
/** | ||
* Insert the scores into the database. To avoid inconsistencies, it deletes all the scores for the given validators and then inserts the new scores. | ||
*/ | ||
export async function persistScores(scores: NewScore[]) { | ||
await useDrizzle().delete(tables.scores).where(or(...scores.map(({ validatorId }) => eq(tables.scores.validatorId, validatorId)))) | ||
await Promise.all(scores.map(async score => await useDrizzle().insert(tables.scores).values(score))) | ||
|
||
// If we ever move out of cloudflare we could use transactions to avoid inconsistencies | ||
// Cloudflare D1 does not support transactions: https://github.com/cloudflare/workerd/blob/e78561270004797ff008f17790dae7cfe4a39629/src/workerd/api/sql-test.js#L252-L253 | ||
// await useDrizzle().transaction(async (tx) => { | ||
// await tx.delete(tables.scores).where(or(...scores.map(({ validatorId }) => eq(tables.scores.validatorId, validatorId)))) | ||
// await tx.insert(tables.scores).values(scores.map(s => ({ ...s, updatedAt }))) | ||
// }) | ||
} | ||
|
||
export async function checkIfScoreExistsInDb(range: Range) { | ||
const scoreAlreadyInDb = await useDrizzle() | ||
.select({ validatorId: tables.scores.validatorId }) | ||
.from(tables.scores) | ||
.where(and( | ||
eq(tables.scores.toEpoch, range.toEpoch), | ||
eq(tables.scores.fromEpoch, range.fromEpoch), | ||
)) | ||
.get() | ||
.then(r => Boolean(r?.validatorId)) | ||
return scoreAlreadyInDb | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
import type { Score, Validator } from './drizzle' | ||
import type { Range } from '~~/packages/nimiq-validators-score/src' | ||
|
||
export type Result<T> = Promise<{ data: T, error: undefined } | { data: undefined, error: string }> | ||
|
||
export type ValidatorScore = | ||
Pick<Validator, 'id' | 'name' | 'address' | 'fee' | 'payoutType' | 'description' | 'icon' | 'tag' | 'website'> | ||
& Pick<Score, 'total' | 'liveness' | 'size' | 'reliability'> | ||
& { range: Range, sizeRatioViaSlots: number } |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,161 @@ | ||
import { readFile, readdir } from 'node:fs/promises' | ||
import path from 'node:path' | ||
import { desc, inArray } from 'drizzle-orm' | ||
// @ts-expect-error no types in the package | ||
import Identicons from '@nimiq/identicons' | ||
import { consola } from 'consola' | ||
import { Address } from '@nimiq/core' | ||
import type { NewValidator, Validator } from './drizzle' | ||
import type { Result, ValidatorScore } from './types' | ||
import { validatorSchema } from './schemas' | ||
|
||
/** | ||
* Given a list of validator addresses, it returns the addresses that are missing in the database. | ||
* This is useful when we are fetching the activity for a range of epochs and we need to check if the validators are already in the database. | ||
* They should be present in the database because the fetch function needs to be run in order to compute the score. | ||
*/ | ||
export async function findMissingValidators(addresses: string[]) { | ||
const existingAddresses = await useDrizzle() | ||
.select({ address: tables.validators.address }) | ||
.from(tables.validators) | ||
.where(inArray(tables.validators.address, addresses)) | ||
.execute().then(r => r.map(r => r.address)) | ||
|
||
const missingAddresses = addresses.filter(a => !existingAddresses.includes(a)) | ||
return missingAddresses | ||
} | ||
|
||
const validators = new Map<string, number>() | ||
|
||
interface StoreValidatorOptions { | ||
/** | ||
* If true, it will store the validator even if it already exists in the database. | ||
* @default false | ||
*/ | ||
force?: boolean | ||
} | ||
|
||
export async function storeValidator( | ||
address: string, | ||
rest: Omit<NewValidator, 'address' | 'icon'> = {}, | ||
options: StoreValidatorOptions = {}, | ||
): Promise<number | undefined> { | ||
try { | ||
Address.fromString(address) | ||
} | ||
catch (error: unknown) { | ||
consola.error(`Invalid address: ${address}. Error: ${JSON.stringify(error)}`) | ||
return | ||
} | ||
|
||
const { force = false } = options | ||
|
||
// If the validator is cached and force is not true, return it | ||
if (!force && validators.has(address)) { | ||
return validators.get(address) | ||
} | ||
|
||
// Check if the validator already exists in the database | ||
let validatorId = await useDrizzle() | ||
.select({ id: tables.validators.id }) | ||
.from(tables.validators) | ||
.where(eq(tables.validators.address, address)) | ||
.get() | ||
.then(r => r?.id) | ||
|
||
// If the validator exists and force is not true, return it | ||
if (validatorId && !force) { | ||
validators.set(address, validatorId) | ||
return validatorId | ||
} | ||
|
||
consola.info(`${force ? 'Updating' : 'Storing'} validator ${address}`) | ||
|
||
const icon = (await Identicons.default?.toDataUrl(address)) || '' | ||
if (validatorId) { | ||
await useDrizzle() | ||
.update(tables.validators) | ||
.set({ ...rest, icon }) | ||
.where(eq(tables.validators.id, validatorId)) | ||
.execute() | ||
} | ||
else { | ||
validatorId = await useDrizzle() | ||
.insert(tables.validators) | ||
.values({ ...rest, address, icon }) | ||
.returning() | ||
.get().then(r => r.id) | ||
} | ||
|
||
validators.set(address, validatorId!) | ||
return validatorId | ||
} | ||
|
||
export async function fetchValidatorsScoreByIds(validatorIds: number[]): Result<ValidatorScore[]> { | ||
const validators = await useDrizzle() | ||
.select({ | ||
id: tables.validators.id, | ||
name: tables.validators.name, | ||
address: tables.validators.address, | ||
fee: tables.validators.fee, | ||
payoutType: tables.validators.payoutType, | ||
description: tables.validators.description, | ||
icon: tables.validators.icon, | ||
tag: tables.validators.tag, | ||
website: tables.validators.website, | ||
liveness: tables.scores.liveness, | ||
total: tables.scores.total, | ||
size: tables.scores.size, | ||
reliability: tables.scores.reliability, | ||
}) | ||
.from(tables.validators) | ||
.leftJoin(tables.scores, eq(tables.validators.id, tables.scores.validatorId)) | ||
.where(inArray(tables.validators.id, validatorIds)) | ||
.groupBy(tables.validators.id) | ||
.orderBy(desc(tables.scores.total)) | ||
.all() as ValidatorScore[] | ||
return { data: validators, error: undefined } | ||
} | ||
|
||
export interface FetchValidatorsOptions { | ||
onlyPools: boolean | ||
} | ||
|
||
export async function fetchValidators({ onlyPools }: FetchValidatorsOptions): Result<Validator[]> { | ||
const validators = await useDrizzle() | ||
.select() | ||
.from(tables.validators) | ||
.where(onlyPools ? eq(tables.validators.payoutType, PayoutType.Restake) : undefined) | ||
.groupBy(tables.validators.id) | ||
.all() | ||
return { data: validators, error: undefined } | ||
} | ||
|
||
/** | ||
* Import validators from a folder containing .json files. | ||
* | ||
* This function is expected to be used when initializing the database with validators, so it will throw | ||
* an error if the files are not valid and the program should stop. | ||
*/ | ||
export async function importValidatorsFromFiles(folderPath: string) { | ||
const allFiles = await readdir(folderPath) | ||
const files = allFiles | ||
.filter(f => path.extname(f) === '.json') | ||
.filter(f => !f.endsWith('.example.json')) | ||
|
||
for (const file of files) { | ||
const filePath = path.join(folderPath, file) | ||
const fileContent = await readFile(filePath, 'utf8') | ||
|
||
// Validate the file content | ||
const jsonData = JSON.parse(fileContent) | ||
validatorSchema.safeParse(jsonData) | ||
|
||
// Check if the address in the title matches the address in the body | ||
const fileNameAddress = path.basename(file, '.json') | ||
if (jsonData.address !== fileNameAddress) | ||
throw new Error(`Address mismatch in file: ${file}`) | ||
|
||
await storeValidator(jsonData.address, jsonData, { force: true }) | ||
} | ||
} |