summaryrefslogtreecommitdiffhomepage
path: root/applications/luci-app-travelmate/htdocs/luci-static
diff options
context:
space:
mode:
Diffstat (limited to 'applications/luci-app-travelmate/htdocs/luci-static')
-rw-r--r--applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/logread.js42
-rw-r--r--applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/overview.js430
-rw-r--r--applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/stations.js879
3 files changed, 1351 insertions, 0 deletions
diff --git a/applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/logread.js b/applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/logread.js
new file mode 100644
index 0000000000..d004ed4d2f
--- /dev/null
+++ b/applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/logread.js
@@ -0,0 +1,42 @@
+'use strict';
+'require view';
+'require poll';
+'require fs';
+
+return view.extend({
+ load: function() {
+ return Promise.all([
+ L.resolveDefault(fs.stat('/sbin/logread'), null),
+ L.resolveDefault(fs.stat('/usr/sbin/logread'), null)
+ ]);
+ },
+ render: function(stat) {
+ var logger = stat[0] ? stat[0].path : stat[1] ? stat[1].path : null;
+ poll.add(function() {
+ return L.resolveDefault(fs.exec_direct(logger, ['-e', 'trm-'])).then(function(res) {
+ var log = document.getElementById("logfile");
+ if (res) {
+ log.value = res.trim();
+ }
+ else {
+ log.value = _('No travelmate related logs yet!');
+ }
+ log.scrollTop = log.scrollHeight;
+ });
+ });
+ return E('div', { class: 'cbi-map' },
+ E('div', { class: 'cbi-section' }, [
+ E('div', { class: 'cbi-section-descr' }, _('The syslog output, pre-filtered for travelmate related messages only.')),
+ E('textarea', {
+ 'id': 'logfile',
+ 'style': 'width: 100% !important; padding: 5px; font-family: monospace',
+ 'readonly': 'readonly',
+ 'wrap': 'off',
+ 'rows': 25
+ })
+ ]));
+ },
+ handleSaveApply: null,
+ handleSave: null,
+ handleReset: null
+});
diff --git a/applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/overview.js b/applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/overview.js
new file mode 100644
index 0000000000..0ea48766e6
--- /dev/null
+++ b/applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/overview.js
@@ -0,0 +1,430 @@
+'use strict';
+'require view';
+'require poll';
+'require fs';
+'require ui';
+'require uci';
+'require form';
+'require tools.widgets as widgets';
+
+/*
+ button handling
+*/
+async function handleAction(ev) {
+ if (ev === 'restart') {
+ fs.exec_direct('/etc/init.d/travelmate', [ev])
+ }
+ if (ev === 'setup') {
+ var ifaceValue = String(uci.get('travelmate', 'global', 'trm_iface') || '');
+ L.ui.showModal(_('Interface Wizard'), [
+ E('p', _('To use Travelmate, you have to set up an uplink interface once. This wizard creates an IPv4- and an IPv6 alias network interface with all required network- and firewall settings.')),
+ E('div', { 'class': 'left', 'style': 'display:flex; flex-direction:column' }, [
+ E('label', { 'class': 'cbi-input-text', 'style': 'padding-top:.5em' }, [
+ E('input', { 'class': 'cbi-input-text', 'id': 'iface', 'placeholder': 'trm_wwan', 'value': ifaceValue, 'maxlength': '15', 'spellcheck': 'false' }, [
+ ]),
+ '\xa0\xa0\xa0',
+ _('The uplink interface name')
+ ]),
+ E('label', { 'class': 'cbi-input-text', 'style': 'padding-top:.5em' }, [
+ E('input', { 'class': 'cbi-input-text', 'id': 'zone', 'placeholder': 'wan', 'maxlength': '15', 'spellcheck': 'false' }),
+ '\xa0\xa0\xa0',
+ _('The firewall zone name')
+ ]),
+ E('label', { 'class': 'cbi-input-text', 'style': 'padding-top:.5em' }, [
+ E('input', { 'class': 'cbi-input-text', 'id': 'metric', 'placeholder': '100', 'maxlength': '3', 'spellcheck': 'false' }),
+ '\xa0\xa0\xa0',
+ _('The interface metric')
+ ])
+ ]),
+ E('div', { 'class': 'right' }, [
+ E('button', {
+ 'class': 'btn',
+ 'click': L.hideModal
+ }, _('Dismiss')),
+ ' ',
+ E('button', {
+ 'class': 'cbi-button cbi-button-positive important',
+ 'click': ui.createHandlerFn(this, function(ev) {
+ var iface = document.getElementById('iface').value || 'trm_wwan',
+ zone = document.getElementById('zone').value || 'wan',
+ metric = document.getElementById('metric').value || '100';
+ L.resolveDefault(fs.exec_direct('/etc/init.d/travelmate', ['setup', iface, zone, metric]))
+ .then(function(res) {
+ if (res) {
+ ui.addNotification(null, E('p', res.trim() + '.'), 'error');
+ } else {
+ ui.addNotification(null, E('p', _('The uplink interface has been updated.')), 'info');
+ }
+ });
+ L.hideModal();
+ })
+ }, _('Save'))
+ ])
+ ]);
+ return document.getElementById('iface').focus();
+ }
+
+ if (ev === 'qrcode') {
+ return Promise.all([
+ uci.load('wireless')
+ ]).then(function() {
+ var w_sid, w_device, w_ssid, w_enc, w_key, w_hidden, result,
+ w_sections = uci.sections('wireless', 'wifi-iface'),
+ optionsAP = [E('option', { value: '' }, [_('-- AP Selection --')])];
+ for (var i = 0; i < w_sections.length; i++) {
+ if (w_sections[i].mode === 'ap' && w_sections[i].disabled !== '1') {
+ w_sid = i;
+ w_device = w_sections[i].device;
+ w_ssid = w_sections[i].ssid;
+ optionsAP.push(E('option', { value: w_sid }, w_device + ', ' + w_ssid));
+ }
+ }
+ var selectAP = E('select', {
+ id: 'selectID',
+ class: 'cbi-input-select',
+ change: function(ev) {
+ result = document.getElementById('qrcode');
+ if (document.getElementById("selectID").value) {
+ w_sid = document.getElementById("selectID").value;
+ w_ssid = w_sections[w_sid].ssid;
+ w_enc = w_sections[w_sid].encryption;
+ w_key = w_sections[w_sid].key;
+ w_hidden = (w_sections[w_sid].hidden == 1 ? 'true' : 'false');
+ if (w_enc.startsWith('psk')) {
+ w_enc = 'WPA';
+ }
+ else if (w_enc === 'none') {
+ w_enc = 'nopass';
+ w_key = 'nokey';
+ }
+ L.resolveDefault(fs.exec_direct('/usr/bin/qrencode', ['--inline', '--8bit', '--type=SVG', '--output=-', 'WIFI:S:' + w_ssid + ';T:' + w_enc + ';P:' + w_key + ';H:' + w_hidden + ';']), null).then(function(res) {
+ if (res) {
+ result.innerHTML = res.trim();
+ }
+ else {
+ result.innerHTML = _('The QR-Code could not be generated!');
+ }
+ });
+ }
+ else {
+ result.innerHTML = '';
+ }
+ }
+ }, optionsAP);
+ L.ui.showModal(_('QR-Code Overview'), [
+ E('p', _('Render the QR-Code of the selected Access Point to comfortably transfer the WLAN credentials to your mobile devices.')),
+ E('div', { 'class': 'left', 'style': 'display:flex; flex-direction:column' }, [
+ E('label', { 'class': 'cbi-input-select', 'style': 'padding-top:.5em' }, [
+ selectAP,
+ ])
+ ]),
+ '\xa0',
+ E('div', {
+ 'id': 'qrcode'
+ }),
+ E('div', { 'class': 'right' }, [
+ E('button', {
+ 'class': 'btn',
+ 'click': L.hideModal
+ }, _('Dismiss'))
+ ])
+ ]);
+ });
+ return;
+ }
+}
+
+return view.extend({
+ load: function() {
+ return Promise.all([
+ uci.load('travelmate')
+ ]);
+ },
+
+ render: function(result) {
+ var m, s, o;
+
+ m = new form.Map('travelmate', 'Travelmate', _('Configuration of the travelmate package to to enable travel router functionality. \
+ For further information <a href="https://github.com/openwrt/packages/blob/master/net/travelmate/files/README.md" target="_blank" rel="noreferrer noopener" >check the online documentation</a>. <br /> \
+ <em>Please note:</em> On first start please call the \'Interface Wizard\' once, to make the necessary network- and firewall settings.'));
+
+ /*
+ poll runtime information
+ */
+ pollData: poll.add(function() {
+ return L.resolveDefault(fs.stat('/tmp/trm_runtime.json'), null).then(function(res) {
+ var status = document.getElementById('status');
+ if (res) {
+ L.resolveDefault(fs.read_direct('/tmp/trm_runtime.json'), null).then(function(res) {
+ if (res) {
+ var info = JSON.parse(res);
+ if (status && info) {
+ status.textContent = (info.data.travelmate_status || '-') + ' / ' + (info.data.travelmate_version || '-');
+ if (info.data.travelmate_status.startsWith('running')) {
+ if (!status.classList.contains("spinning")) {
+ status.classList.add("spinning");
+ }
+ } else {
+ if (status.classList.contains("spinning")) {
+ status.classList.remove("spinning");
+ }
+ }
+ } else if (status) {
+ status.textContent = '-';
+ if (status.classList.contains("spinning")) {
+ status.classList.remove("spinning");
+ }
+ }
+ var station_id = document.getElementById('station_id');
+ if (station_id && info) {
+ station_id.textContent = info.data.station_id || '-';
+ }
+ var station_mac = document.getElementById('station_mac');
+ if (station_mac && info) {
+ station_mac.textContent = info.data.station_mac || '-';
+ }
+ var station_interface = document.getElementById('station_interface');
+ if (station_interface && info) {
+ station_interface.textContent = info.data.station_interface || '-';
+ }
+ var wpa_flags = document.getElementById('wpa_flags');
+ if (wpa_flags && info) {
+ wpa_flags.textContent = info.data.wpa_flags || '-';
+ }
+ var run_flags = document.getElementById('run_flags');
+ if (run_flags && info) {
+ run_flags.textContent = info.data.run_flags || '-';
+ }
+ var ext_hooks = document.getElementById('ext_hooks');
+ if (ext_hooks && info) {
+ ext_hooks.textContent = info.data.ext_hooks || '-';
+ }
+ var run = document.getElementById('run');
+ if (run && info) {
+ run.textContent = info.data.last_run || '-';
+ }
+ }
+ });
+ }
+ else {
+ if (status && status.classList.contains("spinning")) {
+ status.textContent = '-';
+ status.classList.remove("spinning");
+ }
+ }
+ });
+ }, 1);
+
+ /*
+ runtime information and buttons
+ */
+ s = m.section(form.NamedSection, 'global');
+ s.render = L.bind(function(view, section_id) {
+ return E('div', { 'class': 'cbi-section' }, [
+ E('h3', _('Information')),
+ E('div', { 'class': 'cbi-value', 'style': 'margin-bottom:5px' }, [
+ E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Status / Version')),
+ E('div', { 'class': 'cbi-value-field', 'id': 'status', 'style': 'font-weight: bold;margin-bottom:5px;color:#37c' },'-')]),
+ E('div', { 'class': 'cbi-value', 'style': 'margin-bottom:5px' }, [
+ E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Station ID')),
+ E('div', { 'class': 'cbi-value-field', 'id': 'station_id', 'style': 'font-weight: bold;margin-bottom:5px;color:#37c' },'-')]),
+ E('div', { 'class': 'cbi-value', 'style': 'margin-bottom:5px' }, [
+ E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Station MAC')),
+ E('div', { 'class': 'cbi-value-field', 'id': 'station_mac', 'style': 'font-weight: bold;margin-bottom:5px;color:#37c' },'-')]),
+ E('div', { 'class': 'cbi-value', 'style': 'margin-bottom:5px' }, [
+ E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Station Interface')),
+ E('div', { 'class': 'cbi-value-field', 'id': 'station_interface', 'style': 'font-weight: bold;margin-bottom:5px;color:#37c' },'-')]),
+ E('div', { 'class': 'cbi-value', 'style': 'margin-bottom:5px' }, [
+ E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('WPA Flags')),
+ E('div', { 'class': 'cbi-value-field', 'id': 'wpa_flags', 'style': 'font-weight: bold;margin-bottom:5px;color:#37c' },'-')]),
+ E('div', { 'class': 'cbi-value', 'style': 'margin-bottom:5px' }, [
+ E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Run Flags')),
+ E('div', { 'class': 'cbi-value-field', 'id': 'run_flags', 'style': 'font-weight: bold;margin-bottom:5px;color:#37c' },'-')]),
+ E('div', { 'class': 'cbi-value', 'style': 'margin-bottom:5px' }, [
+ E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Ext. Hooks')),
+ E('div', { 'class': 'cbi-value-field', 'id': 'ext_hooks', 'style': 'font-weight: bold;margin-bottom:5px;color:#37c' },'-')]),
+ E('div', { 'class': 'cbi-value', 'style': 'margin-bottom:5px' }, [
+ E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Last Run')),
+ E('div', { 'class': 'cbi-value-field', 'id': 'run', 'style': 'font-weight: bold;margin-bottom:5px;color:#37c' },'-')]),
+ E('div', { class: 'right' }, [
+ E('button', {
+ 'class': 'cbi-button cbi-button-apply',
+ 'id': 'btn_suspend',
+ 'click': ui.createHandlerFn(this, function() {
+ L.resolveDefault(fs.stat('/usr/bin/qrencode'), null).then(function(res) {
+ if (res) {
+ return handleAction('qrcode');
+ }
+ return ui.addNotification(null, E('p', _('Please install the separate \'qrencode\' package.')), 'info');
+ })
+ })
+ }, [ _('AP QR-Codes...') ]),
+ '\xa0',
+ E('button', {
+ 'class': 'cbi-button cbi-button-reset',
+ 'click': ui.createHandlerFn(this, function() {
+ return handleAction('setup');
+ })
+ }, [ _('Interface Wizard...') ])
+ ])
+ ]);
+ }, o, this);
+ this.pollData;
+
+ /*
+ tabbed config section
+ */
+ s = m.section(form.NamedSection, 'global', 'travelmate', _('Settings'));
+ s.addremove = false;
+ s.tab('general', _('General Settings'));
+ s.tab('additional', _('Additional Settings'));
+ s.tab('adv_vpn', _('VPN Settings'), _('Please note: VPN connections require the separate setup of the <em>Wireguard</em> or <em>OpenVPN</em> package.<br /><p>&#xa0;</p>'));
+ s.tab('adv_email', _('E-Mail Settings'), _('Please note: E-Mail notifications require the separate setup of the <em>mstmp</em> package.<br /><p>&#xa0;</p>'));
+
+ /*
+ general settings tab
+ */
+ o = s.taboption('general', form.Flag, 'trm_enabled', _('Enabled'), _('Enable the travelmate service.'));
+ o.rmempty = false;
+
+ o = s.taboption('general', form.Flag, 'trm_debug', _('Verbose Debug Logging'), _('Enable verbose debug logging in case of any processing errors.'));
+ o.rmempty = false;
+
+ o = s.taboption('general', form.Value, 'trm_radio', _('Radio Selection'), _('Restrict travelmate to a single radio or change the overall scanning order (e.g. \'radio1 radio0\').'));
+ o.placeholder = 'radio0';
+ o.rmempty = true;
+
+ o = s.taboption('general', form.Flag, 'trm_captive', _('Captive Portal Detection'), _('Check the internet availability, handle captive portal redirections and keep the uplink connection \'alive\'.'));
+ o.default = 1;
+ o.rmempty = false;
+
+ o = s.taboption('general', form.Flag, 'trm_netcheck', _('Net Error Check'), _('Treat missing internet availability as an error.'));
+ o.depends('trm_captive', '1');
+ o.default = 0;
+ o.rmempty = false;
+
+ o = s.taboption('general', form.Flag, 'trm_proactive', _('ProActive Uplink Switch'), _('Proactively scan and switch to a higher prioritized uplink, despite of an already existing connection.'));
+ o.default = 1;
+ o.rmempty = false;
+
+ o = s.taboption('general', form.Flag, 'trm_autoadd', _('AutoAdd Open Uplinks'), _('Automatically add open uplinks like hotel captive portals to your wireless config.'));
+ o.default = 0;
+ o.rmempty = false;
+
+ o = s.taboption('general', form.Flag, 'trm_randomize', _('Randomize MAC Addresses'), _('Generate a random unicast MAC address for each uplink connection.'));
+ o.default = 0;
+ o.rmempty = false;
+
+ /*
+ additional settings tab
+ */
+ o = s.taboption('additional', form.Value, 'trm_triggerdelay', _('Trigger Delay'), _('Additional trigger delay in seconds before travelmate processing begins.'));
+ o.placeholder = '2';
+ o.datatype = 'range(1,60)';
+ o.rmempty = true;
+
+ o = s.taboption('additional', form.Value, 'trm_maxretry', _('Connection Limit'), _('Retry limit to connect to an uplink.'));
+ o.placeholder = '3';
+ o.datatype = 'range(1,10)';
+ o.rmempty = true;
+
+ o = s.taboption('additional', form.Value, 'trm_minquality', _('Signal Quality Threshold'), _('Minimum signal quality threshold as percent for conditional uplink (dis-) connections.'));
+ o.placeholder = '35';
+ o.datatype = 'range(20,80)';
+ o.rmempty = true;
+
+ o = s.taboption('additional', form.Value, 'trm_maxwait', _('Interface Timeout'), _('How long should travelmate wait for a successful wlan uplink connection.'));
+ o.placeholder = '30';
+ o.datatype = 'range(20,40)';
+ o.rmempty = true;
+
+ o = s.taboption('additional', form.Value, 'trm_timeout', _('Overall Timeout'), _('Overall retry timeout in seconds.'));
+ o.placeholder = '60';
+ o.datatype = 'range(30,300)';
+ o.rmempty = true;
+
+ o = s.taboption('additional', form.Value, 'trm_scanbuffer', _('Scan Buffer Size'), _('Buffer size in bytes to prepare nearby scan results.'));
+ o.placeholder = '1024';
+ o.datatype = 'range(256,4096)';
+ o.rmempty = true;
+
+ o = s.taboption('additional', form.ListValue, 'trm_captiveurl', _('Captive Portal URL'), _('The selected URL will be used for connectivity- and captive portal checks.'));
+ o.value('http://captive.apple.com', 'Apple (default)');
+ o.value('http://connectivity-check.ubuntu.com', 'Ubuntu');
+ o.value('http://connectivitycheck.android.com/generate_204', 'Google');
+ o.value('http://www.msftncsi.com/ncsi.txt', 'Microsoft');
+ o.optional = true;
+ o.rmempty = true;
+
+ o = s.taboption('additional', form.ListValue, 'trm_useragent', _('User Agent'), _('The selected user agent will be used for connectivity- and captive portal checks.'));
+ o.value('Mozilla/5.0 (X11; Linux x86_64; rv:80.0) Gecko/20100101 Firefox/80.0', 'Firefox (default)');
+ o.value('Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/85.0.4183.83 Safari/537.36', 'Chromium');
+ o.value('Mozilla/5.0 (iPhone; CPU iPhone OS 13_7 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/85.0.4183.92 Mobile/15E148 Safari/604.1', 'Safari');
+ o.value('Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/85.0.4183.83 Safari/537.36 Edg/85.0.564.44', 'Edge');
+ o.value('Mozilla/5.0 (Linux; Android 10; SM-G970F) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/85.0.4183.81 Mobile Safari/537.36 OPR/59.1.2926.54067', 'Opera');
+ o.optional = true;
+ o.rmempty = true;
+
+ o = s.taboption('additional', form.ListValue, 'trm_nice', _('Service Priority'), _('The selected priority will be used for travelmate processes.'));
+ o.value('-20', 'Highest Priority');
+ o.value('-10', 'High Priority');
+ o.value('0', 'Normal Priority (default)');
+ o.value('10', 'Less Priority');
+ o.value('19', 'Least Priority');
+ o.optional = true;
+ o.rmempty = true;
+
+ /*
+ advanced vpn settings tab
+ */
+ o = s.taboption('adv_vpn', form.Flag, 'trm_vpn', _('VPN Hook'), _('Automatically handle VPN (re-) connections.'));
+ o.rmempty = false;
+
+ o = s.taboption('adv_vpn', form.ListValue, 'trm_vpnservice', _('VPN Service'));
+ o.depends('trm_vpn', '1');
+ o.value('wireguard');
+ o.value('openvpn');
+ o.rmempty = true;
+
+ o = s.taboption('adv_vpn', widgets.NetworkSelect, 'trm_vpniface', _('VPN Interface'), _('The logical vpn network interface, e.g. \'wg0\' or \'tun0\'.'));
+ o.depends('trm_vpn', '1');
+ o.unspecified = false;
+ o.nocreate = true;
+ o.rmempty = true;
+
+ o = s.taboption('adv_vpn', widgets.DeviceSelect, 'trm_landevice', _('LAN Device'), _('The lan network device, e.g. \'br-lan\'.'));
+ o.depends('trm_vpn', '1');
+ o.unspecified = false;
+ o.nocreate = true;
+ o.rmempty = true;
+
+ /*
+ advanced email settings tab
+ */
+ o = s.taboption('adv_email', form.Flag, 'trm_mail', _('E-Mail Hook'), _('Sends notification E-Mails after every succesful uplink connect.'));
+ o.rmempty = false;
+
+ o = s.taboption('adv_email', form.Value, 'trm_mailreceiver', _('E-Mail Receiver Address'), _('Receiver address for travelmate notification E-Mails.'));
+ o.depends('trm_mail', '1');
+ o.placeholder = 'name@example.com';
+ o.rmempty = true;
+
+ o = s.taboption('adv_email', form.Value, 'trm_mailsender', _('E-Mail Sender Address'), _('Sender address for travelmate notification E-Mails.'));
+ o.depends({ 'trm_mailreceiver': '@', '!contains': true });
+ o.placeholder = 'no-reply@travelmate';
+ o.rmempty = true;
+
+ o = s.taboption('adv_email', form.Value, 'trm_mailtopic', _('E-Mail Topic'), _('Topic for travelmate notification E-Mails.'));
+ o.depends({ 'trm_mailreceiver': '@', '!contains': true });
+ o.placeholder = 'travelmate connection to \'<station>\'';
+ o.rmempty = true;
+
+ o = s.taboption('adv_email', form.Value, 'trm_mailprofile', _('E-Mail Profile'), _('Profile used by \'msmtp\' for travelmate notification E-Mails.'));
+ o.depends({ 'trm_mailreceiver': '@', '!contains': true });
+ o.placeholder = 'trm_notify';
+ o.rmempty = true;
+
+ return m.render();
+ },
+ handleReset: null
+});
diff --git a/applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/stations.js b/applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/stations.js
new file mode 100644
index 0000000000..d997636300
--- /dev/null
+++ b/applications/luci-app-travelmate/htdocs/luci-static/resources/view/travelmate/stations.js
@@ -0,0 +1,879 @@
+'use strict';
+'require view';
+'require poll';
+'require fs';
+'require ui';
+'require uci';
+'require form';
+'require network';
+'require tools.widgets as widgets';
+
+/*
+ remove wireless and stale travelmate sections
+*/
+function handleRemove(sid) {
+ var w_sections, t_sections, match, changes;
+
+ uci.remove('wireless', sid);
+ w_sections = uci.sections('wireless', 'wifi-iface');
+ t_sections = uci.sections('travelmate', 'uplink');
+ for (var i = 0; i < t_sections.length; i++) {
+ match = false;
+ for (var j = 0; j < w_sections.length; j++) {
+ if (t_sections[i].device === w_sections[j].device && t_sections[i].ssid === w_sections[j].ssid && t_sections[i].bssid === w_sections[j].bssid) {
+ match = true;
+ break;
+ }
+ }
+ if (match === false) {
+ uci.remove('travelmate', t_sections[i]['.name']);
+ }
+ }
+ uci.save();
+}
+
+/*
+ add missing travelmate sections
+*/
+function handleSectionsAdd(iface) {
+ var w_sections, t_sections, match, changes;
+
+ w_sections = uci.sections('wireless', 'wifi-iface');
+ t_sections = uci.sections('travelmate', 'uplink');
+ for (var i = 0; i < w_sections.length; i++) {
+ if (w_sections[i].mode !== 'sta' || w_sections[i].network !== iface) {
+ continue;
+ }
+ match = false;
+ for (var j = 0; j < t_sections.length; j++) {
+ if (w_sections[i].device === t_sections[j].device && w_sections[i].ssid === t_sections[j].ssid && w_sections[i].bssid === t_sections[j].bssid) {
+ match = true;
+ break;
+ }
+ }
+ if (match === false) {
+ var sid = uci.add('travelmate', 'uplink');
+ uci.set('travelmate', sid, 'enabled', '1');
+ uci.set('travelmate', sid, 'device', w_sections[i].device);
+ uci.set('travelmate', sid, 'ssid', w_sections[i].ssid);
+ uci.set('travelmate', sid, 'bssid', w_sections[i].bssid);
+ uci.set('travelmate', sid, 'con_start_expiry', '0');
+ uci.set('travelmate', sid, 'con_end_expiry', '0');
+ }
+ }
+}
+
+/*
+ update travelmate sections
+*/
+function handleSectionsVal(action, section_id, option, value) {
+ var date, oldValue, w_device, w_ssid, w_bssid, t_sections;
+
+ w_device = uci.get('wireless', section_id, 'device');
+ w_ssid = uci.get('wireless', section_id, 'ssid');
+ w_bssid = uci.get('wireless', section_id, 'bssid');
+ t_sections = uci.sections('travelmate', 'uplink');
+
+ for (var i = 0; i < t_sections.length; i++) {
+ if (t_sections[i].device === w_device && t_sections[i].ssid === w_ssid && t_sections[i].bssid === w_bssid) {
+ if (action === 'get') {
+ return t_sections[i][option];
+ }
+ else if (action === 'set') {
+ if (option === 'enabled') {
+ oldValue = t_sections[i][option];
+ if (oldValue !== value && value === '0') {
+ date = new Date(new Date().getTime() - new Date().getTimezoneOffset()*60*1000).toISOString().substr(0,19).replace(/-/g, '.').replace('T', '-');
+ uci.set('travelmate', t_sections[i]['.name'], 'con_end', date);
+ }
+ else if (oldValue !== value && value === '1') {
+ uci.unset('travelmate', t_sections[i]['.name'], 'con_end');
+ }
+ }
+ return uci.set('travelmate', t_sections[i]['.name'], option, value);
+ }
+ else if (action === 'del') {
+ return uci.unset('travelmate', t_sections[i]['.name'], option);
+ }
+ }
+ }
+}
+
+/*
+ update travelmate status
+*/
+function handleStatus() {
+ poll.add(function() {
+ L.resolveDefault(fs.stat('/var/run/travelmate.refresh'), null).then(function(res) {
+ if (res) {
+ L.resolveDefault(fs.read_direct('/var/run/travelmate.refresh'), null).then(function(res) {
+ fs.remove('/var/run/travelmate.refresh');
+ if (res && res === 'ui_reload') {
+ location.reload();
+ }
+ else if (res && res === 'cfg_reload') {
+ if (document.readyState === 'complete') {
+ uci.unload('wireless');
+ uci.unload('travelmate');
+ }
+ return Promise.all([
+ uci.load('wireless'),
+ uci.load('travelmate')
+ ]).then(function() {
+ var item, value,
+ container = document.querySelectorAll('.cbi-section-table-row[data-sid]');
+ for (var i = 0; i < container.length; i++) {
+ item = container[i].querySelector('.cbi-value-field[data-title="Enabled"]');
+ value = handleSectionsVal('get', container[i].getAttribute('data-sid'), 'enabled');
+ item.textContent = (value == 0 ? 'No' : 'Yes');
+ }
+ });
+ }
+ });
+ }
+ });
+ return L.resolveDefault(fs.stat('/tmp/trm_runtime.json'), null).then(function(res) {
+ if (res) {
+ L.resolveDefault(fs.read_direct('/tmp/trm_runtime.json'), null).then(function(res) {
+ if (res) {
+ var info = JSON.parse(res);
+ if (info) {
+ var t_device, t_ssid, t_bssid, oldUplinkView, newUplinkView,
+ uplinkId = info.data.station_id.trim().split('/'),
+ oldUplinkView = document.getElementsByName('uplinkStation'),
+ w_sections = uci.sections('wireless', 'wifi-iface');
+
+ t_device = uplinkId[0];
+ t_bssid = uplinkId[uplinkId.length-1];
+ for (var i = 1; i < uplinkId.length-1; i++) {
+ if (!t_ssid) {
+ t_ssid = uplinkId[i];
+ }
+ else {
+ t_ssid = t_ssid + '/' + uplinkId[i];
+ }
+ }
+ if (t_ssid === '-') {
+ if (oldUplinkView.length > 0) {
+ oldUplinkView[0].removeAttribute('style');
+ oldUplinkView[0].removeAttribute('name', 'uplinkStation');
+ }
+ }
+ else {
+ for (var i = 0; i < w_sections.length; i++) {
+ newUplinkView = document.getElementById('cbi-wireless-' + w_sections[i]['.name']);
+ if (t_device === w_sections[i].device && t_ssid === w_sections[i].ssid && t_bssid === (w_sections[i].bssid || '-')) {
+ if (oldUplinkView.length === 0 && newUplinkView) {
+ newUplinkView.setAttribute('name', 'uplinkStation');
+ newUplinkView.setAttribute('style', 'text-align: left !important; color: #37c !important;font-weight: bold !important;');
+ }
+ else if (oldUplinkView.length > 0 && newUplinkView && oldUplinkView[0].getAttribute('id') !== newUplinkView.getAttribute('id')) {
+ oldUplinkView[0].removeAttribute('style');
+ oldUplinkView[0].removeAttribute('name', 'uplinkStation');
+ newUplinkView.setAttribute('name', 'uplinkStation');
+ newUplinkView.setAttribute('style', 'text-align: left !important; color: #37c !important;font-weight: bold !important;');
+ }
+ }
+ }
+ }
+ }
+ }
+ });
+ }
+ });
+ }, 1);
+}
+
+return view.extend({
+ load: function() {
+ return Promise.all([
+ uci.load('wireless'),
+ uci.load('travelmate')
+ ]);
+ },
+
+ render: function() {
+ var m, s, o,
+ iface = uci.get('travelmate', 'global', 'trm_iface') || 'trm_wwan';
+
+ m = new form.Map('wireless');
+ m.chain('travelmate');
+ s = m.section(form.GridSection, 'wifi-iface', null, _('Overview of all configured uplinks for travelmate.<br /> \
+ You can edit, remove or prioritize existing uplinks by drag \&#38; drop and scan for new ones. The currently used uplink is emphasized in blue.'));
+ s.anonymous = true;
+ s.sortable = true;
+ s.filter = function(section_id) {
+ return (uci.get('wireless', section_id, 'network') == iface && uci.get('wireless', section_id, 'mode') == 'sta');
+ };
+ s.tab('wireless', _('Wireless Settings'));
+ s.tab('travelmate', _('Travelmate Settings'));
+ s.renderRowActions = function(section_id) {
+ var btns;
+ btns = [
+ E('button', {
+ 'class': 'btn cbi-button drag-handle center',
+ 'title': _('Drag to reorder'),
+ 'style': 'cursor:move',
+ 'disabled': this.map.readonly || null
+ }, '☰'),
+ E('button', {
+ 'class': 'cbi-button cbi-button-action important',
+ 'title': _('Edit this network'),
+ 'click': ui.createHandlerFn(this, 'renderMoreOptionsModal', section_id)
+ }, _('Edit')),
+ E('button', {
+ 'class': 'cbi-button cbi-button-negative remove',
+ 'title': _('Delete this network'),
+ 'click': ui.createHandlerFn(this, handleRemove, section_id)
+ }, _('Del'))
+ ];
+ return E('div', { 'class': 'td middle cbi-section-actions' }, E('div', btns));
+ };
+
+ o = s.taboption('travelmate', form.Flag, '_enabled', _('Enabled'));
+ o.uciconfig = 'travelmate';
+ o.ucisection = 'uplink';
+ o.ucioption = 'enabled';
+ o.rmempty = false;
+ o.cfgvalue = function(section_id) {
+ return handleSectionsVal('get', section_id, 'enabled');
+ }
+ o.write = function(section_id, value) {
+ return handleSectionsVal('set', section_id, 'enabled', value);
+ }
+
+ o = s.taboption('wireless', form.Value, 'device', _('Device'));
+ o.readonly = true;
+
+ o = s.taboption('wireless', form.Value, 'ssid', _('SSID'));
+ o.datatype = 'maxlength(32)';
+ o.readonly = true;
+
+ o = s.taboption('wireless', form.Value, 'bssid', _('BSSID'));
+ o.datatype = 'macaddr';
+ o.readonly = true;
+
+ o = s.taboption('wireless', form.ListValue, 'encryption', _('Encryption'));
+ o.value('sae', _('WPA3 Pers.'));
+ o.value('sae-mixed', _('WPA2/WPA3 Pers. (CCMP)'));
+ o.value('psk2', _('WPA2 Pers.'));
+ o.value('psk2+ccmp', _('WPA2 Pers. (CCMP)'));
+ o.value('psk2+tkip', _('WPA2 Pers. (TKIP)'));
+ o.value('psk', _('WPA Pers.'));
+ o.value('psk+ccmp', _('WPA Pers. (CCMP)'));
+ o.value('psk+tkip', _('WPA Pers. (TKIP)'));
+ o.value('psk-mixed+ccmp', _('WPA/WPA2 Pers. (CCMP)'));
+ o.value('psk-mixed+tkip', _('WPA/WPA2 Pers. (TKIP)'));
+ o.value('wpa3', _('WPA3 Ent.'));
+ o.value('wpa3-mixed', _('WPA3/WPA2 Ent.'));
+ o.value('wpa2+ccmp', _('WPA2 Ent. (CCMP)'));
+ o.value('wpa2+tkip', _('WPA2 Ent. (TKIP)'));
+ o.value('wpa+ccmp', _('WPA Ent. (CCMP)'));
+ o.value('wpa+tkip', _('WPA Ent. (TKIP)'));
+ o.value('wpa-mixed+ccmp', _('WPA/WPA2 Ent. (CCMP)'));
+ o.value('wpa-mixed+tkip', _('WPA/WPA2 Ent. (TKIP)'));
+ o.value('owe', _('OWE'));
+ o.value('none', _('none'));
+ o.default = 'none';
+ o.textvalue = function(section_id) {
+ var cfgvalue = this.map.data.get('wireless', section_id, 'encryption');
+ switch (cfgvalue) {
+ case 'sae':
+ cfgvalue = 'WPA3 Pers. (SAE)';
+ break;
+ case 'sae-mixed':
+ cfgvalue = 'WPA2/WPA3 Pers. (CCMP)';
+ break;
+ case 'psk2':
+ cfgvalue = 'WPA2 Pers.';
+ break;
+ case 'psk2+ccmp':
+ cfgvalue = 'WPA2 Pers. (CCMP)';
+ break;
+ case 'psk2+tkip':
+ cfgvalue = 'WPA2 Ent. (TKIP)';
+ break;
+ case 'psk':
+ cfgvalue = 'WPA Pers.';
+ break;
+ case 'psk-mixed+ccmp':
+ cfgvalue = 'WPA/WPA2 Pers. (CCMP)';
+ break;
+ case 'psk-mixed+tkip':
+ cfgvalue = 'WPA/WPA2 Pers. (TKIP)';
+ break;
+ case 'wpa3':
+ cfgvalue = 'WPA3 Ent.';
+ break;
+ case 'wpa3-mixed':
+ cfgvalue = 'WPA3/WPA2 Ent.';
+ break;
+ case 'wpa2+ccmp':
+ cfgvalue = 'WPA2 Ent. (CCMP)';
+ break;
+ case 'wpa2+tkip':
+ cfgvalue = 'WPA2 Ent. (TKIP)';
+ break;
+ case 'wpa+ccmp':
+ cfgvalue = 'WPA Ent. (CCMP)';
+ break;
+ case 'wpa+tkip':
+ cfgvalue = 'WPA Ent. (TKIP)';
+ break;
+ case 'wpa-mixed+ccmp':
+ cfgvalue = 'WPA/WPA2 Ent. (CCMP)';
+ break;
+ case 'wpa-mixed+tkip':
+ cfgvalue = 'WPA/WPA2 Ent. (TKIP)';
+ break;
+ case 'owe':
+ cfgvalue = 'WPA3 OWE (CCMP)';
+ break;
+ case 'none':
+ cfgvalue = 'none';
+ break;
+ }
+ return cfgvalue;
+ };
+ handleStatus();
+
+ /*
+ modal wireless tab
+ */
+ o = s.taboption('wireless', form.Value, 'key', _('Password'));
+ o.datatype = 'wpakey';
+ o.depends({ encryption: 'sae', '!contains': true });
+ o.depends({ encryption: 'psk', '!contains': true });
+ o.depends({ encryption: 'wpa', '!contains': true });
+ o.modalonly = true;
+ o.password = true;
+
+ o = s.taboption('wireless', form.ListValue, 'eap_type', _('EAP-Method'));
+ o.value('tls', _('TLS'));
+ o.value('ttls', _('TTLS'));
+ o.value('peap', _('PEAP'));
+ o.value('fast', _('FAST'));
+ o.default = 'peap';
+ o.depends({ encryption: 'wpa', '!contains': true });
+ o.modalonly = true;
+
+ o = s.taboption('wireless', form.ListValue, 'auth', _('Authentication'));
+ o.value('PAP', _('PAP'));
+ o.value('CHAP', _('CHAP'));
+ o.value('MSCHAP', _('MSCHAP'));
+ o.value('MSCHAPV2', _('MSCHAPV2'));
+ o.value('EAP-GTC', _('EAP-GTC'));
+ o.value('EAP-MD5', _('EAP-MD5'));
+ o.value('EAP-MSCHAPV2', _('EAP-MSCHAPV2'));
+ o.value('EAP-TLS', _('EAP-TLS'));
+ o.value('auth=PAP', _('auth=PAP'));
+ o.value('auth=MSCHAPV2', _('auth=MSCHAPV2'));
+ o.default = 'EAP-MSCHAPV2';
+ o.depends({ encryption: 'wpa', '!contains': true });
+ o.modalonly = true;
+
+ o = s.taboption('wireless', form.Value, 'identify', _('Identify'));
+ o.depends({ encryption: 'wpa', '!contains': true });
+ o.modalonly = true;
+
+ o = s.taboption('wireless', form.Value, 'ca_cert', _('Path to CA-Certificate'));
+ o.depends({ eap_type: 'tls' });
+ o.modalonly = true;
+ o.rmempty = true;
+
+ o = s.taboption('wireless', form.Value, 'client_cert', _('Path to Client-Certificate'));
+ o.depends({ eap_type: 'tls' });
+ o.modalonly = true;
+ o.rmempty = true;
+
+ o = s.taboption('wireless', form.Value, 'priv_key', _('Path to Private Key'));
+ o.depends({ eap_type: 'tls' });
+ o.modalonly = true;
+ o.rmempty = true;
+
+ o = s.taboption('wireless', form.Value, 'priv_key_pwd', _('Password of Private Key'));
+ o.datatype = 'wpakey';
+ o.depends({ eap_type: 'tls' });
+ o.modalonly = true;
+ o.password = true;
+ o.rmempty = true;
+
+ /*
+ modal travelmate tab
+ */
+ o = s.taboption('travelmate', form.Value, '_ssid', _('SSID'));
+ o.modalonly = true;
+ o.uciconfig = 'travelmate';
+ o.ucisection = 'uplink';
+ o.ucioption = 'ssid';
+ o.rmempty = false;
+ o.readonly = true;
+ o.cfgvalue = function(section_id) {
+ return handleSectionsVal('get', section_id, 'ssid');
+ }
+
+ o = s.taboption('travelmate', form.Value, '_bssid', _('BSSID'));
+ o.modalonly = true;
+ o.uciconfig = 'travelmate';
+ o.ucisection = 'uplink';
+ o.ucioption = 'bssid';
+ o.rmempty = true;
+ o.readonly = true;
+ o.cfgvalue = function(section_id) {
+ return handleSectionsVal('get', section_id, 'bssid');
+ }
+
+ o = s.taboption('travelmate', form.Value, '_con_start', _('Connection Start'));
+ o.modalonly = true;
+ o.uciconfig = 'travelmate';
+ o.ucisection = 'uplink';
+ o.ucioption = 'con_start';
+ o.rmempty = true;
+ o.readonly = true;
+ o.cfgvalue = function(section_id) {
+ return handleSectionsVal('get', section_id, 'con_start');
+ }
+
+ o = s.taboption('travelmate', form.Value, '_con_end', _('Connection End'));
+ o.modalonly = true;
+ o.uciconfig = 'travelmate';
+ o.ucisection = 'uplink';
+ o.ucioption = 'con_end';
+ o.rmempty = true;
+ o.readonly = true;
+ o.cfgvalue = function(section_id) {
+ return handleSectionsVal('get', section_id, 'con_end');
+ }
+
+ o = s.taboption('travelmate', form.Value, '_con_start_expiry', _('Connection Start Expiry'),
+ _('Automatically disable the uplink after <em>n</em> minutes, e.g. for timed connections.<br /> \
+ The default of \'0\' disables this feature.'));
+ o.modalonly = true;
+ o.uciconfig = 'travelmate';
+ o.ucisection = 'uplink';
+ o.ucioption = 'con_start_expiry';
+ o.rmempty = false;
+ o.placeholder = '0';
+ o.default = '0';
+ o.datatype = 'range(0,720)';
+ o.cfgvalue = function(section_id) {
+ return handleSectionsVal('get', section_id, 'con_start_expiry');
+ }
+ o.write = function(section_id, value) {
+ return handleSectionsVal('set', section_id, 'con_start_expiry', value);
+ }
+
+ o = s.taboption('travelmate', form.Value, '_con_end_expiry', _('Connection End Expiry'),
+ _('Automatically (re-)enable the uplink after <em>n</em> minutes, e.g. after failed login attempts.<br /> \
+ The default of \'0\' disables this feature.'));
+ o.modalonly = true;
+ o.uciconfig = 'travelmate';
+ o.ucisection = 'uplink';
+ o.ucioption = 'con_end_expiry';
+ o.rmempty = false;
+ o.placeholder = '0';
+ o.default = '0';
+ o.datatype = 'range(0,720)';
+ o.cfgvalue = function(section_id) {
+ return handleSectionsVal('get', section_id, 'con_end_expiry');
+ }
+ o.write = function(section_id, value) {
+ return handleSectionsVal('set', section_id, 'con_end_expiry', value);
+ }
+
+ o = s.taboption('travelmate', form.FileUpload, '_script', _('Auto Login Script'),
+ _('External script reference which will be called for automated captive portal logins.'));
+ o.root_directory = '/etc/travelmate';
+ o.enable_remove = false;
+ o.enable_upload = false;
+ o.modalonly = true;
+ o.uciconfig = 'travelmate';
+ o.ucisection = 'uplink';
+ o.ucioption = 'script';
+ o.renderWidget = function(section_id, option_index, cfgvalue) {
+ var browserEl = new ui.FileUpload((cfgvalue != null) ? cfgvalue : this.default, {
+ id: this.cbid(section_id),
+ name: this.cbid(section_id),
+ show_hidden: this.show_hidden,
+ enable_upload: this.enable_upload,
+ enable_remove: this.enable_remove,
+ root_directory: this.root_directory,
+ disabled: (this.readonly != null) ? this.readonly : this.map.readonly
+ });
+ browserEl.renderListing = function(container, path, list) {
+ return ui.FileUpload.prototype.renderListing.apply(this, [
+ container, path,
+ list.filter(function(entry) {
+ return ((entry.type == 'directory') || (entry.type == 'file' && entry.name.match(/\.login$/)));
+ })
+ ]);
+ };
+ return browserEl.render();
+ };
+ o.cfgvalue = function(section_id) {
+ return handleSectionsVal('get', section_id, 'script');
+ }
+ o.write = function(section_id, value) {
+ return handleSectionsVal('set', section_id, 'script', value);
+ }
+ o.remove = function(section_id) {
+ return handleSectionsVal('del', section_id, 'script');
+ }
+
+ o = s.taboption('travelmate', form.Value, '_args', _('Script Arguments'),
+ _('Space separated list of additional arguments passed to the Auto Login Script, i.e. username and password'));
+ o.modalonly = true;
+ o.uciconfig = 'travelmate';
+ o.ucisection = 'uplink';
+ o.ucioption = 'script_args';
+ o.rmempty = true;
+ o.depends({ _script: '/etc/travelmate', '!contains': true });
+ o.cfgvalue = function(section_id) {
+ return handleSectionsVal('get', section_id, 'script_args');
+ }
+ o.write = function(section_id, value) {
+ return handleSectionsVal('set', section_id, 'script_args', value);
+ }
+ o.remove = function(section_id) {
+ return handleSectionsVal('del', section_id, 'script_args');
+ }
+
+ /*
+ scan buttons
+ */
+ s = m.section(form.GridSection, 'wifi-device');
+ s.anonymous = true;
+ s.addremove = false;
+ s.render = function() {
+ return network.getWifiDevices().then(L.bind(function(radios) {
+ var radio, ifname, btns = [];
+ for (var i = 0; i < radios.length; i++) {
+ radio = radios[i].sid;
+ if (radio) {
+ btns.push(E('button', {
+ 'class': 'cbi-button cbi-button-apply',
+ 'id': radio,
+ 'click': ui.createHandlerFn(this, 'handleScan', radio)
+ }, [ _('Scan on ' + radio + '...') ]),
+ '\xa0')
+ }
+ }
+ return E('div', { 'class': 'left', 'style': 'display:flex; flex-direction:column' }, E('div', { 'class': 'left', 'style': 'padding-top:5px; padding-bottom:5px' }, btns));
+ }, this))
+ };
+
+ /*
+ modal 'scan' dialog
+ */
+ s.handleScan = function(radio) {
+ var table = E('div', { 'class': 'table' }, [
+ E('div', { 'class': 'tr table-titles' }, [
+ E('div', { 'class': 'th col-1 middle left' }, _('Strength')),
+ E('div', { 'class': 'th col-1 middle left hide-xs' }, _('Channel')),
+ E('div', { 'class': 'th col-2 middle left' }, _('SSID')),
+ E('div', { 'class': 'th col-2 middle left' }, _('BSSID')),
+ E('div', { 'class': 'th col-3 middle left' }, _('Encryption')),
+ E('div', { 'class': 'th cbi-section-actions right' }, '\xa0')
+ ])
+ ]);
+ cbi_update_table(table, [], E('em', { class: 'spinning' }, _('Starting wireless scan on \'' + radio + '\'...')));
+
+ var md = ui.showModal(_('Wireless Scan'), [
+ table,
+ E('div', { 'class': 'right' }, [
+ E('button', {
+ 'class': 'btn',
+ 'click': ui.hideModal
+ }, _('Dismiss')),
+ '\xa0',
+ E('button', {
+ 'class': 'cbi-button cbi-button-positive important',
+ 'click': L.bind(this.handleScan, this, radio)
+ }, _('Repeat Scan'))
+ ])
+ ]);
+
+ md.style.maxWidth = '90%';
+ md.style.maxHeight = 'none';
+
+ return L.resolveDefault(fs.exec_direct('/etc/init.d/travelmate', [ 'scan', radio ]), null)
+ .then(L.bind(function(res) {
+ if (res) {
+ var lines, strength, channel, encryption, tbl_encryption, bssid, ssid, tbl_ssid, rows = [];
+ lines = res.trim().split('\n');
+ for (var i = 0; i < lines.length; i++) {
+ if (lines[i].match(/^\s+[0-9]/)) {
+ encryption = lines[i].slice(80).trim();
+ if (!encryption.includes('WEP')) {
+ strength = lines[i].slice(4,7).trim();
+ channel = lines[i].slice(15,18).trim();
+ bssid = lines[i].slice(60,77).trim();
+ ssid = lines[i].slice(25,59).trim();
+ if (ssid.startsWith('"')) {
+ ssid = ssid.slice(1, ssid.length-1);
+ tbl_ssid = ssid;
+ }
+ else {
+ ssid = "hidden";
+ tbl_ssid = "<em>hidden</em>";
+ }
+ switch (encryption) {
+ case 'WPA3 PSK (SAE)':
+ encryption = 'sae';
+ tbl_encryption = 'WPA3 Pers. (SAE)';
+ break;
+ case 'mixed WPA2/WPA3 PSK/SAE (CCMP)':
+ encryption = 'sae-mixed';
+ tbl_encryption = 'WPA2/WPA3 Pers. (CCMP)';
+ break;
+ case 'WPA2 PSK (CCMP)':
+ encryption = 'psk2+ccmp';
+ tbl_encryption = 'WPA2 Pers. (CCMP)';
+ break;
+ case 'WPA2 PSK (TKIP)':
+ encryption = 'psk2+tkip';
+ tbl_encryption = 'WPA2 Pers. (TKIP)';
+ break;
+ case 'mixed WPA/WPA2 PSK (TKIP, CCMP)':
+ encryption = 'psk-mixed+ccmp';
+ tbl_encryption = 'WPA/WPA2 Pers. (CCMP)';
+ break;
+ case 'WPA PSK (CCMP)':
+ encryption = 'psk2+ccmp';
+ tbl_encryption = 'WPA Pers. (CCMP)';
+ break;
+ case 'WPA PSK (TKIP)':
+ encryption = 'psk2+tkip';
+ tbl_encryption = 'WPA Pers. (TKIP)';
+ break;
+ case 'WPA2 802.1X (CCMP)':
+ encryption = 'wpa2+ccmp';
+ tbl_encryption = 'WPA2 Ent. (CCMP)';
+ break;
+ case 'WPA3 OWE (CCMP)':
+ encryption = 'owe';
+ tbl_encryption = 'WPA3 OWE (CCMP)';
+ break;
+ case 'none':
+ encryption = 'none';
+ tbl_encryption = 'none';
+ break;
+ }
+ rows.push([
+ strength,
+ channel,
+ tbl_ssid,
+ bssid,
+ tbl_encryption,
+ E('div', { 'class': 'right' }, E('button', {
+ 'class': 'cbi-button cbi-button-action',
+ 'click': ui.createHandlerFn(this, 'handleAdd', radio, iface, ssid, bssid, encryption)
+ }, _('Add Uplink...')))
+ ]);
+ }
+ }
+ else if (lines[i] === '::: No scan results') {
+ rows.push([
+ 'No scan results'
+ ]);
+ }
+ }
+ }
+ else {
+ rows.push([
+ 'No scan results'
+ ]);
+ }
+ cbi_update_table(table, rows);
+ }, this));
+ };
+
+ /*
+ modal 'add' dialog
+ */
+ s.handleAdd = function(radio, iface, ssid, bssid, encryption, ev) {
+ ui.hideModal;
+ var m2, s2, o2;
+
+ m2 = new form.Map('wireless'),
+ s2 = m2.section(form.NamedSection, '_add_trm');
+
+ s2.render = function() {
+ return Promise.all([
+ {},
+ this.renderUCISection('_add_trm')
+ ]).then(this.renderContents.bind(this));
+ };
+
+ o2 = s2.option(form.Value, 'device', _('Device Name'));
+ o2.default = radio;
+ o2.readonly = true;
+
+ o2 = s2.option(form.Value, 'network', _('Interface Name'));
+ o2.default = iface;
+ o2.readonly = true;
+
+ if (ssid === "hidden") {
+ o2 = s2.option(form.Value, 'ssid', _('SSID (hidden)'));
+ o2.placeholder = 'hidden SSID';
+ }
+ else {
+ o2 = s2.option(form.Value, 'ssid', _('SSID'));
+ o2.default = ssid;
+ }
+ o2.datatype = 'maxlength(32)';
+ o2.rmempty = false;
+
+ o2 = s2.option(form.Flag, 'ignore_bssid', _('Ignore BSSID'));
+ if (ssid === "hidden") {
+ o2.default = '0';
+ }
+ else {
+ o2.default = '1';
+ }
+
+ o2 = s2.option(form.Value, 'bssid', _('BSSID'));
+ o2.depends({ ignore_bssid: '0' });
+ o2.datatype = 'macaddr';
+ o2.rmempty = true;
+ o2.default = bssid;
+
+ o2 = s2.option(form.ListValue, 'encryption', _('Encryption'));
+ o2.value('sae', _('WPA3 Pers. (SAE)'));
+ o2.value('sae-mixed', _('WPA2/WPA3 Pers. (CCMP)'));
+ o2.value('psk2', _('WPA2 Pers.'));
+ o2.value('psk2+ccmp', _('WPA2 Pers. (CCMP)'));
+ o2.value('psk2+tkip', _('WPA2 Pers. (TKIP)'));
+ o2.value('psk', _('WPA Pers.'));
+ o2.value('psk+ccmp', _('WPA Pers. (CCMP)'));
+ o2.value('psk+tkip', _('WPA Pers. (TKIP)'));
+ o2.value('psk-mixed+ccmp', _('WPA/WPA2 Pers. (CCMP)'));
+ o2.value('psk-mixed+tkip', _('WPA/WPA2 Pers. (TKIP)'));
+ o2.value('wpa3', _('WPA3 Ent.'));
+ o2.value('wpa3-mixed', _('WPA2/WPA3 Ent.'));
+ o2.value('wpa2+ccmp', _('WPA2 Ent. (CCMP)'));
+ o2.value('wpa2+tkip', _('WPA2 Ent. (TKIP)'));
+ o2.value('wpa+ccmp', _('WPA Ent. (CCMP)'));
+ o2.value('wpa+tkip', _('WPA Ent. (TKIP)'));
+ o2.value('wpa-mixed+ccmp', _('WPA/WPA2 Ent. (CCMP)'));
+ o2.value('wpa-mixed+tkip', _('WPA/WPA2 Ent. (TKIP)'));
+ o2.value('owe', _('WPA3 OWE (CCMP)'));
+ o2.value('none', _('none'));
+ o2.default = encryption;
+
+ o2 = s2.option(form.Value, 'key', _('Password'));
+ o2.depends({ encryption: 'sae', '!contains': true });
+ o2.depends({ encryption: 'psk', '!contains': true });
+ o2.depends({ encryption: 'wpa', '!contains': true });
+ o2.datatype = 'wpakey';
+ o2.password = true;
+
+ o2 = s2.option(form.ListValue, 'eap_type', _('EAP-Method'));
+ o2.depends({ encryption: 'wpa', '!contains': true });
+ o2.value('tls', _('TLS'));
+ o2.value('ttls', _('TTLS'));
+ o2.value('peap', _('PEAP'));
+ o2.value('fast', _('FAST'));
+ o2.default = 'peap';
+
+ o2 = s2.option(form.ListValue, 'auth', _('Authentication'));
+ o2.depends({ encryption: 'wpa', '!contains': true });
+ o2.value('PAP', _('PAP'));
+ o2.value('CHAP', _('CHAP'));
+ o2.value('MSCHAP', _('MSCHAP'));
+ o2.value('MSCHAPV2', _('MSCHAPV2'));
+ o2.value('EAP-GTC', _('EAP-GTC'));
+ o2.value('EAP-MD5', _('EAP-MD5'));
+ o2.value('EAP-MSCHAPV2', _('EAP-MSCHAPV2'));
+ o2.value('EAP-TLS', _('EAP-TLS'));
+ o2.value('auth=PAP', _('auth=PAP'));
+ o2.value('auth=MSCHAPV2', _('auth=MSCHAPV2'));
+ o2.default = 'EAP-MSCHAPV2';
+
+ o2 = s2.option(form.Value, 'identify', _('Identify'));
+ o2.depends({ encryption: 'wpa', '!contains': true });
+
+ o2 = s2.option(form.Value, 'ca_cert', _('Path to CA-Certificate'));
+ o2.depends({ eap_type: 'tls' });
+ o2.rmempty = true;
+
+ o2 = s2.option(form.Value, 'client_cert', _('Path to Client-Certificate'));
+ o2.depends({ eap_type: 'tls' });
+ o2.rmempty = true;
+
+ o2 = s2.option(form.Value, 'priv_key', _('Path to Private Key'));
+ o2.depends({ eap_type: 'tls' });
+ o2.rmempty = true;
+
+ o2 = s2.option(form.Value, 'priv_key_pwd', _('Password of Private Key'));
+ o2.depends({ eap_type: 'tls' });
+ o2.datatype = 'wpakey';
+ o2.password = true;
+ o2.rmempty = true;
+
+ return m2.render().then(L.bind(function(elements) {
+ ui.showModal(_('Add Uplink %q').replace(/%q/, '"%h"'.format(ssid)), [
+ elements,
+ E('div', { 'class': 'right' }, [
+ E('button', {
+ 'class': 'btn',
+ 'click': ui.hideModal
+ }, _('Dismiss')),
+ '\xa0',
+ E('button', {
+ 'class': 'cbi-button cbi-button-positive important',
+ 'click': ui.createHandlerFn(this, 'handleSave', m2)
+ }, _('Save'))
+ ])
+ ]);
+ }, this));
+ };
+
+ /*
+ save new uplink
+ */
+ s.handleSave = function(map, ev) {
+ var w_sections = uci.sections('wireless', 'wifi-iface'),
+ device = L.toArray(map.lookupOption('device', '_add_trm'))[0].formvalue('_add_trm'),
+ network = L.toArray(map.lookupOption('network', '_add_trm'))[0].formvalue('_add_trm'),
+ ssid = L.toArray(map.lookupOption('ssid', '_add_trm'))[0].formvalue('_add_trm'),
+ ignore_bssid = L.toArray(map.lookupOption('ignore_bssid', '_add_trm'))[0].formvalue('_add_trm'),
+ bssid = L.toArray(map.lookupOption('bssid', '_add_trm'))[0].formvalue('_add_trm'),
+ encryption = L.toArray(map.lookupOption('encryption', '_add_trm'))[0].formvalue('_add_trm'),
+ password = L.toArray(map.lookupOption('key', '_add_trm'))[0].formvalue('_add_trm');
+ if (!ssid || ((encryption.includes('psk') || encryption.includes('wpa') || encryption.includes('sae')) && !password)) {
+ return;
+ }
+ for (var i = 0; i < w_sections.length; i++) {
+ if (w_sections[i].device === device && w_sections[i].ssid === ssid) {
+ if (ignore_bssid === '1' || (ignore_bssid === '0' && w_sections[i].bssid === bssid)) {
+ ui.hideModal();
+ return;
+ }
+ }
+ }
+
+ var offset = w_sections.length,
+ new_sid = 'trm_uplink' + (++offset);
+ while (uci.get('wireless', new_sid)) {
+ new_sid = 'trm_uplink' + (++offset);
+ }
+ uci.add('wireless', 'wifi-iface', new_sid);
+ uci.set('wireless', new_sid, 'device', device);
+ uci.set('wireless', new_sid, 'mode', 'sta');
+ uci.set('wireless', new_sid, 'network', network);
+ uci.set('wireless', new_sid, 'ssid', ssid);
+ if (ignore_bssid === '0') {
+ uci.set('wireless', new_sid, 'bssid', bssid);
+ }
+ uci.set('wireless', new_sid, 'encryption', encryption);
+ uci.set('wireless', new_sid, 'key', password);
+ uci.set('wireless', new_sid, 'disabled', '1');
+ handleSectionsAdd(network);
+ uci.save();
+ ui.hideModal();
+ };
+ return m.render();
+ },
+ handleReset: null
+});