Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Diagram class related to #127 #135

Open
wants to merge 14 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 11 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 5 additions & 3 deletions autogole-api/packaging/Dockerfile
Original file line number Diff line number Diff line change
@@ -1,11 +1,12 @@
FROM opensciencegrid/software-base:23-al8-release

RUN yum -y install wget epel-release && \
yum -y install git python3 python3-pyyaml python3-devel python3-pip gcc openssl-devel cronie python3-pyOpenSSL fetch-crl && \
yum -y install git python3 python3-pyyaml python3-devel python3-pip gcc openssl-devel cronie python3-pyOpenSSL fetch-crl graphviz && \
yum clean all

RUN mkdir -p /opt/ && \
mkdir -p /srv/ && \
mkdir -p /srv/icons/ && \
mkdir -p /etc/rtmon/templates/ && \
mkdir -p /var/log/rtmon/ && \
mkdir -p /etc/grid-security/certificates/
Expand All @@ -25,7 +26,8 @@ RUN git clone https://github.com/esnet/sense-rtmon.git /opt/sense-rtmon && \

RUN wget https://raw.githubusercontent.com/sdn-sense/rm-configs/master/CAs/SiteRM.pem -O /etc/grid-security/certificates/e52ac827.0

ADD files/etc/supervisord.d/10-server.conf /etc/supervisord.d/10-server.conf

COPY files/etc/supervisord.d/10-server.conf /etc/supervisord.d/10-server.conf
COPY icons/host.png /srv/icons/host.png
COPY icons/switch.png /srv/icons/switch.png
# Get latest CA's
RUN fetch-crl || echo "Supress warnings."
8 changes: 5 additions & 3 deletions autogole-api/packaging/files/etc/rtmon.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,13 @@
sleep_timer: 30
# Work dir for temp files (api directory). Default /srv
workdir: '/srv/'

image_dir: '/srv/images'
# Grafana settings (for the API)
grafana_host: 'https://autogole-grafana.nrp-nautilus.io'
grafana_api_key: 'REPLACE_ME'
grafana_api_key: ''
baseImageURL: 'http://localhost:8080/images'
grafana_folder: 'Real Time Mon'
grafana_dev: 'Sunami-Image-Server'
# FOR DEVELOPMEENT ONLY
# Enable grafana_dev parameter (can be any string, and will be used as name to create directory inside Grafana)
# Additionally - all dashboards will have this added to name.
Expand All @@ -25,7 +27,7 @@ data_sources:

# Sense endpoints and their auth files.
sense_endpoints:
"sense-o.es.net": "/etc/sense-o-auth-prod.yaml"
#"sense-o.es.net": "/etc/sense-o-auth-prod.yaml"
"sense-o-dev.es.net": "/etc/sense-o-auth.yaml"

# Additional links for the templates
Expand Down
Binary file added autogole-api/packaging/icons/host.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added autogole-api/packaging/icons/switch.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
36 changes: 36 additions & 0 deletions autogole-api/packaging/imageserver.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
from flask import Flask, send_from_directory, render_template_string, abort
import os

app = Flask(__name__)

IMAGE_DIRECTORY = '/srv/images'
@app.route('/')
def home():
return "<h1>Hello, this is the root page!</h1>"

@app.route('/images')
def list_images():
if not os.path.exists(IMAGE_DIRECTORY):
return "<h1>Image directory not found.</h1>", 404

images = [f for f in os.listdir(IMAGE_DIRECTORY) if os.path.isfile(os.path.join(IMAGE_DIRECTORY, f))]

html = """
<h1>Network Topology</h1>
<ul>
{% for image in images %}
<li><a href="/images/{{ image }}">{{ image }}</a></li>
{% endfor %}
</ul>
"""
return render_template_string(html, images=images)

@app.route('/images/<filename>')
def serve_image(filename):
if not os.path.isfile(os.path.join(IMAGE_DIRECTORY, filename)):
abort(404)
return send_from_directory(IMAGE_DIRECTORY, filename)

if __name__ == "__main__":
port = 8000
app.run(host='0.0.0.0', port=port, debug=True)
2 changes: 1 addition & 1 deletion autogole-api/packaging/start-dev.sh
Original file line number Diff line number Diff line change
Expand Up @@ -14,5 +14,5 @@ docker run \
-v $(pwd)/files/etc/grid-security/hostcert.pem:/etc/grid-security/hostcert.pem:ro \
-v $(pwd)/files/etc/grid-security/hostcert.pem:/etc/grid-security/hostkey.pem:ro \
--restart always \
--net=host \
-p 8000:8000 \
rtmon
2 changes: 2 additions & 0 deletions autogole-api/requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -2,3 +2,5 @@ psutil
requests
pyyaml
grafana-client
diagrams
flask
6 changes: 6 additions & 0 deletions autogole-api/src/python/RTMon/worker.py
Original file line number Diff line number Diff line change
Expand Up @@ -84,8 +84,14 @@ def submit_exe(self, filename, fout):
self._updateState(filename, fout)

