"") && is_ipaddr($_POST['gateway']) && $_POST['gateway'] != "dynamic") { if (is_ipaddrv6($_POST['gateway']) && ($_POST['ipprotocol'] == "inet")) { $input_errors[] = sprintf(gettext("The IPv6 gateway address '%s' can not be used as a IPv4 gateway."), $_POST['gateway']); } if (is_ipaddrv4($_POST['gateway']) && ($_POST['ipprotocol'] == "inet6")) { $input_errors[] = sprintf(gettext("The IPv4 gateway address '%s' can not be used as a IPv6 gateway."), $_POST['gateway']); } } /* only allow correct IPv4 and IPv6 monitor addresses */ if (($_POST['monitor'] <> "") && is_ipaddr($_POST['monitor']) && $_POST['monitor'] != "dynamic") { if (is_ipaddrv6($_POST['monitor']) && ($_POST['ipprotocol'] == "inet")) { $input_errors[] = sprintf(gettext("The IPv6 monitor address '%s' can not be used on a IPv4 gateway."), $_POST['monitor']); } if (is_ipaddrv4($_POST['monitor']) && ($_POST['ipprotocol'] == "inet6")) { $input_errors[] = sprintf(gettext("The IPv4 monitor address '%s' can not be used on a IPv6 gateway."), $_POST['monitor']); } } if (isset($_POST['name'])) { /* check for overlaps */ foreach ($a_gateways as $gateway) { if (isset($id) && ($a_gateways[$id]) && ($a_gateways[$id] === $gateway)) { if ($gateway['name'] != $_POST['name']) { $input_errors[] = gettext("Changing name on a gateway is not allowed."); } continue; } if ($_POST['name'] <> "") { if (($gateway['name'] <> "") && ($_POST['name'] == $gateway['name']) && ($gateway['attribute'] !== "system")) { $input_errors[] = sprintf(gettext('The gateway name "%s" already exists.'), $_POST['name']); break; } } if (is_ipaddr($_POST['gateway'])) { if (($gateway['gateway'] <> "") && ($_POST['gateway'] == $gateway['gateway']) && ($gateway['attribute'] !== "system")) { $input_errors[] = sprintf(gettext('The gateway IP address "%s" already exists.'), $_POST['gateway']); break; } } if (is_ipaddr($_POST['monitor'])) { if (($gateway['monitor'] <> "") && ($_POST['monitor'] == $gateway['monitor']) && ($gateway['attribute'] !== "system")) { $input_errors[] = sprintf(gettext('The monitor IP address "%s" is already in use. A different monitor IP must be chosen.'), $_POST['monitor']); break; } } } } /* input validation of dpinger advanced parameters */ $latencylow = $dpinger_default['latencylow']; if ($_POST['latencylow']) { if (!is_numeric($_POST['latencylow'])) { $input_errors[] = gettext("The low latency threshold needs to be a numeric value."); } else if ($_POST['latencylow'] < 1) { $input_errors[] = gettext("The low latency threshold needs to be positive."); } else { $latencylow = $_POST['latencylow']; } } $latencyhigh = $dpinger_default['latencyhigh']; if ($_POST['latencyhigh']) { if (!is_numeric($_POST['latencyhigh'])) { $input_errors[] = gettext("The high latency threshold needs to be a numeric value."); } else if ($_POST['latencyhigh'] < 1) { $input_errors[] = gettext("The high latency threshold needs to be positive."); } else { $latencyhigh = $_POST['latencyhigh']; } } $losslow = $dpinger_default['losslow']; if ($_POST['losslow']) { if (!is_numeric($_POST['losslow'])) { $input_errors[] = gettext("The low Packet Loss threshold needs to be a numeric value."); } else if ($_POST['losslow'] < 1) { $input_errors[] = gettext("The low Packet Loss threshold needs to be positive."); } else if ($_POST['losslow'] >= 100) { $input_errors[] = gettext("The low Packet Loss threshold needs to be less than 100."); } else { $losslow = $_POST['losslow']; } } $losshigh = $dpinger_default['losshigh']; if ($_POST['losshigh']) { if (!is_numeric($_POST['losshigh'])) { $input_errors[] = gettext("The high Packet Loss threshold needs to be a numeric value."); } else if ($_POST['losshigh'] < 1) { $input_errors[] = gettext("The high Packet Loss threshold needs to be positive."); } else if ($_POST['losshigh'] > 100) { $input_errors[] = gettext("The high Packet Loss threshold needs to be 100 or less."); } else { $losshigh = $_POST['losshigh']; } } $time_period = $dpinger_default['time_period']; if ($_POST['time_period']) { if (!is_numeric($_POST['time_period'])) { $input_errors[] = gettext("The time period over which results are averaged needs to be a numeric value."); } else if ($_POST['time_period'] < 1) { $input_errors[] = gettext("The time period over which results are averaged needs to be positive."); } else { $time_period = $_POST['time_period']; } } $interval = $dpinger_default['interval']; if ($_POST['interval']) { if (!is_numeric($_POST['interval'])) { $input_errors[] = gettext("The probe interval needs to be a numeric value."); } else if ($_POST['interval'] < 1) { $input_errors[] = gettext("The probe interval needs to be positive."); } else { $interval = $_POST['interval']; } } $loss_interval = $dpinger_default['loss_interval']; if ($_POST['loss_interval']) { if (!is_numeric($_POST['loss_interval'])) { $input_errors[] = gettext("The loss interval needs to be a numeric value."); } else if ($_POST['loss_interval'] < 1) { $input_errors[] = gettext("The loss interval setting needs to be positive."); } else { $loss_interval = $_POST['loss_interval']; } } $alert_interval = $dpinger_default['alert_interval']; if ($_POST['alert_interval']) { if (!is_numeric($_POST['alert_interval'])) { $input_errors[] = gettext("The alert interval needs to be a numeric value."); } else if ($_POST['alert_interval'] < 1) { $input_errors[] = gettext("The alert interval setting needs to be positive."); } else { $alert_interval = $_POST['alert_interval']; } } if ($latencylow >= $latencyhigh) { $input_errors[] = gettext( "The high latency threshold needs to be greater than the low latency threshold"); } if ($losslow >= $losshigh) { $input_errors[] = gettext( "The high packet loss threshold needs to be higher than the low packet loss threshold"); } // If the loss interval is less than latencyhigh, then high latency could never be recorded // because those high latency packets would be considered as lost. So do not allow that. if ($latencyhigh > $loss_interval) { $input_errors[] = gettext("The loss interval needs to be greater than or equal to the high latency threshold."); } // Ensure that the time period is greater than 2 times the probe interval plus the loss interval. if (($interval * 2 + $loss_interval) >= $time_period) { $input_errors[] = gettext("The time period needs to be greater than twice the probe interval plus the loss interval."); } // There is no point recalculating the average latency and loss more often than the probe interval. // So the alert interval needs to be >= probe interval. if ($interval > $alert_interval) { $input_errors[] = gettext("The alert interval needs to be greater than or equal to the probe interval."); } if (!$input_errors) { $reloadif = ""; $gateway = array(); if (empty($_POST['interface'])) { $gateway['interface'] = $pconfig['friendlyiface']; } else { $gateway['interface'] = $_POST['interface']; } if (is_ipaddr($_POST['gateway'])) { $gateway['gateway'] = $_POST['gateway']; } else { $gateway['gateway'] = "dynamic"; } $gateway['name'] = $_POST['name']; $gateway['weight'] = $_POST['weight']; $gateway['ipprotocol'] = $_POST['ipprotocol']; if ($_POST['interval']) { $gateway['interval'] = $_POST['interval']; } if ($_POST['time_period']) { $gateway['time_period'] = $_POST['time_period']; } if ($_POST['alert_interval']) { $gateway['alert_interval'] = $_POST['alert_interval']; } $gateway['descr'] = $_POST['descr']; if ($_POST['monitor_disable'] == "yes") { $gateway['monitor_disable'] = true; } if ($_POST['nonlocalgateway'] == "yes") { $gateway['nonlocalgateway'] = true; } if ($_POST['force_down'] == "yes") { $gateway['force_down'] = true; } if (is_ipaddr($_POST['monitor'])) { $gateway['monitor'] = $_POST['monitor']; } if (isset($_POST['data_payload']) && $_POST['data_payload'] > 0) { $gateway['data_payload'] = $_POST['data_payload']; } /* NOTE: If gateway ip is changed need to cleanup the old static interface route */ if ($_POST['monitor'] != "dynamic" && !empty($a_gateway_item[$realid]) && is_ipaddr($a_gateway_item[$realid]['gateway']) && $gateway['gateway'] != $a_gateway_item[$realid]['gateway'] && isset($a_gateway_item[$realid]["nonlocalgateway"])) { $realif = get_real_interface($a_gateway_item[$realid]['interface']); $inet = (!is_ipaddrv4($a_gateway_item[$realid]['gateway']) ? "-inet6" : "-inet"); $cmd = "/sbin/route delete $inet " . escapeshellarg($a_gateway_item[$realid]['gateway']) . " -iface " . escapeshellarg($realif); mwexec($cmd); } /* NOTE: If monitor ip is changed need to cleanup the old static route */ if ($_POST['monitor'] != "dynamic" && !empty($a_gateway_item[$realid]) && is_ipaddr($a_gateway_item[$realid]['monitor']) && $_POST['monitor'] != $a_gateway_item[$realid]['monitor'] && $gateway['gateway'] != $a_gateway_item[$realid]['monitor']) { if (is_ipaddrv4($a_gateway_item[$realid]['monitor'])) { mwexec("/sbin/route delete " . escapeshellarg($a_gateway_item[$realid]['monitor'])); } else { mwexec("/sbin/route delete -inet6 " . escapeshellarg($a_gateway_item[$realid]['monitor'])); } } if ($_POST['defaultgw'] == "yes" || $_POST['defaultgw'] == "on") { $i = 0; /* remove the default gateway bits for all gateways with the same address family */ foreach ($a_gateway_item as $gw) { if ($gateway['ipprotocol'] == $gw['ipprotocol']) { unset($config['gateways']['gateway_item'][$i]['defaultgw']); if ($gw['interface'] != $_POST['interface'] && $gw['defaultgw']) { $reloadif = $gw['interface']; } } $i++; } $gateway['defaultgw'] = true; } if ($_POST['latencylow']) { $gateway['latencylow'] = $_POST['latencylow']; } if ($_POST['latencyhigh']) { $gateway['latencyhigh'] = $_POST['latencyhigh']; } if ($_POST['losslow']) { $gateway['losslow'] = $_POST['losslow']; } if ($_POST['losshigh']) { $gateway['losshigh'] = $_POST['losshigh']; } if ($_POST['loss_interval']) { $gateway['loss_interval'] = $_POST['loss_interval']; } if (isset($_POST['disabled'])) { $gateway['disabled'] = true; } else { unset($gateway['disabled']); } /* when saving the manual gateway we use the attribute which has the corresponding id */ if (isset($realid) && $a_gateway_item[$realid]) { $a_gateway_item[$realid] = $gateway; } else { $a_gateway_item[] = $gateway; } mark_subsystem_dirty('staticroutes'); write_config(); if ($_REQUEST['isAjax']) { echo $_POST['name']; exit; } else if (!empty($reloadif)) { send_event("interface reconfigure {$reloadif}"); } header("Location: system_gateways.php"); exit; } else { if ($_REQUEST['isAjax']) { header("HTTP/1.0 500 Internal Server Error"); header("Content-type: text/plain"); foreach ($input_errors as $error) { echo("$error\n"); } exit; } $pconfig = $_POST; if (empty($_POST['friendlyiface'])) { $pconfig['friendlyiface'] = $_POST['interface']; } } } $pgtitle = array(gettext("System"), gettext("Routing"), gettext("Gateways"), gettext("Edit")); $shortcut_section = "gateways"; include("head.inc"); if ($input_errors) { print_input_errors($input_errors); } $form = new Form; /* If this is a system gateway we need this var */ if (($pconfig['attribute'] == "system") || is_numeric($pconfig['attribute'])) { $form->addGlobal(new Form_Input( 'attribute', null, 'hidden', $pconfig['attribute'] )); } if (isset($id) && $a_gateways[$id]) { $form->addGlobal(new Form_Input( 'id', null, 'hidden', $id )); } $form->addGlobal(new Form_Input( 'friendlyiface', null, 'hidden', $pconfig['friendlyiface'] )); $section = new Form_Section('Edit Gateway'); $section->addInput(new Form_Checkbox( 'disabled', 'Disabled', 'Disable this gateway', $pconfig['disabled'] ))->setHelp('Set this option to disable this gateway without removing it from the '. 'list.'); $section->addInput(new Form_Select( 'interface', 'Interface', $pconfig['friendlyiface'], get_configured_interface_with_descr(false, true) ))->setHelp('Choose which interface this gateway applies to.'); $section->addInput(new Form_Select( 'ipprotocol', 'Address Family', $pconfig['ipprotocol'], array( "inet" => "IPv4", "inet6" => "IPv6" ) ))->setHelp('Choose the Internet Protocol this gateway uses.'); $section->addInput(new Form_Input( 'name', 'Name', 'text', $pconfig['name'] ))->setHelp('Gateway name'); $egw = new Form_Input( 'gateway', 'Gateway', 'text', ($pconfig['dynamic'] ? 'dynamic' : $pconfig['gateway']) ); $egw->setHelp('Gateway IP address'); if ($pconfig['dynamic']) { $egw->setReadonly(); } $section->addInput($egw); $section->addInput(new Form_Checkbox( 'defaultgw', 'Default Gateway', 'This will select the above gateway as the default gateway.', $pconfig['defaultgw'] )); $section->addInput(new Form_Checkbox( 'monitor_disable', 'Gateway Monitoring', 'Disable Gateway Monitoring', $pconfig['monitor_disable'] ))->toggles('.toggle-monitor-ip')->setHelp('This will consider this gateway as always being up.'); $group = new Form_Group('Monitor IP'); $group->addClass('toggle-monitor-ip', 'collapse'); if (!$pconfig['monitor_disable']) $group->addClass('in'); $group->add(new Form_Input( 'monitor', null, 'text', ($pconfig['gateway'] == $pconfig['monitor'] ? '' : $pconfig['monitor']) ))->setHelp('Enter an alternative address here to be '. 'used to monitor the link. This is used for the quality RRD graphs as well as the '. 'load balancer entries. Use this if the gateway does not respond to ICMP echo '. 'requests (pings).'); $section->add($group); $section->addInput(new Form_Checkbox( 'force_down', 'Force state', 'Mark Gateway as Down', $pconfig['force_down'] ))->setHelp('This will force this gateway to be considered down.'); $section->addInput(new Form_Input( 'descr', 'Description', 'text', $pconfig['descr'] ))->setHelp('A description may be entered here for reference (not parsed).'); // Add a button to provide access to the advanced fields $btnadv = new Form_Button( 'btnadvopts', 'Display Advanced', null, 'fa-cog' ); $btnadv->setAttribute('type','button')->addClass('btn-info btn-sm'); $section->addInput(new Form_StaticText( null, $btnadv )); $form->add($section); $section = new Form_Section('Advanced'); $section->addClass('adnlopts'); $section->addInput(new Form_Select( 'weight', 'Weight', $pconfig['weight'], array_combine(range(1, 30), range(1, 30)) ))->setHelp('Weight for this gateway when used in a Gateway Group.'); $section->addInput(new Form_Input( 'data_payload', 'Data Payload', 'number', $pconfig['data_payload'], ['placeholder' => $dpinger_default['data_payload']] ))->setHelp('Define data payload to send on ICMP packets to gateway monitor IP.'); $group = new Form_Group('Latency thresholds'); $group->add(new Form_Input( 'latencylow', 'From', 'number', $pconfig['latencylow'], ['placeholder' => $dpinger_default['latencylow']] )); $group->add(new Form_Input( 'latencyhigh', 'To', 'number', $pconfig['latencyhigh'], ['placeholder' => $dpinger_default['latencyhigh']] )); $group->setHelp('Low and high thresholds for latency in milliseconds. Default is %d/%d.', [$dpinger_default['latencylow'], $dpinger_default['latencyhigh']]); $section->add($group); $group = new Form_Group('Packet Loss thresholds'); $group->add(new Form_Input( 'losslow', 'From', 'number', $pconfig['losslow'], ['placeholder' => $dpinger_default['losslow']] )); $group->add(new Form_Input( 'losshigh', 'To', 'number', $pconfig['losshigh'], ['placeholder' => $dpinger_default['losshigh']] )); $group->setHelp('Low and high thresholds for packet loss in %%. Default is %d/%d.', [$dpinger_default['losslow'], $dpinger_default['losshigh']]); $section->add($group); $section->addInput(new Form_Input( 'interval', 'Probe Interval', 'number', $pconfig['interval'], [ 'placeholder' => $dpinger_default['interval'], 'max' => 86400 ] ))->setHelp('How often an ICMP probe will be sent in milliseconds. Default is %d.', [$dpinger_default['interval']]); $section->addInput(new Form_Input( 'loss_interval', 'Loss Interval', 'number', $pconfig['loss_interval'], ['placeholder' => $dpinger_default['loss_interval']] ))->setHelp('Time interval in milliseconds before packets are treated as lost. '. 'Default is %d.', [$dpinger_default['loss_interval']]); $group = new Form_Group('Time Period'); $group->add(new Form_Input( 'time_period', null, 'number', $pconfig['time_period'], [ 'placeholder' => $dpinger_default['time_period'] ] )); $group->setHelp('Time period in milliseconds over which results are averaged. Default is %d.', [$dpinger_default['time_period']]); $section->add($group); $group = new Form_Group('Alert interval'); $group->add(new Form_Input( 'alert_interval', null, 'number', $pconfig['alert_interval'], [ 'placeholder' => $dpinger_default['alert_interval'] ] )); $group->setHelp('Time interval in milliseconds between checking for an alert condition. Default is %d.', [$dpinger_default['alert_interval']]); $section->add($group); $section->addInput(new Form_StaticText( gettext('Additional information'), ''. gettext('The time period, probe interval and loss interval are closely related. The ' . 'ratio between these values control the accuracy of the numbers reported and ' . 'the timeliness of alerts.') . '

