Merge pull request #5539 from BitcoinMechanic/add-miner-name
Show miner name on block timeline
This commit is contained in:
		
						commit
						1f84e1722f
					
				| @ -34,6 +34,7 @@ import { calculateFastBlockCpfp, calculateGoodBlockCpfp } from './cpfp'; | |||||||
| import mempool from './mempool'; | import mempool from './mempool'; | ||||||
| import CpfpRepository from '../repositories/CpfpRepository'; | import CpfpRepository from '../repositories/CpfpRepository'; | ||||||
| import accelerationApi from './services/acceleration'; | import accelerationApi from './services/acceleration'; | ||||||
|  | import { parseDATUMTemplateCreator } from '../utils/bitcoin-script'; | ||||||
| 
 | 
 | ||||||
| class Blocks { | class Blocks { | ||||||
|   private blocks: BlockExtended[] = []; |   private blocks: BlockExtended[] = []; | ||||||
| @ -342,7 +343,12 @@ class Blocks { | |||||||
|           id: pool.uniqueId, |           id: pool.uniqueId, | ||||||
|           name: pool.name, |           name: pool.name, | ||||||
|           slug: pool.slug, |           slug: pool.slug, | ||||||
|  |           minerNames: null, | ||||||
|         }; |         }; | ||||||
|  | 
 | ||||||
|  |         if (extras.pool.name === 'OCEAN') { | ||||||
|  |           extras.pool.minerNames = parseDATUMTemplateCreator(extras.coinbaseRaw); | ||||||
|  |         } | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       extras.matchRate = null; |       extras.matchRate = null; | ||||||
|  | |||||||
| @ -299,6 +299,7 @@ export interface BlockExtension { | |||||||
|     id: number; // Note - This is the `unique_id`, not to mix with the auto increment `id`
 |     id: number; // Note - This is the `unique_id`, not to mix with the auto increment `id`
 | ||||||
|     name: string; |     name: string; | ||||||
|     slug: string; |     slug: string; | ||||||
|  |     minerNames: string[] | null; | ||||||
|   }; |   }; | ||||||
|   avgFee: number; |   avgFee: number; | ||||||
|   avgFeeRate: number; |   avgFeeRate: number; | ||||||
|  | |||||||
| @ -14,6 +14,7 @@ import chainTips from '../api/chain-tips'; | |||||||
| import blocks from '../api/blocks'; | import blocks from '../api/blocks'; | ||||||
| import BlocksAuditsRepository from './BlocksAuditsRepository'; | import BlocksAuditsRepository from './BlocksAuditsRepository'; | ||||||
| import transactionUtils from '../api/transaction-utils'; | import transactionUtils from '../api/transaction-utils'; | ||||||
|  | import { parseDATUMTemplateCreator } from '../utils/bitcoin-script'; | ||||||
| 
 | 
 | ||||||
