/** Build cohort data arrays from brk.metrics */ import { termColors, maxAgeColors, minAgeColors, ageRangeColors, epochColors, geAmountColors, ltAmountColors, amountRangeColors, spendableTypeColors, yearColors, } from "../colors/index.js"; /** * @template {Record} T * @param {T} obj * @returns {[keyof T & string, T[keyof T & string]][]} */ const entries = (obj) => /** @type {[keyof T & string, T[keyof T & string]][]} */ ( Object.entries(obj) ); /** @type {readonly AddressableType[]} */ const ADDRESSABLE_TYPES = [ "p2pk65", "p2pk33", "p2pkh", "p2sh", "p2wpkh", "p2wsh", "p2tr", "p2a", ]; /** @type {(key: SpendableType) => key is AddressableType} */ const isAddressable = (key) => ADDRESSABLE_TYPES.includes(/** @type {any} */ (key)); /** * Build all cohort data from brk tree * @param {Colors} colors * @param {BrkClient} brk */ export function buildCohortData(colors, brk) { const utxoCohorts = brk.metrics.distribution.utxoCohorts; const addressCohorts = brk.metrics.distribution.addressCohorts; const { addrCount } = brk.metrics.distribution; const { TERM_NAMES, EPOCH_NAMES, MAX_AGE_NAMES, MIN_AGE_NAMES, AGE_RANGE_NAMES, GE_AMOUNT_NAMES, LT_AMOUNT_NAMES, AMOUNT_RANGE_NAMES, SPENDABLE_TYPE_NAMES, YEAR_NAMES, } = brk; // Base cohort representing "all" const cohortAll = { name: "", title: "", color: colors.orange, tree: utxoCohorts.all, addrCount: addrCount.all, }; // Term cohorts const shortNames = TERM_NAMES.short; const termShort = { name: shortNames.short, title: shortNames.long, color: colors[termColors.short], tree: utxoCohorts.term.short, }; const longNames = TERM_NAMES.long; const termLong = { name: longNames.short, title: longNames.long, color: colors[termColors.long], tree: utxoCohorts.term.long, }; // Max age cohorts (up to X time) const upToDate = entries(utxoCohorts.maxAge).map(([key, tree]) => { const names = MAX_AGE_NAMES[key]; return { name: names.short, title: `UTXOs ${names.long}`, color: colors[maxAgeColors[key]], tree, }; }); // Min age cohorts (from X time) const fromDate = entries(utxoCohorts.minAge).map(([key, tree]) => { const names = MIN_AGE_NAMES[key]; return { name: names.short, title: `UTXOs ${names.long}`, color: colors[minAgeColors[key]], tree, }; }); // Age range cohorts const dateRange = entries(utxoCohorts.ageRange).map(([key, tree]) => { const names = AGE_RANGE_NAMES[key]; return { name: names.short, title: `UTXOs ${names.long}`, color: colors[ageRangeColors[key]], tree, }; }); // Epoch cohorts const epoch = entries(utxoCohorts.epoch).map(([key, tree]) => { const names = EPOCH_NAMES[key]; return { name: names.short, title: names.long, color: colors[epochColors[key]], tree, }; }); // UTXOs above amount const utxosAboveAmount = entries(utxoCohorts.geAmount).map(([key, tree]) => { const names = GE_AMOUNT_NAMES[key]; return { name: names.short, title: `UTXOs ${names.long}`, color: colors[geAmountColors[key]], tree, }; }); // Addresses above amount const addressesAboveAmount = entries(addressCohorts.geAmount).map( ([key, tree]) => { const names = GE_AMOUNT_NAMES[key]; return { name: names.short, title: `Addresses ${names.long}`, color: colors[geAmountColors[key]], tree, }; }, ); // UTXOs under amount const utxosUnderAmount = entries(utxoCohorts.ltAmount).map(([key, tree]) => { const names = LT_AMOUNT_NAMES[key]; return { name: names.short, title: `UTXOs ${names.long}`, color: colors[ltAmountColors[key]], tree, }; }); // Addresses under amount const addressesUnderAmount = entries(addressCohorts.ltAmount).map( ([key, tree]) => { const names = LT_AMOUNT_NAMES[key]; return { name: names.short, title: `Addresses ${names.long}`, color: colors[ltAmountColors[key]], tree, }; }, ); // UTXOs amount ranges const utxosAmountRanges = entries(utxoCohorts.amountRange).map( ([key, tree]) => { const names = AMOUNT_RANGE_NAMES[key]; return { name: names.short, title: `UTXOs ${names.long}`, color: colors[amountRangeColors[key]], tree, }; }, ); // Addresses amount ranges const addressesAmountRanges = entries(addressCohorts.amountRange).map( ([key, tree]) => { const names = AMOUNT_RANGE_NAMES[key]; return { name: names.short, title: `Addresses ${names.long}`, color: colors[amountRangeColors[key]], tree, }; }, ); // Spendable type cohorts - split by addressability const typeAddressable = ADDRESSABLE_TYPES.map((key) => { const names = SPENDABLE_TYPE_NAMES[key]; return { name: names.short, title: names.short, color: colors[spendableTypeColors[key]], tree: utxoCohorts.type[key], addrCount: addrCount[key], }; }); const typeOther = entries(utxoCohorts.type) .filter(([key]) => !isAddressable(key)) .map(([key, tree]) => { const names = SPENDABLE_TYPE_NAMES[key]; return { name: names.short, title: names.short, color: colors[spendableTypeColors[key]], tree, }; }); // Year cohorts const year = entries(utxoCohorts.year).map(([key, tree]) => { const names = YEAR_NAMES[key]; return { name: names.short, title: names.long, color: colors[yearColors[key]], tree, }; }); return { cohortAll, termShort, termLong, upToDate, fromDate, dateRange, epoch, utxosAboveAmount, addressesAboveAmount, utxosUnderAmount, addressesUnderAmount, utxosAmountRanges, addressesAmountRanges, typeAddressable, typeOther, year, }; }