aa48bf90ea528b772d5bb262bb45ce4ec27abf91
[BackupPC.git] / lib / BackupPC / CGI / EditConfig.pm
1 #============================================================= -*-perl-*-
2 #
3 # BackupPC::CGI::EditConfig package
4 #
5 # DESCRIPTION
6 #
7 #   This module implements the EditConfig action for the CGI interface.
8 #
9 # AUTHOR
10 #   Craig Barratt  <cbarratt@users.sourceforge.net>
11 #
12 # COPYRIGHT
13 #   Copyright (C) 2005-2007  Craig Barratt
14 #
15 #   This program is free software; you can redistribute it and/or modify
16 #   it under the terms of the GNU General Public License as published by
17 #   the Free Software Foundation; either version 2 of the License, or
18 #   (at your option) any later version.
19 #
20 #   This program is distributed in the hope that it will be useful,
21 #   but WITHOUT ANY WARRANTY; without even the implied warranty of
22 #   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
23 #   GNU General Public License for more details.
24 #
25 #   You should have received a copy of the GNU General Public License
26 #   along with this program; if not, write to the Free Software
27 #   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
28 #
29 #========================================================================
30 #
31 # Version 3.1.0beta0, released 3 Sep 2007.
32 #
33 # See http://backuppc.sourceforge.net.
34 #
35 #========================================================================
36
37 package BackupPC::CGI::EditConfig;
38
39 use strict;
40 use BackupPC::CGI::Lib qw(:all);
41 use BackupPC::Config::Meta qw(:all);
42 use BackupPC::Storage;
43 use Data::Dumper;
44 use Encode;
45
46 our %ConfigMenu = (
47     server => {
48         text  => "CfgEdit_Title_Server",
49         param => [
50             {text => "CfgEdit_Title_General_Parameters"},
51             {name => "ServerHost"},
52             {name => "BackupPCUser"},
53             {name => "BackupPCUserVerify"},
54             {name => "MaxOldLogFiles"},
55             {name => "TrashCleanSleepSec"},
56
57             {text => "CfgEdit_Title_Wakeup_Schedule"},
58             {name => "WakeupSchedule"},
59
60             {text => "CfgEdit_Title_Concurrent_Jobs"},
61             {name => "MaxBackups"},
62             {name => "MaxUserBackups"},
63             {name => "MaxPendingCmds"},
64             {name => "MaxBackupPCNightlyJobs"},
65             {name => "BackupPCNightlyPeriod"},
66
67             {text => "CfgEdit_Title_Pool_Filesystem_Limits"},
68             {name => "DfCmd"},
69             {name => "DfMaxUsagePct"},
70             {name => "HardLinkMax"},
71
72             {text => "CfgEdit_Title_Other_Parameters"},
73             {name => "UmaskMode"},
74             {name => "MyPath"},
75             {name => "DHCPAddressRanges"},
76             {name => "PerlModuleLoad"},
77             {name => "ServerInitdPath"},
78             {name => "ServerInitdStartCmd"},
79
80             {text => "CfgEdit_Title_Remote_Apache_Settings"},
81             {name => "ServerPort"},
82             {name => "ServerMesgSecret"},
83
84             {text => "CfgEdit_Title_Program_Paths"},
85             {name => "SshPath"},
86             {name => "NmbLookupPath"},
87             {name => "PingPath"},
88             {name => "DfPath"},
89             {name => "SplitPath"},
90             {name => "ParPath"},
91             {name => "CatPath"},
92             {name => "GzipPath"},
93             {name => "Bzip2Path"},
94
95             {text => "CfgEdit_Title_Install_Paths"},
96             #
97             # Can only edit TopDir and LogDir if we are in FHS mode.
98             # Otherwise they are hardcoded in lib/BackupPC/Lib.pm.
99             #
100             {name => "TopDir",
101                     visible => sub { return $_[1]->useFHS(); } },
102             {name => "LogDir",
103                     visible => sub { return $_[1]->useFHS(); } },
104             {name => "CgiDir"},
105             #
106             # Cannot edit ConfDir or InstallDir, since the real value is hardcoded in
107             # lib/BackupPC/Lib.pm.
108             # {name => "ConfDir"},
109             # {name => "InstallDir"},
110             #
111         ],
112     },
113     email => {
114         text  => "CfgEdit_Title_Email",
115         param => [
116             {text => "CfgEdit_Title_Email_settings"},
117             {name => "SendmailPath"},
118             {name => "EMailNotifyMinDays"},
119             {name => "EMailFromUserName"},
120             {name => "EMailAdminUserName"},
121             {name => "EMailUserDestDomain"},
122
123             {text => "CfgEdit_Title_Email_User_Messages"},
124             {name => "EMailNoBackupEverSubj"},
125             {name => "EMailNoBackupEverMesg"},
126             {name => "EMailNotifyOldBackupDays"},
127             {name => "EMailNoBackupRecentSubj"},
128             {name => "EMailNoBackupRecentMesg"},
129             {name => "EMailNotifyOldOutlookDays"},
130             {name => "EMailOutlookBackupSubj"},
131             {name => "EMailOutlookBackupMesg"},
132             {name => "EMailHeaders"},
133         ],
134     },
135     cgi => {
136         text => "CfgEdit_Title_CGI",
137         param => [
138             {text => "CfgEdit_Title_Admin_Privileges"},
139             {name => "CgiAdminUserGroup"},
140             {name => "CgiAdminUsers"},
141
142             {text => "CfgEdit_Title_Page_Rendering"},
143             {name => "Language"},
144             {name => "CgiNavBarAdminAllHosts"},
145             {name => "CgiSearchBoxEnable"},
146             {name => "CgiNavBarLinks"},
147             {name => "CgiStatusHilightColor"},
148             {name => "CgiDateFormatMMDD"},
149             {name => "CgiHeaders"},
150             {name => "CgiExt2ContentType"},
151             {name => "CgiCSSFile"},
152
153             {text => "CfgEdit_Title_Paths"},
154             {name => "CgiURL"},
155             {name => "CgiImageDir"},
156             {name => "CgiImageDirURL"},
157
158             {text => "CfgEdit_Title_User_URLs"},
159             {name => "CgiUserHomePageCheck"},
160             {name => "CgiUserUrlCreate"},
161
162             {text => "CfgEdit_Title_User_Config_Editing"},
163             {name => "CgiUserConfigEditEnable"},
164             {name => "CgiUserConfigEdit"},
165         ],
166     },
167     xfer => {
168         text => "CfgEdit_Title_Xfer",
169         param => [
170             {text => "CfgEdit_Title_Xfer_Settings"},
171             {name => "XferMethod", onchangeSubmit => 1},
172             {name => "XferLogLevel"},
173             {name => "ClientCharset"},
174             {name => "ClientCharsetLegacy"},
175
176             {text => "CfgEdit_Title_Smb_Settings",
177                 visible => sub { return $_[0]->{XferMethod} eq "smb"; } },
178             {name => "SmbShareName",
179                 visible => sub { return $_[0]->{XferMethod} eq "smb"; } },
180             {name => "SmbShareUserName",
181                 visible => sub { return $_[0]->{XferMethod} eq "smb"; } },
182             {name => "SmbSharePasswd",
183                 visible => sub { return $_[0]->{XferMethod} eq "smb"; } },
184
185             {text => "CfgEdit_Title_Tar_Settings",
186                 visible => sub { return $_[0]->{XferMethod} eq "tar"; } },
187             {name => "TarShareName",
188                 visible => sub { return $_[0]->{XferMethod} eq "tar"; } },
189
190             {text => "CfgEdit_Title_Rsync_Settings",
191                 visible => sub { return $_[0]->{XferMethod} eq "rsync"; } },
192             {text => "CfgEdit_Title_Rsyncd_Settings",
193                 visible => sub { return $_[0]->{XferMethod} eq "rsyncd"; } },
194             {name => "RsyncShareName",
195                 visible => sub { return $_[0]->{XferMethod} =~ /rsync/; } },
196             {name => "RsyncdUserName",
197                 visible => sub { return $_[0]->{XferMethod} eq "rsyncd"; } },
198             {name => "RsyncdPasswd",
199                 visible => sub { return $_[0]->{XferMethod} eq "rsyncd"; } },
200             {name => "RsyncdAuthRequired",
201                 visible => sub { return $_[0]->{XferMethod} eq "rsyncd"; } },
202             {name => "RsyncCsumCacheVerifyProb",
203                 visible => sub { return $_[0]->{XferMethod} =~ /rsync/; } },
204
205             {text => "CfgEdit_Title_BackupPCd_Settings",
206                 visible => sub { return $_[0]->{XferMethod} eq "backuppcd"; } },
207             {name => "BackupPCdShareName",
208                 visible => sub { return $_[0]->{XferMethod} eq "backuppcd"; } },
209             {name => "BackupPCdPath",
210                 visible => sub { return $_[0]->{XferMethod} eq "backuppcd"; } },
211             {name => "BackupPCdCmd",
212                 visible => sub { return $_[0]->{XferMethod} eq "backuppcd"; } },
213             {name => "BackupPCdRestoreCmd",
214                 visible => sub { return $_[0]->{XferMethod} eq "backuppcd"; } },
215
216             {text => "CfgEdit_Title_Archive_Settings",
217                 visible => sub { return $_[0]->{XferMethod} eq "archive"; } },
218             {name => "ArchiveDest",
219                 visible => sub { return $_[0]->{XferMethod} eq "archive"; } },
220             {name => "ArchiveComp",
221                 visible => sub { return $_[0]->{XferMethod} eq "archive"; } },
222             {name => "ArchivePar",
223                 visible => sub { return $_[0]->{XferMethod} eq "archive"; } },
224             {name => "ArchiveSplit",
225                 visible => sub { return $_[0]->{XferMethod} eq "archive"; } },
226
227             {text => "CfgEdit_Title_Include_Exclude",
228                 visible => sub { return $_[0]->{XferMethod} ne "archive"; } },
229             {name => "BackupFilesOnly",
230                 visible => sub { return $_[0]->{XferMethod} ne "archive"; } },
231             {name => "BackupFilesExclude",
232                 visible => sub { return $_[0]->{XferMethod} ne "archive"; } },
233
234             {text => "CfgEdit_Title_Smb_Paths_Commands",
235                 visible => sub { return $_[0]->{XferMethod} eq "smb"; } },
236             {name => "SmbClientPath",
237                 visible => sub { return $_[0]->{XferMethod} eq "smb"; } },
238             {name => "SmbClientFullCmd",
239                 visible => sub { return $_[0]->{XferMethod} eq "smb"; } },
240             {name => "SmbClientIncrCmd",
241                 visible => sub { return $_[0]->{XferMethod} eq "smb"; } },
242             {name => "SmbClientRestoreCmd",
243                 visible => sub { return $_[0]->{XferMethod} eq "smb"; } },
244
245             {text => "CfgEdit_Title_Tar_Paths_Commands",
246                 visible => sub { return $_[0]->{XferMethod} eq "tar"; } },
247             {name => "TarClientPath",
248                 visible => sub { return $_[0]->{XferMethod} eq "tar"; } },
249             {name => "TarClientCmd",
250                 visible => sub { return $_[0]->{XferMethod} eq "tar"; } },
251             {name => "TarFullArgs",
252                 visible => sub { return $_[0]->{XferMethod} eq "tar"; } },
253             {name => "TarIncrArgs",
254                 visible => sub { return $_[0]->{XferMethod} eq "tar"; } },
255             {name => "TarClientRestoreCmd",
256                 visible => sub { return $_[0]->{XferMethod} eq "tar"; } },
257
258             {text => "CfgEdit_Title_Rsync_Paths_Commands_Args",
259                 visible => sub { return $_[0]->{XferMethod} eq "rsync"; } },
260             {text => "CfgEdit_Title_Rsyncd_Port_Args",
261                 visible => sub { return $_[0]->{XferMethod} eq "rsyncd"; } },
262             {name => "RsyncClientPath",
263                 visible => sub { return $_[0]->{XferMethod} eq "rsync"; } },
264             {name => "RsyncClientCmd",
265                 visible => sub { return $_[0]->{XferMethod} eq "rsync"; } },
266             {name => "RsyncClientRestoreCmd",
267                 visible => sub { return $_[0]->{XferMethod} eq "rsync"; } },
268             {name => "RsyncdClientPort",
269                 visible => sub { return $_[0]->{XferMethod} eq "rsyncd"; } },
270             {name => "RsyncArgs",
271                 visible => sub { return $_[0]->{XferMethod} =~ /rsync/; } },
272             {name => "RsyncRestoreArgs",
273                 visible => sub { return $_[0]->{XferMethod} =~ /rsync/; } },
274
275             {text => "CfgEdit_Title_Archive_Paths_Commands",
276                 visible => sub { return $_[0]->{XferMethod} eq "archive"; } },
277             {name => "ArchiveClientCmd",
278                 visible => sub { return $_[0]->{XferMethod} eq "archive"; } },
279
280         ],
281     },
282     schedule => {
283         text => "CfgEdit_Title_Schedule",
284         param => [
285             {text => "CfgEdit_Title_Full_Backups"},
286             {name => "FullPeriod"},
287             {name => "FullKeepCnt"},
288             {name => "FullKeepCntMin"},
289             {name => "FullAgeMax"},
290
291             {text => "CfgEdit_Title_Incremental_Backups"},
292             {name => "IncrPeriod"},
293             {name => "IncrKeepCnt"},
294             {name => "IncrKeepCntMin"},
295             {name => "IncrAgeMax"},
296             {name => "IncrLevels"},
297             {name => "IncrFill"},
298
299             {text => "CfgEdit_Title_Blackouts"},
300             {name => "BackupsDisable"},
301             {name => "BlackoutBadPingLimit"},
302             {name => "BlackoutGoodCnt"},
303             {name => "BlackoutPeriods"},
304
305             {text => "CfgEdit_Title_Other"},
306             {name => "PartialAgeMax"},
307             {name => "RestoreInfoKeepCnt"},
308             {name => "ArchiveInfoKeepCnt"},
309             {name => "BackupZeroFilesIsFatal"},
310         ],
311     },
312     backup => {
313         text => "CfgEdit_Title_Backup_Settings",
314         param => [
315             {text => "CfgEdit_Title_Client_Lookup"},
316             {name => "ClientNameAlias"},
317             {name => "NmbLookupCmd"},
318             {name => "NmbLookupFindHostCmd"},
319             {name => "FixedIPNetBiosNameCheck"},
320             {name => "PingCmd"},
321             {name => "PingMaxMsec"},
322             
323             {text => "CfgEdit_Title_Other"},
324             {name => "ClientTimeout"},
325             {name => "MaxOldPerPCLogFiles"},
326             {name => "CompressLevel"},
327
328             {text => "CfgEdit_Title_User_Commands"},
329             {name => "DumpPreUserCmd"},
330             {name => "DumpPostUserCmd"},
331             {name => "DumpPreShareCmd"},
332             {name => "DumpPostShareCmd"},
333             {name => "RestorePreUserCmd"},
334             {name => "RestorePostUserCmd"},
335             {name => "ArchivePreUserCmd"},
336             {name => "ArchivePostUserCmd"},
337             {name => "UserCmdCheckStatus"},
338         ],
339     },
340     hosts => {
341         text => "CfgEdit_Title_Hosts",
342         param => [
343             {text    => "CfgEdit_Title_Hosts"},
344             {name    => "Hosts",
345              comment => "CfgEdit_Hosts_Comment"},
346         ],
347     },
348 );
349
350 sub action
351 {
352     my $pc_dir = "$TopDir/pc";
353     my($content, $contentHidden, $newConf, $override, $mainConf, $hostConf);
354     my $errors = {};
355
356     my $host = $In{host};
357     my $menu = $In{menu} || "server";
358     my $hosts_path = $Hosts;
359     my $config_path = $host eq "" ? "$TopDir/conf/config.pl"
360                                   : "$TopDir/pc/$host/config.pl";
361
362     my $Privileged = CheckPermission($host)
363                        && ($PrivAdmin || $Conf{CgiUserConfigEditEnable});
364     my $userHost = 1 if ( defined($host) );
365     my $debugText;
366
367     if ( !$Privileged ) {
368         ErrorExit(eval("qq{$Lang->{Only_privileged_users_can_edit_config_files}}"));
369     }
370
371     if ( defined($In{menu}) || $In{saveAction} eq "Save" ) {
372         $errors = errorCheck();
373         if ( %$errors ) {
374             #
375             # If there are errors, then go back to the same menu
376             #
377             $In{saveAction} = "";
378             #$In{newMenu} = "";
379         }
380         if ( (my $var = $In{overrideUncheck}) ne "" ) {
381             #
382             # a compound variable was unchecked; delete extra
383             # variables to make the shape the same.
384             #
385             #print STDERR Dumper(\%In);
386             foreach my $v ( keys(%In) ) {
387                 next if ( $v !~ /^v_zZ_(\Q$var\E(_zZ_.*|$))/ );
388                 delete($In{$v}) if ( !defined($In{"orig_zZ_$1"}) );
389             }
390             delete($In{"vflds.$var"});
391         }
392
393         ($newConf, $override) = inputParse($bpc, $userHost);
394         $override = undef if ( $host eq "" );
395
396     } else {
397         #
398         # First time: pick up the current config settings
399         #
400         $mainConf = $bpc->ConfigDataRead();
401         if ( $host ne "" ) {
402             $hostConf = $bpc->ConfigDataRead($host);
403             $override = {};
404             foreach my $param ( keys(%$hostConf) ) {
405                 $override->{$param} = 1;
406             }
407         } else {
408             my $hostInfo = $bpc->HostInfoRead();
409             $hostConf = {};
410             $mainConf->{Hosts} = [map($hostInfo->{$_}, sort(keys(%$hostInfo)))];
411         }
412         $newConf = { %$mainConf, %$hostConf };
413     }
414
415     if ( $In{saveAction} ne "Save" && $In{newMenu} ne ""
416                     && defined($ConfigMenu{$In{newMenu}}) ) {
417         $menu = $In{newMenu};
418     }
419
420     my %menuDisable;
421     if ( $userHost ) {
422         #
423         # For a non-admin user editing the host config, we need to
424         # figure out which subsets of the menu tree will be visible,
425         # based on what is enabled.  Admin users can edit all the
426         # available per-host settings.
427         #
428         foreach my $m ( keys(%ConfigMenu) ) {
429             my $enabled = 0;
430             my $text = -1;
431             my $n = 0;
432             my @mask = ();
433
434             foreach my $paramInfo ( @{$ConfigMenu{$m}{param}} ) {
435                 my $param = $paramInfo->{name};
436                 if ( defined($paramInfo->{text}) ) {
437                     $text = $n;
438                     $mask[$text] = 1;
439                 } else {
440                     if ( $bpc->{Conf}{CgiUserConfigEdit}{$param}
441                           || (defined($bpc->{Conf}{CgiUserConfigEdit}{$param})
442                                 && $PrivAdmin) ) {
443                         $mask[$text] = 0 if ( $text >= 0 );
444                         $mask[$n] = 0;
445                         $enabled ||= 1;
446                     } else {
447                         $mask[$n] = 1;
448                     }
449                 }
450                 $n++;
451             }
452             $menuDisable{$m}{mask} = \@mask;
453             $menuDisable{$m}{top}  = !$enabled;
454         }
455         if ( $menuDisable{$menu}{top} ) {
456             #
457             # Find an enabled menu if the current menu is empty
458             #
459             foreach my $m ( sort(keys(%menuDisable)) ) {
460                 if ( !$menuDisable{$m}{top} ) {
461                     $menu = $m;
462                     last;
463                 }
464             }
465         }
466     }
467
468     my $groupText;
469     foreach my $m ( keys(%ConfigMenu) ) {
470         next if ( $menuDisable{$m}{top} );
471         my $text = eval("qq($Lang->{$ConfigMenu{$m}{text}})");
472         if ( $m eq $menu ) {
473             $groupText .= <<EOF;
474 <td class="editTabSel"><a href="javascript:menuSubmit('$m')"><b>$text</b></a></td>
475 EOF
476         } else {
477             $groupText .= <<EOF;
478 <td class="editTabNoSel"><a href="javascript:menuSubmit('$m')">$text</a></td>
479 EOF
480         }
481     }
482
483     if ( $host eq "" ) {
484         $content .= eval("qq($Lang->{CfgEdit_Header_Main})");
485     } else {
486         $content .= eval("qq($Lang->{CfgEdit_Header_Host})");
487     }
488
489     my $saveStyle = "";
490     my $saveColor = "#ff0000";
491     
492     if ( $In{modified} && $In{saveAction} ne "Save" && !%$errors ) {
493         $saveStyle = "style=\"color:$saveColor\"";
494     } else {
495         $In{modified} = 0;
496     }
497
498     #
499     # Add action and host to the URL so the nav bar link is
500     # highlighted
501     #
502     my $url = "$MyURL?action=editConfig";
503     $url .= "&host=$host" if ( $host ne "" );
504     $content .= <<EOF;
505 <table border="0" cellpadding="2">
506 <tr>$groupText</tr>
507 <tr>
508 <form method="post" name="editForm" action="$url">
509 <input type="hidden" name="host" value="$host">
510 <input type="hidden" name="menu" value="$menu">
511 <input type="hidden" name="newMenu" value="">
512 <input type="hidden" name="modified" value="$In{modified}">
513 <input type="hidden" name="deleteVar" value="">
514 <input type="hidden" name="insertVar" value="">
515 <input type="hidden" name="overrideUncheck" value="">
516 <input type="hidden" name="addVar" value="">
517 <input type="hidden" name="action" value="editConfig">
518 <input type="hidden" name="saveAction" value="">
519 <input type="button" class="editSaveButton" name="editAction"
520     value="${EscHTML($Lang->{CfgEdit_Button_Save})}" $saveStyle
521     onClick="saveSubmit();">
522 <p>
523
524 <script language="javascript" type="text/javascript">
525 <!--
526
527     function saveSubmit()
528     {
529         if ( document.editForm.modified.value != 0 ) {
530             document.editForm.saveAction.value = 'Save';
531             document.editForm.submit();
532         }
533         return false;
534     }
535
536     function deleteSubmit(varName)
537     {
538         document.editForm.deleteVar.value = varName;
539         document.editForm.modified.value = 1;
540         document.editForm.submit();
541         return;
542     }
543
544     function insertSubmit(varName)
545     {
546         document.editForm.insertVar.value = varName;
547         document.editForm.modified.value = 1;
548         document.editForm.submit();
549         return;
550     }
551
552     function addSubmit(varName, checkKey)
553     {
554         if ( checkKey
555             && eval("document.editForm.addVarKey_" + varName + ".value") == "" ) {
556             alert("New key must be non-empty");
557             return;
558         }
559         document.editForm.addVar.value = varName;
560         document.editForm.modified.value = 1;
561         document.editForm.submit();
562         return;
563     }
564
565     function menuSubmit(menuName)
566     {
567         document.editForm.newMenu.value = menuName;
568         document.editForm.submit();
569     }
570
571     function varChange(varName)
572     {
573         document.editForm.modified.value = 1;
574         document.editForm.editAction.style.color = '$saveColor';
575     }
576
577     function checkboxChange(varName)
578     {
579         document.editForm.modified.value = 1;
580         document.editForm.editAction.style.color = '$saveColor';
581         // Do nothing if the checkbox is now set
582         if ( eval("document.editForm.override_" + varName + ".checked") ) {
583             return false;
584         }
585         var allVars = {};
586         var varRE  = new RegExp("^v_zZ_(" + varName + ".*)");
587         var origRE = new RegExp("^orig_zZ_(" + varName + ".*)");
588         for ( var i = 0 ; i < document.editForm.elements.length ; i++ ) {
589             var e = document.editForm.elements[i];
590             var re;
591             if ( (re = varRE.exec(e.name)) != null ) {
592                 if ( allVars[re[1]] == null ) {
593                     allVars[re[1]] = 0;
594                 }
595                 allVars[re[1]]++;
596                 //debugMsg("found v_zZ_ match with " + re[1]);
597                 //debugMsg("allVars[" + re[1] + "] = " + allVars[re[1]]);
598             } else if ( (re = origRE.exec(e.name)) != null ) {
599                 if ( allVars[re[1]] == null ) {
600                     allVars[re[1]] = 0;
601                 }
602                 allVars[re[1]]--;
603                 //debugMsg("allVars[" + re[1] + "] = " + allVars[re[1]]);
604             }
605         }
606         var sameShape = 1;
607         for ( v in allVars ) {
608             if ( allVars[v] != 0 ) {
609                 //debugMsg("Not the same shape because of " + v);
610                 sameShape = 0;
611             } else {
612                 // copy the original variable values
613                 //debugMsg("setting " + v);
614                 eval("document.editForm.v_zZ_" + v + ".value = document.editForm.orig_zZ_" + v + ".value");
615             }
616         }
617         if ( sameShape ) {
618             return true;
619         } else {
620             // need to rebuild the form since the compound variable
621             // has changed shape
622             document.editForm.overrideUncheck.value = varName;
623             document.editForm.submit();
624             return false;
625         }
626     }
627
628     function checkboxSet(varName)
629     {
630         document.editForm.modified.value = 1;
631         document.editForm.editAction.style.color = '$saveColor';
632         eval("document.editForm.override_" + varName + ".checked = 1;");
633         return false;
634     }
635
636     var debugCounter = 0;
637     function debugMsg(msg)
638     {
639         debugCounter++;
640         var t = document.createTextNode(debugCounter + ": " + msg);
641         var br = document.createElement("br");
642         var debug = document.getElementById("debug");
643         debug.appendChild(t);
644         debug.appendChild(br);
645     }
646
647     function displayHelp(varName)
648     {
649         var help = document.getElementById("id_" + varName);
650         help.style.display = help.style.display == "block" ? "none" : "block";
651     }
652
653 //-->
654 </script>
655
656 <span id="debug">$debugText</span>
657
658 EOF
659
660     $content .= <<EOF;
661 <table border="1" cellspacing="0">
662 EOF
663
664     my $doneParam = {};
665     my $tblContent;
666
667     #
668     # There is a special case of the user deleting just the field
669     # that has the error(s).  So if the delete variable is a match
670     # or parent to all the errors then ignore the errors.
671     #
672     if ( $In{deleteVar} ne "" && %$errors > 0 ) {
673         my $matchAll = 1;
674         foreach my $v ( keys(%$errors) ) {
675             if ( $v ne $In{deleteVar} && $v !~ /^\Q$In{deleteVar}_zZ_/ ) {
676                 $matchAll = 0;
677                 last;
678             }
679         }
680         $errors = {} if ( $matchAll );
681     }
682
683     my $isError = %$errors;
684
685     if ( !$isError && $In{saveAction} eq "Save" ) {
686         my($mesg, $err);
687         if ( $host ne "" ) {
688             $hostConf = $bpc->ConfigDataRead($host) if ( !defined($hostConf) );
689             my %hostConf2 = %$hostConf;
690             foreach my $param ( keys(%$newConf) ) {
691                 if ( $override->{$param} ) {
692                     $hostConf->{$param} = $newConf->{$param}
693                 } else {
694                     delete($hostConf->{$param});
695                 }
696             }
697             $mesg = configDiffMesg($host, \%hostConf2, $hostConf);
698             $err .= $bpc->ConfigDataWrite($host, $hostConf);
699         } else {
700             $mainConf = $bpc->ConfigDataRead() if ( !defined($mainConf) );
701
702             my $hostsSave = [];
703             my($hostsNew, $allHosts, $copyConf);
704             foreach my $entry ( @{$newConf->{Hosts}} ) {
705                 next if ( $entry->{host} eq "" );
706                 $allHosts->{$entry->{host}} = 1;
707                 $allHosts->{$1} = 1 if ( $entry->{host} =~ /(.+?)\s*=/ );
708             }
709             foreach my $entry ( @{$newConf->{Hosts}} ) {
710                 next if ( $entry->{host} eq ""
711                            || defined($hostsNew->{$entry->{host}}) );
712                 if ( $entry->{host} =~ /(.+?)\s*=\s*(.+)/ ) {
713                     if ( defined($allHosts->{$2}) ) {
714                         $entry->{host} = $1;
715                         $copyConf->{$1} = $2;
716                     } else {
717                         my $fullHost = $entry->{host};
718                         my $copyHost = $2;
719                         $err .= eval("qq($Lang->{CfgEdit_Error_Copy_host_does_not_exist})");
720                     }
721                 }
722                 push(@$hostsSave, $entry);
723                 $hostsNew->{$entry->{host}} = $entry;
724             }
725             ($mesg, my $hostChange) = hostsDiffMesg($hostsNew);
726             $bpc->HostInfoWrite($hostsNew) if ( $hostChange );
727             foreach my $host ( keys(%$copyConf) ) {
728                 my $confData = $bpc->ConfigDataRead($copyConf->{$host});
729                 my $fromHost = $copyConf->{$host};
730                 $err  .= $bpc->ConfigDataWrite($host, $confData);
731                 $mesg .= eval("qq($Lang->{CfgEdit_Log_Copy_host_config})");
732             }
733
734             delete($newConf->{Hosts});
735             $mesg .= configDiffMesg(undef, $mainConf, $newConf);
736             $mainConf = { %$mainConf, %$newConf };
737             $err .= $bpc->ConfigDataWrite(undef, $mainConf);
738             $newConf->{Hosts} = $hostsSave;
739         }
740         if ( defined($err) ) {
741             $tblContent .= <<EOF;
742 <tr><td colspan="2" class="border"><span class="editError">$err</span></td></tr>
743 EOF
744         }
745         $bpc->ServerConnect();
746         if ( $mesg ne "" ) {
747             (my $mesgBR = $mesg) =~ s/\n/<br>\n/g;
748              # uncomment this if you want the changes to be displayed
749 #            $tblContent .= <<EOF;
750 #<tr><td colspan="2" class="border"><span class="editComment">$mesgBR</span></td></tr>
751 #EOF
752             foreach my $str ( split(/\n/, $mesg) ) {
753                 $bpc->ServerMesg("log $str") if ( $str ne "" );
754             }
755         }
756         #
757         # Tell the server to reload, unless we only changed
758         # a client config
759         #
760         $bpc->ServerMesg("server reload") if ( $host eq "" );
761     }
762
763     my @mask = @{$menuDisable{$menu}{mask} || []};
764
765     foreach my $paramInfo ( @{$ConfigMenu{$menu}{param}} ) {
766
767         my $param    = $paramInfo->{name};
768         my $disabled = shift(@mask);
769
770         next if ( $disabled || $menuDisable{$menu}{top} );
771         if ( ref($paramInfo->{visible}) eq "CODE"
772                         && !&{$paramInfo->{visible}}($newConf, $bpc) ) {
773             next;
774         }
775
776         if ( defined($paramInfo->{text}) ) {
777             my $text = eval("qq($Lang->{$paramInfo->{text}})");
778             $tblContent .= <<EOF;
779 <tr><td colspan="2" class="editHeader">$text</td></tr>
780 EOF
781             next;
782         }
783
784         #
785         # TODO: get parameter documentation
786         #
787         my $comment = "";
788         #$comment =~ s/\'//g;
789         #$comment =~ s/\"//g;
790         #$comment =~ s/\n/ /g;
791
792         $doneParam->{$param} = 1;
793
794         $tblContent .= fieldEditBuild($ConfigMeta{$param},
795                               $param,
796                               $newConf->{$param},
797                               $errors,
798                               0,
799                               $comment,
800                               $isError,
801                               $paramInfo->{onchangeSubmit},
802                               defined($override) ? $param : undef,
803                               defined($override) ? $override->{$param} : undef
804                         );
805         if ( defined($paramInfo->{comment}) ) {
806             my $topDir = $bpc->TopDir;
807             my $text = eval("qq($Lang->{$paramInfo->{comment}})");
808             $tblContent .= <<EOF;
809 <tr><td colspan="2" class="editComment">$text</td></tr>
810 EOF
811         }
812     }
813
814     #
815     # Emit a summary of all the errors
816     #
817     my $errorTxt;
818
819     if ( %$errors ) {
820         $errorTxt .= <<EOF;
821 <tr><td colspan="2" class="border"><span class="editError">$Lang->{CfgEdit_Error_No_Save}</span></td></tr>
822 EOF
823     }
824
825     foreach my $param ( sort(keys(%$errors)) ) {
826         $errorTxt .= <<EOF;
827 <tr><td colspan="2" class="border"><span class="editError">$errors->{$param}</span></td></tr>
828 EOF
829     }
830
831     $content .= <<EOF;
832 $errorTxt
833 $tblContent
834 </table>
835 EOF
836
837     #
838     # Emit all the remaining editable config settings as hidden values
839     #
840     foreach my $param ( keys(%ConfigMeta) ) {
841         next if ( $doneParam->{$param} );
842         next if ( $userHost
843                       && (!defined($bpc->{Conf}{CgiUserConfigEdit}{$param})
844                          || (!$PrivAdmin
845                              && !$bpc->{Conf}{CgiUserConfigEdit}{$param})) );
846         $content .= fieldHiddenBuild($ConfigMeta{$param},
847                             $param,
848                             $newConf->{$param},
849                             "v"
850                         );
851         if ( defined($override) ) {
852             $content .= <<EOF;
853 <input type="hidden" name="override_$param" value="$override->{$param}">
854 EOF
855         }
856         $doneParam->{$param} = 1;
857     }
858
859     if ( defined($In{menu}) || $In{saveAction} eq "Save" ) {
860         if ( $In{saveAction} eq "Save" && !$userHost ) {
861             #
862             # Emit the new settings as orig_zZ_ parameters
863             #
864             $doneParam = {};
865             foreach my $param ( keys(%ConfigMeta) ) {
866                 next if ( $doneParam->{$param} );
867                 next if ( $userHost
868                           && (!defined($bpc->{Conf}{CgiUserConfigEdit}{$param})
869                              || (!$PrivAdmin
870                                 && !$bpc->{Conf}{CgiUserConfigEdit}{$param})) );
871                 $contentHidden .= fieldHiddenBuild($ConfigMeta{$param},
872                                         $param,
873                                         $newConf->{$param},
874                                         "orig",
875                                     );
876                 $doneParam->{$param} = 1;
877                 $In{modified} = 0;
878             }
879         } else {
880             #
881             # Just switching menus: copy all the orig_zZ_ input parameters
882             #
883             foreach my $var ( keys(%In) ) {
884                 next if ( $var !~ /^orig_zZ_/ );
885                 my $val = decode_utf8($In{$var});
886                 $contentHidden .= <<EOF;
887 <input type="hidden" name="$var" value="${EscHTML($val)}">
888 EOF
889             }
890         }
891     } else {
892         #
893         # First time: emit all the original config settings
894         #
895         $doneParam = {};
896         foreach my $param ( keys(%ConfigMeta) ) {
897             next if ( $doneParam->{$param} );
898             next if ( $userHost
899                           && (!defined($bpc->{Conf}{CgiUserConfigEdit}{$param})
900                              || (!$PrivAdmin
901                                 && !$bpc->{Conf}{CgiUserConfigEdit}{$param})) );
902             $contentHidden .= fieldHiddenBuild($ConfigMeta{$param},
903                                     $param,
904                                     $mainConf->{$param},
905                                     "orig",
906                                 );
907             $doneParam->{$param} = 1;
908         }
909     }
910
911     $content .= <<EOF;
912 $contentHidden
913 </form>
914 </tr>
915 </table>
916 EOF
917
918     Header("Config Edit", $content);
919     Trailer();
920 }
921
922 sub fieldHiddenBuild
923 {
924     my($type, $varName, $varValue, $prefix) = @_;
925     my $content;
926
927     $type = { type => $type } if ( ref($type) ne "HASH" );
928
929     if ( $type->{type} eq "list" ) {
930         $varValue = [] if ( !defined($varValue) );
931         $varValue = [$varValue] if ( ref($varValue) ne "ARRAY" );
932
933         for ( my $i = 0 ; $i < @$varValue ; $i++ ) {
934             $content .= fieldHiddenBuild($type->{child}, "${varName}_zZ_$i",
935                                          $varValue->[$i], $prefix);
936         }
937     } elsif ( $type->{type} eq "hash" || $type->{type} eq "horizHash" ) {
938         $varValue = {} if ( ref($varValue) ne "HASH" );
939         my(@order, $childType);
940
941         if ( defined($type->{order}) ) {
942             @order = @{$type->{order}};
943         } elsif ( defined($type->{child}) ) {
944             @order = sort(keys(%{$type->{child}}));
945         } else {
946             @order = sort(keys(%$varValue));
947         }
948
949         foreach my $fld ( @order ) {
950             if ( defined($type->{child}) ) {
951                 $childType = $type->{child}{$fld};
952             } else {
953                 $childType = $type->{childType};
954                 #
955                 # emit list of fields since they are user-defined
956                 # rather than hard-coded
957                 #
958                 $content .= <<EOF;
959 <input type="hidden" name="vflds.$varName" value="${EscHTML($fld)}">
960 EOF
961             }
962             $content .= fieldHiddenBuild($childType, "${varName}_zZ_$fld",
963                                          $varValue->{$fld}, $prefix);
964         }
965     } elsif ( $type->{type} eq "shortlist" ) {
966         $varValue = [$varValue] if ( ref($varValue) ne "ARRAY" );
967         $varValue = join(", ", @$varValue);
968         $content .= <<EOF;
969 <input type="hidden" name="${prefix}_zZ_$varName" value="${EscHTML($varValue)}">
970 EOF
971     } else {
972         $content .= <<EOF;
973 <input type="hidden" name="${prefix}_zZ_$varName" value="${EscHTML($varValue)}">
974 EOF
975     }
976     return $content;
977 }
978
979 sub fieldEditBuild
980 {
981     my($type, $varName, $varValue, $errors, $level, $comment, $isError,
982        $onchangeSubmit, $overrideVar, $overrideSet) = @_;
983
984     my $content;
985     my $size = 50 - 10 * $level;
986     $type = { type => $type } if ( ref($type) ne "HASH" );
987
988     $size = $type->{size} if ( defined($type->{size}) );
989
990     #
991     # These fragments allow inline content to be turned on and off
992     #
993     # <tr><td colspan="2"><span id="id_$varName" style="display: none" class="editComment">$comment</span></td></tr>
994     # <tr><td class="border"><a href="javascript: displayHelp('$varName')">$varName</a>
995     #
996
997     if ( $level == 0 ) {
998         my $lcVarName = lc($varName);
999         $content .= <<EOF;
1000 <tr><td class="border"><a href="?action=view&type=docs#item__conf_${lcVarName}_">$varName</a>
1001 EOF
1002         if ( defined($overrideVar) ) {
1003             my $override_checked = "";
1004             if ( !$isError && $In{deleteVar}      =~ /^\Q${varName}_zZ_/
1005                    || !$isError && $In{insertVar} =~ /^\Q${varName}\E(_zZ_|$)/
1006                    || !$isError && $In{addVar}    =~ /^\Q${varName}\E(_zZ_|$)/ ) {
1007                 $overrideSet = 1;
1008             }
1009             if ( $overrideSet ) {
1010                 $override_checked = "checked";
1011             }
1012             $content .= <<EOF;
1013 <br><input type="checkbox" name="override_$varName" $override_checked value="1" onClick="checkboxChange('$varName')">\&nbsp;${EscHTML($Lang->{CfgEdit_Button_Override})}
1014 EOF
1015         }
1016         $content .= "</td>\n";
1017     }
1018
1019     if ( $type->{type} eq "list" ) {
1020         $content .= "<td class=\"border\">\n";
1021         $varValue = [] if ( !defined($varValue) );
1022         $varValue = [$varValue] if ( ref($varValue) ne "ARRAY" );
1023         if ( !$isError && $In{deleteVar} =~ /^\Q${varName}_zZ_\E(\d+)$/
1024                 && $1 < @$varValue ) {
1025             #
1026             # User deleted entry in this array
1027             #
1028             splice(@$varValue, $1, 1) if ( @$varValue > 1 || $type->{emptyOk} );
1029             $In{deleteVar} = "";
1030         }
1031         if ( !$isError && $In{insertVar} =~ /^\Q${varName}_zZ_\E(\d+)$/
1032                 && $1 < @$varValue ) {
1033             #
1034             # User inserted entry in this array
1035             #
1036             splice(@$varValue, $1, 0, "")
1037                         if ( @$varValue > 1 || $type->{emptyOk} );
1038             $In{insertVar} = "";
1039         }
1040         if ( !$isError && $In{addVar} eq $varName ) {
1041             #
1042             # User added entry to this array
1043             #
1044             push(@$varValue, undef);
1045             $In{addVar} = "";
1046         }
1047         $content .= "<table border=\"1\" cellspacing=\"0\">\n";
1048         my $colspan;
1049
1050         if ( ref($type) eq "HASH" && ref($type->{child}) eq "HASH"
1051                     && $type->{child}{type} eq "horizHash" ) {
1052             my @order;
1053             if ( defined($type->{child}{order}) ) {
1054                 @order = @{$type->{child}{order}};
1055             } else {
1056                 @order = sort(keys(%{$type->{child}{child}}));
1057             }
1058             $content .= "<tr><td class=\"border\"></td>\n";
1059             for ( my $i = 0 ; $i < @order ; $i++ ) {
1060                 $content .= "<td class=\"tableheader\">$order[$i]</td>\n";
1061             }
1062             $colspan = @order + 1;
1063             $content .= "</tr>\n";
1064             for ( my $i = 0 ; $i < @$varValue ; $i++ ) {
1065                 if ( @$varValue > 1 || $type->{emptyOk} ) {
1066                     $content .= <<EOF;
1067 <td class="border">
1068 <input type="button" name="del_${varName}_zZ_$i" value="${EscHTML($Lang->{CfgEdit_Button_Delete})}"
1069     onClick="deleteSubmit('${varName}_zZ_$i')">
1070 </td>
1071 EOF
1072                 }
1073                 $content .= fieldEditBuild($type->{child}, "${varName}_zZ_$i",
1074                                   $varValue->[$i], $errors, $level + 1, undef,
1075                                   $isError, $onchangeSubmit,
1076                                   $overrideVar, $overrideSet);
1077                 $content .= "</tr>\n";
1078             }
1079         } else {
1080             for ( my $i = 0 ; $i < @$varValue ; $i++ ) {
1081                 $content .= <<EOF;
1082 <tr><td class="border">
1083 <input type="button" name="ins_${varName}_zZ_$i" value="${EscHTML($Lang->{CfgEdit_Button_Insert})}"
1084     onClick="insertSubmit('${varName}_zZ_$i')">
1085 EOF
1086                 if ( @$varValue > 1 || $type->{emptyOk} ) {
1087                     $content .= <<EOF;
1088 <input type="button" name="del_${varName}_zZ_$i" value="${EscHTML($Lang->{CfgEdit_Button_Delete})}"
1089     onClick="deleteSubmit('${varName}_zZ_$i')">
1090 EOF
1091                 }
1092                 $content .= "</td>\n";
1093                 $content .= fieldEditBuild($type->{child}, "${varName}_zZ_$i",
1094                                     $varValue->[$i], $errors, $level + 1, undef,
1095                                     $isError, $onchangeSubmit,
1096                                     $overrideVar, $overrideSet);
1097                 $content .= "</tr>\n";
1098             }
1099             $colspan = 2;
1100         }
1101         $content .= <<EOF;
1102 <tr><td class="border" colspan="$colspan"><input type="button" name="add_$varName" value="${EscHTML($Lang->{CfgEdit_Button_Add})}"
1103     onClick="addSubmit('$varName')"></td></tr>
1104 </table>
1105 EOF
1106         $content .= "</td>\n";
1107     } elsif ( $type->{type} eq "hash" ) {
1108         $content .= "<td class=\"border\">\n";
1109         $content .= "<table border=\"1\" cellspacing=\"0\">\n";
1110         $varValue = {} if ( ref($varValue) ne "HASH" );
1111
1112         if ( !$isError && !$type->{noKeyEdit}
1113                         && $In{deleteVar} !~ /^\Q${varName}_zZ_\E.*_zZ_/
1114                         && $In{deleteVar} =~ /^\Q${varName}_zZ_\E(.*)$/ ) {
1115             #
1116             # User deleted entry in this hash
1117             #
1118             delete($varValue->{$1}) if ( keys(%$varValue) > 1
1119                                             || $type->{emptyOk} );
1120             $In{deleteVar} = "";
1121         }
1122         if ( !$isError && !defined($type->{child})
1123                         && $In{addVar} eq $varName ) {
1124             #
1125             # User added entry to this array
1126             #
1127             $varValue->{$In{"addVarKey_$varName"}} = ""
1128                         if ( !defined($varValue->{$In{"addVarKey_$varName"}}) );
1129             $In{addVar} = "";
1130         }
1131         my(@order, $childType);
1132
1133         if ( defined($type->{order}) ) {
1134             @order = @{$type->{order}};
1135         } elsif ( defined($type->{child}) ) {
1136             @order = sort(keys(%{$type->{child}}));
1137         } else {
1138             @order = sort(keys(%$varValue));
1139         }
1140
1141         foreach my $fld ( @order ) {
1142             $content .= <<EOF;
1143 <tr><td class="border">$fld
1144 EOF
1145             if ( !$type->{noKeyEdit}
1146                     && (keys(%$varValue) > 1 || $type->{emptyOk}) ) {
1147                 $content .= <<EOF;
1148 <input type="submit" name="del_${varName}_zZ_$fld" value="${EscHTML($Lang->{CfgEdit_Button_Delete})}"
1149         onClick="deleteSubmit('${varName}_zZ_$fld')">
1150 EOF
1151             }
1152             if ( defined($type->{child}) ) {
1153                 $childType = $type->{child}{$fld};
1154             } else {
1155                 $childType = $type->{childType};
1156                 #
1157                 # emit list of fields since they are user-defined
1158                 # rather than hard-coded
1159                 #
1160                 $content .= <<EOF;
1161 <input type="hidden" name="vflds.$varName" value="${EscHTML($fld)}">
1162 EOF
1163             }
1164             $content .= "</td>\n";
1165             $content .= fieldEditBuild($childType, "${varName}_zZ_$fld",
1166                             $varValue->{$fld}, $errors, $level + 1, undef,
1167                             $isError, $onchangeSubmit,
1168                             $overrideVar, $overrideSet);
1169             $content .= "</tr>\n";
1170         }
1171
1172         if ( !$type->{noKeyEdit} ) {
1173             $content .= <<EOF;
1174 <tr><td class="border" colspan="2">
1175 $Lang->{CfgEdit_Button_New_Key}: <input type="text" class="editTextInput" name="addVarKey_$varName" size="20" maxlength="256" value="">
1176 <input type="button" name="add_$varName" value="${EscHTML($Lang->{CfgEdit_Button_Add})}" onClick="addSubmit('$varName', 1)">
1177 </td></tr>
1178 EOF
1179         }
1180         $content .= "</table>\n";
1181         $content .= "</td>\n";
1182     } elsif ( $type->{type} eq "horizHash" ) {
1183         $varValue = {} if ( ref($varValue) ne "HASH" );
1184         my(@order, $childType);
1185
1186         if ( defined($type->{order}) ) {
1187             @order = @{$type->{order}};
1188         } elsif ( defined($type->{child}) ) {
1189             @order = sort(keys(%{$type->{child}}));
1190         } else {
1191             @order = sort(keys(%$varValue));
1192         }
1193
1194         foreach my $fld ( @order ) {
1195             if ( defined($type->{child}) ) {
1196                 $childType = $type->{child}{$fld};
1197             } else {
1198                 $childType = $type->{childType};
1199                 #
1200                 # emit list of fields since they are user-defined
1201                 # rather than hard-coded
1202                 #
1203                 $content .= <<EOF;
1204 <input type="hidden" name="vflds.$varName" value="${EscHTML($fld)}">
1205 EOF
1206             }
1207             $content .= fieldEditBuild($childType, "${varName}_zZ_$fld",
1208                             $varValue->{$fld}, $errors, $level + 1, undef,
1209                             $isError, $onchangeSubmit,
1210                             $overrideVar, $overrideSet);
1211         }
1212     } else {
1213         $content .= "<td class=\"border\">\n";
1214         if ( $isError ) {
1215             #
1216             # If there was an error, we use the original post values
1217             # in %In, rather than the parsed values in $varValue.
1218             # This is so that the user's erroneous input is preserved.
1219             #
1220             $varValue = $In{"v_zZ_$varName"} if ( defined($In{"v_zZ_$varName"}) );
1221         }
1222         if ( defined($errors->{$varName}) ) {
1223             $content .= <<EOF;
1224 <span class="editError">$errors->{$varName}</span><br>
1225 EOF
1226         }
1227         my $onChange;
1228         if ( defined($overrideVar) ) {
1229             $onChange .= "checkboxSet('$overrideVar');";
1230         } else {
1231             $onChange .= "varChange('$varName');";
1232         }
1233         if ( $onchangeSubmit ) {
1234             $onChange .= "document.editForm.submit();";
1235         }
1236         if ( $onChange ne "" ) {
1237             $onChange = " onChange=\"$onChange\"";
1238         }
1239         if ( $varValue !~ /\n/ &&
1240                 ($type->{type} eq "integer"
1241                     || $type->{type} eq "string"
1242                     || $type->{type} eq "execPath"
1243                     || $type->{type} eq "shortlist"
1244                     || $type->{type} eq "float") ) {
1245             # simple input box
1246             if ( $type->{type} eq "shortlist" ) {
1247                 $varValue = [$varValue] if ( ref($varValue) ne "ARRAY" );
1248                 $varValue = join(", ", @$varValue);
1249             }
1250             my $textType = ($varName =~ /Passwd/) ? "password" : "text";
1251             $content .= <<EOF;
1252 <input type="$textType" class="editTextInput" name="v_zZ_$varName" size="$size" maxlength="256" value="${EscHTML($varValue)}"$onChange>
1253 EOF
1254         } elsif ( $type->{type} eq "boolean" ) {
1255             # checkbox
1256             my $checked = "checked" if ( $varValue );
1257             $content .= <<EOF;
1258 <input type="checkbox" name="v_zZ_$varName" $checked value="1"$onChange>
1259 EOF
1260         } elsif ( $type->{type} eq "select" ) {
1261             $content .= <<EOF;
1262 <select name="v_zZ_$varName"$onChange>
1263 EOF
1264             foreach my $option ( @{$type->{values}} ) {
1265                 my $sel = " selected" if ( $varValue eq $option );
1266                 $content .= "<option$sel>$option</option>\n";
1267             }
1268             $content .= "</select>\n";
1269         } else {
1270             # multi-line text area - count number of lines
1271             my $rowCnt = $varValue =~ tr/\n//;
1272             $rowCnt = 1 if ( $rowCnt < 1 );
1273             $content .= <<EOF;
1274 <textarea name="v_zZ_$varName" class="editTextArea" cols="$size" rows="$rowCnt"$onChange>${EscHTML($varValue)}</textarea>
1275 EOF
1276         }
1277         $content .= "</td>\n";
1278     }
1279     return $content;
1280 }
1281
1282 sub errorCheck
1283 {
1284     my $errors = {};
1285
1286     foreach my $param ( keys(%ConfigMeta) ) {
1287         fieldErrorCheck($ConfigMeta{$param}, $param, $errors);
1288     }
1289     return $errors;
1290 }
1291
1292 sub fieldErrorCheck
1293 {
1294     my($type, $varName, $errors) = @_;
1295
1296     $type = { type => $type } if ( ref($type) ne "HASH" );
1297
1298     if ( $type->{type} eq "list" ) {
1299         for ( my $i = 0 ; ; $i++ ) {
1300             last if ( fieldErrorCheck($type->{child}, "${varName}_zZ_$i", $errors) );
1301         }
1302     } elsif ( $type->{type} eq "hash" || $type->{type} eq "horizHash" ) {
1303         my(@order, $childType);
1304         my $ret;
1305
1306         if ( defined($type->{order}) ) {
1307             @order = @{$type->{order}};
1308         } elsif ( defined($type->{child}) ) {
1309             @order = sort(keys(%{$type->{child}}));
1310         } else {
1311             @order = split(/\0/, $In{"vflds.$varName"});
1312         }
1313         foreach my $fld ( @order ) {
1314             if ( defined($type->{child}) ) {
1315                 $childType = $type->{child}{$fld};
1316             } else {
1317                 $childType = $type->{childType};
1318             }
1319             $ret ||= fieldErrorCheck($childType, "${varName}_zZ_$fld", $errors);
1320         }
1321         return $ret;
1322     } else {
1323         $In{"v_zZ_$varName"} = "0" if ( $type->{type} eq "boolean"
1324                                         && $In{"v_zZ_$varName"} eq "" );
1325
1326         return 1 if ( !exists($In{"v_zZ_$varName"}) );
1327
1328         (my $var = $varName) =~ s/_zZ_/./g;
1329
1330         if ( $type->{type} eq "integer"
1331                 || $type->{type} eq "boolean" ) {
1332             if ( $In{"v_zZ_$varName"} !~ /^-?\d+\s*$/s
1333                             && $In{"v_zZ_$varName"} ne "" ) {
1334                 $errors->{$varName} = eval("qq{$Lang->{CfgEdit_Error__must_be_an_integer}}");
1335             }
1336         } elsif ( $type->{type} eq "float" ) {
1337             if ( $In{"v_zZ_$varName"} !~ /^-?\d*(\.\d*)?\s*$/s
1338                             && $In{"v_zZ_$varName"} ne "" ) {
1339                 $errors->{$varName}
1340                         = eval("qq{$Lang->{CfgEdit_Error__must_be_real_valued_number}}");
1341             }
1342         } elsif ( $type->{type} eq "shortlist" ) {
1343             my @vals = split(/[,\s]+/, $In{"v_zZ_$varName"});
1344             for ( my $i = 0 ; $i < @vals ; $i++ ) {
1345                 if ( $type->{child} eq "integer"
1346                         && $vals[$i] !~ /^-?\d+\s*$/s
1347                         && $vals[$i] ne "" ) {
1348                     my $k = $i + 1;
1349                     $errors->{$varName} = eval("qq{$Lang->{CfgEdit_Error__entry__must_be_an_integer}}");
1350                 } elsif ( $type->{child} eq "float"
1351                         && $vals[$i] !~ /^-?\d*(\.\d*)?\s*$/s
1352                         && $vals[$i] ne "" ) {
1353                     my $k = $i + 1;
1354                     $errors->{$varName} = eval("qq{$Lang->{CfgEdit_Error__entry__must_be_real_valued_number}}");
1355                 }
1356             }
1357         } elsif ( $type->{type} eq "select" ) {
1358             my $match = 0;
1359             foreach my $option ( @{$type->{values}} ) {
1360                 if ( $In{"v_zZ_$varName"} eq $option ) {
1361                     $match = 1;
1362                     last;
1363                 }
1364             }
1365             $errors->{$varName} = eval("qq{$Lang->{CfgEdit_Error__must_be_valid_option}}")
1366                             if ( !$match );
1367         } elsif ( $type->{type} eq "execPath" ) {
1368             if ( $In{"v_zZ_$varName"} ne "" && !-x $In{"v_zZ_$varName"} ) {
1369                 $errors->{$varName} = eval("qq{$Lang->{CfgEdit_Error__must_be_executable_program}}");
1370             }
1371         } else {
1372             #
1373             # $type->{type} eq "string": no error checking
1374             #
1375         }
1376     }
1377     return 0;
1378 }
1379
1380 sub inputParse
1381 {
1382     my($bpc, $userHost) = @_;
1383     my $conf     = {};
1384     my $override = {};
1385
1386     foreach my $param ( keys(%ConfigMeta) ) {
1387         my $value;
1388         next if ( $userHost
1389                       && (!defined($bpc->{Conf}{CgiUserConfigEdit}{$param})
1390                          || (!$PrivAdmin
1391                             && !$bpc->{Conf}{CgiUserConfigEdit}{$param})) );
1392         fieldInputParse($ConfigMeta{$param}, $param, \$value);
1393         $conf->{$param}     = $value;
1394         $override->{$param} = $In{"override_$param"};
1395     }
1396     return ($conf, $override);
1397 }
1398
1399 sub fieldInputParse
1400 {
1401     my($type, $varName, $value) = @_;
1402
1403     $type = { type => $type } if ( ref($type) ne "HASH" );
1404
1405     if ( $type->{type} eq "list" ) {
1406         $$value = [];
1407         for ( my $i = 0 ; ; $i++ ) {
1408             my $val;
1409             last if ( fieldInputParse($type->{child}, "${varName}_zZ_$i", \$val) );
1410             push(@$$value, $val);
1411         }
1412         $$value = undef if ( $type->{undefIfEmpty} && @$$value == 0 );
1413     } elsif ( $type->{type} eq "hash" || $type->{type} eq "horizHash" ) {
1414         my(@order, $childType);
1415         my $ret;
1416         $$value = {};
1417
1418         if ( defined($type->{order}) ) {
1419             @order = @{$type->{order}};
1420         } elsif ( defined($type->{child}) ) {
1421             @order = sort(keys(%{$type->{child}}));
1422         } else {
1423             @order = split(/\0/, $In{"vflds.$varName"});
1424         }
1425
1426         foreach my $fld ( @order ) {
1427             my $val;
1428             if ( defined($type->{child}) ) {
1429                 $childType = $type->{child}{$fld};
1430             } else {
1431                 $childType = $type->{childType};
1432             }
1433             $ret ||= fieldInputParse($childType, "${varName}_zZ_$fld", \$val);
1434             last if ( $ret );
1435             $$value->{$fld} = $val;
1436         }
1437         return $ret;
1438     } else {
1439         if ( $type->{type} eq "boolean" ) {
1440             $$value = 0 + $In{"v_zZ_$varName"};
1441         } elsif ( !exists($In{"v_zZ_$varName"}) ) {
1442             return 1;
1443         }
1444
1445         my $v = $In{"v_zZ_$varName"};
1446
1447         if ( $type->{type} eq "integer" ) {
1448             if ( $v =~ /^-?\d+\s*$/s || $v eq "" ) {
1449                 $$value = 0 + $v;
1450             } else {
1451                 # error value - keep in string form
1452                 $$value = $v;
1453             }
1454         } elsif ( $type->{type} eq "float" ) {
1455             if ( $v =~ /^-?\d*(\.\d*)?\s*$/s || $v eq "" ) {
1456                 $$value = 0 + $v;
1457             } else {
1458                 # error value - keep in string form
1459                 $$value = $v;
1460             }
1461         } elsif ( $type->{type} eq "shortlist" ) {
1462             $$value = [split(/[,\s]+/, $v)];
1463             if ( $type->{child} eq "float" ) {
1464                 foreach ( @$$value ) {
1465                     if ( /^-?\d*(\.\d*)?\s*$/s || $v eq "" ) {
1466                         $_ += 0;
1467                     }
1468                 }
1469             } elsif ( $type->{child} eq "integer"
1470                         || $type->{child} eq "boolean" ) {
1471                 foreach ( @$$value ) {
1472                     if ( /^-?\d+\s*$/s || $v eq "" ) {
1473                         $_ += 0;
1474                     }
1475                 }
1476             }
1477         } else {
1478             $$value = decode_utf8($In{"v_zZ_$varName"});
1479             $$value =~ s/\r\n/\n/g;
1480             # remove leading space from exec paths
1481             $$value =~ s/^\s+// if ( $type->{type} eq "execPath" );
1482         }
1483         $$value = undef if ( $type->{undefIfEmpty} && $$value eq "" );
1484     }
1485     return 0;
1486 }
1487
1488 sub configDiffMesg
1489 {
1490     my($host, $oldConf, $newConf) = @_;
1491     my $mesg;
1492     my $conf;
1493
1494     if ( $host ne "" ) {
1495         $conf = "host $host config";
1496     } else {
1497         $conf = "main config";
1498     }
1499
1500     foreach my $p ( keys(%ConfigMeta) ) {
1501         if ( !exists($oldConf->{$p}) && !exists($newConf->{$p}) ) {
1502             next;
1503         } elsif ( exists($oldConf->{$p}) && !exists($newConf->{$p}) ) {
1504             $mesg .= eval("qq($Lang->{CfgEdit_Log_Delete_param})");
1505         } elsif ( !exists($oldConf->{$p}) && exists($newConf->{$p}) ) {
1506             my $dump = Data::Dumper->new([$newConf->{$p}]);
1507             $dump->Indent(0);
1508             $dump->Sortkeys(1);
1509             $dump->Terse(1);
1510             my $value = $dump->Dump;
1511             $value =~ s/\n/\\n/g;
1512             $value =~ s/\r/\\r/g;
1513             $mesg .= eval("qq($Lang->{CfgEdit_Log_Add_param_value})");
1514         } else {
1515             my $dump = Data::Dumper->new([$newConf->{$p}]);
1516             $dump->Indent(0);
1517             $dump->Sortkeys(1);
1518             $dump->Terse(1);
1519             my $valueNew = $dump->Dump;
1520
1521             my $v = $oldConf->{$p};
1522             if ( ref($newConf->{$p}) eq "ARRAY" && ref($v) eq "" ) {
1523                 $v = [$v];
1524             }
1525             $dump = Data::Dumper->new([$v]);
1526             $dump->Indent(0);
1527             $dump->Sortkeys(1);
1528             $dump->Terse(1);
1529             my $valueOld = $dump->Dump;
1530
1531             (my $valueNew2 = $valueNew) =~ s/['\n\r]//g;
1532             (my $valueOld2 = $valueOld) =~ s/['\n\r]//g;
1533
1534             next if ( $valueOld2 eq $valueNew2 );
1535
1536             $valueNew =~ s/\n/\\n/g;
1537             $valueOld =~ s/\n/\\n/g;
1538             $valueNew =~ s/\r/\\r/g;
1539             $valueOld =~ s/\r/\\r/g;
1540
1541             $mesg .= eval("qq($Lang->{CfgEdit_Log_Change_param_value})");
1542         }
1543     }
1544     return $mesg;
1545 }
1546
1547 sub hostsDiffMesg
1548 {
1549     my($hostsNew) = @_;
1550     my $hostsOld = $bpc->HostInfoRead();
1551     my($mesg, $hostChange);
1552
1553     foreach my $host ( keys(%$hostsOld) ) {
1554         if ( !defined($hostsNew->{$host}) ) {
1555             $mesg .= eval("qq($Lang->{CfgEdit_Log_Host_Delete})");
1556             $hostChange++;
1557             next;
1558         }
1559         foreach my $key ( keys(%{$hostsNew->{$host}}) ) {
1560             next if ( $hostsNew->{$host}{$key} eq $hostsOld->{$host}{$key} );
1561             my $valueOld = $hostsOld->{$host}{$key};
1562             my $valueNew = $hostsNew->{$host}{$key};
1563             $mesg .= eval("qq($Lang->{CfgEdit_Log_Host_Change})");
1564             $hostChange++;
1565         }
1566     }
1567
1568     foreach my $host ( keys(%$hostsNew) ) {
1569         next if ( defined($hostsOld->{$host}) );
1570         my $dump = Data::Dumper->new([$hostsNew->{$host}]);
1571         $dump->Indent(0);
1572         $dump->Sortkeys(1);
1573         $dump->Terse(1);
1574         my $value = $dump->Dump;
1575         $value =~ s/\n/\\n/g;
1576         $value =~ s/\r/\\r/g;
1577         $mesg .= eval("qq($Lang->{CfgEdit_Log_Host_Add})");
1578         $hostChange++;
1579     }
1580     return ($mesg, $hostChange);
1581 }
1582
1583 1;