fix typo
[mirror/userdir-ldap-cgi.git] / machines.cgi
1 #!/usr/bin/perl
2 # $Id: machines.cgi,v 1.12 2006/12/27 23:00:04 rmurray Exp $
3
4 # (c) 1999 Randolph Chung. Licensed under the GPL. <tausq@debian.org>
5 # (c) 2004 Martin Schulze. Licensed under the GPL. <joey@debian.org>
6 # (c) 2006 Ryan Murray. Licensed under the GPL. <rmurray@debian.org>
7 # (c) 2008 Martin Zobel-Helas. Licensed under the GPL. <zobel@debian.org>
8
9 use lib '.';
10 use strict vars;
11 #use Apache::Registry;
12 use CGI;
13 use Util;
14 use Net::LDAP qw(LDAP_SUCCESS LDAP_PROTOCOL_ERROR);
15 use Fcntl;
16 use POSIX;
17 use MIME::Base64;
18 use Digest::MD5 qw(md5_hex);
19
20 my (%attrs, @attrorder, %summaryattrs, @summaryorder, %summarylistitems);
21
22 # This defines the description of the fields, and which fields are retrieved
23 %attrs = ('hostname' => 'Host name',
24           'admin' => 'Admin contact',
25           'architecture' => 'Architecture',
26           'distribution' => 'Distribution',
27           'access' => 'Access',
28           'sponsor' => 'Sponsor',
29           'sponsor-admin' => 'Sponsor admin',
30           'location' => 'Location',
31           'machine' => 'Processor',
32           'memory' => 'Memory',
33           'disk' => 'Disk space',
34           'bandwidth' => 'Bandwidth',
35           'status' => 'Status',
36           'notes' => 'Notes',
37           'sshrsahostkey' => 'SSH host key',
38           'sshrsahostfprint' => 'SSH host fingerprint',
39           'description' => 'Description',
40           'purpose' => 'purposes of this server',
41 #         'createtimestamp' => 'Entry created',
42 #         'modifytimestamp' => 'Entry modified'
43          );
44
45 # This defines what fields are displayed, and in what order
46 @attrorder = qw(hostname admin architecture distribution access
47                 sponsor sponsor-admin location machine memory
48                 disk bandwidth status notes sshrsahostkey sshrsahostfprint
49                 description purpose);
50
51 # ditto for summary
52 %summaryattrs = ('hostname' => 'Host name',
53                  'host'     => 'just for a link',
54                  'architecture' => 'Architecture',
55                  'status' => 'Status',
56                  'access' => 'Access',
57                  'sponsor' => 'Sponsor',
58                  'purpose' => 'Purpose');
59 @summaryorder = qw{hostname architecture sponsor purpose status access};
60 %summarylistitems = map {$_=>1} qw{purpose sponsor};
61
62 # Global settings...
63 my %config = &Util::ReadConfigFile;
64
65 my ($ldap, $mesg, $dn, $entries, $data, %output, $key, $hostlist, $hostdetails, $selected, %summary);
66 sub DieHandler {
67   $ldap->unbind if (defined($ldap));
68 }
69
70 # human readable fingerprint
71 sub sshfingerprint {
72     my $key = shift;
73
74     return '' if (!$key);
75
76     my @field = split(/ /, $key);
77     my %keytypes = map {$_=>1} (qw{ssh-dss ssh-rsa ecdsa-sha2-nistp256});
78     return '' unless $keytypes{$field[0]};
79     return '' if !$field[1];
80     my $fpr = md5_hex(decode_base64($field[1]));
81     my $hrfpr = $field[0] . " " . substr($fpr,0,2,"");
82     while (length $fpr > 0) {
83        $hrfpr .= ':' . substr($fpr,0,2,"");
84     }
85     return $hrfpr;
86 }
87
88 sub wiki_link($) {
89         my ($in) = @_;
90         # [[hostname|text]] makes a link
91         # [[hostname]] makes a link too
92         # if you add a * after [[ it's still the same, only not used for ssh_known_hosts in ud-generate
93         # [[-hostname]] are not links, but get added to known_hosts.  we should drop the [[- ]] tho
94         $in =~ s#\[\[-(.*?)\]\]#$1#g;
95         $in =~ s#\[\[\*?(.*?)\|(.*?)\]\]#<a href="http://$1">$2</a>#g;
96         $in =~ s#\[\[\*?(.*?)\]\]#<a href="http://$1">$1</a>#g;
97         return $in;
98 }
99
100 # in the purpose field [[host|some other text]] (where some other text is optional)
101 # makes a hyperlink on the web thing. we now also add these hosts to the ssh known_hosts
102 # file.  But so that we don't have to add everything we link we can add an asterisk
103 # and say [[*... to ignore it.  In order to be able to add stuff to ssh without
104 # http linking it we also support [[-hostname]] entries.
105 #
106 # sponsors are also wikified like purpose.  maybe others as well
107 sub item_uplist($) {
108         my ($items) = @_;
109         my $out = undef;
110         my(@tmp) = @$items;
111
112         if (scalar @tmp>= 1) {
113                 $out = "<ul>".
114                         join("", map { 
115                                 "<li>".wiki_link($_)."</li>\n";
116                           } sort {my $A=$a; my $B=$b; $A =~ s/[\[\]\*]//g; $B =~ s/[\[\]\*]//g; $A cmp $B} @tmp
117                         ).
118                         "</ul>";
119         }
120         return $out;
121 }
122
123 #$SIG{__DIE__} = \&DieHandler;
124
125 my $query = new CGI;
126 my $host = lc($query->param('host'));
127 my $sortby = lc($query->param('sortby')) || "host";
128 my $sortorder = lc($query->param('sortorder')) || "asc";
129
130
131 &Util::HTMLSendHeader;
132 $ldap = Net::LDAP->new($config{ldaphost}) || &Util::HTMLError($!);
133 &Util::UpgradeConnection($ldap) unless $config{usessl} eq 'False';
134 $mesg;
135 $ldap->bind;
136
137 $mesg = $ldap->search(base  => $config{hostbasedn}, filter => 'host=*');
138 $mesg->code && &Util::HTMLError($mesg->error);
139 $entries = $mesg->as_struct;
140
141 foreach $dn (sort {$entries->{$a}->{host}->[0] cmp $entries->{$b}->{host}->[0]} keys(%$entries)) {
142   $data = $entries->{$dn};
143
144   my $thishost = $data->{host}->[0];
145   $selected = "";
146   
147   if (lc($thishost) eq $host) {
148     $output{havehostdata} = 1;
149
150     foreach $key (keys(%attrs)) {
151       $output{$key} = $data->{$key}->[0];
152     }
153   
154     $output{hostname} = undef;
155     foreach my $hostname (@{$data->{hostname}}) {
156       $output{hostname} .= sprintf("%s%s", ($output{hostname} ? ', ' : ''), $hostname);
157     }
158
159     # Modified/created time. TODO: maybe add is the name of the creator/modifier
160     $output{modifytimestamp} = &Util::FormatTimestamp($output{modifytimestamp});
161     $output{createtimestamp}  = &Util::FormatTimestamp($output{createtimestamp});
162     
163     # Format email addresses
164     $output{admin} = sprintf("<a href=\"mailto:%s\">%s</a>", $output{admin}, $output{admin});
165     $output{'sponsor-admin'} = sprintf("<a href=\"mailto:%s\">%s</a>", $output{'sponsor-admin'}, $output{'sponsor-admin'});
166
167     $output{sshrsahostkey} = undef;
168     foreach $key (@{$data->{sshrsahostkey}}) {
169       $output{sshrsahostkey} .= $key . "<br>";
170     }
171
172     foreach $key (@{$data->{sshrsahostkey}}) {
173       $output{sshrsahostfprint} .= sshfingerprint($key) . "<br>";
174     }
175     
176     my $sponsor = item_uplist($data->{sponsor});
177     $output{sponsor} = $sponsor if defined $sponsor;
178     my $purpose = item_uplist($data->{purpose});
179     $output{purpose} = $purpose if defined $purpose;
180
181     $selected = " selected ";
182   }
183
184   $hostlist .= "<option value=\"$thishost\"$selected>$thishost\n" unless ($data->{status}->[0] =~ /^unlisted/);
185
186   # collect summary info
187   foreach $key (keys(%summaryattrs)) {
188     if (exists $summarylistitems{$key}) {
189       my $v = item_uplist($data->{$key});
190       $summary{$thishost}{$key} = $v if defined $v;
191     } else {
192       $summary{$thishost}{$key} = $data->{$key}->[0];
193     }
194   }
195
196   $summary{$thishost}{hostname} = undef;
197   foreach my $hostname (@{$data->{hostname}}) {
198     $summary{$thishost}{hostname} .= sprintf("%s<a href=\"machines.cgi?host=%s\">%s</a>", ($summary{$thishost}{hostname} ? '<br>' : ''), $summary{$thishost}{host}, $hostname);
199   }
200 }
201 $ldap->unbind;
202
203 if ($output{havehostdata}) {
204   $hostdetails = "<h1>Information about $output{hostname}</h1>\n";
205   $hostdetails .= "<ul>\n";
206   foreach $key (@attrorder) {
207     if ($output{$key}) {
208       $hostdetails .= "<li><b>$attrs{$key}:</b>$output{$key}\n";
209     }
210   }
211   $hostdetails .= "</ul>\n";
212 } else {
213   # display summary info
214   $hostdetails = "<h1>Summary</h1>\n";
215   $hostdetails .= "<table id=\"machines\" class=\"tablesorter\" border=\"1\" cellpadding=\"0\" cellspacing=\"1\">\n<thead>\n<tr>";
216   foreach $key (@summaryorder) {
217     if ($sortby ne $key) {
218       $hostdetails .= "<th><a class=\"sort\" href=\"machines.cgi?sortby=$key&sortorder=asc\">$summaryattrs{$key}</a></th>";
219     } else {
220       if ($sortorder ne "dsc") {
221         $hostdetails .= "<th><a class=\"sort\" href=\"machines.cgi?sortby=$key&sortorder=dsc\">$summaryattrs{$key}</a></th>";
222       } else {
223         $hostdetails .= "<th><a class=\"sort\" href=\"machines.cgi?sortby=$key&sortorder=asc\">$summaryattrs{$key}</a></th>";
224       }
225     }
226   }
227   $hostdetails .= "</tr>\n</thead>\n<tbody>\n";
228   
229   my @sorted;
230   if ($sortorder eq "asc") {
231      @sorted = sort {($summary{$a}->{$sortby} cmp $summary{$b}->{$sortby}) || ($summary{$a}->{'host'} cmp $summary{$b}->{'host'})} keys(%summary)
232   } else {
233      @sorted = sort {($summary{$b}->{$sortby} cmp $summary{$a}->{$sortby}) || ($summary{$a}->{'host'} cmp $summary{$b}->{'host'})} keys(%summary)
234   }
235   foreach $host (@sorted) {
236     next if $summary{$host}{status} =~ /^unlisted/;
237     $hostdetails .= "<tr>";
238     foreach $key (@summaryorder) {
239       $hostdetails .= "<td>$summary{$host}{$key}&nbsp;</td>";
240     }
241     $hostdetails .= "</tr>\n";
242   }
243   $hostdetails .= "</tbody>\n</table>\n";
244 }
245
246 # Finally, we can write the output... yuck...
247 open (F, "<$config{hosthtml}") || &Util::HTMLError("Cannot open host template");
248 while (<F>) {
249   s/~hostlist~/$hostlist/;
250   s/~hostdetails~/$hostdetails/;
251   print;
252 }
253 close F;