Visualisation with GraphViz
GraphViz is an ingenious package created by IBM to visualise lots of possible object topologies. The package makes it very easy to create all kinds of directed and indirected graphs from raw data.
Hereby I share with you some simple code to create a very basic network topology graph from RackTables database. All updates (expecially fine-tuning the .dot file) are much welcome, since there are hundreds of possible tweaks in dot (the submodule I'm using here) itself.
I use dot (which is the most general topology module) because my tests showed that default layouts of other modules resulted inferior outputs. Still, the .dot file is the same, and you're all free to tweak other settings and show us all. :-)
Code requires perl (DBI / DBD::mysql for database access) and graphviz. If you use accented characters for strings you have to recode the output to UTF-8 somehow, I'm using konwert for that, but recode or other tools would do just fine, too.
dottr.pl
#!/usr/bin/perl ##$Id: dottr.pl,v 5aa2d11fa948 2012/11/10 10:57:04 grin $ # # (c) Peter 'grin' Gervai, 2012 # Released under CreativeCommons-Attribution-ShareAlike-3.0-Unported # # create .dot from racktables links # use DBI; use Encode; use DatabaseLoginData; # $db_user; $db_pw; $db_host my ($db_name, $db_port) = ("racktables_db", 3306); my $dbh = DBI->connect( "dbi:mysql:database=$db_name;host=$DatabaseLoginData::db_host;port=$db_port", $DatabaseLoginData::db_user, $DatabaseLoginData::db_pw, { RaiseError =>1, AutoCommit => 1 } ); ######################################################################### my $sth_links = $dbh->prepare(" SELECT ro1.name AS obj1, p1.name AS port1, Link.cable, p2.name AS port2, ro2.name AS obj2, d.dict_value AS obj1type FROM RackObject AS ro1 JOIN Port AS p1 ON(ro1.id=p1.object_id) JOIN Link ON(p1.id=Link.porta) JOIN Port AS p2 ON(Link.portb=p2.id) JOIN RackObject AS ro2 ON(p2.object_id=ro2.id) LEFT JOIN Dictionary AS d ON(ro1.objtype_id=d.dict_key) ORDER BY obj1, port1"); my $sth_reservations = $dbh->prepare(" SELECT ro.name AS obj1, p.name AS port1, 'RESERVE' AS cable, '' AS port2, CONCAT(reservation_comment,' #',ROUND(RAND()*1000)) AS obj2, d.dict_value AS obj1type FROM RackObject AS ro JOIN Port AS p ON(ro.id=p.object_id) LEFT JOIN Dictionary AS d ON(ro.objtype_id=d.dict_key) WHERE p.reservation_comment IS NOT NULL ORDER BY obj1, port1"); my $res = $sth_links->execute; my $a = $sth_links->fetchall_arrayref; my @data = @$a; $res = $sth_reservations->execute; $a = $sth_reservations->fetchall_arrayref; @data = (@data, @$a); # header print ' graph rackspace_topo { label = "Tarr Racktables topo"; rankdir = LR; edge [ color="#0000a0", decorate=true, fontsize=9, headclip=false ]; node [ shape=box, headport=n, tailport=n ]; '; # generate edges foreach $a (@data) { # node+interface 1 $node{$a->[0]} = $a->[1]; $nodetype{$a->[0]} = $a->[5]; # node+interface 2 $node{$a->[4]} = $a->[3]; # push @link, $a; } # output the nods first foreach my $n (keys %node) { # nodes with empty or "space" names next if $n =~ /^\s*$/; $ntyp = $nodetype{$n}; # color/style depends on node type my $color=""; if( $ntyp eq "Server" ) { $color='color="black"'; } elsif( $ntyp eq "Network switch" ) { $color='color="red" style="bold"'; } elsif( $ntyp eq "Server chassis" ) { $color='color="blue" style="dotted"'; } elsif( $ntyp eq "Network chassis" ) { $color='color="lightblue"'; } elsif( $ntyp eq "MediaConverter" ) { $color='color="gold"'; } elsif( $ntyp eq "Router" ) { $color='color="darkgreen"'; } else { #$color='color="pink"'; # Barbie syndrome FTW } print '"' . $n . "\" [ $color ];\n"; } # and print the edges foreach my $l (@link) { # skip empty... next if $l->[0] =~ /^\s*$/; if( defined( $link{ $l->[0] }{ $l->[4] } ) && $link{ $l->[0] }{ $l->[4] } == $l->[2] ) { # skip duplicate # if you need warnings... here you can. } else { print '"', $l->[0], '" -- "', $l->[4], '" [ label="', $l->[2], "\" color=\"grey\"];\n"; $link{ $l->[0] }{ $l->[4] } = $l->[2]; } } print "\n};\n\n";
create_svg.sh
#!/bin/bash #$Id: create_svg.sh,v 5aa2d11fa948 2012/11/10 10:57:04 grin $ # # (c) Peter 'grin' Gervai, 2012 # Released under CreativeCommons-Attribution-ShareAlike-3.0-Unported # # create .svg from racktables links # recode the file # first create the .dot file, and recode it using konwert # skip recoding if you're happy without accents, screws up svg otherwise echo "Creating .dot file from the database..." ./dottr.pl | konwert isolatin2-utf8 > map.dot # dot output is fine, others (like neato or circo) not that useful echo "Rendering map.svg and map.png..." dot -Tsvg -o map.svg map.dot dot -Tpng -o map.png map.dot echo "That's all, folks!"