def delete_exe(self, filename, fout):

"""Delete Action Execution"""
self.logger.info('Delete Execution: %s, %s', filename, fout)
#Deleting the diagram image
diagram_filename = f"{self.config.get('image_dir', '/srv/images')}/diagram_{fout['referenceUUID']}.png"
if os.path.exists(diagram_filename):
os.remove(diagram_filename)
self.logger.info(f"Removed diagram image {diagram_filename}")
# Delete the dashboard and template from Grafana
for grafDir, dirVals in self.dashboards.items():
for dashbName, dashbVals in dirVals.items():
Expand Down
228 changes: 228 additions & 0 deletions autogole-api/src/python/RTMonLibs/DiagramWorker.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,228 @@
"""
DiagramWorker Class for Network Topology Visualization

sunami09 marked this conversation as resolved.
Show resolved Hide resolved
This module contains the DiagramWorker class, which generates network topology diagrams
by processing input data that includes hosts and switches. It uses the 'diagrams' library
to visualize network components and their interconnections.
"""
import os
from diagrams import Diagram, Cluster, Edge
from diagrams.custom import Custom
from RTMonLibs.GeneralLibs import _processName
# # change later:
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

remove this code

# def _processName(name):
# """Process Name for Mermaid and replace all special chars with _"""
# for repl in [[" ", "_"], [":", "_"], ["/", "_"], ["-", "_"], [".", "_"], ["?", "_"]]:
# name = name.replace(repl[0], repl[1])
# return name
# #########

class DiagramWorker:
sunami09 marked this conversation as resolved.
Show resolved Hide resolved
"""
DiagramWorker class is responsible for generating network topology diagrams
using the input data that contains host and switch information. The class
identifies and visualizes links between network components.
"""
HOST_ICON_PATH = '/srv/icons/host.png'
SWITCH_ICON_PATH = '/srv/icons/switch.png'
# # TEmp
# HOST_ICON_PATH = '/Users/sunami/Desktop/publish/sense-rtmon/autogole-api/packaging/icons/host.png'
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

remove too

# SWITCH_ICON_PATH = '/Users/sunami/Desktop/publish/sense-rtmon/autogole-api/packaging/icons/switch.png'

def __init__(self, indata, instance, manifest):
"""
Initialize the DiagramWorker with input data.

:param indata: List of dictionaries containing host and switch details.
"""
self.indata = indata
self.objects = {}
self.added = {}
self.linksadded = set()
self.popreverse = None
self.instance = instance


def d_find_item(self, fval, fkey):
"""Find Item where fkey == fval"""
for key, vals in self.objects.items():
if vals.get('data', {}).get(fkey, '') == fval:
return key, vals
return None, None

@staticmethod
def d_LinkLabel(vals1, vals2):
"""Get Link Label"""
label = ""
if vals1.get('data', {}).get('Type', '') == "Host":
label = f"Port1: {vals1['data']['Interface']}"
elif vals1.get('data', {}).get('Type', '') == "Switch":
label = f"Port1: {vals1['data']['Name']}"
# Get second side info:
if vals2.get('data', {}).get('Type', '') == "Host":
label += f"\nPort2: {vals2['data']['Interface']}"
elif vals2.get('data', {}).get('Type', '') == "Switch":
label += f"\nPort2: {vals2['data']['Name']}"
if vals1.get('data', {}).get('Vlan', None):
label += f"\nVlan: {vals1['data']['Vlan']}"
elif vals2.get('data', {}).get('Vlan', None):
label += f"\nVlan: {vals2['data']['Vlan']}"
return label

def d_addLink(self, val1, val2, key, fkey):
"""Add Link between 2 objects"""
if val1 and val2 and key and fkey:
if key == fkey:
return

link_keys = tuple(sorted([key, fkey]))
if link_keys in self.linksadded:
return
self.linksadded.add(link_keys)

val1["obj"] >> Edge(label=self.d_LinkLabel(val1, val2)) << val2["obj"]

def d_addLinks(self):
"""Identify Links between items"""
for key, vals in self.objects.items():
data_type = vals.get('data', {}).get('Type', '')
if data_type == "Host":
fKey, fItem = self.d_find_item(key, 'PeerHost')
if fKey and fItem:
self.d_addLink(vals, fItem, key, fKey)
elif data_type == "Switch":
if 'Peer' in vals.get('data', {}) and vals['data']['Peer'] != "?peer?":
fKey, fItem = self.d_find_item(vals['data']['Peer'], "Port")
if fKey and fItem:
self.d_addLink(vals, fItem, key, fKey)
elif 'PeerHost' in vals.get('data', {}):
fKey = vals['data']['PeerHost']
fItem = self.objects.get(fKey)
if fItem:
self.d_addLink(vals, fItem, key, fKey)