| interface DatabaseBlock { | interface DatabaseBlock { | ||||||
|   id: string; |   id: string; | ||||||
| @ -1054,6 +1055,7 @@ class BlocksRepository { | |||||||
|       id: dbBlk.poolId, |       id: dbBlk.poolId, | ||||||
|       name: dbBlk.poolName, |       name: dbBlk.poolName, | ||||||
|       slug: dbBlk.poolSlug, |       slug: dbBlk.poolSlug, | ||||||
|  |       minerNames: null, | ||||||
|     }; |     }; | ||||||
|     extras.avgFee = dbBlk.avgFee; |     extras.avgFee = dbBlk.avgFee; | ||||||
|     extras.avgFeeRate = dbBlk.avgFeeRate; |     extras.avgFeeRate = dbBlk.avgFeeRate; | ||||||
| @ -1123,6 +1125,10 @@ class BlocksRepository { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     if (extras.pool.name === 'OCEAN') { | ||||||
|  |       extras.pool.minerNames = parseDATUMTemplateCreator(extras.coinbaseRaw); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     blk.extras = <BlockExtension>extras; |     blk.extras = <BlockExtension>extras; | ||||||
|     return <BlockExtended>blk; |     return <BlockExtended>blk; | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -201,3 +201,27 @@ export function getVarIntLength(n: number): number { | |||||||
|     return 9; |     return 9; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | /** Extracts miner names from a DATUM coinbase transaction */ | ||||||
|  | export function parseDATUMTemplateCreator(coinbaseRaw: string): string[] | null { | ||||||
|  |   let bytes: number[] = []; | ||||||
|  |   for (let c = 0; c < coinbaseRaw.length; c += 2) { | ||||||
|  |       bytes.push(parseInt(coinbaseRaw.slice(c, c + 2), 16)); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // Skip block height
 | ||||||
|  |   let tagLengthByte = 1 + bytes[0]; | ||||||
|  | 
 | ||||||
|  |   let tagsLength = bytes[tagLengthByte]; | ||||||
|  |   if (tagsLength == 0x4c) { | ||||||
|  |     tagLengthByte += 1; | ||||||
|  |     tagsLength = bytes[tagLengthByte]; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const tagStart = tagLengthByte + 1; | ||||||
|  |   const tags = bytes.slice(tagStart, tagStart + tagsLength); | ||||||
|  |   let tagString = String.fromCharCode(...tags); | ||||||
|  |   tagString = tagString.replace('\x00', ''); | ||||||
|  | 
 | ||||||
|  |   return tagString.split('\x0f'); | ||||||
|  | } | ||||||
| @ -53,14 +53,32 @@ | |||||||
|             <td i18n="block.miner">Miner</td> |             <td i18n="block.miner">Miner</td> | ||||||
|             <td *ngIf="stateService.env.MINING_DASHBOARD"> |             <td *ngIf="stateService.env.MINING_DASHBOARD"> | ||||||
|               <a placement="bottom" [routerLink]="['/mining/pool' | relativeUrl, block.extras.pool.slug]" class="badge" style="color: #FFF;padding:0;"> |               <a placement="bottom" [routerLink]="['/mining/pool' | relativeUrl, block.extras.pool.slug]" class="badge" style="color: #FFF;padding:0;"> | ||||||
|  |                 <ng-container *ngIf="block.extras.pool.minerNames != undefined && block.extras.pool.minerNames.length > 1 && block.extras.pool.minerNames[1] != ''; else centralisedPool"> | ||||||
|  |                   {{ block.extras.pool.minerNames[1] }} | ||||||
|  |                   <div class="on-pool"> | ||||||
|  |                     on | ||||||
|  |                     <img class="pool-logo" [src]="'/resources/mining-pools/' + block.extras.pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + block.extras.pool.name + ' mining pool'"> | ||||||
|  |                     {{ block.extras.pool.name}} | ||||||
|  |                   </div> | ||||||
|  |                 </ng-container> | ||||||
|  |                 <ng-template #centralisedPool> | ||||||
|                   <img class="pool-logo" [src]="'/resources/mining-pools/' + block.extras.pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + block.extras.pool.name + ' mining pool'"> |                   <img class="pool-logo" [src]="'/resources/mining-pools/' + block.extras.pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + block.extras.pool.name + ' mining pool'"> | ||||||
|                   {{ block.extras.pool.name }} |                   {{ block.extras.pool.name }} | ||||||
|  |                 </ng-template> | ||||||
|               </a> |               </a> | ||||||
|             </td> |             </td> | ||||||
|             <td *ngIf="!stateService.env.MINING_DASHBOARD && stateService.env.BASE_MODULE === 'mempool'"> |             <td *ngIf="!stateService.env.MINING_DASHBOARD && stateService.env.BASE_MODULE === 'mempool'"> | ||||||
|               <span [attr.data-cy]="'block-details-miner-badge'" placement="bottom" class="badge" |               <span [attr.data-cy]="'block-details-miner-badge'" placement="bottom" class="badge" | ||||||
|                 [class]="!block?.extras.pool.name || block?.extras.pool.slug === 'unknown' ? 'badge-secondary' : 'badge-primary'"> |                 [class]="!block?.extras.pool.name || block?.extras.pool.slug === 'unknown' ? 'badge-secondary' : 'badge-primary'"> | ||||||
|  |                 <ng-container *ngIf="block?.extras.pool.minerNames != undefined && block?.extras.pool.minerNames.length > 1 && block?.extras.pool.minerNames[1] != ''; else centralisedPool"> | ||||||
|  |                   {{ block?.extras.pool.minerNames[1] }} | ||||||
|  |                   <div class="on-pool"> | ||||||
|  |                     on {{ block?.extras.pool.name }} | ||||||
|  |                   </div> | ||||||
|  |                 </ng-container> | ||||||
|  |                 <ng-template #centralisedPool> | ||||||
|                   {{ block?.extras.pool.name }} |                   {{ block?.extras.pool.name }} | ||||||
|  |                 </ng-template> | ||||||
|               </span> |               </span> | ||||||
|             </td> |             </td> | ||||||
|           </tr> |           </tr> | ||||||
|  | |||||||
| @ -182,8 +182,16 @@ | |||||||
|         <td i18n="block.miner">Miner</td> |         <td i18n="block.miner">Miner</td> | ||||||
|         <td *ngIf="stateService.env.MINING_DASHBOARD"> |         <td *ngIf="stateService.env.MINING_DASHBOARD"> | ||||||
|           <a placement="bottom" [routerLink]="['/mining/pool' | relativeUrl, block.extras.pool.slug]" class="badge" style="color: #FFF;padding:0;"> |           <a placement="bottom" [routerLink]="['/mining/pool' | relativeUrl, block.extras.pool.slug]" class="badge" style="color: #FFF;padding:0;"> | ||||||
|  |             <div class="on-pool-container" *ngIf="block.extras.pool.minerNames != undefined && block.extras.pool.minerNames.length > 1 && block.extras.pool.minerNames[1] != ''; else centralisedPool"> | ||||||
|  |               {{ block.extras.pool.minerNames[1] }} | ||||||
|  |               <div class="on-pool"> | ||||||
|                 <img class="pool-logo" [src]="'/resources/mining-pools/' + block.extras.pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + block.extras.pool.name + ' mining pool'"> |                 <img class="pool-logo" [src]="'/resources/mining-pools/' + block.extras.pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + block.extras.pool.name + ' mining pool'"> | ||||||
|                 {{ block.extras.pool.name }} |                 {{ block.extras.pool.name }} | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |             <ng-template #centralisedPool> | ||||||
|  |               <img class="pool-logo" [src]="'/resources/mining-pools/' + block.extras.pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + block.extras.pool.name + ' mining pool'"> {{ block.extras.pool.name }} | ||||||
|  |             </ng-template> | ||||||
|           </a> |           </a> | ||||||
|         </td> |         </td> | ||||||
|         <td *ngIf="!stateService.env.MINING_DASHBOARD && stateService.env.BASE_MODULE === 'mempool'"> |         <td *ngIf="!stateService.env.MINING_DASHBOARD && stateService.env.BASE_MODULE === 'mempool'"> | ||||||
|  | |||||||
| @ -81,6 +81,27 @@ h1 { | |||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .on-pool-container { | ||||||
|  |   display: inline; | ||||||
|  |   flex-direction: row; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .on-pool { | ||||||
|  |   background-color: var(--bg); | ||||||
|  |   display: inline-block; | ||||||
|  |   margin-top: 4px; | ||||||
|  |   padding: .25em .4em; | ||||||
|  |   border-radius: .25rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .pool-logo { | ||||||
|  |   width: 25px; | ||||||
|  |   height: 25px; | ||||||
|  |   position: relative; | ||||||
|  |   top: -1px; | ||||||
|  |   margin-right: 2px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .row { | .row { | ||||||
| 	flex-direction: column; | 	flex-direction: column; | ||||||
| 	@media (min-width: 768px) { | 	@media (min-width: 768px) { | ||||||
|  | |||||||
| @ -61,8 +61,13 @@ | |||||||
|         </div> |         </div> | ||||||
|         <div class="animated" *ngIf="block.extras?.pool != undefined && showPools"> |         <div class="animated" *ngIf="block.extras?.pool != undefined && showPools"> | ||||||
|           <a [attr.data-cy]="'bitcoin-block-' + offset + '-index-' + i + '-pool'" class="badge" [routerLink]="[('/mining/pool/' + block.extras.pool.slug) | relativeUrl]"> |           <a [attr.data-cy]="'bitcoin-block-' + offset + '-index-' + i + '-pool'" class="badge" [routerLink]="[('/mining/pool/' + block.extras.pool.slug) | relativeUrl]"> | ||||||
|             <img class="pool-logo" [src]="'/resources/mining-pools/' + block.extras.pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + block.extras.pool.name + ' mining pool'">  |             <div *ngIf="block.extras.pool.minerNames != undefined && block.extras.pool.minerNames.length > 1 && block.extras.pool.minerNames[1] != ''; else centralisedPool"> | ||||||
|             {{ block.extras.pool.name}} |               <img [ngbTooltip]="block.extras.pool.name" class="pool-logo faded" [src]="'/resources/mining-pools/' + block.extras.pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + block.extras.pool.name + ' mining pool'"> | ||||||
|  |               {{ block.extras.pool.minerNames[1] }} | ||||||
|  |             </div> | ||||||
|  |             <ng-template #centralisedPool> | ||||||
|  |               <img class="pool-logo" [src]="'/resources/mining-pools/' + block.extras.pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + block.extras.pool.name + ' mining pool'"> {{ block.extras.pool.name }} | ||||||
|  |             </ng-template> | ||||||
|           </a> |           </a> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|  | |||||||
| @ -19,6 +19,38 @@ | |||||||
|   pointer-events: none; |   pointer-events: none; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .on-pool-name-text { | ||||||
|  |   display: inline-block; | ||||||
|  |   padding-top: 2px; | ||||||
|  |   font-weight: normal; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | .on-pool { | ||||||
|  |   align-items: center; | ||||||
|  |   background-color: var(--bg); | ||||||
|  |   display: inline-block; | ||||||
|  |   margin-top: 4px; | ||||||
|  |   padding: .25em .4em; | ||||||
|  |   border-radius: .25rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .on-pool-container { | ||||||
|  |   align-items: center; | ||||||
|  |   position: relative; | ||||||
|  |   top: -8px; | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .on-pool-container.selected { | ||||||
|  |   top: 0px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .pool-container { | ||||||
|  |   margin-top: 12px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .mined-block { | .mined-block { | ||||||
|   position: absolute; |   position: absolute; | ||||||
|   top: 0px; |   top: 0px; | ||||||
| @ -168,6 +200,10 @@ | |||||||
|   margin-right: 2px; |   margin-right: 2px; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .pool-logo.faded { | ||||||
|  |   filter: grayscale(100%) brightness(1.5); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .animated { | .animated { | ||||||
|   transition: all 0.15s ease-in-out; |   transition: all 0.15s ease-in-out; | ||||||
|   white-space: nowrap; |   white-space: nowrap; | ||||||
|  | |||||||
| @ -281,6 +281,15 @@ export class BlockchainBlocksComponent implements OnInit, OnChanges, OnDestroy { | |||||||
|         if (block?.extras) { |         if (block?.extras) { | ||||||
|           block.extras.minFee = this.getMinBlockFee(block); |           block.extras.minFee = this.getMinBlockFee(block); | ||||||
|           block.extras.maxFee = this.getMaxBlockFee(block); |           block.extras.maxFee = this.getMaxBlockFee(block); | ||||||
|  |           if (block.extras.pool?.minerNames) { | ||||||
|  |             block.extras.pool.minerNames = block.extras.pool.minerNames.map((name) => { | ||||||
|  |               name = name.replace(/[^a-zA-Z0-9 ]/g, ''); | ||||||
|  |               if (name.length > 16) { | ||||||
|  |                 return name.slice(0, 16) + '…'; | ||||||
|  |               } | ||||||
|  |               return name; | ||||||
|  |             }); | ||||||
|  |           } | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       this.blocks.push(block || { |       this.blocks.push(block || { | ||||||
| @ -323,6 +332,14 @@ export class BlockchainBlocksComponent implements OnInit, OnChanges, OnDestroy { | |||||||
|       if (block?.extras) { |       if (block?.extras) { | ||||||
|         block.extras.minFee = this.getMinBlockFee(block); |         block.extras.minFee = this.getMinBlockFee(block); | ||||||
|         block.extras.maxFee = this.getMaxBlockFee(block); |         block.extras.maxFee = this.getMaxBlockFee(block); | ||||||
|  |         if (block.extras.pool?.minerNames) { | ||||||
|  |           block.extras.pool.minerNames = block.extras.pool.minerNames.map((name) => { | ||||||
|  |             if (name.length > 16) { | ||||||
|  |               return name.slice(0, 16) + '…'; | ||||||
|  |             } | ||||||
|  |             return name; | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|       } |       } | ||||||
|       this.blocks[blockIndex] = block; |       this.blocks[blockIndex] = block; | ||||||
|       this.blockStyles[blockIndex] = this.getStyleForBlock(block, blockIndex); |       this.blockStyles[blockIndex] = this.getStyleForBlock(block, blockIndex); | ||||||
|  | |||||||
| @ -684,8 +684,17 @@ | |||||||
|         @if (pool) { |         @if (pool) { | ||||||
|           <td class="wrap-cell"> |           <td class="wrap-cell"> | ||||||
|             <a placement="bottom" [routerLink]="['/mining/pool' | relativeUrl, pool.slug]" class="badge" style="color: #FFF;padding:0;"> |             <a placement="bottom" [routerLink]="['/mining/pool' | relativeUrl, pool.slug]" class="badge" style="color: #FFF;padding:0;"> | ||||||
|  |               <div class="on-pool-container" *ngIf="pool.minerNames != undefined && pool.minerNames.length > 1 && pool.minerNames[1] != ''; else centralisedPool"> | ||||||
|  |                 {{ pool.minerNames[1] }} | ||||||
|  |                 <div class="on-pool"> | ||||||
|                   <img class="pool-logo" [src]="'/resources/mining-pools/' + pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + pool.name + ' mining pool'"> |                   <img class="pool-logo" [src]="'/resources/mining-pools/' + pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + pool.name + ' mining pool'"> | ||||||
|                   {{ pool.name }} |                   {{ pool.name }} | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |               <ng-template #centralisedPool> | ||||||
|  |                 <img class="pool-logo" [src]="'/resources/mining-pools/' + pool.slug + '.svg'" onError="this.src = '/resources/mining-pools/default.svg'" [alt]="'Logo of ' + pool.name + ' mining pool'"> | ||||||
|  |                 {{ pool.name }} | ||||||
|  |               </ng-template> | ||||||
|             </a> |             </a> | ||||||
|           </td> |           </td> | ||||||
|         } @else { |         } @else { | ||||||
|  | |||||||
| @ -60,6 +60,27 @@ | |||||||
| 	top: -1px; | 	top: -1px; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .on-pool-container { | ||||||
|  |   display: inline; | ||||||
|  |   flex-direction: row; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .on-pool { | ||||||
|  |   background-color: var(--bg); | ||||||
|  |   display: inline-block; | ||||||
|  |   margin-top: 4px; | ||||||
|  |   padding: .25em .4em; | ||||||
|  |   border-radius: .25rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .pool-logo { | ||||||
|  |   width: 25px; | ||||||
|  |   height: 25px; | ||||||
|  |   position: relative; | ||||||
|  |   top: -1px; | ||||||
|  |   margin-right: 2px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .badge.badge-accelerated { | .badge.badge-accelerated { | ||||||
|   background-color: var(--tertiary); |   background-color: var(--tertiary); | ||||||
|   color: white; |   color: white; | ||||||
|  | |||||||
| @ -42,6 +42,7 @@ interface Pool { | |||||||
|   id: number; |   id: number; | ||||||
|   name: string; |   name: string; | ||||||
|   slug: string; |   slug: string; | ||||||
|  |   minerNames: string[] | null; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export interface TxAuditStatus { | export interface TxAuditStatus { | ||||||
|  | |||||||
| @ -203,6 +203,7 @@ export interface BlockExtension { | |||||||
|     id: number; |     id: number; | ||||||
|     name: string; |     name: string; | ||||||
|     slug: string; |     slug: string; | ||||||
|  |     minerNames: string[] | null; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user