mirror of
				https://github.com/Ysurac/openmptcprouter-feeds.git
				synced 2025-03-09 15:40:03 +00:00 
			
		
		
		
	Fix adding DNS server in interface
This commit is contained in:
		
							parent
							
								
									b6c11bc989
								
							
						
					
					
						commit
						e85a4e766e
					
				
					 11 changed files with 4401 additions and 0 deletions
				
			
		
							
								
								
									
										17
									
								
								luci-mod-network/Makefile
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								luci-mod-network/Makefile
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,17 @@ | |||
| #
 | ||||
| # Copyright (C) 2008-2014 The LuCI Team <luci@lists.subsignal.org>
 | ||||
| #
 | ||||
| # This is free software, licensed under the Apache License, Version 2.0 .
 | ||||
| #
 | ||||
| 
 | ||||
| include $(TOPDIR)/rules.mk | ||||
| 
 | ||||
| LUCI_TITLE:=LuCI Network Administration | ||||
| LUCI_DEPENDS:=+luci-base +libiwinfo-lua +rpcd-mod-iwinfo | ||||
| 
 | ||||
| PKG_LICENSE:=Apache-2.0 | ||||
| 
 | ||||
| include $(TOPDIR)/feeds/luci/luci.mk | ||||
| 
 | ||||
| # call BuildPackage - OpenWrt buildroot signature
 | ||||
| 
 | ||||
|  | @ -0,0 +1,552 @@ | |||
| 'use strict'; | ||||
| 'require rpc'; | ||||
| 'require uci'; | ||||
| 'require form'; | ||||
| 'require validation'; | ||||
| 
 | ||||
| var callHostHints, callDUIDHints, callDHCPLeases, CBILeaseStatus, CBILease6Status; | ||||
| 
 | ||||
| callHostHints = rpc.declare({ | ||||
| 	object: 'luci-rpc', | ||||
| 	method: 'getHostHints', | ||||
| 	expect: { '': {} } | ||||
| }); | ||||
| 
 | ||||
| callDUIDHints = rpc.declare({ | ||||
| 	object: 'luci-rpc', | ||||
| 	method: 'getDUIDHints', | ||||
| 	expect: { '': {} } | ||||
| }); | ||||
| 
 | ||||
| callDHCPLeases = rpc.declare({ | ||||
| 	object: 'luci-rpc', | ||||
| 	method: 'getDHCPLeases', | ||||
| 	expect: { '': {} } | ||||
| }); | ||||
| 
 | ||||
| CBILeaseStatus = form.DummyValue.extend({ | ||||
| 	renderWidget: function(section_id, option_id, cfgvalue) { | ||||
| 		return E([ | ||||
| 			E('h4', _('Active DHCP Leases')), | ||||
| 			E('div', { 'id': 'lease_status_table', 'class': 'table' }, [ | ||||
| 				E('div', { 'class': 'tr table-titles' }, [ | ||||
| 					E('div', { 'class': 'th' }, _('Hostname')), | ||||
| 					E('div', { 'class': 'th' }, _('IPv4-Address')), | ||||
| 					E('div', { 'class': 'th' }, _('MAC-Address')), | ||||
| 					E('div', { 'class': 'th' }, _('Lease time remaining')) | ||||
| 				]), | ||||
| 				E('div', { 'class': 'tr placeholder' }, [ | ||||
| 					E('div', { 'class': 'td' }, E('em', _('Collecting data...'))) | ||||
| 				]) | ||||
| 			]) | ||||
| 		]); | ||||
| 	} | ||||
| }); | ||||
| 
 | ||||
| CBILease6Status = form.DummyValue.extend({ | ||||
| 	renderWidget: function(section_id, option_id, cfgvalue) { | ||||
| 		return E([ | ||||
| 			E('h4', _('Active DHCPv6 Leases')), | ||||
| 			E('div', { 'id': 'lease6_status_table', 'class': 'table' }, [ | ||||
| 				E('div', { 'class': 'tr table-titles' }, [ | ||||
| 					E('div', { 'class': 'th' }, _('Host')), | ||||
| 					E('div', { 'class': 'th' }, _('IPv6-Address')), | ||||
| 					E('div', { 'class': 'th' }, _('DUID')), | ||||
| 					E('div', { 'class': 'th' }, _('Leasetime remaining')) | ||||
| 				]), | ||||
| 				E('div', { 'class': 'tr placeholder' }, [ | ||||
| 					E('div', { 'class': 'td' }, E('em', _('Collecting data...'))) | ||||
| 				]) | ||||
| 			]) | ||||
| 		]); | ||||
| 	} | ||||
| }); | ||||
| 
 | ||||
