diff options
author | Hilman Maulana <hilman0.0maulana@gmail.com> | 2024-01-14 16:33:37 +0700 |
---|---|---|
committer | Paul Donald <itsascambutmailmeanyway@gmail.com> | 2024-02-01 18:08:42 +0100 |
commit | faf6c4f0283f14f2fdc95f298d65f11566c1aefa (patch) | |
tree | b68c7e7b2bf8ea59d8fd37200c0ad7dab9717c8e /applications/luci-app-cloudflared/htdocs/luci-static/resources/view/cloudflared | |
parent | f63e8234dbf43edd56424e54ee4e64771eebca03 (diff) |
luci-app-cloudflared: configure Cloudflare Zero Trust Tunnel
Configure a tunnel and see logs of the daemon.
Signed-off-by: Hilman Maulana <hilman0.0maulana@gmail.com>
Signed-off-by: Sergey Ponomarev <stokito@gmail.com>
Diffstat (limited to 'applications/luci-app-cloudflared/htdocs/luci-static/resources/view/cloudflared')
2 files changed, 243 insertions, 0 deletions
diff --git a/applications/luci-app-cloudflared/htdocs/luci-static/resources/view/cloudflared/config.js b/applications/luci-app-cloudflared/htdocs/luci-static/resources/view/cloudflared/config.js new file mode 100644 index 0000000000..d473a2536d --- /dev/null +++ b/applications/luci-app-cloudflared/htdocs/luci-static/resources/view/cloudflared/config.js @@ -0,0 +1,118 @@ +/* This is free software, licensed under the Apache License, Version 2.0 + * + * Copyright (C) 2024 Hilman Maulana <hilman0.0maulana@gmail.com> + */ + +'use strict'; +'require form'; +'require poll'; +'require rpc'; +'require uci'; +'require view'; + +var callServiceList = rpc.declare({ + object: 'service', + method: 'list', + params: ['name'], + expect: { '': {} } +}); + +function getServiceStatus() { + return L.resolveDefault(callServiceList('cloudflared'), {}).then(function (res) { + var isRunning = false; + try { + isRunning = res['cloudflared']['instances']['cloudflared']['running']; + } catch (ignored) {} + return isRunning; + }); +} + +function renderStatus(isRunning) { + var spanTemp = '<label class="cbi-value-title">Status</label><div class="cbi-value-field"><em><span style="color:%s">%s</span></em></div>'; + var renderHTML; + if (isRunning) { + renderHTML = String.format(spanTemp, 'green', _('Running')); + } else { + renderHTML = String.format(spanTemp, 'red', _('Not Running')); + } + + return renderHTML; +} + +return view.extend({ + load: function () { + return Promise.all([ + uci.load('cloudflared') + ]); + }, + + render: function (data) { + var m, s, o; + + m = new form.Map('cloudflared', _('Cloudflare Zero Trust Tunnel'), + _('Cloudflare Zero Trust Security services help you get maximum security both from outside and within the network.') + '<br />' + + _('Create and manage your network on the <a %s>Cloudflare Zero Trust</a> dashboard.') + .format('href="https://one.dash.cloudflare.com" target="_blank"') + '<br />' + + _('See <a %s>documentation</a>.') + .format('href="https://openwrt.org/docs/guide-user/services/vpn/cloudfare_tunnel" target="_blank"') + ); + + s = m.section(form.NamedSection, 'config', 'cloudflared'); + + o = s.option(form.DummyValue, 'service_status', _('Status')); + o.load = function () { + poll.add(function () { + return L.resolveDefault(getServiceStatus()).then(function (res) { + var view = document.getElementById('cbi-cloudflared-config-service_status'); + if (view) { + view.innerHTML = renderStatus(res); + } + }); + }); + }; + o.value = _('Collecting data...'); + + o = s.option(form.Flag, 'enabled', _('Enable')); + o.rmempty = false; + + o = s.option(form.TextValue, 'token', _('Token'), + _('The tunnel token is shown in the dashboard once you create a tunnel.') + ); + o.optional = true; + o.rmempty = false; + o.monospace = true; + + o = s.option(form.FileUpload, 'config', _('Config file path'), + _('See <a %s>documentation</a>.') + .format('href="https://developers.cloudflare.com/cloudflare-one/connections/connect-networks/configure-tunnels/local-management/configuration-file/" target="_blank"') + ); + o.default = '/etc/cloudflared/config.yml'; + o.root_directory = '/etc/cloudflared/'; + o.optional = true; + + o = s.option(form.FileUpload, 'origincert', _('Certificate of Origin'), + _('The account certificate for your zones authorizing the client to serve as an Origin for that zone') + '<br />' + + _('Obtain a certificate <a %s>here</a>.') + .format('href="https://dash.cloudflare.com/argotunnel" target="_blank"') + ); + o.default = '/etc/cloudflared/cert.pem'; + o.root_directory = '/etc/cloudflared/'; + o.optional = true; + + o = s.option(form.ListValue, 'region', _('Region'), + _('The region to which connections are established.') + ); + o.value('us', _('United States')); + o.optional = true; + + o = s.option(form.ListValue, 'loglevel', _('Debug level')); + o.value('fatal', _('Fatal')); + o.value('error', _('Error')); + o.value('warn', _('Warning')); + o.value('info', _('Info')); + o.value('debug', _('Debug')); + o.default = 'info'; + + return m.render(); + } +});
\ No newline at end of file diff --git a/applications/luci-app-cloudflared/htdocs/luci-static/resources/view/cloudflared/log.js b/applications/luci-app-cloudflared/htdocs/luci-static/resources/view/cloudflared/log.js new file mode 100644 index 0000000000..bc2c9e96fc --- /dev/null +++ b/applications/luci-app-cloudflared/htdocs/luci-static/resources/view/cloudflared/log.js @@ -0,0 +1,125 @@ +/* This is free software, licensed under the Apache License, Version 2.0 + * + * Copyright (C) 2024 Hilman Maulana <hilman0.0maulana@gmail.com> + */ + +'use strict'; +'require fs'; +'require ui'; +'require view'; +'require poll'; + +function formatLogEntry(logObj) { + var formattedTime = new Date(logObj.time).toISOString().replace('T', ' ').split('.')[0]; + var tunnelIDMessage = logObj.tunnelID ? ', ID: ' + logObj.tunnelID : ''; + var errorMessage = logObj.error ? ', Error: ' + logObj.error : ''; + var ipMessage = logObj.ip ? ', IP: ' + logObj.ip : ''; + var configMessage = logObj.config ? ', Config: ' + JSON.stringify(logObj.config) : ''; + var connectionMessage = logObj.connection ? ', Connection: ' + JSON.stringify(logObj.connection) : ''; + var locationMessage = logObj.location ? ', Location: ' + logObj.location : ''; + var protocolMessage = logObj.protocol ? ', Protocol: ' + logObj.protocol : ''; + + return '[' + formattedTime + '] [' + logObj.level + '] : ' + logObj.message + ipMessage + tunnelIDMessage + errorMessage + configMessage + connectionMessage + locationMessage + protocolMessage; +} + +return view.extend({ + handleSaveApply: null, + handleSave: null, + handleReset: null, + load: function() { + poll.add(function () { + return fs.read('/var/log/cloudflared.log').then(function(res) { + if (!res || res.trim() === '') { + ui.addNotification(null, E('p', {}, _('Unable to read the interface info from /var/log/cloudflared.log.'))); + return ''; + } + + var logs = res.trim().split('\n').map(function(entry) { + try { + var logObj = JSON.parse(entry); + return logObj.time && logObj.message && logObj.level + ? formatLogEntry(logObj) + : ''; + } catch (error) { + console.error('Error parsing log entry:', error); + return ''; + } + }); + + logs = logs.filter(function(entry) { + return entry.trim() !== ''; + }); + + var info = logs.join('\n'); + var view = document.getElementById('syslog'); + var filterLevel = document.getElementById('filter-level').value; + var logDirection = document.getElementById('log-direction').value; + + if (view) { + var filteredLogs; + if (filterLevel !== 'all') { + filteredLogs = logs.filter(function(entry) { + var logLevel = entry.match(/\[.*\] \[(.*)\]/)[1].toLowerCase(); + return logLevel.includes(filterLevel.toLowerCase()); + }); + } else { + filteredLogs = logs; + } + + if (logDirection === 'up') { + filteredLogs = filteredLogs.reverse(); + } + + info = filteredLogs.join('\n'); + view.innerHTML = info; + } + + return info; + }); + }); + + return Promise.resolve(''); + }, + render: function(info) { + return E([], [ + E('h2', { 'class': 'section-title' }, _('Log')), + E('div', { 'id': 'logs' }, [ + E('label', { 'for': 'filter-level', 'style': 'margin-right: 8px;' }, _('Filter Level:')), + E('select', { 'id': 'filter-level', 'style': 'margin-right: 8px;' }, [ + E('option', { 'value': 'all', 'selected': 'selected' }, _('All')), + E('option', { 'value': 'info' }, _('Info')), + E('option', { 'value': 'warn' }, _('Warn')), + E('option', { 'value': 'error' }, _('Error')), + ]), + E('label', { 'for': 'log-direction', 'style': 'margin-right: 8px;' }, _('Log Direction:')), + E('select', { 'id': 'log-direction', 'style': 'margin-right: 8px;' }, [ + E('option', { 'value': 'down', 'selected': 'selected' }, _('Down')), + E('option', { 'value': 'up' }, _('Up')), + ]), + E('button', { + 'id': 'download-log', + 'class': 'cbi-button cbi-button-save', + 'click': L.bind(this.handleDownloadLog, this), + 'style': 'margin-bottom: 8px;' + }, _('Download Log')), + E('textarea', { + 'id': 'syslog', + 'class': 'cbi-input-textarea', + 'style': 'height: 500px; overflow-y: scroll;', + 'readonly': 'readonly', + 'wrap': 'off', + 'rows': 1 + }, [ info ]) + ]) + ]); + }, + + handleDownloadLog: function() { + var logs = document.getElementById('syslog').value; + var blob = new Blob([logs], { type: 'text/plain' }); + var link = document.createElement('a'); + link.href = window.URL.createObjectURL(blob); + link.download = 'cloudflared.log'; + link.click(); + } +}); |