IS-IS Watcher is a monitoring tool of IS-IS topology changes for network engineers. It works via passively listening to IS-IS control plane messages through a specially established IS-IS adjacency between IS-IS Watcher and one of the network device. The tool logs IS-IS events and/or export by Logstash to Elastic Stack (ELK), Zabbix, WebHooks and Topolograph monitoring dashboard for keeping the history of events, alerting, instant notification. By encapsulating the solution's elements in containers, it becomes exceptionally quick to start. The only thing is needed to configure manually is GRE tunnel setup on the Linux host.
Note
Upvote in issues/1 if you are interested in tracking IS-IS topology changes via BGP-LS.
- IS-IS neighbor adjacency Up/Down
- IS-IS link cost changes
- IS-IS networks appearance/disappearance from the topology
- IS-IS TE attributes:
- Administrative Group (color, resource class)
- Maximum Link Bandwidth
- Maximum Reservable Link Bandwidth
- Unreserved Bandwidth
- Traffic Engineering Default Metric
The FRR container is isolated in an individual network namespace and the XDP IS-IS filter inspects all outgoing IS-IS advertisements. It checks if FRR instance advertises only locally connected network (assigned on GRE tunnel) and no more. If it advertises multiple networks, IS-IS LSP will be dropped. It prevents the network from populating by unexpected network prefixes.
Note
isiswatcher:v1.0 is compatible with topolograph:v2.38, it means that IS-IS network changes can be shown on the network graph.
Watcher stores topology events/state to show historical network state, whereas Topolograph exhibits present network state and potential future outcomes.
TLV name | TLV | subTLV |
---|---|---|
IS Reachability | 2 | |
Extended IS Reachability (new) | 22 | 3,6,8,9,10,11,12,13,18 |
IPv4 Internal Reachability (old) | 128 | |
IPv4 External Reachability (old) | 130 | |
Extended IPv4 Reachability (new) | 135 | |
IPv6 Reachability | 236 |
Number of watchers is equal to the number of IS-IS areas and each Watcher is placed in individual network namespace. IS-IS LSDB sits in watcher's namespace and doesn't interact with other Watchers keeping it isolated.
The demo shows how IS-IS watcher detected:
- p2p links:
- metric cost changes on eth1 interface on router2 from 10 to 11. There were three duplicated logs for each level: L1 and L2.
- 0100.1001.0002 (router2) went down for L2 after applying "isis circuit-type level-1" on it and router3 detected that.
- metric cost changes on eth1 interface on router2 from 11 to 12 only for L1.
- broadcast links:
- new stub network was added on router6 in L2
- 0100.1001.0004 went down and 0100.1001.0005 (router5) detected that
Logs if IS-IS adjacency was Up/Down or any networks appeared/disappeared.
Latest events about links with admin group 17 with unreserved bandwidth for priority of 0 less than 100Mbits
Monitoring dashboard allows to see events on the timeline and on the topology. Red timelines show link (~adjacency) down events, green one - up link (~adjacency).
Events are grouped by IS-IS level (L1/L2).
Here is a demo of checking events on Monitoring dashboard ./docs/isisdemo_with_topolograph_au.mp4
Zabbix's dashboard with active alarms. It's universal method to track OSPF and IS-IS events. The screenshot is taken from OSPF watcher.
This alarm tracks all new IS-IS/OSPF adjacencies or when a device loses its neighbor. The screenshot is taken from OSPF watcher.
Transit links are all links between active IS-IS/OSPF neighbors. If cost on a link was changed it might affect all actual/shortest paths traffic follows. The screenshot is taken from OSPF watcher.
If a subnet was removed from IS-IS/OSPF node (the node withdrew it from the announcement) it means the network from this node became unavailable for others, this event will be logged too. The screenshot is taken from OSPF watcher.
HTTP POST messages can be easily accepted by messengers, which allows to get instant notifications of IS-IS/OSPF topology changes: The screenshot is taken from OSPF watcher.
Containerlab's topology is placed under containerlab folder. Local README
includes instructions how to run it. IS-IS topology changes are printed by Watcher in a text file only.
./containerlab/frr01/prepare.sh
sudo clab deploy --topo ./containerlab/frr01/frr01.clab.yml
Table below shows different options of possible setups, starting from the bare minimum in case of running Containerlab for testing and ending with maximum setup size with Watcher, Topolograph and ELK. The following setup describes setup №1 and №2.
№ | Deployment size | Number of compose files | Text file logs | View changes on network map | Zabbix/HTTP/Messengers notification | Searching events by any field any time |
---|---|---|---|---|---|---|
1 | Bare minimum. Containerlab | 0 | + | - | - | - |
2 | 1. Local Topolograph 2. local compose file with ELK disabled (commented) |
2 | + | + | + | - |
3 | 1. Local Topolograph 2. local compose file with ELK enabled |
3 | + | + | + | + |
- Choose a Linux host with Docker installed
- Setup Topolograph
- launch your own Topolograph on docker using topolograph-docker
- create a user for API authentication using
Local Registration
form on the Topolograph page, add your IP address inAPI/Authorised source IP ranges
. Set variables in.env
file:
Note
TOPOLOGRAPH_HOST
- set the IP address of your host, where the docker is hosted (if you run all demo on a single machine), do not putlocalhost
, because ELK, Topolograph and IS-IS Watcher run in their private network spaceTOPOLOGRAPH_PORT
- by default8080
TOPOLOGRAPH_WEB_API_USERNAME_EMAIL
- by defaultospf@topolograph.com
or put your recently created userTOPOLOGRAPH_WEB_API_PASSWORD
- by defaultospf
TEST_MODE
- if mode isTrue
, a demo IS-IS events from static file will be uploaded, not from FRR
- Setup ELK (skip it, it's only needed for setup № 3)
- if you already have ELK instance running, fill
ELASTIC_IP
in env file and uncomment Elastic config hereisiswatcher/logstash/pipeline/logstash.conf
. Currently additional manual configuration is needed for Index Templates creation, becausecreate.py
script doesn't accept the certificate of ELK. It's needed to have one in case of security setting enabled. Required mapping for the Index Template is inisiswatcher/logstash/index_template/create.py
. To create Index Templates, run:
sudo docker run -it --rm --env-file=./.env -v ./logstash/index_template/create.py:/home/watcher/watcher/create.py vadims06/isis-watcher:latest python3 ./create.py
To create Index Templates, run:
sudo docker run -it --rm --env-file=./.env -v ./logstash/index_template/create.py:/home/watcher/watcher/create.py vadims06/isis-watcher:latest python3 ./create.py
- if not - boot up a new ELK from docker-elk compose. For demo purporse set license of ELK as basic and turn off security. The setting are in docker-elk/elasticsearch/config/elasticsearch.yml
xpack.license.self_generated.type: basic
xpack.security.enabled: false
Note about having Elastic config commented > When the Elastic output plugin fails to connect to the ELK host, it blocks all other outputs and ignores "EXPORT_TO_ELASTICSEARCH_BOOL" value from env file. Regardless of EXPORT_TO_ELASTICSEARCH_BOOL being False, it tries to connect to Elastic host. The solution - uncomment this portion of config in case of having running ELK.
- Setup IS-IS Watcher
git clone https://github.com/Vadims06/isiswatcher.git
cd isiswatcher
Generate configuration files
vadims06/isis-watcher:v1.1
includes a client for generating configurations for each Watcher for each IS-IS area. To generate individual settings - run the client with --action add_watcher
sudo docker run -it --rm --user $UID -v ./:/home/watcher/watcher/ -v /etc/passwd:/etc/passwd:ro -v /etc/group:/etc/group:ro vadims06/isis-watcher:latest python3 ./client.py --action add_watcher
The output:
+---------------------------+
| Watcher Host | +-------------------+
| +------------+ | | Network device |
| | netns FRR | | | |
| | Tunnel [4] | | Tunnel [4] |
| | gre1 [3]TunnelIP----+-----------------------+[2]TunnelIP |
| | eth1------+-vhost1 | +-----+ | IS-IS area num [5]|
| | | Host IP[6]+-------+ LAN |--------[1]Device IP |
| | | | +-----+ | |
| +------------+ | | |
| | +-------------------+
+---------------------------+
The script will create:
- a folder under
watcher
folder with FRR configuration underrouter
folder - a containerlab configuration file with network settings
- an individual watcher log file in
watcher
folder.
To stop IS-IS routes from being installed in the host's routing table, we the following policy has been applied on the watcher:
# frr/config/isisd.conf
route-map TO_KERNEL deny 200
exit
!
ip protocol isis route-map TO_KERNEL
- Start IS-IS Watcher
Install containerlab To start the watcher run the following command.clab deploy
is like adocker compose up -d
command
sudo clab deploy --topo watcher/watcher1-tun1025/config.yml
It will create:
- Individual network namespace for Watcher and FRR
- A pair of tap interfaces to connect the watcher to Linux host
- GRE tunnel in Watcher's namespace
- NAT settings for GRE traffic
- FRR & Watcher instance
- assign XDP IS-IS filter on watcher's tap interface
- Setup GRE tunnel from the network device to the host with the Watcher. An example for Cisco
interface gigabitether0/1
ip address <GRE tunnel ip address>
tunnel mode gre
tunnel source <router-ip>
tunnel destination <host-ip>
ip router isis <name>
isis network point-to-point
It's needed to have a minimum one GRE tunnel to an area, which is needed to be monitored. If the IS-IS domain has multiple areas, setup one GRE in each area. It's a restriction of Link State architecture to know about new/old adjacency or link cost changes via LSPs per area basis only.
- Start log export to Topolograph and/or ELK
docker-compose up -d
- Index Templates
Have been already created by
logstash-index-creator
container in compose yaml file. OpenManagement -> Stack Management -> Index Management ->[ Index Templates ]
to make sure that the following templates are in the list:
- Index Pattern
Create indices with the same name as index templates
Go to:
old ELK
Stack Management/ Kibana/ Stack Management/ Index Pattern -> Create index pattern
new ELK 8.xManagement -> Stack Management -> Index Management -> [ Indices ]
thenCreate index
- isis-watcher-costs-changes
- isis-watcher-updown-events
- isis-watcher-temetric-changes
- Data View
Create data view for two event types.
Go to
Management -> Stack Management -> Data Views
thenCreate data view
Name: isis-watcher-costs-changes
Index pattern: isis-watcher-costs-changes
Timestamp field: use watcher time
Repeat the same for isis-watcher-updown-events
and for isis-watcher-temetric-changes
As a result, there are two data views should be listed
Note What time to use @timestamp or watcher
It's better to use watcher
time, because connection between Watcher and Logstash can be lost, but the watcher continues to log all topology changes with the correct time. When the connection is repaired, all logs will be added to ELK and you can check the time of the incident. If you choose @timestamp
- the time of all logs will be the time of their addition to ELK.
- Additional checks Make sure that:
.env
hasEXPORT_TO_ELASTICSEARCH_BOOL=True
./logstash/pipeline/logstash.conf
has ELK uncommented
Zabbix settings are available here /docs/zabbix-ui
. There are 4 hosts and items (host and item inside each host has the same names) are required:
- isis_neighbor_up_down
- isis_network_up_down
- isis_link_cost_change
- isis_stub_network_cost_change
- Create a Slack app
- Enable Incoming Webhooks
- Create an Incoming Webhook (generates URL)
- Uncomment
EXPORT_TO_WEBHOOK_URL_BOOL
in.env
, set the URL toWEBHOOK_URL
Text logs are available here isiswatcher/watcher/logs/
If Topolograph is installed, logs are available under Real-Time Monitoring
tab
If ELK is installed, logs are available here http://localhost:5601/ -> Analytics/Discover
watcher-updown-events
.
2023-01-01T00:00:00Z,demo-watcher,1,host,0200.1001.0002,down,0200.1001.0003,01Jan2023_00h00m00s_7_hosts,49.0002,12345
2023-01-01T00:00:00Z
- event timestampdemo-watcher
- name of watcher1
- IS-IS levelhost
- event name:host
,network
,metric
,temetric
0200.1001.0002
- event object. Watcher detected an event related to0200.1001.0002
hostdown
- event status:down
,up
,changed
0200.1001.0003
- event detected by this node.01Jan2023_00h00m00s_7_hosts
- name of graph in Topolograph dashboard49.0002
- area number12345
- AS number10.1.23.3
- Local IP address of detected node0200.1001.0003
(available in 2.0.15)10.1.23.2
- Remote IP address relative to the detected node0200.1001.0003
(available in 2.0.16) Summary:0200.1001.0003
on the interface with10.1.23.3
IP address detected that0200.1001.0002
host went down at2023-01-01T00:00:00Z
in IS-IS level 1 in area 49.0002 in AS 12345
2023-01-01T00:00:00Z,isis-watcher,2,metric,4ffe::192:168:23:2/127,changed,old_cost:10,new_cost:12,0200.1001.0002,stub,0200.1001.0002,01Jan2023_00h00m00s_7_hosts,49.0002,12345,external,1
2023-01-01T00:00:00Z
- event timestampisis-watcher
- name of watcher2
- IS-IS levelmetric
- event name:host
,network
,metric
,temetric
4ffe::192:168:23:2/127
- event object. Watcher detected an event related to 4ffe::192:168:23:2/127` subnetchanged
- event status:down
,up
,changed
10
- old cost12
- new cost0200.1001.0002
- event detected by this node.stub
- subnet type0200.1001.0002
- since it's a stub network it has router id of terminated node.01Jan2023_00h00m00s_7_hosts
- name of graph in Topolograph dashboard49.0002
- area number12345
- AS numberexternal
- subnet type internal|external1
- subnet ext type 1|2. 0 for internal subnets Summary:0200.1001.0002
detected that metric of4ffe::192:168:23:2/127
stub network changed from10
to12
at2023-01-01T00:00:00Z
in IS-IS level 2
2024-12-29T13:20:50.398Z,
isis-watcher,1,temetric,0200.1001.0002,changed,0_17_19_20_21_22_26_29_30,1000000000,1000000000,1000000008_1000000016_1000000024_1000000032_1000000040_1000000048_1000000056,11223344,0200.1001.0003,2024-07-28T18:03:05Z,49.0001,01Jan2023_00h00m00s_7_hosts,10.1.23.3,10.1.23.2
* `2023-01-01T00:00:00Z` - event timestamp
* `demo-watcher` - name of watcher
* `1` - IS-IS level
* `temetric` - event name: `host`, `network`, `metric`, `temetric`
* `0200.1001.0002` - event object. Watcher detected an event related to `0200.1001.0002` host
* `changed` - event status: `down`, `up`, `changed`
* `0_17_19_20_21_22_26_29_30` - 0, 17, 19, 20, 21, 22, 26, 29, 30 admin groups
* `1000000000` - Maximum Link Bandwidth (Sub-TLV 9) bits per sec
* `1000000000` - Maximum Reservable Link Bandwidth (Sub-TLV 10) bits per sec
* `1000000008_1000000016_1000000024_1000000032_1000000040_1000000048_1000000056` - Unreserved Bandwidth (Sub-TLV 11) for priority 0,..7 bits per sec
* `11223344` - Traffic Engineering Default Metric (Sub-TLV 18)
* `0200.1001.0003` - event detected by this node.
* `01Jan2023_00h00m00s_7_hosts` - name of graph in Topolograph dashboard
* `49.0002` - area number
* `12345` - AS number
* `10.1.23.3` - Local IP address of detected node `0200.1001.0003` (available in 2.0.15)
* `10.1.23.2` - Remote IP address relative to the detected node `0200.1001.0003` (available in 2.0.16)
FRR configuration
interface eth1
ip address 20.168.23.1/24
ip router isis lab
ipv6 address 4ffe::192:168:23:2/127
ipv6 router isis lab
isis network point-to-point
link-params
admin-grp 0x647a0001
enable
metric 11223344
max-bw 1e+08
unrsv-bw 0 1.23e+06
unrsv-bw 5 1.23e+06
!
router isis lab
net 49.0002.0200.1001.0002.00
mpls-te on
mpls-te router-address 10.10.10.2
exit
If, for some reason, an extra network is advertised from Watcher, this announcement will be dropped.
This examples shows that 8.8.8.8
prefix was redistributed on Watcher and added into its announcement, but it was dropped by XDP and eventually didn't reach the network.
To check XDP logs, run
sudo cat /sys/kernel/debug/tracing/trace_pipe
Networks changes are not tracked. Log file ./watcher/logs/watcher...log
is empty.
-
Run diagnostic script. It will check IS-IS Watcher <-> Network device connection (iptables, packets from FRR/network device)
sudo docker run -it --rm -v ./:/home/watcher/watcher/ --cap-add=NET_ADMIN -u root --network host vadims06/isis-watcher:latest python3 ./client.py --action diagnostic --watcher_num <num>
-
Login on FRR, check adjancency:
sudo docker exec -it watcher<num>-gre<num>-router vtysh show isis neighbor
if there is no IS-IS neighbor, ping remote end of GRE tunnel from the Watcher. At the same time, make tcpdump on watcher's interface and check counters of iptables
sudo iptables -nvL -t filter --line-numbers sudo iptables -nvL -t nat --line-numbers
Clear connections of GRE tunnel
sudo conntrack -D -p 47
Check ICMP ping packets on Watcher's host and on network device.
Dashboard page is blank. Events are not present on OSPF/IS-IS Monitoring page.
IS-IS Watcher consists of three services: IS-ISd/FRR [1] -> Watcher [2] -> Logstash [3] -> Topolograph & ELK & Zabbix & WebHooks.
-
Check if FRR tracks IS-IS changes in
./watcher/logs/watcher...log
file (previous case)
You should see tracked changes of your network, i.e.2024-10-08T22:54:54Z,watcher1,1,network,4ffe::192:168:145:4/127,changed,old_cost:44,new_cost:4,0200.1001.0004,,49.0002,12345,internal,0
-
Check that logstash container from docker-compose.yml is running via
docker ps
command.- Uncomment
DEBUG_BOOL="True"
in.env
and start continuous logsdocker logs -f logstash
. - Copy and paste the log from the first step in watcher's log file
./watcher/logs/watcher#-gre#-isis.isis.log
.docker logs -f logstash
should print the output. If not - check logstash container.
- Uncomment
-
Check if logs are in Topolograph's DB. Connect to mongoDB and run:
docker exec -it mongo /bin/bash
Inside container (change):
mongo mongodb://$MONGO_INITDB_ROOT_USERNAME:$MONGO_INITDB_ROOT_PASSWORD@mongodb:27017/admin?gssapiServiceName=mongodb use admin
Check the last two/N records in adjacency changes (
adj_change
) or cost changes (cost_change
)db.adj_change.find({}).sort({_id: -1}).limit(2) db.cost_change.find({}).sort({_id: -1}).limit(2)
Note
If you see a single event indocker logs logstash
it means that mongoDB output is blocked, check if you have a connection to MongoDBdocker exec -it logstash curl -v mongodb:27017
FRR 8 perfectly logs any IS-IS LSPs, but doesn't establish IS-IS adjacency over GRE because of internal packet filtering. The filter has been updated to permit IS-IS over GRE through Pull request #12979 and is only supported in FRR 9 versions. However, this version has wrong LSDB output (mix LSPID with hostnames) and is incompatible with the Watcher. Therefore, a custom FRR build based on FRR 8.x version with an altered internal filter for GRE packets is required. vadims06/frr:v8.5.4_isis_over_gre
is prepaired and tested to work IS-IS over GRE tunnel. If you need another 8.x version or want to build your own FRR image - follow the instructions mentioned below.
- clone FRR and choose any FRR 8.x branch
git clone https://github.com/FRRouting/frr.git
cd frr
git fetch --all --tags
git checkout tags/frr-8.5.4 -b frr-8.5.4
- Change isis filter
frr/isisd/isis_pfpacket.c
how it's mentioned in this GRE filter CR.
vi frr/isisd/isis_pfpacket.c
- Build FRR container
sudo docker/alpine/build.sh
- Inspect your new FRR image name using
docker image ls
and replacerouter/image
by your own image name inisiswatcher/watcher/watcher-template/config.yml
7.17.21, this version includes bug fix of issues_281, issues_5115
- https://t.me/topolograph
- admin at topolograph.com
The functionality was tested using Basic ELK license.