def d_addHost(self, item):
"""
Add a host to the network diagram.

:param item: Dictionary containing host details.
:return: Diagram object representing the host.
"""
name = f"Host: {item['Name'].split(':')[1]}"
name += f"\nInterface: {item['Interface']}"
name += f"\nVlan: {item['Vlan']}"
if 'IPv4' in item and item['IPv4'] != "?ipv4?":
name += f"\nIPv4: {item['IPv4']}"
if 'IPv6' in item and item['IPv6'] != "?ipv6?":
name += f"\nIPv6: {item['IPv6']}"

worker = Custom(name, self.HOST_ICON_PATH)
self.objects[item['Name']] = {"obj": worker, "data": item}
return worker

def d_addSwitch(self, item):
"""
Add a switch to the network diagram.

:param item: Dictionary containing switch details.
:return: Diagram object representing the switch.
"""
if item['Node'] in self.added:
self.objects[item['Port']] = {
"obj": self.objects[self.added[item['Node']]]["obj"],
"data": item
}
return
switch1 = Custom(item['Node'].split(":")[1], self.SWITCH_ICON_PATH)
if 'Peer' in item and item['Peer'] != "?peer?":
self.added[item['Node']] = item['Port']
self.objects[item['Port']] = {"obj": switch1, "data": item}
elif 'PeerHost' in item:
uniqname = _processName(f'{item["Node"]}_{item["Name"]}')
self.added[item['Node']] = uniqname
self.objects[uniqname] = {"obj": switch1, "data": item}
# Add IPv4/IPv6 on the switch
for ipkey, ipdef in {'IPv4': '?port_ipv4?', 'IPv6': '?port_ipv6?'}.items():
if ipkey in item and item[ipkey] != ipdef:
ip_node_name = f"{item['Node']}_{ipkey}"
ip_label = item[ipkey]
ip_label2 = ""
sitename = item["Site"]

if self.instance != None:
tempData = self.instance["intents"]
for flow in tempData:
terminals = flow["json"]["data"]["connections"][0]["terminals"]
for connection in terminals:
if connection["uri"] == sitename:
ip_label2 = connection["ipv6_prefix_list"]
break
ip_node = Custom(ip_label + "\n" + ip_label2, self.HOST_ICON_PATH)
switch1 >> Edge() << ip_node

return switch1




def addItem(self, item):
"""
Add an item (host or switch) to the diagram by identifying its type and location (cluster).

:param item: Dictionary containing item details.
:return: Diagram object representing the item.
"""
site = self.identifySite(item)
if item['Type'] == 'Host':
with Cluster(site):
return self.d_addHost(item)
elif item['Type'] == 'Switch':
with Cluster(site):
return self.d_addSwitch(item)

def identifySite(self, item):
"""
Identify the site or cluster to which the item (host or switch) belongs.

:param item: Dictionary containing item details.
:return: The name of the site or cluster.
"""
site = None
if item['Type'] == 'Host':
site = item['Name'].split(':')[0]
elif item['Type'] == 'Switch':
site = item['Node'].split(':')[0]
return site

def setreverse(self, item):
"""
Set the reverse flag for alternating between the first and last items in the input list.

:param item: Dictionary containing item details.
"""
if item['Type'] == 'Host' and self.popreverse is None:
self.popreverse = False
elif item['Type'] == 'Host' and self.popreverse is False:
self.popreverse = True
elif item['Type'] == 'Host' and self.popreverse is True:
self.popreverse = False

def createGraph(self, output_filename):
"""
Create the network topology diagram and save it to a file.

:param output_filename: Path where the output diagram will be saved.
"""
output_dir = os.path.dirname(output_filename)
if not os.path.exists(output_dir):
os.makedirs(output_dir)

with Diagram("Network Topology", show=False, filename=output_filename):
while len(self.indata) > 0:
if self.popreverse in (None, False):
item = self.indata.pop(0)
elif self.popreverse == True:
item = self.indata.pop()
self.addItem(item)
self.setreverse(item)
self.d_addLinks()
6 changes: 6 additions & 0 deletions autogole-api/src/python/RTMonLibs/GeneralLibs.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,12 @@
from yaml import safe_load as yload
from yaml import safe_dump as ydump

def _processName(name):
"""Process Name for Mermaid and replace all special chars with _"""
for repl in [[" ", "_"], [":", "_"], ["/", "_"], ["-", "_"], [".", "_"], ["?", "_"]]:
name = name.replace(repl[0], repl[1])
return name

def getUUID(inputstr):
"""Generate UUID from Input Str"""
hashObject = hashlib.sha256(inputstr.encode('utf-8'))
Expand Down
Loading
Loading