' . gettext('A longer time period will provide smoother results for round trip time ' . 'and loss, but will increase the time before a latency or loss alert is triggered.') . '

' . gettext('A shorter probe interval will decrease the time required before a latency ' . 'or loss alert is triggered, but will use more network resource. Longer ' . 'probe intervals will degrade the accuracy of the quality graphs.') . '

' . gettext('The ratio of the probe interval to the time period (minus the loss interval) ' . 'also controls the resolution of loss reporting. To determine the resolution, ' . 'the following formula can be used:') . '

' . gettext('    100 * probe interval / (time period - loss interval)') . '

' . gettext('Rounding up to the nearest whole number will yield the resolution of loss ' . 'reporting in percent. The default values provide a resolution of 1%.') . '

' . gettext('The default settings are recommended for most use cases. However if ' . 'changing the settings, please observe the following restrictions:') . '

' . gettext('- The time period must be greater than twice the probe interval plus the loss ' . 'interval. This guarantees there is at least one completed probe at all times. ') . '

' . gettext('- The alert interval must be greater than or equal to the probe interval. There ' . 'is no point checking for alerts more often than probes are done.') . '

' . gettext('- The loss interval must be greater than or equal to the high latency threshold.') . '
' )); $section->addInput(new Form_Checkbox( 'nonlocalgateway', 'Use non-local gateway', 'Use non-local gateway through interface specific route.', $pconfig['nonlocalgateway'] ))->setHelp('This will allow use of a gateway outside of this interface\'s subnet. This is usually indicative of a configuration error, but is required for some scenarios.'); $form->add($section); print $form; ?>