2021-11-13 09:03:49 +00:00
|
|
|
<template>
|
|
|
|
<div class="px-0">
|
|
|
|
<b-card>
|
2021-11-13 12:33:19 +00:00
|
|
|
<b-card-title class="d-flex justify-content-between">
|
|
|
|
<span class="text-uppercase"> {{ chain }} </span><small class="text-right"> Height: {{ height }} </small>
|
2021-11-13 09:03:49 +00:00
|
|
|
</b-card-title>
|
|
|
|
<b-alert
|
|
|
|
variant="danger"
|
|
|
|
:show="syncing"
|
|
|
|
>
|
|
|
|
<div class="alert-body">
|
2021-12-22 15:21:58 +00:00
|
|
|
<span>No new blocks have been produced since <strong>{{ latestTime }}</strong> </span>
|
2021-11-13 09:03:49 +00:00
|
|
|
</div>
|
|
|
|
</b-alert>
|
|
|
|
<b-row>
|
2021-12-05 04:20:29 +00:00
|
|
|
<span
|
|
|
|
v-if="uptime.length===0"
|
|
|
|
class="text-danger"
|
|
|
|
> Your node is out of active validator set</span>
|
2021-11-13 09:03:49 +00:00
|
|
|
<b-col
|
|
|
|
v-for="(x,index) in uptime"
|
|
|
|
:key="index"
|
|
|
|
sm="12"
|
|
|
|
class="text-truncate"
|
|
|
|
>
|
2022-03-18 06:58:56 +00:00
|
|
|
<div class="d-flex justify-content-between">
|
|
|
|
<b-form-checkbox
|
|
|
|
v-model="pinned"
|
|
|
|
:value="`${chain}#${x.address}`"
|
|
|
|
class="custom-control-warning"
|
|
|
|
@change="pinValidator(`${chain}#${x.address}`)"
|
|
|
|
>
|
|
|
|
<span class="d-inline-block text-truncate font-weight-bold align-bottom"> {{ x.validator.moniker }} </span>
|
|
|
|
</b-form-checkbox>
|
2022-03-19 03:39:17 +00:00
|
|
|
<span
|
|
|
|
v-if="missing[x.address]"
|
2022-03-18 06:58:56 +00:00
|
|
|
>
|
2022-03-19 03:39:17 +00:00
|
|
|
<b-badge
|
|
|
|
v-if="missing[x.address].missed_blocks_counter > 0"
|
|
|
|
v-b-tooltip.hover.v-danger
|
|
|
|
variant="light-danger"
|
|
|
|
:title="`${missing[x.address].missed_blocks_counter} missed blocks`"
|
|
|
|
class="text-danger text-bolder"
|
|
|
|
>
|
|
|
|
{{ missing[x.address].missed_blocks_counter }}
|
|
|
|
</b-badge>
|
|
|
|
<b-badge
|
|
|
|
v-else
|
|
|
|
v-b-tooltip.hover.v-success
|
|
|
|
variant="light-success"
|
|
|
|
title="Perfect! No missed blocks"
|
|
|
|
>
|
|
|
|
0
|
|
|
|
</b-badge>
|
|
|
|
</span>
|
2022-03-18 06:58:56 +00:00
|
|
|
</div>
|
2021-11-13 09:03:49 +00:00
|
|
|
<div class="d-flex justify-content-between align-self-stretch flex-wrap">
|
|
|
|
<div
|
|
|
|
v-for="(b,i) in blocks"
|
|
|
|
:key="i"
|
|
|
|
style="width:1.5%;"
|
|
|
|
><router-link :to="`./blocks/${b.height}`">
|
|
|
|
<div
|
|
|
|
v-b-tooltip.hover.v-second
|
|
|
|
:title="b.height"
|
|
|
|
:class="b.sigs && b.sigs[x.address] ? b.sigs[x.address] : 'bg-light-success'"
|
|
|
|
class="m-auto"
|
|
|
|
> </div>
|
|
|
|
</router-link>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</b-col>
|
|
|
|
</b-row>
|
|
|
|
</b-card>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
|
|
|
import {
|
2022-03-18 07:35:28 +00:00
|
|
|
BRow, BCol, VBTooltip, BCard, BAlert, BCardTitle, BFormCheckbox, BBadge,
|
2021-11-13 09:03:49 +00:00
|
|
|
} from 'bootstrap-vue'
|
|
|
|
|
|
|
|
import {
|
|
|
|
getLocalChains, timeIn, toDay,
|
2021-12-22 06:21:23 +00:00
|
|
|
} from '@/libs/utils'
|
2022-03-18 06:58:56 +00:00
|
|
|
import { Bech32, toHex } from '@cosmjs/encoding'
|
2021-11-13 09:03:49 +00:00
|
|
|
|
|
|
|
export default {
|
|
|
|
name: 'Blocks',
|
|
|
|
components: {
|
2022-03-18 07:35:28 +00:00
|
|
|
BBadge,
|
2021-11-13 09:03:49 +00:00
|
|
|
BRow,
|
|
|
|
BCol,
|
|
|
|
BCard,
|
|
|
|
BAlert,
|
|
|
|
BCardTitle,
|
|
|
|
BFormCheckbox,
|
|
|
|
},
|
|
|
|
directives: {
|
|
|
|
'b-tooltip': VBTooltip,
|
|
|
|
},
|
|
|
|
props: {
|
|
|
|
chain: {
|
|
|
|
type: String,
|
|
|
|
default: null,
|
|
|
|
},
|
|
|
|
validators: {
|
|
|
|
type: Array,
|
|
|
|
default: () => [],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
data() {
|
|
|
|
const chains = getLocalChains()
|
|
|
|
const pinned = localStorage.getItem('pinned') ? localStorage.getItem('pinned').split(',') : ''
|
|
|
|
return {
|
|
|
|
pinned,
|
|
|
|
config: chains[this.chain],
|
|
|
|
missing: {},
|
|
|
|
blocks: Array.from('0'.repeat(50)).map(x => ({ sigs: {}, height: Number(x) })),
|
|
|
|
syncing: false,
|
|
|
|
latestTime: '',
|
2021-11-13 12:33:19 +00:00
|
|
|
height: '-',
|
2021-11-13 09:03:49 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
computed: {
|
|
|
|
uptime() {
|
|
|
|
const vals = this.validators
|
|
|
|
return vals
|
|
|
|
},
|
|
|
|
},
|
|
|
|
created() {
|
|
|
|
this.initBlocks()
|
2022-03-18 06:58:56 +00:00
|
|
|
this.$http.getSlashingSigningInfo(this.config).then(res => {
|
|
|
|
if (res.info) {
|
|
|
|
res.info.forEach(x => {
|
|
|
|
if (x.address) {
|
|
|
|
const hex = toHex(Bech32.decode(x.address).data).toUpperCase()
|
|
|
|
this.missing[hex] = x
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
2021-11-13 09:03:49 +00:00
|
|
|
},
|
|
|
|
beforeDestroy() {
|
|
|
|
this.blocks = [] // clear running tasks if it is not finish
|
|
|
|
this.syncing = false
|
|
|
|
clearInterval(this.timer)
|
|
|
|
},
|
|
|
|
methods: {
|
|
|
|
pinValidator() {
|
|
|
|
localStorage.setItem('pinned', this.pinned)
|
|
|
|
},
|
|
|
|
initBlocks() {
|
|
|
|
this.$http.getLatestBlock(this.config).then(d => {
|
|
|
|
const { height } = d.block.last_commit
|
2021-11-13 12:33:19 +00:00
|
|
|
this.height = height
|
2021-11-13 09:03:49 +00:00
|
|
|
if (timeIn(d.block.header.time, 3, 'm')) {
|
|
|
|
this.syncing = true
|
|
|
|
} else {
|
|
|
|
this.syncing = false
|
|
|
|
}
|
|
|
|
this.latestTime = toDay(d.block.header.time, 'long')
|
|
|
|
const blocks = []
|
|
|
|
// update height
|
|
|
|
let promise = Promise.resolve()
|
|
|
|
for (let i = height - 1; i > height - 50; i -= 1) {
|
|
|
|
blocks.unshift({ sigs: {}, height: i > 0 ? i : 0 })
|
|
|
|
if (i > height - 48 && i > 0) {
|
|
|
|
promise = promise.then(() => new Promise(resolve => {
|
|
|
|
this.fetch_status(i, resolve)
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const sigs = this.initColor()
|
|
|
|
d.block.last_commit.signatures.forEach(x => {
|
|
|
|
if (x.validator_address) sigs[x.validator_address] = 'bg-success'
|
|
|
|
})
|
|
|
|
blocks.push({ sigs, height })
|
|
|
|
this.blocks = blocks
|
|
|
|
|
|
|
|
this.timer = setInterval(this.fetch_latest, 6000)
|
|
|
|
})
|
|
|
|
},
|
|
|
|
initColor() {
|
|
|
|
const sigs = {}
|
|
|
|
this.validators.forEach(x => {
|
|
|
|
sigs[x.address] = 'bg-danger'
|
|
|
|
})
|
|
|
|
return sigs
|
|
|
|
},
|
|
|
|
fetch_status(height, resolve) {
|
|
|
|
const block = this.blocks.find(b => b.height === height)
|
|
|
|
if (block) {
|
|
|
|
this.$http.getBlockByHeight(height, this.config).then(res => {
|
|
|
|
resolve()
|
|
|
|
const sigs = this.initColor()
|
|
|
|
res.block.last_commit.signatures.forEach(x => {
|
|
|
|
if (x.validator_address) sigs[x.validator_address] = 'bg-success'
|
|
|
|
})
|
|
|
|
this.$set(block, 'sigs', sigs)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
},
|
|
|
|
fetch_latest() {
|
|
|
|
this.$http.getLatestBlock(this.config).then(res => {
|
|
|
|
const sigs = this.initColor()
|
|
|
|
res.block.last_commit.signatures.forEach(x => {
|
|
|
|
if (x.validator_address) sigs[x.validator_address] = 'bg-success'
|
|
|
|
})
|
2021-11-13 12:33:19 +00:00
|
|
|
this.height = res.block.last_commit.height
|
2021-11-13 09:03:49 +00:00
|
|
|
const block = this.blocks.find(b => b[1] === res.block.last_commit.height)
|
|
|
|
if (typeof block === 'undefined') { // mei
|
|
|
|
// this.$set(block, 0, typeof sigs !== 'undefined')
|
|
|
|
if (this.blocks.length >= 50) this.blocks.shift()
|
|
|
|
this.blocks.push({ sigs, height: res.block.last_commit.height })
|
|
|
|
}
|
|
|
|
})
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style></style>
|