| function validateHostname(sid, s) { | ||||
| 	if (s.length > 256) | ||||
| 		return _('Expecting: %s').format(_('valid hostname')); | ||||
| 
 | ||||
| 	var labels = s.replace(/^\.+|\.$/g, '').split(/\./); | ||||
| 
 | ||||
| 	for (var i = 0; i < labels.length; i++) | ||||
| 		if (!labels[i].match(/^[a-z0-9_](?:[a-z0-9-]{0,61}[a-z0-9])?$/i)) | ||||
| 			return _('Expecting: %s').format(_('valid hostname')); | ||||
| 
 | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| function validateAddressList(sid, s) { | ||||
| 	if (s == null || s == '') | ||||
| 		return true; | ||||
| 
 | ||||
| 	var m = s.match(/^\/(.+)\/$/), | ||||
| 	    names = m ? m[1].split(/\//) : [ s ]; | ||||
| 
 | ||||
| 	for (var i = 0; i < names.length; i++) { | ||||
| 		var res = validateHostname(sid, names[i]); | ||||
| 
 | ||||
| 		if (res !== true) | ||||
| 			return res; | ||||
| 	} | ||||
| 
 | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| function validateServerSpec(sid, s) { | ||||
| 	if (s == null || s == '') | ||||
| 		return true; | ||||
| 
 | ||||
| 	var m = s.match(/^\/(.+)\/(.*)$/); | ||||
| 	if (!m) | ||||
| 		return _('Expecting: %s').format(_('valid hostname')); | ||||
| 
 | ||||
| 	var res = validateAddressList(sid, m[1]); | ||||
| 	if (res !== true) | ||||
| 		return res; | ||||
| 
 | ||||
| 	if (m[2] == '' || m[2] == '#') | ||||
| 		return true; | ||||
| 
 | ||||
| 	// ipaddr%scopeid#srvport@source@interface#srcport
 | ||||
| 
 | ||||
| 	m = m[2].match(/^([0-9a-f:.]+)(?:%[^#@]+)?(?:#(\d+))?(?:@([0-9a-f:.]+)(?:@[^#]+)?(?:#(\d+))?)?$/); | ||||
| 
 | ||||
| 	if (!m) | ||||
| 		return _('Expecting: %s').format(_('valid IP address')); | ||||
| 	else if (validation.parseIPv4(m[1]) && m[3] != null && !validation.parseIPv4(m[3])) | ||||
| 		return _('Expecting: %s').format(_('valid IPv4 address')); | ||||
| 	else if (validation.parseIPv6(m[1]) && m[3] != null && !validation.parseIPv6(m[3])) | ||||
| 		return _('Expecting: %s').format(_('valid IPv6 address')); | ||||
| 	else if ((m[2] != null && +m[2] > 65535) || (m[4] != null && +m[4] > 65535)) | ||||
| 		return _('Expecting: %s').format(_('valid port value')); | ||||
| 
 | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| return L.view.extend({ | ||||
| 	load: function() { | ||||
| 		return Promise.all([ | ||||
| 			callHostHints(), | ||||
| 			callDUIDHints() | ||||
| 		]); | ||||
| 	}, | ||||
| 
 | ||||
| 	render: function(hosts_duids) { | ||||
| 		var has_dhcpv6 = L.hasSystemFeature('dnsmasq', 'dhcpv6') || L.hasSystemFeature('odhcpd'), | ||||
| 		    hosts = hosts_duids[0], | ||||
| 		    duids = hosts_duids[1], | ||||
| 		    m, s, o, ss, so; | ||||
| 
 | ||||
| 		m = new form.Map('dhcp', _('DHCP and DNS'), _('Dnsmasq is a combined <abbr title="Dynamic Host Configuration Protocol">DHCP</abbr>-Server and <abbr title="Domain Name System">DNS</abbr>-Forwarder for <abbr title="Network Address Translation">NAT</abbr> firewalls')); | ||||
| 
 | ||||
| 		s = m.section(form.TypedSection, 'dnsmasq', _('Server Settings')); | ||||
| 		s.anonymous = true; | ||||
| 		s.addremove = false; | ||||
| 
 | ||||
| 		s.tab('general', _('General Settings')); | ||||
| 		s.tab('files', _('Resolv and Hosts Files')); | ||||
| 		s.tab('tftp', _('TFTP Settings')); | ||||
| 		s.tab('advanced', _('Advanced Settings')); | ||||
| 		s.tab('leases', _('Static Leases')); | ||||
| 
 | ||||
| 		s.taboption('general', form.Flag, 'domainneeded', | ||||
| 			_('Domain required'), | ||||
| 			_('Don\'t forward <abbr title="Domain Name System">DNS</abbr>-Requests without <abbr title="Domain Name System">DNS</abbr>-Name')); | ||||
| 
 | ||||
| 		s.taboption('general', form.Flag, 'authoritative', | ||||
| 			_('Authoritative'), | ||||
| 			_('This is the only <abbr title="Dynamic Host Configuration Protocol">DHCP</abbr> in the local network')); | ||||
| 
 | ||||
| 
 | ||||
| 		s.taboption('files', form.Flag, 'readethers', | ||||
| 			_('Use <code>/etc/ethers</code>'), | ||||
| 			_('Read <code>/etc/ethers</code> to configure the <abbr title="Dynamic Host Configuration Protocol">DHCP</abbr>-Server')); | ||||
| 
 | ||||
| 		s.taboption('files', form.Value, 'leasefile', | ||||
| 			_('Leasefile'), | ||||
| 			_('file where given <abbr title="Dynamic Host Configuration Protocol">DHCP</abbr>-leases will be stored')); | ||||
| 
 | ||||
| 		s.taboption('files', form.Flag, 'noresolv', | ||||
| 			_('Ignore resolve file')).optional = true; | ||||
| 
 | ||||
| 		o = s.taboption('files', form.Value, 'resolvfile', | ||||
| 			_('Resolve file'), | ||||
| 			_('local <abbr title="Domain Name System">DNS</abbr> file')); | ||||
| 
 | ||||
| 		o.depends('noresolv', ''); | ||||
| 		o.optional = true; | ||||
| 
 | ||||
| 
 | ||||
| 		s.taboption('files', form.Flag, 'nohosts', | ||||
| 			_('Ignore <code>/etc/hosts</code>')).optional = true; | ||||
| 
 | ||||
| 		s.taboption('files', form.DynamicList, 'addnhosts', | ||||
| 			_('Additional Hosts files')).optional = true; | ||||
| 
 | ||||
| 		o = s.taboption('advanced', form.Flag, 'quietdhcp', | ||||
| 			_('Suppress logging'), | ||||
| 			_('Suppress logging of the routine operation of these protocols')); | ||||
| 		o.optional = true; | ||||
| 
 | ||||
| 		o = s.taboption('advanced', form.Flag, 'sequential_ip', | ||||
| 			_('Allocate IP sequentially'), | ||||
| 			_('Allocate IP addresses sequentially, starting from the lowest available address')); | ||||
| 		o.optional = true; | ||||
| 
 | ||||
| 		o = s.taboption('advanced', form.Flag, 'boguspriv', | ||||
| 			_('Filter private'), | ||||
| 			_('Do not forward reverse lookups for local networks')); | ||||
| 		o.default = o.enabled; | ||||
| 
 | ||||
| 		s.taboption('advanced', form.Flag, 'filterwin2k', | ||||
| 			_('Filter useless'), | ||||
| 			_('Do not forward requests that cannot be answered by public name servers')); | ||||
| 
 | ||||
| 
 | ||||
| 		s.taboption('advanced', form.Flag, 'localise_queries', | ||||
| 			_('Localise queries'), | ||||
| 			_('Localise hostname depending on the requesting subnet if multiple IPs are available')); | ||||
| 
 | ||||
| 		if (L.hasSystemFeature('dnsmasq', 'dnssec')) { | ||||
| 			o = s.taboption('advanced', form.Flag, 'dnssec', | ||||
| 				_('DNSSEC')); | ||||
| 			o.optional = true; | ||||
| 
 | ||||
| 			o = s.taboption('advanced', form.Flag, 'dnsseccheckunsigned', | ||||
| 				_('DNSSEC check unsigned'), | ||||
| 				_('Requires upstream supports DNSSEC; verify unsigned domain responses really come from unsigned domains')); | ||||
| 			o.default = o.enabled; | ||||
| 			o.optional = true; | ||||
| 		} | ||||
| 
 | ||||
| 		s.taboption('general', form.Value, 'local', | ||||
| 			_('Local server'), | ||||
| 			_('Local domain specification. Names matching this domain are never forwarded and are resolved from DHCP or hosts files only')); | ||||
| 
 | ||||
| 		s.taboption('general', form.Value, 'domain', | ||||
| 			_('Local domain'), | ||||
| 			_('Local domain suffix appended to DHCP names and hosts file entries')); | ||||
| 
 | ||||
| 		s.taboption('advanced', form.Flag, 'expandhosts', | ||||
| 			_('Expand hosts'), | ||||
| 			_('Add local domain suffix to names served from hosts files')); | ||||
| 
 | ||||
| 		s.taboption('advanced', form.Flag, 'nonegcache', | ||||
| 			_('No negative cache'), | ||||
| 			_('Do not cache negative replies, e.g. for not existing domains')); | ||||
| 
 | ||||
| 		s.taboption('advanced', form.Value, 'serversfile', | ||||
| 			_('Additional servers file'), | ||||
| 			_('This file may contain lines like \'server=/domain/1.2.3.4\' or \'server=1.2.3.4\' for domain-specific or full upstream <abbr title="Domain Name System">DNS</abbr> servers.')); | ||||
| 
 | ||||
| 		s.taboption('advanced', form.Flag, 'strictorder', | ||||
| 			_('Strict order'), | ||||
| 			_('<abbr title="Domain Name System">DNS</abbr> servers will be queried in the order of the resolvfile')).optional = true; | ||||
| 
 | ||||
| 		s.taboption('advanced', form.Flag, 'allservers', | ||||
| 			_('All Servers'), | ||||
| 			_('Query all available upstream <abbr title="Domain Name System">DNS</abbr> servers')).optional = true; | ||||
| 
 | ||||
| 		o = s.taboption('advanced', form.DynamicList, 'bogusnxdomain', _('Bogus NX Domain Override'), | ||||
| 			_('List of hosts that supply bogus NX domain results')); | ||||
| 
 | ||||
| 		o.optional = true; | ||||
| 		o.placeholder = '67.215.65.132'; | ||||
| 
 | ||||
| 
 | ||||
| 		s.taboption('general', form.Flag, 'logqueries', | ||||
| 			_('Log queries'), | ||||
| 			_('Write received DNS requests to syslog')).optional = true; | ||||
| 
 | ||||
| 		o = s.taboption('general', form.DynamicList, 'server', _('DNS forwardings'), | ||||
| 			_('List of <abbr title="Domain Name System">DNS</abbr> servers to forward requests to')); | ||||
| 
 | ||||
| 		o.optional = true; | ||||
| 		o.placeholder = '/example.org/10.1.2.3'; | ||||
| 		//o.validate = validateServerSpec;
 | ||||
| 
 | ||||
| 
 | ||||
| 		o = s.taboption('general', form.Flag, 'rebind_protection', | ||||
| 			_('Rebind protection'), | ||||
| 			_('Discard upstream RFC1918 responses')); | ||||
| 
 | ||||
| 		o.rmempty = false; | ||||
| 
 | ||||
| 
 | ||||
| 		o = s.taboption('general', form.Flag, 'rebind_localhost', | ||||
| 			_('Allow localhost'), | ||||
| 			_('Allow upstream responses in the 127.0.0.0/8 range, e.g. for RBL services')); | ||||
| 
 | ||||
| 		o.depends('rebind_protection', '1'); | ||||
| 
 | ||||
| 
 | ||||
| 		o = s.taboption('general', form.DynamicList, 'rebind_domain', | ||||
| 			_('Domain whitelist'), | ||||
| 			_('List of domains to allow RFC1918 responses for')); | ||||
| 		o.optional = true; | ||||
| 
 | ||||
| 		o.depends('rebind_protection', '1'); | ||||
| 		o.placeholder = 'ihost.netflix.com'; | ||||
| 		o.validate = validateAddressList; | ||||
| 
 | ||||
| 
 | ||||
| 		o = s.taboption('advanced', form.Value, 'port', | ||||
| 			_('<abbr title="Domain Name System">DNS</abbr> server port'), | ||||
| 			_('Listening port for inbound DNS queries')); | ||||
| 
 | ||||
| 		o.optional = true; | ||||
| 		o.datatype = 'port'; | ||||
| 		o.placeholder = 53; | ||||
| 
 | ||||
| 
 | ||||
| 		o = s.taboption('advanced', form.Value, 'queryport', | ||||
| 			_('<abbr title="Domain Name System">DNS</abbr> query port'), | ||||
| 			_('Fixed source port for outbound DNS queries')); | ||||
| 
 | ||||
| 		o.optional = true; | ||||
| 		o.datatype = 'port'; | ||||
| 		o.placeholder = _('any'); | ||||
| 
 | ||||
| 
 | ||||
| 		o = s.taboption('advanced', form.Value, 'dhcpleasemax', | ||||
| 			_('<abbr title="maximal">Max.</abbr> <abbr title="Dynamic Host Configuration Protocol">DHCP</abbr> leases'), | ||||
| 			_('Maximum allowed number of active DHCP leases')); | ||||
| 
 | ||||
| 		o.optional = true; | ||||
| 		o.datatype = 'uinteger'; | ||||
| 		o.placeholder = _('unlimited'); | ||||
| 
 | ||||
| 
 | ||||
| 		o = s.taboption('advanced', form.Value, 'ednspacket_max', | ||||
| 			_('<abbr title="maximal">Max.</abbr> <abbr title="Extension Mechanisms for Domain Name System">EDNS0</abbr> packet size'), | ||||
| 			_('Maximum allowed size of EDNS.0 UDP packets')); | ||||
| 
 | ||||
| 		o.optional = true; | ||||
| 		o.datatype = 'uinteger'; | ||||
| 		o.placeholder = 1280; | ||||
| 
 | ||||
| 
 | ||||
| 		o = s.taboption('advanced', form.Value, 'dnsforwardmax', | ||||
| 			_('<abbr title="maximal">Max.</abbr> concurrent queries'), | ||||
| 			_('Maximum allowed number of concurrent DNS queries')); | ||||
| 
 | ||||
| 		o.optional = true; | ||||
| 		o.datatype = 'uinteger'; | ||||
| 		o.placeholder = 150; | ||||
| 
 | ||||
| 		o = s.taboption('advanced', form.Value, 'cachesize', | ||||
| 			_('Size of DNS query cache'), | ||||
| 			_('Number of cached DNS entries (max is 10000, 0 is no caching)')); | ||||
| 		o.optional = true; | ||||
| 		o.datatype = 'range(0,10000)'; | ||||
| 		o.placeholder = 150; | ||||
| 
 | ||||
| 		s.taboption('tftp', form.Flag, 'enable_tftp', | ||||
| 			_('Enable TFTP server')).optional = true; | ||||
| 
 | ||||
| 		o = s.taboption('tftp', form.Value, 'tftp_root', | ||||
| 			_('TFTP server root'), | ||||
| 			_('Root directory for files served via TFTP')); | ||||
| 
 | ||||
| 		o.optional = true; | ||||
| 		o.depends('enable_tftp', '1'); | ||||
| 		o.placeholder = '/'; | ||||
| 
 | ||||
| 
 | ||||
| 		o = s.taboption('tftp', form.Value, 'dhcp_boot', | ||||
| 			_('Network boot image'), | ||||
| 			_('Filename of the boot image advertised to clients')); | ||||
| 
 | ||||
| 		o.optional = true; | ||||
| 		o.depends('enable_tftp', '1'); | ||||
| 		o.placeholder = 'pxelinux.0'; | ||||
| 
 | ||||
| 		o = s.taboption('general', form.Flag, 'localservice', | ||||
| 			_('Local Service Only'), | ||||
| 			_('Limit DNS service to subnets interfaces on which we are serving DNS.')); | ||||
| 		o.optional = false; | ||||
| 		o.rmempty = false; | ||||
| 
 | ||||
| 		o = s.taboption('general', form.Flag, 'nonwildcard', | ||||
| 			_('Non-wildcard'), | ||||
| 			_('Bind dynamically to interfaces rather than wildcard address (recommended as linux default)')); | ||||
| 		o.default = o.enabled; | ||||
| 		o.optional = false; | ||||
| 		o.rmempty = true; | ||||
| 
 | ||||
| 		o = s.taboption('general', form.DynamicList, 'interface', | ||||
| 			_('Listen Interfaces'), | ||||
| 			_('Limit listening to these interfaces, and loopback.')); | ||||
| 		o.optional = true; | ||||
| 
 | ||||
| 		o = s.taboption('general', form.DynamicList, 'notinterface', | ||||
| 			_('Exclude interfaces'), | ||||
| 			_('Prevent listening on these interfaces.')); | ||||
| 		o.optional = true; | ||||
| 
 | ||||
| 		o = s.taboption('leases', form.SectionValue, '__leases__', form.GridSection, 'host', null, | ||||
| 			_('Static leases are used to assign fixed IP addresses and symbolic hostnames to DHCP clients. They are also required for non-dynamic interface configurations where only hosts with a corresponding lease are served.') + '<br />' + | ||||
| 			_('Use the <em>Add</em> Button to add a new lease entry. The <em>MAC-Address</em> identifies the host, the <em>IPv4-Address</em> specifies the fixed address to use, and the <em>Hostname</em> is assigned as a symbolic name to the requesting host. The optional <em>Lease time</em> can be used to set non-standard host-specific lease time, e.g. 12h, 3d or infinite.')); | ||||
| 
 | ||||
| 		ss = o.subsection; | ||||
| 
 | ||||
| 		ss.addremove = true; | ||||
| 		ss.anonymous = true; | ||||
| 
 | ||||
| 		so = ss.option(form.Value, 'name', _('Hostname')); | ||||
| 		so.datatype = 'hostname("strict")'; | ||||
| 		so.rmempty  = true; | ||||
| 		so.write = function(section, value) { | ||||
| 			uci.set('dhcp', section, 'name', value); | ||||
| 			uci.set('dhcp', section, 'dns', '1'); | ||||
| 		}; | ||||
| 		so.remove = function(section) { | ||||
| 			uci.unset('dhcp', section, 'name'); | ||||
| 			uci.unset('dhcp', section, 'dns'); | ||||
| 		}; | ||||
| 
 | ||||
| 		so = ss.option(form.Value, 'mac', _('<abbr title="Media Access Control">MAC</abbr>-Address')); | ||||
| 		so.datatype = 'list(unique(macaddr))'; | ||||
| 		so.rmempty  = true; | ||||
| 		so.cfgvalue = function(section) { | ||||
| 			var macs = uci.get('dhcp', section, 'mac'), | ||||
| 			    result = []; | ||||
| 
 | ||||
| 			if (!Array.isArray(macs)) | ||||
| 				macs = (macs != null && macs != '') ? macs.split(/\ss+/) : []; | ||||
| 
 | ||||
| 			for (var i = 0, mac; (mac = macs[i]) != null; i++) | ||||
| 				if (/^([0-9a-fA-F]{1,2}):([0-9a-fA-F]{1,2}):([0-9a-fA-F]{1,2}):([0-9a-fA-F]{1,2}):([0-9a-fA-F]{1,2}):([0-9a-fA-F]{1,2})$/.test(mac)) | ||||
| 					result.push('%02X:%02X:%02X:%02X:%02X:%02X'.format( | ||||
| 						parseInt(RegExp.$1, 16), parseInt(RegExp.$2, 16), | ||||
| 						parseInt(RegExp.$3, 16), parseInt(RegExp.$4, 16), | ||||
| 						parseInt(RegExp.$5, 16), parseInt(RegExp.$6, 16))); | ||||
| 
 | ||||
| 			return result.length ? result.join(' ') : null; | ||||
| 		}; | ||||
| 		so.renderWidget = function(section_id, option_index, cfgvalue) { | ||||
| 			var node = form.Value.prototype.renderWidget.apply(this, [section_id, option_index, cfgvalue]), | ||||
| 			    ipopt = this.section.children.filter(function(o) { return o.option == 'ip' })[0]; | ||||
| 
 | ||||
| 			node.addEventListener('cbi-dropdown-change', L.bind(function(ipopt, section_id, ev) { | ||||
| 				var mac = ev.detail.value.value; | ||||
| 				if (mac == null || mac == '' || !hosts[mac] || !hosts[mac].ipv4) | ||||
| 					return; | ||||
| 
 | ||||
| 				var ip = ipopt.formvalue(section_id); | ||||
| 				if (ip != null && ip != '') | ||||
| 					return; | ||||
| 
 | ||||
| 				var node = ipopt.map.findElement('id', ipopt.cbid(section_id)); | ||||
| 				if (node) | ||||
| 					L.dom.callClassMethod(node, 'setValue', hosts[mac].ipv4); | ||||
| 			}, this, ipopt, section_id)); | ||||
| 
 | ||||
| 			return node; | ||||
| 		}; | ||||
| 		Object.keys(hosts).forEach(function(mac) { | ||||
| 			var hint = hosts[mac].name || hosts[mac].ipv4; | ||||
| 			so.value(mac, hint ? '%s (%s)'.format(mac, hint) : mac); | ||||
| 		}); | ||||
| 
 | ||||
| 		so = ss.option(form.Value, 'ip', _('<abbr title="Internet Protocol Version 4">IPv4</abbr>-Address')); | ||||
| 		so.datatype = 'or(ip4addr,"ignore")'; | ||||
| 		so.validate = function(section, value) { | ||||
| 			var mac = this.map.lookupOption('mac', section), | ||||
| 			    name = this.map.lookupOption('name', section), | ||||
| 			    m = mac ? mac[0].formvalue(section) : null, | ||||
| 			    n = name ? name[0].formvalue(section) : null; | ||||
| 
 | ||||
| 			if ((m == null || m == '') && (n == null || n == '')) | ||||
| 				return _('One of hostname or mac address must be specified!'); | ||||
| 
 | ||||
| 			return true; | ||||
| 		}; | ||||
| 		Object.keys(hosts).forEach(function(mac) { | ||||
| 			if (hosts[mac].ipv4) { | ||||
| 				var hint = hosts[mac].name; | ||||
| 				so.value(hosts[mac].ipv4, hint ? '%s (%s)'.format(hosts[mac].ipv4, hint) : hosts[mac].ipv4); | ||||
| 			} | ||||
| 		}); | ||||
| 
 | ||||
| 		so = ss.option(form.Value, 'leasetime', _('Lease time')); | ||||
| 		so.rmempty = true; | ||||
| 
 | ||||
| 		so = ss.option(form.Value, 'duid', _('<abbr title="The DHCP Unique Identifier">DUID</abbr>')); | ||||
| 		so.datatype = 'and(rangelength(20,36),hexstring)'; | ||||
| 		Object.keys(duids).forEach(function(duid) { | ||||
| 			so.value(duid, '%s (%s)'.format(duid, duids[duid].hostname || duids[duid].macaddr || duids[duid].ip6addr || '?')); | ||||
| 		}); | ||||
| 
 | ||||
| 		so = ss.option(form.Value, 'hostid', _('<abbr title="Internet Protocol Version 6">IPv6</abbr>-Suffix (hex)')); | ||||
| 
 | ||||
| 		o = s.taboption('leases', CBILeaseStatus, '__status__'); | ||||
| 
 | ||||
| 		if (has_dhcpv6) | ||||
| 			o = s.taboption('leases', CBILease6Status, '__status6__'); | ||||
| 
 | ||||
| 		return m.render().then(function(mapEl) { | ||||
| 			L.Poll.add(function() { | ||||
| 				return callDHCPLeases().then(function(leaseinfo) { | ||||
| 					var leases = Array.isArray(leaseinfo.dhcp_leases) ? leaseinfo.dhcp_leases : [], | ||||
| 					    leases6 = Array.isArray(leaseinfo.dhcp6_leases) ? leaseinfo.dhcp6_leases : []; | ||||
| 
 | ||||
| 					cbi_update_table(mapEl.querySelector('#lease_status_table'), | ||||
| 						leases.map(function(lease) { | ||||
| 							var exp; | ||||
| 
 | ||||
| 							if (lease.expires === false) | ||||
| 								exp = E('em', _('unlimited')); | ||||
| 							else if (lease.expires <= 0) | ||||
| 								exp = E('em', _('expired')); | ||||
| 							else | ||||
| 								exp = '%t'.format(lease.expires); | ||||
| 
 | ||||
| 							return [ | ||||
| 								lease.hostname || '?', | ||||
| 								lease.ipaddr, | ||||
| 								lease.macaddr, | ||||
| 								exp | ||||
| 							]; | ||||
| 						}), | ||||
| 						E('em', _('There are no active leases'))); | ||||
| 
 | ||||
| 					if (has_dhcpv6) { | ||||
| 						cbi_update_table(mapEl.querySelector('#lease6_status_table'), | ||||
| 							leases6.map(function(lease) { | ||||
| 								var exp; | ||||
| 
 | ||||
| 								if (lease.expires === false) | ||||
| 									exp = E('em', _('unlimited')); | ||||
| 								else if (lease.expires <= 0) | ||||
| 									exp = E('em', _('expired')); | ||||
| 								else | ||||
| 									exp = '%t'.format(lease.expires); | ||||
| 
 | ||||
| 								var hint = lease.macaddr ? hosts[lease.macaddr] : null, | ||||
| 								    name = hint ? (hint.name || hint.ipv4 || hint.ipv6) : null, | ||||
| 								    host = null; | ||||
| 
 | ||||
| 								if (name && lease.hostname && lease.hostname != name && lease.ip6addr != name) | ||||
| 									host = '%s (%s)'.format(lease.hostname, name); | ||||
| 								else if (lease.hostname) | ||||
| 									host = lease.hostname; | ||||
| 								else if (name) | ||||
| 									host = name; | ||||
| 
 | ||||
| 								return [ | ||||
| 									host || '-', | ||||
| 									lease.ip6addrs ? lease.ip6addrs.join(' ') : lease.ip6addr, | ||||
| 									lease.duid, | ||||
| 									exp | ||||
| 								]; | ||||
| 							}), | ||||
| 							E('em', _('There are no active leases'))); | ||||
| 					} | ||||
| 				}); | ||||
| 			}); | ||||
| 
 | ||||
| 			return mapEl; | ||||
| 		}); | ||||
| 	} | ||||
| }); | ||||
|  | @ -0,0 +1,137 @@ | |||
| 'use strict'; | ||||
| 'require fs'; | ||||
| 'require ui'; | ||||
| 'require uci'; | ||||
| 
 | ||||
| return L.view.extend({ | ||||
| 	handleCommand: function(exec, args) { | ||||
| 		var buttons = document.querySelectorAll('.diag-action > .cbi-button'); | ||||
| 
 | ||||
| 		for (var i = 0; i < buttons.length; i++) | ||||
| 			buttons[i].setAttribute('disabled', 'true'); | ||||
| 
 | ||||
| 		return fs.exec(exec, args).then(function(res) { | ||||
| 			var out = document.querySelector('.command-output'); | ||||
| 			    out.style.display = ''; | ||||
| 
 | ||||
| 			L.dom.content(out, [ res.stdout || '', res.stderr || '' ]); | ||||
| 		}).catch(function(err) { | ||||
| 			ui.addNotification(null, E('p', [ err ])) | ||||
| 		}).finally(function() { | ||||
| 			for (var i = 0; i < buttons.length; i++) | ||||
| 				buttons[i].removeAttribute('disabled'); | ||||
| 		}); | ||||
| 	}, | ||||
| 
 | ||||
| 	handlePing: function(ev, cmd) { | ||||
| 		var exec = cmd || 'ping', | ||||
| 		    addr = ev.currentTarget.parentNode.previousSibling.value, | ||||
| 		    args = (exec == 'ping') ? [ '-c', '5', '-W', '1', addr ] : [ '-c', '5', addr ]; | ||||
| 
 | ||||
| 		return this.handleCommand(exec, args); | ||||
| 	}, | ||||
| 
 | ||||
| 	handleTraceroute: function(ev, cmd) { | ||||
| 		var exec = cmd || 'traceroute', | ||||
| 		    addr = ev.currentTarget.parentNode.previousSibling.value, | ||||
| 		    args = (exec == 'traceroute') ? [ '-q', '1', '-w', '1', '-n', addr ] : [ '-q', '1', '-w', '2', '-n', addr ]; | ||||
| 
 | ||||
| 		return this.handleCommand(exec, args); | ||||
| 	}, | ||||
| 
 | ||||
| 	handleNslookup: function(ev, cmd) { | ||||
| 		var addr = ev.currentTarget.parentNode.previousSibling.value; | ||||
| 
 | ||||
| 		return this.handleCommand('nslookup', [ addr ]); | ||||
| 	}, | ||||
| 
 | ||||
| 	load: function() { | ||||
| 		return Promise.all([ | ||||
| 			L.resolveDefault(fs.stat('/bin/ping6'), {}), | ||||
| 			L.resolveDefault(fs.stat('/usr/bin/ping6'), {}), | ||||
| 			L.resolveDefault(fs.stat('/bin/traceroute6'), {}), | ||||
| 			L.resolveDefault(fs.stat('/usr/bin/traceroute6'), {}), | ||||
| 			uci.load('luci') | ||||
| 		]); | ||||
| 	}, | ||||
| 
 | ||||
| 	render: function(res) { | ||||
| 		var has_ping6 = res[0].path || res[1].path, | ||||
| 		    has_traceroute6 = res[2].path || res[3].path, | ||||
| 			dns_host = uci.get('luci', 'diag', 'dns') || 'openwrt.org', | ||||
| 			ping_host = uci.get('luci', 'diag', 'ping') || 'openwrt.org', | ||||
| 			route_host = uci.get('luci', 'diag', 'route') || 'openwrt.org'; | ||||
| 
 | ||||
| 		return E([], [ | ||||
| 			E('h2', {}, [ _('Network Utilities') ]), | ||||
| 			E('div', { 'class': 'table' }, [ | ||||
| 				E('div', { 'class': 'tr' }, [ | ||||
| 					E('div', { 'class': 'td left' }, [ | ||||
| 						E('input', { | ||||
| 							'style': 'margin:5px 0', | ||||
| 							'type': 'text', | ||||
| 							'value': ping_host | ||||
| 						}), | ||||
| 						E('span', { 'class': 'diag-action' }, [ | ||||
| 							has_ping6 ? new ui.ComboButton('ping', { | ||||
| 								'ping': '%s %s'.format(_('IPv4'), _('Ping')), | ||||
| 								'ping6': '%s %s'.format(_('IPv6'), _('Ping')), | ||||
| 							}, { | ||||
| 								'click': ui.createHandlerFn(this, 'handlePing'), | ||||
| 								'classes': { | ||||
| 									'ping': 'cbi-button cbi-button-action', | ||||
| 									'ping6': 'cbi-button cbi-button-action' | ||||
| 								} | ||||
| 							}).render() : E('button', { | ||||
| 								'class': 'cbi-button cbi-button-action', | ||||
| 								'click': ui.createHandlerFn(this, 'handlePing') | ||||
| 							}, [ _('Ping') ]) | ||||
| 						]) | ||||
| 					]), | ||||
| 
 | ||||
| 					E('div', { 'class': 'td left' }, [ | ||||
| 						E('input', { | ||||
| 							'style': 'margin:5px 0', | ||||
| 							'type': 'text', | ||||
| 							'value': route_host | ||||
| 						}), | ||||
| 						E('span', { 'class': 'diag-action' }, [ | ||||
| 							has_traceroute6 ? new ui.ComboButton('traceroute', { | ||||
| 								'traceroute': '%s %s'.format(_('IPv4'), _('Traceroute')), | ||||
| 								'traceroute6': '%s %s'.format(_('IPv6'), _('Traceroute')), | ||||
| 							}, { | ||||
| 								'click': ui.createHandlerFn(this, 'handleTraceroute'), | ||||
| 								'classes': { | ||||
| 									'traceroute': 'cbi-button cbi-button-action', | ||||
| 									'traceroute6': 'cbi-button cbi-button-action' | ||||
| 								} | ||||
| 							}).render() : E('button', { | ||||
| 								'class': 'cbi-button cbi-button-action', | ||||
| 								'click': ui.createHandlerFn(this, 'handleTraceroute') | ||||
| 							}, [ _('Traceroute') ]) | ||||
| 						]) | ||||
| 					]), | ||||
| 
 | ||||
| 					E('div', { 'class': 'td left' }, [ | ||||
| 						E('input', { | ||||
| 							'style': 'margin:5px 0', | ||||
| 							'type': 'text', | ||||
| 							'value': dns_host | ||||
| 						}), | ||||
| 						E('span', { 'class': 'diag-action' }, [ | ||||
| 							E('button', { | ||||
| 								'class': 'cbi-button cbi-button-action', | ||||
| 								'click': ui.createHandlerFn(this, 'handleNslookup') | ||||
| 							}, [ _('Nslookup') ]) | ||||
| 						]) | ||||
| 					]) | ||||
| 				]) | ||||
| 			]), | ||||
| 			E('pre', { 'class': 'command-output', 'style': 'display:none' }) | ||||
| 		]); | ||||
| 	}, | ||||
| 
 | ||||
| 	handleSaveApply: null, | ||||
| 	handleSave: null, | ||||
| 	handleReset: null | ||||
| }); | ||||
|  | @ -0,0 +1,42 @@ | |||
| 'use strict'; | ||||
| 'require rpc'; | ||||
| 'require form'; | ||||
| 
 | ||||
| return L.view.extend({ | ||||
| 	callHostHints: rpc.declare({ | ||||
| 		object: 'luci-rpc', | ||||
| 		method: 'getHostHints', | ||||
| 		expect: { '': {} } | ||||
| 	}), | ||||
| 
 | ||||
| 	load: function() { | ||||
| 		return this.callHostHints(); | ||||
| 	}, | ||||
| 
 | ||||
| 	render: function(hosts) { | ||||
| 		var m, s, o; | ||||
| 
 | ||||
| 		m = new form.Map('dhcp', _('Hostnames')); | ||||
| 
 | ||||
| 		s = m.section(form.GridSection, 'domain', _('Host entries')); | ||||
| 		s.addremove = true; | ||||
| 		s.anonymous = true; | ||||
| 		s.sortable  = true; | ||||
| 
 | ||||
| 		o = s.option(form.Value, 'name', _('Hostname')); | ||||
| 		o.datatype = 'hostname'; | ||||
| 		o.rmempty = true; | ||||
| 
 | ||||
| 		o = s.option(form.Value, 'ip', _('IP address')); | ||||
| 		o.datatype = 'ipaddr'; | ||||
| 		o.rmempty = true; | ||||
| 		L.sortedKeys(hosts, 'ipv4', 'addr').forEach(function(mac) { | ||||
| 			o.value(hosts[mac].ipv4, '%s (%s)'.format( | ||||
| 				hosts[mac].ipv4, | ||||
| 				hosts[mac].name || mac | ||||
| 			)); | ||||
| 		}); | ||||
| 
 | ||||
| 		return m.render(); | ||||
| 	} | ||||
| }); | ||||
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -0,0 +1,102 @@ | |||
| 'use strict'; | ||||
| 'require form'; | ||||
| 'require network'; | ||||
| 'require tools.widgets as widgets'; | ||||
| 
 | ||||
| return L.view.extend({ | ||||
| 	load: function() { | ||||
| 		return network.getDevices(); | ||||
| 	}, | ||||
| 
 | ||||
| 	render: function(netdevs) { | ||||
| 		var m, s, o; | ||||
| 
 | ||||
| 		m = new form.Map('network', _('Routes'), _('Routes specify over which interface and gateway a certain host or network can be reached.')); | ||||
| 		m.tabbed = true; | ||||
| 
 | ||||
| 		for (var i = 4; i <= 6; i += 2) { | ||||
| 			s = m.section(form.GridSection, (i == 4) ? 'route' : 'route6', (i == 4) ? _('Static IPv4 Routes') : _('Static IPv6 Routes')); | ||||
| 			s.anonymous = true; | ||||
| 			s.addremove = true; | ||||
| 			s.sortable = true; | ||||
| 
 | ||||
| 			s.tab('general', _('General Settings')); | ||||
| 			s.tab('advanced', _('Advanced Settings')); | ||||
| 
 | ||||
| 			o = s.taboption('general', widgets.NetworkSelect, 'interface', _('Interface')); | ||||
| 			o.rmempty = false; | ||||
| 			o.nocreate = true; | ||||
| 
 | ||||
| 			o = s.taboption('general', form.Value, 'target', _('Target'), (i == 4) ? _('Host-<abbr title="Internet Protocol Address">IP</abbr> or Network') : _('<abbr title="Internet Protocol Version 6">IPv6</abbr>-Address or Network (CIDR)')); | ||||
| 			o.datatype = (i == 4) ? 'ip4addr' : 'ip6addr'; | ||||
| 			o.rmempty = false; | ||||
| 
 | ||||
| 			if (i == 4) { | ||||
| 				o = s.taboption('general', form.Value, 'netmask', _('<abbr title="Internet Protocol Version 4">IPv4</abbr>-Netmask'), _('if target is a network')); | ||||
| 				o.placeholder = '255.255.255.255'; | ||||
| 				o.datatype = 'ip4addr'; | ||||
| 				o.rmempty = true; | ||||
| 			} | ||||
| 
 | ||||
| 			o = s.taboption('general', form.Value, 'gateway', (i == 4) ? _('<abbr title="Internet Protocol Version 4">IPv4</abbr>-Gateway') : _('<abbr title="Internet Protocol Version 6">IPv6</abbr>-Gateway')); | ||||
| 			o.datatype = (i == 4) ? 'ip4addr' : 'ip6addr'; | ||||
| 			o.rmempty = true; | ||||
| 
 | ||||
| 			o = s.taboption('advanced', form.Value, 'metric', _('Metric')); | ||||
| 			o.placeholder = 0; | ||||
| 			o.datatype = (i == 4) ? 'range(0,255)' : 'range(0,65535)'; | ||||
| 			o.rmempty = true; | ||||
| 			o.textvalue = function(section_id) { | ||||
| 				return this.cfgvalue(section_id) || 0; | ||||
| 			}; | ||||
| 
 | ||||
| 			o = s.taboption('advanced', form.Value, 'mtu', _('MTU')); | ||||
| 			o.placeholder = 1500; | ||||
| 			o.datatype = 'range(64,9000)'; | ||||
| 			o.rmempty = true; | ||||
| 			o.modalonly = true; | ||||
| 
 | ||||
| 			o = s.taboption('advanced', form.ListValue, 'type', _('Route type')); | ||||
| 			o.value('', 'unicast'); | ||||
| 			o.value('local'); | ||||
| 			o.value('broadcast'); | ||||
| 			o.value('multicast'); | ||||
| 			o.value('unreachable'); | ||||
| 			o.value('prohibit'); | ||||
| 			o.value('blackhole'); | ||||
| 			o.value('anycast'); | ||||
| 			o.default = ''; | ||||
| 			o.rmempty = true; | ||||
| 			o.modalonly = true; | ||||
| 
 | ||||
| 			o = s.taboption('advanced', form.Value, 'table', _('Route table')); | ||||
| 			o.value('local', 'local (255)'); | ||||
| 			o.value('main', 'main (254)'); | ||||
| 			o.value('default', 'default (253)'); | ||||
| 			o.rmempty = true; | ||||
| 			o.modalonly = true; | ||||
| 			o.cfgvalue = function(section_id) { | ||||
| 				var cfgvalue = this.super('cfgvalue', [section_id]); | ||||
| 				return cfgvalue || 'main'; | ||||
| 			}; | ||||
| 
 | ||||
| 			o = s.taboption('advanced', form.Value, 'source', _('Source Address')); | ||||
| 			o.placeholder = E('em', _('automatic')); | ||||
| 			for (var j = 0; j < netdevs.length; j++) { | ||||
| 				var addrs = netdevs[j].getIPAddrs(); | ||||
| 				for (var k = 0; k < addrs.length; k++) | ||||
| 					o.value(addrs[k].split('/')[0]); | ||||
| 			} | ||||
| 			o.datatype = (i == 4) ? 'ip4addr' : 'ip6addr'; | ||||
| 			o.default = ''; | ||||
| 			o.rmempty = true; | ||||
| 			o.modalonly = true; | ||||
| 
 | ||||
| 			o = s.taboption('advanced', form.Flag, 'onlink', _('On-Link route')); | ||||
| 			o.default = o.disabled; | ||||
| 			o.rmempty = true; | ||||
| 		} | ||||
| 
 | ||||
| 		return m.render(); | ||||
| 	} | ||||
| }); | ||||
|  | @ -0,0 +1,370 @@ | |||
| 'use strict'; | ||||
| 'require ui'; | ||||
| 'require rpc'; | ||||
| 'require uci'; | ||||
| 'require form'; | ||||
| 'require network'; | ||||
| 
 | ||||
| function parse_portvalue(section_id) { | ||||
| 	var ports = L.toArray(uci.get('network', section_id, 'ports')); | ||||
| 
 | ||||
| 	for (var i = 0; i < ports.length; i++) { | ||||
| 		var m = ports[i].match(/^(\d+)([tu]?)/); | ||||
| 
 | ||||
| 		if (m && m[1] == this.option) | ||||
| 			return m[2] || 'u'; | ||||
| 	} | ||||
| 
 | ||||
| 	return ''; | ||||
| } | ||||
| 
 | ||||
| function validate_portvalue(section_id, value) { | ||||
| 	if (value != 'u') | ||||
| 		return true; | ||||
| 
 | ||||
| 	var sections = this.section.cfgsections(); | ||||
| 
 | ||||
| 	for (var i = 0; i < sections.length; i++) { | ||||
| 		if (sections[i] == section_id) | ||||
| 			continue; | ||||
| 
 | ||||
| 		if (this.formvalue(sections[i]) == 'u') | ||||
| 			return _('%s is untagged in multiple VLANs!').format(this.title); | ||||
| 	} | ||||
| 
 | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| function update_interfaces(old_ifname, new_ifname) { | ||||
| 	var interfaces = uci.sections('network', 'interface'); | ||||
| 
 | ||||
| 	for (var i = 0; i < interfaces.length; i++) { | ||||
| 		var old_ifnames = L.toArray(interfaces[i].ifname), | ||||
| 		    new_ifnames = [], | ||||
| 		    changed = false; | ||||
| 
 | ||||
| 		for (var j = 0; j < old_ifnames.length; j++) { | ||||
| 			if (old_ifnames[j] == old_ifname) { | ||||
| 				new_ifnames.push(new_ifname); | ||||
| 				changed = true; | ||||
| 			} | ||||
| 			else { | ||||
| 				new_ifnames.push(old_ifnames[j]); | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if (changed) { | ||||
| 			uci.set('network', interfaces[i]['.name'], 'ifname', new_ifnames.join(' ')); | ||||
| 
 | ||||
| 			ui.addNotification(null, E('p', _('Interface %q device auto-migrated from %q to %q.') | ||||
| 				.replace(/%q/g, '"%s"').format(interfaces[i]['.name'], old_ifname, new_ifname))); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| function render_port_status(node, portstate) { | ||||
| 	if (!node) | ||||
| 		return null; | ||||
| 
 | ||||
| 	if (!portstate.link) | ||||
| 		L.dom.content(node, [ | ||||
| 			E('img', { src: L.resource('icons/port_down.png') }), | ||||
| 			E('br'), | ||||
| 			_('no link') | ||||
| 		]); | ||||
| 	else | ||||
| 		L.dom.content(node, [ | ||||
| 			E('img', { src: L.resource('icons/port_up.png') }), | ||||
| 			E('br'), | ||||
| 			'%d'.format(portstate.speed) + _('baseT'), | ||||
| 			E('br'), | ||||
| 			portstate.duplex ? _('full-duplex') : _('half-duplex') | ||||
| 		]); | ||||
| 
 | ||||
| 	return node; | ||||
| } | ||||
| 
 | ||||
| function update_port_status(topologies) { | ||||
| 	var tasks = []; | ||||
| 
 | ||||
| 	for (var switch_name in topologies) | ||||
| 		tasks.push(callSwconfigPortState(switch_name).then(L.bind(function(switch_name, ports) { | ||||
| 			for (var i = 0; i < ports.length; i++) { | ||||
| 				var node = document.querySelector('[data-switch="%s"][data-port="%d"]'.format(switch_name, ports[i].port)); | ||||
| 				render_port_status(node, ports[i]); | ||||
| 			} | ||||
| 		}, topologies[switch_name], switch_name))); | ||||
| 
 | ||||
| 	return Promise.all(tasks); | ||||
| } | ||||
| 
 | ||||
| var callSwconfigFeatures = rpc.declare({ | ||||
| 	object: 'luci', | ||||
| 	method: 'getSwconfigFeatures', | ||||
| 	params: [ 'switch' ], | ||||
| 	expect: { '': {} } | ||||
| }); | ||||
| 
 | ||||
| var callSwconfigPortState = rpc.declare({ | ||||
| 	object: 'luci', | ||||
| 	method: 'getSwconfigPortState', | ||||
| 	params: [ 'switch' ], | ||||
| 	expect: { result: [] } | ||||
| }); | ||||
| 
 | ||||
| return L.view.extend({ | ||||
| 	load: function() { | ||||
| 		return network.getSwitchTopologies().then(function(topologies) { | ||||
| 			var tasks = []; | ||||
| 
 | ||||
| 			for (var switch_name in topologies) { | ||||
| 				tasks.push(callSwconfigFeatures(switch_name).then(L.bind(function(features) { | ||||
| 					this.features = features; | ||||
| 				}, topologies[switch_name]))); | ||||
| 				tasks.push(callSwconfigPortState(switch_name).then(L.bind(function(ports) { | ||||
| 					this.portstate = ports; | ||||
| 				}, topologies[switch_name]))); | ||||
| 			} | ||||
| 
 | ||||
| 			return Promise.all(tasks).then(function() { return topologies }); | ||||
| 		}); | ||||
| 	}, | ||||
| 
 | ||||
| 	render: function(topologies) { | ||||
| 		var m, s, o; | ||||
| 
 | ||||
| 		m = new form.Map('network', _('Switch'), _('The network ports on this device can be combined to several <abbr title=\"Virtual Local Area Network\">VLAN</abbr>s in which computers can communicate directly with each other. <abbr title=\"Virtual Local Area Network\">VLAN</abbr>s are often used to separate different network segments. Often there is by default one Uplink port for a connection to the next greater network like the internet and other ports for a local network.')); | ||||
| 
 | ||||
| 		var switchSections = uci.sections('network', 'switch'); | ||||
| 
 | ||||
| 		for (var i = 0; i < switchSections.length; i++) { | ||||
| 			var switchSection   = switchSections[i], | ||||
| 			    sid             = switchSection['.name'], | ||||
| 			    switch_name     = switchSection.name || sid, | ||||
| 			    topology        = topologies[switch_name]; | ||||
| 
 | ||||
| 			if (!topology) { | ||||
| 				ui.addNotification(null, _('Switch %q has an unknown topology - the VLAN settings might not be accurate.').replace(/%q/, switch_name)); | ||||
| 
 | ||||
| 				topology = { | ||||
| 					features: {}, | ||||
| 					netdevs: { | ||||
| 						5: 'eth0' | ||||
| 					}, | ||||
| 					ports: [ | ||||
| 						{ num: 0, label: 'Port 1' }, | ||||
| 						{ num: 1, label: 'Port 2' }, | ||||
| 						{ num: 2, label: 'Port 3' }, | ||||
| 						{ num: 3, label: 'Port 4' }, | ||||
| 						{ num: 4, label: 'Port 5' }, | ||||
| 						{ num: 5, label: 'CPU (eth0)', device: 'eth0', need_tag: false } | ||||
| 					] | ||||
| 				}; | ||||
| 			} | ||||
| 
 | ||||
| 			var feat = topology.features, | ||||
| 			    min_vid = feat.min_vid || 0, | ||||
| 			    max_vid = feat.max_vid || 16, | ||||
| 			    num_vlans = feat.num_vlans || 16, | ||||
| 			    switch_title = _('Switch %q').replace(/%q/, '"%s"'.format(switch_name)), | ||||
| 			    vlan_title = _('VLANs on %q').replace(/%q/, '"%s"'.format(switch_name)); | ||||
| 
 | ||||
| 			if (feat.switch_title) { | ||||
| 				switch_title += ' (%s)'.format(feat.switch_title); | ||||
| 				vlan_title += ' (%s)'.format(feat.switch_title); | ||||
| 			} | ||||
| 
 | ||||
| 			s = m.section(form.NamedSection, sid, 'switch', switch_title); | ||||
| 			s.addremove = false; | ||||
| 
 | ||||
| 			if (feat.vlan_option) | ||||
| 				s.option(form.Flag, feat.vlan_option, _('Enable VLAN functionality')); | ||||
| 
 | ||||
| 			if (feat.learning_option) { | ||||
| 				o = s.option(form.Flag, feat.learning_option, _('Enable learning and aging')); | ||||
| 				o.default = o.enabled; | ||||
| 			} | ||||
| 
 | ||||
| 			if (feat.jumbo_option) { | ||||
| 				o = s.option(form.Flag, feat.jumbo_option, _('Enable Jumbo Frame passthrough')); | ||||
| 				o.enabled = '3'; | ||||
| 				o.rmempty = true; | ||||
| 			} | ||||
| 
 | ||||
| 			if (feat.mirror_option) { | ||||
| 				s.option(form.Flag, 'enable_mirror_rx', _('Enable mirroring of incoming packets')); | ||||
| 				s.option(form.Flag, 'enable_mirror_tx', _('Enable mirroring of outgoing packets')); | ||||
| 
 | ||||
| 				var sp = s.option(form.ListValue, 'mirror_source_port', _('Mirror source port')), | ||||
| 				    mp = s.option(form.ListValue, 'mirror_monitor_port', _('Mirror monitor port')); | ||||
| 
 | ||||
| 				sp.depends('enable_mirror_rx', '1'); | ||||
| 				sp.depends('enable_mirror_tx', '1'); | ||||
| 
 | ||||
| 				mp.depends('enable_mirror_rx', '1'); | ||||
| 				mp.depends('enable_mirror_tx', '1'); | ||||
| 
 | ||||
| 				for (var j = 0; j < topology.ports.length; j++) { | ||||
| 					sp.value(topology.ports[j].num, topology.ports[j].label); | ||||
| 					mp.value(topology.ports[j].num, topology.ports[j].label); | ||||
| 				} | ||||
| 			} | ||||
| 
 | ||||
| 			s = m.section(form.TableSection, 'switch_vlan', vlan_title); | ||||
| 			s.anonymous = true; | ||||
| 			s.addremove = true; | ||||
| 			s.addbtntitle = _('Add VLAN'); | ||||
| 			s.topology = topology; | ||||
| 			s.device = switch_name; | ||||
| 
 | ||||
| 			s.filter = function(section_id) { | ||||
| 				var device = uci.get('network', section_id, 'device'); | ||||
| 				return (device == switch_name); | ||||
| 			}; | ||||
| 
 | ||||
| 			s.cfgsections = function() { | ||||
| 				var sections = form.TableSection.prototype.cfgsections.apply(this); | ||||
| 
 | ||||
| 				return sections.sort(function(a, b) { | ||||
| 					var vidA = feat.vid_option ? uci.get('network', a, feat.vid_option) : null, | ||||
| 					    vidB = feat.vid_option ? uci.get('network', b, feat.vid_option) : null; | ||||
| 
 | ||||
| 					vidA = +(vidA != null ? vidA : uci.get('network', a, 'vlan') || 9999); | ||||
| 					vidB = +(vidB != null ? vidB : uci.get('network', b, 'vlan') || 9999); | ||||
| 
 | ||||
| 					return (vidA - vidB); | ||||
| 				}); | ||||
| 			}; | ||||
| 
 | ||||
| 			s.handleAdd = function(ev) { | ||||
| 				var sections = uci.sections('network', 'switch_vlan'), | ||||
| 				    section_id = uci.add('network', 'switch_vlan'), | ||||
| 				    max_vlan = 0, | ||||
| 				    max_vid = 0; | ||||
| 
 | ||||
| 				for (var j = 0; j < sections.length; j++) { | ||||
| 					if (sections[j].device != s.device) | ||||
| 						continue; | ||||
| 
 | ||||
| 					var vlan = +sections[j].vlan, | ||||
| 					    vid = feat.vid_option ? +sections[j][feat.vid_option] : null; | ||||
| 
 | ||||
| 					if (vlan > max_vlan) | ||||
| 						max_vlan = vlan; | ||||
| 
 | ||||
| 					if (vid > max_vid) | ||||
| 						max_vid = vid; | ||||
| 				} | ||||
| 
 | ||||
| 				uci.set('network', section_id, 'device', s.device); | ||||
| 				uci.set('network', section_id, 'vlan', max_vlan + 1); | ||||
| 
 | ||||
| 				if (feat.vid_option) | ||||
| 					uci.set('network', section_id, feat.vid_option, max_vid + 1); | ||||
| 
 | ||||
| 				return this.map.save(null, true); | ||||
| 			}; | ||||
| 
 | ||||
| 			var port_opts = []; | ||||
| 
 | ||||
| 			o = s.option(form.Value, feat.vid_option || 'vlan', 'VLAN ID'); | ||||
| 			o.rmempty = false; | ||||
| 			o.forcewrite = true; | ||||
| 			o.vlan_used = {}; | ||||
| 			o.datatype = 'range(%u,%u)'.format(min_vid, feat.vid_option ? 4094 : num_vlans - 1); | ||||
| 			o.description = _('Port status:'); | ||||
| 
 | ||||
| 			o.validate = function(section_id, value) { | ||||
| 				var v = +value, | ||||
| 				    m = feat.vid_option ? 4094 : num_vlans - 1; | ||||
| 
 | ||||
| 				if (isNaN(v) || v < min_vid || v > m) | ||||
| 					return _('Invalid VLAN ID given! Only IDs between %d and %d are allowed.').format(min_vid, m); | ||||
| 
 | ||||
| 				var sections = this.section.cfgsections(); | ||||
| 
 | ||||
| 				for (var i = 0; i < sections.length; i++) { | ||||
| 					if (sections[i] == section_id) | ||||
| 						continue; | ||||
| 
 | ||||
| 					if (this.formvalue(sections[i]) == v) | ||||
| 						return _('Invalid VLAN ID given! Only unique IDs are allowed'); | ||||
| 				} | ||||
| 
 | ||||
| 				return true; | ||||
| 			}; | ||||
| 
 | ||||
| 			o.write = function(section_id, value) { | ||||
| 				var topology = this.section.topology, | ||||
| 				    values = []; | ||||
| 
 | ||||
| 				for (var i = 0; i < port_opts.length; i++) { | ||||
| 					var tagging = port_opts[i].formvalue(section_id), | ||||
| 					    portspec = Array.isArray(topology.ports) ? topology.ports[i] : null; | ||||
| 
 | ||||
| 					if (tagging == 't') | ||||
| 						values.push(port_opts[i].option + tagging); | ||||
| 					else if (tagging == 'u') | ||||
| 						values.push(port_opts[i].option); | ||||
| 
 | ||||
| 					if (portspec && portspec.device) { | ||||
| 						var old_tag = port_opts[i].cfgvalue(section_id), | ||||
| 						    old_vid = this.cfgvalue(section_id); | ||||
| 
 | ||||
| 						if (old_tag != tagging || old_vid != value) { | ||||
| 							var old_ifname = portspec.device + (old_tag != 'u' ? '.' + old_vid : ''), | ||||
| 							    new_ifname = portspec.device + (tagging != 'u' ? '.' + value : ''); | ||||
| 
 | ||||
| 							if (old_ifname != new_ifname) | ||||
| 								update_interfaces(old_ifname, new_ifname); | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 
 | ||||
| 				if (feat.vlan4k_option) | ||||
| 					uci.set('network', sid, feat.vlan4k_option, '1'); | ||||
| 
 | ||||
| 				uci.set('network', section_id, 'ports', values.join(' ')); | ||||
| 
 | ||||
| 				return form.Value.prototype.write.apply(this, [section_id, value]); | ||||
| 			}; | ||||
| 
 | ||||
| 			o.cfgvalue = function(section_id) { | ||||
| 				var value = feat.vid_option ? uci.get('network', section_id, feat.vid_option) : null; | ||||
| 				return (value || uci.get('network', section_id, 'vlan')); | ||||
| 			}; | ||||
| 
 | ||||
| 			for (var j = 0; Array.isArray(topology.ports) && j < topology.ports.length; j++) { | ||||
| 				var portspec = topology.ports[j], | ||||
| 				    portstate = Array.isArray(topology.portstate) ? topology.portstate[portspec.num] : null; | ||||
| 
 | ||||
| 				o = s.option(form.ListValue, String(portspec.num), portspec.label); | ||||
| 				o.value('', _('off')); | ||||
| 
 | ||||
| 				if (!portspec.need_tag) | ||||
| 					o.value('u', _('untagged')); | ||||
| 
 | ||||
| 				o.value('t', _('tagged')); | ||||
| 
 | ||||
| 				o.cfgvalue = parse_portvalue; | ||||
| 				o.validate = validate_portvalue; | ||||
| 				o.write    = function() {}; | ||||
| 
 | ||||
| 				o.description = render_port_status(E('small', { | ||||
| 					'data-switch': switch_name, | ||||
| 					'data-port': portspec.num | ||||
| 				}), portstate); | ||||
| 
 | ||||
| 				port_opts.push(o); | ||||
| 			} | ||||
| 
 | ||||
| 			port_opts.sort(function(a, b) { | ||||
| 				return a.option < b.option; | ||||
| 			}); | ||||
| 		} | ||||
| 
 | ||||
| 		L.Poll.add(L.bind(update_port_status, m, topologies)); | ||||
| 
 | ||||
| 		return m.render(); | ||||
| 	} | ||||
| }); | ||||
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										22
									
								
								luci-mod-network/root/etc/uci-defaults/50_luci-mod-admin-full
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										22
									
								
								luci-mod-network/root/etc/uci-defaults/50_luci-mod-admin-full
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,22 @@ | |||
| #!/bin/sh | ||||
| 
 | ||||
| if [ "$(uci -q get luci.diag)" != "internal" ]; then | ||||
| 	host="" | ||||
| 
 | ||||
| 	if [ -s /etc/os-release ]; then | ||||
| 		. /etc/os-release | ||||
| 		host="${HOME_URL:-${BUG_URL:-$OPENWRT_DEVICE_MANUFACTURER_URL}}" | ||||
| 		host="${host#*://}" | ||||
| 		host="${host%%/*}" | ||||
| 	fi | ||||
| 
 | ||||
| 	uci -q batch <<-EOF >/dev/null | ||||
| 		set luci.diag=internal | ||||
| 		set luci.diag.dns='${host:-openwrt.org}' | ||||
| 		set luci.diag.ping='${host:-openwrt.org}' | ||||
| 		set luci.diag.route='${host:-openwrt.org}' | ||||
| 		commit luci | ||||
| 	EOF | ||||
| fi | ||||
| 
 | ||||
| exit 0 | ||||
							
								
								
									
										46
									
								
								luci-mod-network/root/usr/libexec/luci-peeraddr
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										46
									
								
								luci-mod-network/root/usr/libexec/luci-peeraddr
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,46 @@ | |||
| #!/bin/sh | ||||
| 
 | ||||
| NL=" | ||||
| " | ||||
| 
 | ||||
| function ifaces_by_device() { | ||||
| 	ubus call network.interface dump 2>/dev/null | \ | ||||
| 		jsonfilter -e "@.interface[@.device='$1' || @.l3_device='$1'].interface" | ||||
| } | ||||
| 
 | ||||
| function device_by_addr() { | ||||
| 	set -- $(ip route get "$1" ${2:+from "$2"} 2>/dev/null) | ||||
| 	echo "$5" | ||||
| } | ||||
| 
 | ||||
| for inbound_device in $(device_by_addr "$REMOTE_ADDR" "$SERVER_ADDR"); do | ||||
| 	inbound_devices="$inbound_device" | ||||
| 	inbound_interfaces="" | ||||
| 
 | ||||
| 	for iface in $(ifaces_by_device "$inbound_device"); do | ||||
| 		inbound_interfaces="${inbound_interfaces:+$inbound_interfaces$NL}$iface" | ||||
| 
 | ||||
| 		for peeraddr in $(uci get "network.$iface.peeraddr"); do | ||||
| 			for ipaddr in $(resolveip -t 1 "$peeraddr" 2>/dev/null); do | ||||
| 				for peerdev in $(device_by_addr "$ipaddr"); do | ||||
| 					for iface in $(ifaces_by_device "$peerdev"); do | ||||
| 						inbound_devices="${inbound_devices:+$inbound_devices$NL}$peerdev" | ||||
| 						inbound_interfaces="${inbound_interfaces:+$inbound_interfaces$NL}$iface" | ||||
| 					done | ||||
| 				done | ||||
| 			done | ||||
| 		done | ||||
| 	done | ||||
| done | ||||
| 
 | ||||
| inbound_devices="$(echo "$inbound_devices" | sort -u | sed ':a;N;$!ba;s/\n/", "/g')" | ||||
| inbound_interfaces="$(echo "$inbound_interfaces" | sort -u | sed ':a;N;$!ba;s/\n/", "/g')" | ||||
| 
 | ||||
| cat <<JSON | ||||
| { | ||||
| 	"remote_addr": "$REMOTE_ADDR", | ||||
| 	"server_addr": "$SERVER_ADDR", | ||||
| 	"inbound_devices": [ ${inbound_devices:+\"$inbound_devices\"} ], | ||||
| 	"inbound_interfaces": [ ${inbound_interfaces:+\"$inbound_interfaces\"} ] | ||||
| } | ||||
| JSON | ||||
|  | @ -0,0 +1,85 @@ | |||
| { | ||||
| 	"admin/network/switch": { | ||||
| 		"title": "Switch", | ||||
| 		"order": 20, | ||||
| 		"action": { | ||||
| 			"type": "view", | ||||
| 			"path": "network/switch" | ||||
| 		}, | ||||
| 		"depends": { | ||||
| 			"fs": { "/sbin/swconfig": "executable" }, | ||||
| 			"uci": { "network": { "@switch": true } } | ||||
| 		} | ||||
| 	}, | ||||
| 
 | ||||
| 	"admin/network/wireless": { | ||||
| 		"title": "Wireless", | ||||
| 		"order": 15, | ||||
| 		"action": { | ||||
| 			"type": "view", | ||||
| 			"path": "network/wireless" | ||||
| 		}, | ||||
| 		"depends": { | ||||
| 			"uci": { "wireless": { "@wifi-device": true } } | ||||
| 		} | ||||
| 	}, | ||||
| 
 | ||||
| 	"admin/network/remote_addr/*": { | ||||
| 		"action": { | ||||
| 			"type": "call", | ||||
| 			"module": "luci.controller.admin.network", | ||||
| 			"function": "remote_addr" | ||||
| 		} | ||||
| 	}, | ||||
| 
 | ||||
| 	"admin/network/network": { | ||||
| 		"title": "Interfaces", | ||||
| 		"order": 10, | ||||
| 		"action": { | ||||
| 			"type": "view", | ||||
| 			"path": "network/interfaces" | ||||
| 		} | ||||
| 	}, | ||||
| 
 | ||||
| 	"admin/network/dhcp": { | ||||
| 		"title": "DHCP and DNS", | ||||
| 		"order": 30, | ||||
| 		"action": { | ||||
| 			"type": "view", | ||||
| 			"path": "network/dhcp" | ||||
| 		}, | ||||
| 		"depends": { | ||||
| 			"uci": { "dhcp": true } | ||||
| 		} | ||||
| 	}, | ||||
| 
 | ||||
| 	"admin/network/hosts": { | ||||
| 		"title": "Hostnames", | ||||
| 		"order": 40, | ||||
| 		"action": { | ||||
| 			"type": "view", | ||||
| 			"path": "network/hosts" | ||||
| 		}, | ||||
| 		"depends": { | ||||
| 			"uci": { "dhcp": true } | ||||
| 		} | ||||
| 	}, | ||||
| 
 | ||||
| 	"admin/network/routes": { | ||||
| 		"title": "Static Routes", | ||||
| 		"order": 50, | ||||
| 		"action": { | ||||
| 			"type": "view", | ||||
| 			"path": "network/routes" | ||||
| 		} | ||||
| 	}, | ||||
| 
 | ||||
| 	"admin/network/diagnostics": { | ||||
| 		"title": "Diagnostics", | ||||
| 		"order": 60, | ||||
| 		"action": { | ||||
| 			"type": "view", | ||||
| 			"path": "network/diagnostics" | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue