Project

General

Profile

Download (24.5 KB) Statistics
| Branch: | Tag: | Revision:
1
<?php
2
/*
3
 * system.php
4
 *
5
 * part of pfSense (https://www.pfsense.org)
6
 * Copyright (c) 2004-2018 Rubicon Communications, LLC (Netgate)
7
 * All rights reserved.
8
 *
9
 * originally based on m0n0wall (http://m0n0.ch/wall)
10
 * Copyright (c) 2003-2004 Manuel Kasper <mk@neon1.net>.
11
 * All rights reserved.
12
 *
13
 * Licensed under the Apache License, Version 2.0 (the "License");
14
 * you may not use this file except in compliance with the License.
15
 * You may obtain a copy of the License at
16
 *
17
 * http://www.apache.org/licenses/LICENSE-2.0
18
 *
19
 * Unless required by applicable law or agreed to in writing, software
20
 * distributed under the License is distributed on an "AS IS" BASIS,
21
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
22
 * See the License for the specific language governing permissions and
23
 * limitations under the License.
24
 */
25

    
26
##|+PRIV
27
##|*IDENT=page-system-generalsetup
28
##|*NAME=System: General Setup
29
##|*DESCR=Allow access to the 'System: General Setup' page.
30
##|*MATCH=system.php*
31
##|-PRIV
32

    
33
require_once("guiconfig.inc");
34
require_once("functions.inc");
35
require_once("filter.inc");
36
require_once("shaper.inc");
37
require_once("system.inc");
38

    
39
$pconfig['hostname'] = $config['system']['hostname'];
40
$pconfig['domain'] = $config['system']['domain'];
41
$pconfig['dnsserver'] = $config['system']['dnsserver'];
42

    
43
$arr_gateways = return_gateways_array();
44

    
45
// set default columns to two if unset
46
if (!isset($config['system']['webgui']['dashboardcolumns'])) {
47
	$config['system']['webgui']['dashboardcolumns'] = 2;
48
}
49

    
50
// set default language if unset
51
if (!isset($config['system']['language'])) {
52
	$config['system']['language'] = $g['language'];
53
}
54

    
55
$dnsgw_counter = 1;
56

    
57
while (isset($config["system"]["dns{$dnsgw_counter}gw"])) {
58
	$pconfig_dnsgw_counter = $dnsgw_counter - 1;
59
	$pconfig["dnsgw{$pconfig_dnsgw_counter}"] = $config["system"]["dns{$dnsgw_counter}gw"];
60
	$dnsgw_counter++;
61
}
62

    
63
$pconfig['dnsallowoverride'] = isset($config['system']['dnsallowoverride']);
64
$pconfig['timezone'] = $config['system']['timezone'];
65
$pconfig['timeservers'] = $config['system']['timeservers'];
66
$pconfig['language'] = $config['system']['language'];
67
$pconfig['webguicss'] = $config['system']['webgui']['webguicss'];
68
$pconfig['logincss'] = $config['system']['webgui']['logincss'];
69
$pconfig['webguifixedmenu'] = $config['system']['webgui']['webguifixedmenu'];
70
$pconfig['dashboardcolumns'] = $config['system']['webgui']['dashboardcolumns'];
71
$pconfig['interfacessort'] = isset($config['system']['webgui']['interfacessort']);
72
$pconfig['webguileftcolumnhyper'] = isset($config['system']['webgui']['webguileftcolumnhyper']);
73
$pconfig['disablealiaspopupdetail'] = isset($config['system']['webgui']['disablealiaspopupdetail']);
74
$pconfig['dashboardavailablewidgetspanel'] = isset($config['system']['webgui']['dashboardavailablewidgetspanel']);
75
$pconfig['systemlogsfilterpanel'] = isset($config['system']['webgui']['systemlogsfilterpanel']);
76
$pconfig['systemlogsmanagelogpanel'] = isset($config['system']['webgui']['systemlogsmanagelogpanel']);
77
$pconfig['statusmonitoringsettingspanel'] = isset($config['system']['webgui']['statusmonitoringsettingspanel']);
78
$pconfig['webguihostnamemenu'] = $config['system']['webgui']['webguihostnamemenu'];
79
$pconfig['dnslocalhost'] = isset($config['system']['dnslocalhost']);
80
//$pconfig['dashboardperiod'] = isset($config['widgets']['period']) ? $config['widgets']['period']:"10";
81
$pconfig['roworderdragging'] = isset($config['system']['webgui']['roworderdragging']);
82
$pconfig['loginshowhost'] = isset($config['system']['webgui']['loginshowhost']);
83
$pconfig['requirestatefilter'] = isset($config['system']['webgui']['requirestatefilter']);
84

    
85
if (!$pconfig['timezone']) {
86
	if (isset($g['default_timezone']) && !empty($g['default_timezone'])) {
87
		$pconfig['timezone'] = $g['default_timezone'];
88
	} else {
89
		$pconfig['timezone'] = "Etc/UTC";
90
	}
91
}
92

    
93
if (!$pconfig['timeservers']) {
94
	$pconfig['timeservers'] = "pool.ntp.org";
95
}
96

    
97
$changedesc = gettext("System") . ": ";
98
$changecount = 0;
99

    
100
function is_timezone($elt) {
101
	return !preg_match("/\/$/", $elt);
102
}
103

    
104
if ($pconfig['timezone'] <> $_POST['timezone']) {
105
	filter_pflog_start(true);
106
}
107

    
108
$timezonelist = system_get_timezone_list();
109
$timezonedesc = $timezonelist;
110

    
111
/*
112
 * Etc/GMT entries work the opposite way to what people expect.
113
 * Ref: https://github.com/eggert/tz/blob/master/etcetera and Redmine issue 7089
114
 * Add explanatory text to entries like:
115
 * Etc/GMT+1 and Etc/GMT-1
116
 * but not:
117
 * Etc/GMT or Etc/GMT+0
118
 */
