re-work loadsa stuff
All checks were successful
continuous-integration/drone/push Build is passing

This commit is contained in:
Simon Marsh 2020-10-18 13:55:13 +01:00
parent 3e0802c16b
commit 8cde2ae854
No known key found for this signature in database
GPG Key ID: 30B29A716A54DBB3
4 changed files with 384 additions and 150 deletions

View File

@ -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
}) })

View File

@ -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>&nbsp;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
View File

@ -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
} }
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////

View File

@ -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)
}
} }
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////