123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718 |
- import {ApiPromise, WsProvider} from "@polkadot/api";
- import {types} from '@joystream/types'
- import {
- AccountId,
- Balance,
- BalanceOf,
- BlockNumber,
- EraIndex,
- EventRecord,
- Hash,
- Moment
- } from "@polkadot/types/interfaces";
- import {
- CacheEvent,
- Media,
- MintStatistics,
- Statistics,
- WorkersInfo, Channel, SpendingProposals, Bounty
- } from "./types";
- import {Option, u32, Vec} from "@polkadot/types";
- import {ElectionStake, SealedVote, Seats} from "@joystream/types/council";
- import {Mint, MintId} from "@joystream/types/mint";
- import {ContentId, DataObject} from "@joystream/types/media";
- import {ChannelId, PostId, ThreadId} from "@joystream/types/common";
- import {CategoryId} from "@joystream/types/forum";
- import {MemberId, Membership} from "@joystream/types/members";
- import {RewardRelationship, RewardRelationshipId} from "@joystream/types/recurring-rewards";
- import {Stake} from "@joystream/types/stake";
- import {WorkerId} from "@joystream/types/working-group";
- import {Entity, EntityId, PropertyType} from "@joystream/types/content-directory";
- import {ProposalId, Video, VideoId, WorkerOf, } from "@joystream/types/augment-codec/all";
- import {ProposalDetails, ProposalOf} from "@joystream/types/augment/types";
- import {SpendingParams} from "@joystream/types/proposals";
- import * as constants from "constants";
- const fsSync = require('fs');
- const fs = fsSync.promises;
- const parse = require('csv-parse/lib/sync');
- const BURN_ADDRESS = '5D5PhZQNJzcJXVBxwJxZcsutjKPqUPydrvpu6HeiBfMaeKQu';
- const COUNCIL_ROUND_OFFSET = 2;
- const PROVIDER_URL = "ws://localhost:9944";
- const CACHE_FOLDER = "cache";
- const VIDEO_CLASS_iD = 10;
- const CHANNEL_CLASS_iD = 1;
- const SPENDING_PROPOSALS_CATEGORIES_FILE = __dirname + '/../../../documentation/spending_proposal_categories.csv';
- export class StatisticsCollector {
- private api?: ApiPromise;
- private blocksEventsCache: Map<number, CacheEvent[]>;
- private statistics: Statistics;
- constructor() {
- this.blocksEventsCache = new Map<number, CacheEvent[]>();
- this.statistics = new Statistics();
- }
- async getStatistics(startBlock: number, endBlock: number): Promise<Statistics> {
- this.api = await StatisticsCollector.connectApi();
- let startHash = (await this.api.rpc.chain.getBlockHash(startBlock)) as Hash;
- let endHash = (await this.api.rpc.chain.getBlockHash(endBlock)) as Hash;
- this.statistics.startBlock = startBlock;
- this.statistics.endBlock = endBlock;
- this.statistics.newBlocks = endBlock - startBlock;
- this.statistics.percNewBlocks = StatisticsCollector.convertToPercentage(startBlock, endBlock);
- await this.buildBlocksEventCache(startBlock, endBlock);
- await this.fillBasicInfo(startHash, endHash);
- await this.fillTokenGenerationInfo(startBlock, endBlock, startHash, endHash);
- await this.fillMintsInfo(startHash, endHash);
- await this.fillCouncilInfo(startHash, endHash);
- await this.fillCouncilElectionInfo(startBlock);
- await this.fillValidatorInfo(startHash, endHash);
- await this.fillStorageProviderInfo(startBlock, endBlock, startHash, endHash);
- await this.fillCuratorInfo(startHash, endHash);
- await this.fillOperationsInfo(startBlock, endBlock, startHash, endHash);
- await this.fillMembershipInfo(startHash, endHash);
- await this.fillMediaUploadInfo(startHash, endHash);
- await this.fillForumInfo(startHash, endHash);
- await this.api.disconnect();
- return this.statistics;
- }
- async getApprovedBounties() {
- try {
- await fs.access(SPENDING_PROPOSALS_CATEGORIES_FILE, constants.R_OK);
- } catch {
- console.warn('File with the spending proposal categories not found');
- return [];
- }
- const fileContent = await fs.readFile(SPENDING_PROPOSALS_CATEGORIES_FILE);
- let rawBounties = parse(fileContent);
- rawBounties.shift();
- rawBounties = rawBounties.filter((line: string[]) => line[8] == 'Bounties');
- let bounties = rawBounties.map((rawBounty: any) => {
- return new Bounty(rawBounty[0], rawBounty[1], rawBounty[2], rawBounty[3], rawBounty[4], rawBounty[5]);
- });
- return bounties.filter((bounty: Bounty) => bounty.status == "Approved" && bounty.testnet == "Antioch");
- }
- async fillValidatorsRewards() {
- for (let [key, blockEvents] of this.blocksEventsCache) {
- let validatorRewards = blockEvents.filter((event) => {
- return event.section == "staking" && event.method == "Reward";
- });
- for (let validatorReward of validatorRewards) {
- this.statistics.newValidatorRewards += Number(validatorReward.data[1]);
- }
- }
- }
- async computeTokensBurn(){
- let tokensBurned = 0;
- for (let [, blockEvents] of this.blocksEventsCache) {
- let transfers = blockEvents.filter((event) => {
- return event.section == "balances" && event.method == "Transfer";
- });
- for (let transfer of transfers) {
- let receiver = transfer.data[1] as AccountId;
- let amount = transfer.data[2] as Balance;
- if (receiver.toString() == BURN_ADDRESS) {
- tokensBurned += Number(amount);
- }
- }
- }
- return tokensBurned;
- }
- async getFinalizedSpendingProposals(endHash: Hash): Promise<Array<SpendingProposals>> {
- let spendingProposals = new Array<SpendingProposals>();
- for (let [key, blockEvents] of this.blocksEventsCache) {
- let proposalEvents = blockEvents.filter((event) => {
- return event.section == "proposalsEngine" && event.method == "ProposalStatusUpdated";
- });
- for (let proposalEvent of proposalEvents) {
- let statusUpdateData = proposalEvent.data[1] as any;
- if (!(statusUpdateData.finalized && statusUpdateData.finalized.finalizedAt)) {
- continue;
- }
- let proposalId = proposalEvent.data[0] as ProposalId;
- let proposalInfo = await this.api.query.proposalsEngine.proposals.at(endHash, proposalId) as ProposalOf;
- const finalizedData = proposalInfo.status.asFinalized;
- let proposalDetail = await this.api.query.proposalsCodex.proposalDetailsByProposalId.at(endHash, proposalId) as ProposalDetails;
- if (!finalizedData.proposalStatus.isApproved || !proposalDetail.isSpending) {
- continue;
- }
- let approvedData = finalizedData.proposalStatus.asApproved;
- if (!approvedData.isExecuted) {
- continue;
- }
- let spendingParams = proposalDetail.asSpending;
- if (!spendingProposals.some(spendingProposal => (spendingProposal.id == Number(proposalId)))){
- spendingProposals.push(new SpendingProposals(Number(proposalId), proposalInfo.title.toString(), Number(spendingParams[0])));
- }
- }
- }
- return spendingProposals;
- }
- async fillBasicInfo(startHash: Hash, endHash: Hash) {
- let startDate = (await this.api.query.timestamp.now.at(startHash)) as Moment;
- let endDate = (await this.api.query.timestamp.now.at(endHash)) as Moment;
- this.statistics.dateStart = new Date(startDate.toNumber()).toLocaleDateString("en-US");
- this.statistics.dateEnd = new Date(endDate.toNumber()).toLocaleDateString("en-US");
- }
- async fillTokenGenerationInfo(startBlock: number, endBlock: number, startHash: Hash, endHash: Hash) {
- this.statistics.startIssuance = (await this.api.query.balances.totalIssuance.at(startHash) as Balance).toNumber();
- this.statistics.endIssuance = (await this.api.query.balances.totalIssuance.at(endHash) as Balance).toNumber();
- this.statistics.newIssuance = this.statistics.endIssuance - this.statistics.startIssuance;
- this.statistics.percNewIssuance = StatisticsCollector.convertToPercentage(this.statistics.startIssuance, this.statistics.endIssuance);
- this.statistics.newTokensBurn = await this.computeTokensBurn();
- let bounties = await this.getApprovedBounties();
- let spendingProposals = await this.getFinalizedSpendingProposals(endHash);
- this.statistics.bountiesTotalPaid = 0;
- if (bounties) {
- for (let bounty of bounties) {
- let bountySpendingProposal = spendingProposals.find((spendingProposal) => spendingProposal.id == bounty.proposalId);
- if (bountySpendingProposal) {
- this.statistics.bountiesTotalPaid += bountySpendingProposal.spentAmount;
- }
- }
- }
- if (!this.statistics.bountiesTotalPaid) {
- console.warn('No bounties found in ' + SPENDING_PROPOSALS_CATEGORIES_FILE +', trying to find spending proposals of bounties, please check the values!...');
- for (const spendingProposal of spendingProposals) {
- if (spendingProposal.title.toLowerCase().includes("bounty")) {
- this.statistics.bountiesTotalPaid += spendingProposal.spentAmount;
- }
- }
- }
- this.statistics.spendingProposalsTotal = spendingProposals.reduce((n, spendingProposal) => n + spendingProposal.spentAmount, 0);
- let roundNrBlocks = endBlock - startBlock;
- this.statistics.newCouncilRewards = await this.computeCouncilReward(roundNrBlocks, endHash);
- this.statistics.newCouncilRewards = Number(this.statistics.newCouncilRewards.toFixed(2));
- this.statistics.newCuratorRewards = await this.computeCuratorsReward(roundNrBlocks, startHash, endHash);
- this.statistics.newCuratorRewards = Number(this.statistics.newCuratorRewards.toFixed(2));
- }
- async computeCouncilReward(roundNrBlocks: number, endHash: Hash): Promise<number> {
- const payoutInterval = Number((await this.api.query.council.payoutInterval.at(endHash) as Option<BlockNumber>).unwrapOr(0));
- const amountPerPayout = (await this.api.query.council.amountPerPayout.at(endHash) as BalanceOf).toNumber();
- const announcing_period = (await this.api.query.councilElection.announcingPeriod.at(endHash)) as BlockNumber;
- const voting_period = (await this.api.query.councilElection.votingPeriod.at(endHash)) as BlockNumber;
- const revealing_period = (await this.api.query.councilElection.revealingPeriod.at(endHash)) as BlockNumber;
- const new_term_duration = (await this.api.query.councilElection.newTermDuration.at(endHash)) as BlockNumber;
- const termDuration = new_term_duration.toNumber();
- const votingPeriod = voting_period.toNumber();
- const revealingPeriod = revealing_period.toNumber();
- const announcingPeriod = announcing_period.toNumber();
- const nrCouncilMembers = (await this.api.query.council.activeCouncil.at(endHash) as Seats).length
- const totalCouncilRewardsPerBlock = (amountPerPayout && payoutInterval)
- ? (amountPerPayout * nrCouncilMembers) / payoutInterval
- : 0;
- const councilTermDurationRatio = termDuration / (termDuration + votingPeriod + revealingPeriod + announcingPeriod);
- const avgCouncilRewardPerBlock = councilTermDurationRatio * totalCouncilRewardsPerBlock;
- return avgCouncilRewardPerBlock * roundNrBlocks;
- }
- async computeWorkingGroupReward(roundNrBlocks: number, startHash: Hash, endHash: Hash, workingGroup: string): Promise<WorkersInfo> {
- let nextWorkerId = (await this.api.query[workingGroup + 'WorkingGroup'].nextWorkerId.at(startHash) as WorkerId).toNumber();
- let info = new WorkersInfo();
- for (let i = 0; i < nextWorkerId; ++i) {
- let worker = await this.api.query[workingGroup + 'WorkingGroup'].workerById.at(endHash, i) as WorkerOf;
- if (!worker.is_active) {
- continue;
- }
- if (worker.role_stake_profile.isSome) {
- let roleStakeProfile = worker.role_stake_profile.unwrap();
- let stake = await this.api.query.stake.stakes.at(endHash, roleStakeProfile.stake_id) as Stake;
- info.startStake += stake.value.toNumber();
- }
- }
- nextWorkerId = (await this.api.query[workingGroup + 'WorkingGroup'].nextWorkerId.at(endHash) as WorkerId).toNumber();
- let rewardRelationshipIds = Array<RewardRelationshipId>();
- for (let i = 0; i < nextWorkerId; ++i) {
- let worker = await this.api.query[workingGroup + 'WorkingGroup'].workerById.at(endHash, i) as WorkerOf;
- if (!worker.is_active) {
- continue;
- }
- if (worker.reward_relationship.isSome) {
- rewardRelationshipIds.push(worker.reward_relationship.unwrap());
- }
- if (worker.role_stake_profile.isSome) {
- let roleStakeProfile = worker.role_stake_profile.unwrap();
- let stake = await this.api.query.stake.stakes.at(endHash, roleStakeProfile.stake_id) as Stake;
- info.endStake += stake.value.toNumber();
- }
- }
- info.rewards = await this.computeReward(roundNrBlocks, rewardRelationshipIds, endHash);
- info.endNrOfWorkers = nextWorkerId;
- return info;
- }
- async computeCuratorsReward(roundNrBlocks: number, startHash: Hash, endHash: Hash) {
- let nextCuratorId = (await this.api.query.contentDirectoryWorkingGroup.nextWorkerId.at(endHash) as WorkerId).toNumber();
- let rewardRelationshipIds = Array<RewardRelationshipId>();
- for (let i = 0; i < nextCuratorId; ++i) {
- let worker = await this.api.query.contentDirectoryWorkingGroup.workerById.at(endHash, i) as WorkerOf;
- if (!worker.is_active) {
- continue;
- }
- if (worker.reward_relationship.isSome) {
- rewardRelationshipIds.push(worker.reward_relationship.unwrap());
- }
- }
- return this.computeReward(roundNrBlocks, rewardRelationshipIds, endHash);
- }
- async computeReward(roundNrBlocks: number, rewardRelationshipIds: RewardRelationshipId[], hash: Hash) {
- let recurringRewards = await Promise.all(rewardRelationshipIds.map(async (rewardRelationshipId) => {
- return await this.api.query.recurringRewards.rewardRelationships.at(hash, rewardRelationshipId) as RewardRelationship;
- }));
- let rewardPerBlock = 0;
- for (let recurringReward of recurringRewards) {
- const amount = recurringReward.amount_per_payout.toNumber();
- const payoutInterval = recurringReward.payout_interval.unwrapOr(null);
- if (amount && payoutInterval) {
- rewardPerBlock += amount / payoutInterval;
- }
- }
- return rewardPerBlock * roundNrBlocks;
- }
- async fillMintsInfo(startHash: Hash, endHash: Hash) {
- let startNrMints = parseInt((await this.api.query.minting.mintsCreated.at(startHash)).toString());
- let endNrMints = parseInt((await this.api.query.minting.mintsCreated.at(endHash)).toString());
- this.statistics.newMints = endNrMints - startNrMints;
- // statistics.startMinted = 0;
- // statistics.endMinted = 0;
- for (let i = 0; i < startNrMints; ++i) {
- let startMint = (await this.api.query.minting.mints.at(startHash, i)) as Mint;
- // if (!startMint) {
- // continue;
- // }
- let endMint = (await this.api.query.minting.mints.at(endHash, i)) as Mint;
- // let = endMintResult[0];
- // if (!endMint) {
- // continue;
- // }
- let startMintTotal = parseInt(startMint.getField("total_minted").toString());
- let endMintTotal = parseInt(endMint.getField("total_minted").toString());
- // statistics.startMinted += startMintTotal;
- this.statistics.totalMinted += endMintTotal - startMintTotal;
- this.statistics.totalMintCapacityIncrease += parseInt(endMint.getField("capacity").toString()) - parseInt(startMint.getField("capacity").toString());
- }
- for (let i = startNrMints; i < endNrMints; ++i) {
- let endMint = await this.api.query.minting.mints.at(endHash, i) as Mint;
- if (!endMint) {
- return;
- }
- this.statistics.totalMinted = parseInt(endMint.getField("total_minted").toString());
- }
- let councilMint = (await this.api.query.council.councilMint.at(endHash)) as MintId;
- let councilMintStatistics = await this.computeMintInfo(councilMint, startHash, endHash);
- this.statistics.startCouncilMinted = councilMintStatistics.startMinted;
- this.statistics.endCouncilMinted = councilMintStatistics.endMinted;
- this.statistics.newCouncilMinted = councilMintStatistics.diffMinted;
- this.statistics.percNewCouncilMinted = councilMintStatistics.percMinted;
- let curatorMint = (await this.api.query.contentDirectoryWorkingGroup.mint.at(endHash)) as MintId;
- let curatorMintStatistics = await this.computeMintInfo(curatorMint, startHash, endHash);
- this.statistics.startCuratorMinted = curatorMintStatistics.startMinted;
- this.statistics.endCuratorMinted = curatorMintStatistics.endMinted;
- this.statistics.newCuratorMinted = curatorMintStatistics.diffMinted;
- this.statistics.percCuratorMinted = curatorMintStatistics.percMinted;
- let storageProviderMint = (await this.api.query.storageWorkingGroup.mint.at(endHash)) as MintId;
- let storageProviderMintStatistics = await this.computeMintInfo(storageProviderMint, startHash, endHash);
- this.statistics.startStorageMinted = storageProviderMintStatistics.startMinted;
- this.statistics.endStorageMinted = storageProviderMintStatistics.endMinted;
- this.statistics.newStorageMinted = storageProviderMintStatistics.diffMinted;
- this.statistics.percStorageMinted = storageProviderMintStatistics.percMinted;
- let operationsProviderMint = (await this.api.query.operationsWorkingGroup.mint.at(endHash)) as MintId;
- let operationsProviderMintStatistics = await this.computeMintInfo(operationsProviderMint, startHash, endHash);
- this.statistics.startOperationsMinted = operationsProviderMintStatistics.startMinted;
- this.statistics.endOperationsMinted = operationsProviderMintStatistics.endMinted;
- this.statistics.newOperationsMinted = operationsProviderMintStatistics.diffMinted;
- this.statistics.percOperationsMinted = operationsProviderMintStatistics.percMinted;
- }
- async computeMintInfo(mintId: MintId, startHash: Hash, endHash: Hash): Promise<MintStatistics> {
- // if (mintId.toString() == "0") {
- // return new MintStatistics(0, 0, 0);
- // }
- let startMint = await this.api.query.minting.mints.at(startHash, mintId) as Mint;
- // let startMint = startMintResult[0] as unknown as Mint;
- // if (!startMint) {
- // return new MintStatistics(0, 0, 0);
- // }
- let endMint = await this.api.query.minting.mints.at(endHash, mintId) as Mint;
- // let endMint = endMintResult[0] as unknown as Mint;
- // if (!endMint) {
- // return new MintStatistics(0, 0, 0);
- // }
- let mintStatistics = new MintStatistics();
- mintStatistics.startMinted = parseInt(startMint.getField('total_minted').toString());
- mintStatistics.endMinted = parseInt(endMint.getField('total_minted').toString());
- mintStatistics.diffMinted = mintStatistics.endMinted - mintStatistics.startMinted;
- mintStatistics.percMinted = StatisticsCollector.convertToPercentage(mintStatistics.startMinted, mintStatistics.endMinted);
- return mintStatistics;
- }
- async fillCouncilInfo(startHash: Hash, endHash: Hash) {
- this.statistics.councilRound = (await this.api.query.councilElection.round.at(startHash) as u32).toNumber() - COUNCIL_ROUND_OFFSET;
- this.statistics.councilMembers = (await this.api.query.councilElection.councilSize.at(startHash) as u32).toNumber();
- let startNrProposals = await this.api.query.proposalsEngine.proposalCount.at(startHash) as u32;
- let endNrProposals = await this.api.query.proposalsEngine.proposalCount.at(endHash) as u32;
- this.statistics.newProposals = endNrProposals.toNumber() - startNrProposals.toNumber();
- let approvedProposals = new Set();
- for (let [key, blockEvents] of this.blocksEventsCache) {
- for (let event of blockEvents) {
- if (event.section == "proposalsEngine" && event.method == "ProposalStatusUpdated") {
- let statusUpdateData = event.data[1] as any;
- let finalizeData = statusUpdateData.finalized as any
- if (finalizeData && finalizeData.proposalStatus.approved) {
- approvedProposals.add(Number(event.data[0]));
- }
- }
- }
- }
- this.statistics.newApprovedProposals = approvedProposals.size;
- }
- async fillCouncilElectionInfo(startBlock: number) {
- let startBlockHash = await this.api.rpc.chain.getBlockHash(startBlock);
- let events = await this.api.query.system.events.at(startBlockHash) as Vec<EventRecord>;
- let isStartBlockFirstCouncilBlock = events.some((event) => {
- return event.event.section == "councilElection" && event.event.method == "CouncilElected";
- });
- if (!isStartBlockFirstCouncilBlock) {
- console.warn('Note: The given start block is not the first block of the council round so council election information will be empty');
- return;
- }
- let previousCouncilRoundLastBlock = startBlock - 1;
- let previousCouncilRoundLastBlockHash = await this.api.rpc.chain.getBlockHash(previousCouncilRoundLastBlock);
- let applicants = await this.api.query.councilElection.applicants.at(previousCouncilRoundLastBlockHash) as Vec<AccountId>;
- this.statistics.electionApplicants = applicants.length;
- for (let applicant of applicants) {
- let applicantStakes = await this.api.query.councilElection.applicantStakes.at(previousCouncilRoundLastBlockHash, applicant) as unknown as ElectionStake;
- this.statistics.electionApplicantsStakes += applicantStakes.new.toNumber();
- }
- // let seats = await this.api.query.council.activeCouncil.at(startBlockHash) as Seats;
- //TODO: Find a more accurate way of getting the votes
- const votes = await this.api.query.councilElection.commitments.at(previousCouncilRoundLastBlockHash) as Vec<Hash>;
- this.statistics.electionVotes = votes.length;
- }
- async fillValidatorInfo(startHash: Hash, endHash: Hash) {
- let startTimestamp = await this.api.query.timestamp.now.at(startHash) as unknown as Moment;
- let endTimestamp = await this.api.query.timestamp.now.at(endHash) as unknown as Moment;
- let avgBlockProduction = (((endTimestamp.toNumber() - startTimestamp.toNumber())
- / 1000) / this.statistics.newBlocks);
- this.statistics.avgBlockProduction = Number(avgBlockProduction.toFixed(2));
- let maxStartValidators = (await this.api.query.staking.validatorCount.at(startHash) as u32).toNumber();
- let startValidators = await this.findActiveValidators(startHash, false);
- this.statistics.startValidators = startValidators.length + " / " + maxStartValidators;
- let maxEndValidators = (await this.api.query.staking.validatorCount.at(endHash) as u32).toNumber();
- let endValidators = await this.findActiveValidators(endHash, true);
- this.statistics.endValidators = endValidators.length + " / " + maxEndValidators;
- this.statistics.percValidators = StatisticsCollector.convertToPercentage(startValidators.length, endValidators.length);
- const startEra = await this.api.query.staking.currentEra.at(startHash) as Option<EraIndex>;
- this.statistics.startValidatorsStake = (await this.api.query.staking.erasTotalStake.at(startHash, startEra.unwrap())).toNumber();
- const endEra = await this.api.query.staking.currentEra.at(endHash) as Option<EraIndex>;
- this.statistics.endValidatorsStake = (await this.api.query.staking.erasTotalStake.at(endHash, endEra.unwrap())).toNumber();
- this.statistics.percNewValidatorsStake = StatisticsCollector.convertToPercentage(this.statistics.startValidatorsStake, this.statistics.endValidatorsStake);
- await this.fillValidatorsRewards();
- }
- async findActiveValidators(hash: Hash, searchPreviousBlocks: boolean): Promise<AccountId[]> {
- const block = await this.api.rpc.chain.getBlock(hash);
- let currentBlockNr = block.block.header.number.toNumber();
- let activeValidators;
- do {
- let currentHash = (await this.api.rpc.chain.getBlockHash(currentBlockNr)) as Hash;
- let allValidators = await this.api.query.staking.snapshotValidators.at(currentHash) as Option<Vec<AccountId>>;
- if (!allValidators.isEmpty) {
- let max = (await this.api.query.staking.validatorCount.at(currentHash)).toNumber();
- activeValidators = Array.from(allValidators.unwrap()).slice(0, max);
- }
- if (searchPreviousBlocks) {
- --currentBlockNr;
- } else {
- ++currentBlockNr;
- }
- } while (activeValidators == undefined);
- return activeValidators;
- }
- async fillStorageProviderInfo(startBlock: number, endBlock: number, startHash: Hash, endHash: Hash) {
- let roundNrBlocks = endBlock - startBlock;
- let storageProvidersRewards = await this.computeWorkingGroupReward(roundNrBlocks, startHash, endHash, 'storage');
- this.statistics.newStorageProviderReward = storageProvidersRewards.rewards;
- this.statistics.newStorageProviderReward = Number(this.statistics.newStorageProviderReward.toFixed(2));
- this.statistics.startStorageProvidersStake = storageProvidersRewards.startStake;
- this.statistics.endStorageProvidersStake = storageProvidersRewards.endStake;
- this.statistics.percNewStorageProviderStake = StatisticsCollector.convertToPercentage(this.statistics.startStorageProvidersStake, this.statistics.endStorageProvidersStake);
- this.statistics.startStorageProviders = await this.api.query.storageWorkingGroup.activeWorkerCount.at(startHash);
- this.statistics.endStorageProviders = await this.api.query.storageWorkingGroup.activeWorkerCount.at(endHash);
- this.statistics.percNewStorageProviders = StatisticsCollector.convertToPercentage(this.statistics.startStorageProviders, this.statistics.endStorageProviders);
- let nextWorkerId = Number(await this.api.query.storageWorkingGroup.nextWorkerId.at(endHash));
- this.statistics.storageProviders = "";
- for (let i = 0; i < nextWorkerId; ++i) {
- let storageProvider = await this.api.query.storageWorkingGroup.workerById.at(endHash, i) as WorkerOf;
- if (!storageProvider.is_active) {
- continue;
- }
- let membership = await this.api.query.members.membershipById.at(endHash, storageProvider.member_id) as Membership;
- this.statistics.storageProviders += "@" + membership.handle + " | (" + membership.root_account + ") \n";
- }
- }
- async fillCuratorInfo(startHash: Hash, endHash: Hash) {
- this.statistics.startCurators = Number(await this.api.query.contentDirectoryWorkingGroup.activeWorkerCount.at(startHash));
- this.statistics.endCurators = Number(await this.api.query.contentDirectoryWorkingGroup.activeWorkerCount.at(endHash));
- this.statistics.percNewCurators = StatisticsCollector.convertToPercentage(this.statistics.startCurators, this.statistics.endCurators);
- let nextCuratorId = Number(await this.api.query.contentDirectoryWorkingGroup.nextWorkerId.at(endHash));
- this.statistics.curators = "";
- for (let i = 0; i < nextCuratorId; i++) {
- let worker = await this.api.query.contentDirectoryWorkingGroup.workerById.at(endHash, i) as WorkerOf;
- if (!worker.is_active) {
- continue;
- }
- let curatorMembership = await this.api.query.members.membershipById.at(endHash, worker.member_id) as Membership;
- this.statistics.curators += "@" + curatorMembership.handle + " | (" + curatorMembership.root_account + ") \n";
- }
- }
- async fillOperationsInfo(startBlock: number, endBlock: number, startHash: Hash, endHash: Hash) {
- let roundNrBlocks = endBlock - startBlock;
- let operationsRewards = await this.computeWorkingGroupReward(roundNrBlocks, startHash, endHash, 'operations');
- this.statistics.newOperationsReward = operationsRewards.rewards;
- this.statistics.newOperationsReward = Number(this.statistics.newOperationsReward.toFixed(2));
- this.statistics.startOperationsStake = operationsRewards.startStake;
- this.statistics.endOperationsStake = operationsRewards.endStake;
- this.statistics.percNewOperationstake = StatisticsCollector.convertToPercentage(this.statistics.startOperationsStake, this.statistics.endOperationsStake);
- this.statistics.startOperationsWorkers = Number(await this.api.query.operationsWorkingGroup.activeWorkerCount.at(startHash));
- this.statistics.endOperationsWorkers = Number(await this.api.query.operationsWorkingGroup.activeWorkerCount.at(endHash));
- this.statistics.percNewOperationsWorkers = StatisticsCollector.convertToPercentage(this.statistics.startOperationsWorkers, this.statistics.endOperationsWorkers);
- let nextOperationsWorkerId = Number(await this.api.query.operationsWorkingGroup.nextWorkerId.at(endHash));
- this.statistics.operations = "";
- for (let i = 0; i < nextOperationsWorkerId; i++) {
- let worker = await this.api.query.operationsWorkingGroup.workerById.at(endHash, i) as WorkerOf;
- if (!worker.is_active) {
- continue;
- }
- let operationMembership = await this.api.query.members.membershipById.at(endHash, worker.member_id) as Membership;
- this.statistics.operations += "@" + operationMembership.handle + " | (" + operationMembership.root_account + ") \n";
- }
- }
- async fillMembershipInfo(startHash: Hash, endHash: Hash) {
- this.statistics.startMembers = (await this.api.query.members.nextMemberId.at(startHash) as MemberId).toNumber();
- this.statistics.endMembers = (await this.api.query.members.nextMemberId.at(endHash) as MemberId).toNumber();
- this.statistics.newMembers = this.statistics.endMembers - this.statistics.startMembers;
- this.statistics.percNewMembers = StatisticsCollector.convertToPercentage(this.statistics.startMembers, this.statistics.endMembers);
- }
- async fillMediaUploadInfo(startHash: Hash, endHash: Hash) {
- let startVideos = (await this.api.query.content.nextVideoId.at(startHash) as VideoId).toNumber();
- let endVideos = (await this.api.query.content.nextVideoId.at(endHash) as VideoId).toNumber();
- this.statistics.startMedia = startVideos;
- this.statistics.endMedia = endVideos;
- this.statistics.percNewMedia = StatisticsCollector.convertToPercentage(this.statistics.startMedia, this.statistics.endMedia);
- let startChannels = (await this.api.query.content.nextChannelId.at(startHash) as ChannelId).toNumber();
- let endChannels = (await this.api.query.content.nextChannelId.at(endHash) as ChannelId).toNumber();
- this.statistics.startChannels = startChannels;
- this.statistics.endChannels = endChannels;
- this.statistics.percNewChannels = StatisticsCollector.convertToPercentage(this.statistics.startChannels, this.statistics.endChannels);
- let dataObjects = await this.api.query.dataDirectory.dataByContentId.entries() as unknown as Map<ContentId, DataObject>;
- let startObjects = new Map<ContentId, DataObject>();
- let endObjects = new Map<ContentId, DataObject>();
- const startBlock = await this.api.rpc.chain.getBlock(startHash);
- const endBlock = await this.api.rpc.chain.getBlock(endHash);
- for (let [key, dataObject] of dataObjects) {
- if (dataObject.added_at.block.toNumber() < startBlock.block.header.number.toNumber()) {
- startObjects.set(key, dataObject);
- this.statistics.startUsedSpace += dataObject.size_in_bytes.toNumber() / 1024 / 1024;
- }
- if (dataObject.added_at.block.toNumber() < endBlock.block.header.number.toNumber()) {
- endObjects.set(key, dataObject);
- this.statistics.endUsedSpace += dataObject.size_in_bytes.toNumber() / 1024 / 1024;
- }
- }
- this.statistics.startUsedSpace = Number(this.statistics.startUsedSpace.toFixed(2));
- this.statistics.endUsedSpace = Number(this.statistics.endUsedSpace.toFixed(2));
- this.statistics.percNewUsedSpace = StatisticsCollector.convertToPercentage(this.statistics.startUsedSpace, this.statistics.endUsedSpace);
- }
- async fillForumInfo(startHash: Hash, endHash: Hash) {
- let startPostId = await this.api.query.forum.nextPostId.at(startHash) as PostId;
- let endPostId = await this.api.query.forum.nextPostId.at(endHash) as PostId;
- this.statistics.startPosts = startPostId.toNumber();
- this.statistics.endPosts = endPostId.toNumber();
- this.statistics.newPosts = this.statistics.endPosts - this.statistics.startPosts;
- this.statistics.percNewPosts = StatisticsCollector.convertToPercentage(this.statistics.startPosts, this.statistics.endPosts);
- let startThreadId = ((await this.api.query.forum.nextThreadId.at(startHash)) as unknown) as ThreadId;
- let endThreadId = ((await this.api.query.forum.nextThreadId.at(endHash)) as unknown) as ThreadId;
- this.statistics.startThreads = startThreadId.toNumber();
- this.statistics.endThreads = endThreadId.toNumber();
- this.statistics.newThreads = this.statistics.endThreads - this.statistics.startThreads;
- this.statistics.percNewThreads = StatisticsCollector.convertToPercentage(this.statistics.startThreads, this.statistics.endThreads);
- let startCategoryId = (await this.api.query.forum.nextCategoryId.at(startHash)) as CategoryId;
- let endCategoryId = (await this.api.query.forum.nextCategoryId.at(endHash)) as CategoryId;
- this.statistics.startCategories = startCategoryId.toNumber();
- this.statistics.endCategories = endCategoryId.toNumber();
- this.statistics.newCategories = this.statistics.endCategories - this.statistics.startCategories;
- this.statistics.perNewCategories = StatisticsCollector.convertToPercentage(this.statistics.startCategories, this.statistics.endCategories);
- }
- static convertToPercentage(previousValue: number, newValue: number): number {
- if (previousValue == 0) {
- return newValue > 0 ? Infinity : 0;
- }
- return Number((newValue * 100 / previousValue - 100).toFixed(2));
- }
- async buildBlocksEventCache(startBlock: number, endBlock: number) {
- let cacheFile = CACHE_FOLDER + '/' + startBlock + '-' + endBlock + '.json';
- let exists = await fs.access(cacheFile, fsSync.constants.R_OK).then(() => true)
- .catch(() => false);
- // let exists = false;
- if (!exists) {
- console.log('Building events cache...');
- let blocksEvents = new Map<number, CacheEvent[]>();
- for (let i = startBlock; i < endBlock; ++i) {
- process.stdout.write('\rCaching block: ' + i + ' until ' + endBlock);
- const blockHash: Hash = await this.api.rpc.chain.getBlockHash(i);
- let eventRecord = await this.api.query.system.events.at(blockHash) as Vec<EventRecord>;
- let cacheEvents = new Array<CacheEvent>();
- for (let event of eventRecord) {
- cacheEvents.push(new CacheEvent(event.event.section, event.event.method, event.event.data));
- }
- blocksEvents.set(i, cacheEvents);
- }
- console.log('\nFinish events cache...');
- let jsonOutput = JSON.stringify(Array.from(blocksEvents.entries()), null, 2);
- await fs.writeFile(cacheFile, jsonOutput);
- this.blocksEventsCache = new Map(JSON.parse(jsonOutput));
- } else {
- console.log('Cache file found, loading it...');
- let fileData = await fs.readFile(cacheFile);
- this.blocksEventsCache = new Map(JSON.parse(fileData));
- console.log('Cache file loaded...');
- }
- }
- static async connectApi(): Promise<ApiPromise> {
- // const provider = new WsProvider('wss://testnet.joystream.org:9944');
- const provider = new WsProvider(PROVIDER_URL);
- // Create the API and wait until ready
- return await ApiPromise.create({provider, types});
- }
- }
|