119
foreach ($timezonedesc as $idx => $desc) {
120
	if (substr($desc, 0, 7) != "Etc/GMT" || substr($desc, 8, 1) == "0") {
121
		continue;
122
	}
123

    
124
	$direction = substr($desc, 7, 1);
125

    
126
	switch ($direction) {
127
	case '-':
128
		$direction_str = gettext('AHEAD of');
129
		break;
130
	case '+':
131
		$direction_str = gettext('BEHIND');
132
		break;
133
	default:
134
		continue;
135
	}
136

    
137
	$hr_offset = substr($desc, 8);
138
	$timezonedesc[$idx] = $desc . " " .
139
	    sprintf(ngettext('(%1$s hour %2$s GMT)', '(%1$s hours %2$s GMT)', intval($hr_offset)), $hr_offset, $direction_str);
140
}
141

    
142
$multiwan = 0;
143
$interfaces = get_configured_interface_list();
144
foreach ($interfaces as $interface) {
145
	if (interface_has_gateway($interface)) {
146
		$multiwan++;
147
	}
148
}
149

    
150
if ($_POST) {
151

    
152
	$changecount++;
153

    
154
	unset($input_errors);
155
	$pconfig = $_POST;
156

    
157
	/* input validation */
158
	$reqdfields = explode(" ", "hostname domain");
159
	$reqdfieldsn = array(gettext("Hostname"), gettext("Domain"));
160

    
161
	do_input_validation($_POST, $reqdfields, $reqdfieldsn, $input_errors);
162

    
163
	if ($_POST['hostname']) {
164
		if (!is_hostname($_POST['hostname'])) {
165
			$input_errors[] = gettext("The hostname can only contain the characters A-Z, 0-9 and '-'. It may not start or end with '-'.");
166
		} else {
167
			if (!is_unqualified_hostname($_POST['hostname'])) {
168
				$input_errors[] = gettext("A valid hostname is specified, but the domain name part should be omitted");
169
			}
170
		}
171
	}
172
	if ($_POST['domain'] && !is_domain($_POST['domain'])) {
173
		$input_errors[] = gettext("The domain may only contain the characters a-z, 0-9, '-' and '.'.");
174
	}
175
	validate_webguicss_field($input_errors, $_POST['webguicss']);
176
	validate_webguifixedmenu_field($input_errors, $_POST['webguifixedmenu']);
177
	validate_webguihostnamemenu_field($input_errors, $_POST['webguihostnamemenu']);
178
	validate_dashboardcolumns_field($input_errors, $_POST['dashboardcolumns']);
179

    
180
	$dnslist = $ignore_posted_dnsgw = array();
181

    
182
	$dnscounter = 0;
183
	$dnsname = "dns{$dnscounter}";
184

    
185
	while (isset($_POST[$dnsname])) {
186
		$dnsgwname = "dnsgw{$dnscounter}";
187
		$dnslist[] = $_POST[$dnsname];
188

    
189
		if (($_POST[$dnsname] && !is_ipaddr($_POST[$dnsname]))) {
190
			$input_errors[] = sprintf(gettext("A valid IP address must be specified for DNS server %s."), $dnscounter+1);
191
		} else {
192
			if (($_POST[$dnsgwname] <> "") && ($_POST[$dnsgwname] <> "none")) {
193
				// A real gateway has been selected.
194
				if (is_ipaddr($_POST[$dnsname])) {
195
					if ((is_ipaddrv4($_POST[$dnsname])) && (validate_address_family($_POST[$dnsname], $_POST[$dnsgwname]) === false)) {
196
						$input_errors[] = sprintf(gettext('The IPv6 gateway "%1$s" can not be specified for IPv4 DNS server "%2$s".'), $_POST[$dnsgwname], $_POST[$dnsname]);
197
					}
198
					if ((is_ipaddrv6($_POST[$dnsname])) && (validate_address_family($_POST[$dnsname], $_POST[$dnsgwname]) === false)) {
199
						$input_errors[] = sprintf(gettext('The IPv4 gateway "%1$s" can not be specified for IPv6 DNS server "%2$s".'), $_POST[$dnsgwname], $_POST[$dnsname]);
200
					}
201
				} else {
202
					// The user selected a gateway but did not provide a DNS address. Be nice and set the gateway back to "none".
203
					$ignore_posted_dnsgw[$dnsgwname] = true;
204
				}
205
			}
206
		}
207
		$dnscounter++;
208
		$dnsname = "dns{$dnscounter}";
209
	}
210

    
211
	if (count(array_filter($dnslist)) != count(array_unique(array_filter($dnslist)))) {
212
		$input_errors[] = gettext('Each configured DNS server must have a unique IP address. Remove the duplicated IP.');
213
	}
214

    
215
	$dnscounter = 0;
216
	$dnsname = "dns{$dnscounter}";
217

    
218
	$direct_networks_list = explode(" ", filter_get_direct_networks_list());
219
	while (isset($_POST[$dnsname])) {
220
		$dnsgwname = "dnsgw{$dnscounter}";
221
		if ($_POST[$dnsgwname] && ($_POST[$dnsgwname] <> "none")) {
222
			foreach ($direct_networks_list as $direct_network) {
223
				if (ip_in_subnet($_POST[$dnsname], $direct_network)) {
224
					$input_errors[] = sprintf(gettext("A gateway cannot be specified for %s because that IP address is part of a directly connected subnet %s. To use that nameserver, change its Gateway to `none`."), $_POST[$dnsname], $direct_network);
225
				}
226
			}
227
		}
228
		$dnscounter++;
229
		$dnsname = "dns{$dnscounter}";
230
	}
231

    
232
	# it's easy to have a little too much whitespace in the field, clean it up for the user before processing.
233
	$_POST['timeservers'] = preg_replace('/[[:blank:]]+/', ' ', $_POST['timeservers']);
234
	$_POST['timeservers'] = trim($_POST['timeservers']);
235
	foreach (explode(' ', $_POST['timeservers']) as $ts) {
236
		if (!is_domain($ts)) {
237
			$input_errors[] = gettext("A NTP Time Server name may only contain the characters a-z, 0-9, '-' and '.'.");
238
		}
239
	}
240

    
241
	if ($input_errors) {
242
		// Put the user-entered list back into place so it will be redisplayed for correction.
243
		$pconfig['dnsserver'] = $dnslist;
244
	} else {
245
		// input validation passed, so we can proceed with removing static routes for dead DNS gateways
246
		if (is_array($config['system']['dnsserver'])) {
247
		  	$dns_servers_arr = $config['system']['dnsserver'];
248
	 		foreach ($dns_servers_arr as $arr_index => $this_dnsserver) {
249
	   			$i = (int)$arr_index + 1;
250
	   			$this_dnsgw = $config['system']['dns'.$i.'gw'];
251
				unset($gatewayip);
252
				unset($inet6);
253
				if ((!empty($this_dnsgw)) && ($this_dnsgw != 'none') && (!empty($this_dnsserver))) {
254
					$gatewayip = lookup_gateway_ip_by_name($this_dnsgw);
255
					$inet6 = is_ipaddrv6($gatewayip) ? '-inet6 ' : '';
256
					mwexec("/sbin/route -q delete -host {$inet6}{$this_dnsserver}");
257
				}
258
			}
259
		}
260

    
261
		update_if_changed("hostname", $config['system']['hostname'], $_POST['hostname']);
262
		update_if_changed("domain", $config['system']['domain'], $_POST['domain']);
263
		update_if_changed("timezone", $config['system']['timezone'], $_POST['timezone']);
264
		update_if_changed("NTP servers", $config['system']['timeservers'], strtolower($_POST['timeservers']));
265

    
266
		if ($_POST['language'] && $_POST['language'] != $config['system']['language']) {
267
			$config['system']['language'] = $_POST['language'];
268
			set_language();
269
		}
270

    
271
		unset($config['system']['webgui']['interfacessort']);
272
		$config['system']['webgui']['interfacessort'] = $_POST['interfacessort'] ? true : false;
273

    
274
		unset($config['system']['webgui']['webguileftcolumnhyper']);
275
		$config['system']['webgui']['webguileftcolumnhyper'] = $_POST['webguileftcolumnhyper'] ? true : false;
276

    
277
		unset($config['system']['webgui']['disablealiaspopupdetail']);
278
		$config['system']['webgui']['disablealiaspopupdetail'] = $_POST['disablealiaspopupdetail'] ? true : false;
279

    
280
		unset($config['system']['webgui']['dashboardavailablewidgetspanel']);
281
		$config['system']['webgui']['dashboardavailablewidgetspanel'] = $_POST['dashboardavailablewidgetspanel'] ? true : false;
282

    
283
		unset($config['system']['webgui']['systemlogsfilterpanel']);
284
		$config['system']['webgui']['systemlogsfilterpanel'] = $_POST['systemlogsfilterpanel'] ? true : false;
285

    
286
		unset($config['system']['webgui']['systemlogsmanagelogpanel']);
287
		$config['system']['webgui']['systemlogsmanagelogpanel'] = $_POST['systemlogsmanagelogpanel'] ? true : false;
288

    
289
		unset($config['system']['webgui']['statusmonitoringsettingspanel']);
290
		$config['system']['webgui']['statusmonitoringsettingspanel'] = $_POST['statusmonitoringsettingspanel'] ? true : false;
291

    
292
//		if ($_POST['dashboardperiod']) {
293
//			$config['widgets']['period'] = $_POST['dashboardperiod'];
294
//		}
295

    
296
		if ($_POST['webguicss']) {
297
			$config['system']['webgui']['webguicss'] = $_POST['webguicss'];
298
		} else {
299
			unset($config['system']['webgui']['webguicss']);
300
		}
301

    
302
		$config['system']['webgui']['roworderdragging'] = $_POST['roworderdragging'] ? true:false;
303

    
304
		if ($_POST['logincss']) {
305
			$config['system']['webgui']['logincss'] = $_POST['logincss'];
306
		} else {
307
			unset($config['system']['webgui']['logincss']);
308
		}
309

    
310
		$config['system']['webgui']['loginshowhost'] = $_POST['loginshowhost'] ? true:false;
311

    
312
		if ($_POST['webguifixedmenu']) {
313
			$config['system']['webgui']['webguifixedmenu'] = $_POST['webguifixedmenu'];
314
		} else {
315
			unset($config['system']['webgui']['webguifixedmenu']);
316
		}
317

    
318
		if ($_POST['webguihostnamemenu']) {
319
			$config['system']['webgui']['webguihostnamemenu'] = $_POST['webguihostnamemenu'];
320
		} else {
321
			unset($config['system']['webgui']['webguihostnamemenu']);
322
		}
323

    
324
		if ($_POST['dashboardcolumns']) {
325
			$config['system']['webgui']['dashboardcolumns'] = $_POST['dashboardcolumns'];
326
		} else {
327
			unset($config['system']['webgui']['dashboardcolumns']);
328
		}
329

    
330
		$config['system']['webgui']['requirestatefilter'] = $_POST['requirestatefilter'] ? true : false;
331

    
332
		/* XXX - billm: these still need updating after figuring out how to check if they actually changed */
333
		$olddnsservers = $config['system']['dnsserver'];
334
		unset($config['system']['dnsserver']);
335

    
336
		$dnscounter = 0;
337
		$dnsname = "dns{$dnscounter}";
338

    
339
		while (isset($_POST[$dnsname])) {
340
			if ($_POST[$dnsname]) {
341
				$config['system']['dnsserver'][] = $_POST[$dnsname];
342
			}
343
			$dnscounter++;
344
			$dnsname = "dns{$dnscounter}";
345
		}
346

    
347
		// Remember the new list for display also.
348
		$pconfig['dnsserver'] = $config['system']['dnsserver'];
349

    
350
		$olddnsallowoverride = $config['system']['dnsallowoverride'];
351

    
352
		unset($config['system']['dnsallowoverride']);
353
		$config['system']['dnsallowoverride'] = $_POST['dnsallowoverride'] ? true : false;
354

    
355
		if ($_POST['dnslocalhost'] == "yes") {
356
			$config['system']['dnslocalhost'] = true;
357
		} else {
358
			unset($config['system']['dnslocalhost']);
359
		}
360

    
361
		/* which interface should the dns servers resolve through? */
362
		$dnscounter = 0;
363
		// The $_POST array key of the DNS IP (starts from 0)
364
		$dnsname = "dns{$dnscounter}";
365
		$outdnscounter = 0;
366
		while (isset($_POST[$dnsname])) {
367
			// The $_POST array key of the corresponding gateway (starts from 0)
368
			$dnsgwname = "dnsgw{$dnscounter}";
369
			// The numbering of DNS GW entries in the config starts from 1
370
			$dnsgwconfigcounter = $dnscounter + 1;
371
			// So this is the array key of the DNS GW entry in $config['system']
372
			$dnsgwconfigname = "dns{$dnsgwconfigcounter}gw";
373

    
374
			$olddnsgwname = $config['system'][$dnsgwconfigname];
375

    
376
			if ($ignore_posted_dnsgw[$dnsgwname]) {
377
				$thisdnsgwname = "none";
378
			} else {
379
				$thisdnsgwname = $pconfig[$dnsgwname];
380
			}
381

    
382
			// "Blank" out the settings for this index, then we set them below using the "outdnscounter" index.
383
			$config['system'][$dnsgwconfigname] = "none";
384
			$pconfig[$dnsgwname] = "none";
385
			$pconfig[$dnsname] = "";
386

    
387
			if ($_POST[$dnsname]) {
388
				// Only the non-blank DNS servers were put into the config above.
389
				// So we similarly only add the corresponding gateways sequentially to the config (and to pconfig), as we find non-blank DNS servers.
390
				// This keeps the DNS server IP and corresponding gateway "lined up" when the user blanks out a DNS server IP in the middle of the list.
391

    
392
				// The $pconfig array key of the DNS IP (starts from 0)
393
				$outdnsname = "dns{$outdnscounter}";
394
				// The $pconfig array key of the corresponding gateway (starts from 0)
395
				$outdnsgwname = "dnsgw{$outdnscounter}";
396
				// The numbering of DNS GW entries in the config starts from 1
397
				$outdnsgwconfigcounter = $outdnscounter + 1;
398
				// So this is the array key of the output DNS GW entry in $config['system']
399
				$outdnsgwconfigname = "dns{$outdnsgwconfigcounter}gw";
400

    
401
				$pconfig[$outdnsname] = $_POST[$dnsname];
402
				if ($_POST[$dnsgwname]) {
403
					$config['system'][$outdnsgwconfigname] = $thisdnsgwname;
404
					$pconfig[$outdnsgwname] = $thisdnsgwname;
405
				} else {
406
					// Note: when no DNS GW name is chosen, the entry is set to "none", so actually this case never happens.
407
					unset($config['system'][$outdnsgwconfigname]);
408
					$pconfig[$outdnsgwname] = "";
409
				}
410
				$outdnscounter++;
411
			}
412

    
413
			$dnscounter++;
414
			// The $_POST array key of the DNS IP (starts from 0)
415
			$dnsname = "dns{$dnscounter}";
416
		}
417

    
418
		// clean up dnsgw orphans
419
		$oldgwcounter = 1;
420
		$olddnsgwconfigname = "dns{$oldgwcounter}gw";
421
		while (isset($config['system'][$olddnsgwconfigname])) {
422
			if (empty($config['system']['dnsserver'][$oldgwcounter - 1])) {
423
				unset($config['system'][$olddnsgwconfigname]);
424
			}
425
			$oldgwcounter++;
426
			$olddnsgwconfigname = "dns{$oldgwcounter}gw";
427
		}
428
		unset($oldgwcounter);
429
		unset($olddnsgwconfigname);
430

    
431
		if ($changecount > 0) {
432
			write_config($changedesc);
433
		}
434

    
435
		$changes_applied = true;
436
		$retval = 0;
437
		$retval |= system_hostname_configure();
438
		$retval |= system_hosts_generate();
439
		$retval |= system_resolvconf_generate();
440
		if (isset($config['dnsmasq']['enable'])) {
441
			$retval |= services_dnsmasq_configure();
442
		} elseif (isset($config['unbound']['enable'])) {
443
			$retval |= services_unbound_configure();
444
		}
445
		$retval |= system_timezone_configure();
446
		$retval |= system_ntp_configure();
447

    
448
		if ($olddnsallowoverride != $config['system']['dnsallowoverride']) {
449
			$retval |= send_event("service reload dns");
450
		}
451

    
452
		// Reload the filter - plugins might need to be run.
453
		$retval |= filter_configure();
454
	}
455

    
456
	unset($ignore_posted_dnsgw);
457
}
458

    
459
$pgtitle = array(gettext("System"), gettext("General Setup"));
460
include("head.inc");
461

    
462
if ($input_errors) {
463
	print_input_errors($input_errors);
464
}
465

    
466
if ($changes_applied) {
467
	print_apply_result_box($retval);
468
}
469
?>
470
<div id="container">
471
<?php
472

    
473
$form = new Form;
474
$section = new Form_Section('System');
475
$section->addInput(new Form_Input(
476
	'hostname',
477
	'*Hostname',
478
	'text',
479
	$pconfig['hostname'],
480
	['placeholder' => 'pfSense']
481
))->setHelp('Name of the firewall host, without domain part');
482

    
483
$section->addInput(new Form_Input(
484
	'domain',
485
	'*Domain',
486
	'text',
487
	$pconfig['domain'],
488
	['placeholder' => 'mycorp.com, home, office, private, etc.']
489
))->setHelp('Do not use \'.local\' as the final part of the domain (TLD), The \'.local\' domain is %1$swidely used%2$s by '.
490
	'mDNS (including Avahi and Apple OS X\'s Bonjour/Rendezvous/Airprint/Airplay), and some Windows systems and networked devices. ' .
491
	'These will not network correctly if the router uses \'.local\'. Alternatives such as \'.local.lan\' or \'.mylocal\' are safe.',
492
	 '<a target="_blank" href="https://www.unbound.net/pipermail/unbound-users/2011-March/001735.html">',
493
	 '</a>'
494
);
495

    
496
$form->add($section);
497

    
498
$section = new Form_Section('DNS Server Settings');
499

    
500
if (!is_array($pconfig['dnsserver'])) {
501
	$pconfig['dnsserver'] = array();
502
}
503

    
504
$dnsserver_count = count($pconfig['dnsserver']);
505
$dnsserver_num = 0;
506
$dnsserver_help = gettext("Address") . '<br/>' . gettext("Enter IP addresses to be used by the system for DNS resolution.") . " " .
507
	gettext("These are also used for the DHCP service, DNS Forwarder and DNS Resolver when it has DNS Query Forwarding enabled.");
