diff --git a/files/app/main/httpproxy.ut b/files/app/main/httpproxy.ut new file mode 100755 index 00000000..270850b2 --- /dev/null +++ b/files/app/main/httpproxy.ut @@ -0,0 +1,72 @@ +{% +/* + * Part of AREDN® -- Used for creating Amateur Radio Emergency Data Networks + * Copyright (C) 2024 Tim Wilkinson + * See Contributors file for additional contributors + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation version 3 of the License. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + * + * Additional Terms: + * + * Additional use restrictions exist on the AREDN® trademark and logo. + * See AREDNLicense.txt for more info. + * + * Attributions to the AREDN® Project must be retained in the source code. + * If importing this code into a new or existing project attribution + * to the AREDN® project must be added to the source code. + * + * You must not misrepresent the origin of the material contained within. + * + * Modified versions must be modified to attribute to the original source + * and be marked in reasonable ways as differentiate it from the original + * version + */ +%} +{% +const url = request.env.QUERY_STRING; +const m = match(url, /^https?:\/\/(.+@)?([^/]+)\//); +let valid = false; +if (m) { + const host = replace(m[2], ".local.mesh", ""); + const s = fs.open("/etc/hosts"); + if (s) { + const k = lc(`\t${host}`); + for (let line = s.read("line"); length(line); line = s.read("line")) { + if (index(lc(line), k) !== -1) { + valid = true; + break; + } + } + s.close(); + } +} + +response.override = true; +if (valid) { + uhttpd.send("Status: 200 OK\r\nCache-Control: no-store\r\n\r\n"); + const s = fs.popen(`/usr/bin/wget -O - ${url}`); + if (s) { + for (;;) { + const d = s.read(10240); + if (!length(d)) { + break; + } + uhttpd.send(d); + } + s.close(); + } +} +else { + uhttpd.send("Status: 404 Not Found\r\nCache-Control: no-store\r\n\r\n"); +} +%} diff --git a/files/app/main/redirect.ut b/files/app/main/redirect.ut new file mode 100755 index 00000000..09bfd107 --- /dev/null +++ b/files/app/main/redirect.ut @@ -0,0 +1,61 @@ +{% +/* + * Part of AREDN® -- Used for creating Amateur Radio Emergency Data Networks + * Copyright (C) 2024 Tim Wilkinson + * See Contributors file for additional contributors + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation version 3 of the License. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + * + * Additional Terms: + * + * Additional use restrictions exist on the AREDN® trademark and logo. + * See AREDNLicense.txt for more info. + * + * Attributions to the AREDN® Project must be retained in the source code. + * If importing this code into a new or existing project attribution + * to the AREDN® project must be added to the source code. + * + * You must not misrepresent the origin of the material contained within. + * + * Modified versions must be modified to attribute to the original source + * and be marked in reasonable ways as differentiate it from the original + * version + */ +%} +{% +const url = request.env.QUERY_STRING; +const m = match(url, /^https?:\/\/(.+@)?([^/]+)\//); +let valid = false; +if (m) { + const host = replace(m[2], ".local.mesh", ""); + const s = fs.open("/etc/hosts"); + if (s) { + const k = lc(`\t${host}`); + for (let line = s.read("line"); length(line); line = s.read("line")) { + if (index(lc(line), k) !== -1) { + valid = true; + break; + } + } + s.close(); + } +} + +response.override = true; +if (valid) { + uhttpd.send(`Status: 307 Temporary Redirect\r\nLocation: ${url}\r\nCache-Control: no-store\r\n\r\n`); +} +else { + uhttpd.send("Status: 404 Not Found\r\nCache-Control: no-store\r\n\r\n"); +} +%} diff --git a/files/app/main/status/e/local-services.ut b/files/app/main/status/e/local-services.ut index 5f3ca75f..fac5c2ad 100755 --- a/files/app/main/status/e/local-services.ut +++ b/files/app/main/status/e/local-services.ut @@ -97,6 +97,7 @@ const templates = [ { name: "Axis camera", type: "camera", protocol: "http", port: 80, path: "jpg/image.jpg" }, { name: "Sunba Lite camera", type: "camera", protocol: "http", port: 80, path: "webcapture.jpg?command=snap&channel=1&user=USERNAME&password=PASSWORD" }, { name: "Sunba Performance camera", type: "camera", protocol: "http", port: 80, path: "images/snapshot.jpg" }, + { name: "Sunba Performance camera (redirect)", type: "camera", protocol: "http", port: 80, path: "a/redirect?http://USERNAME:PASSWORD@CAMERANAME/images/snapshot.jpg" }, { name: "Sunba Smart camera", type: "camera", protocol: "http", port: 80, path: "ISAPI/Custom/snapshot?authInfo=USERNAME:PASSWORD" }, { name: "NTP Server", type: "time", protocol: "ntp", port: 123 }, { name: "Streaming video", type: "video", protocol: "rtsp", port: 554 }, @@ -108,6 +109,8 @@ const templates = [ { name: "Proxmox Server", type: "server", protocol: "https", port: 8006 }, { name: "Generic HTTP", protocol: "http", port: 80 }, { name: "Generic HTTPS", protocol: "https", port: 443 }, + { name: "HTTP Proxy", protocol: "http", port: 80, path: "a/httpproxy?URL" }, + { name: "Redirect", protocol: "http", port: 80, path: "a/redirect?URL" }, ]; const services = []; const dhcp = configuration.getDHCP(); @@ -254,7 +257,7 @@ if (f) { %} : - / + / {% } %} @@ -519,7 +522,7 @@ if (f) { :// : - / + /