This commit is contained in:
parent
3e0802c16b
commit
8cde2ae854
@ -7,6 +7,7 @@ const ExplorerURL='https://explorer.burble.com/#'
|
|||||||
//////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////
|
||||||
// smaller display components
|
// smaller display components
|
||||||
|
|
||||||
|
// display an ASN
|
||||||
Vue.component('reg-asn', {
|
Vue.component('reg-asn', {
|
||||||
template: '#reg-asn',
|
template: '#reg-asn',
|
||||||
props: [ 'asn' ],
|
props: [ 'asn' ],
|
||||||
@ -27,6 +28,7 @@ Vue.component('reg-asn', {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// display a path of ASNs
|
||||||
Vue.component('reg-path', {
|
Vue.component('reg-path', {
|
||||||
template: '#reg-path',
|
template: '#reg-path',
|
||||||
props: [ 'path' ],
|
props: [ 'path' ],
|
||||||
@ -40,6 +42,7 @@ Vue.component('reg-path', {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// display a inet{6,}num
|
||||||
Vue.component('reg-prefix', {
|
Vue.component('reg-prefix', {
|
||||||
template: '#reg-prefix',
|
template: '#reg-prefix',
|
||||||
props: [ 'prefix' ],
|
props: [ 'prefix' ],
|
||||||
@ -63,7 +66,88 @@ Vue.component('reg-prefix', {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// display mntner(s)
|
||||||
|
Vue.component('reg-mnts', {
|
||||||
|
template: '#reg-mnts',
|
||||||
|
props: [ 'mnts' ],
|
||||||
|
data() {
|
||||||
|
return { }
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// display mntner
|
||||||
|
Vue.component('reg-mnt', {
|
||||||
|
template: '#reg-mnt',
|
||||||
|
props: [ 'mnt' ],
|
||||||
|
data() {
|
||||||
|
return { }
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
url: function() {
|
||||||
|
return ExplorerURL + "/mntner/" + this.mnt
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
// display a set of AS Paths
|
||||||
|
Vue.component('rt-paths', {
|
||||||
|
template: '#rt-paths',
|
||||||
|
props: [ 'paths' ],
|
||||||
|
data() {
|
||||||
|
return { }
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
showPaths: function() {
|
||||||
|
const e = this.$createElement
|
||||||
|
|
||||||
|
// create a new array of reversed paths
|
||||||
|
var paths = [ ]
|
||||||
|
this.paths.forEach(path => {
|
||||||
|
paths.push(path.split(" ").reverse())
|
||||||
|
})
|
||||||
|
// and sort for easier comparing
|
||||||
|
paths.sort(
|
||||||
|
(a,b) => a.join(" ").localeCompare(b.join(" "))
|
||||||
|
)
|
||||||
|
|
||||||
|
// now build the modal dialog
|
||||||
|
|
||||||
|
// for each path in the list
|
||||||
|
var pathList = [ e('p', { }, 'Paths are reverse sorted') ]
|
||||||
|
paths.forEach(path => {
|
||||||
|
|
||||||
|
// for each ASN in the path
|
||||||
|
var asnList = [ ]
|
||||||
|
path.forEach(asn => {
|
||||||
|
asnList.push(e('b-link', {
|
||||||
|
class: 'px-1 my-0 py-0',
|
||||||
|
props: {
|
||||||
|
href: ExplorerURL + "/aut-num/" + asn
|
||||||
|
}
|
||||||
|
}, asn))
|
||||||
|
})
|
||||||
|
|
||||||
|
pathList.push(e('li', {
|
||||||
|
class: 'my-0 py-0'
|
||||||
|
}, asnList))
|
||||||
|
})
|
||||||
|
|
||||||
|
var vnode = e('ul', { }, pathList)
|
||||||
|
|
||||||
|
this.$bvModal.msgBoxOk([vnode], {
|
||||||
|
title: 'AS Paths',
|
||||||
|
size: 'xl',
|
||||||
|
centered: true
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
count: function() {
|
||||||
|
return this.paths.length
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////
|
||||||
// flap list component
|
// flap list component
|
||||||
@ -73,52 +157,47 @@ Vue.component('app-flaps', {
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
filter: '',
|
filter: '',
|
||||||
currentPage: 1,
|
|
||||||
total: 0,
|
|
||||||
fields: [
|
|
||||||
{
|
|
||||||
key: 'prefix',
|
|
||||||
label: 'Prefix',
|
|
||||||
sortable: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
key: 'path',
|
|
||||||
label: 'Path',
|
|
||||||
sortable: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
key: 'count',
|
|
||||||
label: 'Updates',
|
|
||||||
sortable: true,
|
|
||||||
sortDirection: 'desc'
|
|
||||||
}
|
|
||||||
],
|
|
||||||
flaps: [ ]
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
rows: function() {
|
|
||||||
return this.flaps.length;
|
total: function() {
|
||||||
}
|
return this.$root.flapsTotal
|
||||||
},
|
},
|
||||||
methods: {
|
|
||||||
update: function(data) {
|
// converts the map to an array for iterating,
|
||||||
this.flaps.splice(0);
|
// applying any filtering that is required
|
||||||
|
list: function() {
|
||||||
|
|
||||||
this.total = data.total;
|
var list = [ ]
|
||||||
this.flaps = data.list.map((update) => {
|
var flaps = this.$root.flaps
|
||||||
var tmp = update.path.split(" ");
|
var total = this.total
|
||||||
update.path = tmp.map((asn) => {
|
|
||||||
return "AS" + asn;
|
Object.keys(flaps).forEach((prefix) => {
|
||||||
}).join(" ")
|
|
||||||
return update;
|
if (this.filter == '' || prefix.includes(this.filter)) {
|
||||||
|
list.push({
|
||||||
|
prefix: prefix,
|
||||||
|
count: flaps[prefix].count,
|
||||||
|
percent: Math.round(flaps[prefix].count*100 / total),
|
||||||
|
paths: flaps[prefix].paths,
|
||||||
|
mntner: flaps[prefix].mntner
|
||||||
|
})
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
list.sort(
|
||||||
|
(a,b) => b.count - a.count
|
||||||
|
)
|
||||||
|
|
||||||
|
return list.slice(0, 10)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
this.$root.$on('flaps-update', data => {
|
this.$root.$on('flaps-update', data => {
|
||||||
this.update(data)
|
this.update(data)
|
||||||
})
|
})
|
||||||
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
@ -131,37 +210,53 @@ Vue.component('app-roa', {
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
filter4: '',
|
filter4: '',
|
||||||
filter6: '',
|
filter6: ''
|
||||||
roaFields: [
|
|
||||||
{
|
|
||||||
key: 'prefix',
|
|
||||||
label: 'Prefix',
|
|
||||||
sortable: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
key: 'origin',
|
|
||||||
label: 'Origin',
|
|
||||||
sortable: true
|
|
||||||
}
|
|
||||||
],
|
|
||||||
roa4: [ ],
|
|
||||||
roa6: [ ]
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
computed: {
|
||||||
update: function(data) {
|
list4: function() {
|
||||||
this.roa4.splice(0);
|
var list = [ ]
|
||||||
this.roa6.splice(0);
|
|
||||||
|
this.$root.roas.forEach((roa) => {
|
||||||
data.list.forEach((roa) => {
|
if (!roa.prefix.includes(':') &&
|
||||||
var nroa = {
|
(this.filter4 == '' ||
|
||||||
origin: "AS" + roa.origin,
|
roa.prefix.includes(this.filter4))) {
|
||||||
prefix: roa.prefix
|
list.push(roa)
|
||||||
};
|
}
|
||||||
if (roa.prefix.includes(":")) { this.roa6.push(nroa) }
|
|
||||||
else { this.roa4.push(nroa) }
|
|
||||||
})
|
})
|
||||||
}
|
|
||||||
|
// sort by origin
|
||||||
|
list.sort((a,b) => {
|
||||||
|
var asna = a.origin.substr(2)
|
||||||
|
var asnb = b.origin.substr(2)
|
||||||
|
return asna - asnb
|
||||||
|
})
|
||||||
|
|
||||||
|
return list
|
||||||
|
},
|
||||||
|
|
||||||
|
list6: function() {
|
||||||
|
var list = [ ]
|
||||||
|
|
||||||
|
this.$root.roas.forEach((roa) => {
|
||||||
|
if (roa.prefix.includes(':') &&
|
||||||
|
(this.filter6 == '' ||
|
||||||
|
roa.prefix.includes(this.filter6))) {
|
||||||
|
list.push(roa)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// sort by origin
|
||||||
|
list.sort((a,b) => {
|
||||||
|
var asna = a.origin.substr(2)
|
||||||
|
var asnb = b.origin.substr(2)
|
||||||
|
return asna - asnb
|
||||||
|
})
|
||||||
|
|
||||||
|
return list
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
this.$root.$on('roa-update', data => {
|
this.$root.$on('roa-update', data => {
|
||||||
@ -178,7 +273,8 @@ Vue.component('app-timer', {
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
seconds: 0,
|
seconds: 0,
|
||||||
timer: 0
|
timer: 0,
|
||||||
|
roaCounter: 0
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
@ -186,15 +282,65 @@ Vue.component('app-timer', {
|
|||||||
if (this.seconds <= 0) {
|
if (this.seconds <= 0) {
|
||||||
this.seconds = 61;
|
this.seconds = 61;
|
||||||
|
|
||||||
|
// refresh flaps data
|
||||||
axios.get('/api/flaps').then(response => {
|
axios.get('/api/flaps').then(response => {
|
||||||
this.$root.$emit('flaps-update', response.data)
|
|
||||||
|
// build a new map of the prefixes and total updates
|
||||||
|
var flaps = { }
|
||||||
|
var total = 0
|
||||||
|
|
||||||
|
response.data.list.forEach((update) => {
|
||||||
|
|
||||||
|
// initialise if no existing entry for this prefix
|
||||||
|
if (!(update.prefix in flaps)) {
|
||||||
|
flaps[update.prefix] = {
|
||||||
|
count: 0,
|
||||||
|
paths: [ ],
|
||||||
|
mntner: this.$root.mntmap[update.prefix] || [ ]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// canonicalise the ASN path
|
||||||
|
var tmp = update.path.split(" ")
|
||||||
|
var path = tmp.map((asn) => {
|
||||||
|
return "AS" + asn
|
||||||
|
}).join(" ")
|
||||||
|
|
||||||
|
// finally update the prefix entry
|
||||||
|
flaps[update.prefix].count += update.count
|
||||||
|
flaps[update.prefix].paths.push(path)
|
||||||
|
|
||||||
|
total += update.count
|
||||||
|
})
|
||||||
|
|
||||||
|
this.$root.flaps = flaps
|
||||||
|
this.$root.flapsTotal = total
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
axios.get('/api/roa').then(response => {
|
if (this.roaCounter == 0) {
|
||||||
this.$root.$emit('roa-update', response.data)
|
this.roaCounter = 59
|
||||||
});
|
|
||||||
|
// refresh roa data
|
||||||
|
axios.get('/api/roa').then(response => {
|
||||||
|
var roas = [ ]
|
||||||
|
response.data.list.forEach((roa) => {
|
||||||
|
var asn = "AS" + roa.origin
|
||||||
|
roas.push({
|
||||||
|
origin: asn,
|
||||||
|
prefix: roa.prefix,
|
||||||
|
mntner: this.$root.mntmap[asn] || [ ]
|
||||||
|
})
|
||||||
|
})
|
||||||
|
this.$root.roas = roas
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
this.roaCounter -= 1
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
// countdown every second
|
||||||
this.seconds -= 1;
|
this.seconds -= 1;
|
||||||
this.timer = setTimeout(() => this.trigger(), 1000);
|
this.timer = setTimeout(() => this.trigger(), 1000);
|
||||||
}
|
}
|
||||||
@ -220,7 +366,46 @@ const router = new VueRouter({
|
|||||||
// and the main app instance
|
// and the main app instance
|
||||||
const vm = new Vue({
|
const vm = new Vue({
|
||||||
el: '#grc_realtime',
|
el: '#grc_realtime',
|
||||||
data: { },
|
data: {
|
||||||
|
flaps: { },
|
||||||
|
flapsTotal: 0,
|
||||||
|
roas: [ ],
|
||||||
|
mntmap: { }
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
axios.get('https://explorer.burble.com/api/registry/*inet/*/mnt-by?raw').then(response => {
|
||||||
|
|
||||||
|
// put the response in the mntmap hash
|
||||||
|
Object.keys(response.data).forEach(index => {
|
||||||
|
var slash = index.indexOf('/')
|
||||||
|
var prefix = index.substr(slash + 1)
|
||||||
|
var prefix = prefix.replace('_', '/')
|
||||||
|
this.mntmap[prefix] = response.data[index]['mnt-by']
|
||||||
|
})
|
||||||
|
|
||||||
|
// update the flaps list if required
|
||||||
|
Object.keys(this.flaps).forEach(prefix => {
|
||||||
|
this.flaps[prefix].mntner = this.mntmap[prefix]
|
||||||
|
})
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
axios.get('https://explorer.burble.com/api/registry/aut-num/*/mnt-by?raw').then(response => {
|
||||||
|
|
||||||
|
// put the response in the mntmap hash
|
||||||
|
Object.keys(response.data).forEach(index => {
|
||||||
|
var slash = index.indexOf('/')
|
||||||
|
var autnum = index.substr(slash + 1)
|
||||||
|
this.mntmap[autnum] = response.data[index]['mnt-by']
|
||||||
|
})
|
||||||
|
|
||||||
|
// and roas
|
||||||
|
this.roas.forEach(roa => {
|
||||||
|
roa.mntner = this.mntmap[roa.origin]
|
||||||
|
})
|
||||||
|
|
||||||
|
})
|
||||||
|
},
|
||||||
router
|
router
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -76,47 +76,43 @@
|
|||||||
<script type="text/x-template" id="app-flaps-template">
|
<script type="text/x-template" id="app-flaps-template">
|
||||||
<b-container fluid="sm">
|
<b-container fluid="sm">
|
||||||
<b-row class="justify-content-center py-3">
|
<b-row class="justify-content-center py-3">
|
||||||
<h2>Most Updated Routes</h2>
|
<h2>Most updated prefixes</h2>
|
||||||
</b-row>
|
</b-row>
|
||||||
<b-row class="px-3">
|
<b-row class="px-3">
|
||||||
<b-col>
|
<b-col>
|
||||||
<b-table
|
<b-table-simple responsive>
|
||||||
id="flaps-table"
|
<b-thead>
|
||||||
:fields="fields"
|
<b-tr>
|
||||||
:items="flaps"
|
<b-th>Updates/min</b-th>
|
||||||
:filter="filter"
|
<b-th>Prefix</b-th>
|
||||||
:sort-by="'count'"
|
<b-th>MNTNER</b-th>
|
||||||
:sort-desc="true"
|
<b-th>Paths</b-th>
|
||||||
:per-page="20"
|
</b-tr>
|
||||||
>
|
</b-thead>
|
||||||
<template v-slot:cell(prefix)="data">
|
<b-tbody>
|
||||||
<reg-prefix v-bind:prefix="data.value"></reg-prefix>
|
<b-tr v-for="item in this.list">
|
||||||
</template>
|
<b-td class="py-1 my-0" v-text="item.count + ' - ' + item.percent + '%'"></b-td>
|
||||||
<template v-slot:cell(path)="data">
|
<b-td class="py-1 my-0"><reg-prefix v-bind:prefix="item.prefix"></reg-prefix></b-td>
|
||||||
<reg-path v-bind:path="data.value"></reg-path>
|
<b-td class="py-1 my-0"><reg-mnts v-bind:mnts="item.mntner"></reg-mnts></b-td>
|
||||||
</template>
|
<b-td class="py-1 my-0"><rt-paths v-bind:paths="item.paths"></rt-paths></b-td>
|
||||||
</b-table>
|
</b-tr>
|
||||||
|
</b-tbody>
|
||||||
|
</b-table-simple>
|
||||||
</b-col>
|
</b-col>
|
||||||
|
|
||||||
<b-col class="text-left pr-5">
|
<b-col class="text-left pr-5">
|
||||||
<p>Total Updates: {{ this.total }}</p>
|
<p>Total Updates: {{ this.total }}<br/>Showing top ten results</p>
|
||||||
<p><b-pagination class="pb-4"
|
<p class="mt-5">Type to filter prefixes:<br/><b-input-group size="sm">
|
||||||
v-model="currentPage"
|
<b-form-input
|
||||||
:total-rows="rows"
|
v-model="filter"
|
||||||
:per-page="20"
|
type="search"
|
||||||
aria-controls="flaps-table"
|
id="filterInput"
|
||||||
></b-pagination>
|
placeholder="filter"
|
||||||
</p>
|
debounce="200"
|
||||||
<p><b-input-group size="sm">
|
></b-form-input>
|
||||||
<b-form-input
|
</b-input-group></p>
|
||||||
v-model="filter"
|
</b-col>
|
||||||
type="search"
|
</b-row>
|
||||||
id="filterInput"
|
|
||||||
placeholder="Type to filter results"
|
|
||||||
debounce="200"
|
|
||||||
></b-form-input>
|
|
||||||
</b-input-group></p>
|
|
||||||
</b-col>
|
|
||||||
</b-row>
|
|
||||||
</b-container>
|
</b-container>
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@ -131,26 +127,28 @@
|
|||||||
</b-row>
|
</b-row>
|
||||||
<b-row class="px-3">
|
<b-row class="px-3">
|
||||||
<b-col>
|
<b-col>
|
||||||
<b-table
|
<b-table-simple responsive>
|
||||||
:fields="roaFields"
|
<b-thead>
|
||||||
:items="roa4"
|
<b-th>Prefix</b-th>
|
||||||
:filter="filter4"
|
<b-th>Origin</b-th>
|
||||||
>
|
<b-th>MNTNER</b-th>
|
||||||
<template v-slot:cell(prefix)="data">
|
</b-thead>
|
||||||
<reg-prefix v-bind:prefix="data.value"></reg-prefix>
|
<b-tbody>
|
||||||
</template>
|
<b-tr v-for="roa in this.list4">
|
||||||
<template v-slot:cell(origin)="data">
|
<b-td class="py-1 my-0"><reg-prefix v-bind:prefix="roa.prefix"></reg-prefix></b-td>
|
||||||
<reg-asn v-bind:asn="data.value"></reg-asn>
|
<b-td class="py-1 my-0"><reg-asn v-bind:asn="roa.origin"></reg-asn></b-td>
|
||||||
</template>
|
<b-td class="py-1 my-0"><reg-mnts v-bind:mnts="roa.mntner"></reg-mnts></b-td>
|
||||||
</b-table>
|
</b-tr>
|
||||||
|
</b-tbody>
|
||||||
|
</b-table-simple>
|
||||||
</b-col>
|
</b-col>
|
||||||
<b-col class="text-left pr-5">
|
<b-col class="text-left pr-5">
|
||||||
<b-input-group size="sm">
|
<p>Type to filter prefixes</p><b-input-group size="sm">
|
||||||
<b-form-input
|
<b-form-input
|
||||||
v-model="filter4"
|
v-model="filter4"
|
||||||
type="search"
|
type="search"
|
||||||
id="filterInput4"
|
id="filterInput4"
|
||||||
placeholder="Type to filter results"
|
placeholder="filter"
|
||||||
debounce="200"
|
debounce="200"
|
||||||
></b-form-input>
|
></b-form-input>
|
||||||
</b-input-group>
|
</b-input-group>
|
||||||
@ -164,26 +162,28 @@
|
|||||||
</b-row>
|
</b-row>
|
||||||
<b-row class="px-3">
|
<b-row class="px-3">
|
||||||
<b-col>
|
<b-col>
|
||||||
<b-table
|
<b-table-simple responsive>
|
||||||
:fields="roaFields"
|
<b-thead>
|
||||||
:items="roa6"
|
<b-th>Prefix</b-th>
|
||||||
:filter="filter6"
|
<b-th>Origin</b-th>
|
||||||
>
|
<b-th>MNTNER</b-th>
|
||||||
<template v-slot:cell(prefix)="data">
|
</b-thead>
|
||||||
<reg-prefix v-bind:prefix="data.value"></reg-prefix>
|
<b-tbody>
|
||||||
</template>
|
<b-tr v-for="roa in this.list6">
|
||||||
<template v-slot:cell(origin)="data">
|
<b-td class="py-1 my-0"><reg-prefix v-bind:prefix="roa.prefix"></reg-prefix></b-td>
|
||||||
<reg-asn v-bind:asn="data.value"></reg-asn>
|
<b-td class="py-1 my-0"><reg-asn v-bind:asn="roa.origin"></reg-asn></b-td>
|
||||||
</template>
|
<b-td class="py-1 my-0"><reg-mnts v-bind:mnts="roa.mntner"></reg-mnts></b-td>
|
||||||
</b-table>
|
</b-tr>
|
||||||
</b-col>
|
</b-tbody>
|
||||||
|
</b-table-simple>
|
||||||
|
</b-col>
|
||||||
<b-col class="text-left pr-5">
|
<b-col class="text-left pr-5">
|
||||||
<b-input-group size="sm">
|
<p>Type to filter prefixes</p><b-input-group size="sm">
|
||||||
<b-form-input
|
<b-form-input
|
||||||
v-model="filter6"
|
v-model="filter6"
|
||||||
type="search"
|
type="search"
|
||||||
id="filterInput6"
|
id="filterInput6"
|
||||||
placeholder="Type to filter results"
|
placeholder="filter"
|
||||||
debounce="200"
|
debounce="200"
|
||||||
></b-form-input>
|
></b-form-input>
|
||||||
</b-input-group>
|
</b-input-group>
|
||||||
@ -203,6 +203,18 @@
|
|||||||
<b-link v-bind:href="url" v-text="prefix"></b-link>
|
<b-link v-bind:href="url" v-text="prefix"></b-link>
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
<script type="text/x-template" id="reg-mnts">
|
||||||
|
<div>
|
||||||
|
<p class="py-0 my-0" v-for="mnt in this.mnts">
|
||||||
|
<reg-mnt v-bind:mnt="mnt"></reg-mnt>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/x-template" id="reg-mnt">
|
||||||
|
<b-link class="text-nowrap" v-bind:href="url" v-text="mnt"></b-link>
|
||||||
|
</script>
|
||||||
|
|
||||||
<script type="text/x-template" id="reg-path">
|
<script type="text/x-template" id="reg-path">
|
||||||
<ul class="text-nowrap">
|
<ul class="text-nowrap">
|
||||||
<li class="d-inline p-1" v-for="asn in list">
|
<li class="d-inline p-1" v-for="asn in list">
|
||||||
@ -211,6 +223,10 @@
|
|||||||
</ul>
|
</ul>
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
<script type="text/x-template" id="rt-paths">
|
||||||
|
<b-button @click="this.showPaths" variant="dark" size="sm"><span v-text="this.count"></span> paths</b-button>
|
||||||
|
</script>
|
||||||
|
|
||||||
<script type="text/x-template" id="app-timer">
|
<script type="text/x-template" id="app-timer">
|
||||||
<p class="p-1 pr-3">Update in {{ this.seconds }}s</p>
|
<p class="p-1 pr-3">Update in {{ this.seconds }}s</p>
|
||||||
</script>
|
</script>
|
||||||
|
47
data.go
47
data.go
@ -70,7 +70,8 @@ type SnapshotData struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type DataStruct struct {
|
type DataStruct struct {
|
||||||
shutdown context.CancelFunc
|
shutdown context.CancelFunc
|
||||||
|
roaCounter uint
|
||||||
|
|
||||||
// sets for currently cellecting and previous data
|
// sets for currently cellecting and previous data
|
||||||
active *ActiveData
|
active *ActiveData
|
||||||
@ -119,25 +120,49 @@ func (data *DataStruct) swap(ctx context.Context) {
|
|||||||
|
|
||||||
case <-ticker.C:
|
case <-ticker.C:
|
||||||
// tiemr expired
|
// tiemr expired
|
||||||
log.Debug("Data Update")
|
log.Debug("Route Data Update")
|
||||||
|
|
||||||
var upp *unsafe.Pointer
|
var upp *unsafe.Pointer
|
||||||
var up unsafe.Pointer
|
var up unsafe.Pointer
|
||||||
var oldp unsafe.Pointer
|
var oldp unsafe.Pointer
|
||||||
|
|
||||||
// swap in a new empty data set
|
// swap in new route data
|
||||||
upp = (*unsafe.Pointer)(unsafe.Pointer(&data.active))
|
upp = (*unsafe.Pointer)(unsafe.Pointer(&data.active.route))
|
||||||
up = unsafe.Pointer(data.newActiveData())
|
up = unsafe.Pointer(data.newRouteData())
|
||||||
oldp = atomic.SwapPointer(upp, up)
|
oldp = atomic.SwapPointer(upp, up)
|
||||||
|
|
||||||
// generate the reporting snapshot
|
// create a snapshot of the old data
|
||||||
active := (*ActiveData)(oldp)
|
routed := (*RouteData)(oldp)
|
||||||
snapshot := active.snapshot()
|
snapshot := routed.snapshot()
|
||||||
|
|
||||||
// then swap in the new snapshot
|
// then swap in the new snapshot
|
||||||
upp = (*unsafe.Pointer)(unsafe.Pointer(&data.snapshot))
|
upp = (*unsafe.Pointer)(unsafe.Pointer(&data.snapshot.route))
|
||||||
up = unsafe.Pointer(snapshot)
|
up = unsafe.Pointer(snapshot)
|
||||||
atomic.SwapPointer(upp, up)
|
atomic.SwapPointer(upp, up)
|
||||||
|
|
||||||
|
if data.roaCounter == 0 {
|
||||||
|
data.roaCounter = 59
|
||||||
|
|
||||||
|
log.Debug("ROA Data Update")
|
||||||
|
|
||||||
|
// swap in new roa data
|
||||||
|
upp = (*unsafe.Pointer)(unsafe.Pointer(&data.active.roa))
|
||||||
|
up = unsafe.Pointer(data.newROAData())
|
||||||
|
oldp = atomic.SwapPointer(upp, up)
|
||||||
|
|
||||||
|
// create a snapshot of the old data
|
||||||
|
road := (*ROAData)(oldp)
|
||||||
|
snapshot := road.snapshot()
|
||||||
|
|
||||||
|
// then swap in the new snapshot
|
||||||
|
upp = (*unsafe.Pointer)(unsafe.Pointer(&data.snapshot.roa))
|
||||||
|
up = unsafe.Pointer(snapshot)
|
||||||
|
atomic.SwapPointer(upp, up)
|
||||||
|
|
||||||
|
} else {
|
||||||
|
data.roaCounter -= 1
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -171,8 +196,8 @@ func (route *RouteData) add(path string) {
|
|||||||
route.paths[path] += 1
|
route.paths[path] += 1
|
||||||
}
|
}
|
||||||
|
|
||||||
func (roa *ROAData) add(path string) {
|
func (roa *ROAData) add(prefix string) {
|
||||||
roa.roas[path] += 1
|
roa.roas[prefix] += 1
|
||||||
}
|
}
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
14
ingest.go
14
ingest.go
@ -98,8 +98,9 @@ func (ingest *Ingest) tail(ctx context.Context, sockPath string) {
|
|||||||
|
|
||||||
// trim the trailing newline
|
// trim the trailing newline
|
||||||
line = strings.TrimSuffix(line, "\n")
|
line = strings.TrimSuffix(line, "\n")
|
||||||
|
ix := strings.Index(line, "<INFO> ")
|
||||||
if len(line) > 3 {
|
if (ix != -1) && (len(line) > (ix + 10)) {
|
||||||
|
line = line[ix+7:]
|
||||||
|
|
||||||
log.WithFields(log.Fields{
|
log.WithFields(log.Fields{
|
||||||
"line": line,
|
"line": line,
|
||||||
@ -137,7 +138,14 @@ func (ingest *Ingest) route(data string) {
|
|||||||
"data": data,
|
"data": data,
|
||||||
}).Debug("route update")
|
}).Debug("route update")
|
||||||
|
|
||||||
ingest.data.active.route.add(data)
|
space := strings.IndexByte(data, ' ')
|
||||||
|
if space != -1 {
|
||||||
|
|
||||||
|
prefix := data[:space]
|
||||||
|
path := data[space+7 : len(data)-1]
|
||||||
|
|
||||||
|
ingest.data.active.route.add(prefix + " " + path)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
Loading…
x
Reference in New Issue
Block a user