- updated version to BackupPC-2.1.0beta0
[BackupPC.git] / lib / BackupPC / CGI / Lib.pm
1 #============================================================= -*-perl-*-
2 #
3 # BackupPC::CGI::Lib package
4 #
5 # DESCRIPTION
6 #
7 #   This library defines a BackupPC::Lib class and a variety of utility
8 #   functions used by BackupPC.
9 #
10 # AUTHOR
11 #   Craig Barratt  <cbarratt@users.sourceforge.net>
12 #
13 # COPYRIGHT
14 #   Copyright (C) 2003  Craig Barratt
15 #
16 #   This program is free software; you can redistribute it and/or modify
17 #   it under the terms of the GNU General Public License as published by
18 #   the Free Software Foundation; either version 2 of the License, or
19 #   (at your option) any later version.
20 #
21 #   This program is distributed in the hope that it will be useful,
22 #   but WITHOUT ANY WARRANTY; without even the implied warranty of
23 #   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
24 #   GNU General Public License for more details.
25 #
26 #   You should have received a copy of the GNU General Public License
27 #   along with this program; if not, write to the Free Software
28 #   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
29 #
30 #========================================================================
31 #
32 # Version 2.1.0beta0, released 20 Mar 2004.
33 #
34 # See http://backuppc.sourceforge.net.
35 #
36 #========================================================================
37
38 package BackupPC::CGI::Lib;
39
40 use strict;
41 use BackupPC::Lib;
42
43 require Exporter;
44
45 use vars qw( @ISA @EXPORT @EXPORT_OK %EXPORT_TAGS );
46
47 use vars qw($Cgi %In $MyURL $User %Conf $TopDir $BinDir $bpc);
48 use vars qw(%Status %Info %Jobs @BgQueue @UserQueue @CmdQueue
49             %QueueLen %StatusHost);
50 use vars qw($Hosts $HostsMTime $ConfigMTime $PrivAdmin);
51 use vars qw(%UserEmailInfo $UserEmailInfoMTime %RestoreReq %ArchiveReq);
52 use vars qw($Lang);
53
54 @ISA = qw(Exporter);
55
56 @EXPORT    = qw( );
57
58 @EXPORT_OK = qw(
59                     timeStamp2
60                     HostLink
61                     UserLink
62                     EscHTML
63                     EscURI
64                     ErrorExit
65                     ServerConnect
66                     GetStatusInfo
67                     ReadUserEmailInfo
68                     CheckPermission
69                     GetUserHosts
70                     ConfirmIPAddress
71                     Header
72                     Trailer
73                     NavSectionTitle
74                     NavSectionStart
75                     NavSectionEnd
76                     NavLink
77                     h1
78                     h2
79                     $Cgi %In $MyURL $User %Conf $TopDir $BinDir $bpc
80                     %Status %Info %Jobs @BgQueue @UserQueue @CmdQueue
81                     %QueueLen %StatusHost
82                     $Hosts $HostsMTime $ConfigMTime $PrivAdmin
83                     %UserEmailInfo $UserEmailInfoMTime %RestoreReq %ArchiveReq
84                     $Lang
85              );
86
87 %EXPORT_TAGS = (
88     'all'    => [ @EXPORT_OK ],
89 );
90
91 sub NewRequest
92 {
93     $Cgi = new CGI;
94     %In = $Cgi->Vars;
95
96     if ( !defined($bpc) ) {
97         ErrorExit($Lang->{BackupPC__Lib__new_failed__check_apache_error_log})
98             if ( !($bpc = BackupPC::Lib->new(undef, undef, 1)) );
99         $TopDir = $bpc->TopDir();
100         $BinDir = $bpc->BinDir();
101         %Conf   = $bpc->Conf();
102         $Lang   = $bpc->Lang();
103         $ConfigMTime = $bpc->ConfigMTime();
104     } elsif ( $bpc->ConfigMTime() != $ConfigMTime ) {
105         $bpc->ServerMesg("log Re-read config file because mtime changed");
106         $bpc->ServerMesg("server reload");
107     }
108
109     #
110     # Default REMOTE_USER so in a miminal installation the user
111     # has a sensible default.
112     #
113     $ENV{REMOTE_USER} = $Conf{BackupPCUser} if ( $ENV{REMOTE_USER} eq "" );
114
115     #
116     # We require that Apache pass in $ENV{SCRIPT_NAME} and $ENV{REMOTE_USER}.
117     # The latter requires .ht_access style authentication.  Replace this
118     # code if you are using some other type of authentication, and have
119     # a different way of getting the user name.
120     #
121     $MyURL  = $ENV{SCRIPT_NAME};
122     $User   = $ENV{REMOTE_USER};
123
124     #
125     # Clean up %ENV for taint checking
126     #
127     delete @ENV{qw(IFS CDPATH ENV BASH_ENV)};
128     $ENV{PATH} = $Conf{MyPath};
129
130     #
131     # Verify we are running as the correct user
132     #
133     if ( $Conf{BackupPCUserVerify}
134             && $> != (my $uid = (getpwnam($Conf{BackupPCUser}))[2]) ) {
135         ErrorExit(eval("qq{$Lang->{Wrong_user__my_userid_is___}}"), <<EOF);
136 This script needs to run as the user specified in \$Conf{BackupPCUser},
137 which is set to $Conf{BackupPCUser}.
138 <p>
139 This is an installation problem.  If you are using mod_perl then
140 it appears that Apache is not running as user $Conf{BackupPCUser}.
141 If you are not using mod_perl, then most like setuid is not working
142 properly on BackupPC_Admin.  Check the permissions on
143 $Conf{CgiDir}/BackupPC_Admin and look at the documentation.
144 EOF
145     }
146
147     if ( !defined($Hosts) || $bpc->HostsMTime() != $HostsMTime ) {
148         $HostsMTime = $bpc->HostsMTime();
149         $Hosts = $bpc->HostInfoRead();
150
151         # turn moreUsers list into a hash for quick lookups
152         foreach my $host (keys %$Hosts) {
153            $Hosts->{$host}{moreUsers} =
154                {map {$_, 1} split(",", $Hosts->{$host}{moreUsers}) }
155         }
156     }
157 }
158
159 sub timeStamp2
160 {
161     my($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst)
162               = localtime($_[0] == 0 ? time : $_[0] );
163     $mon++;
164     if ( $Conf{CgiDateFormatMMDD} ) {
165         return sprintf("$mon/$mday %02d:%02d", $hour, $min);
166     } else {
167         return sprintf("$mday/$mon %02d:%02d", $hour, $min);
168     }
169 }
170
171 sub HostLink
172 {
173     my($host) = @_;
174     my($s);
175     if ( defined($Hosts->{$host}) || defined($Status{$host}) ) {
176         $s = "<a href=\"$MyURL?host=${EscURI($host)}\">$host</a>";
177     } else {
178         $s = $host;
179     }
180     return \$s;
181 }
182
183 sub UserLink
184 {
185     my($user) = @_;
186     my($s);
187
188     return \$user if ( $user eq ""
189                     || $Conf{CgiUserUrlCreate} eq "" );
190     if ( $Conf{CgiUserHomePageCheck} eq ""
191             || -f sprintf($Conf{CgiUserHomePageCheck}, $user, $user, $user) ) {
192         $s = "<a href=\""
193              . sprintf($Conf{CgiUserUrlCreate}, $user, $user, $user)
194              . "\">$user</a>";
195     } else {
196         $s = $user;
197     }
198     return \$s;
199 }
200
201 sub EscHTML
202 {
203     my($s) = @_;
204     $s =~ s/&/&amp;/g;
205     $s =~ s/\"/&quot;/g;
206     $s =~ s/>/&gt;/g;
207     $s =~ s/</&lt;/g;
208     $s =~ s{([^[:print:]])}{sprintf("&\#x%02X;", ord($1));}eg;
209     return \$s;
210 }
211
212 sub EscURI
213 {
214     my($s) = @_;
215     $s =~ s{([^\w.\/-])}{sprintf("%%%02X", ord($1));}eg;
216     return \$s;
217 }
218
219 sub ErrorExit
220 {
221     my(@mesg) = @_;
222     my($head) = shift(@mesg);
223     my($mesg) = join("</p>\n<p>", @mesg);
224
225     if ( !defined($ENV{REMOTE_USER}) ) {
226         $mesg .= <<EOF;
227 <p>
228 Note: \$ENV{REMOTE_USER} is not set, which could mean there is an
229 installation problem.  BackupPC_Admin expects Apache to authenticate
230 the user and pass their user name into this script as the REMOTE_USER
231 environment variable.  See the documentation.
232 EOF
233     }
234
235     $bpc->ServerMesg("log User $User (host=$In{host}) got CGI error: $head")
236                             if ( defined($bpc) );
237     if ( !defined($Lang->{Error}) ) {
238         $mesg = <<EOF if ( !defined($mesg) );
239 There is some problem with the BackupPC installation.
240 Please check the permissions on BackupPC_Admin.
241 EOF
242         my $content = <<EOF;
243 ${h1("Error: Unable to read config.pl or language strings!!")}
244 <p>$mesg</p>
245 EOF
246         Header("BackupPC: Error", $content);
247         Trailer();
248     } else {
249         my $content = eval("qq{$Lang->{Error____head}}");
250         Header(eval("qq{$Lang->{Error}}"), $content);
251         Trailer();
252     }
253     exit(1);
254 }
255
256 sub ServerConnect
257 {
258     #
259     # Verify that the server connection is ok
260     #
261     return if ( $bpc->ServerOK() );
262     $bpc->ServerDisconnect();
263     if ( my $err = $bpc->ServerConnect($Conf{ServerHost}, $Conf{ServerPort}) ) {
264         if ( CheckPermission() 
265           && -f $Conf{ServerInitdPath}
266           && $Conf{ServerInitdStartCmd} ne "" ) {
267             my $content = eval("qq{$Lang->{Admin_Start_Server}}");
268             Header(eval("qq{$Lang->{Unable_to_connect_to_BackupPC_server}}"), $content);
269             Trailer();
270             exit(1);
271         } else {
272             ErrorExit(eval("qq{$Lang->{Unable_to_connect_to_BackupPC_server}}"));
273         }
274     }
275 }
276
277 sub GetStatusInfo
278 {
279     my($status) = @_;
280     ServerConnect();
281     my $reply = $bpc->ServerMesg("status $status");
282     $reply = $1 if ( $reply =~ /(.*)/s );
283     eval($reply);
284     # ignore status related to admin and trashClean jobs
285     if ( $status =~ /\bhosts\b/ ) {
286         delete($Status{$bpc->adminJob});
287         delete($Status{$bpc->trashJob});
288     }
289 }
290
291 sub ReadUserEmailInfo
292 {
293     if ( (stat("$TopDir/log/UserEmailInfo.pl"))[9] != $UserEmailInfoMTime ) {
294         do "$TopDir/log/UserEmailInfo.pl";
295         $UserEmailInfoMTime = (stat("$TopDir/log/UserEmailInfo.pl"))[9];
296     }
297 }
298
299 #
300 # Check if the user is privileged.  A privileged user can access
301 # any information (backup files, logs, status pages etc).
302 #
303 # A user is privileged if they belong to the group
304 # $Conf{CgiAdminUserGroup}, or they are in $Conf{CgiAdminUsers}
305 # or they are the user assigned to a host in the host file.
306 #
307 sub CheckPermission
308 {
309     my($host) = @_;
310     my $Privileged = 0;
311
312     return 0 if ( $User eq "" && $Conf{CgiAdminUsers} ne "*"
313                || $host ne "" && !defined($Hosts->{$host}) );
314     if ( $Conf{CgiAdminUserGroup} ne "" ) {
315         my($n,$p,$gid,$mem) = getgrnam($Conf{CgiAdminUserGroup});
316         $Privileged ||= ($mem =~ /\b$User\b/);
317     }
318     if ( $Conf{CgiAdminUsers} ne "" ) {
319         $Privileged ||= ($Conf{CgiAdminUsers} =~ /\b$User\b/);
320         $Privileged ||= $Conf{CgiAdminUsers} eq "*";
321     }
322     $PrivAdmin = $Privileged;
323     $Privileged ||= $User eq $Hosts->{$host}{user};
324     $Privileged ||= defined($Hosts->{$host}{moreUsers}{$User});
325
326     return $Privileged;
327 }
328
329 #
330 # Returns the list of hosts that should appear in the navigation bar
331 # for this user.  If $getAll is set, the admin gets all the hosts.
332 # Otherwise, regular users get hosts for which they are the user or
333 # are listed in the moreUsers column in the hosts file.
334 #
335 sub GetUserHosts
336 {
337     my($host, $getAll) = @_;
338     my @hosts;
339
340     if ( $getAll && CheckPermission() ) {
341         @hosts = sort keys %$Hosts;
342     } else {
343         @hosts = sort grep { $Hosts->{$_}{user} eq $User ||
344                        defined($Hosts->{$_}{moreUsers}{$User}) } keys(%$Hosts);
345     }
346     return @hosts;
347 }
348
349 #
350 # Given a host name tries to find the IP address.  For non-dhcp hosts
351 # we just return the host name.  For dhcp hosts we check the address
352 # the user is using ($ENV{REMOTE_ADDR}) and also the last-known IP
353 # address for $host.  (Later we should replace this with a broadcast
354 # nmblookup.)
355 #
356 sub ConfirmIPAddress
357 {
358     my($host) = @_;
359     my $ipAddr = $host;
360
361     if ( defined($Hosts->{$host}) && $Hosts->{$host}{dhcp}
362                && $ENV{REMOTE_ADDR} =~ /^(\d+[\.\d]*)$/ ) {
363         $ipAddr = $1;
364         my($netBiosHost, $netBiosUser) = $bpc->NetBiosInfoGet($ipAddr);
365         if ( $netBiosHost ne $host ) {
366             my($tryIP);
367             GetStatusInfo("host(${EscURI($host)})");
368             if ( defined($StatusHost{dhcpHostIP})
369                         && $StatusHost{dhcpHostIP} ne $ipAddr ) {
370                 $tryIP = eval("qq{$Lang->{tryIP}}");
371                 ($netBiosHost, $netBiosUser)
372                         = $bpc->NetBiosInfoGet($StatusHost{dhcpHostIP});
373             }
374             if ( $netBiosHost ne $host ) {
375                 ErrorExit(eval("qq{$Lang->{Can_t_find_IP_address_for}}"),
376                           eval("qq{$Lang->{host_is_a_DHCP_host}}"));
377             }
378             $ipAddr = $StatusHost{dhcpHostIP};
379         }
380     }
381     return $ipAddr;
382 }
383
384 ###########################################################################
385 # HTML layout subroutines
386 ###########################################################################
387
388 sub Header
389 {
390     my($title, $content, $noBrowse, $contentSub, $contentPost) = @_;
391     my @adminLinks = (
392         { link => "",                         name => $Lang->{Status},
393                                               priv => 1},
394         { link => "?action=adminOpts",        name => $Lang->{Admin_Options} },
395         { link => "?action=summary",          name => $Lang->{PC_Summary},
396                                               priv => 1},
397         { link => "?action=view&type=LOG",    name => $Lang->{LOG_file} },
398         { link => "?action=LOGlist",          name => $Lang->{Old_LOGs} },
399         { link => "?action=emailSummary",     name => $Lang->{Email_summary} },
400         { link => "?action=view&type=config", name => $Lang->{Config_file} },
401         { link => "?action=view&type=hosts",  name => $Lang->{Hosts_file} },
402         { link => "?action=queue",            name => $Lang->{Current_queues} },
403         { link => "?action=view&type=docs",   name => $Lang->{Documentation},
404                                               priv => 1},
405         { link => "http://backuppc.sourceforge.net/faq", name => "FAQ",
406                                               priv => 1},
407         { link => "http://backuppc.sourceforge.net", name => "SourceForge",
408                                               priv => 1},
409     );
410     my $host = $In{host};
411
412     print $Cgi->header();
413     print <<EOF;
414 <!doctype html public "-//W3C//DTD HTML 4.01 Transitional//EN">
415 <html><head>
416 <title>$title</title>
417 $Conf{CSSstylesheet}
418 $Conf{CgiHeaders}
419 </head><body onLoad="document.getElementById('NavMenu').style.height=document.body.scrollHeight">
420 <img src="$Conf{CgiImageDirURL}/logo.gif" hspace="5" vspace="7"><br>
421 EOF
422
423     if ( defined($Hosts) && defined($host) && defined($Hosts->{$host}) ) {
424         print "<div class=\"NavMenu\">";
425         NavSectionTitle("${EscURI($host)}");
426         print <<EOF;
427 </div>
428 <div class="NavMenu">
429 EOF
430         NavLink("?host=${EscURI($host)}",
431                 "$host $Lang->{Home}", " class=\"navbar\"");
432         NavLink("?action=browse&host=${EscURI($host)}",
433                 $Lang->{Browse}, " class=\"navbar\"") if ( !$noBrowse );
434         NavLink("?action=view&type=LOG&host=${EscURI($host)}",
435                 $Lang->{LOG_file}, " class=\"navbar\"");
436         NavLink("?action=LOGlist&host=${EscURI($host)}",
437                 $Lang->{LOG_files}, " class=\"navbar\"");
438         if ( -f "$TopDir/pc/$host/SmbLOG.bad"
439                     || -f "$TopDir/pc/$host/SmbLOG.bad.z"
440                     || -f "$TopDir/pc/$host/XferLOG.bad"
441                     || -f "$TopDir/pc/$host/XferLOG.bad.z" ) {
442            NavLink("?action=view&type=XferLOGbad&host=${EscURI($host)}",
443                     $Lang->{Last_bad_XferLOG}, " class=\"navbar\"");
444            NavLink("?action=view&type=XferErrbad&host=${EscURI($host)}",
445                     $Lang->{Last_bad_XferLOG_errors_only},
446                     " class=\"navbar\"");
447         }
448         if ( -f "$TopDir/pc/$host/config.pl" ) {
449             NavLink("?action=view&type=config&host=${EscURI($host)}",
450                     $Lang->{Config_file}, " class=\"navbar\"");
451         }
452         print "</div>\n";
453     }
454     print("<div id=\"Content\">\n$content\n");
455     if ( defined($contentSub) && ref($contentSub) eq "CODE" ) {
456         while ( (my $s = &$contentSub()) ne "" ) {
457             print($s);
458         }
459     }
460     print($contentPost) if ( defined($contentPost) );
461     print <<EOF;
462 <br><br><br>
463 </div>
464 <div class="NavMenu" id="NavMenu" style="height:100%">
465 EOF
466     my $hostSelectbox = "<option value=\"#\">$Lang->{Select_a_host}</option>";
467     my @hosts = GetUserHosts($In{host}, $Conf{CgiNavBarAdminAllHosts});
468     if ( defined($Hosts) && %$Hosts > 0 && @hosts ) {
469         NavSectionTitle($Lang->{Hosts});
470         foreach my $host ( @hosts ) {
471             NavLink("?host=${EscURI($host)}", $host)
472                     if ( @hosts < $Conf{CgiNavBarAdminAllHosts} );
473             my $sel = " selected" if ( $host eq $In{host} );
474             $hostSelectbox .= "<option value=\"?host=${EscURI($host)}\"$sel>"
475                             . "$host</option>";
476         }
477     }
478     if ( @hosts >= $Conf{CgiNavBarAdminAllHosts} ) {
479         print <<EOF;
480 <br>
481 <select onChange="document.location=this.value">
482 $hostSelectbox
483 </select>
484 <br><br>
485 EOF
486     }
487     print <<EOF;
488 <form action="$MyURL" method="get">
489     <input type="text" name="host" size="14" maxlength="64">
490     <input type="hidden" name="action" value="hostInfo"><input type="submit" value="$Lang->{Go}" name="ignore">
491     </form>
492 EOF
493     NavSectionTitle($Lang->{NavSectionTitle_});
494     foreach my $l ( @adminLinks ) {
495         if ( $PrivAdmin || $l->{priv} ) {
496             NavLink($l->{link}, $l->{name});
497     }
498 }
499
500 print <<EOF;
501 <br><br><br>
502 </div>
503 EOF
504 }
505
506 sub Trailer
507 {
508     print <<EOF;
509 </body></html>
510 EOF
511 }
512
513
514 sub NavSectionTitle
515 {
516     my($head) = @_;
517     print <<EOF;
518 <div class="NavTitle">$head</div>
519 EOF
520 }
521
522 sub NavSectionStart
523 {
524 }
525
526 sub NavSectionEnd
527 {
528 }
529
530 sub NavLink
531 {
532     my($link, $text) = @_;
533     if ( defined($link) ) {
534         my($class);
535         $class = " class=\"NavCurrent\""
536                 if ( length($link) && $ENV{REQUEST_URI} =~ /\Q$link\E$/
537                     || $link eq "" && $ENV{REQUEST_URI} !~ /\?/ );
538         $link = "$MyURL$link" if ( $link eq "" || $link =~ /^\?/ );
539         print <<EOF;
540 <a href="$link"$class>$text</a>
541 EOF
542     } else {
543         print <<EOF;
544 $text<br>
545 EOF
546     }
547 }
548
549 sub h1
550 {
551     my($str) = @_;
552     return \<<EOF;
553 <div class="h1">$str</div>
554 EOF
555 }
556
557 sub h2
558 {
559     my($str) = @_;
560     return \<<EOF;
561 <div class="h2">$str</div>
562 EOF
563 }