'use strict'; 'require form'; 'require network'; 'require rpc'; 'require tools.widgets as widgets'; 'require uci'; 'require ui'; network.registerPatternVirtual(/^yggdrasil-.+$/); function validatePrivateKey(section_id,value) { if (value.length == 0) { return true; }; if (!value.match(/^([0-9a-fA-F]){128}$/)) { if (value != "auto") { return _('Invalid private key string %s').format(value); } return true; } return true; }; function validatePublicKey(section_id,value) { if (value.length == 0) { return true; }; if (!value.match(/^([0-9a-fA-F]){64}$/)) return _('Invalid public key string %s').format(value); return true; }; function validateYggdrasilListenUri(section_id,value) { if (value.length == 0) { return true; }; if (!value.match(/^(tls|tcp|unix|quic):\/\//)) return _('Unsupported URI scheme in %s').format(value); return true; }; function validateYggdrasilPeerUri(section_id,value) { if (!value.match(/^(tls|tcp|unix|quic|socks|sockstls):\/\//)) return _('URI scheme %s not supported').format(value); return true; }; var cbiKeyPairGenerate = form.DummyValue.extend({ cfgvalue: function(section_id, value) { return E('button', { 'class':'btn', 'click':ui.createHandlerFn(this, function(section_id,ev) { var prv = this.section.getUIElement(section_id,'private_key'), map = this.map; return generateKey().then(function(keypair){ prv.setValue(keypair.priv); map.save(null,true); }); },section_id) },[_('Generate new key pair')]); } }); function updateActivePeers(ifname) { getPeers(ifname).then(function(peers){ var table = document.querySelector('#yggdrasil-active-peerings-' + ifname); if (table) { while (table.rows.length > 1) { table.deleteRow(1); } peers.forEach(function(peer) { var row = table.insertRow(-1); row.style.fontSize = "xx-small"; if (!peer.up) { row.style.opacity = "66%"; } var cell = row.insertCell(-1) cell.className = "td" cell.textContent = peer.remote; cell = row.insertCell(-1) cell.className = "td" cell.textContent = peer.up ? "Up" : "Down"; cell = row.insertCell(-1) cell.className = "td" cell.textContent = peer.inbound ? "In" : "Out"; cell = row.insertCell(-1) cell.className = "td" cell.innerHTML = "" + peer.address + "" cell.dataToggle = "tooltip"; cell.title = "Key: " + peer.key; cell = row.insertCell(-1) cell.className = "td" cell.textContent = '%t'.format(peer.uptime); cell = row.insertCell(-1) cell.className = "td" cell.textContent = '%.2mB'.format(peer.bytes_recvd); cell = row.insertCell(-1) cell.className = "td" cell.textContent = '%.2mB'.format(peer.bytes_sent); cell = row.insertCell(-1) cell.className = "td" cell.textContent = peer.priority; cell = row.insertCell(-1) cell.className = "td" if (!peer.up) { cell.innerHTML = "%t ago".format(peer.last_error_time) cell.dataToggle = "tooltip" cell.title = peer.last_error } else { cell.innerHTML = "-" } }); setTimeout(updateActivePeers.bind(this, ifname), 5000); } }); } var cbiActivePeers = form.DummyValue.extend({ cfgvalue: function(section_id, value) { updateActivePeers(this.option); return E('table', { 'class': 'table', 'id': 'yggdrasil-active-peerings-' + this.option, },[ E('tr', {'class': 'tr'}, [ E('th', {'class': 'th'}, _('URI')), E('th', {'class': 'th'}, _('State')), E('th', {'class': 'th'}, _('Dir')), E('th', {'class': 'th'}, _('IP Address')), E('th', {'class': 'th'}, _('Uptime')), E('th', {'class': 'th'}, _('RX')), E('th', {'class': 'th'}, _('TX')), E('th', {'class': 'th'}, _('Priority')), E('th', {'class': 'th'}, _('Last Error')), ]) ]); } }); var generateKey = rpc.declare({ object:'luci.yggdrasil', method:'generateKeyPair', expect:{keys:{}} }); var getPeers = rpc.declare({ object:'luci.yggdrasil', method:'getPeers', params:['interface'], expect:{peers:[]} }); return network.registerProtocol('yggdrasil', { getI18n: function() { return _('Yggdrasil Network'); }, getIfname: function() { return this._ubus('l3_device') || this.sid; }, getType: function() { return "tunnel"; }, getOpkgPackage: function() { return 'yggdrasil'; }, isFloating: function() { return true; }, isVirtual: function() { return true; }, getDevices: function() { return null; }, containsDevice: function(ifname) { return(network.getIfnameOf(ifname)==this.getIfname()); }, renderFormOptions: function(s) { var o, ss; o=s.taboption('general',form.Value,'private_key',_('Private key'),_('The private key for your Yggdrasil node')); o.optional=false; o.password=true; o.validate=validatePrivateKey; s.taboption('general',cbiKeyPairGenerate,'_gen_server_keypair',' '); o=s.taboption('advanced',form.Value,'mtu',_('MTU'),_('A default MTU of 65535 is set by Yggdrasil. It is recomended to utilize the default.')); o.optional=true; o.placeholder=65535; o.datatype='range(1280, 65535)'; o=s.taboption('general',form.TextValue,'node_info',_('Node info'),_('Optional node info. This must be a { "key": "value", ... } map or set as null. This is entirely optional but, if set, is visible to the whole network on request.')); o.optional=true; o.placeholder="{}"; o=s.taboption('general',form.Flag,'node_info_privacy',_('Node info privacy'),_('Enable node info privacy so that only items specified in "Node info" are sent back. Otherwise defaults including the platform, architecture and Yggdrasil version are included.')); o.default=o.disabled; try { s.tab('peers',_('Peers')); } catch(e) {}; o=s.taboption('peers', form.SectionValue, '_active', form.NamedSection, this.sid, "interface", _("Active peers")) ss=o.subsection; ss.option(cbiActivePeers, this.sid); o=s.taboption('peers', form.SectionValue, '_listen', form.NamedSection, this.sid, "interface", _("Listen for peers")) ss=o.subsection; o=ss.option(form.DynamicList,'listen_address',_('Listen addresses'), _('Add listeners in order to accept incoming peerings from non-local nodes. Multicast peer discovery works regardless of listeners set here. URI Format: tls://0.0.0.0:0 or tls://[::]:0 to listen on all interfaces. Choose an acceptable URI tls://, tcp://, unix:// or quic://')); o.placeholder="tls://0.0.0.0:0" o.validate=validateYggdrasilListenUri; o=s.taboption('peers',form.DynamicList,'allowed_public_key',_('Accept from public keys'),_('If empty, all incoming connections will be allowed (default). This does not affect outgoing peerings, nor link-local peers discovered via multicast.')); o.validate=validatePublicKey; o=s.taboption('peers', form.SectionValue, '_peers', form.TableSection, 'yggdrasil_%s_peer'.format(this.sid), _("Peer addresses")) ss=o.subsection; ss.addremove=true; ss.anonymous=true; ss.addbtntitle=_("Add peer address"); o=ss.option(form.Value,"address",_("Peer URI")); o.placeholder="tls://0.0.0.0:0" o.validate=validateYggdrasilPeerUri; ss.option(widgets.NetworkSelect,"interface",_("Peer interface")); o=s.taboption('peers', form.SectionValue, '_interfaces', form.TableSection, 'yggdrasil_%s_interface'.format(this.sid), _("Multicast rules")) ss=o.subsection; ss.addbtntitle=_("Add multicast rule"); ss.addremove=true; ss.anonymous=true; o=ss.option(widgets.DeviceSelect,"interface",_("Devices")); o.multiple=true; ss.option(form.Flag,"beacon",_("Send multicast beacon")); ss.option(form.Flag,"listen",_("Listen to multicast beacons")); o=ss.option(form.Value,"port",_("Port")); o.optional=true; o.datatype='range(1, 65535)'; o=ss.option(form.Value,"password",_("Password")); o.optional=true; return; }, deleteConfiguration: function() { uci.sections('network', 'yggdrasil_%s_interface'.format(this.sid), function(s) { uci.remove('network', s['.name']); }); uci.sections('network', 'yggdrasil_%s_peer'.format(this.sid), function(s) { uci.remove('network', s['.name']); }); } } );