508
$dnsgw_help = gettext("Gateway") . '<br/>'. gettext("Optionally select the gateway for each DNS server.") . " " .
509
	gettext("When using multiple WAN connections there should be at least one unique DNS server per gateway.");
510

    
511
// If there are no DNS servers, make an empty entry for initial display.
512
if ($dnsserver_count == 0) {
513
	$pconfig['dnsserver'][] = '';
514
}
515

    
516
foreach ($pconfig['dnsserver'] as $dnsserver) {
517

    
518
	$is_last_dnsserver = ($dnsserver_num == $dnsserver_count - 1);
519
	$group = new Form_Group($dnsserver_num == 0 ? 'DNS Servers':'');
520
	$group->addClass('repeatable');
521

    
522
	$group->add(new Form_Input(
523
		'dns' . $dnsserver_num,
524
		'DNS Server',
525
		'text',
526
		$dnsserver
527
	))->setHelp(($is_last_dnsserver) ? $dnsserver_help:null);
528

    
529
	if ($multiwan > 1) {
530
		$options = array('none' => 'none');
531

    
532
		foreach ($arr_gateways as $gwname => $gwitem) {
533
			if ((is_ipaddrv4(lookup_gateway_ip_by_name($pconfig[$dnsgw])) && (is_ipaddrv6($gwitem['gateway'])))) {
534
				continue;
535
			}
536

    
537
			if ((is_ipaddrv6(lookup_gateway_ip_by_name($pconfig[$dnsgw])) && (is_ipaddrv4($gwitem['gateway'])))) {
538
				continue;
539
			}
540

    
541
			$options[$gwname] = $gwname.' - '.$gwitem['friendlyiface'].' - '.$gwitem['gateway'];
542
		}
543

    
544
		$group->add(new Form_Select(
545
			'dnsgw' . $dnsserver_num,
546
			'Gateway',
547
			$pconfig['dnsgw' . $dnsserver_num],
548
			$options
549
		))->setWidth(4)->setHelp(($is_last_dnsserver) ? $dnsgw_help:null);
550
	}
551

    
552
	$group->add(new Form_Button(
553
		'deleterow' . $dnsserver_num,
554
		'Delete',
555
		null,
556
		'fa-trash'
557
	))->setWidth(2)->addClass('btn-warning');
558

    
559
	$section->add($group);
560
	$dnsserver_num++;
561
}
562

    
563
$section->addInput(new Form_Button(
564
	'addrow',
565
	'Add DNS Server',
566
	null,
567
	'fa-plus'
568
))->addClass('btn-success addbtn');
569

    
570
$section->addInput(new Form_Checkbox(
571
	'dnsallowoverride',
572
	'DNS Server Override',
573
	'Allow DNS server list to be overridden by DHCP/PPP on WAN',
574
	$pconfig['dnsallowoverride']
575
))->setHelp('If this option is set, %s will use DNS servers '.
576
	'assigned by a DHCP/PPP server on WAN for its own purposes (including '.
577
	'the DNS Forwarder/DNS Resolver). However, they will not be assigned to DHCP '.
578
	'clients.', $g['product_name']);
