diff options
Diffstat (limited to 'config/dyntables')
-rw-r--r-- | config/dyntables/pkg/dyntables.inc | 117 | ||||
-rw-r--r-- | config/dyntables/pkg/dyntables.xml | 139 | ||||
-rw-r--r-- | config/dyntables/pkg/dyntables_classdefs.inc | 146 | ||||
-rw-r--r-- | config/dyntables/www/js/dyntables.js | 287 | ||||
-rw-r--r-- | config/dyntables/www/php/diag_dhcp_leases.php | 440 |
5 files changed, 1129 insertions, 0 deletions
diff --git a/config/dyntables/pkg/dyntables.inc b/config/dyntables/pkg/dyntables.inc new file mode 100644 index 00000000..97e8a422 --- /dev/null +++ b/config/dyntables/pkg/dyntables.inc @@ -0,0 +1,117 @@ +<?php +/* $Id$ */ +/* ========================================================================== */ +/* + dyntables.inc + part of pfSense (http://www.pfSense.com) + Copyright (C) 2007 Daniel S. Haischt <me@daniel.stefan.haischt.name> + All rights reserved. + + Based on m0n0wall (http://m0n0.ch/wall) + Copyright (C) 2003-2006 Manuel Kasper <mk@neon1.net>. + All rights reserved. + */ +/* ========================================================================== */ +/* + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + + THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, + INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY + AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, + OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + POSSIBILITY OF SUCH DAMAGE. + */ +/* ========================================================================== */ + +require_once("dyntables_classdefs.inc"); + +function syncPackageDyntables() { + +} + +function installPackageDyntables() { + /* move temp files */ + @rename('/usr/local/www/diag_dhcp_leases.php', '/usr/local/pkg/diag_dhcp_leases.php.org'); + @rename('/usr/local/pkg/dyntables.js', '/usr/local/www/javascript/dyntables.js'); + @rename('/usr/local/pkg/diag_dhcp_leases.php', '/usr/local/www/diag_dhcp_leases.php'); +} + +function deinstallPackageDyntables() { + /* move backup files */ + @unlink('/usr/local/www/diag_dhcp_leases.php'); + @unlink('/usr/local/www/javascript/dyntables.js'); + @rename('/usr/local/pkg/diag_dhcp_leases.php.org', '/usr/local/www/diag_dhcp_leases.php'); +} + +function checkForExtension($alertClass) { + if(! extension_loaded( 'json' )) { + if(! @dl( 'json.so' )) { + return openNoExtDialog($alertClass); + } + } +} + +/** + * openNoExtDialog() + * + * @param mixed $effectClass + * @return + */ +function openNoExtDialog($effectClass) { + $alertMessage = gettext("PHP extension json.so was not found or could not be loaded. Please check you PHP installation."); + $dialogScript = " + <script type='text/javascript'> + var anchor = document.getElementById('popupanchor'); + + function openNoExtDialog(html) { + var effect = new PopupEffect(html, {className: '${effectClass}'}); + Dialog.alert('${alertMessage}',{className:'alphacube', top:150, width:400, height:null, showEffect:effect.show.bind(effect), hideEffect:effect.hide.bind(effect)}); + } + + openNoExtDialog(anchor); + </script> + "; + + return $dialogScript; +} + +/** +* getWindowJSScriptRefs() +* +* @return an array of <script /> elements +*/ +function getWindowJSScriptRefs(){ + $result = array('<script type="text/javascript" src="/javascript/windows-js/javascript/window.js"></script>', + '<script type="text/javascript" src="/javascript/windows-js/javascript/window_effects.js"></script>', + '<script type="text/javascript" src="/javascript/windows-js/javascript/debug.js"></script>'); + + return $result; +} + +/** +* getWindowJSStyleRefs() +* +* @return an array of <style /> tags +*/ +function getWindowJSStyleRefs(){ + $result = array('<link href="/javascript/windows-js/themes/default.css" rel="stylesheet" type="text/css" />', + '<link href="/javascript/windows-js/themes/alert.css" rel="stylesheet" type="text/css" />', + '<link href="/javascript/windows-js/themes/alphacube.css" rel="stylesheet" type="text/css" />'); + + return $result; +} + +?>
\ No newline at end of file diff --git a/config/dyntables/pkg/dyntables.xml b/config/dyntables/pkg/dyntables.xml new file mode 100644 index 00000000..8a249966 --- /dev/null +++ b/config/dyntables/pkg/dyntables.xml @@ -0,0 +1,139 @@ +<?xml version="1.0" encoding="utf-8" ?> +<!DOCTYPE packagegui SYSTEM "../../schema/packages.dtd"> +<?xml-stylesheet type="text/xsl" href="../../xsl/package.xsl"?> +<packagegui> + <copyright> + <![CDATA[ +/* $Id$ */ +/* ========================================================================== */ +/* + dyntables.xml + part of pfSense (http://www.pfSense.com) + Copyright (C) 2007 Daniel S. Haischt <me@daniel.stefan.haischt.name> + All rights reserved. + + Based on m0n0wall (http://m0n0.ch/wall) + Copyright (C) 2003-2006 Manuel Kasper <mk@neon1.net>. + All rights reserved. + */ +/* ========================================================================== */ +/* + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + + THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, + INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY + AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, + OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + POSSIBILITY OF SUCH DAMAGE. + */ +/* ========================================================================== */ + ]]> + </copyright> + <description> + This package adds dynamic reloading support of table data to + various pfSense web pages. Reloading table data is done using + Ajax. + </description> + <requirements> + This package is supposed to be run on RELENG based pfSense systems. + </requirements> + <faq>Currently there are no FAQ items provided.</faq> + <name>dyntables</name> + <version>1.0</version> + <title>System: Dynamic Tables</title> + <include_file>/usr/local/pkg/dyntables.inc</include_file> + <!-- Menu is where this packages menu will appear --> + <!-- + <menu> + <name>Menu Name</name> + <section>Menu Section</section> + <url>/phpfile.php</url> + </menu> + --> + <!-- + <service> + <name>yourservice</name> + <rcfile>/usr/local/etc/rc.d/yourservice.sh</rcfile> + </service> + --> + <tabs /> + <!-- + configpath gets expanded out automatically and config items + will be stored in that location + --> + <configpath>['installedpackages']['dyntables']['config']</configpath> + <!-- + | + | PHP files (user management) + | + --> + <additional_files_needed> + <prefix>/usr/local/pkg/</prefix> + <chmod>0755</chmod> + <item>http://www.pfsense.com/packages/config/dyntables/www/php/diag_dhcp_leases.php</item> + </additional_files_needed> + <!-- + | + | JavaScript files (*.js and *.inc.) + | + --> + <additional_files_needed> + <prefix>/usr/local/pkg/</prefix> + <chmod>0755</chmod> + <item>http://www.pfsense.com/packages/config/dyntables/www/js/dyntables.js</item> + </additional_files_needed> + <!-- + | + | Include files (class defs etc.) + | + --> + <additional_files_needed> + <prefix>/usr/local/pkg/</prefix> + <chmod>0755</chmod> + <item>http://www.pfsense.com/packages/config/dyntables/pkg/dyntables.inc</item> + </additional_files_needed> + <additional_files_needed> + <prefix>/usr/local/pkg/</prefix> + <chmod>0755</chmod> + <item>http://www.pfsense.com/packages/config/dyntables/pkg/dyntables_classdefs.inc</item> + </additional_files_needed> + <!-- + | + | Binary files + | + --> + <additional_files_needed> + <prefix>/usr/local/lib/php/extensions/no-debug-non-zts-20020429/</prefix> + <chmod>0755</chmod> + <item>http://www.pfsense.com/packages/config/dyntables/bin/json.so</item> + </additional_files_needed> + <!-- + fields gets invoked when the user adds or edits a item. The following items + will be parsed and rendered for the user as a gui with input, and selectboxes. + --> + <!-- + Arbitrary PHP Code, that gets executed if a certain event gets triggered. + --> + <custom_php_resync_config_command> + syncPackageDyntables(); + </custom_php_resync_config_command> + <custom_php_install_command> + installPackageDyntables(); + </custom_php_install_command> + <custom_php_deinstall_command> + deinstallPackageDyntables(); + </custom_php_deinstall_command> +</packagegui> diff --git a/config/dyntables/pkg/dyntables_classdefs.inc b/config/dyntables/pkg/dyntables_classdefs.inc new file mode 100644 index 00000000..696170d1 --- /dev/null +++ b/config/dyntables/pkg/dyntables_classdefs.inc @@ -0,0 +1,146 @@ +<?php +/* $Id$ */ +/* ========================================================================== */ +/* + classdefs.inc + part of pfSense (http://www.pfSense.com) + Copyright (C) 2007 Daniel S. Haischt <me@daniel.stefan.haischt.name> + All rights reserved. + + Based on m0n0wall (http://m0n0.ch/wall) + Copyright (C) 2003-2006 Manuel Kasper <mk@neon1.net>. + All rights reserved. + */ +/* ========================================================================== */ +/* + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + + THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, + INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY + AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, + OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + POSSIBILITY OF SUCH DAMAGE. + */ +/* ========================================================================== */ + +if (! class_exists("Object")) { + class Object { + function Object() { + $args = func_get_args(); + if (method_exists($this, '__destruct')) { + register_shutdown_function(array(&$this, '__destruct')); + } + call_user_func_array(array(&$this, '__construct'), $args); + } + + function __construct() { + } + } +} + +class Column extends Object { + var $defaultClass = "listhdrr"; + var $class = null; + var $name = null; + var $sortID = null; + + function __construct($name) { + // Perform object initialization here. + parent::__construct(); + $this->name = $name; + } + + function setClass($class) { + $this->class = $class; + } + + function getClass() { + return $this->class; + } + + function setSortID($id) { + $this->sortID = $id; + } + + function getSortID() { + return $this->sortID; + } + + function toHTML() { + global $_GET; + + $classString = $this->class != null ? $this->class : $this->defaultClass; + $allString = isset($_GET['all']) ? ", {$_GET['all']}" : ""; + return "<td class='{$classString}'><a id='{$this->sortID}' href='#' onclick='sortTable(this{$allString});'>{$this->name}</a></td>"; + } +} + +class Table extends Object { + var $columns = null; + var $data = null; + + function __construct() { + // Perform object initialization here. + parent::__construct(); + $this->columns = array(); + $this->data = array(); + } + + function addColumn($column, $id = null, $class = null) { + $column = new Column($column); + + if ($id != null) { + $column->setSortID($id); + } + if ($class != null) { + $column->setClass($class); + } + + $this->columns[] = $column; + } + + function addDataSet($dataSet) { + if (is_array($dataSet)) { + if (count($dataSet) == count($this->columns)) { + $this->data[] = $dataSet; + } else { + trigger_error("Dataset does not match column size."); + } + } else { + trigger_error("Adding non arrays as datasets to a table is not supported."); + } + } + + function setData(&$data) { + $this->data = $data; + } + + function getData() { + return $this->data; + } + + function getColumnHTML() { + $htmlString = ""; + + foreach($this->columns as $column) { + $htmlString .= $column->toHtml(); + } + + return "<tr>{$htmlString}</tr>"; + } +} + +?>
\ No newline at end of file diff --git a/config/dyntables/www/js/dyntables.js b/config/dyntables/www/js/dyntables.js new file mode 100644 index 00000000..0e90a3c1 --- /dev/null +++ b/config/dyntables/www/js/dyntables.js @@ -0,0 +1,287 @@ +/* $Id$ */ +/* ========================================================================== */ +/* + authng_wizard.js + part of pfSense (http://www.pfSense.com) + Copyright (C) 2007 Daniel S. Haischt <me@daniel.stefan.haischt.name> + All rights reserved. + + Based on m0n0wall (http://m0n0.ch/wall) + Copyright (C) 2003-2006 Manuel Kasper <mk@neon1.net>. + All rights reserved. + */ +/* ========================================================================== */ +/* + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + + THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, + INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY + AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, + OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + POSSIBILITY OF SUCH DAMAGE. + */ +/* ========================================================================== */ + +function openInfoDialog() { + Dialog.info("Reloading table data ...", + {className:"alphacube", width:200, height:75, top:100, showProgress: true}); +} + +/** + * + * @access public + * @return void + **/ +function getFirstTD(node, index){ + var firstTD = node.childNodes[index]; + + if (firstTD.tagName == 'TD') { + return firstTD; + } else { + return getFirstTD(node, ++index); + } +} + +/** + * + * @access public + * @return void + **/ +function getTableRow(nodes, index){ + var row = nodes[index]; + + if (row.tagName == 'TR') { + return row; + } else { + return getTableRow(nodes, ++index); + } +} + +/** + * + * @access public + * @return void + **/ +function emtyDhcpLeaseTable(table) { + var tbody = table.childNodes[1]; + var trNodes = tbody.childNodes; + + for (i = 0; i < trNodes.length; i++) { + var currentRow = getTableRow(trNodes, i); + var firstTd = getFirstTD(currentRow, 0); + var classAttrib = firstTd.className; + + if (classAttrib == 'listhdrr') { + continue; + } else if (currentRow.tagName == 'TR') { + tbody.removeChild(currentRow); + } + } +} + +/** + * TODO: data[if], $fspans, $fspane, + * + **/ +function dhcpLeaseTableToHTML(table, json, theme) { + var tbody = table.childNodes[1]; + + if (json && json.length > 0) { + for (i = 0; i < json.length; i++) { + /* text nodes */ + var newActTxt = document.createTextNode(json[i]['act']); + var newEndTxt = document.createTextNode(json[i]['end']); + var newStartTxt = document.createTextNode(json[i]['start']); + var newHostnameTxt = document.createTextNode(json[i]['hostname']); + var newIPTxt = document.createTextNode(json[i]['ip']); + var newMACTxt = document.createTextNode(json[i]['mac']); + var newOnlineTxt = document.createTextNode(json[i]['online']); + var newTypeTxt = document.createTextNode(json[i]['type']); + + var newTR = document.createElement("tr"); + var newTRIdAttrib = document.createAttribute("id"); + newTRIdAttrib.nodeValue = "dhcpRow" + i; + newTR.setAttributeNode(newTRIdAttrib); + + /* IP td element */ + var newIPTD = document.createElement("td"); + /* TD attributes */ + var newIPTDClassAttrib = document.createAttribute("class"); + newIPTDClassAttrib.nodeValue = "listlr"; + /* assign attribs */ + newIPTD.setAttributeNode(newIPTDClassAttrib); + + /* MAC td element */ + var newMACTD = document.createElement("td"); + /* TD attributes */ + var newMACTDClassAttrib = document.createAttribute("class"); + newMACTDClassAttrib.nodeValue = "listr"; + /* assign attribs */ + newMACTD.setAttributeNode(newMACTDClassAttrib); + + /* Hostname td element */ + var newHostnameTD = document.createElement("td"); + /* TD attributes */ + var newHostnameTDClassAttrib = document.createAttribute("class"); + newHostnameTDClassAttrib.nodeValue = "listr"; + /* assign attribs */ + newHostnameTD.setAttributeNode(newHostnameTDClassAttrib); + + /* Start td element */ + var newStartTD = document.createElement("td"); + /* TD attributes */ + var newStartTDClassAttrib = document.createAttribute("class"); + newStartTDClassAttrib.nodeValue = "listr"; + /* assign attribs */ + newStartTD.setAttributeNode(newStartTDClassAttrib); + + /* End td element */ + var newEndTD = document.createElement("td"); + /* TD attributes */ + var newEndTDClassAttrib = document.createAttribute("class"); + newEndTDClassAttrib.nodeValue = "listr"; + /* assign attribs */ + newEndTD.setAttributeNode(newEndTDClassAttrib); + + /* Online td element */ + var newOnlineTD = document.createElement("td"); + /* TD attributes */ + var newOnlineTDClassAttrib = document.createAttribute("class"); + newOnlineTDClassAttrib.nodeValue = "listr"; + /* assign attribs */ + newOnlineTD.setAttributeNode(newOnlineTDClassAttrib); + + /* Lease td element */ + var newLeaseTD = document.createElement("td"); + /* TD attributes */ + var newLeaseTDClassAttrib = document.createAttribute("class"); + newLeaseTDClassAttrib.nodeValue = "listr"; + /* assign attribs */ + newLeaseTD.setAttributeNode(newLeaseTDClassAttrib); + + /* Mapping td element */ + var newMappingTD = document.createElement("td"); + /* TD attributes */ + var newMappingTDClassAttrib = document.createAttribute("class"); + newMappingTDClassAttrib.nodeValue = "list"; + var newMappingTDValignAttrib = document.createAttribute("valign"); + newMappingTDValignAttrib.nodeValue = "middle"; + /* assign attribs */ + newMappingTD.setAttributeNode(newMappingTDClassAttrib); + newMappingTD.setAttributeNode(newMappingTDValignAttrib); + + /* WOL td element */ + var newWOLTD = document.createElement("td"); + /* TD attributes */ + var newWOLTDValignAttrib = document.createAttribute("valign"); + newWOLTDValignAttrib.nodeValue = "middle"; + /* assign attribs */ + newWOLTD.setAttributeNode(newWOLTDValignAttrib); + + /* Mapping anchor */ + var newMappingAnchor = document.createElement("a"); + /* Anchor attribs */ + var newMappingAnchorHrefAttrib = document.createAttribute("href"); + newMappingAnchorHrefAttrib.nodeValue = "services_dhcp_edit.php?if=lan&mac=" + json[i]['mac'] + "&hostname=" + json[i]['hostname']; + /* assign attribs */ + newMappingAnchor.setAttributeNode(newMappingAnchorHrefAttrib); + + /* Mapping button */ + var newMappingButton = document.createElement("img"); + /* Button attribs */ + var newMappingButtonSrcAttrib = document.createAttribute("src"); + newMappingButtonSrcAttrib.nodeValue = "/themes/" + theme + "/images/icons/icon_plus.gif"; + var newMappingButtonWidthAttrib = document.createAttribute("width"); + newMappingButtonWidthAttrib.nodeValue = "17"; + var newMappingButtonHeightAttrib = document.createAttribute("height"); + newMappingButtonHeightAttrib.nodeValue = "17"; + var newMappingButtonBorderAttrib = document.createAttribute("border"); + newMappingButtonBorderAttrib.nodeValue = "0"; + var newMappingButtonTitleAttrib = document.createAttribute("title"); + newMappingButtonTitleAttrib.nodeValue = "add a static mapping for this MAC address"; + var newMappingButtonAltAttrib = document.createAttribute("alt"); + newMappingButtonAltAttrib.nodeValue = "add a static mapping for this MAC address"; + /* assign attribs */ + newMappingButton.setAttributeNode(newMappingButtonSrcAttrib); + newMappingButton.setAttributeNode(newMappingButtonWidthAttrib); + newMappingButton.setAttributeNode(newMappingButtonHeightAttrib); + newMappingButton.setAttributeNode(newMappingButtonBorderAttrib); + newMappingButton.setAttributeNode(newMappingButtonTitleAttrib); + newMappingButton.setAttributeNode(newMappingButtonAltAttrib); + + /* WOL anchor */ + var newWOLAnchor = document.createElement("a"); + /* Anchor attribs */ + var newWOLAnchorHrefAttrib = document.createAttribute("href"); + newWOLAnchorHrefAttrib.nodeValue = "services_wol_edit.php?if=lan&mac=" + json[i]['mac'] + "&descr=pfSense"; + /* assign attribs */ + newWOLAnchor.setAttributeNode(newWOLAnchorHrefAttrib); + + /* WOL button */ + var newWOLButton = document.createElement("img"); + /* Button attribs */ + var newWOLButtonSrcAttrib = document.createAttribute("src"); + newWOLButtonSrcAttrib.nodeValue = "/themes/" + theme + "/images/icons/icon_wol_all.gif"; + var newWOLButtonWidthAttrib = document.createAttribute("width"); + newWOLButtonWidthAttrib.nodeValue = "17"; + var newWOLButtonHeightAttrib = document.createAttribute("height"); + newWOLButtonHeightAttrib.nodeValue = "17"; + var newWOLButtonBorderAttrib = document.createAttribute("border"); + newWOLButtonBorderAttrib.nodeValue = "0"; + var newWOLButtonTitleAttrib = document.createAttribute("title"); + newWOLButtonTitleAttrib.nodeValue = "add a Wake on Lan mapping for this MAC address"; + var newWOLButtonAltAttrib = document.createAttribute("alt"); + newWOLButtonAltAttrib.nodeValue = "add a Wake on Lan mapping for this MAC address"; + /* assign attribs */ + newWOLButton.setAttributeNode(newWOLButtonSrcAttrib); + newWOLButton.setAttributeNode(newWOLButtonWidthAttrib); + newWOLButton.setAttributeNode(newWOLButtonHeightAttrib); + newWOLButton.setAttributeNode(newWOLButtonBorderAttrib); + newWOLButton.setAttributeNode(newWOLButtonTitleAttrib); + newWOLButton.setAttributeNode(newWOLButtonAltAttrib); + + /* assign buttons to anchor elements */ + newMappingAnchor.appendChild(newMappingButton); + newWOLAnchor.appendChild(newWOLButton); + + /* assign anchors to TD elements */ + newMappingTD.appendChild(newMappingAnchor); + newWOLTD.appendChild(newWOLAnchor); + + /* assign text nodes to TD elements */ + newIPTD.appendChild(newIPTxt); + newMACTD.appendChild(newMACTxt); + newHostnameTD.appendChild(newHostnameTxt); + newStartTD.appendChild(newStartTxt); + newEndTD.appendChild(newEndTxt); + newOnlineTD.appendChild(newOnlineTxt); + newLeaseTD.appendChild(newActTxt); + + /* populate table body */ + newTR.appendChild(newIPTD); + newTR.appendChild(newMACTD); + newTR.appendChild(newHostnameTD); + newTR.appendChild(newStartTD); + newTR.appendChild(newEndTD); + newTR.appendChild(newOnlineTD); + newTR.appendChild(newLeaseTD); + newTR.appendChild(newMappingTD); + newTR.appendChild(newWOLTD); + + tbody.appendChild(newTR); + } + } +}
\ No newline at end of file diff --git a/config/dyntables/www/php/diag_dhcp_leases.php b/config/dyntables/www/php/diag_dhcp_leases.php new file mode 100644 index 00000000..4ff8e241 --- /dev/null +++ b/config/dyntables/www/php/diag_dhcp_leases.php @@ -0,0 +1,440 @@ +<?php +/* $Id$ */ +/* + diag_dhcp_leases.php + Copyright (C) 2004 Scott Ullrich + All rights reserved. + + originially part of m0n0wall (http://m0n0.ch/wall) + Copyright (C) 2003-2004 Manuel Kasper <mk@neon1.net>. + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + + THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, + INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY + AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, + OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + POSSIBILITY OF SUCH DAMAGE. +*/ + +require("guiconfig.inc"); +require("dyntables.inc"); + +/* ========================================================================== */ +/* ======================= NON GUI RELATED FUNCTIONS ======================== */ +/* ========================================================================== */ + +function leasecmp($a, $b) { + return strcmp($a[$_GET['order']], $b[$_GET['order']]); +} + +function adjust_gmt($dt) { + $ts = strtotime($dt . " GMT"); + return strftime("%Y/%m/%d %H:%M:%S", $ts); +} + +function remove_duplicate($array, $field) +{ + foreach ($array as $sub) + $cmp[] = $sub[$field]; + $unique = array_unique($cmp); + foreach ($unique as $k => $rien) + $new[] = $array[$k]; + return $new; +} + +$leasesfile = "{$g['dhcpd_chroot_path']}/var/db/dhcpd.leases"; +$awk = "/usr/bin/awk"; +/* this pattern sticks comments into a single array item */ +$cleanpattern = "'{ gsub(\"#.*\", \"\");} { gsub(\";\", \"\"); print;}'"; +/* We then split the leases file by } */ +$splitpattern = "'BEGIN { RS=\"}\";} {for (i=1; i<=NF; i++) printf \"%s \", \$i; printf \"}\\n\";}'"; + +/* stuff the leases file in a proper format into a array by line */ +exec("cat {$leasesfile} | {$awk} {$cleanpattern} | {$awk} {$splitpattern}", $leases_content); +$leases_count = count($leases_content); + +$pools = array(); +$leases = array(); +$i = 0; +$l = 0; +$p = 0; + +// Put everything together again +while($i < $leases_count) { + /* split the line by space */ + $data = explode(" ", $leases_content[$i]); + /* walk the fields */ + $f = 0; + $fcount = count($data); + /* with less then 20 fields there is nothing useful */ + if($fcount < 20) { + $i++; + continue; + } + while($f < $fcount) { + switch($data[$f]) { + case "failover": + $pools[$p]['name'] = $data[$f+2]; + $pools[$p]['mystate'] = $data[$f+7]; + $pools[$p]['peerstate'] = $data[$f+14]; + $pools[$p]['mydate'] = $data[$f+10]; + $pools[$p]['mydate'] .= " " . $data[$f+11]; + $pools[$p]['peerdate'] = $data[$f+17]; + $pools[$p]['peerdate'] .= " " . $data[$f+18]; + $p++; + $i++; + continue 3; + case "lease": + $leases[$l]['ip'] = $data[$f+1]; + $leases[$l]['type'] = "dynamic"; + $f = $f+2; + break; + case "starts": + $leases[$l]['start'] = $data[$f+2]; + $leases[$l]['start'] .= " " . $data[$f+3]; + $f = $f+3; + break; + case "ends": + $leases[$l]['end'] = $data[$f+2]; + $leases[$l]['end'] .= " " . $data[$f+3]; + $f = $f+3; + break; + case "tstp": + $f = $f+3; + break; + case "tsfp": + $f = $f+3; + break; + case "atsfp": + $f = $f+3; + break; + case "cltt": + $f = $f+3; + break; + case "binding": + switch($data[$f+2]) { + case "active": + $leases[$l]['act'] = "active"; + break; + case "free": + $leases[$l]['act'] = "expired"; + $leases[$l]['online'] = "offline"; + break; + case "backup": + $leases[$l]['act'] = "reserved"; + $leases[$l]['online'] = "offline"; + break; + } + $f = $f+1; + break; + case "next": + /* skip the next binding statement */ + $f = $f+3; + break; + case "hardware": + $leases[$l]['mac'] = $data[$f+2]; + /* check if it's online and the lease is active */ + if($leases[$l]['act'] == "active") { + $online = exec("/usr/sbin/arp -an |/usr/bin/awk '/{$leases[$l]['ip']}/ {print}'|wc -l"); + if ($online == 1) { + $leases[$l]['online'] = 'online'; + } else { + $leases[$l]['online'] = 'offline'; + } + } + $f = $f+2; + break; + case "client-hostname": + if($data[$f+1] <> "") { + $leases[$l]['hostname'] = preg_replace('/"/','',$data[$f+1]); + } else { + $hostname = gethostbyaddr($leases[$l]['ip']); + if($hostname <> "") { + $leases[$l]['hostname'] = $hostname; + } + } + $f = $f+1; + break; + case "uid": + $f = $f+1; + break; + } + $f++; + } + $l++; + $i++; +} + +/* remove duplicate items by mac address */ +if(count($leases) > 0) { + $leases = remove_duplicate($leases,"ip"); +} + +if(count($pools) > 0) { + $pools = remove_duplicate($pools,"name"); + asort($pools); +} + +foreach($config['interfaces'] as $ifname => $ifarr) { + if (is_array($config['dhcpd'][$ifname]['staticmap'])) { + foreach($config['dhcpd'][$ifname]['staticmap'] as $static) { + $slease = array(); + $slease['ip'] = $static['ipaddr']; + $slease['type'] = "static"; + $slease['mac'] = $static['mac']; + $slease['start'] = gmdate("M d Y H:i:s", time()); + $slease['end'] = gmdate("M d Y H:i:s", time()); + $slease['end'] = gmdate("M d Y H:i:s", strtotime('+5 minutes')); + $slease['hostname'] = $static['hostname']; + $slease['act'] = "static"; + $online = exec("/usr/sbin/arp -an |/usr/bin/grep {$slease['mac']}| /usr/bin/wc -l|/usr/bin/awk '{print $1;}'"); + if ($online == 1) { + $slease['online'] = 'online'; + } else { + $slease['online'] = 'offline'; + } + $leases[] = $slease; + } + } +} + +/* ========================================================================== */ +/* ====================== PROCESS POST AND GET VARS ========================= */ +/* ========================================================================== */ + +if ($_GET['order']) { + usort($leases, "leasecmp"); +} else if ($_GET['refresh']) { + dl( 'json.so' ); + echo json_encode($leases); + exit; +} + +/* ========================================================================== */ +/* =================== GUI RELATED FUNCTIONS AND STUFF ====================== */ +/* ========================================================================== */ + +$pgtitle = "Diagnostics: DHCP leases"; + +$effectStyle = ' + <style type="text/css"> + .popup_dialog { + background: #000000; + opacity: 0.2; + } + </style> +'; + +include("head.inc"); + +// add extra stuff to HTML head +if (isset($pfSenseHead)) { + $pfSenseHead->setCloseHead(false); + print($pfSenseHead->getHTML()); +} else { + $closehead = false; +} + +$scriptaculousRefs = ' + <script type="text/javascript" src="/javascript/scriptaculous/prototype.js"></script> + <script type="text/javascript" src="/javascript/scriptaculous/scriptaculous.js"></script> +'; + +$ajaxRequest = " +<script type='text/javascript' src='/javascript/dyntables.js'></script> +<script type='text/javascript'> +<!-- + function refreshTable(column, all) { + var myAjax = new Ajax.Request( + '{$_SERVER['SCRIPT_NAME']}', + { + method: 'get', + parameters: { refresh: 'true' }, + onLoading: openInfoDialog, + onSuccess: function(transport) { + var json = transport.responseText.evalJSON(); + var table = document.getElementById('sortabletable'); + + emtyDhcpLeaseTable(table); + dhcpLeaseTableToHTML(table, json, '{$g['theme']}'); + Dialog.closeInfo(); + } + }); + } +//--> +</script> +"; + +print($ajaxRequest); +print($scriptaculousRefs); + +foreach(getWindowJSScriptRefs() as $jscript){ + print("{$jscript}\n"); +} + +foreach(getWindowJSStyleRefs() as $style){ + print("{$style}\n"); +} + +print($effectStyle); + +?> +<script type="text/javascript" src="/javascript/sorttable.js"></script> +</head> + +<body link="#0000CC" vlink="#0000CC" alink="#0000CC"> +<?php include("fbegin.inc"); ?> +<p class="pgtitle"><?=$pgtitle?></p> +<?php +/* only print pool status when we have one */ +if(count($pools) > 0) { +?> +<table class="sortable" id="sortabletable" name="sortabletable" width="100%" border="0" cellpadding="0" cellspacing="0"> + <tr> + <td class="listhdrr">Failover Group</a></td> + <td class="listhdrr">My State</a></td> + <td class="listhdrr">Since</a></td> + <td class="listhdrr">Peer State</a></td> + <td class="listhdrr">Since</a></td> + </tr> +<?php +foreach ($pools as $data) { + echo "<tr>\n"; + echo "<td class=\"listlr\">{$fspans}{$data['name']}{$fspane} </td>\n"; + echo "<td class=\"listr\">{$fspans}{$data['mystate']}{$fspane} </td>\n"; + echo "<td class=\"listr\">{$fspans}" . adjust_gmt($data['mydate']) . "{$fspane} </td>\n"; + echo "<td class=\"listr\">{$fspans}{$data['peerstate']}{$fspane} </td>\n"; + echo "<td class=\"listr\">{$fspans}" . adjust_gmt($data['peerdate']) . "{$fspane} </td>\n"; + echo "<td class=\"list\" valign=\"middle\" width=\"17\"> </td>\n"; + echo "<td class=\"list\" valign=\"middle\" width=\"17\"> </td>\n"; + echo "</tr>\n"; +} + +?> +</table> + +<?php +/* only print pool status when we have one */ +} + +$table = new Table(); + +// assemble columns +$table->addColumn("IP address", "ip"); +$table->addColumn("MAC address", "mac"); +$table->addColumn("Hostname", "hostname"); +$table->addColumn("Start", "start"); +$table->addColumn("End", "end"); +$table->addColumn("Online", "online"); +$table->addColumn("Lease Type", "act", "listhdr"); + +// populate table data +$table->setData($leases); + +?> + +<p> + +<table class="sortable" id="sortabletable" name="sortabletable" width="100%" border="0" cellpadding="0" cellspacing="0"> +<?= $table->getColumnHTML(); ?> +<?php +foreach ($table->getData() as $data) { + if (($data['act'] == "active") || ($data['act'] == "static") || ($_GET['all'] == 1)) { + if ($data['act'] != "active" && $data['act'] != "static") { + $fspans = "<span class=\"gray\">"; + $fspane = "</span>"; + } else { + $fspans = $fspane = ""; + } + $lip = ip2long($data['ip']); + if ($data['act'] == "static") { + foreach ($config['dhcpd'] as $dhcpif => $dhcpifconf) { + if(is_array($dhcpifconf['staticmap'])) { + foreach ($dhcpifconf['staticmap'] as $staticent) { + if ($data['ip'] == $staticent['ipaddr']) { + $data['if'] = $dhcpif; + break; + } + } + } + /* exit as soon as we have an interface */ + if ($data['if'] != "") + break; + } + } else { + foreach ($config['dhcpd'] as $dhcpif => $dhcpifconf) { + if (($lip >= ip2long($dhcpifconf['range']['from'])) && ($lip <= ip2long($dhcpifconf['range']['to']))) { + $data['if'] = $dhcpif; + break; + } + } + } + echo "<tr>\n"; + echo "<td class=\"listlr\">{$fspans}{$data['ip']}{$fspane} </td>\n"; + if ($data['online'] != "online") { + echo "<td class=\"listr\">{$fspans}<a href=\"services_wol.php?if={$data['if']}&mac={$data['mac']}\" title=\"send Wake on Lan packet to mac\">{$data['mac']}</a>{$fspane} </td>\n"; + } else { + echo "<td class=\"listr\">{$fspans}{$data['mac']}{$fspane} </td>\n"; + } + echo "<td class=\"listr\">{$fspans}{$data['hostname']}{$fspane} </td>\n"; + echo "<td class=\"listr\">{$fspans}" . adjust_gmt($data['start']) . "{$fspane} </td>\n"; + echo "<td class=\"listr\">{$fspans}" . adjust_gmt($data['end']) . "{$fspane} </td>\n"; + echo "<td class=\"listr\">{$fspans}{$data['online']}{$fspane} </td>\n"; + echo "<td class=\"listr\">{$fspans}{$data['act']}{$fspane} </td>\n"; + + if ($data['type'] == "dynamic") { + echo "<td class=\"list\" valign=\"middle\"><a href=\"services_dhcp_edit.php?if={$data['if']}&mac={$data['mac']}&hostname={$data['hostname']}\">"; + echo "<img src=\"/themes/{$g['theme']}/images/icons/icon_plus.gif\" width=\"17\" height=\"17\" border=\"0\" title=\"add a static mapping for this MAC address\"></a></td>\n"; + } else { + echo "<td class=\"list\" valign=\"middle\">"; + echo "<img src=\"/themes/{$g['theme']}/images/icons/icon_plus_mo.gif\" width=\"17\" height=\"17\" border=\"0\"></td>\n"; + } + + echo "<td valign=\"middle\"><a href=\"services_wol_edit.php?if={$data['if']}&mac={$data['mac']}&descr={$data['hostname']}\">"; + echo "<img src=\"/themes/{$g['theme']}/images/icons/icon_wol_all.gif\" width=\"17\" height=\"17\" border=\"0\" title=\"add a Wake on Lan mapping for this MAC address\"></a></td>\n"; + echo "</tr>\n"; + } +} + +?> +</table> +<p> +<form action="diag_dhcp_leases.php" method="GET"> +<input type="hidden" name="order" value="<?=$_GET['order'];?>"> +<?php if ($_GET['all']): ?> +<input type="hidden" name="all" value="0"> +<input type="submit" class="formbtn" value="Show active and static leases only"> +<?php else: ?> +<input type="hidden" name="all" value="1"> +<input type="submit" class="formbtn" value="Show all configured leases"> +<?php endif; ?> +</form> +<?php if($leases == 0): ?> +<p><strong>No leases file found. Is the DHCP server active?</strong></p> +<?php endif; ?> + +<div id="popupanchor"> </div> +<?= checkForExtension("popup_dialog"); ?> +<?php include("fend.inc"); ?> +<script type="text/javascript"> +<!-- + window.setInterval("refreshTable()", 20000); +//--> +</script> +</body> +</html> |