mirror of
https://github.com/nttgin/BGPalerter.git
synced 2024-05-19 06:50:08 +00:00
initial commit of ris dump connector
This commit is contained in:
@@ -12,6 +12,9 @@ connectors:
|
|||||||
socketOptions:
|
socketOptions:
|
||||||
includeRaw: false
|
includeRaw: false
|
||||||
|
|
||||||
|
- file: connectorRISDump
|
||||||
|
name: dmp
|
||||||
|
|
||||||
monitors:
|
monitors:
|
||||||
- file: monitorHijack
|
- file: monitorHijack
|
||||||
channel: hijack
|
channel: hijack
|
||||||
|
@@ -132,6 +132,9 @@ Parameters for this connector module:
|
|||||||
|carefulSubscription| If this parameter is set to true (default), the RIS server will stream only the data related to our prefix. This is an advanced parameter useful only for research purposes. |
|
|carefulSubscription| If this parameter is set to true (default), the RIS server will stream only the data related to our prefix. This is an advanced parameter useful only for research purposes. |
|
||||||
|perMessageDeflate| Enable gzip compression on the connection. |
|
|perMessageDeflate| Enable gzip compression on the connection. |
|
||||||
|
|
||||||
|
#### connectorRISDump
|
||||||
|
It connects to the RIPEstat's BGPlay API and retrieves a RIS dump about the monitored resources. The retrieved dump is 2 hours old, due to limitations on the API side.
|
||||||
|
|
||||||
#### connectorTest
|
#### connectorTest
|
||||||
|
|
||||||
Connector used for testing purposes, it provokes all types of alerting. Needed to run the tests (`npm run test`) .
|
Connector used for testing purposes, it provokes all types of alerting. Needed to run the tests (`npm run test`) .
|
||||||
|
@@ -359,7 +359,7 @@ export default class ConnectorRIS extends Connector {
|
|||||||
prefix,
|
prefix,
|
||||||
peer,
|
peer,
|
||||||
timestamp
|
timestamp
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return components;
|
return components;
|
||||||
|
171
src/connectors/connectorRISDump.js
Normal file
171
src/connectors/connectorRISDump.js
Normal file
@@ -0,0 +1,171 @@
|
|||||||
|
/*
|
||||||
|
* BSD 3-Clause License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2019, NTT Ltd.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* Redistributions of source code must retain the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* Neither the name of the copyright holder nor the names of its
|
||||||
|
* contributors may be used to endorse or promote products derived from
|
||||||
|
* this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||||
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import Connector from "./connector";
|
||||||
|
import batchPromises from "batch-promises";
|
||||||
|
import brembo from "brembo";
|
||||||
|
import moment from "moment";
|
||||||
|
import {AS, Path} from "../model";
|
||||||
|
|
||||||
|
export default class ConnectorRISDump extends Connector {
|
||||||
|
|
||||||
|
constructor(name, params, env) {
|
||||||
|
super(name, params, env);
|
||||||
|
this.withdrawNotVisible = this.params.withdrawNotVisible || false;
|
||||||
|
this.storage = env.storage;
|
||||||
|
this.lastRun = null;
|
||||||
|
|
||||||
|
if (this.storage) {
|
||||||
|
this.storage
|
||||||
|
.get(`run-${this.name}`)
|
||||||
|
.then(date => {
|
||||||
|
if (date && !isNaN(date)) {
|
||||||
|
this.lastRun = moment.unix(parseInt(date)).utc();
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
this.logger.log({
|
||||||
|
level: 'error',
|
||||||
|
message: error
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
_shouldDownloadDump = () => {
|
||||||
|
return !this.lastRun || this.lastRun.diff(moment(), 'hours') > 2;
|
||||||
|
};
|
||||||
|
|
||||||
|
connect = () =>
|
||||||
|
new Promise((resolve, reject) => {
|
||||||
|
resolve(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
_loadResource = (resource) => {
|
||||||
|
const stop = moment().subtract(2, "hours").utc();
|
||||||
|
const url = brembo.build("https://stat.ripe.net/data/bgplay/data.json", {
|
||||||
|
params: {
|
||||||
|
resource,
|
||||||
|
rrcs: "0,11,13,14,15,16",
|
||||||
|
"unix_timestamps": "TRUE",
|
||||||
|
starttime: moment(stop).subtract(2, "minutes").unix(),
|
||||||
|
stoptime: stop.unix()
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return this.axios({
|
||||||
|
responseType: "json",
|
||||||
|
url
|
||||||
|
})
|
||||||
|
.then(data => {
|
||||||
|
if (data && data.data && data.data.data && data.data.data.initial_state) {
|
||||||
|
const dump = data.data.data.initial_state;
|
||||||
|
const sent = {};
|
||||||
|
|
||||||
|
for (let entry of dump) {
|
||||||
|
const path = new Path((entry.path|| []).map(i => new AS(i)));
|
||||||
|
sent[entry.target_prefix] = true;
|
||||||
|
this._message({
|
||||||
|
type: "announcement",
|
||||||
|
prefix: entry.target_prefix,
|
||||||
|
peer: entry.source_id.split("-")[1],
|
||||||
|
path,
|
||||||
|
originAS: path.getLast(),
|
||||||
|
nextHop: null,
|
||||||
|
aggregator: null,
|
||||||
|
timestamp: stop.valueOf(),
|
||||||
|
communities: entry.community
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.withdrawNotVisible) { // This feature is not reachable for now
|
||||||
|
for (let entry of dump) {
|
||||||
|
if (!sent[entry.target_prefix]) {
|
||||||
|
this._message({
|
||||||
|
type: "withdrawal",
|
||||||
|
prefix: entry.target_prefix,
|
||||||
|
peer: null,
|
||||||
|
timestamp: stop.valueOf()
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
this.logger.log({
|
||||||
|
level: 'error',
|
||||||
|
message: `Cannot download historic RIS data ${error}`
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
_subscribe = (input) => {
|
||||||
|
if (this._shouldDownloadDump()) {
|
||||||
|
const asns = input.getMonitoredASns().map(i => i.asn);
|
||||||
|
const prefixes = input.getMonitoredPrefixes().filter(i => !asns.includes(i.asn)).map(i => i.prefix);
|
||||||
|
const dumps = [...asns, ...prefixes];
|
||||||
|
|
||||||
|
if (dumps.length) {
|
||||||
|
this.storage
|
||||||
|
.set(`run-${this.name}`, moment.utc().unix())
|
||||||
|
.catch(error => {
|
||||||
|
this.logger.log({
|
||||||
|
level: 'error',
|
||||||
|
message: error
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return batchPromises(1, dumps, this._loadResource);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
subscribe = (input) => {
|
||||||
|
this._subscribe(input);
|
||||||
|
|
||||||
|
input.onChange(() => {
|
||||||
|
if (this._timeoutFileChange) {
|
||||||
|
clearTimeout(this._timeoutFileChange);
|
||||||
|
}
|
||||||
|
this._timeoutFileChange = setTimeout(() => {
|
||||||
|
this._subscribe(input);
|
||||||
|
}, 2000);
|
||||||
|
});
|
||||||
|
|
||||||
|
return Promise.resolve();
|
||||||
|
};
|
||||||
|
|
||||||
|
static transform = (message) => {
|
||||||
|
return [ message ];
|
||||||
|
};
|
||||||
|
};
|
@@ -140,7 +140,6 @@ export default class Input {
|
|||||||
return p;
|
return p;
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
// if (!this.cache.af[p.prefix] || !this.cache.binaries[p.prefix]) {
|
|
||||||
if (!this.cache.af[p.prefix]) {
|
if (!this.cache.af[p.prefix]) {
|
||||||
this.cache.af[p.prefix] = ipUtils.getAddressFamily(p.prefix);
|
this.cache.af[p.prefix] = ipUtils.getAddressFamily(p.prefix);
|
||||||
this.cache.binaries[p.prefix] = ipUtils.getNetmask(p.prefix, this.cache.af[p.prefix]);
|
this.cache.binaries[p.prefix] = ipUtils.getNetmask(p.prefix, this.cache.af[p.prefix]);
|
||||||
|
Reference in New Issue
Block a user