summaryrefslogtreecommitdiffhomepage
path: root/protocols/luci-proto-modemmanager
diff options
context:
space:
mode:
Diffstat (limited to 'protocols/luci-proto-modemmanager')
-rw-r--r--protocols/luci-proto-modemmanager/htdocs/luci-static/resources/modemmanager_helper.js101
-rw-r--r--protocols/luci-proto-modemmanager/htdocs/luci-static/resources/protocol/modemmanager.js113
-rw-r--r--protocols/luci-proto-modemmanager/htdocs/luci-static/resources/view/modemmanager/status.js181
-rw-r--r--protocols/luci-proto-modemmanager/root/usr/share/luci/menu.d/luci-proto-modemmanager.json13
-rw-r--r--protocols/luci-proto-modemmanager/root/usr/share/rpcd/acl.d/luci-proto-modemmanager.json6
5 files changed, 367 insertions, 47 deletions
diff --git a/protocols/luci-proto-modemmanager/htdocs/luci-static/resources/modemmanager_helper.js b/protocols/luci-proto-modemmanager/htdocs/luci-static/resources/modemmanager_helper.js
new file mode 100644
index 0000000000..b8558b885b
--- /dev/null
+++ b/protocols/luci-proto-modemmanager/htdocs/luci-static/resources/modemmanager_helper.js
@@ -0,0 +1,101 @@
+'use strict';
+'require baseclass';
+'require fs';
+
+return baseclass.extend({
+
+ _mmcliBin: '/usr/bin/mmcli',
+
+ _emptyStringValue: '--',
+
+ _parseIndex: function (dbusPath) {
+ var index = dbusPath.split('/').slice(-1);
+ return parseInt(index);
+ },
+
+ _parseOutput: function (output) {
+ try {
+ return this._removeEmptyStrings(JSON.parse(output));
+ } catch (err) {
+ return null;
+ }
+ },
+
+ _removeEmptyStrings: function (obj) {
+ if (obj == null) {
+ return obj;
+ }
+
+ if (typeof obj == 'string') {
+ if (obj == this._emptyStringValue) {
+ obj = null;
+ }
+ } else if (Array.isArray()) {
+ obj = obj.map(L.bind(function (it) {
+ return this._removeEmptyStrings(it);
+ }, this));
+ } else {
+ var keys = Object.keys(obj);
+ keys.forEach(L.bind(function (key) {
+ obj[key] = this._removeEmptyStrings(obj[key]);
+ }, this));
+ }
+
+ return obj;
+ },
+
+ getModems: function () {
+ return fs.exec_direct(this._mmcliBin, [ '-L', '-J' ]).then(L.bind(function (res) {
+ var json = this._parseOutput(res);
+ if (json == null) {
+ return [];
+ }
+ var modems = json['modem-list'];
+ var tasks = [];
+
+ modems.forEach(L.bind(function (modem) {
+ var index = this._parseIndex(modem);
+ if (!isNaN(index)) {
+ tasks.push(this.getModem(index));
+ }
+ }, this));
+ return Promise.all(tasks);
+ }, this));
+ },
+
+ getModem: function (index) {
+ return fs.exec_direct(this._mmcliBin, [ '-m', index, '-J' ]).then(L.bind(function (modem) {
+ return this._parseOutput(modem);
+ }, this));
+ },
+
+ getModemSims: function (modem) {
+ var tasks = [];
+ var simSlots = modem.generic['sim-slots'];
+ var sim = modem.generic.sim;
+ if (sim != null && !simSlots.includes(sim)) {
+ simSlots.push(sim);
+ }
+
+ simSlots.forEach(L.bind(function (modem) {
+ var index = this._parseIndex(modem);
+ if (!isNaN(index)) {
+ tasks.push(this.getSim(index));
+ }
+ }, this));
+ return Promise.all(tasks);
+ },
+
+ getSim: function (index) {
+ return fs.exec_direct(this._mmcliBin, [ '-i', index, '-J' ]).then(L.bind(function (sim) {
+ return this._parseOutput(sim);
+ }, this));
+ },
+
+ getModemLocation: function (modem) {
+ var index = this._parseIndex(modem['dbus-path']);
+ return fs.exec_direct(this._mmcliBin, [ '-m', index, '--location-get', '-J' ]).then(L.bind(function (location) {
+ return this._parseOutput(location);
+ }, this));
+ }
+});
diff --git a/protocols/luci-proto-modemmanager/htdocs/luci-static/resources/protocol/modemmanager.js b/protocols/luci-proto-modemmanager/htdocs/luci-static/resources/protocol/modemmanager.js
index ad8cc1484a..b13dd310c7 100644
--- a/protocols/luci-proto-modemmanager/htdocs/luci-static/resources/protocol/modemmanager.js
+++ b/protocols/luci-proto-modemmanager/htdocs/luci-static/resources/protocol/modemmanager.js
@@ -2,50 +2,19 @@
'require fs';
'require form';
'require network';
-
-function getModemList() {
- return fs.exec_direct('/usr/bin/mmcli', [ '-L' ]).then(function(res) {
- var lines = (res || '').split(/\n/),
- tasks = [];
-
- for (var i = 0; i < lines.length; i++) {
- var m = lines[i].match(/\/Modem\/(\d+)/);
- if (m)
- tasks.push(fs.exec_direct('/usr/bin/mmcli', [ '-m', m[1] ]));
- }
-
- return Promise.all(tasks).then(function(res) {
- var modems = [];
-
- for (var i = 0; i < res.length; i++) {
- var man = res[i].match(/manufacturer: ([^\n]+)/),
- mod = res[i].match(/model: ([^\n]+)/),
- dev = res[i].match(/device: ([^\n]+)/);
-
- if (dev) {
- modems.push({
- device: dev[1].trim(),
- manufacturer: (man ? man[1].trim() : '') || '?',
- model: (mod ? mod[1].trim() : '') || dev[1].trim()
- });
- }
- }
-
- return modems;
- });
- });
-}
+'require modemmanager_helper as helper';
network.registerPatternVirtual(/^mobiledata-.+$/);
network.registerErrorCode('MM_CONNECT_FAILED', _('Connection attempt failed.'));
-network.registerErrorCode('MM_DISCONNECT_IN_PROGRESS', _('Modem disconnection in progress. Please wait.'));
network.registerErrorCode('MM_CONNECT_IN_PROGRESS', _('Modem connection in progress. Please wait. This process will timeout after 2 minutes.'));
-network.registerErrorCode('MM_TEARDOWN_IN_PROGRESS', _('Modem bearer teardown in progress.'));
-network.registerErrorCode('MM_MODEM_DISABLED', _('Modem is disabled.'));
network.registerErrorCode('DEVICE_NOT_MANAGED', _('Device not managed by ModemManager.'));
network.registerErrorCode('INVALID_BEARER_LIST', _('Invalid bearer list. Possibly too many bearers created. This protocol supports one and only one bearer.'));
network.registerErrorCode('UNKNOWN_METHOD', _('Unknown and unsupported connection method.'));
network.registerErrorCode('DISCONNECT_FAILED', _('Disconnection attempt failed.'));
+network.registerErrorCode('MM_INVALID_ALLOWED_MODES_LIST', _('Unable to set allowed mode list.'));
+network.registerErrorCode('MM_NO_PREFERRED_MODE_CONFIGURED', _('No preferred mode configuration found.'));
+network.registerErrorCode('MM_NO_ALLOWED_MODE_CONFIGURED', _('No allowed mode configuration found.'));
+network.registerErrorCode('MM_FAILED_SETTING_PREFERRED_MODE', _('Unable to set preferred mode.'));
return network.registerProtocol('modemmanager', {
getI18n: function() {
@@ -79,19 +48,33 @@ return network.registerProtocol('modemmanager', {
renderFormOptions: function(s) {
var dev = this.getL3Device() || this.getDevice(), o;
- o = s.taboption('general', form.ListValue, 'device', _('Modem device'));
+ o = s.taboption('general', form.ListValue, '_modem_device', _('Modem device'));
+ o.ucioption = 'device';
o.rmempty = false;
o.load = function(section_id) {
- return getModemList().then(L.bind(function(devices) {
- for (var i = 0; i < devices.length; i++)
- this.value(devices[i].device,
- '%s - %s'.format(devices[i].manufacturer, devices[i].model));
+ return helper.getModems().then(L.bind(function(devices) {
+ for (var i = 0; i < devices.length; i++) {
+ var generic = devices[i].modem.generic;
+ this.value(generic.device,
+ '%s - %s'.format(generic.manufacturer, generic.model));
+ }
return form.Value.prototype.load.apply(this, [section_id]);
}, this));
};
- s.taboption('general', form.Value, 'apn', _('APN'));
- s.taboption('general', form.Value, 'pincode', _('PIN'));
+ o = s.taboption('general', form.Value, 'apn', _('APN'));
+ o.validate = function(section_id, value) {
+ if (value == null || value == '')
+ return true;
+
+ if (!/^[a-zA-Z0-9\-.]*[a-zA-Z0-9]$/.test(value))
+ return _('Invalid APN provided');
+
+ return true;
+ };
+
+ o = s.taboption('general', form.Value, 'pincode', _('PIN'));
+ o.datatype = 'and(uinteger,minlength(4),maxlength(8))';
o = s.taboption('general', form.ListValue, 'auth', _('Authentication Type'));
o.value('both', _('PAP/CHAP (both)'));
@@ -100,6 +83,45 @@ return network.registerProtocol('modemmanager', {
o.value('none', _('None'));
o.default = 'none';
+ o = s.taboption('general', form.ListValue, 'allowedmode', _('Allowed network technology'),
+ _('Setting the allowed network technology.'));
+ o.value('2g');
+ o.value('3g');
+ o.value('3g|2g');
+ o.value('4g');
+ o.value('4g|2g');
+ o.value('4g|3g');
+ o.value('4g|3g|2g');
+ o.value('5g');
+ o.value('5g|2g');
+ o.value('5g|3g');
+ o.value('5g|3g|2g');
+ o.value('5g|4g');
+ o.value('5g|4g|2g');
+ o.value('5g|4g|3g');
+ o.value('5g|4g|3g|2g');
+ o.value('',_('any'));
+ o.default = '';
+
+ o = s.taboption('general', form.ListValue, 'preferredmode', _('Preferred network technology'),
+ _('Setting the preferred network technology.'));
+ o.value('2g');
+ o.value('3g');
+ o.value('4g');
+ o.value('5g');
+ o.value('none', _('None'));
+ o.depends('allowedmode','3g|2g');
+ o.depends('allowedmode','4g|2g');
+ o.depends('allowedmode','4g|3g');
+ o.depends('allowedmode','4g|3g|2g');
+ o.depends('allowedmode','5g|2g');
+ o.depends('allowedmode','5g|3g');
+ o.depends('allowedmode','5g|3g|2g');
+ o.depends('allowedmode','5g|4g');
+ o.depends('allowedmode','5g|4g|2g');
+ o.depends('allowedmode','5g|4g|3g');
+ o.depends('allowedmode','5g|4g|3g|2g');
+
o = s.taboption('general', form.Value, 'username', _('PAP/CHAP username'));
o.depends('auth', 'pap');
o.depends('auth', 'chap');
@@ -117,11 +139,12 @@ return network.registerProtocol('modemmanager', {
o.value('ipv6', _('IPv6 only'));
o.default = 'ipv4v6';
- s.taboption('general', form.Value, 'signalrate', _('Signal refresh rate'));
-
o = s.taboption('advanced', form.Value, 'mtu', _('Override MTU'));
o.placeholder = dev ? (dev.getMTU() || '1500') : '1500';
o.datatype = 'max(9200)';
+
+ o = s.taboption('general', form.Value, 'signalrate', _('Signal Refresh Rate'), _("In seconds"));
+ o.datatype = 'uinteger';
s.taboption('general', form.Value, 'metric', _('Gateway metric'));
diff --git a/protocols/luci-proto-modemmanager/htdocs/luci-static/resources/view/modemmanager/status.js b/protocols/luci-proto-modemmanager/htdocs/luci-static/resources/view/modemmanager/status.js
new file mode 100644
index 0000000000..3db79051d3
--- /dev/null
+++ b/protocols/luci-proto-modemmanager/htdocs/luci-static/resources/view/modemmanager/status.js
@@ -0,0 +1,181 @@
+'use strict';
+'require ui';
+'require view';
+'require poll';
+'require dom';
+'require modemmanager_helper as helper';
+
+return view.extend({
+ load: function() {
+ return helper.getModems().then(function (modems) {
+ return Promise.all(modems.filter(function (modem){
+ return modem != null;
+ }).map(function (modem) {
+ return helper.getModemSims(modem.modem).then(function (sims) {
+ modem.sims = sims.filter(function (sim) {
+ return sim != null;
+ });
+
+ return helper.getModemLocation(modem.modem).then(function (location) {
+ modem.location = location;
+ return modem;
+ });
+ });
+ }));
+ });
+ },
+
+ pollData: function (container) {
+ poll.add(L.bind(function () {
+ return this.load().then(L.bind(function (modems) {
+ dom.content(container, this.renderContent(modems));
+ }, this));
+ }, this));
+ },
+
+ renderSections: function (name, sections) {
+ if (sections.length == 0) {
+ sections.push(E('div', { 'class': 'cbi-section' }, [
+ E('span', {}, _('Section %s is empty.').format(name))
+ ]));
+ }
+
+ return E('div', { 'class': 'cbi-section' }, [
+ E('h1', {}, name),
+ ...sections
+ ]);
+ },
+
+ renderSection: function (name, table) {
+ var rowNodes = table.filter(function (row) {
+ return row[1] != null;
+ }).map(function (row) {
+ return E('tr', { 'class': 'tr' }, [
+ E('td', { 'class': 'td', 'width': '33%' }, E('strong', {}, [row[0]])),
+ E('td', { 'class': 'td' }, [row[1]])
+ ]);
+ });
+
+ var tableNode;
+ if (rowNodes.length == 0) {
+ tableNode = E('div', { 'class': 'cbi-section' }, [
+ E('span', {}, _('Section %s is empty.').format(name))
+ ])
+ } else {
+ tableNode = E('table', { 'class': 'table', }, rowNodes);
+ }
+
+ return E('div', { 'class': 'cbi-section' }, [
+ E('h2', {}, [name]),
+ tableNode
+ ]);
+ },
+
+ renderContent: function (modems) {
+ var node = E('div', {}, E('div'));
+
+ modems.forEach(L.bind(function (modem) {
+ var generic = modem.modem.generic;
+ var modem3gpp = modem.modem['3gpp'];
+
+ var modemSection = this.renderSection(_('Modem Info'), [
+ [_('Manufacturer'), generic.manufacturer],
+ [_('Model'), generic.model],
+ [_('Revision'), generic.revision],
+ [E('abbr', { 'title': _('International Mobile Station Equipment Identity') }, [
+ _('IMEI')
+ ]), modem3gpp.imei],
+ [_('Device Identifier'), generic['device-identifier']],
+ [_('Power State'), generic['power-state']],
+ [_('State'), generic.state],
+ [_('Failed Reason'), generic['state-failed-reason']]
+ ]);
+
+ var ownNumbersStr = generic['own-numbers'].join(', ');
+ var accessTechnologiesStr = generic['access-technologies'].join(', ');
+ var signalQualityValue = parseInt(generic['signal-quality'].value);
+ var networkSection = this.renderSection(_('Network Registration'), [
+ [_('Own Numbers'), ownNumbersStr],
+ [_('Access Technologies'), accessTechnologiesStr],
+ [_('Operator') , modem3gpp['operator-name']],
+ [_('Operator Code'), modem3gpp['operator-code']],
+ [_('Registration State'), modem3gpp['registration-state']],
+ [_('Packet Service State'), modem3gpp['packet-service-state']],
+ [_('Signal Quality'), E('div', { 'class': 'cbi-progressbar', 'title': '%d %'.format(signalQualityValue) }, [
+ E('div', { 'style': 'width: %d%%'.format(signalQualityValue) })
+ ])]
+ ]);
+
+ var location3gpp = {};
+ if (modem.location != null) {
+ location3gpp = modem.location.modem.location['3gpp'];
+ }
+ var locationSection = this.renderSection(_('Cell Location'), [
+ [E('abbr', { 'title': _('Cell ID') }, [
+ 'CID'
+ ]), location3gpp.cid],
+ [E('abbr', { 'title': _('Location Area Code') }, [
+ 'LAC'
+ ]), location3gpp.lac],
+ [E('abbr', { 'title': _('Mobile Country Code') }, [
+ 'MCC'
+ ]), location3gpp.mcc],
+ [E('abbr', { 'title': _('Mobile Network Code') }, [
+ 'MNC'
+ ]), location3gpp.mnc],
+ [E('abbr', { 'title': _('Tracking Area Code') }, [
+ 'TAC'
+ ]), location3gpp.tac]
+ ]);
+
+ var simTables = modem.sims.map(function (sim) {
+ var properties = sim.sim.properties;
+ return [
+ [_('Active'), properties.active],
+ [_('Operator Name'), properties['operator-name']],
+ [E('abbr', { 'title': _('Integrated Circuit Card Identifier') }, [
+ 'ICCID'
+ ]), properties.iccid],
+ [E('abbr', { 'title': _('International Mobile Subscriber Identity') }, [
+ 'IMSI'
+ ]), properties.imsi]
+ ];
+ });
+ var simSubSections = simTables.map(L.bind(function (table, index) {
+ return this.renderSection(_('SIM %d').format(index + 1), table)
+ }, this));
+ var simSection = this.renderSections(_('SIMs'), simSubSections);
+
+ var sections = [
+ E('div', { 'class': 'cbi-map-descr'}, []),
+ modemSection,
+ networkSection,
+ locationSection,
+ simSection
+ ].filter(function (section) {
+ return section != null;
+ });
+ node.firstElementChild.appendChild(E('div', { 'data-tab': generic.device, 'data-tab-title': generic.device }, sections));
+ }, this));
+ ui.tabs.initTabGroup(node.firstElementChild.childNodes);
+
+ return node;
+ },
+
+ render: function (modems) {
+ var content = E([], [
+ E('h2', {}, [_('Mobile Service')]),
+ E('div')
+ ]);
+ var container = content.lastElementChild;
+
+ dom.content(container, this.renderContent(modems));
+ this.pollData(container);
+
+ return content;
+ },
+
+ handleSaveApply: null,
+ handleSave: null,
+ handleReset: null
+});
diff --git a/protocols/luci-proto-modemmanager/root/usr/share/luci/menu.d/luci-proto-modemmanager.json b/protocols/luci-proto-modemmanager/root/usr/share/luci/menu.d/luci-proto-modemmanager.json
new file mode 100644
index 0000000000..7104f221ce
--- /dev/null
+++ b/protocols/luci-proto-modemmanager/root/usr/share/luci/menu.d/luci-proto-modemmanager.json
@@ -0,0 +1,13 @@
+{
+ "admin/status/modemmanager": {
+ "title": "Mobile Service",
+ "order": 10,
+ "action": {
+ "type": "view",
+ "path": "modemmanager/status"
+ },
+ "depends": {
+ "acl": [ "luci-proto-modemmanager" ]
+ }
+ }
+}
diff --git a/protocols/luci-proto-modemmanager/root/usr/share/rpcd/acl.d/luci-proto-modemmanager.json b/protocols/luci-proto-modemmanager/root/usr/share/rpcd/acl.d/luci-proto-modemmanager.json
index 716f4c4655..cde3e9cbb0 100644
--- a/protocols/luci-proto-modemmanager/root/usr/share/rpcd/acl.d/luci-proto-modemmanager.json
+++ b/protocols/luci-proto-modemmanager/root/usr/share/rpcd/acl.d/luci-proto-modemmanager.json
@@ -4,8 +4,10 @@
"read": {
"cgi-io": [ "exec" ],
"file": {
- "/usr/bin/mmcli -L": [ "exec" ],
- "/usr/bin/mmcli -m [0-9]": [ "exec" ]
+ "/usr/bin/mmcli -L -J": [ "exec" ],
+ "/usr/bin/mmcli -m [0-9]* -J": [ "exec" ],
+ "/usr/bin/mmcli -i [0-9]* -J": [ "exec" ],
+ "/usr/bin/mmcli -m [0-9]* --location-get -J": [ "exec" ]
}
}
}