579

    
580
$section->addInput(new Form_Checkbox(
581
	'dnslocalhost',
582
	'Disable DNS Forwarder',
583
	'Do not use the DNS Forwarder/DNS Resolver as a DNS server for the firewall',
584
	$pconfig['dnslocalhost']
585
))->setHelp('By default localhost (127.0.0.1) will be used as the first DNS '.
586
	'server where the DNS Forwarder or DNS Resolver is enabled and set to '.
587
	'listen on localhost, so system can use the local DNS service to perform '.
588
	'lookups. Checking this box omits localhost from the list of DNS servers in resolv.conf.');
589

    
590
$form->add($section);
591

    
592
$section = new Form_Section('Localization');
593

    
594
$section->addInput(new Form_Select(
595
	'timezone',
596
	'*Timezone',
597
	$pconfig['timezone'],
598
	array_combine($timezonelist, $timezonedesc)
599
))->setHelp('Select a geographic region name (Continent/Location) to determine the timezone for the firewall. %1$s' .
600
	'Choose a special or "Etc" zone only in cases where the geographic zones do not properly handle the clock offset required for this firewall.', '<br/>');
601

    
602
$section->addInput(new Form_Input(
603
	'timeservers',
604
	'Timeservers',
605
	'text',
606
	$pconfig['timeservers']
607
))->setHelp('Use a space to separate multiple hosts (only one required). '.
608
	'Remember to set up at least one DNS server if a host name is entered here!');
609

    
610
$section->addInput(new Form_Select(
611
	'language',
612
	'*Language',
613
	$pconfig['language'],
614
	get_locale_list()
615
))->setHelp('Choose a language for the webConfigurator');
616

    
617
$form->add($section);
618

    
619
$section = new Form_Section('webConfigurator');
620

    
621
gen_webguicss_field($section, $pconfig['webguicss']);
622
gen_webguifixedmenu_field($section, $pconfig['webguifixedmenu']);
623
gen_webguihostnamemenu_field($section, $pconfig['webguihostnamemenu']);
624
gen_dashboardcolumns_field($section, $pconfig['dashboardcolumns']);
625
gen_interfacessort_field($section, $pconfig['interfacessort']);
626
gen_associatedpanels_fields(
627
	$section,
628
	$pconfig['dashboardavailablewidgetspanel'],
629
	$pconfig['systemlogsfilterpanel'],
630
	$pconfig['systemlogsmanagelogpanel'],
631
	$pconfig['statusmonitoringsettingspanel']);
632
gen_requirestatefilter_field($section, $pconfig['requirestatefilter']);
633
gen_webguileftcolumnhyper_field($section, $pconfig['webguileftcolumnhyper']);
634
gen_disablealiaspopupdetail_field($section, $pconfig['disablealiaspopupdetail']);
635

    
636
$section->addInput(new Form_Checkbox(
637
	'roworderdragging',
638
	'Disable dragging',
639
	'Disable dragging of firewall/NAT rules',
640
	$pconfig['roworderdragging']
641
))->setHelp('Disables dragging rows to allow selecting and copying row contents and avoid accidental changes.');
642

    
643
$section->addInput(new Form_Select(
644
	'logincss',
645
	'Login page color',
646
	$pconfig['logincss'],
647
	["1e3f75;" => gettext("Blue"), "003300" => gettext("Green"), "770101" => gettext("Red"),
648
	 "4b1263" => gettext("Purple"), "424142" => gettext("Gray"), "333333" => gettext("Dark gray"),
649
	 "633215" => gettext("Brown" ), "bf7703" => gettext("Orange")]
650
))->setHelp('Choose a color for the login page');
651

    
652
$section->addInput(new Form_Checkbox(
653
	'loginshowhost',
654
	'Login hostname',
655
	'Show hostname on login banner',
656
	$pconfig['loginshowhost']
657
));
658
/*
659
$section->addInput(new Form_Input(
660
	'dashboardperiod',
661
	'Dashboard update period',
662
	'number',
663
	$pconfig['dashboardperiod'],
664
	['min' => '5', 'max' => '600']
665
))->setHelp('Time in seconds between dashboard widget updates. Small values cause ' .
666
			'more frequent updates but increase the load on the web server. ' .
667
			'Minimum is 5 seconds, maximum 600 seconds');
668
*/
669
$form->add($section);
670

    
671
print $form;
672

    
673
$csswarning = sprintf(gettext("%sUser-created themes are unsupported, use at your own risk."), "<br />");
674

    
675
?>
676
</div>
677

    
678
<script type="text/javascript">
679
//<![CDATA[
680
events.push(function() {
681

    
682
	function setThemeWarning() {
683
		if ($('#webguicss').val().startsWith("pfSense")) {
684
			$('#csstxt').html("").addClass("text-default");
685
		} else {
686
			$('#csstxt').html("<?=$csswarning?>").addClass("text-danger");
687
		}
688
	}
689

    
690
	$('#webguicss').change(function() {
691
		setThemeWarning();
692
	});
693

    
694
	setThemeWarning();
695

    
696
	// Suppress "Delete row" button if there are fewer than two rows
697
	checkLastRow();
698
});
699
//]]>
700
</script>
701

    
702
<?php
703
include("foot.inc");
704
?>
(192-192/234)