mirror of
				https://github.com/tobychui/zoraxy.git
				synced 2025-10-24 19:44:03 +02:00 
			
		
		
		
	Compare commits
	
		
			149 Commits
		
	
	
		
			v3.1.6
			...
			0e74ff69c3
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 0e74ff69c3 | ||
|   | f0fa71c5b4 | ||
|   | 8cb47e19fa | ||
|   | 9d2b8f224c | ||
|   | 877692695e | ||
|   | b9c609e413 | ||
|   | 7426cc2bb1 | ||
|   | bd71335f47 | ||
|   | 8bcc8e7095 | ||
|   | b1824a66a3 | ||
|   | e2882b6436 | ||
|   | 61b873451f | ||
|   | cc6501db12 | ||
|   | 70d95bd4e4 | ||
|   | b590e15ef2 | ||
|   | b25f8aab3e | ||
|   | c0578a33b6 | ||
|   | 55a525106a | ||
|   | e3b68b9aad | ||
|   | 3f1c50c009 | ||
|   | 8f046a0b47 | ||
|   | 0e5550487e | ||
|   | 9781735983 | ||
|   | a98d86a303 | ||
|   | 73e6530862 | ||
|   | 0c753ae531 | ||
|   | 6353cc532a | ||
|   | e049761f36 | ||
|   | 4dc7175588 | ||
|   | ffc67ede12 | ||
|   | 6750c7fe3d | ||
|   | 36c2c9a00e | ||
|   | 4f026e8c07 | ||
|   | 72b100aab0 | ||
|   | 291f12e5ea | ||
|   | 0c8dfd8aa0 | ||
|   | 76e2861fea | ||
|   | b23b967165 | ||
|   | d682d52eb7 | ||
|   | 23eeeee701 | ||
|   | e961e52dea | ||
|   | b863a9720f | ||
|   | ca7cd0476c | ||
|   | a3cccee162 | ||
|   | b9b992a817 | ||
|   | 19d5695f1a | ||
|   | bcfc777d15 | ||
|   | caa64ada76 | ||
|   | ac91a3fef1 | ||
|   | 05f1743ecd | ||
|   | d4c1225f75 | ||
|   | f245a61d32 | ||
|   | 5c2b8e4c31 | ||
|   | f6eef46d3f | ||
|   | 3adc669db9 | ||
|   | 85201885f0 | ||
|   | 44b65d1bfa | ||
|   | 6cb9e8e427 | ||
|   | d4b1cc8c57 | ||
|   | 0e749e8a41 | ||
|   | 2c219eceef | ||
|   | 92a27cbeb8 | ||
|   | b8a47dc620 | ||
|   | c4266559be | ||
|   | 136989f2ea | ||
|   | 3e031605fc | ||
|   | eb265e3e94 | ||
|   | 8504ff16cb | ||
|   | b71437058f | ||
|   | 4d16758e0a | ||
|   | f2b4c47805 | ||
|   | 7dff4f83b4 | ||
|   | eb24bc0391 | ||
|   | dac3e8c925 | ||
|   | 3f1c1f1395 | ||
|   | cd15fdf3c1 | ||
|   | 0fdfda436b | ||
|   | f8270e46c2 | ||
|   | 4a99afa2f0 | ||
|   | dfd5ef5578 | ||
|   | 3e57a90bb6 | ||
|   | 23d4df1ed7 | ||
|   | 39d6d16c2a | ||
|   | b7e3888513 | ||
|   | fd41a1cb91 | ||
|   | 75c351e7e2 | ||
|   | 6a8057c3a7 | ||
|   | ebf6ad6600 | ||
|   | 549e492ffd | ||
|   | 6351f25c00 | ||
|   | 560b0058cd | ||
|   | 28a0a837ba | ||
|   | 14e1341c34 | ||
|   | 5abc4ac606 | ||
|   | 214b69b0b8 | ||
|   | 3993ac954c | ||
|   | 53657e8716 | ||
|   | bddff0cf2f | ||
|   | dd4df0b4db | ||
|   | 85709dacf6 | ||
|   | ad13b33283 | ||
|   | 20959cd6cc | ||
|   | 394cf50e1d | ||
|   | 1116b643b5 | ||
|   | 2e9d70da83 | ||
|   | 6130459f7c | ||
|   | 2d29065812 | ||
|   | 2be7f711ba | ||
|   | de9d3bfb65 | ||
|   | 3e4c66b34f | ||
|   | 895ee1e53f | ||
|   | caf4ab331b | ||
|   | 36c1f149e6 | ||
|   | b0dc4d6670 | ||
|   | 5d8bec7f24 | ||
|   | 32f60dfba6 | ||
|   | 0abe4c12cf | ||
|   | 7555611ba5 | ||
|   | e624227dae | ||
|   | 27695584ab | ||
|   | e47a7a8357 | ||
|   | 3246f8ea2c | ||
|   | ccbda6d7c2 | ||
|   | a7285438af | ||
|   | 693dba07b7 | ||
|   | 9b64278200 | ||
|   | d04eff2bda | ||
|   | 3320b56b19 | ||
|   | 99728144b3 | ||
|   | 05511ed4ca | ||
|   | 70abfe6fcf | ||
|   | 6ab91c377f | ||
|   | 1863af0d63 | ||
|   | 2a9d87787d | ||
|   | f753becd66 | ||
|   | bb2d0d5b46 | ||
|   | 07dc63a82c | ||
|   | 97a6cf016a | ||
|   | 8df68f1f4e | ||
|   | e4ad505f2a | ||
|   | a402c4f326 | ||
|   | 791fbfa1b4 | ||
|   | c49f2fd1db | ||
|   | 7d9f240d56 | ||
|   | e20f816080 | ||
|   | eeb438eb18 | ||
|   | bfd64a885e | ||
|   | 45f61b3053 | ||
|   | 0d4c71d0f6 | 
							
								
								
									
										8
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							| @@ -28,12 +28,16 @@ If applicable, add screenshots to help explain your problem. | ||||
|  - Browser [e.g. chrome, safari] | ||||
|  - Version [e.g. 22] | ||||
|  | ||||
| **Host Environment (please complete the following information):** | ||||
| **Host Environment (please complete following information, DO NOT REMOVE ANY FIELD(S)):** | ||||
| -  Arch: [e.g. arm64] | ||||
|  - Device: [e.g. Bananapi R2 PRO] | ||||
|  - OS: [e.g. Armbian] | ||||
|  - Version [e.g.  23.02 Bullseye ] | ||||
|  - Docker Version (if you are running Zoraxy in docker): [e.g. 3.0.4] | ||||
|  - Are you using Docker? (yes / no) | ||||
|  - Docker Version (fill in "N/A" for native deployment): [e.g. 3.0.4] | ||||
|  | ||||
| **Supplementary links** | ||||
| If your issue is related to a particular open source project, paste the link here. | ||||
|  | ||||
| **Additional context** | ||||
| Add any other context about the problem here. | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							| @@ -25,9 +25,13 @@ jobs: | ||||
|           username: ${{ secrets.DOCKER_USERNAME }} | ||||
|           password: ${{ secrets.DOCKER_PASSWORD }} | ||||
|  | ||||
|       - name: Pull last image for layer reuse | ||||
|         run: | | ||||
|           docker pull docker.io/zoraxydocker/zoraxy:latest | ||||
|        | ||||
|       - name: Setup building file structure | ||||
|         run: | | ||||
|           cp -lr $GITHUB_WORKSPACE/src/ $GITHUB_WORKSPACE/docker/ | ||||
|           cp -lr $GITHUB_WORKSPACE/src/ $GITHUB_WORKSPACE/docker/src/ | ||||
|  | ||||
|       - name: Build and push Docker image | ||||
|         uses: docker/build-push-action@v6 | ||||
|   | ||||
							
								
								
									
										17
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										17
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -40,3 +40,20 @@ src/www/html/index.html | ||||
| src/sys.uuid | ||||
| src/zoraxy | ||||
| src/log/ | ||||
|  | ||||
|  | ||||
| # dev-tags | ||||
| /Dockerfile | ||||
| /Entrypoint.sh | ||||
|  | ||||
| # plugins | ||||
| example/plugins/ztnc/ztnc.db | ||||
| example/plugins/ztnc/authtoken.secret | ||||
| example/plugins/ztnc/ztnc.db.lock | ||||
| .idea | ||||
| conf | ||||
| log | ||||
| tmp | ||||
| sys.* | ||||
| www/html/index.html | ||||
| *.exe | ||||
							
								
								
									
										41
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										41
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,3 +1,44 @@ | ||||
| # v3.1.9 1 Mar 2025 | ||||
|  | ||||
| + Fixed netstat underflow bug | ||||
| + Fixed origin picker cookie bug [#550](https://github.com/tobychui/zoraxy/issues/550) | ||||
| + Added prototype plugin system | ||||
| + Added plugin examples | ||||
| + Added notice for build-in Zerotier network controller deprecation (and will be moved to plugins) | ||||
| + Added country code display for quickban list [#247](https://github.com/tobychui/zoraxy/issues/247) | ||||
| + Removed passive load balancer and default to active lb only [#554](https://github.com/tobychui/zoraxy/issues/554) | ||||
|  | ||||
|  | ||||
| # v3.1.8 16 Feb 2025 | ||||
|  | ||||
| + Exposed timeout value from dpcore to UI | ||||
| + Added active load balancing (if uptime monitor is enabled on that rule) | ||||
| + Re-factorized io stats and remove dependencies over wmic by [eyerrock](https://github.com/eyerrock) | ||||
| + Removed SMTP input validation [#497](https://github.com/tobychui/zoraxy/issues/497) | ||||
| + Fixed sticky session bug | ||||
| + Fixed passive load balancer bug | ||||
| + Fixed dockerfile bug by [PassiveLemon](https://github.com/PassiveLemon) | ||||
|  | ||||
| # v3.1.7 08 Feb 2025 | ||||
|  | ||||
| + Merged and added new tagging system for HTTP Proxy rules [by @adoolaard](https://github.com/adoolaard) | ||||
| + Added inline editing for redirection rules [#510](https://github.com/tobychui/zoraxy/issues/510) | ||||
| + Added uptime monitor status dot detail info (now clickable) [#467](https://github.com/tobychui/zoraxy/issues/467) | ||||
| + Added close connection support to port 80 listener [#405](https://github.com/tobychui/zoraxy/issues/450) | ||||
| + Optimized port collision check on startup | ||||
| + Optimized dark theme color scheme (Free consultation by 3S Design studio) | ||||
| + Fixed capital letter rule unable to delete bug [#507](https://github.com/tobychui/zoraxy/issues/507) | ||||
| + Fixed docker statistic not save bug [by @PassiveLemon](https://github.com/PassiveLemon) [#505](https://github.com/tobychui/zoraxy/issues/505) | ||||
|  | ||||
|  | ||||
| # v3.1.6 31 Dec 2024 | ||||
|  | ||||
|  | ||||
| + Exposed log file, sys.uuid and static web server path to start flag (customizable conf and sys.db path is still wip) | ||||
| + Optimized connection close implementation | ||||
| + Added toggle for uptime monitor | ||||
| + Added optional copy HTTP custom headers to websocket connection [#444](https://github.com/tobychui/zoraxy/issues/444) | ||||
|  | ||||
| # v3.1.5 28 Dec 2024 | ||||
|  | ||||
| + Fixed hostname case sensitive bug [#435](https://github.com/tobychui/zoraxy/issues/435) | ||||
|   | ||||
							
								
								
									
										30
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								README.md
									
									
									
									
									
								
							| @@ -101,32 +101,42 @@ Usage of zoraxy: | ||||
|         ACME auto TLS/SSL certificate renew check interval (seconds) (default 86400) | ||||
|   -cfgupgrade | ||||
|         Enable auto config upgrade if breaking change is detected (default true) | ||||
|   -db string | ||||
|         Database backend to use (leveldb, boltdb, auto) Note that fsdb will be used on unsupported platforms like RISCV (default "auto") | ||||
|   -default_inbound_enabled | ||||
|         If web server is enabled by default (default true) | ||||
|   -default_inbound_port int | ||||
|         Default web server listening port (default 443) | ||||
|   -docker | ||||
|         Run Zoraxy in docker compatibility mode | ||||
|   -earlyrenew int | ||||
|         Number of days to early renew a soon expiring certificate (days) (default 30) | ||||
|   -fastgeoip | ||||
|         Enable high speed geoip lookup, require 1GB extra memory (Not recommend for low end devices) | ||||
|   -log string | ||||
|         Log folder path (default "./log") | ||||
|   -mdns | ||||
|         Enable mDNS scanner and transponder (default true) | ||||
|   -mdnsname string | ||||
|         mDNS name, leave empty to use default (zoraxy_{node-uuid}.local) | ||||
|   -noauth | ||||
|         Disable authentication for management interface | ||||
|   -plugin string | ||||
|         Plugin folder path (default "./plugins") | ||||
|   -port string | ||||
|         Management web interface listening port (default ":8000") | ||||
|   -sshlb | ||||
|         Allow loopback web ssh connection (DANGER) | ||||
|   -update_geoip | ||||
|         Download the latest GeoIP data and exit | ||||
|   -uuid string | ||||
|         sys.uuid file path (default "./sys.uuid") | ||||
|   -version | ||||
|         Show version of this server | ||||
|   -webfm | ||||
|         Enable web file manager for static web server root folder (default true) | ||||
|   -webroot string | ||||
|         Static web server root folder. Only allow chnage in start paramters (default "./www") | ||||
|   -ztauth string | ||||
|         ZeroTier authtoken for the local node | ||||
|   -ztport int | ||||
|         ZeroTier controller API port (default 9993) | ||||
|         Static web server root folder. Only allow change in start paramters (default "./www") | ||||
| ``` | ||||
|  | ||||
| ### External Permission Management Mode | ||||
| @@ -185,6 +195,16 @@ Loopback web SSH connections, by default, are disabled. This means that if you a | ||||
| ./zoraxy -sshlb=true | ||||
| ``` | ||||
|  | ||||
| ## Community Maintained Sections | ||||
|  | ||||
| Some section of Zoraxy are contributed by our amazing community and if you have any issues regarding those sections, it would be more efficient if you can tag them directly when creating an issue report. | ||||
|  | ||||
| - Authelia Support added by [@7brend7](https://github.com/7brend7) | ||||
| - Authentik Support added by [@JokerQyou](https://github.com/JokerQyou) | ||||
| - Docker Container List by [@eyerrock](https://github.com/eyerrock) | ||||
|  | ||||
| Thank you so much for your contributions! | ||||
|  | ||||
| ## Sponsor This Project | ||||
|  | ||||
| If you like the project and want to support us, please consider a donation. You can use the links below | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| ## Build Zoraxy | ||||
| FROM docker.io/golang:alpine AS build-zoraxy | ||||
|  | ||||
| RUN mkdir -p /opt/zoraxy/source/ &&\ | ||||
| @@ -12,31 +13,56 @@ RUN go mod tidy &&\ | ||||
|     go build -o /usr/local/bin/zoraxy &&\ | ||||
|     chmod 755 /usr/local/bin/zoraxy | ||||
|  | ||||
| FROM docker.io/ubuntu:latest AS build-zerotier | ||||
|  | ||||
| ## Build ZeroTier | ||||
| FROM docker.io/rust:1.79-alpine AS build-zerotier | ||||
|  | ||||
| RUN mkdir -p /opt/zerotier/source/ &&\ | ||||
|     mkdir -p /usr/local/bin/ | ||||
|  | ||||
| WORKDIR /opt/zerotier/source/ | ||||
|  | ||||
| RUN apt-get update -y &&\ | ||||
|     apt-get install -y curl jq build-essential pkg-config clang cargo libssl-dev | ||||
| RUN apk add --update --no-cache curl make gcc g++ linux-headers openssl-dev nano | ||||
|  | ||||
| RUN curl -Lo ZeroTierOne.tar.gz https://codeload.github.com/zerotier/ZeroTierOne/tar.gz/refs/tags/1.10.6 &&\ | ||||
|     tar -xzvf ZeroTierOne.tar.gz &&\ | ||||
|     cd ZeroTierOne-* &&\ | ||||
|     make &&\ | ||||
|     cd ZeroTierOne-*/zeroidc &&\ | ||||
|     cargo update -p getrandom &&\ | ||||
|     cd .. &&\ | ||||
|     make -f make-linux.mk &&\ | ||||
|     mv ./zerotier-one /usr/local/bin/zerotier-one &&\ | ||||
|     chmod 755 /usr/local/bin/zerotier-one | ||||
|  | ||||
| FROM docker.io/ubuntu:latest | ||||
|  | ||||
| RUN apt-get update -y &&\ | ||||
|     apt-get install -y bash sudo netcat-openbsd libssl-dev ca-certificates | ||||
| ## Fetch plugin | ||||
| FROM docker.io/golang:alpine AS fetch-plugin | ||||
|  | ||||
| RUN mkdir -p /opt/zoraxy/zoraxy_plugin/ | ||||
|  | ||||
| RUN apk add --update --no-cache git | ||||
|  | ||||
| WORKDIR /opt/zoraxy/ | ||||
|  | ||||
| RUN git clone https://github.com/aroz-online/zoraxy-official-plugins &&\ | ||||
|     cp -r ./zoraxy-official-plugins/src/ztnc/mod/zoraxy_plugin/ /opt/zoraxy/zoraxy_plugin/ | ||||
|  | ||||
|  | ||||
| ## Main | ||||
| FROM docker.io/golang:alpine | ||||
|  | ||||
| # If you build it yourself, you will need to add the example directory into the docker directory. | ||||
|  | ||||
| COPY --chmod=700 ./entrypoint.sh /opt/zoraxy/ | ||||
| COPY --from=build-zoraxy /usr/local/bin/zoraxy /usr/local/bin/zoraxy | ||||
| COPY --chmod=700 ./build_plugins.sh /usr/local/bin/build_plugins | ||||
|  | ||||
| COPY --from=fetch-plugin --chmod=700 /opt/zoraxy/zoraxy_plugin/ /opt/zoraxy/zoraxy_plugin/ | ||||
|  | ||||
| COPY --from=build-zerotier /usr/local/bin/zerotier-one /usr/local/bin/zerotier-one | ||||
| COPY --from=build-zoraxy /usr/local/bin/zoraxy /usr/local/bin/zoraxy | ||||
|  | ||||
| RUN apk add --update --no-cache bash sudo netcat-openbsd libressl-dev openssh ca-certificates libc6-compat libstdc++ &&\ | ||||
|     mkdir -p /opt/zoraxy/plugin/ &&\ | ||||
|     echo "tun" | tee -a /etc/modules | ||||
|  | ||||
| WORKDIR /opt/zoraxy/config/ | ||||
|  | ||||
| @@ -51,17 +77,18 @@ ENV FASTGEOIP="false" | ||||
| ENV MDNS="true" | ||||
| ENV MDNSNAME="''" | ||||
| ENV NOAUTH="false" | ||||
| ENV PLUGIN="/opt/zoraxy/plugin/" | ||||
| ENV PORT="8000" | ||||
| ENV SSHLB="false" | ||||
| ENV UPDATE_GEOIP="false" | ||||
| ENV VERSION="false" | ||||
| ENV WEBFM="true" | ||||
| ENV WEBROOT="./www" | ||||
| ENV ZTAUTH="" | ||||
| ENV ZTPORT="9993" | ||||
|  | ||||
| VOLUME [ "/opt/zoraxy/config/" ] | ||||
|  | ||||
| LABEL com.imuslab.zoraxy.container-identifier="Zoraxy" | ||||
|  | ||||
| ENTRYPOINT [ "/opt/zoraxy/entrypoint.sh" ] | ||||
|  | ||||
| HEALTHCHECK --interval=15s --timeout=5s --start-period=10s --retries=3 CMD nc -vz 127.0.0.1 $PORT || exit 1 | ||||
|   | ||||
| @@ -19,10 +19,12 @@ Once setup, access the webui at `http://<host-ip>:8000` to configure Zoraxy. Cha | ||||
| docker run -d \ | ||||
|   --name zoraxy \ | ||||
|   --restart unless-stopped \ | ||||
|   --add-host=host.docker.internal:host-gateway \ | ||||
|   -p 80:80 \ | ||||
|   -p 443:443 \ | ||||
|   -p 8000:8000 \ | ||||
|   -v /path/to/zoraxy/config/:/opt/zoraxy/config/ \ | ||||
|   -v /path/to/zoraxy/plugin/:/opt/zoraxy/plugin/ \ | ||||
|   -v /var/run/docker.sock:/var/run/docker.sock \ | ||||
|   -v /etc/localtime:/etc/localtime \ | ||||
|   -e FASTGEOIP="true" \ | ||||
| @@ -43,8 +45,11 @@ services: | ||||
|       - 8000:8000 | ||||
|     volumes: | ||||
|       - /path/to/zoraxy/config/:/opt/zoraxy/config/ | ||||
|       - /path/to/zoraxy/plugin/:/opt/zoraxy/plugin/ | ||||
|       - /var/run/docker.sock:/var/run/docker.sock | ||||
|       - /etc/localtime:/etc/localtime | ||||
|     extra_hosts: | ||||
|       - "host.docker.internal:host-gateway" | ||||
|     environment: | ||||
|       FASTGEOIP: "true" | ||||
| ``` | ||||
| @@ -62,9 +67,15 @@ services: | ||||
| | Volume | Details | | ||||
| |:-|:-| | ||||
| | `/opt/zoraxy/config/` | Zoraxy configuration. | | ||||
| | `/opt/zoraxy/plugin/` | Zoraxy plugins. | | ||||
| | `/var/run/docker.sock` | Docker socket. Used for additional functionality with Zoraxy. | | ||||
| | `/etc/localtime` | Localtime. Set to ensure the host and container are synchronized. | | ||||
|  | ||||
| ### Extra Hosts | ||||
| | Host | Details | | ||||
| |:-|:-| | ||||
| | `host.docker.internal:host-gateway` | Resolves host.docker.internal to the host’s gateway IP on the Docker bridge network, allowing containers to access services running on the host machine. | | ||||
|  | ||||
| ### Environment | ||||
|  | ||||
| Variables are the same as those in [Start Parameters](https://github.com/tobychui/zoraxy?tab=readme-ov-file#start-paramters). | ||||
| @@ -80,6 +91,7 @@ Variables are the same as those in [Start Parameters](https://github.com/tobychu | ||||
| | `MDNS` | `true` (Boolean) | Enable mDNS scanner and transponder. | | ||||
| | `MDNSNAME` | `''` (String) | mDNS name, leave empty to use default (zoraxy_{node-uuid}.local). | | ||||
| | `NOAUTH` | `false` (Boolean) | Disable authentication for management interface. | | ||||
| | `PLUGIN` | `/opt/zoraxy/plugin/` (String) | Set the path for Zoraxy plugins. Only change this if you know what you are doing. | | ||||
| | `PORT` | `8000` (Integer) | Management web interface listening port | | ||||
| | `SSHLB` | `false` (Boolean) | Allow loopback web ssh connection (DANGER). | | ||||
| | `UPDATE_GEOIP` | `false` (Boolean) | Download the latest GeoIP data and exit. | | ||||
| @@ -87,17 +99,38 @@ Variables are the same as those in [Start Parameters](https://github.com/tobychu | ||||
| | `WEBFM` | `true` (Boolean) | Enable web file manager for static web server root folder. | | ||||
| | `WEBROOT` | `./www` (String) | Static web server root folder. Only allow change in start parameters. | | ||||
| | `ZEROTIER` | `false` (Boolean) | Enable ZeroTier functionality for GAN. | | ||||
| | `ZTAUTH` | `""` (String) | ZeroTier authtoken for the local node. | | ||||
| | `ZTPORT` | `9993` (Integer) | ZeroTier controller API port. | | ||||
|  | ||||
| > [!IMPORTANT] | ||||
| > Contrary to the Zoraxy README, Docker usage of the port flag should NOT include the colon. Ex: `-e PORT="8000"` for Docker run and `PORT: "8000"` for Docker compose. | ||||
|  | ||||
| ### ZeroTier | ||||
|  | ||||
| If you are running with ZeroTier, make sure to add the following flags to ensure ZeroTier functionality: | ||||
|    | ||||
| `--cap_add NET_ADMIN` and `--device /dev/net/tun:/dev/net/tun` | ||||
|  | ||||
| Or for Docker Compose: | ||||
| ``` | ||||
|   cap_add: | ||||
|     - NET_ADMIN | ||||
|   devices: | ||||
|     - /dev/net/tun:/dev/net/tun | ||||
| ``` | ||||
|  | ||||
| ### Plugins | ||||
|  | ||||
| You can find official plugins at https://github.com/aroz-online/zoraxy-official-plugins | ||||
|  | ||||
| Place your plugins inside the volume `/path/to/zoraxy/plugin/:/opt/zoraxy/plugin/` (Adjust to your actual install location). Any plugins you have added will then be built and used on the next restart. | ||||
|  | ||||
| > [!IMPORTANT] | ||||
| > Plugins are currently experimental. | ||||
|  | ||||
| ### Building | ||||
|  | ||||
| To build the Docker image: | ||||
|   - Check out the repository/branch. | ||||
|   - Copy the Zoraxy `src/` directory into the `docker/` (here) directory. | ||||
|   - Copy the Zoraxy `src/` and `example/` directory into the `docker/` (here) directory. | ||||
|   - Run the build command with `docker build -t zoraxy_build .` | ||||
|   - You can now use the image `zoraxy_build` | ||||
|     - If you wish to change the image name, then modify`zoraxy_build` in the previous step and then build again. | ||||
|   | ||||
							
								
								
									
										19
									
								
								docker/build_plugins.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								docker/build_plugins.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| echo "Copying zoraxy_plugin to all mods..." | ||||
| for dir in "$1"/*; do | ||||
|   if [ -d "$dir" ]; then | ||||
|     cp -r "/opt/zoraxy/zoraxy_plugin/" "$dir/mod/" | ||||
|   fi | ||||
| done | ||||
|  | ||||
| echo "Running go mod tidy and go build for all directories..." | ||||
| for dir in "$1"/*; do | ||||
|   if [ -d "$dir" ]; then | ||||
|     cd "$dir" || exit 1 | ||||
|     go mod tidy | ||||
|     go build | ||||
|     cd "$1" || exit 1 | ||||
|   fi | ||||
| done | ||||
|  | ||||
							
								
								
									
										18
									
								
								docker/docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								docker/docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| services: | ||||
|   zoraxy: | ||||
|     image: zoraxydocker/zoraxy:latest | ||||
|     container_name: zoraxy | ||||
|     restart: unless-stopped | ||||
|     ports: | ||||
|       - 80:80 | ||||
|       - 443:443 | ||||
|       - 8000:8000 | ||||
|     volumes: | ||||
|       - /path/to/zoraxy/config/:/opt/zoraxy/config/ | ||||
|       - /path/to/zoraxy/plugin/:/opt/zoraxy/plugin/ | ||||
|       - /var/run/docker.sock:/var/run/docker.sock | ||||
|       - /etc/localtime:/etc/localtime | ||||
|     extra_hosts: | ||||
|       - "host.docker.internal:host-gateway" | ||||
|     environment: | ||||
|       FASTGEOIP: "true" | ||||
| @@ -1,22 +1,36 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| update-ca-certificates | ||||
| echo "CA certificates updated." | ||||
| cleanup() { | ||||
|   echo "Stop signal received. Shutting down..." | ||||
|   kill -TERM "$(pidof zoraxy)" &> /dev/null && echo "Zoraxy stopped." | ||||
|   kill -TERM "$(pidof zerotier-one)" &> /dev/null && echo "ZeroTier-One stopped." | ||||
|   unlink /var/lib/zerotier-one/zerotier/ | ||||
|   exit 0 | ||||
| } | ||||
|  | ||||
| zoraxy -update_geoip=true | ||||
| echo "Updated GeoIP data." | ||||
| trap cleanup SIGTERM SIGINT TERM INT | ||||
|  | ||||
| update-ca-certificates && echo "CA certificates updated." | ||||
| zoraxy -update_geoip=true && echo "GeoIP data updated ." | ||||
|  | ||||
| echo "Building plugins..." | ||||
| cd /opt/zoraxy/plugin/ || exit 1 | ||||
| build_plugins "$PWD" | ||||
| echo "Plugins built." | ||||
| cd /opt/zoraxy/config/ || exit 1 | ||||
|  | ||||
| if [ "$ZEROTIER" = "true" ]; then | ||||
|   if [ ! -d "/opt/zoraxy/config/zerotier/" ]; then | ||||
|     mkdir -p /opt/zoraxy/config/zerotier/ | ||||
|   fi | ||||
|   ln -s /opt/zoraxy/config/zerotier/ /var/lib/zerotier-one | ||||
|   zerotier-one -d | ||||
|   zerotier-one -d & | ||||
|   zerotierpid=$! | ||||
|   echo "ZeroTier daemon started." | ||||
| fi | ||||
|  | ||||
| echo "Starting Zoraxy..." | ||||
| exec zoraxy \ | ||||
| zoraxy \ | ||||
|   -autorenew="$AUTORENEW" \ | ||||
|   -cfgupgrade="$CFGUPGRADE" \ | ||||
|   -db="$DB" \ | ||||
| @@ -26,12 +40,16 @@ exec zoraxy \ | ||||
|   -mdns="$MDNS" \ | ||||
|   -mdnsname="$MDNSNAME" \ | ||||
|   -noauth="$NOAUTH" \ | ||||
|   -plugin="$PLUGIN" \ | ||||
|   -port=:"$PORT" \ | ||||
|   -sshlb="$SSHLB" \ | ||||
|   -update_geoip="$UPDATE_GEOIP" \ | ||||
|   -version="$VERSION" \ | ||||
|   -webfm="$WEBFM" \ | ||||
|   -webroot="$WEBROOT" \ | ||||
|   -ztauth="$ZTAUTH" \ | ||||
|   -ztport="$ZTPORT" | ||||
|   & | ||||
|  | ||||
| zoraxypid=$! | ||||
| wait "$zoraxypid" | ||||
| wait "$zerotierpid" | ||||
|  | ||||
|   | ||||
							
								
								
									
										451
									
								
								docs/GNU Free Documentation License.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										451
									
								
								docs/GNU Free Documentation License.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,451 @@ | ||||
|  | ||||
|                 GNU Free Documentation License | ||||
|                  Version 1.3, 3 November 2008 | ||||
|  | ||||
|  | ||||
|  Copyright (C) 2000, 2001, 2002, 2007, 2008 Free Software Foundation, Inc. | ||||
|      <https://fsf.org/> | ||||
|  Everyone is permitted to copy and distribute verbatim copies | ||||
|  of this license document, but changing it is not allowed. | ||||
|  | ||||
| 0. PREAMBLE | ||||
|  | ||||
| The purpose of this License is to make a manual, textbook, or other | ||||
| functional and useful document "free" in the sense of freedom: to | ||||
| assure everyone the effective freedom to copy and redistribute it, | ||||
| with or without modifying it, either commercially or noncommercially. | ||||
| Secondarily, this License preserves for the author and publisher a way | ||||
| to get credit for their work, while not being considered responsible | ||||
| for modifications made by others. | ||||
|  | ||||
| This License is a kind of "copyleft", which means that derivative | ||||
| works of the document must themselves be free in the same sense.  It | ||||
| complements the GNU General Public License, which is a copyleft | ||||
| license designed for free software. | ||||
|  | ||||
| We have designed this License in order to use it for manuals for free | ||||
| software, because free software needs free documentation: a free | ||||
| program should come with manuals providing the same freedoms that the | ||||
| software does.  But this License is not limited to software manuals; | ||||
| it can be used for any textual work, regardless of subject matter or | ||||
| whether it is published as a printed book.  We recommend this License | ||||
| principally for works whose purpose is instruction or reference. | ||||
|  | ||||
|  | ||||
| 1. APPLICABILITY AND DEFINITIONS | ||||
|  | ||||
| This License applies to any manual or other work, in any medium, that | ||||
| contains a notice placed by the copyright holder saying it can be | ||||
| distributed under the terms of this License.  Such a notice grants a | ||||
| world-wide, royalty-free license, unlimited in duration, to use that | ||||
| work under the conditions stated herein.  The "Document", below, | ||||
| refers to any such manual or work.  Any member of the public is a | ||||
| licensee, and is addressed as "you".  You accept the license if you | ||||
| copy, modify or distribute the work in a way requiring permission | ||||
| under copyright law. | ||||
|  | ||||
| A "Modified Version" of the Document means any work containing the | ||||
| Document or a portion of it, either copied verbatim, or with | ||||
| modifications and/or translated into another language. | ||||
|  | ||||
| A "Secondary Section" is a named appendix or a front-matter section of | ||||
| the Document that deals exclusively with the relationship of the | ||||
| publishers or authors of the Document to the Document's overall | ||||
| subject (or to related matters) and contains nothing that could fall | ||||
| directly within that overall subject.  (Thus, if the Document is in | ||||
| part a textbook of mathematics, a Secondary Section may not explain | ||||
| any mathematics.)  The relationship could be a matter of historical | ||||
| connection with the subject or with related matters, or of legal, | ||||
| commercial, philosophical, ethical or political position regarding | ||||
| them. | ||||
|  | ||||
| The "Invariant Sections" are certain Secondary Sections whose titles | ||||
| are designated, as being those of Invariant Sections, in the notice | ||||
| that says that the Document is released under this License.  If a | ||||
| section does not fit the above definition of Secondary then it is not | ||||
| allowed to be designated as Invariant.  The Document may contain zero | ||||
| Invariant Sections.  If the Document does not identify any Invariant | ||||
| Sections then there are none. | ||||
|  | ||||
| The "Cover Texts" are certain short passages of text that are listed, | ||||
| as Front-Cover Texts or Back-Cover Texts, in the notice that says that | ||||
| the Document is released under this License.  A Front-Cover Text may | ||||
| be at most 5 words, and a Back-Cover Text may be at most 25 words. | ||||
|  | ||||
| A "Transparent" copy of the Document means a machine-readable copy, | ||||
| represented in a format whose specification is available to the | ||||
| general public, that is suitable for revising the document | ||||
| straightforwardly with generic text editors or (for images composed of | ||||
| pixels) generic paint programs or (for drawings) some widely available | ||||
| drawing editor, and that is suitable for input to text formatters or | ||||
| for automatic translation to a variety of formats suitable for input | ||||
| to text formatters.  A copy made in an otherwise Transparent file | ||||
| format whose markup, or absence of markup, has been arranged to thwart | ||||
| or discourage subsequent modification by readers is not Transparent. | ||||
| An image format is not Transparent if used for any substantial amount | ||||
| of text.  A copy that is not "Transparent" is called "Opaque". | ||||
|  | ||||
| Examples of suitable formats for Transparent copies include plain | ||||
| ASCII without markup, Texinfo input format, LaTeX input format, SGML | ||||
| or XML using a publicly available DTD, and standard-conforming simple | ||||
| HTML, PostScript or PDF designed for human modification.  Examples of | ||||
| transparent image formats include PNG, XCF and JPG.  Opaque formats | ||||
| include proprietary formats that can be read and edited only by | ||||
| proprietary word processors, SGML or XML for which the DTD and/or | ||||
| processing tools are not generally available, and the | ||||
| machine-generated HTML, PostScript or PDF produced by some word | ||||
| processors for output purposes only. | ||||
|  | ||||
| The "Title Page" means, for a printed book, the title page itself, | ||||
| plus such following pages as are needed to hold, legibly, the material | ||||
| this License requires to appear in the title page.  For works in | ||||
| formats which do not have any title page as such, "Title Page" means | ||||
| the text near the most prominent appearance of the work's title, | ||||
| preceding the beginning of the body of the text. | ||||
|  | ||||
| The "publisher" means any person or entity that distributes copies of | ||||
| the Document to the public. | ||||
|  | ||||
| A section "Entitled XYZ" means a named subunit of the Document whose | ||||
| title either is precisely XYZ or contains XYZ in parentheses following | ||||
| text that translates XYZ in another language.  (Here XYZ stands for a | ||||
| specific section name mentioned below, such as "Acknowledgements", | ||||
| "Dedications", "Endorsements", or "History".)  To "Preserve the Title" | ||||
| of such a section when you modify the Document means that it remains a | ||||
| section "Entitled XYZ" according to this definition. | ||||
|  | ||||
| The Document may include Warranty Disclaimers next to the notice which | ||||
| states that this License applies to the Document.  These Warranty | ||||
| Disclaimers are considered to be included by reference in this | ||||
| License, but only as regards disclaiming warranties: any other | ||||
| implication that these Warranty Disclaimers may have is void and has | ||||
| no effect on the meaning of this License. | ||||
|  | ||||
| 2. VERBATIM COPYING | ||||
|  | ||||
| You may copy and distribute the Document in any medium, either | ||||
| commercially or noncommercially, provided that this License, the | ||||
| copyright notices, and the license notice saying this License applies | ||||
| to the Document are reproduced in all copies, and that you add no | ||||
| other conditions whatsoever to those of this License.  You may not use | ||||
| technical measures to obstruct or control the reading or further | ||||
| copying of the copies you make or distribute.  However, you may accept | ||||
| compensation in exchange for copies.  If you distribute a large enough | ||||
| number of copies you must also follow the conditions in section 3. | ||||
|  | ||||
| You may also lend copies, under the same conditions stated above, and | ||||
| you may publicly display copies. | ||||
|  | ||||
|  | ||||
| 3. COPYING IN QUANTITY | ||||
|  | ||||
| If you publish printed copies (or copies in media that commonly have | ||||
| printed covers) of the Document, numbering more than 100, and the | ||||
| Document's license notice requires Cover Texts, you must enclose the | ||||
| copies in covers that carry, clearly and legibly, all these Cover | ||||
| Texts: Front-Cover Texts on the front cover, and Back-Cover Texts on | ||||
| the back cover.  Both covers must also clearly and legibly identify | ||||
| you as the publisher of these copies.  The front cover must present | ||||
| the full title with all words of the title equally prominent and | ||||
| visible.  You may add other material on the covers in addition. | ||||
| Copying with changes limited to the covers, as long as they preserve | ||||
| the title of the Document and satisfy these conditions, can be treated | ||||
| as verbatim copying in other respects. | ||||
|  | ||||
| If the required texts for either cover are too voluminous to fit | ||||
| legibly, you should put the first ones listed (as many as fit | ||||
| reasonably) on the actual cover, and continue the rest onto adjacent | ||||
| pages. | ||||
|  | ||||
| If you publish or distribute Opaque copies of the Document numbering | ||||
| more than 100, you must either include a machine-readable Transparent | ||||
| copy along with each Opaque copy, or state in or with each Opaque copy | ||||
| a computer-network location from which the general network-using | ||||
| public has access to download using public-standard network protocols | ||||
| a complete Transparent copy of the Document, free of added material. | ||||
| If you use the latter option, you must take reasonably prudent steps, | ||||
| when you begin distribution of Opaque copies in quantity, to ensure | ||||
| that this Transparent copy will remain thus accessible at the stated | ||||
| location until at least one year after the last time you distribute an | ||||
| Opaque copy (directly or through your agents or retailers) of that | ||||
| edition to the public. | ||||
|  | ||||
| It is requested, but not required, that you contact the authors of the | ||||
| Document well before redistributing any large number of copies, to | ||||
| give them a chance to provide you with an updated version of the | ||||
| Document. | ||||
|  | ||||
|  | ||||
| 4. MODIFICATIONS | ||||
|  | ||||
| You may copy and distribute a Modified Version of the Document under | ||||
| the conditions of sections 2 and 3 above, provided that you release | ||||
| the Modified Version under precisely this License, with the Modified | ||||
| Version filling the role of the Document, thus licensing distribution | ||||
| and modification of the Modified Version to whoever possesses a copy | ||||
| of it.  In addition, you must do these things in the Modified Version: | ||||
|  | ||||
| A. Use in the Title Page (and on the covers, if any) a title distinct | ||||
|    from that of the Document, and from those of previous versions | ||||
|    (which should, if there were any, be listed in the History section | ||||
|    of the Document).  You may use the same title as a previous version | ||||
|    if the original publisher of that version gives permission. | ||||
| B. List on the Title Page, as authors, one or more persons or entities | ||||
|    responsible for authorship of the modifications in the Modified | ||||
|    Version, together with at least five of the principal authors of the | ||||
|    Document (all of its principal authors, if it has fewer than five), | ||||
|    unless they release you from this requirement. | ||||
| C. State on the Title page the name of the publisher of the | ||||
|    Modified Version, as the publisher. | ||||
| D. Preserve all the copyright notices of the Document. | ||||
| E. Add an appropriate copyright notice for your modifications | ||||
|    adjacent to the other copyright notices. | ||||
| F. Include, immediately after the copyright notices, a license notice | ||||
|    giving the public permission to use the Modified Version under the | ||||
|    terms of this License, in the form shown in the Addendum below. | ||||
| G. Preserve in that license notice the full lists of Invariant Sections | ||||
|    and required Cover Texts given in the Document's license notice. | ||||
| H. Include an unaltered copy of this License. | ||||
| I. Preserve the section Entitled "History", Preserve its Title, and add | ||||
|    to it an item stating at least the title, year, new authors, and | ||||
|    publisher of the Modified Version as given on the Title Page.  If | ||||
|    there is no section Entitled "History" in the Document, create one | ||||
|    stating the title, year, authors, and publisher of the Document as | ||||
|    given on its Title Page, then add an item describing the Modified | ||||
|    Version as stated in the previous sentence. | ||||
| J. Preserve the network location, if any, given in the Document for | ||||
|    public access to a Transparent copy of the Document, and likewise | ||||
|    the network locations given in the Document for previous versions | ||||
|    it was based on.  These may be placed in the "History" section. | ||||
|    You may omit a network location for a work that was published at | ||||
|    least four years before the Document itself, or if the original | ||||
|    publisher of the version it refers to gives permission. | ||||
| K. For any section Entitled "Acknowledgements" or "Dedications", | ||||
|    Preserve the Title of the section, and preserve in the section all | ||||
|    the substance and tone of each of the contributor acknowledgements | ||||
|    and/or dedications given therein. | ||||
| L. Preserve all the Invariant Sections of the Document, | ||||
|    unaltered in their text and in their titles.  Section numbers | ||||
|    or the equivalent are not considered part of the section titles. | ||||
| M. Delete any section Entitled "Endorsements".  Such a section | ||||
|    may not be included in the Modified Version. | ||||
| N. Do not retitle any existing section to be Entitled "Endorsements" | ||||
|    or to conflict in title with any Invariant Section. | ||||
| O. Preserve any Warranty Disclaimers. | ||||
|  | ||||
| If the Modified Version includes new front-matter sections or | ||||
| appendices that qualify as Secondary Sections and contain no material | ||||
| copied from the Document, you may at your option designate some or all | ||||
| of these sections as invariant.  To do this, add their titles to the | ||||
| list of Invariant Sections in the Modified Version's license notice. | ||||
| These titles must be distinct from any other section titles. | ||||
|  | ||||
| You may add a section Entitled "Endorsements", provided it contains | ||||
| nothing but endorsements of your Modified Version by various | ||||
| parties--for example, statements of peer review or that the text has | ||||
| been approved by an organization as the authoritative definition of a | ||||
| standard. | ||||
|  | ||||
| You may add a passage of up to five words as a Front-Cover Text, and a | ||||
| passage of up to 25 words as a Back-Cover Text, to the end of the list | ||||
| of Cover Texts in the Modified Version.  Only one passage of | ||||
| Front-Cover Text and one of Back-Cover Text may be added by (or | ||||
| through arrangements made by) any one entity.  If the Document already | ||||
| includes a cover text for the same cover, previously added by you or | ||||
| by arrangement made by the same entity you are acting on behalf of, | ||||
| you may not add another; but you may replace the old one, on explicit | ||||
| permission from the previous publisher that added the old one. | ||||
|  | ||||
| The author(s) and publisher(s) of the Document do not by this License | ||||
| give permission to use their names for publicity for or to assert or | ||||
| imply endorsement of any Modified Version. | ||||
|  | ||||
|  | ||||
| 5. COMBINING DOCUMENTS | ||||
|  | ||||
| You may combine the Document with other documents released under this | ||||
| License, under the terms defined in section 4 above for modified | ||||
| versions, provided that you include in the combination all of the | ||||
| Invariant Sections of all of the original documents, unmodified, and | ||||
| list them all as Invariant Sections of your combined work in its | ||||
| license notice, and that you preserve all their Warranty Disclaimers. | ||||
|  | ||||
| The combined work need only contain one copy of this License, and | ||||
| multiple identical Invariant Sections may be replaced with a single | ||||
| copy.  If there are multiple Invariant Sections with the same name but | ||||
| different contents, make the title of each such section unique by | ||||
| adding at the end of it, in parentheses, the name of the original | ||||
| author or publisher of that section if known, or else a unique number. | ||||
| Make the same adjustment to the section titles in the list of | ||||
| Invariant Sections in the license notice of the combined work. | ||||
|  | ||||
| In the combination, you must combine any sections Entitled "History" | ||||
| in the various original documents, forming one section Entitled | ||||
| "History"; likewise combine any sections Entitled "Acknowledgements", | ||||
| and any sections Entitled "Dedications".  You must delete all sections | ||||
| Entitled "Endorsements". | ||||
|  | ||||
|  | ||||
| 6. COLLECTIONS OF DOCUMENTS | ||||
|  | ||||
| You may make a collection consisting of the Document and other | ||||
| documents released under this License, and replace the individual | ||||
| copies of this License in the various documents with a single copy | ||||
| that is included in the collection, provided that you follow the rules | ||||
| of this License for verbatim copying of each of the documents in all | ||||
| other respects. | ||||
|  | ||||
| You may extract a single document from such a collection, and | ||||
| distribute it individually under this License, provided you insert a | ||||
| copy of this License into the extracted document, and follow this | ||||
| License in all other respects regarding verbatim copying of that | ||||
| document. | ||||
|  | ||||
|  | ||||
| 7. AGGREGATION WITH INDEPENDENT WORKS | ||||
|  | ||||
| A compilation of the Document or its derivatives with other separate | ||||
| and independent documents or works, in or on a volume of a storage or | ||||
| distribution medium, is called an "aggregate" if the copyright | ||||
| resulting from the compilation is not used to limit the legal rights | ||||
| of the compilation's users beyond what the individual works permit. | ||||
| When the Document is included in an aggregate, this License does not | ||||
| apply to the other works in the aggregate which are not themselves | ||||
| derivative works of the Document. | ||||
|  | ||||
| If the Cover Text requirement of section 3 is applicable to these | ||||
| copies of the Document, then if the Document is less than one half of | ||||
| the entire aggregate, the Document's Cover Texts may be placed on | ||||
| covers that bracket the Document within the aggregate, or the | ||||
| electronic equivalent of covers if the Document is in electronic form. | ||||
| Otherwise they must appear on printed covers that bracket the whole | ||||
| aggregate. | ||||
|  | ||||
|  | ||||
| 8. TRANSLATION | ||||
|  | ||||
| Translation is considered a kind of modification, so you may | ||||
| distribute translations of the Document under the terms of section 4. | ||||
| Replacing Invariant Sections with translations requires special | ||||
| permission from their copyright holders, but you may include | ||||
| translations of some or all Invariant Sections in addition to the | ||||
| original versions of these Invariant Sections.  You may include a | ||||
| translation of this License, and all the license notices in the | ||||
| Document, and any Warranty Disclaimers, provided that you also include | ||||
| the original English version of this License and the original versions | ||||
| of those notices and disclaimers.  In case of a disagreement between | ||||
| the translation and the original version of this License or a notice | ||||
| or disclaimer, the original version will prevail. | ||||
|  | ||||
| If a section in the Document is Entitled "Acknowledgements", | ||||
| "Dedications", or "History", the requirement (section 4) to Preserve | ||||
| its Title (section 1) will typically require changing the actual | ||||
| title. | ||||
|  | ||||
|  | ||||
| 9. TERMINATION | ||||
|  | ||||
| You may not copy, modify, sublicense, or distribute the Document | ||||
| except as expressly provided under this License.  Any attempt | ||||
| otherwise to copy, modify, sublicense, or distribute it is void, and | ||||
| will automatically terminate your rights under this License. | ||||
|  | ||||
| However, if you cease all violation of this License, then your license | ||||
| from a particular copyright holder is reinstated (a) provisionally, | ||||
| unless and until the copyright holder explicitly and finally | ||||
| terminates your license, and (b) permanently, if the copyright holder | ||||
| fails to notify you of the violation by some reasonable means prior to | ||||
| 60 days after the cessation. | ||||
|  | ||||
| Moreover, your license from a particular copyright holder is | ||||
| reinstated permanently if the copyright holder notifies you of the | ||||
| violation by some reasonable means, this is the first time you have | ||||
| received notice of violation of this License (for any work) from that | ||||
| copyright holder, and you cure the violation prior to 30 days after | ||||
| your receipt of the notice. | ||||
|  | ||||
| Termination of your rights under this section does not terminate the | ||||
| licenses of parties who have received copies or rights from you under | ||||
| this License.  If your rights have been terminated and not permanently | ||||
| reinstated, receipt of a copy of some or all of the same material does | ||||
| not give you any rights to use it. | ||||
|  | ||||
|  | ||||
| 10. FUTURE REVISIONS OF THIS LICENSE | ||||
|  | ||||
| The Free Software Foundation may publish new, revised versions of the | ||||
| GNU Free Documentation License from time to time.  Such new versions | ||||
| will be similar in spirit to the present version, but may differ in | ||||
| detail to address new problems or concerns.  See | ||||
| https://www.gnu.org/licenses/. | ||||
|  | ||||
| Each version of the License is given a distinguishing version number. | ||||
| If the Document specifies that a particular numbered version of this | ||||
| License "or any later version" applies to it, you have the option of | ||||
| following the terms and conditions either of that specified version or | ||||
| of any later version that has been published (not as a draft) by the | ||||
| Free Software Foundation.  If the Document does not specify a version | ||||
| number of this License, you may choose any version ever published (not | ||||
| as a draft) by the Free Software Foundation.  If the Document | ||||
| specifies that a proxy can decide which future versions of this | ||||
| License can be used, that proxy's public statement of acceptance of a | ||||
| version permanently authorizes you to choose that version for the | ||||
| Document. | ||||
|  | ||||
| 11. RELICENSING | ||||
|  | ||||
| "Massive Multiauthor Collaboration Site" (or "MMC Site") means any | ||||
| World Wide Web server that publishes copyrightable works and also | ||||
| provides prominent facilities for anybody to edit those works.  A | ||||
| public wiki that anybody can edit is an example of such a server.  A | ||||
| "Massive Multiauthor Collaboration" (or "MMC") contained in the site | ||||
| means any set of copyrightable works thus published on the MMC site. | ||||
|  | ||||
| "CC-BY-SA" means the Creative Commons Attribution-Share Alike 3.0  | ||||
| license published by Creative Commons Corporation, a not-for-profit  | ||||
| corporation with a principal place of business in San Francisco,  | ||||
| California, as well as future copyleft versions of that license  | ||||
| published by that same organization. | ||||
|  | ||||
| "Incorporate" means to publish or republish a Document, in whole or in  | ||||
| part, as part of another Document. | ||||
|  | ||||
| An MMC is "eligible for relicensing" if it is licensed under this  | ||||
| License, and if all works that were first published under this License  | ||||
| somewhere other than this MMC, and subsequently incorporated in whole or  | ||||
| in part into the MMC, (1) had no cover texts or invariant sections, and  | ||||
| (2) were thus incorporated prior to November 1, 2008. | ||||
|  | ||||
| The operator of an MMC Site may republish an MMC contained in the site | ||||
| under CC-BY-SA on the same site at any time before August 1, 2009, | ||||
| provided the MMC is eligible for relicensing. | ||||
|  | ||||
|  | ||||
| ADDENDUM: How to use this License for your documents | ||||
|  | ||||
| To use this License in a document you have written, include a copy of | ||||
| the License in the document and put the following copyright and | ||||
| license notices just after the title page: | ||||
|  | ||||
|     Copyright (c)  YEAR  YOUR NAME. | ||||
|     Permission is granted to copy, distribute and/or modify this document | ||||
|     under the terms of the GNU Free Documentation License, Version 1.3 | ||||
|     or any later version published by the Free Software Foundation; | ||||
|     with no Invariant Sections, no Front-Cover Texts, and no Back-Cover Texts. | ||||
|     A copy of the license is included in the section entitled "GNU | ||||
|     Free Documentation License". | ||||
|  | ||||
| If you have Invariant Sections, Front-Cover Texts and Back-Cover Texts, | ||||
| replace the "with...Texts." line with this: | ||||
|  | ||||
|     with the Invariant Sections being LIST THEIR TITLES, with the | ||||
|     Front-Cover Texts being LIST, and with the Back-Cover Texts being LIST. | ||||
|  | ||||
| If you have Invariant Sections without Cover Texts, or some other | ||||
| combination of the three, merge those two alternatives to suit the | ||||
| situation. | ||||
|  | ||||
| If your document contains nontrivial examples of program code, we | ||||
| recommend releasing these examples in parallel under your choice of | ||||
| free software license, such as the GNU General Public License, | ||||
| to permit their use in free software. | ||||
							
								
								
									
										1
									
								
								docs/dom-i18n.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								docs/dom-i18n.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| !function(a,b){"use strict";"function"==typeof define&&define.amd?define([],function(){return a.domI18n=b()}):"object"==typeof exports?module.exports=b():a.domI18n=b()}(this,function(){"use strict";return function(a){function b(a){return a||(a=window.navigator.languages?window.navigator.languages[0]:window.navigator.language||window.navigator.userLanguage),-1===q.indexOf(a)&&(r&&console.warn(a+" is not available on the list of languages provided"),a=a.indexOf("-")?a.split("-")[0]:a),-1===q.indexOf(a)&&(r&&console.error(a+" is not compatible with any language provided"),a=p),a}function c(a){v=b(a),l()}function d(){u={}}function e(a){var b=a.getAttribute("data-dom-i18n-id");return b&&u&&u[b]}function f(a,b){var c="i18n"+Date.now()+1e3*Math.random();a.setAttribute("data-dom-i18n-id",c),u[c]=b}function g(a){return u&&u[a.getAttribute("data-dom-i18n-id")]}function h(a,b){var c={},d=a.firstElementChild,e=!d&&a[b].split(o);return q.forEach(function(b,f){var g;d?(g=a.children[f],g&&g.cloneNode&&(c[b]=g.cloneNode(!0))):(g=e[f],g&&(c[b]=String(g)))}),c}function i(a){var b,c,d=a.getAttribute(t),i=null!==a.getAttribute(s),k=d?d:"textContent";!i&&e(a)?b=g(a):(b=h(a,k),i||f(a,b)),c=b[v],"string"==typeof c?a[k]=c:"object"==typeof c&&j(a,c)}function j(a,b){k(a),a.appendChild(b)}function k(a){for(;a.lastChild;)a.removeChild(a.lastChild)}function l(){for(var a="string"==typeof n||n instanceof String?m.querySelectorAll(n):n,b=0;b<a.length;++b)i(a[b])}a=a||{};var m=a.rootElement||window.document,n=a.selector||"[data-translatable]",o=a.separator||" // ",p=a.defaultLanguage||"en",q=a.languages||["en"],r=void 0!==a.enableLog?a.enableLog:!0,s="data-no-cache",t="data-translatable-attr",u={},v=b(a.currentLanguage);return l(n),{changeLanguage:c,clearCachedElements:d}}}); | ||||
							
								
								
									
										
											BIN
										
									
								
								docs/img/logo_white.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/img/logo_white.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 6.5 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/img/preview-mobile.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/img/preview-mobile.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 140 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/img/preview-mobile.psd
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/img/preview-mobile.psd
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								docs/img/preview-pc.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/img/preview-pc.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 194 KiB | 
							
								
								
									
										867
									
								
								docs/index.html
									
									
									
									
									
								
							
							
						
						
									
										867
									
								
								docs/index.html
									
									
									
									
									
								
							| @@ -1,128 +1,121 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
| <!doctype html> | ||||
| <html> | ||||
|   <head> | ||||
|     <meta charset="utf-8"> | ||||
|     <meta content="width=device-width, initial-scale=1.0" name="viewport"> | ||||
|     <meta content="Reverse Proxy, Cluster, Gateway, Go, Homelab, Network Tools" name="keywords"> | ||||
|     <meta content="A reverse proxy server and cluster network gateway for noobs" name="description"> | ||||
|     <meta content="Reverse Proxy, Open Source, Aroz, Go, OS, NAS, Cloud" name="keywords"> | ||||
|     <meta content="Simplify your self-hosted services with Zoraxy, the ultimate homelab networking toolbox" name="description"> | ||||
|     <meta name="author" content="tobychui"> | ||||
|  | ||||
|     <!-- HTML Meta Tags --> | ||||
|     <title>Reverse Proxy Server | Zoraxy</title> | ||||
|     <meta name="description" content="A reverse proxy server and cluster network gateway for noobs"> | ||||
|     <title>Homelab Gateway | Zoraxy</title> | ||||
|     <meta name="description" content="Simplify your self-hosted services with Zoraxy, the ultimate homelab networking toolbox"> | ||||
|  | ||||
|     <!-- Facebook Meta Tags --> | ||||
|     <meta property="og:url" content="https://zoraxy.aroz.org/"> | ||||
|     <meta property="og:type" content="website"> | ||||
|     <meta property="og:title" content="Cluster Proxy Gateway | Zoraxy"> | ||||
|     <meta property="og:description" content="A reverse proxy server and cluster network gateway for noobs"> | ||||
|     <meta property="og:title" content="Hello Zoraxy"> | ||||
|     <meta property="og:description" content="Simplify your self-hosted services with Zoraxy, the ultimate homelab networking toolbox"> | ||||
|     <meta property="og:image" content="https://zoraxy.aroz.org/img/og.png"> | ||||
|  | ||||
|     <!-- Twitter Meta Tags --> | ||||
|     <meta name="twitter:card" content="summary_large_image"> | ||||
|     <meta property="twitter:domain" content="aroz.org"> | ||||
|     <meta property="twitter:domain" content="os.aroz.org"> | ||||
|     <meta property="twitter:url" content="https://zoraxy.aroz.org/"> | ||||
|     <meta name="twitter:title" content="Cluster Proxy Gateway | Zoraxy"> | ||||
|     <meta name="twitter:description" content="A reverse proxy server and cluster network gateway for noobs"> | ||||
|     <meta name="twitter:title" content="Hello Zoraxy"> | ||||
|     <meta name="twitter:description" content="Simplify your self-hosted services with Zoraxy, the ultimate homelab networking toolbox"> | ||||
|     <meta name="twitter:image" content="https://zoraxy.aroz.org/img/og.png"> | ||||
|  | ||||
|     <!-- JavaScript Libs--> | ||||
|     <script src="https://code.jquery.com/jquery-3.7.1.min.js" integrity="sha256-/JqT3SQfawRcv/BIHPThkBvs0OEvtFFmqPF/lYI/Cxo=" crossorigin="anonymous"></script> | ||||
|     <script src="dom-i18n.min.js"></script> | ||||
|     <link href="main.css" rel="stylesheet"> | ||||
|  | ||||
|     <!-- Css stuffs--> | ||||
|     <script src="https://cdnjs.cloudflare.com/ajax/libs/fomantic-ui/2.9.3/semantic.min.js" integrity="sha512-gnoBksrDbaMnlE0rhhkcx3iwzvgBGz6mOEj4/Y5ZY09n55dYddx6+WYc72A55qEesV8VX2iMomteIwobeGK1BQ==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> | ||||
|     <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/fomantic-ui/2.9.3/semantic.min.css" integrity="sha512-3quBdRGJyLy79hzhDDcBzANW+mVqPctrGCfIPosHQtMKb3rKsCxfyslzwlz2wj1dT8A7UX+sEvDjaUv+WExQrA==" crossorigin="anonymous" referrerpolicy="no-referrer" /> | ||||
|  | ||||
|     <!-- Favicons --> | ||||
|     <link href="favicon.png" rel="icon"> | ||||
|     <link href="img/apple-touch-icon.png" rel="apple-touch-icon"> | ||||
|  | ||||
|     <!-- Google Fonts --> | ||||
|     <link rel="preconnect" href="https://fonts.googleapis.com"> | ||||
|     <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin> | ||||
|     <link href="https://fonts.googleapis.com/css2?family=Source+Sans+Pro:wght@100;300;400;600;700;900&display=swap" rel="stylesheet"> | ||||
|     <link rel="stylesheet" href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL,GRAD@48,400,0,0" /> | ||||
|     <link href="https://fonts.googleapis.com/css2?family=Source+Sans+Pro:wght@300;400;600;700;900&display=swap" rel="stylesheet"> | ||||
|     <link href="https://fonts.googleapis.com/css2?family=Noto+Sans+TC:wght@100;300;400&display=swap" rel="stylesheet"> | ||||
|     <link href="https://fonts.googleapis.com/css2?family=M+PLUS+1p&family=Noto+Sans+JP:wght@100..900&display=swap" rel="stylesheet"> | ||||
|     <link href="https://fonts.googleapis.com/css2?family=Noto+Sans+SC:wght@100;300;400&display=swap" rel="stylesheet"> | ||||
|  | ||||
|     <!-- Main Stylesheet File --> | ||||
|     <link href="style.css" rel="stylesheet"> | ||||
|     <script | ||||
|       src="https://code.jquery.com/jquery-3.7.0.min.js" | ||||
|       integrity="sha256-2Pmvv0kuTBOenSvLm6bvfBSSHrUJ+3A7x6P5Ebd07/g=" | ||||
|       crossorigin="anonymous"></script> | ||||
|     <script src="https://cdnjs.cloudflare.com/ajax/libs/fomantic-ui/2.9.2/semantic.min.js" integrity="sha512-5cguXwRllb+6bcc2pogwIeQmQPXEzn2ddsqAexIBhh7FO1z5Hkek1J9mrK2+rmZCTU6b6pERxI7acnp1MpAg4Q==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> | ||||
|     <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/fomantic-ui/2.9.2/semantic.min.css" integrity="sha512-n//BDM4vMPvyca4bJjZPDh7hlqsQ7hqbP9RH18GF2hTXBY5amBwM2501M0GPiwCU/v9Tor2m13GOTFjk00tkQA==" crossorigin="anonymous" referrerpolicy="no-referrer" /> | ||||
|     <style> | ||||
|       p,a,div,span,h1,h2,h3,h4,h5,h6{ | ||||
|         font-family: 'Source Sans Pro', sans-serif !important; | ||||
|         color: #404040; | ||||
|       } | ||||
|     </style> | ||||
|     <!-- AOS.js--> | ||||
|     <script src="https://cdnjs.cloudflare.com/ajax/libs/aos/2.3.4/aos.js" integrity="sha512-A7AYk1fGKX6S2SsHywmPkrnzTZHrgiVT7GcQkLGDe2ev0aWb8zejytzS8wjo7PGEXKqJOrjQ4oORtnimIRZBtw==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> | ||||
|     <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/aos/2.3.4/aos.css" integrity="sha512-1cK78a1o+ht2JcaW6g8OXYwqpev9+6GqOkz9xmBN9iUUhIndKtxwILGWYOSibOKjLsEdjyjZvYDq/cZwNeak0w==" crossorigin="anonymous" referrerpolicy="no-referrer" /> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div class="main section"> | ||||
|       <div class="left-menu"> | ||||
|         <div class="iconWrapper"> | ||||
|           <a href="index.html"><img class="ui fluid image" src="img/icon.png"></a> | ||||
|         </div> | ||||
|         <a href="#home" class="menu-item active" align="center"> | ||||
|           <img src="img/icons/home.svg"> | ||||
|         </a> | ||||
|         <a href="#features" class="menu-item" align="center"> | ||||
|           <img src="img/icons/awesome.svg"> | ||||
|         </a> | ||||
|         <a href="#screenshots" class="menu-item" align="center"> | ||||
|           <img src="img/icons/screenshots.svg"> | ||||
|         </a> | ||||
|         <a href="#plugins" class="menu-item" align="center"> | ||||
|           <img src="img/icons/plugin.svg"> | ||||
|         </a> | ||||
|         <a href="#source" class="menu-item" align="center"> | ||||
|           <img src="img/icons/code.svg"> | ||||
|         </a> | ||||
|       </div> | ||||
|       <div class="right-content"> | ||||
|         <!-- Hero Banner Section --> | ||||
|         <div class="headbanner"></div> | ||||
|         <div id="home" class="herotext"> | ||||
|           <div class="ui basic segment"> | ||||
|             <div class="bannerHeaderWrapper"> | ||||
|               <h1 class="bannerHeader">Zoraxy</h1> | ||||
| 			  <div class="ui divider"></div><br> | ||||
|               <p class="bannerSubheader">Beyond Reverse Proxy: Your Ultimate Homelab Network Tool</p> | ||||
|     <div id="backToTopBtn" class="ui big icon button" onclick="backToTop();"> | ||||
|        <i class="ui arrow up icon"></i> | ||||
|     </div> | ||||
|     <button id="rwdmenubtn" class="ui black big icon button"><i class="ui bars icon"></i></button> | ||||
|     <div id="mainmenu" class="ui segment"> | ||||
|         <div class="ui container"> | ||||
|             <div class="ui small stackable secondary menu"> | ||||
|                 <div class="item"> | ||||
|                     <img class="ui tiny image" src="img/logo.png"> | ||||
|                 </div> | ||||
|                 <a class="item" href="#mainmenu" i18n> | ||||
|                     Home // 主頁 // Startseite | ||||
|                 </a> | ||||
|                 <a class="item" href="#about" i18n> | ||||
|                     About Zoraxy // 關於 Zoraxy // Über Zoraxy | ||||
|                 </a> | ||||
|                 <a class="item" href="#features" i18n> | ||||
|                     Screenshots // 系統截圖 // Bildschirmfotos | ||||
|                 </a> | ||||
|                 <a class="item" href="#techspec" i18n> | ||||
|                     Videos // 介紹影片 // Videos | ||||
|                 </a> | ||||
|                 <a class="item" href="#download" i18n> | ||||
|                     Download // 下載 // Herunterladen | ||||
|                 </a> | ||||
|                 <a class="item" href="#learnmore" i18n> | ||||
|                     Learn More // 了解更多 // Mehr erfahren | ||||
|                 </a> | ||||
|                 <a class="right floated item"> | ||||
|                     <div class="ui small selection dropdown"> | ||||
|                         <input type="hidden" id="language"> | ||||
|                         <div class="default text" style="color: #6cacff;"><i class="language icon"></i> Default</div> | ||||
|                         <i class="dropdown icon"></i> | ||||
|                         <div class="menu"> | ||||
|                             <div class="item" data-value="en">English</div> | ||||
|                             <div class="item" data-value="zh">中文(正體)</div> | ||||
|                             <div class="item" data-value="de">Deutsch</div> | ||||
|                         </div> | ||||
|                     </div> | ||||
|                 </a> | ||||
|             </div> | ||||
|             <br><br> | ||||
|             <a class="ui basic big button" style="background-color: white;" href="#features"><i class="ui blue arrow down icon"></i> Learn More</a> | ||||
|             <br><br> | ||||
|             <table class="ui very basic collapsing unstackable celled table"> | ||||
|               <thead> | ||||
|                 <tr><th colspan="2">Quick Access</th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|               <tr> | ||||
|                 <td> | ||||
|                   <h4 class="ui image header"> | ||||
|                     <i class="ui download icon"></i> | ||||
|                     <div class="content"> | ||||
|                       Download | ||||
|                       <div class="sub header">Prebuild Binary | ||||
|                     </div> | ||||
|                   </div> | ||||
|                 </h4></td> | ||||
|                 <td> | ||||
|                   <a href="https://github.com/tobychui/zoraxy/releases" target="_blank">Open <i class="ui external icon"></i></a> | ||||
|                 </td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td> | ||||
|                   <h4 class="ui image header"> | ||||
|                     <i class="ui github icon"></i> | ||||
|                     <div class="content"> | ||||
|                       Github | ||||
|                       <div class="sub header">Source Code | ||||
|                     </div> | ||||
|                   </div> | ||||
|                 </h4></td> | ||||
|                 <td> | ||||
|                   <a href="https://github.com/tobychui/zoraxy" target="_blank">Open <i class="ui external icon"></i></a> | ||||
|                 </td> | ||||
|               </tr> | ||||
|             </table> | ||||
|           </div> | ||||
|         </div> | ||||
|     </div> | ||||
|     <div class="messageBanner"> | ||||
|         <div class="ui text container"> | ||||
|             <p i18n>This site is currently under development. Some information might not be ready. | ||||
|                 // 本網站目前仍在開發中,部分資訊可能尚未準備好。 | ||||
|                 // Diese Seite ist in Entwicklung. Einige Informationen sind möglicherweise nicht verfügbar. | ||||
|             </p> | ||||
|         </div> | ||||
|     </div> | ||||
|     <div id="slideshowBanner"> | ||||
|         <div class="title"> | ||||
|             <h1 i18n>Zoraxy</h1> | ||||
|             <div class="ui divider" style="border-top: 1px solid rgba(255,255,255,0.5); "></div> | ||||
|             <p i18n>The ultimate homelab networking toolbox for self-hosted services | ||||
|                 // 簡化自家伺服器部署之事,初學者居家網絡必備良器 | ||||
|                 // Das ultimative Homelab-Netzwerk-Toolbox für selbstgehostete Dienste | ||||
|             </p> | ||||
|             <a href="https://github.com/tobychui/zoraxy/releases" class="ui basic white button" target="_blank"><i style="color:white;" class="ui download icon"></i><span i18n>Download // 立即下載 // Herunterladen </span></a> | ||||
|             <a href="https://github.com/tobychui/zoraxy" class="ui basic white button" target="_blank"><i style="color: white;" class="ui code icon"></i><span i18n>Source Code // 查看原始碼 // Quellcode</span></a> | ||||
|              | ||||
| 		  <div id="wavesWrapper"> | ||||
|         </div> | ||||
|         <div id="wavesWrapper"> | ||||
|             <!-- CSS waves--> | ||||
|             <svg class="waves" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|             viewBox="0 24 150 28" preserveAspectRatio="none" shape-rendering="auto"> | ||||
| @@ -137,250 +130,470 @@ | ||||
|                 </g> | ||||
|             </svg> | ||||
|         </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Features --> | ||||
|         <div id="features" class="section"> | ||||
|           <div class="ui container"> | ||||
|             <div class="ui basic segment"> | ||||
|               <h1 class="ui header"> | ||||
|                 <img class="ui small image" src="img/icons/awesome.svg"> | ||||
|                 <div class="content"> | ||||
|                     Features | ||||
|                     <div class="sub header">Highlighting a few important features of Zoraxy</div> | ||||
|                 </div> | ||||
|               </h1> | ||||
|               <br> | ||||
|               <div class="ui stackable grid featureList"> | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/proxy.svg"> | ||||
|                     <div class="content"> | ||||
|                       Reverse Proxy | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Simple to use noob-friendly reverse proxy server that can be easily set up using a web form and a few toggle switches.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/redirect.svg"> | ||||
|                     <div class="content"> | ||||
|                       Redirection | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Direct and intuitive redirection rules with basic rewrite options. Suitable for most simple use cases.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/blacklist.svg"> | ||||
|                     <div class="content"> | ||||
|                       Geo-IP & Blacklist | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Blacklist with GeoIP support. Allows easy setup for regional services.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/gan.svg"> | ||||
|                     <div class="content"> | ||||
|                       Global Area Network | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>ZeroTier controller integrated GAN. Enable unlimited nodes in your network with a few clicks.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <!-- Row 2--> | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/terminal.svg"> | ||||
|                     <div class="content"> | ||||
|                       Web SSH | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Integration with Gotty Web SSH terminal allows one-stop management of your nodes inside private LAN via gateway nodes.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/stats.svg"> | ||||
|                     <div class="content"> | ||||
|                       Real Time Statistics | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Traffic data collection and real-time analytic tools provide you the best insight of visitors data without cookies.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/scan.svg"> | ||||
|                     <div class="content"> | ||||
|                       Scanner & Utilities | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Build in IP scanner and mDNS discovery service to enable automatic service discovery within LAN.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/code.svg"> | ||||
|                     <div class="content"> | ||||
|                       Open Source | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Project is open-source under AGPL on Github. Feel free to contribute on missing functions you need! </p> | ||||
|                 </div> | ||||
|               </div> | ||||
|     </div> | ||||
|     <br><br><br><br> | ||||
|     <!-- About ArozOS--> | ||||
|     <div id="about" class="ui text container"> | ||||
|         <div class="ui stackable grid" data-aos="fade-up"> | ||||
|             <div class="six wide column" align="right"> | ||||
|                 <img class="ui medium image" src="img/preview-pc.png"> | ||||
|             </div> | ||||
|             <div class="ten wide column"> | ||||
|                 <div class="about-text-wrapper"> | ||||
|                     <p class="about-title"><b i18n>Reverse Proxy // 反向代理 // Reverse-Proxy</b></p> | ||||
|                     <p><span i18n>Easy setups with dynamic updates // 讓你想不到般簡單易用、迅速設定、動態更新 // Einfache Einrichtung mit dynamischen Updates</span></p> | ||||
|                     <p i18n>Access your reverse proxy and self-hosted services from any computer with a browser, anytime, anywhere. | ||||
|                         // 透過瀏覽器,隨時隨地在任何裝置上存取您的反向代理及自家伺服器服務。 | ||||
|                         // Greifen Sie jederzeit und überall von jedem Gerät aus auf Ihren Reverse-Proxy und selbst gehostete | ||||
|                     </p> | ||||
|                     <div class="ui list"> | ||||
|                         <div class="item"> | ||||
|                             <i class="caret right icon"></i> | ||||
|                             <div class="content" i18n> | ||||
|                                 Simple setups with web UI | ||||
|                                 // 透過網頁介面簡單設定即可使用 | ||||
|                                 // Einfache Einrichtung mit Web-UI | ||||
|                             </div> | ||||
|                         </div> | ||||
|                         <div class="item"> | ||||
|                             <i class="caret right icon"></i> | ||||
|                             <div class="content" i18n> | ||||
|                                 Change settings on the fly without restarting | ||||
|                                 // 即時更改設定,無需重新啟動 | ||||
|                                 // Einstellungen ohne Neustart ändern | ||||
|                             </div> | ||||
|                         </div> | ||||
|                         <div class="item"> | ||||
|                             <i class="caret right icon"></i> | ||||
|                             <div class="content" i18n> | ||||
|                                 One of the best reverse proxy manager for beginners | ||||
|                                 // 可能是最適合初學者的反向代理管理器之一 | ||||
|                                 // Einer der besten Reverse-Proxy-Manager für Anfänger | ||||
|                             </div> | ||||
|                         </div> | ||||
|                         <div class="item"> | ||||
|                             <i class="caret right icon"></i> | ||||
|                             <div class="content" i18n> | ||||
|                                 Easily install plugins and edit configurations | ||||
|                                 // 輕鬆安裝插件並編輯設定 | ||||
|                                 // Plugins einfach installieren und Konfigurationen bearbeiten | ||||
|                             </div> | ||||
|                         </div> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Screenshots --> | ||||
|         <div id="screenshots" class="ui container"> | ||||
|           <div class="ui basic segment"> | ||||
|             <br> | ||||
|             <h1 class="ui header"> | ||||
|               <img class="ui small image" src="img/icons/screenshots.svg"> | ||||
|               <div class="content"> | ||||
|                   Screenshots | ||||
|                   <div class="sub header">A quick overview of the UI designs</div> | ||||
|               </div> | ||||
|         <div class="ui stackable grid" data-aos="fade-up"> | ||||
|             <div class="six wide column" align="right"> | ||||
|                 <img class="ui medium image" src="img/preview-mobile.png"> | ||||
|             </div> | ||||
|             <div class="ten wide column"> | ||||
|                 <div class="about-text-wrapper"> | ||||
|                     <p class="about-title"><b i18n>Real-time Analytics // 即時流量分析 // Echtzeit-Analysen</b></p> | ||||
|                     <p><span i18n>Dynamic statistic and access control // 動態流量數據、權限與路由設定 // Dynamische Statistik und Zugriffskontrolle</span></p> | ||||
|                     <p i18n>Provide real time statistical overview, take advantage of the real time traffic and situations to make better decisions. | ||||
|                         // 提供即時統計概覽,利用即時流量和情況做出更好的決策。 | ||||
|                         // Bietet eine Echtzeit-Übersicht über die Statistiken, um bessere Entscheidungen zu treffen. | ||||
|                     </p> | ||||
|                     <div class="ui list"> | ||||
|                         <div class="item"> | ||||
|                             <i class="caret right icon"></i> | ||||
|                             <div class="content" i18n> | ||||
|                                 Real time visitor statistic | ||||
|                                 // 即時訪客統計概覽 | ||||
|                                 // Echtzeit-Besucherstatistik | ||||
|                             </div> | ||||
|                         </div> | ||||
|                         <div class="item"> | ||||
|                             <i class="caret right icon"></i> | ||||
|                             <div class="content" i18n> | ||||
|                                 Instant network utilitization overview | ||||
|                                 // 即時網路使用率概覽 | ||||
|                                 // Sofortige Netzwerkübersicht | ||||
|                             </div> | ||||
|                         </div> | ||||
|                         <div class="item"> | ||||
|                             <i class="caret right icon"></i> | ||||
|                             <div class="content" i18n> | ||||
|                                 No-reload access control and settings | ||||
|                                 // 即時生效存取控制和設定 | ||||
|                                 // Zugriffskontrolle und Einstellungen ohne Neustart | ||||
|                             </div> | ||||
|                         </div> | ||||
|                         <div class="item"> | ||||
|                             <i class="caret right icon"></i> | ||||
|                             <div class="content" i18n> | ||||
|                                 One-click setting change with no downtime | ||||
|                                 // 一鍵設定更改,無需停機 | ||||
|                                 // Einstellungsänderung mit einem Klick ohne Ausfallzeiten | ||||
|                             </div> | ||||
|                         </div> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
|     <br><br><br><br> | ||||
|     <!-- Features --> | ||||
|     <div class="ui divider"></div> | ||||
|     <div id="features" class="ui container"> | ||||
|         <div class="centered title"> | ||||
|             <h1 i18n>Screenshots  | ||||
|                 // 系統截圖 | ||||
|                 // Bildschirmfotos | ||||
|             </h1> | ||||
|              | ||||
|             <div class="ui three column stackable grid"> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/1.png" target="_blank"><img src="img/screenshots/1.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/2.png" target="_blank"><img src="img/screenshots/2.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/3.png" target="_blank"><img src="img/screenshots/3.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/4.png" target="_blank"><img src="img/screenshots/4.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/5.png" target="_blank"><img src="img/screenshots/5.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/6.png" target="_blank"><img src="img/screenshots/6.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/7.png" target="_blank"><img src="img/screenshots/7.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/8.png" target="_blank"><img src="img/screenshots/8.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/9.png" target="_blank"><img src="img/screenshots/9.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/10.png" target="_blank"><img src="img/screenshots/10.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|             </div>  | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Plugin Developments --> | ||||
|         <div id="plugins" class="ui container"> | ||||
|           <div class="ui basic segment"> | ||||
|             <br> | ||||
|             <h1 class="ui header"> | ||||
|               <img class="ui small image" src="img/icons/plugin.svg"> | ||||
|               <div class="content"> | ||||
|                   Plugins | ||||
|                   <div class="sub header">Add custom routing rules via simple scripts</div> | ||||
|               </div> | ||||
|             </h1> | ||||
|             <div style="width: 100%; text-align: center;"> | ||||
|               <br> | ||||
|               <p>Documentation work in progress</p> | ||||
|         <div class="ui three column stackable grid"> | ||||
|             <div class="column"> | ||||
|                 <img class="ui fluid image screenshot" src="img/screenshots/1.png"> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Source code --> | ||||
|         <div id="source" class="ui container"> | ||||
|           <div class="ui basic segment"> | ||||
|             <br> | ||||
|             <h1 class="ui header"> | ||||
|               <img class="ui small image" src="img/icons/code.svg"> | ||||
|               <div class="content"> | ||||
|                   Source Code | ||||
|                   <div class="sub header">Feel free to give us a ⭐ star ⭐.</div> | ||||
|               </div> | ||||
|             </h1> | ||||
|             <br> | ||||
|             <div class="ui two column stackable grid"> | ||||
|               <div class="column"> | ||||
|                   <h3 class="ui header"> | ||||
|                     <i class="grey github icon"></i> | ||||
|                     <div class="content" style="text-align: left;"> | ||||
|                       <a href="https://github.com/tobychui/zoraxy"> | ||||
|                         Github | ||||
|                         <div class="sub header">https://github.com/tobychui/zoraxy</div> | ||||
|                       </a> | ||||
|                     </div> | ||||
|                   </h3> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <h3 class="ui header"> | ||||
|                   <i class="blue mail icon"></i> | ||||
|                   <div class="content" style="text-align: left;"> | ||||
|                     <a href="mailto:toby@imuslab.com"> | ||||
|                       Email Contact | ||||
|                       <div class="sub header">toby@imuslab.com</div> | ||||
|                     </a> | ||||
|                   </div> | ||||
|                 </h3> | ||||
|               </div> | ||||
|             <div class="column"> | ||||
|                 <img class="ui fluid image screenshot" src="img/screenshots/2.png"> | ||||
|             </div> | ||||
|             <div class="column"> | ||||
|                 <img class="ui fluid image screenshot" src="img/screenshots/3.png"> | ||||
|             </div> | ||||
|             <div class="column"> | ||||
|                 <img class="ui fluid image screenshot" src="img/screenshots/4.png"> | ||||
|             </div>   | ||||
|             <div class="column"> | ||||
|                 <img class="ui fluid image screenshot" src="img/screenshots/5.png"> | ||||
|             </div>   | ||||
|             <div class="column"> | ||||
|                 <img class="ui fluid image screenshot" src="img/screenshots/6.png"> | ||||
|             </div> | ||||
|             <!-- <div class="column"> | ||||
|                 <img class="ui fluid image screenshot" src="img/screenshots/7.png"> | ||||
|             </div> --> | ||||
|             <div class="column"> | ||||
|                 <img class="ui fluid image screenshot" src="img/screenshots/8.png"> | ||||
|             </div> | ||||
|             <div class="column"> | ||||
|                 <img class="ui fluid image screenshot" src="img/screenshots/9.png"> | ||||
|             </div> | ||||
|             <div class="column"> | ||||
|                 <img class="ui fluid image screenshot" src="img/screenshots/10.png"> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <br><br> | ||||
|         <div class="ui container"> | ||||
|           <p style="color: #3a3a3a">CopyRight Zoraxy Project and its authors © 2021 - <span class="year"></span></p> | ||||
|         </div> | ||||
|         <br><br><br> | ||||
|       </div> | ||||
|     </div> | ||||
|     <br> | ||||
|  | ||||
|     <!-- Spec --> | ||||
|     <div id="techspec" class="blackbanner"> | ||||
|         <br><br> | ||||
|         <div class="centered title"> | ||||
|             <h1 style="font-weight: 600;" i18n> | ||||
|                 Review Videos  | ||||
|                 // 介紹影片  | ||||
|                 // Videos | ||||
|             </h1> | ||||
|         </div> | ||||
|         <div> | ||||
|             <div class="videoScrollBar"> | ||||
|                 <div class="introvideo"><iframe width="560" height="315" src="https://www.youtube.com/embed/5-lps8DC6_Y?si=rkfePn9kiYKCvYUZ" title="YouTube video player" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share" referrerpolicy="strict-origin-when-cross-origin" allowfullscreen></iframe></div> | ||||
|                 <div class="introvideo"><iframe width="560" height="315" src="https://www.youtube.com/embed/49xQYLpmedE?si=fgba2iK55s1760Xr" title="YouTube video player" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share" referrerpolicy="strict-origin-when-cross-origin" allowfullscreen></iframe></div> | ||||
|                 <div class="introvideo"><iframe width="560" height="315" src="https://www.youtube.com/embed/I_F97he5F2A?si=qKEXwDcjkX1nPejq" title="YouTube video player" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share" referrerpolicy="strict-origin-when-cross-origin" allowfullscreen></iframe></div> | ||||
|                 <div class="introvideo"><iframe width="560" height="315" src="https://www.youtube.com/embed/FNU08-ufByM?si=I2hq9vsapeXB2Oqb" title="YouTube video player" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share" referrerpolicy="strict-origin-when-cross-origin" allowfullscreen></iframe></div> | ||||
|             </div>    | ||||
|         </div> | ||||
|         <br><br><br> | ||||
|     </div> | ||||
|  | ||||
|     <!-- Download --> | ||||
|     <div id="download" class="ui text container"> | ||||
|         <br><br> | ||||
|         <div class="centered title"> | ||||
|             <h1 i18n> | ||||
|                 Download  | ||||
|                 // 下載 | ||||
|                 // Herunterladen | ||||
|             </h1> | ||||
|         </div> | ||||
|          | ||||
|         <div class="downloadTabWrapper"> | ||||
|             <div class="ui top attached fluid stackable tabular menu"> | ||||
|                 <a class="active item" data-tab="linux"><i class="grey linux icon"></i> Linux</a> | ||||
|                 <a class="item" data-tab="windows"><i class="blue windows icon"></i> Windows</a> | ||||
|                 <a class="item" data-tab="rpi"><i class="red raspberry pi icon"></i><span i18n>SBCs // ARM 開發板 // SBCs</span></a> | ||||
|                 <a class="item" data-tab="build"><i class="code icon"></i> <span i18n>Build from source // 從原始碼建置 // Aus dem Quellcode erstellen</span> </a> | ||||
|             </div> | ||||
|             <div class="ui bottom attached active tab segment" data-tab="linux"> | ||||
|                 <p i18n> | ||||
|                     Install with command line  | ||||
|                     // 使用 CLI 下載並執行發行版本  | ||||
|                     // Installieren Sie mit der Befehlszeile | ||||
|                 </p> | ||||
|                 <div class="ui black message"> | ||||
|                     <code> | ||||
|                         wget https://github.com/tobychui/zoraxy/releases/latest/download/zoraxy_linux_amd64<br> | ||||
|                         chmod +x ./zoraxy_linux_amd64<br> | ||||
|                         sudo ./zoraxy_linux_amd64 | ||||
|                     </code> | ||||
|                 </div> | ||||
|                 <br> | ||||
|                 <p i18n> | ||||
|                     Install with precompiled binary  | ||||
|                     // 下載發行版本  | ||||
|                     // Installieren Sie mit vorkompilierten Binärdateien | ||||
|                 </p> | ||||
|                 <button class="ui basic downloadButton button" onclick='handleDownload("zoraxy_linux_amd64");'> | ||||
|                     <i class="black linux icon"></i>  | ||||
|                     <span i18n>Download x64  | ||||
|                         // 下載 64位元 執行檔  | ||||
|                         // Herunterladen x64 | ||||
|                     </span> | ||||
|                 </button> | ||||
|                 <span style="font-size: 1.2em; font-weight: 600; margin-right: 0.4em">OR</span> | ||||
|                 <button class="ui basic downloadButton button" onclick='handleDownload("zoraxy_linux_386");'> | ||||
|                     <i class="black linux icon"></i>  | ||||
|                     <span i18n>Download x32  | ||||
|                         // 下載 32位元 執行檔  | ||||
|                         // Herunterladen x32 | ||||
|                     </span> | ||||
|                 </button> | ||||
|             </div> | ||||
|             <div class="ui bottom attached tab segment" data-tab="windows"> | ||||
|                 <p i18n> | ||||
|                     Install with precompiled binary  | ||||
|                     // 下載發行版本  | ||||
|                     // Installieren Sie mit vorkompilierten Binärdateien | ||||
|                 </p> | ||||
|                 <button class="ui basic downloadButton button" onclick='handleDownload("zoraxy_windows_amd64.exe");'> | ||||
|                     <i class="blue windows icon"></i>  | ||||
|                     <span i18n> | ||||
|                         Download Zoraxy for Windows  | ||||
|                         // 下載 Windows 版 Zoraxy  | ||||
|                         // Zoraxy für Windows herunterladen | ||||
|                     </span> | ||||
|                 </button> | ||||
|                 <br><br> | ||||
|             </div> | ||||
|             <div class="ui bottom attached tab segment" data-tab="rpi"> | ||||
|                 <p i18n>Install with command line (armv6-7, arm64, x86)  | ||||
|                     // 使用 CLI 下載並執行 (armv6-7, arm64, x86) | ||||
|                     // Installieren Sie mit der Befehlszeile (armv6-7, arm64, x86) | ||||
|                 </p> | ||||
|                 <div class="ui black message"> | ||||
|                     <code> | ||||
|                         # Check your CPU architecture<br> | ||||
|                         uname -m <br> | ||||
|                         <br> | ||||
|                         # For arm64 (aarch64) CPU<br> | ||||
|                         wget -O zoraxy https://github.com/tobychui/zoraxy/releases/latest/download/zoraxy_linux_arm64<br> | ||||
|                         <br> | ||||
|                         # For armv6 (armv6l) / armv7 (armv7l) CPU<br> | ||||
|                         wget -O zoraxy https://github.com/tobychui/zoraxy/releases/latest/download/zoraxy_linux_arm<br> | ||||
|                         <br> | ||||
|                         # For RISC-V (riscv64) CPU<br> | ||||
|                         wget -O zoraxy https://github.com/tobychui/zoraxy/releases/latest/download/zoraxy_linux_riscv64<br> | ||||
|                         <br> | ||||
|  | ||||
|                         chmod +x ./zoraxy<br> | ||||
|                         sudo ./zoraxy <br> | ||||
|                     </code> | ||||
|                 </div> | ||||
|                 <br> | ||||
|                 <p i18n>Install with precompiled binary  | ||||
|                     // 下載發行版本 | ||||
|                     // Installieren Sie mit vorkompilierten Binärdateien | ||||
|                 </p> | ||||
|                 <button class="ui basic downloadButton button" onclick='handleDownload("zoraxy_linux_arm");'><i class="black download icon"></i> <span i18n></span>arm (v6, v7)</button> | ||||
|                 <button class="ui basic downloadButton button" onclick='handleDownload("zoraxy_linux_arm64");'><i class="black download icon"></i> <span i18n></span>arm64</button> | ||||
|                 <button class="ui basic downloadButton button" onclick='handleDownload("zoraxy_linux_riscv64");'><i class="grey download icon"></i> <span i18n></span>riscv64</button> | ||||
|             </div> | ||||
|             <div class="ui bottom attached tab segment" data-tab="build"> | ||||
|                 <p i18n>Require Go (Golang) compiler. Details build from source instruction can be found on Zoraxy Github README file.  | ||||
|                     // 需要 Go (Go 語言)編譯器。建置詳情可以在 Zoraxy Github README 檔案中找到。 | ||||
|                     // Erfordert den Go (Golang) Compiler. Detaillierte Anweisungen zum Erstellen aus dem Quellcode finden Sie in der Zoraxy Github README-Datei. | ||||
|                 </p> | ||||
|                 <div class="ui black message"> | ||||
|                     <code> | ||||
|                         git clone https://github.com/tobychui/zoraxy<br> | ||||
|                         cd ./zoraxy/src/<br> | ||||
|                         go mod tidy<br> | ||||
|                         go build<br> | ||||
|                         sudo ./zoraxy <br> | ||||
|                     </code> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|         <p> | ||||
|             <span i18n>After Zoraxy is started, navigate to  | ||||
|                 // 當 Zoraxy 執行檔 / 服務啟動後,使用瀏覽器開啟 | ||||
|                 // Nachdem Zoraxy gestartet wurde, navigieren Sie zu | ||||
|             </span>  | ||||
|             <a href="http://localhost:8000" target="_blank">http://localhost:8000</a>  | ||||
|             <span i18n>to continue account and system setup.  | ||||
|                 // 以繼續帳戶和系統設定。 | ||||
|                 // um die Konto- und Systemeinrichtung fortzusetzen. | ||||
|             </span> | ||||
|         </p> | ||||
|         <br><br> | ||||
|     </div> | ||||
|  | ||||
|     <!-- Learn More --> | ||||
|     <div class="ui divider"></div> | ||||
|     <div id="learnmore" class="ui text container"> | ||||
|         <div class="centered title" style="margin-bottom: 0px;"> | ||||
|             <h1 i18n>Learn More  | ||||
|                 // 了解更多 | ||||
|                 // Mehr erfahren | ||||
|             </h1> | ||||
|             <p i18n>If you like this project, please feel free to give us a ⭐ star ⭐. | ||||
|                 // 如果您喜歡這個開源專案,歡迎來給我們一顆 ⭐星星⭐ 喔!! | ||||
|                 // Wenn Ihnen dieses Projekt gefällt, geben Sie uns bitte einen ⭐ Stern ⭐. | ||||
|             </p> | ||||
|         </div> | ||||
|         <div class="ui basic segment linkicons"> | ||||
|             <div class="ui big breadcrumb"> | ||||
|                 <a class="section externallink" href="https://github.com/tobychui/zoraxy" target="_blank"> | ||||
|                     <div class="ui icon header"> | ||||
|                         <i class="black github icon"></i> | ||||
|                         <div class="content" i18n> | ||||
|                             Github | ||||
|                             // 源碼 | ||||
|                             // Quellcode | ||||
|                         </div> | ||||
|                     </div> | ||||
|                 </a> | ||||
|                 <i class="divider"> </i> | ||||
|                 <a class="section externallink" href="" target="_blank"> | ||||
|                     <div class="ui icon header"> | ||||
|                         <i class="green code icon"></i> | ||||
|                         <div class="content" i18n> | ||||
|                             Plugin Devs | ||||
|                             // 插件開發 | ||||
|                             // Plugin-Entwickler | ||||
|                         </div> | ||||
|                     </div> | ||||
|                 </a> | ||||
|                 <i class=" divider"> </i> | ||||
|                 <a class="section externallink" href="mailto:toby@imuslab.com" target="_blank"> | ||||
|                     <div class="ui icon header"> | ||||
|                         <i class="yellow mail icon"></i> | ||||
|                         <div class="content" i18n> | ||||
|                             Email | ||||
|                             // 電子郵件 | ||||
|                             // E-Mail | ||||
|                         </div> | ||||
|                     </div> | ||||
|                 </a> | ||||
|                 <i class=" divider"> </i> | ||||
|                 <a class="section externallink" href="https://t.me/ArOZBeta" target="_blank"> | ||||
|                     <div class="ui icon header"> | ||||
|                         <i class="blue telegram icon"></i> | ||||
|                         <div class="content"> | ||||
|                             Telegram | ||||
|                         </div> | ||||
|                     </div> | ||||
|                 </a> | ||||
|             </div>   | ||||
|         </div> | ||||
|         <br><br><br><br> | ||||
|     </div> | ||||
|  | ||||
|      <!-- Footer --> | ||||
|      <div id="footer"> | ||||
|         <div class="ui container"> | ||||
|             <br><br> | ||||
|             <div class="ui stackable grid" style="height: 100%;"> | ||||
|                 <div class="six wide column" style="height: 100%;"> | ||||
|                     <a href="https://zoraxy.aroz.org"><img src="img/logo_white.png" class="ui small image"></a> | ||||
|                     <p><span style="font-weight: 300;">The Zoraxy Project</span><br> | ||||
|                         © Toby Chui</p> | ||||
|                         | ||||
|                     <div class="bottom-attach"> | ||||
|                        <br><br> | ||||
|                         <div class="ui breadcrumb" style="margin-top: 0.4em;"> | ||||
|                             <div class="section" i18n><a style="color: white;" href="https://zoraxy.aroz.org" target="_blank">zoraxy.aroz.org</a></div> | ||||
|                             <div class="divider"> / </div> | ||||
|                             <div class="section">2018 - <span class="year">now</span></div> | ||||
|                         </div> | ||||
|                     </div> | ||||
|                 </div> | ||||
|                 <div class="three wide column"> | ||||
|                     <div class="ui list"> | ||||
|                         <div class="item title" i18n>Developer Tools  | ||||
|                             // 開發者工具 | ||||
|                             // Entwicklerwerkzeuge | ||||
|                         </div> | ||||
|                         <div class="item"><a href="https://github.com/tobychui/zoraxy/wiki" target="_blank">Zoraxy Wiki</a></div> | ||||
|                         <div class="item"><a href="https://github.com/tobychui/zoraxy" target="_blank">Source Code</a></div> | ||||
|                         <div class="item"><a href="" target="_blank">Offical Plugin List</a></div> | ||||
|                         <div class="item"><a href="" target="_blank">Plugin Development Guide</a></div> | ||||
|                     </div> | ||||
|                 </div>	 | ||||
|                 <div class="three wide column"> | ||||
|                     <div class="ui list"> | ||||
|                         <div class="item title" i18n>Project Spin-offs  | ||||
|                             // 衍生開源計劃 | ||||
|                             // Projekt-Ableger | ||||
|                         </div> | ||||
|                         <div class="item"><a href="https://os.aroz.org" target="_blank">ArozOS</a></div> | ||||
|                     </div> | ||||
|                 </div> | ||||
|                 <div class="three wide column"> | ||||
|                     <div class="ui list"> | ||||
|                         <div class="item title" i18n>Related Links  | ||||
|                             // 相關連接 | ||||
|                             // Verwandte Links | ||||
|                         </div> | ||||
|                         <div class="item"><a href="https://github.com/tobychui/zoraxy/wiki/Getting-Started" target="_blank" i18n>Getting Started</a></div> | ||||
|                         <div class="item"><a href="https://github.com/tobychui/zoraxy/releases" target="_blank">Zoraxy Release</a></div> | ||||
|                         <div class="item"><a href="https://hub.docker.com/r/zoraxydocker/zoraxy" target="_blank">Zoraxy Docker</a></div> | ||||
|                         <div class="item"><a href="https://imuslab.com" target="_blank">imuslab</a></div> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|         <br><br><br><br> | ||||
|     </div> | ||||
|      | ||||
|     <script> | ||||
|       $(".year").text(new Date().getFullYear()  ); | ||||
|         AOS.init(); | ||||
|         $(".year").text(new Date().getFullYear()); | ||||
|  | ||||
|       $(".menu-item").on("click", function(){ | ||||
|         $(".menu-item.active").removeClass("active"); | ||||
|         $(this).addClass("active"); | ||||
|       }); | ||||
|         // Function to open the modal with the clicked image using jQuery | ||||
|         function openModal(src) { | ||||
|             // Remove the old modal if it exists | ||||
|             $('#imageModal').remove(); | ||||
|  | ||||
|       $(".right-content").on("scroll", function() { | ||||
|         var scrollPos = $(".right-content").scrollTop(); | ||||
|         if (scrollPos < 10){ | ||||
|           //Reaching the top | ||||
|           $('.menu-item.active').removeClass("active"); | ||||
|           $($('.menu-item')[0]).addClass('active'); | ||||
|           return; | ||||
|         }else if ($(".right-content")[0].scrollHeight  ==  $(".right-content").scrollTop() + window.innerHeight ){ | ||||
|           //Reaching the bottom | ||||
|           $('.menu-item.active').removeClass("active"); | ||||
|           $($('.menu-item').get().reverse()[0]).addClass('active'); | ||||
|           return | ||||
|             const modal = $('<div style="display:none;">', { id: 'imageModal' }).css({ | ||||
|                 position: 'fixed', | ||||
|                 top: '0', | ||||
|                 left: '0', | ||||
|                 width: '100%', | ||||
|                 height: '100%', | ||||
|                 backgroundColor: 'rgba(0, 0, 0, 0.8)', | ||||
|                 display: 'flex', | ||||
|                 alignItems: 'center', | ||||
|                 justifyContent: 'center', | ||||
|                 zIndex: '1000' | ||||
|             }); | ||||
|  | ||||
|             const img = $('<img>', { src: src }).css({ | ||||
|                 maxWidth: '80%', | ||||
|                 maxHeight: '80%', | ||||
|                 boxShadow: '0 0 10px rgba(0, 0, 0, 0.5)' | ||||
|             }); | ||||
|  | ||||
|             modal.append(img); | ||||
|             $("body").css("overflow", "hidden"); | ||||
|  | ||||
|             modal.on('click', function() { | ||||
|                 modal.remove(); | ||||
|                 $("body").css("overflow", "auto"); | ||||
|             }); | ||||
|  | ||||
|             $('body').append(modal); | ||||
|         } | ||||
|         $('.menu-item').each(function() { | ||||
|           var currLink = $(this); | ||||
|           var refElement = $(currLink.attr("href")); | ||||
|           if (refElement.offset().top <= (window.innerHeight / 2)) { | ||||
|             $('.menu-item.active').removeClass("active"); | ||||
|             currLink.addClass("active"); | ||||
|             console.log(currLink.attr("href")); | ||||
|           } | ||||
|         }); | ||||
|       }); | ||||
|  | ||||
|         // Add click event listener to all screenshot images using jQuery | ||||
|         $('.screenshot').on('click', function() { | ||||
|             openModal($(this).attr('src')); | ||||
|         }); | ||||
|     </script> | ||||
|     <!-- Locales --> | ||||
|     <script src="main.js" defer></script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										386
									
								
								docs/index_legacy.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										386
									
								
								docs/index_legacy.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,386 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|   <head> | ||||
|     <meta charset="utf-8"> | ||||
|     <meta content="width=device-width, initial-scale=1.0" name="viewport"> | ||||
|     <meta content="Reverse Proxy, Cluster, Gateway, Go, Homelab, Network Tools" name="keywords"> | ||||
|     <meta content="A reverse proxy server and cluster network gateway for noobs" name="description"> | ||||
|     <meta name="author" content="tobychui"> | ||||
|  | ||||
|     <!-- HTML Meta Tags --> | ||||
|     <title>Reverse Proxy Server | Zoraxy</title> | ||||
|     <meta name="description" content="A reverse proxy server and cluster network gateway for noobs"> | ||||
|  | ||||
|     <!-- Facebook Meta Tags --> | ||||
|     <meta property="og:url" content="https://zoraxy.aroz.org/"> | ||||
|     <meta property="og:type" content="website"> | ||||
|     <meta property="og:title" content="Cluster Proxy Gateway | Zoraxy"> | ||||
|     <meta property="og:description" content="A reverse proxy server and cluster network gateway for noobs"> | ||||
|     <meta property="og:image" content="https://zoraxy.aroz.org/img/og.png"> | ||||
|  | ||||
|     <!-- Twitter Meta Tags --> | ||||
|     <meta name="twitter:card" content="summary_large_image"> | ||||
|     <meta property="twitter:domain" content="aroz.org"> | ||||
|     <meta property="twitter:url" content="https://zoraxy.aroz.org/"> | ||||
|     <meta name="twitter:title" content="Cluster Proxy Gateway | Zoraxy"> | ||||
|     <meta name="twitter:description" content="A reverse proxy server and cluster network gateway for noobs"> | ||||
|     <meta name="twitter:image" content="https://zoraxy.aroz.org/img/og.png"> | ||||
|  | ||||
|     <!-- Favicons --> | ||||
|     <link href="favicon.png" rel="icon"> | ||||
|  | ||||
|     <!-- Google Fonts --> | ||||
|     <link rel="preconnect" href="https://fonts.googleapis.com"> | ||||
|     <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin> | ||||
|     <link href="https://fonts.googleapis.com/css2?family=Source+Sans+Pro:wght@100;300;400;600;700;900&display=swap" rel="stylesheet"> | ||||
|     <link rel="stylesheet" href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL,GRAD@48,400,0,0" /> | ||||
|  | ||||
|     <!-- Main Stylesheet File --> | ||||
|     <link href="style.css" rel="stylesheet"> | ||||
|     <script | ||||
|       src="https://code.jquery.com/jquery-3.7.0.min.js" | ||||
|       integrity="sha256-2Pmvv0kuTBOenSvLm6bvfBSSHrUJ+3A7x6P5Ebd07/g=" | ||||
|       crossorigin="anonymous"></script> | ||||
|     <script src="https://cdnjs.cloudflare.com/ajax/libs/fomantic-ui/2.9.2/semantic.min.js" integrity="sha512-5cguXwRllb+6bcc2pogwIeQmQPXEzn2ddsqAexIBhh7FO1z5Hkek1J9mrK2+rmZCTU6b6pERxI7acnp1MpAg4Q==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> | ||||
|     <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/fomantic-ui/2.9.2/semantic.min.css" integrity="sha512-n//BDM4vMPvyca4bJjZPDh7hlqsQ7hqbP9RH18GF2hTXBY5amBwM2501M0GPiwCU/v9Tor2m13GOTFjk00tkQA==" crossorigin="anonymous" referrerpolicy="no-referrer" /> | ||||
|     <style> | ||||
|       p,a,div,span,h1,h2,h3,h4,h5,h6{ | ||||
|         font-family: 'Source Sans Pro', sans-serif !important; | ||||
|         color: #404040; | ||||
|       } | ||||
|     </style> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div class="main section"> | ||||
|       <div class="left-menu"> | ||||
|         <div class="iconWrapper"> | ||||
|           <a href="index.html"><img class="ui fluid image" src="img/icon.png"></a> | ||||
|         </div> | ||||
|         <a href="#home" class="menu-item active" align="center"> | ||||
|           <img src="img/icons/home.svg"> | ||||
|         </a> | ||||
|         <a href="#features" class="menu-item" align="center"> | ||||
|           <img src="img/icons/awesome.svg"> | ||||
|         </a> | ||||
|         <a href="#screenshots" class="menu-item" align="center"> | ||||
|           <img src="img/icons/screenshots.svg"> | ||||
|         </a> | ||||
|         <a href="#plugins" class="menu-item" align="center"> | ||||
|           <img src="img/icons/plugin.svg"> | ||||
|         </a> | ||||
|         <a href="#source" class="menu-item" align="center"> | ||||
|           <img src="img/icons/code.svg"> | ||||
|         </a> | ||||
|       </div> | ||||
|       <div class="right-content"> | ||||
|         <!-- Hero Banner Section --> | ||||
|         <div class="headbanner"></div> | ||||
|         <div id="home" class="herotext"> | ||||
|           <div class="ui basic segment"> | ||||
|             <div class="bannerHeaderWrapper"> | ||||
|               <h1 class="bannerHeader">Zoraxy</h1> | ||||
| 			  <div class="ui divider"></div><br> | ||||
|               <p class="bannerSubheader">Beyond Reverse Proxy: Your Ultimate Homelab Network Tool</p> | ||||
|             </div> | ||||
|             <br><br> | ||||
|             <a class="ui basic big button" style="background-color: white;" href="#features"><i class="ui blue arrow down icon"></i> Learn More</a> | ||||
|             <br><br> | ||||
|             <table class="ui very basic collapsing unstackable celled table"> | ||||
|               <thead> | ||||
|                 <tr><th colspan="2">Quick Access</th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|               <tr> | ||||
|                 <td> | ||||
|                   <h4 class="ui image header"> | ||||
|                     <i class="ui download icon"></i> | ||||
|                     <div class="content"> | ||||
|                       Download | ||||
|                       <div class="sub header">Prebuild Binary | ||||
|                     </div> | ||||
|                   </div> | ||||
|                 </h4></td> | ||||
|                 <td> | ||||
|                   <a href="https://github.com/tobychui/zoraxy/releases" target="_blank">Open <i class="ui external icon"></i></a> | ||||
|                 </td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td> | ||||
|                   <h4 class="ui image header"> | ||||
|                     <i class="ui github icon"></i> | ||||
|                     <div class="content"> | ||||
|                       Github | ||||
|                       <div class="sub header">Source Code | ||||
|                     </div> | ||||
|                   </div> | ||||
|                 </h4></td> | ||||
|                 <td> | ||||
|                   <a href="https://github.com/tobychui/zoraxy" target="_blank">Open <i class="ui external icon"></i></a> | ||||
|                 </td> | ||||
|               </tr> | ||||
|             </table> | ||||
|           </div> | ||||
| 		   | ||||
| 		  <div id="wavesWrapper"> | ||||
|             <!-- CSS waves--> | ||||
|             <svg class="waves" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" | ||||
|             viewBox="0 24 150 28" preserveAspectRatio="none" shape-rendering="auto"> | ||||
|                 <defs> | ||||
|                 <path id="gentle-wave" d="M-160 44c30 0 58-18 88-18s 58 18 88 18 58-18 88-18 58 18 88 18 v44h-352z" /> | ||||
|                 </defs> | ||||
|                 <g class="parallax"> | ||||
|                 <use xlink:href="#gentle-wave" x="48" y="0" fill="rgba(255,255,255,0.7" /> | ||||
|                 <use xlink:href="#gentle-wave" x="48" y="3" fill="rgba(255,255,255,0.5)" /> | ||||
|                 <use xlink:href="#gentle-wave" x="48" y="5" fill="rgba(255,255,255,0.3)" /> | ||||
|                 <use xlink:href="#gentle-wave" x="48" y="7" fill="#fff" /> | ||||
|                 </g> | ||||
|             </svg> | ||||
|         </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Features --> | ||||
|         <div id="features" class="section"> | ||||
|           <div class="ui container"> | ||||
|             <div class="ui basic segment"> | ||||
|               <h1 class="ui header"> | ||||
|                 <img class="ui small image" src="img/icons/awesome.svg"> | ||||
|                 <div class="content"> | ||||
|                     Features | ||||
|                     <div class="sub header">Highlighting a few important features of Zoraxy</div> | ||||
|                 </div> | ||||
|               </h1> | ||||
|               <br> | ||||
|               <div class="ui stackable grid featureList"> | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/proxy.svg"> | ||||
|                     <div class="content"> | ||||
|                       Reverse Proxy | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Simple to use noob-friendly reverse proxy server that can be easily set up using a web form and a few toggle switches.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/redirect.svg"> | ||||
|                     <div class="content"> | ||||
|                       Redirection | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Direct and intuitive redirection rules with basic rewrite options. Suitable for most simple use cases.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/blacklist.svg"> | ||||
|                     <div class="content"> | ||||
|                       Geo-IP & Blacklist | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Blacklist with GeoIP support. Allows easy setup for regional services.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/gan.svg"> | ||||
|                     <div class="content"> | ||||
|                       Global Area Network | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>ZeroTier controller integrated GAN. Enable unlimited nodes in your network with a few clicks.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <!-- Row 2--> | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/terminal.svg"> | ||||
|                     <div class="content"> | ||||
|                       Web SSH | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Integration with Gotty Web SSH terminal allows one-stop management of your nodes inside private LAN via gateway nodes.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/stats.svg"> | ||||
|                     <div class="content"> | ||||
|                       Real Time Statistics | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Traffic data collection and real-time analytic tools provide you the best insight of visitors data without cookies.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/scan.svg"> | ||||
|                     <div class="content"> | ||||
|                       Scanner & Utilities | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Build in IP scanner and mDNS discovery service to enable automatic service discovery within LAN.</p> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="four wide column featureItem"> | ||||
|                   <h3 class="ui header featureHeader"> | ||||
|                     <img class="ui image" src="img/icons/code.svg"> | ||||
|                     <div class="content"> | ||||
|                       Open Source | ||||
|                     </div> | ||||
|                   </h3> | ||||
|                   <p>Project is open-source under AGPL on Github. Feel free to contribute on missing functions you need! </p> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Screenshots --> | ||||
|         <div id="screenshots" class="ui container"> | ||||
|           <div class="ui basic segment"> | ||||
|             <br> | ||||
|             <h1 class="ui header"> | ||||
|               <img class="ui small image" src="img/icons/screenshots.svg"> | ||||
|               <div class="content"> | ||||
|                   Screenshots | ||||
|                   <div class="sub header">A quick overview of the UI designs</div> | ||||
|               </div> | ||||
|             </h1> | ||||
|              | ||||
|             <div class="ui three column stackable grid"> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/1.png" target="_blank"><img src="img/screenshots/1.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/2.png" target="_blank"><img src="img/screenshots/2.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/3.png" target="_blank"><img src="img/screenshots/3.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/4.png" target="_blank"><img src="img/screenshots/4.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/5.png" target="_blank"><img src="img/screenshots/5.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/6.png" target="_blank"><img src="img/screenshots/6.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/7.png" target="_blank"><img src="img/screenshots/7.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/8.png" target="_blank"><img src="img/screenshots/8.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/9.png" target="_blank"><img src="img/screenshots/9.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <a href="img/screenshots/10.png" target="_blank"><img src="img/screenshots/10.png" class="ui fluid image screenshot"></a> | ||||
|               </div> | ||||
|             </div>  | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Plugin Developments --> | ||||
|         <div id="plugins" class="ui container"> | ||||
|           <div class="ui basic segment"> | ||||
|             <br> | ||||
|             <h1 class="ui header"> | ||||
|               <img class="ui small image" src="img/icons/plugin.svg"> | ||||
|               <div class="content"> | ||||
|                   Plugins | ||||
|                   <div class="sub header">Add custom routing rules via simple scripts</div> | ||||
|               </div> | ||||
|             </h1> | ||||
|             <div style="width: 100%; text-align: center;"> | ||||
|               <br> | ||||
|               <p>Documentation work in progress</p> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Source code --> | ||||
|         <div id="source" class="ui container"> | ||||
|           <div class="ui basic segment"> | ||||
|             <br> | ||||
|             <h1 class="ui header"> | ||||
|               <img class="ui small image" src="img/icons/code.svg"> | ||||
|               <div class="content"> | ||||
|                   Source Code | ||||
|                   <div class="sub header">Feel free to give us a ⭐ star ⭐.</div> | ||||
|               </div> | ||||
|             </h1> | ||||
|             <br> | ||||
|             <div class="ui two column stackable grid"> | ||||
|               <div class="column"> | ||||
|                   <h3 class="ui header"> | ||||
|                     <i class="grey github icon"></i> | ||||
|                     <div class="content" style="text-align: left;"> | ||||
|                       <a href="https://github.com/tobychui/zoraxy"> | ||||
|                         Github | ||||
|                         <div class="sub header">https://github.com/tobychui/zoraxy</div> | ||||
|                       </a> | ||||
|                     </div> | ||||
|                   </h3> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|                 <h3 class="ui header"> | ||||
|                   <i class="blue mail icon"></i> | ||||
|                   <div class="content" style="text-align: left;"> | ||||
|                     <a href="mailto:toby@imuslab.com"> | ||||
|                       Email Contact | ||||
|                       <div class="sub header">toby@imuslab.com</div> | ||||
|                     </a> | ||||
|                   </div> | ||||
|                 </h3> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <br><br> | ||||
|         <div class="ui container"> | ||||
|           <p style="color: #3a3a3a">CopyRight Zoraxy Project and its authors © 2021 - <span class="year"></span></p> | ||||
|         </div> | ||||
|         <br><br><br> | ||||
|       </div> | ||||
|     </div> | ||||
|     <br> | ||||
|     <script> | ||||
|       $(".year").text(new Date().getFullYear()  ); | ||||
|  | ||||
|       $(".menu-item").on("click", function(){ | ||||
|         $(".menu-item.active").removeClass("active"); | ||||
|         $(this).addClass("active"); | ||||
|       }); | ||||
|  | ||||
|       $(".right-content").on("scroll", function() { | ||||
|         var scrollPos = $(".right-content").scrollTop(); | ||||
|         if (scrollPos < 10){ | ||||
|           //Reaching the top | ||||
|           $('.menu-item.active').removeClass("active"); | ||||
|           $($('.menu-item')[0]).addClass('active'); | ||||
|           return; | ||||
|         }else if ($(".right-content")[0].scrollHeight  ==  $(".right-content").scrollTop() + window.innerHeight ){ | ||||
|           //Reaching the bottom | ||||
|           $('.menu-item.active').removeClass("active"); | ||||
|           $($('.menu-item').get().reverse()[0]).addClass('active'); | ||||
|           return | ||||
|         } | ||||
|         $('.menu-item').each(function() { | ||||
|           var currLink = $(this); | ||||
|           var refElement = $(currLink.attr("href")); | ||||
|           if (refElement.offset().top <= (window.innerHeight / 2)) { | ||||
|             $('.menu-item.active').removeClass("active"); | ||||
|             currLink.addClass("active"); | ||||
|             console.log(currLink.attr("href")); | ||||
|           } | ||||
|         }); | ||||
|       }); | ||||
|  | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										462
									
								
								docs/main.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										462
									
								
								docs/main.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,462 @@ | ||||
| /* Global */ | ||||
|  | ||||
| p,a,div,span,h1,h2,h3,h4,h5,h6{ | ||||
|     font-family: 'Source Sans Pro', sans-serif; | ||||
| } | ||||
|  | ||||
| body.en *:not(i){ | ||||
|     font-family: 'Source Sans Pro', 'Noto Sans TC',sans-serif !important; | ||||
| } | ||||
|  | ||||
| body.zh *:not(i){ | ||||
|     font-family: 'Noto Sans TC',sans-serif !important; | ||||
| } | ||||
|  | ||||
| body.jp *:not(i){ | ||||
|     font-family: "Noto Sans JP", sans-serif !important; | ||||
| } | ||||
|  | ||||
| body.zh-cn *:not(i){ | ||||
|     font-family: 'Noto Sans SC',sans-serif !important; | ||||
| } | ||||
|  | ||||
|  | ||||
| .centered.title{ | ||||
|     padding: 2em; | ||||
|     margin-bottom: 2em; | ||||
|     text-align: center; | ||||
| } | ||||
| .centered.title h1{ | ||||
|     font-weight: 300 !important; | ||||
| } | ||||
|  | ||||
| .messageBanner{ | ||||
|     width: 100%; | ||||
|     background: #6cacff; | ||||
|     text-align: center; | ||||
|     color: white; | ||||
|     padding: 10px; | ||||
| } | ||||
| .messageBanner .header{ | ||||
|     font-weight: 500; | ||||
| } | ||||
|  | ||||
| #backToTopBtn{ | ||||
|     position: fixed; | ||||
|     bottom: 1em; | ||||
|     right: 1em; | ||||
|     display:none; | ||||
|     z-index: 999; | ||||
|     border: 1px solid white; | ||||
|     background: #6cacff; | ||||
| } | ||||
|  | ||||
| #backToTopBtn:hover{ | ||||
|     opacity: 0.8; | ||||
| } | ||||
|  | ||||
| #backToTopBtn i{ | ||||
|     color: white; | ||||
| } | ||||
|  | ||||
| /* Main Menu */ | ||||
| #mainmenu{ | ||||
|     padding-top: 0.4em; | ||||
|     padding-bottom: 0.4em; | ||||
|     border-radius: 0; | ||||
|     margin-bottom: 0; | ||||
|     margin-top: 0; | ||||
|     background: transparent !important; | ||||
| } | ||||
|  | ||||
| #slideshowBanner .ui.basic.white.button{ | ||||
|     color: white; | ||||
|     box-shadow: 0 0 0 1px rgb(231, 231, 231) inset; | ||||
|     border-radius: 0.4em; | ||||
|     background: none !important; | ||||
| } | ||||
| #slideshowBanner .ui.basic.white.button:hover{ | ||||
|     background-color: rgba(255, 255, 255, 0.3) !important; | ||||
| } | ||||
|  | ||||
| #slideshowBanner .ui.basic.white.button:active{ | ||||
|     background: rgba(255, 255, 255, 0.5) !important; | ||||
| } | ||||
|  | ||||
| #rwdmenubtn{ | ||||
|     display:none; | ||||
|     position: absolute; | ||||
|     background: white; | ||||
|     border: 1px solid #6cacff; | ||||
|     color: #6cacff; | ||||
| } | ||||
|  | ||||
| #mainmenu .ui.secondary.inverted.menu .link.item:not(.disabled), .ui.secondary.inverted.menu a.item:not(.disabled){ | ||||
|     font-size: 1.1em; | ||||
|     font-weight: 500; | ||||
|     border-bottom: 1px solid transparent; | ||||
|     transition: border-bottom ease-in-out 0.1s; | ||||
|     color: white !important; | ||||
|     border-radius: 0; | ||||
| } | ||||
|  | ||||
| #mainmenu #mainmenu .ui.secondary.inverted.menu .link.item:not(.disabled), .ui.secondary.inverted.menu a.item:not(.disabled):hover{ | ||||
|     background-color: transparent; | ||||
|     border-bottom: 1px solid #82adfc; | ||||
|     color: #82adfc !important; | ||||
| } | ||||
|  | ||||
| /* Image Sldiers */ | ||||
| #slideshowBanner{ | ||||
|     background: rgb(108,172,255); | ||||
|     background: linear-gradient(48deg, rgba(108,172,255,1) 8%, rgba(141,235,255,1) 65%);  | ||||
|     position: relative; | ||||
|     height: 80vh; | ||||
| } | ||||
|  | ||||
| .slideshow { | ||||
|     width: 100%; | ||||
|     overflow: hidden; | ||||
|     border-radius: 0; | ||||
|     max-height: 500px; | ||||
| } | ||||
|  | ||||
| .slideshow .slides { | ||||
|     display: flex; | ||||
|     transition: transform 1s ease-in-out; | ||||
|     opacity: 0.6; | ||||
|     filter: blur(2px); | ||||
|     pointer-events: none; | ||||
|     user-select: none; | ||||
| } | ||||
|  | ||||
| .slideshow .slide { | ||||
|     min-width: 100%; | ||||
|     box-sizing: border-box; | ||||
| } | ||||
|  | ||||
| .slideshow .slide img { | ||||
|     width: 100%; | ||||
|     display: block; | ||||
| } | ||||
|  | ||||
| .slideshow .dots{ | ||||
|     text-align: center; | ||||
|     position: absolute; | ||||
|     bottom: 15px; | ||||
|     width: 100%; | ||||
| } | ||||
|  | ||||
| .slideshow .dot { | ||||
|     display: inline-block; | ||||
|     width: 10px; | ||||
|     height: 10px; | ||||
|     margin: 0 5px; | ||||
|     background-color: #bebebe; | ||||
|     border-radius: 50%; | ||||
|     cursor: pointer; | ||||
|     transition: background-color 0.6s ease; | ||||
| } | ||||
|  | ||||
| .dot.active { | ||||
|     background-color: #ffffff; | ||||
| } | ||||
|  | ||||
| #slideshowBanner .title{ | ||||
|     display: inline-block; | ||||
|     width: 100%; | ||||
|     max-width: 500px; | ||||
|     text-align: left; | ||||
|     position: absolute; | ||||
|     top: 50%; | ||||
|     margin-left: 10%; | ||||
|     transform: translateX(0%) translateY(-50%); | ||||
|     color: white; | ||||
| } | ||||
|  | ||||
| #slideshowBanner .title .scrolldownTips{ | ||||
|     display: none; | ||||
| } | ||||
|  | ||||
| #slideshowBanner .title h1{ | ||||
|     font-size: 4em; | ||||
|     font-weight: 600; | ||||
|     margin-bottom: 0; | ||||
| } | ||||
|  | ||||
| #slideshowBanner .title p{ | ||||
|     font-size: 1.2em; | ||||
| } | ||||
|  | ||||
| /* About Zoraxy */ | ||||
| .about-text-wrapper{ | ||||
|     margin-top: 3em; | ||||
| } | ||||
| .about-text-wrapper p, .about-text-wrapper .list .item{ | ||||
|     font-weight: 300; | ||||
| } | ||||
| .about-title{ | ||||
|     font-size: 2.4em; | ||||
|     font-weight: 300; | ||||
|     margin-bottom: 0em; | ||||
| } | ||||
| .about-title b{ | ||||
|     font-weight: 800; | ||||
| } | ||||
| .about-text-wrapper .ui.list .item{ | ||||
|     margin-bottom: 0.6em; | ||||
| } | ||||
| .about-text-wrapper .ui.list .item .icon{ | ||||
|     padding-top: 0.15em; | ||||
| } | ||||
|  | ||||
| /* Screenshots */ | ||||
| #features{ | ||||
|     margin-bottom: 3em; | ||||
| } | ||||
|  | ||||
| #features .screenshot{ | ||||
|     transition: opacity 0.1s ease-in-out; | ||||
|     cursor: pointer; | ||||
| } | ||||
|  | ||||
| #features .screenshot:hover{ | ||||
|     opacity: 0.5; | ||||
| } | ||||
|  | ||||
| /* Videos */ | ||||
| #techspec .centered.title{ | ||||
|     color: white; | ||||
| } | ||||
|  | ||||
| #techspec p { | ||||
|     color: white; | ||||
| } | ||||
|  | ||||
| #techspec .videoScrollBar{ | ||||
|     overflow-x: scroll;  | ||||
|     display: block;  | ||||
|     white-space: nowrap;  | ||||
|     scrollbar-color: #e7e7e7 rgba(0, 0, 0, 0.1); | ||||
|     padding-top: 2em; | ||||
|     padding-bottom: 3em; | ||||
| } | ||||
|  | ||||
| .introvideo{ | ||||
|     display: inline-block !important; | ||||
|      | ||||
| } | ||||
|  | ||||
| .blackbanner{ | ||||
|     width: 100%; | ||||
|     background: rgb(108,172,255); | ||||
|     background: linear-gradient(48deg, rgba(108,172,255,1) 8%, rgba(141,235,255,1) 65%);  | ||||
|     min-height: 300px; | ||||
|  | ||||
| } | ||||
|  | ||||
| /* Download */ | ||||
| .downloadButton { | ||||
|     margin-top: 0.4em !important; | ||||
| } | ||||
|  | ||||
| .downloadTabWrapper{ | ||||
|     width: 100%; | ||||
|     overflow-x: hidden; | ||||
| } | ||||
|  | ||||
| #download .ui.black.message{ | ||||
|     word-break: break-all; | ||||
| } | ||||
|  | ||||
| /* Learn More */ | ||||
| #learnmore .linkicons{ | ||||
|     text-align: center; | ||||
|     width: 100%; | ||||
| } | ||||
|  | ||||
| #learnmore .linkicons .divider{ | ||||
|     margin-left: 1em; | ||||
|     margin-right: 1em; | ||||
| } | ||||
|  | ||||
| #learnmore .linkicons .externallink{ | ||||
|     margin-bottom: 0.6em; | ||||
|     transition: opacity 0.1s ease-in-out; | ||||
| } | ||||
|  | ||||
| #learnmore .linkicons .externallink i{ | ||||
|     /* color: #1b1c1d; */ | ||||
|     font-weight: 300; | ||||
|     font-size: 1.5em; | ||||
| } | ||||
|  | ||||
| #learnmore .linkicons .externallink:hover{ | ||||
|     opacity: 0.8; | ||||
| } | ||||
|  | ||||
|  | ||||
| #learnmore .linkicons .externallink .content{ | ||||
|     color: #1b1c1d; | ||||
|     font-weight: 500; | ||||
|     font-size: 0.6em; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* Footer */ | ||||
| #footer{ | ||||
|     background: rgb(85,131,238); | ||||
|     background: linear-gradient(48deg, rgba(85,131,238,1) 21%, rgba(108,172,255,1) 73%); | ||||
|     color: rgb(255, 255, 255); | ||||
| } | ||||
|  | ||||
| #footer a { | ||||
|     color: rgb(209, 224, 255); | ||||
| } | ||||
|  | ||||
| #footer a:hover{ | ||||
|     color: rgb(255, 255, 255); | ||||
| } | ||||
|  | ||||
| #footer .bottom-attach .divider{ | ||||
|     color: rgb(212, 212, 212); | ||||
| } | ||||
|  | ||||
| #footer .ui.list .title{ | ||||
|     margin-bottom: 0.6em; | ||||
| } | ||||
|  | ||||
| /* RWD Rules */ | ||||
| @media (max-width:960px) { | ||||
|     /* Main menu */ | ||||
|     #mainmenu{ | ||||
|         display:none; | ||||
|         z-index: 99; | ||||
|         position: absolute; | ||||
|         top: 0; | ||||
|         left: 0; | ||||
|         width: 100%; | ||||
|         background: #fdfdfd !important; | ||||
|     } | ||||
|  | ||||
|     #rwdmenubtn{ | ||||
|         display: block; | ||||
|         position: absolute; | ||||
|         top: 0.4em; | ||||
|         right: 0.4em; | ||||
|         z-index: 100; | ||||
|     } | ||||
|  | ||||
|     /* Slideshows */ | ||||
|     .slideshow { | ||||
|         min-height: 100vh; | ||||
|     } | ||||
|  | ||||
|     .slideshow .slide{ | ||||
|         height: 100% !important; | ||||
|         min-width: none; | ||||
|     } | ||||
|  | ||||
|     .slideshow .slide img{ | ||||
|         height: 100%; | ||||
|         width: auto; | ||||
|     } | ||||
|  | ||||
|     #slideshowBanner .title{ | ||||
|         padding: 1em; | ||||
|         margin-left: 0; | ||||
|     } | ||||
|  | ||||
|     #slideshowBanner .title .scrolldownTips{ | ||||
|         margin-top: 2em; | ||||
|         display: block; | ||||
|     } | ||||
|  | ||||
|     #slideshowBanner .title .scrolldownTips img{ | ||||
|         left: 50%; | ||||
|         transform: translateX(-50%); | ||||
|     } | ||||
|  | ||||
|     #techspec .videoScrollBar{ | ||||
|         overflow-x: auto;  | ||||
|         display: block;   | ||||
|         scrollbar-color: #e7e7e7 rgba(0, 0, 0, 0.1); | ||||
|         padding-top: 2em; | ||||
|         padding-bottom: 3em; | ||||
|     } | ||||
|  | ||||
|     .introvideo { | ||||
|         display: block !important; | ||||
|         width: 100%; | ||||
|         margin-bottom: 1em; | ||||
|     } | ||||
|  | ||||
|     .introvideo iframe{ | ||||
|         width: 100%; | ||||
|     } | ||||
|  | ||||
|     #download .stackable.tabular.menu .active.item{ | ||||
|         background-color: rgb(243, 243, 243); | ||||
|         border-width: 0; | ||||
|         border-radius: 0.4em !important; | ||||
|     }    | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| /*  | ||||
| 	Waves CSS  | ||||
| */ | ||||
|  | ||||
| #wavesWrapper{ | ||||
| 	position: absolute; | ||||
| 	bottom: 0; | ||||
| 	width: 100%; | ||||
| 	left: 0; | ||||
| } | ||||
|  | ||||
| .waves { | ||||
| 	position:relative; | ||||
| 	width: 100%; | ||||
| 	height:15vh; | ||||
| 	margin-bottom:-7px; /*Fix for safari gap*/ | ||||
| 	min-height:100px; | ||||
| 	max-height:150px; | ||||
| } | ||||
|  | ||||
|  | ||||
| .parallax > use { | ||||
| 	animation: move-forever 25s cubic-bezier(.55,.5,.45,.5)     infinite; | ||||
| } | ||||
| .parallax > use:nth-child(1) { | ||||
| 	animation-delay: -8s; | ||||
| 	animation-duration: 28s; | ||||
| } | ||||
| .parallax > use:nth-child(2) { | ||||
| 	animation-delay: -12s; | ||||
| 	animation-duration: 40s; | ||||
| } | ||||
| .parallax > use:nth-child(3) { | ||||
| 	animation-delay: -16s; | ||||
| 	animation-duration: 52s; | ||||
| } | ||||
| .parallax > use:nth-child(4) { | ||||
| 	animation-delay: -20s; | ||||
| 	animation-duration: 80s; | ||||
| } | ||||
| @keyframes move-forever { | ||||
| 	0% { | ||||
| 		transform: translate3d(-90px,0,0); | ||||
| 	} | ||||
| 	100% {  | ||||
| 		transform: translate3d(85px,0,0); | ||||
| 	} | ||||
| } | ||||
| /*Shrinking for mobile*/ | ||||
| @media (max-width: 768px) { | ||||
| 	.waves { | ||||
| 		height:40px; | ||||
| 		min-height:40px; | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										84
									
								
								docs/main.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								docs/main.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,84 @@ | ||||
| /* | ||||
|     Localization | ||||
|  | ||||
|     To add more locales, add to the html file with // (translated text) | ||||
|     after each DOM elements with attr i18n | ||||
|  | ||||
|     And then add the language ISO key to the list below. | ||||
| */ | ||||
| let languages = ['en', 'zh', 'de']; | ||||
|  | ||||
|  | ||||
| //Bind language change dropdown events | ||||
| $(".dropdown").dropdown(); | ||||
| $("#language").on("change",function(){ | ||||
|    let newLang = $("#language").parent().dropdown("get value"); | ||||
|    i18n.changeLanguage(newLang); | ||||
|    $("body").attr("class", newLang); | ||||
| }); | ||||
|  | ||||
| //Initialize the i18n dom library | ||||
| var i18n = domI18n({ | ||||
|     selector: '[i18n]', | ||||
|     separator: ' // ', | ||||
|     languages: languages, | ||||
|     defaultLanguage: 'en' | ||||
| }); | ||||
|  | ||||
| $(document).ready(function(){ | ||||
|     let userLang = navigator.language || navigator.userLanguage; | ||||
|     console.log("User language: " + userLang); | ||||
|     userLang = userLang.split("-")[0]; | ||||
|     if (!languages.includes(userLang)) { | ||||
|         userLang = 'en'; | ||||
|     } | ||||
|     i18n.changeLanguage(userLang); | ||||
|     $("body").attr("class", userLang); | ||||
| }); | ||||
|  | ||||
|  | ||||
| /* Main Menu */ | ||||
| $("#rwdmenubtn").on("click", function(){ | ||||
|     $("#mainmenu").slideToggle("fast"); | ||||
| }) | ||||
|  | ||||
| //Handle resize  | ||||
| $(window).on("resize", function(){ | ||||
|     if (window.innerWidth > 960){ | ||||
|         $("#mainmenu").show(); | ||||
|     }else{ | ||||
|         $("#mainmenu").hide(); | ||||
|     } | ||||
| }) | ||||
|  | ||||
| /* | ||||
|     Download | ||||
| */ | ||||
|  | ||||
| $('.menu .item').tab(); | ||||
|  | ||||
| //Download webpack and binary at the same time | ||||
| function handleDownload(releasename){ | ||||
|     let binaryURL = "https://github.com/tobychui/zoraxy/releases/latest/download/" + releasename; | ||||
|     window.open(binaryURL); | ||||
| } | ||||
|  | ||||
| /* RWD */ | ||||
| window.addEventListener('scroll', function() { | ||||
|     var scrollPosition = window.scrollY || window.pageYOffset; | ||||
|     var windowHeight = window.innerHeight; | ||||
|     var hiddenDiv = document.querySelector('#backToTopBtn'); | ||||
|  | ||||
|     if (scrollPosition > windowHeight / 2) { | ||||
|     hiddenDiv.style.display = 'block'; | ||||
|     } else { | ||||
|     hiddenDiv.style.display = 'none'; | ||||
|     } | ||||
| }); | ||||
|  | ||||
|  | ||||
| function backToTop(){ | ||||
|     $('html, body').animate({scrollTop : 0},800, function(){ | ||||
|         window.location.hash = ""; | ||||
|     }); | ||||
| } | ||||
							
								
								
									
										32
									
								
								example/plugins/build_all.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								example/plugins/build_all.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| #!/bin/bash | ||||
| # This script builds all the plugins in the current directory | ||||
|  | ||||
| echo "Copying zoraxy_plugin to all mods" | ||||
| for dir in ./*; do | ||||
|     if [ -d "$dir" ]; then | ||||
|         cp -r ../mod/plugins/zoraxy_plugin "$dir/mod" | ||||
|     fi | ||||
| done | ||||
|  | ||||
|  | ||||
| # Iterate over all directories in the current directory | ||||
| echo "Running go mod tidy and go build for all directories" | ||||
| for dir in */; do | ||||
|     if [ -d "$dir" ]; then | ||||
|         echo "Processing directory: $dir" | ||||
|         cd "$dir" | ||||
|          | ||||
|         # Execute go mod tidy | ||||
|         echo "Running go mod tidy in $dir" | ||||
|         go mod tidy | ||||
|          | ||||
|         # Execute go build | ||||
|         echo "Running go build in $dir" | ||||
|         go build | ||||
|          | ||||
|         # Return to the parent directory | ||||
|         cd .. | ||||
|     fi | ||||
| done | ||||
|  | ||||
| echo "Build process completed for all directories." | ||||
							
								
								
									
										3
									
								
								example/plugins/debugger/go.mod
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								example/plugins/debugger/go.mod
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| module aroz.org/zoraxy/debugger | ||||
|  | ||||
| go 1.23.6 | ||||
							
								
								
									
										140
									
								
								example/plugins/debugger/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										140
									
								
								example/plugins/debugger/main.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,140 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"sort" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	plugin "aroz.org/zoraxy/debugger/mod/zoraxy_plugin" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	PLUGIN_ID              = "org.aroz.zoraxy.debugger" | ||||
| 	UI_PATH                = "/debug" | ||||
| 	STATIC_CAPTURE_INGRESS = "/s_capture" | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
| 	// Serve the plugin intro spect | ||||
| 	// This will print the plugin intro spect and exit if the -introspect flag is provided | ||||
| 	runtimeCfg, err := plugin.ServeAndRecvSpec(&plugin.IntroSpect{ | ||||
| 		ID:            "org.aroz.zoraxy.debugger", | ||||
| 		Name:          "Plugin Debugger", | ||||
| 		Author:        "aroz.org", | ||||
| 		AuthorContact: "https://aroz.org", | ||||
| 		Description:   "A debugger for Zoraxy <-> plugin communication pipeline", | ||||
| 		URL:           "https://zoraxy.aroz.org", | ||||
| 		Type:          plugin.PluginType_Router, | ||||
| 		VersionMajor:  1, | ||||
| 		VersionMinor:  0, | ||||
| 		VersionPatch:  0, | ||||
|  | ||||
| 		StaticCapturePaths: []plugin.StaticCaptureRule{ | ||||
| 			{ | ||||
| 				CapturePath: "/test_a", | ||||
| 			}, | ||||
| 			{ | ||||
| 				CapturePath: "/test_b", | ||||
| 			}, | ||||
| 		}, | ||||
| 		StaticCaptureIngress: "/s_capture", | ||||
|  | ||||
| 		DynamicCaptureSniff:   "/d_sniff", | ||||
| 		DynamicCaptureIngress: "/d_capture", | ||||
|  | ||||
| 		UIPath: UI_PATH, | ||||
|  | ||||
| 		/* | ||||
| 			SubscriptionPath: "/subept", | ||||
| 			SubscriptionsEvents: []plugin.SubscriptionEvent{ | ||||
| 		*/ | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		//Terminate or enter standalone mode here | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	// Setup the path router | ||||
| 	pathRouter := plugin.NewPathRouter() | ||||
| 	pathRouter.SetDebugPrintMode(true) | ||||
|  | ||||
| 	/* | ||||
| 		Static Routers | ||||
| 	*/ | ||||
| 	pathRouter.RegisterPathHandler("/test_a", http.HandlerFunc(HandleCaptureA)) | ||||
| 	pathRouter.RegisterPathHandler("/test_b", http.HandlerFunc(HandleCaptureB)) | ||||
| 	pathRouter.SetDefaultHandler(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		//In theory this should never be called | ||||
| 		//but just in case the request is not captured by the path handlers | ||||
| 		//this will be the fallback handler | ||||
| 		w.Header().Set("Content-Type", "text/html") | ||||
| 		w.Write([]byte("This request is captured by the default handler!<br>Request URI: " + r.URL.String())) | ||||
| 	})) | ||||
| 	pathRouter.RegisterStaticCaptureHandle(STATIC_CAPTURE_INGRESS, http.DefaultServeMux) | ||||
|  | ||||
| 	/* | ||||
| 		Dynamic Captures | ||||
| 	*/ | ||||
| 	pathRouter.RegisterDynamicSniffHandler("/d_sniff", http.DefaultServeMux, func(dsfr *plugin.DynamicSniffForwardRequest) plugin.SniffResult { | ||||
| 		//fmt.Println("Dynamic Capture Sniffed Request:") | ||||
| 		//fmt.Println("Request URI: " + dsfr.RequestURI) | ||||
|  | ||||
| 		//In this example, we want to capture all URI | ||||
| 		//that start with /test_ and forward it to the dynamic capture handler | ||||
| 		if strings.HasPrefix(dsfr.RequestURI, "/test_") { | ||||
| 			reqUUID := dsfr.GetRequestUUID() | ||||
| 			fmt.Println("Accepting request with UUID: " + reqUUID) | ||||
| 			return plugin.SniffResultAccpet | ||||
| 		} | ||||
|  | ||||
| 		return plugin.SniffResultSkip | ||||
| 	}) | ||||
| 	pathRouter.RegisterDynamicCaptureHandle("/d_capture", http.DefaultServeMux, func(w http.ResponseWriter, r *http.Request) { | ||||
| 		// This is the dynamic capture handler where it actually captures and handle the request | ||||
| 		w.WriteHeader(http.StatusOK) | ||||
| 		w.Write([]byte("Welcome to the dynamic capture handler!")) | ||||
|  | ||||
| 		// Print all the request info to the response writer | ||||
| 		w.Write([]byte("\n\nRequest Info:\n")) | ||||
| 		w.Write([]byte("Request URI: " + r.RequestURI + "\n")) | ||||
| 		w.Write([]byte("Request Method: " + r.Method + "\n")) | ||||
| 		w.Write([]byte("Request Headers:\n")) | ||||
| 		headers := make([]string, 0, len(r.Header)) | ||||
| 		for key := range r.Header { | ||||
| 			headers = append(headers, key) | ||||
| 		} | ||||
| 		sort.Strings(headers) | ||||
| 		for _, key := range headers { | ||||
| 			for _, value := range r.Header[key] { | ||||
| 				w.Write([]byte(fmt.Sprintf("%s: %s\n", key, value))) | ||||
| 			} | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	http.HandleFunc(UI_PATH+"/", RenderDebugUI) | ||||
| 	fmt.Println("Debugger started at http://127.0.0.1:" + strconv.Itoa(runtimeCfg.Port)) | ||||
| 	http.ListenAndServe("127.0.0.1:"+strconv.Itoa(runtimeCfg.Port), nil) | ||||
| } | ||||
|  | ||||
| // Handle the captured request | ||||
| func HandleCaptureA(w http.ResponseWriter, r *http.Request) { | ||||
| 	/*for key, values := range r.Header { | ||||
| 		for _, value := range values { | ||||
| 			fmt.Printf("%s: %s\n", key, value) | ||||
| 		} | ||||
| 	}*/ | ||||
| 	w.Header().Set("Content-Type", "text/html") | ||||
| 	w.Write([]byte("This request is captured by A handler!<br>Request URI: " + r.URL.String())) | ||||
| } | ||||
|  | ||||
| func HandleCaptureB(w http.ResponseWriter, r *http.Request) { | ||||
| 	/*for key, values := range r.Header { | ||||
| 		for _, value := range values { | ||||
| 			fmt.Printf("%s: %s\n", key, value) | ||||
| 		} | ||||
| 	}*/ | ||||
| 	w.Header().Set("Content-Type", "text/html") | ||||
| 	w.Write([]byte("This request is captured by the B handler!<br>Request URI: " + r.URL.String())) | ||||
| } | ||||
							
								
								
									
										19
									
								
								example/plugins/debugger/mod/zoraxy_plugin/README.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								example/plugins/debugger/mod/zoraxy_plugin/README.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| # Zoraxy Plugin | ||||
|  | ||||
| ## Overview | ||||
| This module serves as a template for building your own plugins for the Zoraxy Reverse Proxy. By copying this module to your plugin mod folder, you can create a new plugin with the necessary structure and components. | ||||
|  | ||||
| ## Instructions | ||||
|  | ||||
| 1. **Copy the Module:** | ||||
|     - Copy the entire `zoraxy_plugin` module to your plugin mod folder. | ||||
|  | ||||
| 2. **Include the Structure:** | ||||
|     - Ensure that you maintain the directory structure and file organization as provided in this module. | ||||
|  | ||||
| 3. **Modify as Needed:** | ||||
|     - Customize the copied module to implement the desired functionality for your plugin. | ||||
|  | ||||
| ## Directory Structure | ||||
|  zoraxy_plugin: Handle -introspect and -configuration process required for plugin loading and startup | ||||
|  embed_webserver: Handle embeded web server routing and injecting csrf token to your plugin served UI pages | ||||
							
								
								
									
										145
									
								
								example/plugins/debugger/mod/zoraxy_plugin/dev_webserver.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								example/plugins/debugger/mod/zoraxy_plugin/dev_webserver.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type PluginUiDebugRouter struct { | ||||
| 	PluginID         string //The ID of the plugin | ||||
| 	TargetDir        string //The directory where the UI files are stored | ||||
| 	HandlerPrefix    string //The prefix of the handler used to route this router, e.g. /ui | ||||
| 	EnableDebug      bool   //Enable debug mode | ||||
| 	terminateHandler func() //The handler to be called when the plugin is terminated | ||||
| } | ||||
|  | ||||
| // NewPluginFileSystemUIRouter creates a new PluginUiRouter with file system | ||||
| // The targetDir is the directory where the UI files are stored (e.g. ./www) | ||||
| // The handlerPrefix is the prefix of the handler used to route this router | ||||
| // The handlerPrefix should start with a slash (e.g. /ui) that matches the http.Handle path | ||||
| // All prefix should not end with a slash | ||||
| func NewPluginFileSystemUIRouter(pluginID string, targetDir string, handlerPrefix string) *PluginUiDebugRouter { | ||||
| 	//Make sure all prefix are in /prefix format | ||||
| 	if !strings.HasPrefix(handlerPrefix, "/") { | ||||
| 		handlerPrefix = "/" + handlerPrefix | ||||
| 	} | ||||
| 	handlerPrefix = strings.TrimSuffix(handlerPrefix, "/") | ||||
|  | ||||
| 	//Return the PluginUiRouter | ||||
| 	return &PluginUiDebugRouter{ | ||||
| 		PluginID:      pluginID, | ||||
| 		TargetDir:     targetDir, | ||||
| 		HandlerPrefix: handlerPrefix, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *PluginUiDebugRouter) populateCSRFToken(r *http.Request, fsHandler http.Handler) http.Handler { | ||||
| 	//Get the CSRF token from header | ||||
| 	csrfToken := r.Header.Get("X-Zoraxy-Csrf") | ||||
| 	if csrfToken == "" { | ||||
| 		csrfToken = "missing-csrf-token" | ||||
| 	} | ||||
|  | ||||
| 	//Return the middleware | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		// Check if the request is for an HTML file | ||||
| 		if strings.HasSuffix(r.URL.Path, ".html") { | ||||
| 			//Read the target file from file system | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetDir + "/" + targetFilePath | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			targetFileContent, err := os.ReadFile(targetFilePath) | ||||
| 			if err != nil { | ||||
| 				http.Error(w, "File not found", http.StatusNotFound) | ||||
| 				return | ||||
| 			} | ||||
| 			body := string(targetFileContent) | ||||
| 			body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 			w.Header().Set("Content-Type", "text/html") | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte(body)) | ||||
| 			return | ||||
| 		} else if strings.HasSuffix(r.URL.Path, "/") { | ||||
| 			//Check if the request is for a directory | ||||
| 			//Check if the directory has an index.html file | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetDir + "/" + targetFilePath + "index.html" | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			if _, err := os.Stat(targetFilePath); err == nil { | ||||
| 				//Serve the index.html file | ||||
| 				targetFileContent, err := os.ReadFile(targetFilePath) | ||||
| 				if err != nil { | ||||
| 					http.Error(w, "File not found", http.StatusNotFound) | ||||
| 					return | ||||
| 				} | ||||
| 				body := string(targetFileContent) | ||||
| 				body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 				w.Header().Set("Content-Type", "text/html") | ||||
| 				w.WriteHeader(http.StatusOK) | ||||
| 				w.Write([]byte(body)) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Call the next handler | ||||
| 		fsHandler.ServeHTTP(w, r) | ||||
| 	}) | ||||
|  | ||||
| } | ||||
|  | ||||
| // GetHttpHandler returns the http.Handler for the PluginUiRouter | ||||
| func (p *PluginUiDebugRouter) Handler() http.Handler { | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		//Remove the plugin UI handler path prefix | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Print("Request URL:", r.URL.Path, " rewriting to ") | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, p.HandlerPrefix) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		r.URL.Path = rewrittenURL | ||||
| 		r.RequestURI = rewrittenURL | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Println(r.URL.Path) | ||||
| 		} | ||||
|  | ||||
| 		//Serve the file from the file system | ||||
| 		fsHandler := http.FileServer(http.Dir(p.TargetDir)) | ||||
|  | ||||
| 		// Replace {{csrf_token}} with the actual CSRF token and serve the file | ||||
| 		p.populateCSRFToken(r, fsHandler).ServeHTTP(w, r) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // RegisterTerminateHandler registers the terminate handler for the PluginUiRouter | ||||
| // The terminate handler will be called when the plugin is terminated from Zoraxy plugin manager | ||||
| // if mux is nil, the handler will be registered to http.DefaultServeMux | ||||
| func (p *PluginUiDebugRouter) RegisterTerminateHandler(termFunc func(), mux *http.ServeMux) { | ||||
| 	p.terminateHandler = termFunc | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
| 	mux.HandleFunc(p.HandlerPrefix+"/term", func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.terminateHandler() | ||||
| 		w.WriteHeader(http.StatusOK) | ||||
| 		go func() { | ||||
| 			//Make sure the response is sent before the plugin is terminated | ||||
| 			time.Sleep(100 * time.Millisecond) | ||||
| 			os.Exit(0) | ||||
| 		}() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // Attach the file system UI handler to the target http.ServeMux | ||||
| func (p *PluginUiDebugRouter) AttachHandlerToMux(mux *http.ServeMux) { | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
|  | ||||
| 	p.HandlerPrefix = strings.TrimSuffix(p.HandlerPrefix, "/") | ||||
| 	mux.Handle(p.HandlerPrefix+"/", p.Handler()) | ||||
| } | ||||
							
								
								
									
										162
									
								
								example/plugins/debugger/mod/zoraxy_plugin/dynamic_router.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										162
									
								
								example/plugins/debugger/mod/zoraxy_plugin/dynamic_router.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,162 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| /* | ||||
|  | ||||
| 	Dynamic Path Handler | ||||
|  | ||||
| */ | ||||
|  | ||||
| type SniffResult int | ||||
|  | ||||
| const ( | ||||
| 	SniffResultAccpet SniffResult = iota // Forward the request to this plugin dynamic capture ingress | ||||
| 	SniffResultSkip                      // Skip this plugin and let the next plugin handle the request | ||||
| ) | ||||
|  | ||||
| type SniffHandler func(*DynamicSniffForwardRequest) SniffResult | ||||
|  | ||||
| /* | ||||
| RegisterDynamicSniffHandler registers a dynamic sniff handler for a path | ||||
| You can decide to accept or skip the request based on the request header and paths | ||||
| */ | ||||
| func (p *PathRouter) RegisterDynamicSniffHandler(sniff_ingress string, mux *http.ServeMux, handler SniffHandler) { | ||||
| 	if !strings.HasSuffix(sniff_ingress, "/") { | ||||
| 		sniff_ingress = sniff_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(sniff_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Println("Request captured by dynamic sniff path: " + r.RequestURI) | ||||
| 		} | ||||
|  | ||||
| 		// Decode the request payload | ||||
| 		jsonBytes, err := io.ReadAll(r.Body) | ||||
| 		if err != nil { | ||||
| 			if p.enableDebugPrint { | ||||
| 				fmt.Println("Error reading request body:", err) | ||||
| 			} | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
| 		payload, err := DecodeForwardRequestPayload(jsonBytes) | ||||
| 		if err != nil { | ||||
| 			if p.enableDebugPrint { | ||||
| 				fmt.Println("Error decoding request payload:", err) | ||||
| 				fmt.Print("Payload: ") | ||||
| 				fmt.Println(string(jsonBytes)) | ||||
| 			} | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		// Get the forwarded request UUID | ||||
| 		forwardUUID := r.Header.Get("X-Zoraxy-RequestID") | ||||
| 		payload.requestUUID = forwardUUID | ||||
| 		payload.rawRequest = r | ||||
|  | ||||
| 		sniffResult := handler(&payload) | ||||
| 		if sniffResult == SniffResultAccpet { | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte("OK")) | ||||
| 		} else { | ||||
| 			w.WriteHeader(http.StatusNotImplemented) | ||||
| 			w.Write([]byte("SKIP")) | ||||
| 		} | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| // RegisterDynamicCaptureHandle register the dynamic capture ingress path with a handler | ||||
| func (p *PathRouter) RegisterDynamicCaptureHandle(capture_ingress string, mux *http.ServeMux, handlefunc func(http.ResponseWriter, *http.Request)) { | ||||
| 	if !strings.HasSuffix(capture_ingress, "/") { | ||||
| 		capture_ingress = capture_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(capture_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Println("Request captured by dynamic capture path: " + r.RequestURI) | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, capture_ingress) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		if rewrittenURL == "" { | ||||
| 			rewrittenURL = "/" | ||||
| 		} | ||||
| 		if !strings.HasPrefix(rewrittenURL, "/") { | ||||
| 			rewrittenURL = "/" + rewrittenURL | ||||
| 		} | ||||
| 		r.RequestURI = rewrittenURL | ||||
|  | ||||
| 		handlefunc(w, r) | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| /* | ||||
| 	Sniffing and forwarding | ||||
|  | ||||
| 	The following functions are here to help with | ||||
| 	sniffing and forwarding requests to the dynamic | ||||
| 	router. | ||||
| */ | ||||
| // A custom request object to be used in the dynamic sniffing | ||||
| type DynamicSniffForwardRequest struct { | ||||
| 	Method     string              `json:"method"` | ||||
| 	Hostname   string              `json:"hostname"` | ||||
| 	URL        string              `json:"url"` | ||||
| 	Header     map[string][]string `json:"header"` | ||||
| 	RemoteAddr string              `json:"remote_addr"` | ||||
| 	Host       string              `json:"host"` | ||||
| 	RequestURI string              `json:"request_uri"` | ||||
| 	Proto      string              `json:"proto"` | ||||
| 	ProtoMajor int                 `json:"proto_major"` | ||||
| 	ProtoMinor int                 `json:"proto_minor"` | ||||
|  | ||||
| 	/* Internal use */ | ||||
| 	rawRequest  *http.Request `json:"-"` | ||||
| 	requestUUID string        `json:"-"` | ||||
| } | ||||
|  | ||||
| // GetForwardRequestPayload returns a DynamicSniffForwardRequest object from an http.Request object | ||||
| func EncodeForwardRequestPayload(r *http.Request) DynamicSniffForwardRequest { | ||||
| 	return DynamicSniffForwardRequest{ | ||||
| 		Method:     r.Method, | ||||
| 		Hostname:   r.Host, | ||||
| 		URL:        r.URL.String(), | ||||
| 		Header:     r.Header, | ||||
| 		RemoteAddr: r.RemoteAddr, | ||||
| 		Host:       r.Host, | ||||
| 		RequestURI: r.RequestURI, | ||||
| 		Proto:      r.Proto, | ||||
| 		ProtoMajor: r.ProtoMajor, | ||||
| 		ProtoMinor: r.ProtoMinor, | ||||
| 		rawRequest: r, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // DecodeForwardRequestPayload decodes JSON bytes into a DynamicSniffForwardRequest object | ||||
| func DecodeForwardRequestPayload(jsonBytes []byte) (DynamicSniffForwardRequest, error) { | ||||
| 	var payload DynamicSniffForwardRequest | ||||
| 	err := json.Unmarshal(jsonBytes, &payload) | ||||
| 	if err != nil { | ||||
| 		return DynamicSniffForwardRequest{}, err | ||||
| 	} | ||||
| 	return payload, nil | ||||
| } | ||||
|  | ||||
| // GetRequest returns the original http.Request object, for debugging purposes | ||||
| func (dsfr *DynamicSniffForwardRequest) GetRequest() *http.Request { | ||||
| 	return dsfr.rawRequest | ||||
| } | ||||
|  | ||||
| // GetRequestUUID returns the request UUID | ||||
| // if this UUID is empty string, that might indicate the request | ||||
| // is not coming from the dynamic router | ||||
| func (dsfr *DynamicSniffForwardRequest) GetRequestUUID() string { | ||||
| 	return dsfr.requestUUID | ||||
| } | ||||
							
								
								
									
										156
									
								
								example/plugins/debugger/mod/zoraxy_plugin/embed_webserver.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								example/plugins/debugger/mod/zoraxy_plugin/embed_webserver.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	"fmt" | ||||
| 	"io/fs" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type PluginUiRouter struct { | ||||
| 	PluginID         string    //The ID of the plugin | ||||
| 	TargetFs         *embed.FS //The embed.FS where the UI files are stored | ||||
| 	TargetFsPrefix   string    //The prefix of the embed.FS where the UI files are stored, e.g. /web | ||||
| 	HandlerPrefix    string    //The prefix of the handler used to route this router, e.g. /ui | ||||
| 	EnableDebug      bool      //Enable debug mode | ||||
| 	terminateHandler func()    //The handler to be called when the plugin is terminated | ||||
| } | ||||
|  | ||||
| // NewPluginEmbedUIRouter creates a new PluginUiRouter with embed.FS | ||||
| // The targetFsPrefix is the prefix of the embed.FS where the UI files are stored | ||||
| // The targetFsPrefix should be relative to the root of the embed.FS | ||||
| // The targetFsPrefix should start with a slash (e.g. /web) that corresponds to the root folder of the embed.FS | ||||
| // The handlerPrefix is the prefix of the handler used to route this router | ||||
| // The handlerPrefix should start with a slash (e.g. /ui) that matches the http.Handle path | ||||
| // All prefix should not end with a slash | ||||
| func NewPluginEmbedUIRouter(pluginID string, targetFs *embed.FS, targetFsPrefix string, handlerPrefix string) *PluginUiRouter { | ||||
| 	//Make sure all prefix are in /prefix format | ||||
| 	if !strings.HasPrefix(targetFsPrefix, "/") { | ||||
| 		targetFsPrefix = "/" + targetFsPrefix | ||||
| 	} | ||||
| 	targetFsPrefix = strings.TrimSuffix(targetFsPrefix, "/") | ||||
|  | ||||
| 	if !strings.HasPrefix(handlerPrefix, "/") { | ||||
| 		handlerPrefix = "/" + handlerPrefix | ||||
| 	} | ||||
| 	handlerPrefix = strings.TrimSuffix(handlerPrefix, "/") | ||||
|  | ||||
| 	//Return the PluginUiRouter | ||||
| 	return &PluginUiRouter{ | ||||
| 		PluginID:       pluginID, | ||||
| 		TargetFs:       targetFs, | ||||
| 		TargetFsPrefix: targetFsPrefix, | ||||
| 		HandlerPrefix:  handlerPrefix, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *PluginUiRouter) populateCSRFToken(r *http.Request, fsHandler http.Handler) http.Handler { | ||||
| 	//Get the CSRF token from header | ||||
| 	csrfToken := r.Header.Get("X-Zoraxy-Csrf") | ||||
| 	if csrfToken == "" { | ||||
| 		csrfToken = "missing-csrf-token" | ||||
| 	} | ||||
|  | ||||
| 	//Return the middleware | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		// Check if the request is for an HTML file | ||||
| 		if strings.HasSuffix(r.URL.Path, ".html") { | ||||
| 			//Read the target file from embed.FS | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetFsPrefix + "/" + targetFilePath | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			targetFileContent, err := fs.ReadFile(*p.TargetFs, targetFilePath) | ||||
| 			if err != nil { | ||||
| 				http.Error(w, "File not found", http.StatusNotFound) | ||||
| 				return | ||||
| 			} | ||||
| 			body := string(targetFileContent) | ||||
| 			body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 			w.Header().Set("Content-Type", "text/html") | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte(body)) | ||||
| 			return | ||||
| 		} else if strings.HasSuffix(r.URL.Path, "/") { | ||||
| 			// Check if the directory has an index.html file | ||||
| 			indexFilePath := strings.TrimPrefix(r.URL.Path, "/") + "index.html" | ||||
| 			indexFilePath = p.TargetFsPrefix + "/" + indexFilePath | ||||
| 			indexFilePath = strings.TrimPrefix(indexFilePath, "/") | ||||
| 			indexFileContent, err := fs.ReadFile(*p.TargetFs, indexFilePath) | ||||
| 			if err == nil { | ||||
| 				body := string(indexFileContent) | ||||
| 				body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 				w.Header().Set("Content-Type", "text/html") | ||||
| 				w.WriteHeader(http.StatusOK) | ||||
| 				w.Write([]byte(body)) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Call the next handler | ||||
| 		fsHandler.ServeHTTP(w, r) | ||||
| 	}) | ||||
|  | ||||
| } | ||||
|  | ||||
| // GetHttpHandler returns the http.Handler for the PluginUiRouter | ||||
| func (p *PluginUiRouter) Handler() http.Handler { | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		//Remove the plugin UI handler path prefix | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Print("Request URL:", r.URL.Path, " rewriting to ") | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, p.HandlerPrefix) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		r.URL, _ = url.Parse(rewrittenURL) | ||||
| 		r.RequestURI = rewrittenURL | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Println(r.URL.Path) | ||||
| 		} | ||||
|  | ||||
| 		//Serve the file from the embed.FS | ||||
| 		subFS, err := fs.Sub(*p.TargetFs, strings.TrimPrefix(p.TargetFsPrefix, "/")) | ||||
| 		if err != nil { | ||||
| 			fmt.Println(err.Error()) | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		// Replace {{csrf_token}} with the actual CSRF token and serve the file | ||||
| 		p.populateCSRFToken(r, http.FileServer(http.FS(subFS))).ServeHTTP(w, r) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // RegisterTerminateHandler registers the terminate handler for the PluginUiRouter | ||||
| // The terminate handler will be called when the plugin is terminated from Zoraxy plugin manager | ||||
| // if mux is nil, the handler will be registered to http.DefaultServeMux | ||||
| func (p *PluginUiRouter) RegisterTerminateHandler(termFunc func(), mux *http.ServeMux) { | ||||
| 	p.terminateHandler = termFunc | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
| 	mux.HandleFunc(p.HandlerPrefix+"/term", func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.terminateHandler() | ||||
| 		w.WriteHeader(http.StatusOK) | ||||
| 		go func() { | ||||
| 			//Make sure the response is sent before the plugin is terminated | ||||
| 			time.Sleep(100 * time.Millisecond) | ||||
| 			os.Exit(0) | ||||
| 		}() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // Attach the embed UI handler to the target http.ServeMux | ||||
| func (p *PluginUiRouter) AttachHandlerToMux(mux *http.ServeMux) { | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
|  | ||||
| 	p.HandlerPrefix = strings.TrimSuffix(p.HandlerPrefix, "/") | ||||
| 	mux.Handle(p.HandlerPrefix+"/", p.Handler()) | ||||
| } | ||||
							
								
								
									
										105
									
								
								example/plugins/debugger/mod/zoraxy_plugin/static_router.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								example/plugins/debugger/mod/zoraxy_plugin/static_router.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type PathRouter struct { | ||||
| 	enableDebugPrint bool | ||||
| 	pathHandlers     map[string]http.Handler | ||||
| 	defaultHandler   http.Handler | ||||
| } | ||||
|  | ||||
| // NewPathRouter creates a new PathRouter | ||||
| func NewPathRouter() *PathRouter { | ||||
| 	return &PathRouter{ | ||||
| 		enableDebugPrint: false, | ||||
| 		pathHandlers:     make(map[string]http.Handler), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // RegisterPathHandler registers a handler for a path | ||||
| func (p *PathRouter) RegisterPathHandler(path string, handler http.Handler) { | ||||
| 	path = strings.TrimSuffix(path, "/") | ||||
| 	p.pathHandlers[path] = handler | ||||
| } | ||||
|  | ||||
| // RemovePathHandler removes a handler for a path | ||||
| func (p *PathRouter) RemovePathHandler(path string) { | ||||
| 	delete(p.pathHandlers, path) | ||||
| } | ||||
|  | ||||
| // SetDefaultHandler sets the default handler for the router | ||||
| // This handler will be called if no path handler is found | ||||
| func (p *PathRouter) SetDefaultHandler(handler http.Handler) { | ||||
| 	p.defaultHandler = handler | ||||
| } | ||||
|  | ||||
| // SetDebugPrintMode sets the debug print mode | ||||
| func (p *PathRouter) SetDebugPrintMode(enable bool) { | ||||
| 	p.enableDebugPrint = enable | ||||
| } | ||||
|  | ||||
| // StartStaticCapture starts the static capture ingress | ||||
| func (p *PathRouter) RegisterStaticCaptureHandle(capture_ingress string, mux *http.ServeMux) { | ||||
| 	if !strings.HasSuffix(capture_ingress, "/") { | ||||
| 		capture_ingress = capture_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(capture_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.staticCaptureServeHTTP(w, r) | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| // staticCaptureServeHTTP serves the static capture path using user defined handler | ||||
| func (p *PathRouter) staticCaptureServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
| 	capturePath := r.Header.Get("X-Zoraxy-Capture") | ||||
| 	if capturePath != "" { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Printf("Using capture path: %s\n", capturePath) | ||||
| 		} | ||||
| 		originalURI := r.Header.Get("X-Zoraxy-Uri") | ||||
| 		r.URL.Path = originalURI | ||||
| 		if handler, ok := p.pathHandlers[capturePath]; ok { | ||||
| 			handler.ServeHTTP(w, r) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	p.defaultHandler.ServeHTTP(w, r) | ||||
| } | ||||
|  | ||||
| func (p *PathRouter) PrintRequestDebugMessage(r *http.Request) { | ||||
| 	if p.enableDebugPrint { | ||||
| 		fmt.Printf("Capture Request with path: %s \n\n**Request Headers** \n\n", r.URL.Path) | ||||
| 		keys := make([]string, 0, len(r.Header)) | ||||
| 		for key := range r.Header { | ||||
| 			keys = append(keys, key) | ||||
| 		} | ||||
| 		sort.Strings(keys) | ||||
| 		for _, key := range keys { | ||||
| 			for _, value := range r.Header[key] { | ||||
| 				fmt.Printf("%s: %s\n", key, value) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		fmt.Printf("\n\n**Request Details**\n\n") | ||||
| 		fmt.Printf("Method: %s\n", r.Method) | ||||
| 		fmt.Printf("URL: %s\n", r.URL.String()) | ||||
| 		fmt.Printf("Proto: %s\n", r.Proto) | ||||
| 		fmt.Printf("Host: %s\n", r.Host) | ||||
| 		fmt.Printf("RemoteAddr: %s\n", r.RemoteAddr) | ||||
| 		fmt.Printf("RequestURI: %s\n", r.RequestURI) | ||||
| 		fmt.Printf("ContentLength: %d\n", r.ContentLength) | ||||
| 		fmt.Printf("TransferEncoding: %v\n", r.TransferEncoding) | ||||
| 		fmt.Printf("Close: %v\n", r.Close) | ||||
| 		fmt.Printf("Form: %v\n", r.Form) | ||||
| 		fmt.Printf("PostForm: %v\n", r.PostForm) | ||||
| 		fmt.Printf("MultipartForm: %v\n", r.MultipartForm) | ||||
| 		fmt.Printf("Trailer: %v\n", r.Trailer) | ||||
| 		fmt.Printf("RemoteAddr: %s\n", r.RemoteAddr) | ||||
| 		fmt.Printf("RequestURI: %s\n", r.RequestURI) | ||||
|  | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										176
									
								
								example/plugins/debugger/mod/zoraxy_plugin/zoraxy_plugin.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								example/plugins/debugger/mod/zoraxy_plugin/zoraxy_plugin.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| /* | ||||
| 	Plugins Includes.go | ||||
|  | ||||
| 	This file is copied from Zoraxy source code | ||||
| 	You can always find the latest version under mod/plugins/includes.go | ||||
| 	Usually this file are backward compatible | ||||
| */ | ||||
|  | ||||
| type PluginType int | ||||
|  | ||||
| const ( | ||||
| 	PluginType_Router    PluginType = 0 //Router Plugin, used for handling / routing / forwarding traffic | ||||
| 	PluginType_Utilities PluginType = 1 //Utilities Plugin, used for utilities like Zerotier or Static Web Server that do not require interception with the dpcore | ||||
| ) | ||||
|  | ||||
| type StaticCaptureRule struct { | ||||
| 	CapturePath string `json:"capture_path"` | ||||
| 	//To be expanded | ||||
| } | ||||
|  | ||||
| type ControlStatusCode int | ||||
|  | ||||
| const ( | ||||
| 	ControlStatusCode_CAPTURED  ControlStatusCode = 280 //Traffic captured by plugin, ask Zoraxy not to process the traffic | ||||
| 	ControlStatusCode_UNHANDLED ControlStatusCode = 284 //Traffic not handled by plugin, ask Zoraxy to process the traffic | ||||
| 	ControlStatusCode_ERROR     ControlStatusCode = 580 //Error occurred while processing the traffic, ask Zoraxy to process the traffic and log the error | ||||
| ) | ||||
|  | ||||
| type SubscriptionEvent struct { | ||||
| 	EventName   string `json:"event_name"` | ||||
| 	EventSource string `json:"event_source"` | ||||
| 	Payload     string `json:"payload"` //Payload of the event, can be empty | ||||
| } | ||||
|  | ||||
| type RuntimeConstantValue struct { | ||||
| 	ZoraxyVersion    string `json:"zoraxy_version"` | ||||
| 	ZoraxyUUID       string `json:"zoraxy_uuid"` | ||||
| 	DevelopmentBuild bool   `json:"development_build"` //Whether the Zoraxy is a development build or not | ||||
| } | ||||
|  | ||||
| /* | ||||
| IntroSpect Payload | ||||
|  | ||||
| When the plugin is initialized with -introspect flag, | ||||
| the plugin shell return this payload as JSON and exit | ||||
| */ | ||||
| type IntroSpect struct { | ||||
| 	/* Plugin metadata */ | ||||
| 	ID            string     `json:"id"`             //Unique ID of your plugin, recommended using your own domain in reverse like com.yourdomain.pluginname | ||||
| 	Name          string     `json:"name"`           //Name of your plugin | ||||
| 	Author        string     `json:"author"`         //Author name of your plugin | ||||
| 	AuthorContact string     `json:"author_contact"` //Author contact of your plugin, like email | ||||
| 	Description   string     `json:"description"`    //Description of your plugin | ||||
| 	URL           string     `json:"url"`            //URL of your plugin | ||||
| 	Type          PluginType `json:"type"`           //Type of your plugin, Router(0) or Utilities(1) | ||||
| 	VersionMajor  int        `json:"version_major"`  //Major version of your plugin | ||||
| 	VersionMinor  int        `json:"version_minor"`  //Minor version of your plugin | ||||
| 	VersionPatch  int        `json:"version_patch"`  //Patch version of your plugin | ||||
|  | ||||
| 	/* | ||||
|  | ||||
| 		Endpoint Settings | ||||
|  | ||||
| 	*/ | ||||
|  | ||||
| 	/* | ||||
| 		Static Capture Settings | ||||
|  | ||||
| 		Once plugin is enabled these rules always applies to the enabled HTTP Proxy rule | ||||
| 		This is faster than dynamic capture, but less flexible | ||||
| 	*/ | ||||
| 	StaticCapturePaths   []StaticCaptureRule `json:"static_capture_paths"`   //Static capture paths of your plugin, see Zoraxy documentation for more details | ||||
| 	StaticCaptureIngress string              `json:"static_capture_ingress"` //Static capture ingress path of your plugin (e.g. /s_handler) | ||||
|  | ||||
| 	/* | ||||
| 		Dynamic Capture Settings | ||||
|  | ||||
| 		Once plugin is enabled, these rules will be captured and forward to plugin sniff | ||||
| 		if the plugin sniff returns 280, the traffic will be captured | ||||
| 		otherwise, the traffic will be forwarded to the next plugin | ||||
| 		This is slower than static capture, but more flexible | ||||
| 	*/ | ||||
| 	DynamicCaptureSniff   string `json:"dynamic_capture_sniff"`   //Dynamic capture sniff path of your plugin (e.g. /d_sniff) | ||||
| 	DynamicCaptureIngress string `json:"dynamic_capture_ingress"` //Dynamic capture ingress path of your plugin (e.g. /d_handler) | ||||
|  | ||||
| 	/* UI Path for your plugin */ | ||||
| 	UIPath string `json:"ui_path"` //UI path of your plugin (e.g. /ui), will proxy the whole subpath tree to Zoraxy Web UI as plugin UI | ||||
|  | ||||
| 	/* Subscriptions Settings */ | ||||
| 	SubscriptionPath    string            `json:"subscription_path"`    //Subscription event path of your plugin (e.g. /notifyme), a POST request with SubscriptionEvent as body will be sent to this path when the event is triggered | ||||
| 	SubscriptionsEvents map[string]string `json:"subscriptions_events"` //Subscriptions events of your plugin, see Zoraxy documentation for more details | ||||
| } | ||||
|  | ||||
| /* | ||||
| ServeIntroSpect Function | ||||
|  | ||||
| This function will check if the plugin is initialized with -introspect flag, | ||||
| if so, it will print the intro spect and exit | ||||
|  | ||||
| Place this function at the beginning of your plugin main function | ||||
| */ | ||||
| func ServeIntroSpect(pluginSpect *IntroSpect) { | ||||
| 	if len(os.Args) > 1 && os.Args[1] == "-introspect" { | ||||
| 		//Print the intro spect and exit | ||||
| 		jsonData, _ := json.MarshalIndent(pluginSpect, "", " ") | ||||
| 		fmt.Println(string(jsonData)) | ||||
| 		os.Exit(0) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* | ||||
| ConfigureSpec Payload | ||||
|  | ||||
| Zoraxy will start your plugin with -configure flag, | ||||
| the plugin shell read this payload as JSON and configure itself | ||||
| by the supplied values like starting a web server at given port | ||||
| that listens to 127.0.0.1:port | ||||
| */ | ||||
| type ConfigureSpec struct { | ||||
| 	Port         int                  `json:"port"`          //Port to listen | ||||
| 	RuntimeConst RuntimeConstantValue `json:"runtime_const"` //Runtime constant values | ||||
| 	//To be expanded | ||||
| } | ||||
|  | ||||
| /* | ||||
| RecvExecuteConfigureSpec Function | ||||
|  | ||||
| This function will read the configure spec from Zoraxy | ||||
| and return the ConfigureSpec object | ||||
|  | ||||
| Place this function after ServeIntroSpect function in your plugin main function | ||||
| */ | ||||
| func RecvConfigureSpec() (*ConfigureSpec, error) { | ||||
| 	for i, arg := range os.Args { | ||||
| 		if strings.HasPrefix(arg, "-configure=") { | ||||
| 			var configSpec ConfigureSpec | ||||
| 			if err := json.Unmarshal([]byte(arg[11:]), &configSpec); err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			return &configSpec, nil | ||||
| 		} else if arg == "-configure" { | ||||
| 			var configSpec ConfigureSpec | ||||
| 			var nextArg string | ||||
| 			if len(os.Args) > i+1 { | ||||
| 				nextArg = os.Args[i+1] | ||||
| 				if err := json.Unmarshal([]byte(nextArg), &configSpec); err != nil { | ||||
| 					return nil, err | ||||
| 				} | ||||
| 			} else { | ||||
| 				return nil, fmt.Errorf("No port specified after -configure flag") | ||||
| 			} | ||||
| 			return &configSpec, nil | ||||
| 		} | ||||
| 	} | ||||
| 	return nil, fmt.Errorf("No -configure flag found") | ||||
| } | ||||
|  | ||||
| /* | ||||
| ServeAndRecvSpec Function | ||||
|  | ||||
| This function will serve the intro spect and return the configure spec | ||||
| See the ServeIntroSpect and RecvConfigureSpec for more details | ||||
| */ | ||||
| func ServeAndRecvSpec(pluginSpect *IntroSpect) (*ConfigureSpec, error) { | ||||
| 	ServeIntroSpect(pluginSpect) | ||||
| 	return RecvConfigureSpec() | ||||
| } | ||||
							
								
								
									
										26
									
								
								example/plugins/debugger/ui_info.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								example/plugins/debugger/ui_info.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	_ "embed" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"sort" | ||||
| ) | ||||
|  | ||||
| // Render the debug UI | ||||
| func RenderDebugUI(w http.ResponseWriter, r *http.Request) { | ||||
| 	fmt.Fprint(w, "**Plugin UI Debug Interface**\n\n[Recv Headers] \n") | ||||
|  | ||||
| 	headerKeys := make([]string, 0, len(r.Header)) | ||||
| 	for name := range r.Header { | ||||
| 		headerKeys = append(headerKeys, name) | ||||
| 	} | ||||
| 	sort.Strings(headerKeys) | ||||
| 	for _, name := range headerKeys { | ||||
| 		values := r.Header[name] | ||||
| 		for _, value := range values { | ||||
| 			fmt.Fprintf(w, "%s: %s\n", name, value) | ||||
| 		} | ||||
| 	} | ||||
| 	w.Header().Set("Content-Type", "text/html") | ||||
| } | ||||
							
								
								
									
										3
									
								
								example/plugins/helloworld/go.mod
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								example/plugins/helloworld/go.mod
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| module example.com/zoraxy/helloworld | ||||
|  | ||||
| go 1.23.6 | ||||
							
								
								
									
										
											BIN
										
									
								
								example/plugins/helloworld/icon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								example/plugins/helloworld/icon.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 14 KiB | 
							
								
								
									
										63
									
								
								example/plugins/helloworld/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								example/plugins/helloworld/main.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	_ "embed" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"strconv" | ||||
|  | ||||
| 	plugin "example.com/zoraxy/helloworld/mod/zoraxy_plugin" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	PLUGIN_ID = "com.example.helloworld" | ||||
| 	UI_PATH   = "/" | ||||
| 	WEB_ROOT  = "/www" | ||||
| ) | ||||
|  | ||||
| //go:embed www/* | ||||
| var content embed.FS | ||||
|  | ||||
| func main() { | ||||
| 	// Serve the plugin intro spect | ||||
| 	// This will print the plugin intro spect and exit if the -introspect flag is provided | ||||
| 	runtimeCfg, err := plugin.ServeAndRecvSpec(&plugin.IntroSpect{ | ||||
| 		ID:            "com.example.helloworld", | ||||
| 		Name:          "Hello World Plugin", | ||||
| 		Author:        "foobar", | ||||
| 		AuthorContact: "admin@example.com", | ||||
| 		Description:   "A simple hello world plugin", | ||||
| 		URL:           "https://example.com", | ||||
| 		Type:          plugin.PluginType_Utilities, | ||||
| 		VersionMajor:  1, | ||||
| 		VersionMinor:  0, | ||||
| 		VersionPatch:  0, | ||||
|  | ||||
| 		// As this is a utility plugin, we don't need to capture any traffic | ||||
| 		// but only serve the UI, so we set the UI (relative to the plugin path) to "/" | ||||
| 		UIPath: UI_PATH, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		//Terminate or enter standalone mode here | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	// Create a new PluginEmbedUIRouter that will serve the UI from web folder | ||||
| 	// The router will also help to handle the termination of the plugin when | ||||
| 	// a user wants to stop the plugin via Zoraxy Web UI | ||||
| 	embedWebRouter := plugin.NewPluginEmbedUIRouter(PLUGIN_ID, &content, WEB_ROOT, UI_PATH) | ||||
| 	embedWebRouter.RegisterTerminateHandler(func() { | ||||
| 		// Do cleanup here if needed | ||||
| 		fmt.Println("Hello World Plugin Exited") | ||||
| 	}, nil) | ||||
|  | ||||
| 	// Serve the hello world page in the www folder | ||||
| 	http.Handle(UI_PATH, embedWebRouter.Handler()) | ||||
| 	fmt.Println("Hello World started at http://127.0.0.1:" + strconv.Itoa(runtimeCfg.Port)) | ||||
| 	err = http.ListenAndServe("127.0.0.1:"+strconv.Itoa(runtimeCfg.Port), nil) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										19
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/README.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/README.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| # Zoraxy Plugin | ||||
|  | ||||
| ## Overview | ||||
| This module serves as a template for building your own plugins for the Zoraxy Reverse Proxy. By copying this module to your plugin mod folder, you can create a new plugin with the necessary structure and components. | ||||
|  | ||||
| ## Instructions | ||||
|  | ||||
| 1. **Copy the Module:** | ||||
|     - Copy the entire `zoraxy_plugin` module to your plugin mod folder. | ||||
|  | ||||
| 2. **Include the Structure:** | ||||
|     - Ensure that you maintain the directory structure and file organization as provided in this module. | ||||
|  | ||||
| 3. **Modify as Needed:** | ||||
|     - Customize the copied module to implement the desired functionality for your plugin. | ||||
|  | ||||
| ## Directory Structure | ||||
|  zoraxy_plugin: Handle -introspect and -configuration process required for plugin loading and startup | ||||
|  embed_webserver: Handle embeded web server routing and injecting csrf token to your plugin served UI pages | ||||
							
								
								
									
										145
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/dev_webserver.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/dev_webserver.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type PluginUiDebugRouter struct { | ||||
| 	PluginID         string //The ID of the plugin | ||||
| 	TargetDir        string //The directory where the UI files are stored | ||||
| 	HandlerPrefix    string //The prefix of the handler used to route this router, e.g. /ui | ||||
| 	EnableDebug      bool   //Enable debug mode | ||||
| 	terminateHandler func() //The handler to be called when the plugin is terminated | ||||
| } | ||||
|  | ||||
| // NewPluginFileSystemUIRouter creates a new PluginUiRouter with file system | ||||
| // The targetDir is the directory where the UI files are stored (e.g. ./www) | ||||
| // The handlerPrefix is the prefix of the handler used to route this router | ||||
| // The handlerPrefix should start with a slash (e.g. /ui) that matches the http.Handle path | ||||
| // All prefix should not end with a slash | ||||
| func NewPluginFileSystemUIRouter(pluginID string, targetDir string, handlerPrefix string) *PluginUiDebugRouter { | ||||
| 	//Make sure all prefix are in /prefix format | ||||
| 	if !strings.HasPrefix(handlerPrefix, "/") { | ||||
| 		handlerPrefix = "/" + handlerPrefix | ||||
| 	} | ||||
| 	handlerPrefix = strings.TrimSuffix(handlerPrefix, "/") | ||||
|  | ||||
| 	//Return the PluginUiRouter | ||||
| 	return &PluginUiDebugRouter{ | ||||
| 		PluginID:      pluginID, | ||||
| 		TargetDir:     targetDir, | ||||
| 		HandlerPrefix: handlerPrefix, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *PluginUiDebugRouter) populateCSRFToken(r *http.Request, fsHandler http.Handler) http.Handler { | ||||
| 	//Get the CSRF token from header | ||||
| 	csrfToken := r.Header.Get("X-Zoraxy-Csrf") | ||||
| 	if csrfToken == "" { | ||||
| 		csrfToken = "missing-csrf-token" | ||||
| 	} | ||||
|  | ||||
| 	//Return the middleware | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		// Check if the request is for an HTML file | ||||
| 		if strings.HasSuffix(r.URL.Path, ".html") { | ||||
| 			//Read the target file from file system | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetDir + "/" + targetFilePath | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			targetFileContent, err := os.ReadFile(targetFilePath) | ||||
| 			if err != nil { | ||||
| 				http.Error(w, "File not found", http.StatusNotFound) | ||||
| 				return | ||||
| 			} | ||||
| 			body := string(targetFileContent) | ||||
| 			body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 			w.Header().Set("Content-Type", "text/html") | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte(body)) | ||||
| 			return | ||||
| 		} else if strings.HasSuffix(r.URL.Path, "/") { | ||||
| 			//Check if the request is for a directory | ||||
| 			//Check if the directory has an index.html file | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetDir + "/" + targetFilePath + "index.html" | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			if _, err := os.Stat(targetFilePath); err == nil { | ||||
| 				//Serve the index.html file | ||||
| 				targetFileContent, err := os.ReadFile(targetFilePath) | ||||
| 				if err != nil { | ||||
| 					http.Error(w, "File not found", http.StatusNotFound) | ||||
| 					return | ||||
| 				} | ||||
| 				body := string(targetFileContent) | ||||
| 				body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 				w.Header().Set("Content-Type", "text/html") | ||||
| 				w.WriteHeader(http.StatusOK) | ||||
| 				w.Write([]byte(body)) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Call the next handler | ||||
| 		fsHandler.ServeHTTP(w, r) | ||||
| 	}) | ||||
|  | ||||
| } | ||||
|  | ||||
| // GetHttpHandler returns the http.Handler for the PluginUiRouter | ||||
| func (p *PluginUiDebugRouter) Handler() http.Handler { | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		//Remove the plugin UI handler path prefix | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Print("Request URL:", r.URL.Path, " rewriting to ") | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, p.HandlerPrefix) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		r.URL.Path = rewrittenURL | ||||
| 		r.RequestURI = rewrittenURL | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Println(r.URL.Path) | ||||
| 		} | ||||
|  | ||||
| 		//Serve the file from the file system | ||||
| 		fsHandler := http.FileServer(http.Dir(p.TargetDir)) | ||||
|  | ||||
| 		// Replace {{csrf_token}} with the actual CSRF token and serve the file | ||||
| 		p.populateCSRFToken(r, fsHandler).ServeHTTP(w, r) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // RegisterTerminateHandler registers the terminate handler for the PluginUiRouter | ||||
| // The terminate handler will be called when the plugin is terminated from Zoraxy plugin manager | ||||
| // if mux is nil, the handler will be registered to http.DefaultServeMux | ||||
| func (p *PluginUiDebugRouter) RegisterTerminateHandler(termFunc func(), mux *http.ServeMux) { | ||||
| 	p.terminateHandler = termFunc | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
| 	mux.HandleFunc(p.HandlerPrefix+"/term", func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.terminateHandler() | ||||
| 		w.WriteHeader(http.StatusOK) | ||||
| 		go func() { | ||||
| 			//Make sure the response is sent before the plugin is terminated | ||||
| 			time.Sleep(100 * time.Millisecond) | ||||
| 			os.Exit(0) | ||||
| 		}() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // Attach the file system UI handler to the target http.ServeMux | ||||
| func (p *PluginUiDebugRouter) AttachHandlerToMux(mux *http.ServeMux) { | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
|  | ||||
| 	p.HandlerPrefix = strings.TrimSuffix(p.HandlerPrefix, "/") | ||||
| 	mux.Handle(p.HandlerPrefix+"/", p.Handler()) | ||||
| } | ||||
							
								
								
									
										162
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/dynamic_router.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										162
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/dynamic_router.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,162 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| /* | ||||
|  | ||||
| 	Dynamic Path Handler | ||||
|  | ||||
| */ | ||||
|  | ||||
| type SniffResult int | ||||
|  | ||||
| const ( | ||||
| 	SniffResultAccpet SniffResult = iota // Forward the request to this plugin dynamic capture ingress | ||||
| 	SniffResultSkip                      // Skip this plugin and let the next plugin handle the request | ||||
| ) | ||||
|  | ||||
| type SniffHandler func(*DynamicSniffForwardRequest) SniffResult | ||||
|  | ||||
| /* | ||||
| RegisterDynamicSniffHandler registers a dynamic sniff handler for a path | ||||
| You can decide to accept or skip the request based on the request header and paths | ||||
| */ | ||||
| func (p *PathRouter) RegisterDynamicSniffHandler(sniff_ingress string, mux *http.ServeMux, handler SniffHandler) { | ||||
| 	if !strings.HasSuffix(sniff_ingress, "/") { | ||||
| 		sniff_ingress = sniff_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(sniff_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Println("Request captured by dynamic sniff path: " + r.RequestURI) | ||||
| 		} | ||||
|  | ||||
| 		// Decode the request payload | ||||
| 		jsonBytes, err := io.ReadAll(r.Body) | ||||
| 		if err != nil { | ||||
| 			if p.enableDebugPrint { | ||||
| 				fmt.Println("Error reading request body:", err) | ||||
| 			} | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
| 		payload, err := DecodeForwardRequestPayload(jsonBytes) | ||||
| 		if err != nil { | ||||
| 			if p.enableDebugPrint { | ||||
| 				fmt.Println("Error decoding request payload:", err) | ||||
| 				fmt.Print("Payload: ") | ||||
| 				fmt.Println(string(jsonBytes)) | ||||
| 			} | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		// Get the forwarded request UUID | ||||
| 		forwardUUID := r.Header.Get("X-Zoraxy-RequestID") | ||||
| 		payload.requestUUID = forwardUUID | ||||
| 		payload.rawRequest = r | ||||
|  | ||||
| 		sniffResult := handler(&payload) | ||||
| 		if sniffResult == SniffResultAccpet { | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte("OK")) | ||||
| 		} else { | ||||
| 			w.WriteHeader(http.StatusNotImplemented) | ||||
| 			w.Write([]byte("SKIP")) | ||||
| 		} | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| // RegisterDynamicCaptureHandle register the dynamic capture ingress path with a handler | ||||
| func (p *PathRouter) RegisterDynamicCaptureHandle(capture_ingress string, mux *http.ServeMux, handlefunc func(http.ResponseWriter, *http.Request)) { | ||||
| 	if !strings.HasSuffix(capture_ingress, "/") { | ||||
| 		capture_ingress = capture_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(capture_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Println("Request captured by dynamic capture path: " + r.RequestURI) | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, capture_ingress) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		if rewrittenURL == "" { | ||||
| 			rewrittenURL = "/" | ||||
| 		} | ||||
| 		if !strings.HasPrefix(rewrittenURL, "/") { | ||||
| 			rewrittenURL = "/" + rewrittenURL | ||||
| 		} | ||||
| 		r.RequestURI = rewrittenURL | ||||
|  | ||||
| 		handlefunc(w, r) | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| /* | ||||
| 	Sniffing and forwarding | ||||
|  | ||||
| 	The following functions are here to help with | ||||
| 	sniffing and forwarding requests to the dynamic | ||||
| 	router. | ||||
| */ | ||||
| // A custom request object to be used in the dynamic sniffing | ||||
| type DynamicSniffForwardRequest struct { | ||||
| 	Method     string              `json:"method"` | ||||
| 	Hostname   string              `json:"hostname"` | ||||
| 	URL        string              `json:"url"` | ||||
| 	Header     map[string][]string `json:"header"` | ||||
| 	RemoteAddr string              `json:"remote_addr"` | ||||
| 	Host       string              `json:"host"` | ||||
| 	RequestURI string              `json:"request_uri"` | ||||
| 	Proto      string              `json:"proto"` | ||||
| 	ProtoMajor int                 `json:"proto_major"` | ||||
| 	ProtoMinor int                 `json:"proto_minor"` | ||||
|  | ||||
| 	/* Internal use */ | ||||
| 	rawRequest  *http.Request `json:"-"` | ||||
| 	requestUUID string        `json:"-"` | ||||
| } | ||||
|  | ||||
| // GetForwardRequestPayload returns a DynamicSniffForwardRequest object from an http.Request object | ||||
| func EncodeForwardRequestPayload(r *http.Request) DynamicSniffForwardRequest { | ||||
| 	return DynamicSniffForwardRequest{ | ||||
| 		Method:     r.Method, | ||||
| 		Hostname:   r.Host, | ||||
| 		URL:        r.URL.String(), | ||||
| 		Header:     r.Header, | ||||
| 		RemoteAddr: r.RemoteAddr, | ||||
| 		Host:       r.Host, | ||||
| 		RequestURI: r.RequestURI, | ||||
| 		Proto:      r.Proto, | ||||
| 		ProtoMajor: r.ProtoMajor, | ||||
| 		ProtoMinor: r.ProtoMinor, | ||||
| 		rawRequest: r, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // DecodeForwardRequestPayload decodes JSON bytes into a DynamicSniffForwardRequest object | ||||
| func DecodeForwardRequestPayload(jsonBytes []byte) (DynamicSniffForwardRequest, error) { | ||||
| 	var payload DynamicSniffForwardRequest | ||||
| 	err := json.Unmarshal(jsonBytes, &payload) | ||||
| 	if err != nil { | ||||
| 		return DynamicSniffForwardRequest{}, err | ||||
| 	} | ||||
| 	return payload, nil | ||||
| } | ||||
|  | ||||
| // GetRequest returns the original http.Request object, for debugging purposes | ||||
| func (dsfr *DynamicSniffForwardRequest) GetRequest() *http.Request { | ||||
| 	return dsfr.rawRequest | ||||
| } | ||||
|  | ||||
| // GetRequestUUID returns the request UUID | ||||
| // if this UUID is empty string, that might indicate the request | ||||
| // is not coming from the dynamic router | ||||
| func (dsfr *DynamicSniffForwardRequest) GetRequestUUID() string { | ||||
| 	return dsfr.requestUUID | ||||
| } | ||||
							
								
								
									
										156
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/embed_webserver.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/embed_webserver.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	"fmt" | ||||
| 	"io/fs" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type PluginUiRouter struct { | ||||
| 	PluginID         string    //The ID of the plugin | ||||
| 	TargetFs         *embed.FS //The embed.FS where the UI files are stored | ||||
| 	TargetFsPrefix   string    //The prefix of the embed.FS where the UI files are stored, e.g. /web | ||||
| 	HandlerPrefix    string    //The prefix of the handler used to route this router, e.g. /ui | ||||
| 	EnableDebug      bool      //Enable debug mode | ||||
| 	terminateHandler func()    //The handler to be called when the plugin is terminated | ||||
| } | ||||
|  | ||||
| // NewPluginEmbedUIRouter creates a new PluginUiRouter with embed.FS | ||||
| // The targetFsPrefix is the prefix of the embed.FS where the UI files are stored | ||||
| // The targetFsPrefix should be relative to the root of the embed.FS | ||||
| // The targetFsPrefix should start with a slash (e.g. /web) that corresponds to the root folder of the embed.FS | ||||
| // The handlerPrefix is the prefix of the handler used to route this router | ||||
| // The handlerPrefix should start with a slash (e.g. /ui) that matches the http.Handle path | ||||
| // All prefix should not end with a slash | ||||
| func NewPluginEmbedUIRouter(pluginID string, targetFs *embed.FS, targetFsPrefix string, handlerPrefix string) *PluginUiRouter { | ||||
| 	//Make sure all prefix are in /prefix format | ||||
| 	if !strings.HasPrefix(targetFsPrefix, "/") { | ||||
| 		targetFsPrefix = "/" + targetFsPrefix | ||||
| 	} | ||||
| 	targetFsPrefix = strings.TrimSuffix(targetFsPrefix, "/") | ||||
|  | ||||
| 	if !strings.HasPrefix(handlerPrefix, "/") { | ||||
| 		handlerPrefix = "/" + handlerPrefix | ||||
| 	} | ||||
| 	handlerPrefix = strings.TrimSuffix(handlerPrefix, "/") | ||||
|  | ||||
| 	//Return the PluginUiRouter | ||||
| 	return &PluginUiRouter{ | ||||
| 		PluginID:       pluginID, | ||||
| 		TargetFs:       targetFs, | ||||
| 		TargetFsPrefix: targetFsPrefix, | ||||
| 		HandlerPrefix:  handlerPrefix, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *PluginUiRouter) populateCSRFToken(r *http.Request, fsHandler http.Handler) http.Handler { | ||||
| 	//Get the CSRF token from header | ||||
| 	csrfToken := r.Header.Get("X-Zoraxy-Csrf") | ||||
| 	if csrfToken == "" { | ||||
| 		csrfToken = "missing-csrf-token" | ||||
| 	} | ||||
|  | ||||
| 	//Return the middleware | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		// Check if the request is for an HTML file | ||||
| 		if strings.HasSuffix(r.URL.Path, ".html") { | ||||
| 			//Read the target file from embed.FS | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetFsPrefix + "/" + targetFilePath | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			targetFileContent, err := fs.ReadFile(*p.TargetFs, targetFilePath) | ||||
| 			if err != nil { | ||||
| 				http.Error(w, "File not found", http.StatusNotFound) | ||||
| 				return | ||||
| 			} | ||||
| 			body := string(targetFileContent) | ||||
| 			body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 			w.Header().Set("Content-Type", "text/html") | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte(body)) | ||||
| 			return | ||||
| 		} else if strings.HasSuffix(r.URL.Path, "/") { | ||||
| 			// Check if the directory has an index.html file | ||||
| 			indexFilePath := strings.TrimPrefix(r.URL.Path, "/") + "index.html" | ||||
| 			indexFilePath = p.TargetFsPrefix + "/" + indexFilePath | ||||
| 			indexFilePath = strings.TrimPrefix(indexFilePath, "/") | ||||
| 			indexFileContent, err := fs.ReadFile(*p.TargetFs, indexFilePath) | ||||
| 			if err == nil { | ||||
| 				body := string(indexFileContent) | ||||
| 				body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 				w.Header().Set("Content-Type", "text/html") | ||||
| 				w.WriteHeader(http.StatusOK) | ||||
| 				w.Write([]byte(body)) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Call the next handler | ||||
| 		fsHandler.ServeHTTP(w, r) | ||||
| 	}) | ||||
|  | ||||
| } | ||||
|  | ||||
| // GetHttpHandler returns the http.Handler for the PluginUiRouter | ||||
| func (p *PluginUiRouter) Handler() http.Handler { | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		//Remove the plugin UI handler path prefix | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Print("Request URL:", r.URL.Path, " rewriting to ") | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, p.HandlerPrefix) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		r.URL, _ = url.Parse(rewrittenURL) | ||||
| 		r.RequestURI = rewrittenURL | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Println(r.URL.Path) | ||||
| 		} | ||||
|  | ||||
| 		//Serve the file from the embed.FS | ||||
| 		subFS, err := fs.Sub(*p.TargetFs, strings.TrimPrefix(p.TargetFsPrefix, "/")) | ||||
| 		if err != nil { | ||||
| 			fmt.Println(err.Error()) | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		// Replace {{csrf_token}} with the actual CSRF token and serve the file | ||||
| 		p.populateCSRFToken(r, http.FileServer(http.FS(subFS))).ServeHTTP(w, r) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // RegisterTerminateHandler registers the terminate handler for the PluginUiRouter | ||||
| // The terminate handler will be called when the plugin is terminated from Zoraxy plugin manager | ||||
| // if mux is nil, the handler will be registered to http.DefaultServeMux | ||||
| func (p *PluginUiRouter) RegisterTerminateHandler(termFunc func(), mux *http.ServeMux) { | ||||
| 	p.terminateHandler = termFunc | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
| 	mux.HandleFunc(p.HandlerPrefix+"/term", func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.terminateHandler() | ||||
| 		w.WriteHeader(http.StatusOK) | ||||
| 		go func() { | ||||
| 			//Make sure the response is sent before the plugin is terminated | ||||
| 			time.Sleep(100 * time.Millisecond) | ||||
| 			os.Exit(0) | ||||
| 		}() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // Attach the embed UI handler to the target http.ServeMux | ||||
| func (p *PluginUiRouter) AttachHandlerToMux(mux *http.ServeMux) { | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
|  | ||||
| 	p.HandlerPrefix = strings.TrimSuffix(p.HandlerPrefix, "/") | ||||
| 	mux.Handle(p.HandlerPrefix+"/", p.Handler()) | ||||
| } | ||||
							
								
								
									
										105
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/static_router.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/static_router.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type PathRouter struct { | ||||
| 	enableDebugPrint bool | ||||
| 	pathHandlers     map[string]http.Handler | ||||
| 	defaultHandler   http.Handler | ||||
| } | ||||
|  | ||||
| // NewPathRouter creates a new PathRouter | ||||
| func NewPathRouter() *PathRouter { | ||||
| 	return &PathRouter{ | ||||
| 		enableDebugPrint: false, | ||||
| 		pathHandlers:     make(map[string]http.Handler), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // RegisterPathHandler registers a handler for a path | ||||
| func (p *PathRouter) RegisterPathHandler(path string, handler http.Handler) { | ||||
| 	path = strings.TrimSuffix(path, "/") | ||||
| 	p.pathHandlers[path] = handler | ||||
| } | ||||
|  | ||||
| // RemovePathHandler removes a handler for a path | ||||
| func (p *PathRouter) RemovePathHandler(path string) { | ||||
| 	delete(p.pathHandlers, path) | ||||
| } | ||||
|  | ||||
| // SetDefaultHandler sets the default handler for the router | ||||
| // This handler will be called if no path handler is found | ||||
| func (p *PathRouter) SetDefaultHandler(handler http.Handler) { | ||||
| 	p.defaultHandler = handler | ||||
| } | ||||
|  | ||||
| // SetDebugPrintMode sets the debug print mode | ||||
| func (p *PathRouter) SetDebugPrintMode(enable bool) { | ||||
| 	p.enableDebugPrint = enable | ||||
| } | ||||
|  | ||||
| // StartStaticCapture starts the static capture ingress | ||||
| func (p *PathRouter) RegisterStaticCaptureHandle(capture_ingress string, mux *http.ServeMux) { | ||||
| 	if !strings.HasSuffix(capture_ingress, "/") { | ||||
| 		capture_ingress = capture_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(capture_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.staticCaptureServeHTTP(w, r) | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| // staticCaptureServeHTTP serves the static capture path using user defined handler | ||||
| func (p *PathRouter) staticCaptureServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
| 	capturePath := r.Header.Get("X-Zoraxy-Capture") | ||||
| 	if capturePath != "" { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Printf("Using capture path: %s\n", capturePath) | ||||
| 		} | ||||
| 		originalURI := r.Header.Get("X-Zoraxy-Uri") | ||||
| 		r.URL.Path = originalURI | ||||
| 		if handler, ok := p.pathHandlers[capturePath]; ok { | ||||
| 			handler.ServeHTTP(w, r) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	p.defaultHandler.ServeHTTP(w, r) | ||||
| } | ||||
|  | ||||
| func (p *PathRouter) PrintRequestDebugMessage(r *http.Request) { | ||||
| 	if p.enableDebugPrint { | ||||
| 		fmt.Printf("Capture Request with path: %s \n\n**Request Headers** \n\n", r.URL.Path) | ||||
| 		keys := make([]string, 0, len(r.Header)) | ||||
| 		for key := range r.Header { | ||||
| 			keys = append(keys, key) | ||||
| 		} | ||||
| 		sort.Strings(keys) | ||||
| 		for _, key := range keys { | ||||
| 			for _, value := range r.Header[key] { | ||||
| 				fmt.Printf("%s: %s\n", key, value) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		fmt.Printf("\n\n**Request Details**\n\n") | ||||
| 		fmt.Printf("Method: %s\n", r.Method) | ||||
| 		fmt.Printf("URL: %s\n", r.URL.String()) | ||||
| 		fmt.Printf("Proto: %s\n", r.Proto) | ||||
| 		fmt.Printf("Host: %s\n", r.Host) | ||||
| 		fmt.Printf("RemoteAddr: %s\n", r.RemoteAddr) | ||||
| 		fmt.Printf("RequestURI: %s\n", r.RequestURI) | ||||
| 		fmt.Printf("ContentLength: %d\n", r.ContentLength) | ||||
| 		fmt.Printf("TransferEncoding: %v\n", r.TransferEncoding) | ||||
| 		fmt.Printf("Close: %v\n", r.Close) | ||||
| 		fmt.Printf("Form: %v\n", r.Form) | ||||
| 		fmt.Printf("PostForm: %v\n", r.PostForm) | ||||
| 		fmt.Printf("MultipartForm: %v\n", r.MultipartForm) | ||||
| 		fmt.Printf("Trailer: %v\n", r.Trailer) | ||||
| 		fmt.Printf("RemoteAddr: %s\n", r.RemoteAddr) | ||||
| 		fmt.Printf("RequestURI: %s\n", r.RequestURI) | ||||
|  | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										176
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/zoraxy_plugin.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								example/plugins/helloworld/mod/zoraxy_plugin/zoraxy_plugin.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| /* | ||||
| 	Plugins Includes.go | ||||
|  | ||||
| 	This file is copied from Zoraxy source code | ||||
| 	You can always find the latest version under mod/plugins/includes.go | ||||
| 	Usually this file are backward compatible | ||||
| */ | ||||
|  | ||||
| type PluginType int | ||||
|  | ||||
| const ( | ||||
| 	PluginType_Router    PluginType = 0 //Router Plugin, used for handling / routing / forwarding traffic | ||||
| 	PluginType_Utilities PluginType = 1 //Utilities Plugin, used for utilities like Zerotier or Static Web Server that do not require interception with the dpcore | ||||
| ) | ||||
|  | ||||
| type StaticCaptureRule struct { | ||||
| 	CapturePath string `json:"capture_path"` | ||||
| 	//To be expanded | ||||
| } | ||||
|  | ||||
| type ControlStatusCode int | ||||
|  | ||||
| const ( | ||||
| 	ControlStatusCode_CAPTURED  ControlStatusCode = 280 //Traffic captured by plugin, ask Zoraxy not to process the traffic | ||||
| 	ControlStatusCode_UNHANDLED ControlStatusCode = 284 //Traffic not handled by plugin, ask Zoraxy to process the traffic | ||||
| 	ControlStatusCode_ERROR     ControlStatusCode = 580 //Error occurred while processing the traffic, ask Zoraxy to process the traffic and log the error | ||||
| ) | ||||
|  | ||||
| type SubscriptionEvent struct { | ||||
| 	EventName   string `json:"event_name"` | ||||
| 	EventSource string `json:"event_source"` | ||||
| 	Payload     string `json:"payload"` //Payload of the event, can be empty | ||||
| } | ||||
|  | ||||
| type RuntimeConstantValue struct { | ||||
| 	ZoraxyVersion    string `json:"zoraxy_version"` | ||||
| 	ZoraxyUUID       string `json:"zoraxy_uuid"` | ||||
| 	DevelopmentBuild bool   `json:"development_build"` //Whether the Zoraxy is a development build or not | ||||
| } | ||||
|  | ||||
| /* | ||||
| IntroSpect Payload | ||||
|  | ||||
| When the plugin is initialized with -introspect flag, | ||||
| the plugin shell return this payload as JSON and exit | ||||
| */ | ||||
| type IntroSpect struct { | ||||
| 	/* Plugin metadata */ | ||||
| 	ID            string     `json:"id"`             //Unique ID of your plugin, recommended using your own domain in reverse like com.yourdomain.pluginname | ||||
| 	Name          string     `json:"name"`           //Name of your plugin | ||||
| 	Author        string     `json:"author"`         //Author name of your plugin | ||||
| 	AuthorContact string     `json:"author_contact"` //Author contact of your plugin, like email | ||||
| 	Description   string     `json:"description"`    //Description of your plugin | ||||
| 	URL           string     `json:"url"`            //URL of your plugin | ||||
| 	Type          PluginType `json:"type"`           //Type of your plugin, Router(0) or Utilities(1) | ||||
| 	VersionMajor  int        `json:"version_major"`  //Major version of your plugin | ||||
| 	VersionMinor  int        `json:"version_minor"`  //Minor version of your plugin | ||||
| 	VersionPatch  int        `json:"version_patch"`  //Patch version of your plugin | ||||
|  | ||||
| 	/* | ||||
|  | ||||
| 		Endpoint Settings | ||||
|  | ||||
| 	*/ | ||||
|  | ||||
| 	/* | ||||
| 		Static Capture Settings | ||||
|  | ||||
| 		Once plugin is enabled these rules always applies to the enabled HTTP Proxy rule | ||||
| 		This is faster than dynamic capture, but less flexible | ||||
| 	*/ | ||||
| 	StaticCapturePaths   []StaticCaptureRule `json:"static_capture_paths"`   //Static capture paths of your plugin, see Zoraxy documentation for more details | ||||
| 	StaticCaptureIngress string              `json:"static_capture_ingress"` //Static capture ingress path of your plugin (e.g. /s_handler) | ||||
|  | ||||
| 	/* | ||||
| 		Dynamic Capture Settings | ||||
|  | ||||
| 		Once plugin is enabled, these rules will be captured and forward to plugin sniff | ||||
| 		if the plugin sniff returns 280, the traffic will be captured | ||||
| 		otherwise, the traffic will be forwarded to the next plugin | ||||
| 		This is slower than static capture, but more flexible | ||||
| 	*/ | ||||
| 	DynamicCaptureSniff   string `json:"dynamic_capture_sniff"`   //Dynamic capture sniff path of your plugin (e.g. /d_sniff) | ||||
| 	DynamicCaptureIngress string `json:"dynamic_capture_ingress"` //Dynamic capture ingress path of your plugin (e.g. /d_handler) | ||||
|  | ||||
| 	/* UI Path for your plugin */ | ||||
| 	UIPath string `json:"ui_path"` //UI path of your plugin (e.g. /ui), will proxy the whole subpath tree to Zoraxy Web UI as plugin UI | ||||
|  | ||||
| 	/* Subscriptions Settings */ | ||||
| 	SubscriptionPath    string            `json:"subscription_path"`    //Subscription event path of your plugin (e.g. /notifyme), a POST request with SubscriptionEvent as body will be sent to this path when the event is triggered | ||||
| 	SubscriptionsEvents map[string]string `json:"subscriptions_events"` //Subscriptions events of your plugin, see Zoraxy documentation for more details | ||||
| } | ||||
|  | ||||
| /* | ||||
| ServeIntroSpect Function | ||||
|  | ||||
| This function will check if the plugin is initialized with -introspect flag, | ||||
| if so, it will print the intro spect and exit | ||||
|  | ||||
| Place this function at the beginning of your plugin main function | ||||
| */ | ||||
| func ServeIntroSpect(pluginSpect *IntroSpect) { | ||||
| 	if len(os.Args) > 1 && os.Args[1] == "-introspect" { | ||||
| 		//Print the intro spect and exit | ||||
| 		jsonData, _ := json.MarshalIndent(pluginSpect, "", " ") | ||||
| 		fmt.Println(string(jsonData)) | ||||
| 		os.Exit(0) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* | ||||
| ConfigureSpec Payload | ||||
|  | ||||
| Zoraxy will start your plugin with -configure flag, | ||||
| the plugin shell read this payload as JSON and configure itself | ||||
| by the supplied values like starting a web server at given port | ||||
| that listens to 127.0.0.1:port | ||||
| */ | ||||
| type ConfigureSpec struct { | ||||
| 	Port         int                  `json:"port"`          //Port to listen | ||||
| 	RuntimeConst RuntimeConstantValue `json:"runtime_const"` //Runtime constant values | ||||
| 	//To be expanded | ||||
| } | ||||
|  | ||||
| /* | ||||
| RecvExecuteConfigureSpec Function | ||||
|  | ||||
| This function will read the configure spec from Zoraxy | ||||
| and return the ConfigureSpec object | ||||
|  | ||||
| Place this function after ServeIntroSpect function in your plugin main function | ||||
| */ | ||||
| func RecvConfigureSpec() (*ConfigureSpec, error) { | ||||
| 	for i, arg := range os.Args { | ||||
| 		if strings.HasPrefix(arg, "-configure=") { | ||||
| 			var configSpec ConfigureSpec | ||||
| 			if err := json.Unmarshal([]byte(arg[11:]), &configSpec); err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			return &configSpec, nil | ||||
| 		} else if arg == "-configure" { | ||||
| 			var configSpec ConfigureSpec | ||||
| 			var nextArg string | ||||
| 			if len(os.Args) > i+1 { | ||||
| 				nextArg = os.Args[i+1] | ||||
| 				if err := json.Unmarshal([]byte(nextArg), &configSpec); err != nil { | ||||
| 					return nil, err | ||||
| 				} | ||||
| 			} else { | ||||
| 				return nil, fmt.Errorf("No port specified after -configure flag") | ||||
| 			} | ||||
| 			return &configSpec, nil | ||||
| 		} | ||||
| 	} | ||||
| 	return nil, fmt.Errorf("No -configure flag found") | ||||
| } | ||||
|  | ||||
| /* | ||||
| ServeAndRecvSpec Function | ||||
|  | ||||
| This function will serve the intro spect and return the configure spec | ||||
| See the ServeIntroSpect and RecvConfigureSpec for more details | ||||
| */ | ||||
| func ServeAndRecvSpec(pluginSpect *IntroSpect) (*ConfigureSpec, error) { | ||||
| 	ServeIntroSpect(pluginSpect) | ||||
| 	return RecvConfigureSpec() | ||||
| } | ||||
							
								
								
									
										35
									
								
								example/plugins/helloworld/www/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								example/plugins/helloworld/www/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
| <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <!-- CSRF token, if your plugin need to make POST request to backend --> | ||||
|     <meta name="zoraxy.csrf.Token" content="{{.csrfToken}}"> | ||||
|     <link rel="stylesheet" href="/script/semantic/semantic.min.css"> | ||||
|     <script src="/script/jquery-3.6.0.min.js"></script> | ||||
|     <script src="/script/semantic/semantic.min.js"></script> | ||||
|     <script src="/script/utils.js"></script> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||
|     <link rel="stylesheet" href="/main.css"> | ||||
|     <title>Hello World</title> | ||||
|     <style> | ||||
|         body { | ||||
|             display: flex; | ||||
|             justify-content: center; | ||||
|             align-items: center; | ||||
|             height: 100vh; | ||||
|             margin: 0; | ||||
|             font-family: Arial, sans-serif; | ||||
|             background:none; | ||||
|         } | ||||
|     </style> | ||||
| </head> | ||||
| <body> | ||||
|     <!-- Dark theme script must be included after body tag--> | ||||
|     <link rel="stylesheet" href="/darktheme.css"> | ||||
|     <script src="/script/darktheme.js"></script> | ||||
|     <div style="text-align: center;"> | ||||
|         <h1>Hello World</h1> | ||||
|         <p>Welcome to your first Zoraxy plugin</p> | ||||
|     </div> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										327
									
								
								example/plugins/upnp/api.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										327
									
								
								example/plugins/upnp/api.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,327 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"net/http" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| /* | ||||
| 	API Handlers | ||||
| */ | ||||
|  | ||||
| func handleUsableState(w http.ResponseWriter, r *http.Request) { | ||||
| 	if r.Method == "GET" { | ||||
| 		js, _ := json.Marshal(upnpRouterExists) | ||||
| 		SendJSONResponse(w, string(js)) | ||||
| 	} else if r.Method == "POST" { | ||||
| 		//Try to probe the UPnP router again | ||||
| 		TryStartUPnPClient() | ||||
| 		if upnpRouterExists { | ||||
| 			SendOK(w) | ||||
| 		} else { | ||||
| 			SendErrorResponse(w, "UPnP router not found") | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Get or set the enable state of the plugin | ||||
| func handleEnableState(w http.ResponseWriter, r *http.Request) { | ||||
| 	if r.Method == "GET" { | ||||
| 		js, _ := json.Marshal(upnpRuntimeConfig.Enabled) | ||||
| 		SendJSONResponse(w, string(js)) | ||||
| 	} else if r.Method == "POST" { | ||||
| 		enable, err := PostBool(r, "enable") | ||||
| 		if err != nil { | ||||
| 			SendErrorResponse(w, err.Error()) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		if !enable { | ||||
| 			//Close all the port forwards if UPnP client is available | ||||
| 			if upnpClient != nil { | ||||
| 				for _, record := range upnpRuntimeConfig.ForwardRules { | ||||
| 					err = upnpClient.ClosePort(record.PortNumber) | ||||
| 					if err != nil { | ||||
| 						SendErrorResponse(w, err.Error()) | ||||
| 						return | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} else { | ||||
| 			if upnpClient == nil { | ||||
| 				SendErrorResponse(w, "No UPnP router in network") | ||||
| 				return | ||||
| 			} | ||||
|  | ||||
| 			//Forward all the ports if UPnP client is available | ||||
| 			if upnpClient != nil { | ||||
| 				for _, record := range upnpRuntimeConfig.ForwardRules { | ||||
| 					err = upnpClient.ForwardPort(record.PortNumber, record.RuleName) | ||||
| 					if err != nil { | ||||
| 						SendErrorResponse(w, err.Error()) | ||||
| 						return | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		upnpRuntimeConfig.Enabled = enable | ||||
| 		SaveRuntimeConfig() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func handleForwardPortEdit(w http.ResponseWriter, r *http.Request) { | ||||
| 	if r.Method == "POST" { | ||||
| 		port, err := PostInt(r, "port") | ||||
| 		if err != nil { | ||||
| 			SendErrorResponse(w, err.Error()) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		oldPort, err := PostInt(r, "oldPort") | ||||
| 		if err != nil { | ||||
| 			SendErrorResponse(w, err.Error()) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		name, err := PostPara(r, "name") | ||||
| 		if err != nil { | ||||
| 			SendErrorResponse(w, err.Error()) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		if port < 1 || port > 65535 { | ||||
| 			SendErrorResponse(w, "invalid port number") | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		//Check if the old port exists | ||||
| 		found := false | ||||
| 		for _, record := range upnpRuntimeConfig.ForwardRules { | ||||
| 			if record.PortNumber == oldPort { | ||||
| 				found = true | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if !found { | ||||
| 			SendErrorResponse(w, "editing forward rule not found") | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		//Delete the old port forward | ||||
| 		if oldPort != port && upnpClient != nil { | ||||
| 			//Remove the port forward if UPnP client is available | ||||
| 			err = upnpClient.ClosePort(oldPort) | ||||
| 			if err != nil { | ||||
| 				SendErrorResponse(w, err.Error()) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Remove from runtime config | ||||
| 		for i, record := range upnpRuntimeConfig.ForwardRules { | ||||
| 			if record.PortNumber == oldPort { | ||||
| 				upnpRuntimeConfig.ForwardRules = append(upnpRuntimeConfig.ForwardRules[:i], upnpRuntimeConfig.ForwardRules[i+1:]...) | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Create the new forward rule | ||||
| 		if upnpClient != nil { | ||||
| 			//Forward the port if UPnP client is available | ||||
| 			err = upnpClient.ForwardPort(port, name) | ||||
| 			if err != nil { | ||||
| 				SendErrorResponse(w, err.Error()) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Add to runtime config | ||||
| 		upnpRuntimeConfig.ForwardRules = append(upnpRuntimeConfig.ForwardRules, &PortForwardRecord{ | ||||
| 			RuleName:   name, | ||||
| 			PortNumber: port, | ||||
| 		}) | ||||
|  | ||||
| 		//Save the runtime config | ||||
| 		SaveRuntimeConfig() | ||||
| 		SendOK(w) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Remove a port forward | ||||
| func handleForwardPortRemove(w http.ResponseWriter, r *http.Request) { | ||||
| 	if r.Method == "POST" { | ||||
| 		port, err := PostInt(r, "port") | ||||
| 		if err != nil { | ||||
| 			SendErrorResponse(w, err.Error()) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		if upnpClient != nil { | ||||
| 			//Remove the port forward if UPnP client is available | ||||
| 			err = upnpClient.ClosePort(port) | ||||
| 			if err != nil { | ||||
| 				SendErrorResponse(w, err.Error()) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Remove from runtime config | ||||
| 		for i, record := range upnpRuntimeConfig.ForwardRules { | ||||
| 			if record.PortNumber == port { | ||||
| 				upnpRuntimeConfig.ForwardRules = append(upnpRuntimeConfig.ForwardRules[:i], upnpRuntimeConfig.ForwardRules[i+1:]...) | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Save the runtime config | ||||
| 		SaveRuntimeConfig() | ||||
| 		SendOK(w) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Handle the port forward operations | ||||
| func handleForwardPort(w http.ResponseWriter, r *http.Request) { | ||||
| 	if r.Method == "GET" { | ||||
| 		// List all the forwarded ports | ||||
| 		js, _ := json.Marshal(upnpRuntimeConfig.ForwardRules) | ||||
| 		SendJSONResponse(w, string(js)) | ||||
| 	} else if r.Method == "POST" { | ||||
| 		//Add a new port forward | ||||
| 		port, err := PostInt(r, "port") | ||||
| 		if err != nil { | ||||
| 			SendErrorResponse(w, err.Error()) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		name, err := PostPara(r, "name") | ||||
| 		if err != nil { | ||||
| 			SendErrorResponse(w, err.Error()) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		if port < 1 || port > 65535 { | ||||
| 			SendErrorResponse(w, "invalid port number") | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		if upnpClient != nil { | ||||
| 			//Forward the port if UPnP client is available | ||||
| 			err = upnpClient.ForwardPort(port, name) | ||||
| 			if err != nil { | ||||
| 				SendErrorResponse(w, err.Error()) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Add to runtime config | ||||
| 		upnpRuntimeConfig.ForwardRules = append(upnpRuntimeConfig.ForwardRules, &PortForwardRecord{ | ||||
| 			RuleName:   name, | ||||
| 			PortNumber: port, | ||||
| 		}) | ||||
|  | ||||
| 		//Save the runtime config | ||||
| 		SaveRuntimeConfig() | ||||
| 		SendOK(w) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* | ||||
| 	Network Utilities | ||||
| */ | ||||
|  | ||||
| // Send JSON response, with an extra json header | ||||
| func SendJSONResponse(w http.ResponseWriter, json string) { | ||||
| 	w.Header().Set("Content-Type", "application/json") | ||||
| 	w.Write([]byte(json)) | ||||
| } | ||||
|  | ||||
| func SendErrorResponse(w http.ResponseWriter, errMsg string) { | ||||
| 	w.Header().Set("Content-Type", "application/json") | ||||
| 	w.Write([]byte("{\"error\":\"" + errMsg + "\"}")) | ||||
| } | ||||
|  | ||||
| func SendOK(w http.ResponseWriter) { | ||||
| 	w.Header().Set("Content-Type", "application/json") | ||||
| 	w.Write([]byte("\"OK\"")) | ||||
| } | ||||
|  | ||||
| // Get GET parameter | ||||
| func GetPara(r *http.Request, key string) (string, error) { | ||||
| 	// Get first value from the URL query | ||||
| 	value := r.URL.Query().Get(key) | ||||
| 	if len(value) == 0 { | ||||
| 		return "", errors.New("invalid " + key + " given") | ||||
| 	} | ||||
| 	return value, nil | ||||
| } | ||||
|  | ||||
| // Get GET paramter as boolean, accept 1 or true | ||||
| func GetBool(r *http.Request, key string) (bool, error) { | ||||
| 	x, err := GetPara(r, key) | ||||
| 	if err != nil { | ||||
| 		return false, err | ||||
| 	} | ||||
|  | ||||
| 	// Convert to lowercase and trim spaces just once to compare | ||||
| 	switch strings.ToLower(strings.TrimSpace(x)) { | ||||
| 	case "1", "true", "on": | ||||
| 		return true, nil | ||||
| 	case "0", "false", "off": | ||||
| 		return false, nil | ||||
| 	} | ||||
|  | ||||
| 	return false, errors.New("invalid boolean given") | ||||
| } | ||||
|  | ||||
| // Get POST parameter | ||||
| func PostPara(r *http.Request, key string) (string, error) { | ||||
| 	// Try to parse the form | ||||
| 	if err := r.ParseForm(); err != nil { | ||||
| 		return "", err | ||||
| 	} | ||||
| 	// Get first value from the form | ||||
| 	x := r.Form.Get(key) | ||||
| 	if len(x) == 0 { | ||||
| 		return "", errors.New("invalid " + key + " given") | ||||
| 	} | ||||
| 	return x, nil | ||||
| } | ||||
|  | ||||
| // Get POST paramter as boolean, accept 1 or true | ||||
| func PostBool(r *http.Request, key string) (bool, error) { | ||||
| 	x, err := PostPara(r, key) | ||||
| 	if err != nil { | ||||
| 		return false, err | ||||
| 	} | ||||
|  | ||||
| 	// Convert to lowercase and trim spaces just once to compare | ||||
| 	switch strings.ToLower(strings.TrimSpace(x)) { | ||||
| 	case "1", "true", "on": | ||||
| 		return true, nil | ||||
| 	case "0", "false", "off": | ||||
| 		return false, nil | ||||
| 	} | ||||
|  | ||||
| 	return false, errors.New("invalid boolean given") | ||||
| } | ||||
|  | ||||
| // Get POST paramter as int | ||||
| func PostInt(r *http.Request, key string) (int, error) { | ||||
| 	x, err := PostPara(r, key) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
|  | ||||
| 	x = strings.TrimSpace(x) | ||||
| 	rx, err := strconv.Atoi(x) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
|  | ||||
| 	return rx, nil | ||||
| } | ||||
							
								
								
									
										13
									
								
								example/plugins/upnp/go.mod
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								example/plugins/upnp/go.mod
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| module plugins.zoraxy.aroz.org/zoraxy/upnp | ||||
|  | ||||
| go 1.23.6 | ||||
|  | ||||
| require gitlab.com/NebulousLabs/go-upnp v0.0.0-20211002182029-11da932010b6 | ||||
|  | ||||
| require ( | ||||
| 	gitlab.com/NebulousLabs/fastrand v0.0.0-20181126182046-603482d69e40 // indirect | ||||
| 	golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2 // indirect | ||||
| 	golang.org/x/net v0.0.0-20210410081132-afb366fc7cd1 // indirect | ||||
| 	golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44 // indirect | ||||
| 	golang.org/x/text v0.3.6 // indirect | ||||
| ) | ||||
							
								
								
									
										17
									
								
								example/plugins/upnp/go.sum
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								example/plugins/upnp/go.sum
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| gitlab.com/NebulousLabs/fastrand v0.0.0-20181126182046-603482d69e40 h1:dizWJqTWjwyD8KGcMOwgrkqu1JIkofYgKkmDeNE7oAs= | ||||
| gitlab.com/NebulousLabs/fastrand v0.0.0-20181126182046-603482d69e40/go.mod h1:rOnSnoRyxMI3fe/7KIbVcsHRGxe30OONv8dEgo+vCfA= | ||||
| gitlab.com/NebulousLabs/go-upnp v0.0.0-20211002182029-11da932010b6 h1:WKij6HF8ECp9E7K0E44dew9NrRDGiNR5u4EFsXnJUx4= | ||||
| gitlab.com/NebulousLabs/go-upnp v0.0.0-20211002182029-11da932010b6/go.mod h1:vhrHTGDh4YR7wK8Z+kRJ+x8SF/6RUM3Vb64Si5FD0L8= | ||||
| golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2 h1:It14KIkyBFYkHkwZ7k45minvA9aorojkyjGk9KJ5B/w= | ||||
| golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | ||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20210410081132-afb366fc7cd1 h1:4qWs8cYYH6PoEFy4dfhDFgoMGkwAcETd+MmPdCPMzUc= | ||||
| golang.org/x/net v0.0.0-20210410081132-afb366fc7cd1/go.mod h1:9tjilg8BloeKEkVJvy7fQ90B1CfIiPueXVOjqfkSzI8= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44 h1:Bli41pIlzTzf3KEY06n+xnzK/BESIg2ze4Pgfh/aI8c= | ||||
| golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6 h1:aRYxNxv6iGQlyVaZmk6ZgYEDa+Jg18DxebPSrd6bg1M= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
							
								
								
									
										
											BIN
										
									
								
								example/plugins/upnp/icon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								example/plugins/upnp/icon.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 7.8 KiB | 
							
								
								
									
										
											BIN
										
									
								
								example/plugins/upnp/icon.psd
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								example/plugins/upnp/icon.psd
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										194
									
								
								example/plugins/upnp/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										194
									
								
								example/plugins/upnp/main.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,194 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	_ "embed" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"strconv" | ||||
| 	"time" | ||||
|  | ||||
| 	"plugins.zoraxy.aroz.org/zoraxy/upnp/mod/upnpc" | ||||
| 	plugin "plugins.zoraxy.aroz.org/zoraxy/upnp/mod/zoraxy_plugin" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	PLUGIN_ID           = "org.aroz.zoraxy.plugins.upnp" | ||||
| 	UI_PATH             = "/ui" | ||||
| 	WEB_ROOT            = "/www" | ||||
| 	CONFIG_FILE         = "upnp.json" | ||||
| 	AUTO_RENEW_INTERVAL = 12 * 60 * 60 // 12 hours | ||||
| ) | ||||
|  | ||||
| type PortForwardRecord struct { | ||||
| 	RuleName   string | ||||
| 	PortNumber int | ||||
| } | ||||
|  | ||||
| type UPnPConfig struct { | ||||
| 	ForwardRules []*PortForwardRecord | ||||
| 	Enabled      bool | ||||
| } | ||||
|  | ||||
| //go:embed www/* | ||||
| var content embed.FS | ||||
|  | ||||
| // Runtime variables | ||||
| var ( | ||||
| 	upnpRouterExists  bool        = false | ||||
| 	upnpRuntimeConfig *UPnPConfig = &UPnPConfig{ | ||||
| 		ForwardRules: []*PortForwardRecord{}, | ||||
| 		Enabled:      false, | ||||
| 	} | ||||
| 	upnpClient      *upnpc.UPnPClient = nil | ||||
| 	renewTickerStop chan bool | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
| 	//Handle introspect | ||||
| 	runtimeCfg, err := plugin.ServeAndRecvSpec(&plugin.IntroSpect{ | ||||
| 		ID:            PLUGIN_ID, | ||||
| 		Name:          "UPnP Forwarder", | ||||
| 		Author:        "aroz.org", | ||||
| 		AuthorContact: "https://github.com/aroz-online", | ||||
| 		Description:   "A UPnP Port Forwarder Plugin for Zoraxy", | ||||
| 		URL:           "https://github.com/aroz-online", | ||||
| 		Type:          plugin.PluginType_Utilities, | ||||
| 		VersionMajor:  1, | ||||
| 		VersionMinor:  0, | ||||
| 		VersionPatch:  0, | ||||
| 		UIPath:        UI_PATH, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		//Terminate or enter standalone mode here | ||||
| 		fmt.Println("This is a plugin for Zoraxy and should not be run standalone\n Visit zoraxy.aroz.org to download Zoraxy.") | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	//Read the configuration from file | ||||
| 	if _, err := os.Stat(CONFIG_FILE); os.IsNotExist(err) { | ||||
| 		err = os.WriteFile(CONFIG_FILE, []byte("{}"), 0644) | ||||
| 		if err != nil { | ||||
| 			panic(err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	cfgBytes, err := os.ReadFile(CONFIG_FILE) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	//Load the configuration | ||||
| 	err = json.Unmarshal(cfgBytes, &upnpRuntimeConfig) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	//Start upnp client and auto-renew ticker | ||||
| 	go func() { | ||||
| 		TryStartUPnPClient() | ||||
| 	}() | ||||
|  | ||||
| 	//Serve the plugin UI | ||||
| 	embedWebRouter := plugin.NewPluginEmbedUIRouter(PLUGIN_ID, &content, WEB_ROOT, UI_PATH) | ||||
| 	// For debugging, use the following line instead | ||||
| 	//embedWebRouter := plugin.NewPluginFileSystemUIRouter(PLUGIN_ID, "."+WEB_ROOT, UI_PATH) | ||||
| 	//embedWebRouter.EnableDebug = true | ||||
| 	embedWebRouter.RegisterTerminateHandler(func() { | ||||
| 		if renewTickerStop != nil { | ||||
| 			renewTickerStop <- true | ||||
| 		} | ||||
| 		// Do cleanup here if needed | ||||
| 		upnpClient.Close() | ||||
| 	}, nil) | ||||
| 	embedWebRouter.AttachHandlerToMux(nil) | ||||
|  | ||||
| 	//Serve the API | ||||
| 	RegisterAPIs() | ||||
|  | ||||
| 	//Start the IO server | ||||
| 	fmt.Println("UPnP Forwarder started at http://127.0.0.1:" + strconv.Itoa(runtimeCfg.Port)) | ||||
| 	err = http.ListenAndServe("127.0.0.1:"+strconv.Itoa(runtimeCfg.Port), nil) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // RegisterAPIs registers the APIs for the plugin | ||||
| func RegisterAPIs() { | ||||
| 	http.HandleFunc(UI_PATH+"/api/usable", handleUsableState) | ||||
| 	http.HandleFunc(UI_PATH+"/api/enable", handleEnableState) | ||||
| 	http.HandleFunc(UI_PATH+"/api/forward", handleForwardPort) | ||||
| 	http.HandleFunc(UI_PATH+"/api/edit", handleForwardPortEdit) | ||||
| 	http.HandleFunc(UI_PATH+"/api/remove", handleForwardPortRemove) | ||||
| } | ||||
|  | ||||
| // TryStartUPnPClient tries to start the UPnP client | ||||
| func TryStartUPnPClient() { | ||||
| 	if renewTickerStop != nil { | ||||
| 		renewTickerStop <- true | ||||
| 	} | ||||
|  | ||||
| 	// Create UPnP client | ||||
| 	upnpClient, err := upnpc.NewUPNPClient() | ||||
| 	if err != nil { | ||||
| 		upnpRouterExists = false | ||||
| 		upnpRuntimeConfig.Enabled = false | ||||
| 		fmt.Println("UPnP router not found") | ||||
| 		SaveRuntimeConfig() | ||||
| 		return | ||||
| 	} | ||||
| 	upnpRouterExists = true | ||||
|  | ||||
| 	//Check if the client is enabled by default | ||||
| 	if upnpRuntimeConfig.Enabled { | ||||
| 		// Forward all the ports | ||||
| 		for _, rule := range upnpRuntimeConfig.ForwardRules { | ||||
| 			err = upnpClient.ForwardPort(rule.PortNumber, rule.RuleName) | ||||
| 			if err != nil { | ||||
| 				fmt.Println("Unable to forward port", rule.PortNumber, ":", err) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Start the auto-renew ticker | ||||
| 	_, renewTickerStop = SetupAutoRenewTicker() | ||||
| } | ||||
|  | ||||
| // SetupAutoRenewTicker sets up a ticker for auto-renewing the port forwarding rules | ||||
| func SetupAutoRenewTicker() (*time.Ticker, chan bool) { | ||||
| 	ticker := time.NewTicker(AUTO_RENEW_INTERVAL * time.Second) | ||||
| 	closeChan := make(chan bool) | ||||
| 	go func() { | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-closeChan: | ||||
| 				ticker.Stop() | ||||
| 				return | ||||
| 			case <-ticker.C: | ||||
| 				if upnpClient != nil { | ||||
| 					upnpClient.RenewForwardRules() | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| 	return ticker, closeChan | ||||
| } | ||||
|  | ||||
| // SaveRuntimeConfig saves the runtime configuration to file | ||||
| func SaveRuntimeConfig() error { | ||||
| 	cfgBytes, err := json.Marshal(upnpRuntimeConfig) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	err = os.WriteFile(CONFIG_FILE, cfgBytes, 0644) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										135
									
								
								example/plugins/upnp/mod/upnpc/upnpc.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										135
									
								
								example/plugins/upnp/mod/upnpc/upnpc.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,135 @@ | ||||
| package upnpc | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"gitlab.com/NebulousLabs/go-upnp" | ||||
| ) | ||||
|  | ||||
| /* | ||||
| 	uPNP Module | ||||
|  | ||||
| 	This module handles uPNP Connections to the gateway router and create a port forward entry | ||||
| 	for the host system at the given port (set with -port paramter) | ||||
| */ | ||||
|  | ||||
| type UPnPClient struct { | ||||
| 	Connection    *upnp.IGD //UPnP conenction object | ||||
| 	ExternalIP    string    //Storage of external IP address | ||||
| 	RequiredPorts []int     //All the required ports will be recored | ||||
| 	PolicyNames   sync.Map  //Name for the required port nubmer | ||||
| } | ||||
|  | ||||
| // NewUPNPClient creates a new UPnPClient object | ||||
| func NewUPNPClient() (*UPnPClient, error) { | ||||
| 	//Create uPNP forwarding in the NAT router | ||||
| 	fmt.Println("Discovering UPnP router in Local Area Network...") | ||||
| 	d, err := upnp.Discover() | ||||
| 	if err != nil { | ||||
| 		return &UPnPClient{}, err | ||||
| 	} | ||||
|  | ||||
| 	// discover external IP | ||||
| 	ip, err := d.ExternalIP() | ||||
| 	if err != nil { | ||||
| 		return &UPnPClient{}, err | ||||
| 	} | ||||
|  | ||||
| 	//Create the final obejcts | ||||
| 	newUPnPObject := &UPnPClient{ | ||||
| 		Connection:    d, | ||||
| 		ExternalIP:    ip, | ||||
| 		RequiredPorts: []int{}, | ||||
| 	} | ||||
|  | ||||
| 	return newUPnPObject, nil | ||||
| } | ||||
|  | ||||
| // ForwardPort forwards a port to the host | ||||
| func (u *UPnPClient) ForwardPort(portNumber int, ruleName string) error { | ||||
| 	fmt.Println("UPnP forwarding new port: ", portNumber, "for "+ruleName+" service") | ||||
|  | ||||
| 	//Check if port already forwarded | ||||
| 	_, ok := u.PolicyNames.Load(portNumber) | ||||
| 	if ok { | ||||
| 		//Port already forward. Ignore this request | ||||
| 		return errors.New("port already forwarded") | ||||
| 	} | ||||
|  | ||||
| 	// forward a port | ||||
| 	err := u.Connection.Forward(uint16(portNumber), ruleName) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	u.RequiredPorts = append(u.RequiredPorts, portNumber) | ||||
| 	u.PolicyNames.Store(portNumber, ruleName) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // ClosePort closes the port forwarding | ||||
| func (u *UPnPClient) ClosePort(portNumber int) error { | ||||
| 	//Check if port is opened | ||||
| 	portOpened := false | ||||
| 	newRequiredPort := []int{} | ||||
| 	for _, thisPort := range u.RequiredPorts { | ||||
| 		if thisPort != portNumber { | ||||
| 			newRequiredPort = append(newRequiredPort, thisPort) | ||||
| 		} else { | ||||
| 			portOpened = true | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if portOpened { | ||||
| 		//Update the port list | ||||
| 		u.RequiredPorts = newRequiredPort | ||||
|  | ||||
| 		// Close the port | ||||
| 		fmt.Println("Closing UPnP Port Forward: ", portNumber) | ||||
| 		err := u.Connection.Clear(uint16(portNumber)) | ||||
|  | ||||
| 		//Delete the name registry | ||||
| 		u.PolicyNames.Delete(portNumber) | ||||
|  | ||||
| 		if err != nil { | ||||
| 			fmt.Println(err) | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Renew forward rules, prevent router lease time from flushing the Upnp config | ||||
| func (u *UPnPClient) RenewForwardRules() { | ||||
| 	if u.Connection == nil { | ||||
| 		//UPnP router gone | ||||
| 		return | ||||
| 	} | ||||
| 	portsToRenew := u.RequiredPorts | ||||
| 	for _, thisPort := range portsToRenew { | ||||
| 		ruleName, ok := u.PolicyNames.Load(thisPort) | ||||
| 		if !ok { | ||||
| 			continue | ||||
| 		} | ||||
| 		u.ClosePort(thisPort) | ||||
| 		time.Sleep(100 * time.Millisecond) | ||||
| 		u.ForwardPort(thisPort, ruleName.(string)) | ||||
| 	} | ||||
| 	fmt.Println("UPnP Port Forward rule renew completed") | ||||
| } | ||||
|  | ||||
| func (u *UPnPClient) Close() error { | ||||
| 	//Shutdown the default UPnP Object | ||||
| 	if u != nil { | ||||
| 		for _, portNumber := range u.RequiredPorts { | ||||
| 			err := u.Connection.Clear(uint16(portNumber)) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										19
									
								
								example/plugins/upnp/mod/zoraxy_plugin/README.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								example/plugins/upnp/mod/zoraxy_plugin/README.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| # Zoraxy Plugin | ||||
|  | ||||
| ## Overview | ||||
| This module serves as a template for building your own plugins for the Zoraxy Reverse Proxy. By copying this module to your plugin mod folder, you can create a new plugin with the necessary structure and components. | ||||
|  | ||||
| ## Instructions | ||||
|  | ||||
| 1. **Copy the Module:** | ||||
|     - Copy the entire `zoraxy_plugin` module to your plugin mod folder. | ||||
|  | ||||
| 2. **Include the Structure:** | ||||
|     - Ensure that you maintain the directory structure and file organization as provided in this module. | ||||
|  | ||||
| 3. **Modify as Needed:** | ||||
|     - Customize the copied module to implement the desired functionality for your plugin. | ||||
|  | ||||
| ## Directory Structure | ||||
|  zoraxy_plugin: Handle -introspect and -configuration process required for plugin loading and startup | ||||
|  embed_webserver: Handle embeded web server routing and injecting csrf token to your plugin served UI pages | ||||
							
								
								
									
										145
									
								
								example/plugins/upnp/mod/zoraxy_plugin/dev_webserver.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								example/plugins/upnp/mod/zoraxy_plugin/dev_webserver.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type PluginUiDebugRouter struct { | ||||
| 	PluginID         string //The ID of the plugin | ||||
| 	TargetDir        string //The directory where the UI files are stored | ||||
| 	HandlerPrefix    string //The prefix of the handler used to route this router, e.g. /ui | ||||
| 	EnableDebug      bool   //Enable debug mode | ||||
| 	terminateHandler func() //The handler to be called when the plugin is terminated | ||||
| } | ||||
|  | ||||
| // NewPluginFileSystemUIRouter creates a new PluginUiRouter with file system | ||||
| // The targetDir is the directory where the UI files are stored (e.g. ./www) | ||||
| // The handlerPrefix is the prefix of the handler used to route this router | ||||
| // The handlerPrefix should start with a slash (e.g. /ui) that matches the http.Handle path | ||||
| // All prefix should not end with a slash | ||||
| func NewPluginFileSystemUIRouter(pluginID string, targetDir string, handlerPrefix string) *PluginUiDebugRouter { | ||||
| 	//Make sure all prefix are in /prefix format | ||||
| 	if !strings.HasPrefix(handlerPrefix, "/") { | ||||
| 		handlerPrefix = "/" + handlerPrefix | ||||
| 	} | ||||
| 	handlerPrefix = strings.TrimSuffix(handlerPrefix, "/") | ||||
|  | ||||
| 	//Return the PluginUiRouter | ||||
| 	return &PluginUiDebugRouter{ | ||||
| 		PluginID:      pluginID, | ||||
| 		TargetDir:     targetDir, | ||||
| 		HandlerPrefix: handlerPrefix, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *PluginUiDebugRouter) populateCSRFToken(r *http.Request, fsHandler http.Handler) http.Handler { | ||||
| 	//Get the CSRF token from header | ||||
| 	csrfToken := r.Header.Get("X-Zoraxy-Csrf") | ||||
| 	if csrfToken == "" { | ||||
| 		csrfToken = "missing-csrf-token" | ||||
| 	} | ||||
|  | ||||
| 	//Return the middleware | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		// Check if the request is for an HTML file | ||||
| 		if strings.HasSuffix(r.URL.Path, ".html") { | ||||
| 			//Read the target file from file system | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetDir + "/" + targetFilePath | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			targetFileContent, err := os.ReadFile(targetFilePath) | ||||
| 			if err != nil { | ||||
| 				http.Error(w, "File not found", http.StatusNotFound) | ||||
| 				return | ||||
| 			} | ||||
| 			body := string(targetFileContent) | ||||
| 			body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 			w.Header().Set("Content-Type", "text/html") | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte(body)) | ||||
| 			return | ||||
| 		} else if strings.HasSuffix(r.URL.Path, "/") { | ||||
| 			//Check if the request is for a directory | ||||
| 			//Check if the directory has an index.html file | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetDir + "/" + targetFilePath + "index.html" | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			if _, err := os.Stat(targetFilePath); err == nil { | ||||
| 				//Serve the index.html file | ||||
| 				targetFileContent, err := os.ReadFile(targetFilePath) | ||||
| 				if err != nil { | ||||
| 					http.Error(w, "File not found", http.StatusNotFound) | ||||
| 					return | ||||
| 				} | ||||
| 				body := string(targetFileContent) | ||||
| 				body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 				w.Header().Set("Content-Type", "text/html") | ||||
| 				w.WriteHeader(http.StatusOK) | ||||
| 				w.Write([]byte(body)) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Call the next handler | ||||
| 		fsHandler.ServeHTTP(w, r) | ||||
| 	}) | ||||
|  | ||||
| } | ||||
|  | ||||
| // GetHttpHandler returns the http.Handler for the PluginUiRouter | ||||
| func (p *PluginUiDebugRouter) Handler() http.Handler { | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		//Remove the plugin UI handler path prefix | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Print("Request URL:", r.URL.Path, " rewriting to ") | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, p.HandlerPrefix) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		r.URL.Path = rewrittenURL | ||||
| 		r.RequestURI = rewrittenURL | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Println(r.URL.Path) | ||||
| 		} | ||||
|  | ||||
| 		//Serve the file from the file system | ||||
| 		fsHandler := http.FileServer(http.Dir(p.TargetDir)) | ||||
|  | ||||
| 		// Replace {{csrf_token}} with the actual CSRF token and serve the file | ||||
| 		p.populateCSRFToken(r, fsHandler).ServeHTTP(w, r) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // RegisterTerminateHandler registers the terminate handler for the PluginUiRouter | ||||
| // The terminate handler will be called when the plugin is terminated from Zoraxy plugin manager | ||||
| // if mux is nil, the handler will be registered to http.DefaultServeMux | ||||
| func (p *PluginUiDebugRouter) RegisterTerminateHandler(termFunc func(), mux *http.ServeMux) { | ||||
| 	p.terminateHandler = termFunc | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
| 	mux.HandleFunc(p.HandlerPrefix+"/term", func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.terminateHandler() | ||||
| 		w.WriteHeader(http.StatusOK) | ||||
| 		go func() { | ||||
| 			//Make sure the response is sent before the plugin is terminated | ||||
| 			time.Sleep(100 * time.Millisecond) | ||||
| 			os.Exit(0) | ||||
| 		}() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // Attach the file system UI handler to the target http.ServeMux | ||||
| func (p *PluginUiDebugRouter) AttachHandlerToMux(mux *http.ServeMux) { | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
|  | ||||
| 	p.HandlerPrefix = strings.TrimSuffix(p.HandlerPrefix, "/") | ||||
| 	mux.Handle(p.HandlerPrefix+"/", p.Handler()) | ||||
| } | ||||
							
								
								
									
										162
									
								
								example/plugins/upnp/mod/zoraxy_plugin/dynamic_router.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										162
									
								
								example/plugins/upnp/mod/zoraxy_plugin/dynamic_router.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,162 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| /* | ||||
|  | ||||
| 	Dynamic Path Handler | ||||
|  | ||||
| */ | ||||
|  | ||||
| type SniffResult int | ||||
|  | ||||
| const ( | ||||
| 	SniffResultAccpet SniffResult = iota // Forward the request to this plugin dynamic capture ingress | ||||
| 	SniffResultSkip                      // Skip this plugin and let the next plugin handle the request | ||||
| ) | ||||
|  | ||||
| type SniffHandler func(*DynamicSniffForwardRequest) SniffResult | ||||
|  | ||||
| /* | ||||
| RegisterDynamicSniffHandler registers a dynamic sniff handler for a path | ||||
| You can decide to accept or skip the request based on the request header and paths | ||||
| */ | ||||
| func (p *PathRouter) RegisterDynamicSniffHandler(sniff_ingress string, mux *http.ServeMux, handler SniffHandler) { | ||||
| 	if !strings.HasSuffix(sniff_ingress, "/") { | ||||
| 		sniff_ingress = sniff_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(sniff_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Println("Request captured by dynamic sniff path: " + r.RequestURI) | ||||
| 		} | ||||
|  | ||||
| 		// Decode the request payload | ||||
| 		jsonBytes, err := io.ReadAll(r.Body) | ||||
| 		if err != nil { | ||||
| 			if p.enableDebugPrint { | ||||
| 				fmt.Println("Error reading request body:", err) | ||||
| 			} | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
| 		payload, err := DecodeForwardRequestPayload(jsonBytes) | ||||
| 		if err != nil { | ||||
| 			if p.enableDebugPrint { | ||||
| 				fmt.Println("Error decoding request payload:", err) | ||||
| 				fmt.Print("Payload: ") | ||||
| 				fmt.Println(string(jsonBytes)) | ||||
| 			} | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		// Get the forwarded request UUID | ||||
| 		forwardUUID := r.Header.Get("X-Zoraxy-RequestID") | ||||
| 		payload.requestUUID = forwardUUID | ||||
| 		payload.rawRequest = r | ||||
|  | ||||
| 		sniffResult := handler(&payload) | ||||
| 		if sniffResult == SniffResultAccpet { | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte("OK")) | ||||
| 		} else { | ||||
| 			w.WriteHeader(http.StatusNotImplemented) | ||||
| 			w.Write([]byte("SKIP")) | ||||
| 		} | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| // RegisterDynamicCaptureHandle register the dynamic capture ingress path with a handler | ||||
| func (p *PathRouter) RegisterDynamicCaptureHandle(capture_ingress string, mux *http.ServeMux, handlefunc func(http.ResponseWriter, *http.Request)) { | ||||
| 	if !strings.HasSuffix(capture_ingress, "/") { | ||||
| 		capture_ingress = capture_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(capture_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Println("Request captured by dynamic capture path: " + r.RequestURI) | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, capture_ingress) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		if rewrittenURL == "" { | ||||
| 			rewrittenURL = "/" | ||||
| 		} | ||||
| 		if !strings.HasPrefix(rewrittenURL, "/") { | ||||
| 			rewrittenURL = "/" + rewrittenURL | ||||
| 		} | ||||
| 		r.RequestURI = rewrittenURL | ||||
|  | ||||
| 		handlefunc(w, r) | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| /* | ||||
| 	Sniffing and forwarding | ||||
|  | ||||
| 	The following functions are here to help with | ||||
| 	sniffing and forwarding requests to the dynamic | ||||
| 	router. | ||||
| */ | ||||
| // A custom request object to be used in the dynamic sniffing | ||||
| type DynamicSniffForwardRequest struct { | ||||
| 	Method     string              `json:"method"` | ||||
| 	Hostname   string              `json:"hostname"` | ||||
| 	URL        string              `json:"url"` | ||||
| 	Header     map[string][]string `json:"header"` | ||||
| 	RemoteAddr string              `json:"remote_addr"` | ||||
| 	Host       string              `json:"host"` | ||||
| 	RequestURI string              `json:"request_uri"` | ||||
| 	Proto      string              `json:"proto"` | ||||
| 	ProtoMajor int                 `json:"proto_major"` | ||||
| 	ProtoMinor int                 `json:"proto_minor"` | ||||
|  | ||||
| 	/* Internal use */ | ||||
| 	rawRequest  *http.Request `json:"-"` | ||||
| 	requestUUID string        `json:"-"` | ||||
| } | ||||
|  | ||||
| // GetForwardRequestPayload returns a DynamicSniffForwardRequest object from an http.Request object | ||||
| func EncodeForwardRequestPayload(r *http.Request) DynamicSniffForwardRequest { | ||||
| 	return DynamicSniffForwardRequest{ | ||||
| 		Method:     r.Method, | ||||
| 		Hostname:   r.Host, | ||||
| 		URL:        r.URL.String(), | ||||
| 		Header:     r.Header, | ||||
| 		RemoteAddr: r.RemoteAddr, | ||||
| 		Host:       r.Host, | ||||
| 		RequestURI: r.RequestURI, | ||||
| 		Proto:      r.Proto, | ||||
| 		ProtoMajor: r.ProtoMajor, | ||||
| 		ProtoMinor: r.ProtoMinor, | ||||
| 		rawRequest: r, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // DecodeForwardRequestPayload decodes JSON bytes into a DynamicSniffForwardRequest object | ||||
| func DecodeForwardRequestPayload(jsonBytes []byte) (DynamicSniffForwardRequest, error) { | ||||
| 	var payload DynamicSniffForwardRequest | ||||
| 	err := json.Unmarshal(jsonBytes, &payload) | ||||
| 	if err != nil { | ||||
| 		return DynamicSniffForwardRequest{}, err | ||||
| 	} | ||||
| 	return payload, nil | ||||
| } | ||||
|  | ||||
| // GetRequest returns the original http.Request object, for debugging purposes | ||||
| func (dsfr *DynamicSniffForwardRequest) GetRequest() *http.Request { | ||||
| 	return dsfr.rawRequest | ||||
| } | ||||
|  | ||||
| // GetRequestUUID returns the request UUID | ||||
| // if this UUID is empty string, that might indicate the request | ||||
| // is not coming from the dynamic router | ||||
| func (dsfr *DynamicSniffForwardRequest) GetRequestUUID() string { | ||||
| 	return dsfr.requestUUID | ||||
| } | ||||
							
								
								
									
										156
									
								
								example/plugins/upnp/mod/zoraxy_plugin/embed_webserver.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								example/plugins/upnp/mod/zoraxy_plugin/embed_webserver.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	"fmt" | ||||
| 	"io/fs" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type PluginUiRouter struct { | ||||
| 	PluginID         string    //The ID of the plugin | ||||
| 	TargetFs         *embed.FS //The embed.FS where the UI files are stored | ||||
| 	TargetFsPrefix   string    //The prefix of the embed.FS where the UI files are stored, e.g. /web | ||||
| 	HandlerPrefix    string    //The prefix of the handler used to route this router, e.g. /ui | ||||
| 	EnableDebug      bool      //Enable debug mode | ||||
| 	terminateHandler func()    //The handler to be called when the plugin is terminated | ||||
| } | ||||
|  | ||||
| // NewPluginEmbedUIRouter creates a new PluginUiRouter with embed.FS | ||||
| // The targetFsPrefix is the prefix of the embed.FS where the UI files are stored | ||||
| // The targetFsPrefix should be relative to the root of the embed.FS | ||||
| // The targetFsPrefix should start with a slash (e.g. /web) that corresponds to the root folder of the embed.FS | ||||
| // The handlerPrefix is the prefix of the handler used to route this router | ||||
| // The handlerPrefix should start with a slash (e.g. /ui) that matches the http.Handle path | ||||
| // All prefix should not end with a slash | ||||
| func NewPluginEmbedUIRouter(pluginID string, targetFs *embed.FS, targetFsPrefix string, handlerPrefix string) *PluginUiRouter { | ||||
| 	//Make sure all prefix are in /prefix format | ||||
| 	if !strings.HasPrefix(targetFsPrefix, "/") { | ||||
| 		targetFsPrefix = "/" + targetFsPrefix | ||||
| 	} | ||||
| 	targetFsPrefix = strings.TrimSuffix(targetFsPrefix, "/") | ||||
|  | ||||
| 	if !strings.HasPrefix(handlerPrefix, "/") { | ||||
| 		handlerPrefix = "/" + handlerPrefix | ||||
| 	} | ||||
| 	handlerPrefix = strings.TrimSuffix(handlerPrefix, "/") | ||||
|  | ||||
| 	//Return the PluginUiRouter | ||||
| 	return &PluginUiRouter{ | ||||
| 		PluginID:       pluginID, | ||||
| 		TargetFs:       targetFs, | ||||
| 		TargetFsPrefix: targetFsPrefix, | ||||
| 		HandlerPrefix:  handlerPrefix, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *PluginUiRouter) populateCSRFToken(r *http.Request, fsHandler http.Handler) http.Handler { | ||||
| 	//Get the CSRF token from header | ||||
| 	csrfToken := r.Header.Get("X-Zoraxy-Csrf") | ||||
| 	if csrfToken == "" { | ||||
| 		csrfToken = "missing-csrf-token" | ||||
| 	} | ||||
|  | ||||
| 	//Return the middleware | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		// Check if the request is for an HTML file | ||||
| 		if strings.HasSuffix(r.URL.Path, ".html") { | ||||
| 			//Read the target file from embed.FS | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetFsPrefix + "/" + targetFilePath | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			targetFileContent, err := fs.ReadFile(*p.TargetFs, targetFilePath) | ||||
| 			if err != nil { | ||||
| 				http.Error(w, "File not found", http.StatusNotFound) | ||||
| 				return | ||||
| 			} | ||||
| 			body := string(targetFileContent) | ||||
| 			body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 			w.Header().Set("Content-Type", "text/html") | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte(body)) | ||||
| 			return | ||||
| 		} else if strings.HasSuffix(r.URL.Path, "/") { | ||||
| 			// Check if the directory has an index.html file | ||||
| 			indexFilePath := strings.TrimPrefix(r.URL.Path, "/") + "index.html" | ||||
| 			indexFilePath = p.TargetFsPrefix + "/" + indexFilePath | ||||
| 			indexFilePath = strings.TrimPrefix(indexFilePath, "/") | ||||
| 			indexFileContent, err := fs.ReadFile(*p.TargetFs, indexFilePath) | ||||
| 			if err == nil { | ||||
| 				body := string(indexFileContent) | ||||
| 				body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 				w.Header().Set("Content-Type", "text/html") | ||||
| 				w.WriteHeader(http.StatusOK) | ||||
| 				w.Write([]byte(body)) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Call the next handler | ||||
| 		fsHandler.ServeHTTP(w, r) | ||||
| 	}) | ||||
|  | ||||
| } | ||||
|  | ||||
| // GetHttpHandler returns the http.Handler for the PluginUiRouter | ||||
| func (p *PluginUiRouter) Handler() http.Handler { | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		//Remove the plugin UI handler path prefix | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Print("Request URL:", r.URL.Path, " rewriting to ") | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, p.HandlerPrefix) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		r.URL, _ = url.Parse(rewrittenURL) | ||||
| 		r.RequestURI = rewrittenURL | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Println(r.URL.Path) | ||||
| 		} | ||||
|  | ||||
| 		//Serve the file from the embed.FS | ||||
| 		subFS, err := fs.Sub(*p.TargetFs, strings.TrimPrefix(p.TargetFsPrefix, "/")) | ||||
| 		if err != nil { | ||||
| 			fmt.Println(err.Error()) | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		// Replace {{csrf_token}} with the actual CSRF token and serve the file | ||||
| 		p.populateCSRFToken(r, http.FileServer(http.FS(subFS))).ServeHTTP(w, r) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // RegisterTerminateHandler registers the terminate handler for the PluginUiRouter | ||||
| // The terminate handler will be called when the plugin is terminated from Zoraxy plugin manager | ||||
| // if mux is nil, the handler will be registered to http.DefaultServeMux | ||||
| func (p *PluginUiRouter) RegisterTerminateHandler(termFunc func(), mux *http.ServeMux) { | ||||
| 	p.terminateHandler = termFunc | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
| 	mux.HandleFunc(p.HandlerPrefix+"/term", func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.terminateHandler() | ||||
| 		w.WriteHeader(http.StatusOK) | ||||
| 		go func() { | ||||
| 			//Make sure the response is sent before the plugin is terminated | ||||
| 			time.Sleep(100 * time.Millisecond) | ||||
| 			os.Exit(0) | ||||
| 		}() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // Attach the embed UI handler to the target http.ServeMux | ||||
| func (p *PluginUiRouter) AttachHandlerToMux(mux *http.ServeMux) { | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
|  | ||||
| 	p.HandlerPrefix = strings.TrimSuffix(p.HandlerPrefix, "/") | ||||
| 	mux.Handle(p.HandlerPrefix+"/", p.Handler()) | ||||
| } | ||||
							
								
								
									
										105
									
								
								example/plugins/upnp/mod/zoraxy_plugin/static_router.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								example/plugins/upnp/mod/zoraxy_plugin/static_router.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type PathRouter struct { | ||||
| 	enableDebugPrint bool | ||||
| 	pathHandlers     map[string]http.Handler | ||||
| 	defaultHandler   http.Handler | ||||
| } | ||||
|  | ||||
| // NewPathRouter creates a new PathRouter | ||||
| func NewPathRouter() *PathRouter { | ||||
| 	return &PathRouter{ | ||||
| 		enableDebugPrint: false, | ||||
| 		pathHandlers:     make(map[string]http.Handler), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // RegisterPathHandler registers a handler for a path | ||||
| func (p *PathRouter) RegisterPathHandler(path string, handler http.Handler) { | ||||
| 	path = strings.TrimSuffix(path, "/") | ||||
| 	p.pathHandlers[path] = handler | ||||
| } | ||||
|  | ||||
| // RemovePathHandler removes a handler for a path | ||||
| func (p *PathRouter) RemovePathHandler(path string) { | ||||
| 	delete(p.pathHandlers, path) | ||||
| } | ||||
|  | ||||
| // SetDefaultHandler sets the default handler for the router | ||||
| // This handler will be called if no path handler is found | ||||
| func (p *PathRouter) SetDefaultHandler(handler http.Handler) { | ||||
| 	p.defaultHandler = handler | ||||
| } | ||||
|  | ||||
| // SetDebugPrintMode sets the debug print mode | ||||
| func (p *PathRouter) SetDebugPrintMode(enable bool) { | ||||
| 	p.enableDebugPrint = enable | ||||
| } | ||||
|  | ||||
| // StartStaticCapture starts the static capture ingress | ||||
| func (p *PathRouter) RegisterStaticCaptureHandle(capture_ingress string, mux *http.ServeMux) { | ||||
| 	if !strings.HasSuffix(capture_ingress, "/") { | ||||
| 		capture_ingress = capture_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(capture_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.staticCaptureServeHTTP(w, r) | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| // staticCaptureServeHTTP serves the static capture path using user defined handler | ||||
| func (p *PathRouter) staticCaptureServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
| 	capturePath := r.Header.Get("X-Zoraxy-Capture") | ||||
| 	if capturePath != "" { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Printf("Using capture path: %s\n", capturePath) | ||||
| 		} | ||||
| 		originalURI := r.Header.Get("X-Zoraxy-Uri") | ||||
| 		r.URL.Path = originalURI | ||||
| 		if handler, ok := p.pathHandlers[capturePath]; ok { | ||||
| 			handler.ServeHTTP(w, r) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	p.defaultHandler.ServeHTTP(w, r) | ||||
| } | ||||
|  | ||||
| func (p *PathRouter) PrintRequestDebugMessage(r *http.Request) { | ||||
| 	if p.enableDebugPrint { | ||||
| 		fmt.Printf("Capture Request with path: %s \n\n**Request Headers** \n\n", r.URL.Path) | ||||
| 		keys := make([]string, 0, len(r.Header)) | ||||
| 		for key := range r.Header { | ||||
| 			keys = append(keys, key) | ||||
| 		} | ||||
| 		sort.Strings(keys) | ||||
| 		for _, key := range keys { | ||||
| 			for _, value := range r.Header[key] { | ||||
| 				fmt.Printf("%s: %s\n", key, value) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		fmt.Printf("\n\n**Request Details**\n\n") | ||||
| 		fmt.Printf("Method: %s\n", r.Method) | ||||
| 		fmt.Printf("URL: %s\n", r.URL.String()) | ||||
| 		fmt.Printf("Proto: %s\n", r.Proto) | ||||
| 		fmt.Printf("Host: %s\n", r.Host) | ||||
| 		fmt.Printf("RemoteAddr: %s\n", r.RemoteAddr) | ||||
| 		fmt.Printf("RequestURI: %s\n", r.RequestURI) | ||||
| 		fmt.Printf("ContentLength: %d\n", r.ContentLength) | ||||
| 		fmt.Printf("TransferEncoding: %v\n", r.TransferEncoding) | ||||
| 		fmt.Printf("Close: %v\n", r.Close) | ||||
| 		fmt.Printf("Form: %v\n", r.Form) | ||||
| 		fmt.Printf("PostForm: %v\n", r.PostForm) | ||||
| 		fmt.Printf("MultipartForm: %v\n", r.MultipartForm) | ||||
| 		fmt.Printf("Trailer: %v\n", r.Trailer) | ||||
| 		fmt.Printf("RemoteAddr: %s\n", r.RemoteAddr) | ||||
| 		fmt.Printf("RequestURI: %s\n", r.RequestURI) | ||||
|  | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										176
									
								
								example/plugins/upnp/mod/zoraxy_plugin/zoraxy_plugin.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								example/plugins/upnp/mod/zoraxy_plugin/zoraxy_plugin.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| /* | ||||
| 	Plugins Includes.go | ||||
|  | ||||
| 	This file is copied from Zoraxy source code | ||||
| 	You can always find the latest version under mod/plugins/includes.go | ||||
| 	Usually this file are backward compatible | ||||
| */ | ||||
|  | ||||
| type PluginType int | ||||
|  | ||||
| const ( | ||||
| 	PluginType_Router    PluginType = 0 //Router Plugin, used for handling / routing / forwarding traffic | ||||
| 	PluginType_Utilities PluginType = 1 //Utilities Plugin, used for utilities like Zerotier or Static Web Server that do not require interception with the dpcore | ||||
| ) | ||||
|  | ||||
| type StaticCaptureRule struct { | ||||
| 	CapturePath string `json:"capture_path"` | ||||
| 	//To be expanded | ||||
| } | ||||
|  | ||||
| type ControlStatusCode int | ||||
|  | ||||
| const ( | ||||
| 	ControlStatusCode_CAPTURED  ControlStatusCode = 280 //Traffic captured by plugin, ask Zoraxy not to process the traffic | ||||
| 	ControlStatusCode_UNHANDLED ControlStatusCode = 284 //Traffic not handled by plugin, ask Zoraxy to process the traffic | ||||
| 	ControlStatusCode_ERROR     ControlStatusCode = 580 //Error occurred while processing the traffic, ask Zoraxy to process the traffic and log the error | ||||
| ) | ||||
|  | ||||
| type SubscriptionEvent struct { | ||||
| 	EventName   string `json:"event_name"` | ||||
| 	EventSource string `json:"event_source"` | ||||
| 	Payload     string `json:"payload"` //Payload of the event, can be empty | ||||
| } | ||||
|  | ||||
| type RuntimeConstantValue struct { | ||||
| 	ZoraxyVersion    string `json:"zoraxy_version"` | ||||
| 	ZoraxyUUID       string `json:"zoraxy_uuid"` | ||||
| 	DevelopmentBuild bool   `json:"development_build"` //Whether the Zoraxy is a development build or not | ||||
| } | ||||
|  | ||||
| /* | ||||
| IntroSpect Payload | ||||
|  | ||||
| When the plugin is initialized with -introspect flag, | ||||
| the plugin shell return this payload as JSON and exit | ||||
| */ | ||||
| type IntroSpect struct { | ||||
| 	/* Plugin metadata */ | ||||
| 	ID            string     `json:"id"`             //Unique ID of your plugin, recommended using your own domain in reverse like com.yourdomain.pluginname | ||||
| 	Name          string     `json:"name"`           //Name of your plugin | ||||
| 	Author        string     `json:"author"`         //Author name of your plugin | ||||
| 	AuthorContact string     `json:"author_contact"` //Author contact of your plugin, like email | ||||
| 	Description   string     `json:"description"`    //Description of your plugin | ||||
| 	URL           string     `json:"url"`            //URL of your plugin | ||||
| 	Type          PluginType `json:"type"`           //Type of your plugin, Router(0) or Utilities(1) | ||||
| 	VersionMajor  int        `json:"version_major"`  //Major version of your plugin | ||||
| 	VersionMinor  int        `json:"version_minor"`  //Minor version of your plugin | ||||
| 	VersionPatch  int        `json:"version_patch"`  //Patch version of your plugin | ||||
|  | ||||
| 	/* | ||||
|  | ||||
| 		Endpoint Settings | ||||
|  | ||||
| 	*/ | ||||
|  | ||||
| 	/* | ||||
| 		Static Capture Settings | ||||
|  | ||||
| 		Once plugin is enabled these rules always applies to the enabled HTTP Proxy rule | ||||
| 		This is faster than dynamic capture, but less flexible | ||||
| 	*/ | ||||
| 	StaticCapturePaths   []StaticCaptureRule `json:"static_capture_paths"`   //Static capture paths of your plugin, see Zoraxy documentation for more details | ||||
| 	StaticCaptureIngress string              `json:"static_capture_ingress"` //Static capture ingress path of your plugin (e.g. /s_handler) | ||||
|  | ||||
| 	/* | ||||
| 		Dynamic Capture Settings | ||||
|  | ||||
| 		Once plugin is enabled, these rules will be captured and forward to plugin sniff | ||||
| 		if the plugin sniff returns 280, the traffic will be captured | ||||
| 		otherwise, the traffic will be forwarded to the next plugin | ||||
| 		This is slower than static capture, but more flexible | ||||
| 	*/ | ||||
| 	DynamicCaptureSniff   string `json:"dynamic_capture_sniff"`   //Dynamic capture sniff path of your plugin (e.g. /d_sniff) | ||||
| 	DynamicCaptureIngress string `json:"dynamic_capture_ingress"` //Dynamic capture ingress path of your plugin (e.g. /d_handler) | ||||
|  | ||||
| 	/* UI Path for your plugin */ | ||||
| 	UIPath string `json:"ui_path"` //UI path of your plugin (e.g. /ui), will proxy the whole subpath tree to Zoraxy Web UI as plugin UI | ||||
|  | ||||
| 	/* Subscriptions Settings */ | ||||
| 	SubscriptionPath    string            `json:"subscription_path"`    //Subscription event path of your plugin (e.g. /notifyme), a POST request with SubscriptionEvent as body will be sent to this path when the event is triggered | ||||
| 	SubscriptionsEvents map[string]string `json:"subscriptions_events"` //Subscriptions events of your plugin, see Zoraxy documentation for more details | ||||
| } | ||||
|  | ||||
| /* | ||||
| ServeIntroSpect Function | ||||
|  | ||||
| This function will check if the plugin is initialized with -introspect flag, | ||||
| if so, it will print the intro spect and exit | ||||
|  | ||||
| Place this function at the beginning of your plugin main function | ||||
| */ | ||||
| func ServeIntroSpect(pluginSpect *IntroSpect) { | ||||
| 	if len(os.Args) > 1 && os.Args[1] == "-introspect" { | ||||
| 		//Print the intro spect and exit | ||||
| 		jsonData, _ := json.MarshalIndent(pluginSpect, "", " ") | ||||
| 		fmt.Println(string(jsonData)) | ||||
| 		os.Exit(0) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* | ||||
| ConfigureSpec Payload | ||||
|  | ||||
| Zoraxy will start your plugin with -configure flag, | ||||
| the plugin shell read this payload as JSON and configure itself | ||||
| by the supplied values like starting a web server at given port | ||||
| that listens to 127.0.0.1:port | ||||
| */ | ||||
| type ConfigureSpec struct { | ||||
| 	Port         int                  `json:"port"`          //Port to listen | ||||
| 	RuntimeConst RuntimeConstantValue `json:"runtime_const"` //Runtime constant values | ||||
| 	//To be expanded | ||||
| } | ||||
|  | ||||
| /* | ||||
| RecvExecuteConfigureSpec Function | ||||
|  | ||||
| This function will read the configure spec from Zoraxy | ||||
| and return the ConfigureSpec object | ||||
|  | ||||
| Place this function after ServeIntroSpect function in your plugin main function | ||||
| */ | ||||
| func RecvConfigureSpec() (*ConfigureSpec, error) { | ||||
| 	for i, arg := range os.Args { | ||||
| 		if strings.HasPrefix(arg, "-configure=") { | ||||
| 			var configSpec ConfigureSpec | ||||
| 			if err := json.Unmarshal([]byte(arg[11:]), &configSpec); err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			return &configSpec, nil | ||||
| 		} else if arg == "-configure" { | ||||
| 			var configSpec ConfigureSpec | ||||
| 			var nextArg string | ||||
| 			if len(os.Args) > i+1 { | ||||
| 				nextArg = os.Args[i+1] | ||||
| 				if err := json.Unmarshal([]byte(nextArg), &configSpec); err != nil { | ||||
| 					return nil, err | ||||
| 				} | ||||
| 			} else { | ||||
| 				return nil, fmt.Errorf("No port specified after -configure flag") | ||||
| 			} | ||||
| 			return &configSpec, nil | ||||
| 		} | ||||
| 	} | ||||
| 	return nil, fmt.Errorf("No -configure flag found") | ||||
| } | ||||
|  | ||||
| /* | ||||
| ServeAndRecvSpec Function | ||||
|  | ||||
| This function will serve the intro spect and return the configure spec | ||||
| See the ServeIntroSpect and RecvConfigureSpec for more details | ||||
| */ | ||||
| func ServeAndRecvSpec(pluginSpect *IntroSpect) (*ConfigureSpec, error) { | ||||
| 	ServeIntroSpect(pluginSpect) | ||||
| 	return RecvConfigureSpec() | ||||
| } | ||||
							
								
								
									
										1
									
								
								example/plugins/upnp/upnp.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								example/plugins/upnp/upnp.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| {"ForwardRules":[],"Enabled":false} | ||||
							
								
								
									
										302
									
								
								example/plugins/upnp/www/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										302
									
								
								example/plugins/upnp/www/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,302 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
| <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <!-- CSRF token, if your plugin need to make POST request to backend --> | ||||
|     <meta name="zoraxy.csrf.Token" content="{{.csrfToken}}"> | ||||
|     <title>UPnP Port Forwarder | Zoraxy</title> | ||||
|     <link rel="stylesheet" href="/script/semantic/semantic.min.css"> | ||||
|     <script src="/script/jquery-3.6.0.min.js"></script> | ||||
|     <script src="/script/semantic/semantic.min.js"></script> | ||||
|     <script src="/script/utils.js"></script> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||
|     <link rel="stylesheet" href="/main.css"> | ||||
|     <style> | ||||
|         body { | ||||
|             background:none; | ||||
|         } | ||||
|     </style> | ||||
| </head> | ||||
| <body> | ||||
|     <!-- Dark theme script must be included after body tag--> | ||||
|     <link rel="stylesheet" href="/darktheme.css"> | ||||
|     <script src="/script/darktheme.js"></script> | ||||
|     <div id="upnpForwarder" class="standardContainer"> | ||||
|         <div id="upnpRouterNotFoundWarning" class="ui basic segment" style="display: none;"> | ||||
|             <h2>UPnP Port Forwarder</h2> | ||||
|             <p>Port forward using UPnP protocol, only works with some of the supported gateway routers</p> | ||||
|         </div> | ||||
|         <div class="ui basic segment"> | ||||
|             <div class="ui message"> | ||||
|                 <div class="header"><i class="yellow exclamation triangle icon"></i> UPnP Gateway Not Found</div> | ||||
|                 <p>No UPnP router found in network. Please ensure your router supports UPnP and is enabled.</p> | ||||
|                 <button id="retryBtn" onclick="searchUpnpRouter();" class="ui basic small button"><i class="green refresh icon"></i> Search again</button> | ||||
|             </div> | ||||
|             <div class="ui basic segment"> | ||||
|                 <h3 class="ui header">UPnP Port Forwarder</h3> | ||||
|                 <div class="ui toggle checkbox"> | ||||
|                     <input type="checkbox" id="upnpToggle" onchange="toggleUpnpState();"> | ||||
|                     <label>Enable UPnP Forwarding</label> | ||||
|                 </div> | ||||
|             </div> | ||||
|  | ||||
|             <div class="ui divider"></div> | ||||
|             <table class="ui celled table"> | ||||
|                 <thead> | ||||
|                     <tr> | ||||
|                         <th>Rule Name</th> | ||||
|                         <th>Forwarded Port</th> | ||||
|                         <th>Action</th> | ||||
|                     </tr> | ||||
|                 </thead> | ||||
|                 <tbody id="forwardList"> | ||||
|                     <tr> | ||||
|                         <td>Example Rule</td> | ||||
|                         <td>8080</td> | ||||
|                         <td> | ||||
|                             <button class="ui button">Edit</button> | ||||
|                             <button class="ui button">Remove</button> | ||||
|                         </td> | ||||
|                     </tr> | ||||
|                 </tbody> | ||||
|             </table> | ||||
|             <div class="ui divider"></div> | ||||
|             <div> | ||||
|                 <h4>Port Forward Rules</h4> | ||||
|                 <form class="ui form" id="addRuleForm"> | ||||
|                     <div class="field"> | ||||
|                         <label>Rule Name</label> | ||||
|                         <input type="text" name="ruleName" placeholder="Rule Name"> | ||||
|                     </div> | ||||
|                     <div class="field"> | ||||
|                         <label>Port</label> | ||||
|                         <input type="number" name="port" placeholder="Port" min="1" max="65535"> | ||||
|                     </div> | ||||
|                     <button onclick="handleAddForward(event);" class="ui small basic button"><i class="ui blue add icon"></i> Add Rule</button> | ||||
|                 </form> | ||||
|  | ||||
|             </div> | ||||
|  | ||||
|         </div> | ||||
|     </div> | ||||
|     <script> | ||||
|  | ||||
|         function toggleUpnpState() { | ||||
|             let isChecked = $("#upnpToggle").prop("checked"); | ||||
|             $.cjax({ | ||||
|                 url: './api/toggle', | ||||
|                 method: "POST", | ||||
|                 data: { | ||||
|                     enable: isChecked | ||||
|                 }, | ||||
|                 success: function(data) { | ||||
|                     if (data.error != undefined) { | ||||
|                         // Error | ||||
|                         parent.msgbox(data.error, false); | ||||
|                     } else { | ||||
|                         // Success | ||||
|                         parent.msgbox("UPnP Forwarding " + (isChecked ? "enabled" : "disabled"), true); | ||||
|                     } | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         function initUPnPEnableState(){ | ||||
|             $.cjax({ | ||||
|                 url: './api/enable', | ||||
|                 success: function(data) { | ||||
|                     if (data == true){ | ||||
|                         //Upnp forwarding enabled | ||||
|                         $("#upnpToggle").prop("checked", true); | ||||
|                     }else{ | ||||
|                         //Upnp forwarding disabled | ||||
|                         $("#upnpToggle").prop("checked", false); | ||||
|                     } | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|         initUPnPEnableState(); | ||||
|  | ||||
|         function searchUpnpRouter(){ | ||||
|             $("#retryBtn").addClass("loading"); | ||||
|             parent.msgbox("Searching for UPnP router (will take a few minutes)...", true); | ||||
|             $.cjax({ | ||||
|                 url: './api/usable', | ||||
|                 method: "POST", | ||||
|                 success: function(data) { | ||||
|                     if (data.error != undefined){ | ||||
|                         //Not found | ||||
|                         parent.msgbox("UPnP router not found", false); | ||||
|                     }else{ | ||||
|                         //Found | ||||
|                         parent.msgbox("UPnP router discovered", true); | ||||
|                     } | ||||
|                     initUpnpUsableState(); | ||||
|                     $("#retryBtn").removeClass("loading"); | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         //Check if UPnP is usable | ||||
|         function initUpnpUsableState(){ | ||||
|             $.cjax({ | ||||
|                 url: './api/usable', | ||||
|                 success: function(data) { | ||||
|                     if (data == true){ | ||||
|                         //Upnp router found in network, enable the page | ||||
|                         $('#upnpRouterNotFoundWarning').hide(); | ||||
|                     }else{ | ||||
|                         //No upnp router found in network, disable the page | ||||
|                         $('#upnpForwarder').addClass('disabled'); | ||||
|                         $('#upnpRouterNotFoundWarning').show(); | ||||
|                     } | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         function handleAddForward(event){ | ||||
|             event.preventDefault(); | ||||
|             let ruleName = $("#addRuleForm input[name='ruleName']").val(); | ||||
|             let port = $("#addRuleForm input[name='port']").val(); | ||||
|             if (ruleName == "" || port == ""){ | ||||
|                 parent.msgbox("Please fill in all fields", false); | ||||
|                 return; | ||||
|             } | ||||
|             $.cjax({ | ||||
|                 url: './api/forward', | ||||
|                 method: "POST", | ||||
|                 data: { | ||||
|                     name: ruleName, | ||||
|                     port: port | ||||
|                 }, | ||||
|                 success: function(data) { | ||||
|                     if (data.error != undefined){ | ||||
|                         //Error | ||||
|                         parent.msgbox(data.error, false); | ||||
|                     }else{ | ||||
|                         //Success | ||||
|                         parent.msgbox("Rule added successfully", true); | ||||
|                         initForwardList(); | ||||
|                     } | ||||
|  | ||||
|                     $("#addRuleForm input[name='ruleName']").val(''); | ||||
|                     $("#addRuleForm input[name='port']").val(''); | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         function handleRemoveForward(portNo){ | ||||
|             $.cjax({ | ||||
|                 url: './api/remove', | ||||
|                 method: "POST", | ||||
|                 data: { | ||||
|                     port: portNo | ||||
|                 }, | ||||
|                 success: function(data) { | ||||
|                     if (data.error != undefined){ | ||||
|                         //Error | ||||
|                         parent.msgbox(data.error, false); | ||||
|                     }else{ | ||||
|                         //Success | ||||
|                         parent.msgbox("Rule removed successfully", true); | ||||
|                         initForwardList(); | ||||
|                     } | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         function editForwardRule(row){ | ||||
|             let ruleName = $(row).closest('tr').find('td:eq(0)').text(); | ||||
|             let portNumber = $(row).closest('tr').find('td:eq(1)').text(); | ||||
|             $(row).closest('tr').html(` | ||||
|                 <td> | ||||
|                     <div class="ui fluid input"> | ||||
|                         <input type="text" value="${ruleName}" onkeypress="if(event.key === 'Enter') $(this).closest('tr').find('td:eq(1) input').focus();"> | ||||
|                     </div> | ||||
|                 </td> | ||||
|                 <td> | ||||
|                     <div class="ui fluid input"> | ||||
|                         <input type="number" value="${portNumber}" class="ui input" min="1" max="65535" onkeypress="if(event.key === 'Enter') saveForwardRule(this, '${portNumber}');"> | ||||
|                     </div> | ||||
|                 </td> | ||||
|                 <td> | ||||
|                     <button onclick="saveForwardRule(this, '${portNumber}');" class="ui basic small circular icon button"><i class="ui green save icon"></i></button> | ||||
|                     <button onclick="cancelEditForwardRule(this, '${ruleName}', '${portNumber}');" class="ui basic small circular icon button"><i class="ui cancel icon"></i></button> | ||||
|                 </td> | ||||
|             `); | ||||
|         } | ||||
|  | ||||
|         function cancelEditForwardRule(){ | ||||
|             initForwardList(); | ||||
|         } | ||||
|  | ||||
|         function saveForwardRule(row, portNo){ | ||||
|             let ruleName = $(row).closest('tr').find('td:eq(0) input').val(); | ||||
|             let newPortNo = $(row).closest('tr').find('td:eq(1) input').val(); | ||||
|             if (ruleName == "" || newPortNo == ""){ | ||||
|                 parent.msgbox("Please fill in all fields", false); | ||||
|                 return; | ||||
|             } | ||||
|             $.cjax({ | ||||
|                 url: './api/edit', | ||||
|                 method: "POST", | ||||
|                 data: { | ||||
|                     name: ruleName, | ||||
|                     port: newPortNo, | ||||
|                     oldPort: portNo | ||||
|                 }, | ||||
|                 success: function(data) { | ||||
|                     if (data.error != undefined){ | ||||
|                         //Error | ||||
|                         parent.msgbox(data.error, false); | ||||
|                     }else{ | ||||
|                         //Success | ||||
|                         parent.msgbox("Rule updated successfully", true); | ||||
|                         initForwardList(); | ||||
|                     } | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|          | ||||
|         //Load forward list | ||||
|         function initForwardList(){ | ||||
|             $("#forwardList").html('<tr><td colspan="3"> <i class="ui loading spinner icon"></i> Loading...</tr>'); | ||||
|             $.cjax({ | ||||
|                 url: './api/forward', | ||||
|                 success: function(data) { | ||||
|                     if (data.error != undefined){ | ||||
|                         //Error | ||||
|                         parent.msgbox(data.error, false); | ||||
|                     }else{ | ||||
|                         //Success | ||||
|                         $("#forwardList").empty(); | ||||
|                         let rows = ''; | ||||
|                         data.forEach(row => { | ||||
|                             $("#forwardList").append(` | ||||
|                                 <tr> | ||||
|                                     <td>${row.RuleName}</td> | ||||
|                                     <td>${row.PortNumber}</td> | ||||
|                                     <td> | ||||
|                                         <button onclick="editForwardRule(this);" class="ui basic small circular icon button"><i class="ui edit icon"></i></button> | ||||
|                                         <button onclick="handleRemoveForward(${row.PortNumber});" class="ui basic small red circular icon button"><i class="ui red trash icon"></i></button> | ||||
|                                     </td> | ||||
|                                 </tr> | ||||
|                             `); | ||||
|                         }); | ||||
|  | ||||
|                         if (data == null || data.length == 0){ | ||||
|                             //No rules | ||||
|                             $("#forwardList").append(` | ||||
|                                 <tr> | ||||
|                                     <td colspan="3"><i class="ui green check circle icon"></i> No running port forward rules</td> | ||||
|                                 </tr>`); | ||||
|                         } | ||||
|                     }  | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|         initUpnpUsableState(); | ||||
|         initForwardList(); | ||||
|     </script> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										11
									
								
								example/plugins/ztnc/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								example/plugins/ztnc/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| ## Global Area Network Plugin | ||||
|  | ||||
| This plugin implements a user interface for ZeroTier Network Controller in Zoraxy | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## License | ||||
|  | ||||
| AGPL | ||||
							
								
								
									
										11
									
								
								example/plugins/ztnc/go.mod
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								example/plugins/ztnc/go.mod
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| module aroz.org/zoraxy/ztnc | ||||
|  | ||||
| go 1.23.6 | ||||
|  | ||||
| require ( | ||||
| 	github.com/boltdb/bolt v1.3.1 | ||||
| 	github.com/syndtr/goleveldb v1.0.0 | ||||
| 	golang.org/x/sys v0.30.0 | ||||
| ) | ||||
|  | ||||
| require github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db // indirect | ||||
							
								
								
									
										30
									
								
								example/plugins/ztnc/go.sum
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								example/plugins/ztnc/go.sum
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| github.com/boltdb/bolt v1.3.1 h1:JQmyP4ZBrce+ZQu0dY660FMfatumYDLun9hBCUVIkF4= | ||||
| github.com/boltdb/bolt v1.3.1/go.mod h1:clJnj/oiGkjum5o1McbSZDSLxVThjynRyGBgiAx27Ps= | ||||
| github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | ||||
| github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db h1:woRePGFeVFfLKN/pOkfl+p/TAqKOfFu+7KPlMVpok/w= | ||||
| github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= | ||||
| github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI= | ||||
| github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||
| github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/ginkgo v1.7.0 h1:WSHQ+IS43OoUrWtD1/bbclrwK8TTH5hzp+umCiuxHgs= | ||||
| github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/gomega v1.4.3 h1:RE1xgDvH7imwFD45h+u2SgIfERHlS2yNG4DObb5BSKU= | ||||
| github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= | ||||
| github.com/syndtr/goleveldb v1.0.0 h1:fBdIW9lB4Iz0n9khmH8w27SJ3QEJ7+IgjPEwGSZiFdE= | ||||
| github.com/syndtr/goleveldb v1.0.0/go.mod h1:ZVVdQEZoIme9iO1Ch2Jdy24qqXrMMOU6lpPAyBWyWuQ= | ||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd h1:nTDtHvHSdCn1m6ITfMRqtOd/9+7a3s8RBNOZ3eYZzJA= | ||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.30.0 h1:QjkSwP/36a20jFYWkSue1YwXzLmsV5Gfq7Eiy72C1uc= | ||||
| golang.org/x/sys v0.30.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/text v0.3.0 h1:g61tztE5qeGQ89tm6NTjjM9VPIm088od1l6aSorWRWg= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4= | ||||
| gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | ||||
| gopkg.in/yaml.v2 v2.2.1 h1:mUhvW9EsL+naU5Q3cakzfE91YhliOondGd6ZrsDBHQE= | ||||
| gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
							
								
								
									
										
											BIN
										
									
								
								example/plugins/ztnc/icon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								example/plugins/ztnc/icon.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 7.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								example/plugins/ztnc/icon.psd
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								example/plugins/ztnc/icon.psd
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										83
									
								
								example/plugins/ztnc/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								example/plugins/ztnc/main.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,83 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"strconv" | ||||
|  | ||||
| 	"embed" | ||||
|  | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database" | ||||
| 	"aroz.org/zoraxy/ztnc/mod/ganserv" | ||||
| 	plugin "aroz.org/zoraxy/ztnc/mod/zoraxy_plugin" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	PLUGIN_ID       = "org.aroz.zoraxy.ztnc" | ||||
| 	UI_RELPATH      = "/ui" | ||||
| 	EMBED_FS_ROOT   = "/web" | ||||
| 	DB_FILE_PATH    = "ztnc.db" | ||||
| 	AUTH_TOKEN_PATH = "./authtoken.secret" | ||||
| ) | ||||
|  | ||||
| //go:embed web/* | ||||
| var content embed.FS | ||||
|  | ||||
| var ( | ||||
| 	sysdb      *database.Database | ||||
| 	ganManager *ganserv.NetworkManager | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
| 	// Serve the plugin intro spect | ||||
| 	runtimeCfg, err := plugin.ServeAndRecvSpec(&plugin.IntroSpect{ | ||||
| 		ID:            PLUGIN_ID, | ||||
| 		Name:          "ztnc", | ||||
| 		Author:        "aroz.org", | ||||
| 		AuthorContact: "zoraxy.aroz.org", | ||||
| 		Description:   "UI for ZeroTier Network Controller", | ||||
| 		URL:           "https://zoraxy.aroz.org", | ||||
| 		Type:          plugin.PluginType_Utilities, | ||||
| 		VersionMajor:  1, | ||||
| 		VersionMinor:  0, | ||||
| 		VersionPatch:  0, | ||||
|  | ||||
| 		// As this is a utility plugin, we don't need to capture any traffic | ||||
| 		// but only serve the UI, so we set the UI (relative to the plugin path) to "/ui/" to match the HTTP Handler | ||||
| 		UIPath: UI_RELPATH, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		//Terminate or enter standalone mode here | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	// Create a new PluginEmbedUIRouter that will serve the UI from web folder | ||||
| 	uiRouter := plugin.NewPluginEmbedUIRouter(PLUGIN_ID, &content, EMBED_FS_ROOT, UI_RELPATH) | ||||
| 	uiRouter.EnableDebug = true | ||||
|  | ||||
| 	// Register the shutdown handler | ||||
| 	uiRouter.RegisterTerminateHandler(func() { | ||||
| 		// Do cleanup here if needed | ||||
| 		if sysdb != nil { | ||||
| 			sysdb.Close() | ||||
| 		} | ||||
| 		fmt.Println("ztnc Exited") | ||||
| 	}, nil) | ||||
|  | ||||
| 	// This will serve the index.html file embedded in the binary | ||||
| 	targetHandler := uiRouter.Handler() | ||||
| 	http.Handle(UI_RELPATH+"/", targetHandler) | ||||
|  | ||||
| 	// Start the GAN Network Controller | ||||
| 	err = startGanNetworkController() | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	// Initiate the API endpoints | ||||
| 	initApiEndpoints() | ||||
|  | ||||
| 	// Start the HTTP server, only listen to loopback interface | ||||
| 	fmt.Println("Plugin UI server started at http://127.0.0.1:" + strconv.Itoa(runtimeCfg.Port) + UI_RELPATH) | ||||
| 	http.ListenAndServe("127.0.0.1:"+strconv.Itoa(runtimeCfg.Port), nil) | ||||
| } | ||||
							
								
								
									
										146
									
								
								example/plugins/ztnc/mod/database/database.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										146
									
								
								example/plugins/ztnc/mod/database/database.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,146 @@ | ||||
| package database | ||||
|  | ||||
| /* | ||||
| 	ArOZ Online Database Access Module | ||||
| 	author: tobychui | ||||
|  | ||||
| 	This is an improved Object oriented base solution to the original | ||||
| 	aroz online database script. | ||||
| */ | ||||
|  | ||||
| import ( | ||||
| 	"log" | ||||
| 	"runtime" | ||||
|  | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database/dbinc" | ||||
| ) | ||||
|  | ||||
| type Database struct { | ||||
| 	Db          interface{} //This will be nil on openwrt, leveldb.DB on x64 platforms or bolt.DB on other platforms | ||||
| 	BackendType dbinc.BackendType | ||||
| 	Backend     dbinc.Backend | ||||
| } | ||||
|  | ||||
| func NewDatabase(dbfile string, backendType dbinc.BackendType) (*Database, error) { | ||||
| 	if runtime.GOARCH == "riscv64" { | ||||
| 		log.Println("RISCV hardware detected, ignoring the backend type and using FS emulated database") | ||||
| 	} | ||||
| 	return newDatabase(dbfile, backendType) | ||||
| } | ||||
|  | ||||
| // Get the recommended backend type for the current system | ||||
| func GetRecommendedBackendType() dbinc.BackendType { | ||||
| 	//Check if the system is running on RISCV hardware | ||||
| 	if runtime.GOARCH == "riscv64" { | ||||
| 		//RISCV hardware, currently only support FS emulated database | ||||
| 		return dbinc.BackendFSOnly | ||||
| 	} else if runtime.GOOS == "windows" || (runtime.GOOS == "linux" && runtime.GOARCH == "amd64") { | ||||
| 		//Powerful hardware | ||||
| 		return dbinc.BackendBoltDB | ||||
| 		//return dbinc.BackendLevelDB | ||||
| 	} | ||||
|  | ||||
| 	//Default to BoltDB, the safest option | ||||
| 	return dbinc.BackendBoltDB | ||||
| } | ||||
|  | ||||
| /* | ||||
| 	Create / Drop a table | ||||
| 	Usage: | ||||
| 	err := sysdb.NewTable("MyTable") | ||||
| 	err := sysdb.DropTable("MyTable") | ||||
| */ | ||||
|  | ||||
| // Create a new table | ||||
| func (d *Database) NewTable(tableName string) error { | ||||
| 	return d.newTable(tableName) | ||||
| } | ||||
|  | ||||
| // Check is table exists | ||||
| func (d *Database) TableExists(tableName string) bool { | ||||
| 	return d.tableExists(tableName) | ||||
| } | ||||
|  | ||||
| // Drop the given table | ||||
| func (d *Database) DropTable(tableName string) error { | ||||
| 	return d.dropTable(tableName) | ||||
| } | ||||
|  | ||||
| /* | ||||
| Write to database with given tablename and key. Example Usage: | ||||
|  | ||||
| 	type demo struct{ | ||||
| 		content string | ||||
| 	} | ||||
|  | ||||
| 	thisDemo := demo{ | ||||
| 		content: "Hello World", | ||||
| 	} | ||||
|  | ||||
| err := sysdb.Write("MyTable", "username/message",thisDemo); | ||||
| */ | ||||
| func (d *Database) Write(tableName string, key string, value interface{}) error { | ||||
| 	return d.write(tableName, key, value) | ||||
| } | ||||
|  | ||||
| /* | ||||
| 	Read from database and assign the content to a given datatype. Example Usage: | ||||
|  | ||||
| 	type demo struct{ | ||||
| 		content string | ||||
| 	} | ||||
| 	thisDemo := new(demo) | ||||
| 	err := sysdb.Read("MyTable", "username/message",&thisDemo); | ||||
| */ | ||||
|  | ||||
| func (d *Database) Read(tableName string, key string, assignee interface{}) error { | ||||
| 	return d.read(tableName, key, assignee) | ||||
| } | ||||
|  | ||||
| /* | ||||
| Check if a key exists in the database table given tablename and key | ||||
|  | ||||
| 	if sysdb.KeyExists("MyTable", "username/message"){ | ||||
| 		log.Println("Key exists") | ||||
| 	} | ||||
| */ | ||||
| func (d *Database) KeyExists(tableName string, key string) bool { | ||||
| 	return d.keyExists(tableName, key) | ||||
| } | ||||
|  | ||||
| /* | ||||
| Delete a value from the database table given tablename and key | ||||
|  | ||||
| err := sysdb.Delete("MyTable", "username/message"); | ||||
| */ | ||||
| func (d *Database) Delete(tableName string, key string) error { | ||||
| 	return d.delete(tableName, key) | ||||
| } | ||||
|  | ||||
| /* | ||||
| 	//List table example usage | ||||
| 	//Assume the value is stored as a struct named "groupstruct" | ||||
|  | ||||
| 	entries, err := sysdb.ListTable("test") | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
| 	for _, keypairs := range entries{ | ||||
| 		log.Println(string(keypairs[0])) | ||||
| 		group := new(groupstruct) | ||||
| 		json.Unmarshal(keypairs[1], &group) | ||||
| 		log.Println(group); | ||||
| 	} | ||||
|  | ||||
| */ | ||||
|  | ||||
| func (d *Database) ListTable(tableName string) ([][][]byte, error) { | ||||
| 	return d.listTable(tableName) | ||||
| } | ||||
|  | ||||
| /* | ||||
| Close the database connection | ||||
| */ | ||||
| func (d *Database) Close() { | ||||
| 	d.close() | ||||
| } | ||||
							
								
								
									
										70
									
								
								example/plugins/ztnc/mod/database/database_core.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								example/plugins/ztnc/mod/database/database_core.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | ||||
| //go:build !mipsle && !riscv64 | ||||
| // +build !mipsle,!riscv64 | ||||
|  | ||||
| package database | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
|  | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database/dbbolt" | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database/dbinc" | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database/dbleveldb" | ||||
| ) | ||||
|  | ||||
| func newDatabase(dbfile string, backendType dbinc.BackendType) (*Database, error) { | ||||
| 	if backendType == dbinc.BackendFSOnly { | ||||
| 		return nil, errors.New("Unsupported backend type for this platform") | ||||
| 	} | ||||
|  | ||||
| 	if backendType == dbinc.BackendLevelDB { | ||||
| 		db, err := dbleveldb.NewDB(dbfile) | ||||
| 		return &Database{ | ||||
| 			Db:          nil, | ||||
| 			BackendType: backendType, | ||||
| 			Backend:     db, | ||||
| 		}, err | ||||
| 	} | ||||
|  | ||||
| 	db, err := dbbolt.NewBoltDatabase(dbfile) | ||||
| 	return &Database{ | ||||
| 		Db:          nil, | ||||
| 		BackendType: backendType, | ||||
| 		Backend:     db, | ||||
| 	}, err | ||||
| } | ||||
|  | ||||
| func (d *Database) newTable(tableName string) error { | ||||
| 	return d.Backend.NewTable(tableName) | ||||
| } | ||||
|  | ||||
| func (d *Database) tableExists(tableName string) bool { | ||||
| 	return d.Backend.TableExists(tableName) | ||||
| } | ||||
|  | ||||
| func (d *Database) dropTable(tableName string) error { | ||||
| 	return d.Backend.DropTable(tableName) | ||||
| } | ||||
|  | ||||
| func (d *Database) write(tableName string, key string, value interface{}) error { | ||||
| 	return d.Backend.Write(tableName, key, value) | ||||
| } | ||||
|  | ||||
| func (d *Database) read(tableName string, key string, assignee interface{}) error { | ||||
| 	return d.Backend.Read(tableName, key, assignee) | ||||
| } | ||||
|  | ||||
| func (d *Database) keyExists(tableName string, key string) bool { | ||||
| 	return d.Backend.KeyExists(tableName, key) | ||||
| } | ||||
|  | ||||
| func (d *Database) delete(tableName string, key string) error { | ||||
| 	return d.Backend.Delete(tableName, key) | ||||
| } | ||||
|  | ||||
| func (d *Database) listTable(tableName string) ([][][]byte, error) { | ||||
| 	return d.Backend.ListTable(tableName) | ||||
| } | ||||
|  | ||||
| func (d *Database) close() { | ||||
| 	d.Backend.Close() | ||||
| } | ||||
							
								
								
									
										196
									
								
								example/plugins/ztnc/mod/database/database_openwrt.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										196
									
								
								example/plugins/ztnc/mod/database/database_openwrt.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,196 @@ | ||||
| //go:build mipsle || riscv64 | ||||
| // +build mipsle riscv64 | ||||
|  | ||||
| package database | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"strings" | ||||
|  | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database/dbinc" | ||||
| ) | ||||
|  | ||||
| /* | ||||
| 	OpenWRT or RISCV backend | ||||
|  | ||||
| 	For OpenWRT or RISCV platform, we will use the filesystem as the database backend | ||||
| 	as boltdb or leveldb is not supported on these platforms, including boltDB and LevelDB | ||||
| 	in conditional compilation will create a build error on these platforms | ||||
| */ | ||||
|  | ||||
| func newDatabase(dbfile string, backendType dbinc.BackendType) (*Database, error) { | ||||
| 	dbRootPath := filepath.ToSlash(filepath.Clean(dbfile)) | ||||
| 	dbRootPath = "fsdb/" + dbRootPath | ||||
| 	err := os.MkdirAll(dbRootPath, 0755) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	log.Println("Filesystem Emulated Key-value Database Service Started: " + dbRootPath) | ||||
| 	return &Database{ | ||||
| 		Db:          dbRootPath, | ||||
| 		BackendType: dbinc.BackendFSOnly, | ||||
| 		Backend:     nil, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| func (d *Database) dump(filename string) ([]string, error) { | ||||
| 	//Get all file objects from root | ||||
| 	rootfiles, err := filepath.Glob(filepath.Join(d.Db.(string), "/*")) | ||||
| 	if err != nil { | ||||
| 		return []string{}, err | ||||
| 	} | ||||
|  | ||||
| 	//Filter out the folders | ||||
| 	rootFolders := []string{} | ||||
| 	for _, file := range rootfiles { | ||||
| 		if !isDirectory(file) { | ||||
| 			rootFolders = append(rootFolders, filepath.Base(file)) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return rootFolders, nil | ||||
| } | ||||
|  | ||||
| func (d *Database) newTable(tableName string) error { | ||||
|  | ||||
| 	tablePath := filepath.Join(d.Db.(string), filepath.Base(tableName)) | ||||
| 	if !fileExists(tablePath) { | ||||
| 		return os.MkdirAll(tablePath, 0755) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (d *Database) tableExists(tableName string) bool { | ||||
| 	tablePath := filepath.Join(d.Db.(string), filepath.Base(tableName)) | ||||
| 	if _, err := os.Stat(tablePath); errors.Is(err, os.ErrNotExist) { | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	if !isDirectory(tablePath) { | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	return true | ||||
| } | ||||
|  | ||||
| func (d *Database) dropTable(tableName string) error { | ||||
|  | ||||
| 	tablePath := filepath.Join(d.Db.(string), filepath.Base(tableName)) | ||||
| 	if d.tableExists(tableName) { | ||||
| 		return os.RemoveAll(tablePath) | ||||
| 	} else { | ||||
| 		return errors.New("table not exists") | ||||
| 	} | ||||
|  | ||||
| } | ||||
|  | ||||
| func (d *Database) write(tableName string, key string, value interface{}) error { | ||||
|  | ||||
| 	tablePath := filepath.Join(d.Db.(string), filepath.Base(tableName)) | ||||
| 	js, err := json.Marshal(value) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	key = strings.ReplaceAll(key, "/", "-SLASH_SIGN-") | ||||
|  | ||||
| 	return os.WriteFile(filepath.Join(tablePath, key+".entry"), js, 0755) | ||||
| } | ||||
|  | ||||
| func (d *Database) read(tableName string, key string, assignee interface{}) error { | ||||
| 	if !d.keyExists(tableName, key) { | ||||
| 		return errors.New("key not exists") | ||||
| 	} | ||||
|  | ||||
| 	key = strings.ReplaceAll(key, "/", "-SLASH_SIGN-") | ||||
|  | ||||
| 	tablePath := filepath.Join(d.Db.(string), filepath.Base(tableName)) | ||||
| 	entryPath := filepath.Join(tablePath, key+".entry") | ||||
| 	content, err := os.ReadFile(entryPath) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	err = json.Unmarshal(content, &assignee) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (d *Database) keyExists(tableName string, key string) bool { | ||||
| 	key = strings.ReplaceAll(key, "/", "-SLASH_SIGN-") | ||||
| 	tablePath := filepath.Join(d.Db.(string), filepath.Base(tableName)) | ||||
| 	entryPath := filepath.Join(tablePath, key+".entry") | ||||
| 	return fileExists(entryPath) | ||||
| } | ||||
|  | ||||
| func (d *Database) delete(tableName string, key string) error { | ||||
|  | ||||
| 	if !d.keyExists(tableName, key) { | ||||
| 		return errors.New("key not exists") | ||||
| 	} | ||||
| 	key = strings.ReplaceAll(key, "/", "-SLASH_SIGN-") | ||||
| 	tablePath := filepath.Join(d.Db.(string), filepath.Base(tableName)) | ||||
| 	entryPath := filepath.Join(tablePath, key+".entry") | ||||
|  | ||||
| 	return os.Remove(entryPath) | ||||
| } | ||||
|  | ||||
| func (d *Database) listTable(tableName string) ([][][]byte, error) { | ||||
| 	if !d.tableExists(tableName) { | ||||
| 		return [][][]byte{}, errors.New("table not exists") | ||||
| 	} | ||||
| 	tablePath := filepath.Join(d.Db.(string), filepath.Base(tableName)) | ||||
| 	entries, err := filepath.Glob(filepath.Join(tablePath, "/*.entry")) | ||||
| 	if err != nil { | ||||
| 		return [][][]byte{}, err | ||||
| 	} | ||||
|  | ||||
| 	var results [][][]byte = [][][]byte{} | ||||
| 	for _, entry := range entries { | ||||
| 		if !isDirectory(entry) { | ||||
| 			//Read it | ||||
| 			key := filepath.Base(entry) | ||||
| 			key = strings.TrimSuffix(key, filepath.Ext(key)) | ||||
| 			key = strings.ReplaceAll(key, "-SLASH_SIGN-", "/") | ||||
|  | ||||
| 			bkey := []byte(key) | ||||
| 			bval := []byte("") | ||||
| 			c, err := os.ReadFile(entry) | ||||
| 			if err != nil { | ||||
| 				break | ||||
| 			} | ||||
|  | ||||
| 			bval = c | ||||
| 			results = append(results, [][]byte{bkey, bval}) | ||||
| 		} | ||||
| 	} | ||||
| 	return results, nil | ||||
| } | ||||
|  | ||||
| func (d *Database) close() { | ||||
| 	//Nothing to close as it is file system | ||||
| } | ||||
|  | ||||
| func isDirectory(path string) bool { | ||||
| 	fileInfo, err := os.Stat(path) | ||||
| 	if err != nil { | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	return fileInfo.IsDir() | ||||
| } | ||||
|  | ||||
| func fileExists(name string) bool { | ||||
| 	_, err := os.Stat(name) | ||||
| 	if err == nil { | ||||
| 		return true | ||||
| 	} | ||||
| 	if errors.Is(err, os.ErrNotExist) { | ||||
| 		return false | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
							
								
								
									
										141
									
								
								example/plugins/ztnc/mod/database/dbbolt/dbbolt.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								example/plugins/ztnc/mod/database/dbbolt/dbbolt.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,141 @@ | ||||
| package dbbolt | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
|  | ||||
| 	"github.com/boltdb/bolt" | ||||
| ) | ||||
|  | ||||
| type Database struct { | ||||
| 	Db interface{} //This is the bolt database object | ||||
| } | ||||
|  | ||||
| func NewBoltDatabase(dbfile string) (*Database, error) { | ||||
| 	db, err := bolt.Open(dbfile, 0600, nil) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return &Database{ | ||||
| 		Db: db, | ||||
| 	}, err | ||||
| } | ||||
|  | ||||
| // Create a new table | ||||
| func (d *Database) NewTable(tableName string) error { | ||||
| 	err := d.Db.(*bolt.DB).Update(func(tx *bolt.Tx) error { | ||||
| 		_, err := tx.CreateBucketIfNotExists([]byte(tableName)) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
|  | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| // Check is table exists | ||||
| func (d *Database) TableExists(tableName string) bool { | ||||
| 	return d.Db.(*bolt.DB).View(func(tx *bolt.Tx) error { | ||||
| 		b := tx.Bucket([]byte(tableName)) | ||||
| 		if b == nil { | ||||
| 			return errors.New("table not exists") | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) == nil | ||||
| } | ||||
|  | ||||
| // Drop the given table | ||||
| func (d *Database) DropTable(tableName string) error { | ||||
| 	err := d.Db.(*bolt.DB).Update(func(tx *bolt.Tx) error { | ||||
| 		err := tx.DeleteBucket([]byte(tableName)) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| // Write to table | ||||
| func (d *Database) Write(tableName string, key string, value interface{}) error { | ||||
| 	jsonString, err := json.Marshal(value) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = d.Db.(*bolt.DB).Update(func(tx *bolt.Tx) error { | ||||
| 		_, err := tx.CreateBucketIfNotExists([]byte(tableName)) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		b := tx.Bucket([]byte(tableName)) | ||||
| 		err = b.Put([]byte(key), jsonString) | ||||
| 		return err | ||||
| 	}) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (d *Database) Read(tableName string, key string, assignee interface{}) error { | ||||
| 	err := d.Db.(*bolt.DB).View(func(tx *bolt.Tx) error { | ||||
| 		b := tx.Bucket([]byte(tableName)) | ||||
| 		v := b.Get([]byte(key)) | ||||
| 		json.Unmarshal(v, &assignee) | ||||
| 		return nil | ||||
| 	}) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (d *Database) KeyExists(tableName string, key string) bool { | ||||
| 	resultIsNil := false | ||||
| 	if !d.TableExists(tableName) { | ||||
| 		//Table not exists. Do not proceed accessing key | ||||
| 		//log.Println("[DB] ERROR: Requesting key from table that didn't exist!!!") | ||||
| 		return false | ||||
| 	} | ||||
| 	err := d.Db.(*bolt.DB).View(func(tx *bolt.Tx) error { | ||||
| 		b := tx.Bucket([]byte(tableName)) | ||||
| 		v := b.Get([]byte(key)) | ||||
| 		if v == nil { | ||||
| 			resultIsNil = true | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return false | ||||
| 	} else { | ||||
| 		if resultIsNil { | ||||
| 			return false | ||||
| 		} else { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (d *Database) Delete(tableName string, key string) error { | ||||
| 	err := d.Db.(*bolt.DB).Update(func(tx *bolt.Tx) error { | ||||
| 		tx.Bucket([]byte(tableName)).Delete([]byte(key)) | ||||
| 		return nil | ||||
| 	}) | ||||
|  | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (d *Database) ListTable(tableName string) ([][][]byte, error) { | ||||
| 	var results [][][]byte | ||||
| 	err := d.Db.(*bolt.DB).View(func(tx *bolt.Tx) error { | ||||
| 		b := tx.Bucket([]byte(tableName)) | ||||
| 		c := b.Cursor() | ||||
|  | ||||
| 		for k, v := c.First(); k != nil; k, v = c.Next() { | ||||
| 			results = append(results, [][]byte{k, v}) | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
| 	return results, err | ||||
| } | ||||
|  | ||||
| func (d *Database) Close() { | ||||
| 	d.Db.(*bolt.DB).Close() | ||||
| } | ||||
							
								
								
									
										67
									
								
								example/plugins/ztnc/mod/database/dbbolt/dbbolt_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								example/plugins/ztnc/mod/database/dbbolt/dbbolt_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | ||||
| package dbbolt_test | ||||
|  | ||||
| import ( | ||||
| 	"os" | ||||
| 	"testing" | ||||
|  | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database/dbbolt" | ||||
| ) | ||||
|  | ||||
| func TestNewBoltDatabase(t *testing.T) { | ||||
| 	dbfile := "test.db" | ||||
| 	defer os.Remove(dbfile) | ||||
|  | ||||
| 	db, err := dbbolt.NewBoltDatabase(dbfile) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new Bolt database: %v", err) | ||||
| 	} | ||||
| 	defer db.Close() | ||||
|  | ||||
| 	if db.Db == nil { | ||||
| 		t.Fatalf("Expected non-nil database object") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNewTable(t *testing.T) { | ||||
| 	dbfile := "test.db" | ||||
| 	defer os.Remove(dbfile) | ||||
|  | ||||
| 	db, err := dbbolt.NewBoltDatabase(dbfile) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new Bolt database: %v", err) | ||||
| 	} | ||||
| 	defer db.Close() | ||||
|  | ||||
| 	err = db.NewTable("testTable") | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new table: %v", err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestTableExists(t *testing.T) { | ||||
| 	dbfile := "test.db" | ||||
| 	defer os.Remove(dbfile) | ||||
|  | ||||
| 	db, err := dbbolt.NewBoltDatabase(dbfile) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new Bolt database: %v", err) | ||||
| 	} | ||||
| 	defer db.Close() | ||||
|  | ||||
| 	tableName := "testTable" | ||||
| 	err = db.NewTable(tableName) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new table: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	exists := db.TableExists(tableName) | ||||
| 	if !exists { | ||||
| 		t.Fatalf("Expected table %s to exist", tableName) | ||||
| 	} | ||||
|  | ||||
| 	nonExistentTable := "nonExistentTable" | ||||
| 	exists = db.TableExists(nonExistentTable) | ||||
| 	if exists { | ||||
| 		t.Fatalf("Expected table %s to not exist", nonExistentTable) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										39
									
								
								example/plugins/ztnc/mod/database/dbinc/dbinc.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								example/plugins/ztnc/mod/database/dbinc/dbinc.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| package dbinc | ||||
|  | ||||
| /* | ||||
| 	dbinc is the interface for all database backend | ||||
| */ | ||||
| type BackendType int | ||||
|  | ||||
| const ( | ||||
| 	BackendBoltDB  BackendType = iota //Default backend | ||||
| 	BackendFSOnly                     //OpenWRT or RISCV backend | ||||
| 	BackendLevelDB                    //LevelDB backend | ||||
|  | ||||
| 	BackEndAuto = BackendBoltDB | ||||
| ) | ||||
|  | ||||
| type Backend interface { | ||||
| 	NewTable(tableName string) error | ||||
| 	TableExists(tableName string) bool | ||||
| 	DropTable(tableName string) error | ||||
| 	Write(tableName string, key string, value interface{}) error | ||||
| 	Read(tableName string, key string, assignee interface{}) error | ||||
| 	KeyExists(tableName string, key string) bool | ||||
| 	Delete(tableName string, key string) error | ||||
| 	ListTable(tableName string) ([][][]byte, error) | ||||
| 	Close() | ||||
| } | ||||
|  | ||||
| func (b BackendType) String() string { | ||||
| 	switch b { | ||||
| 	case BackendBoltDB: | ||||
| 		return "BoltDB" | ||||
| 	case BackendFSOnly: | ||||
| 		return "File System Emulated Key-Value Store" | ||||
| 	case BackendLevelDB: | ||||
| 		return "LevelDB" | ||||
| 	default: | ||||
| 		return "Unknown" | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										152
									
								
								example/plugins/ztnc/mod/database/dbleveldb/dbleveldb.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										152
									
								
								example/plugins/ztnc/mod/database/dbleveldb/dbleveldb.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,152 @@ | ||||
| package dbleveldb | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"log" | ||||
| 	"path/filepath" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database/dbinc" | ||||
| 	"github.com/syndtr/goleveldb/leveldb" | ||||
| 	"github.com/syndtr/goleveldb/leveldb/util" | ||||
| ) | ||||
|  | ||||
| // Ensure the DB struct implements the Backend interface | ||||
| var _ dbinc.Backend = (*DB)(nil) | ||||
|  | ||||
| type DB struct { | ||||
| 	db               *leveldb.DB | ||||
| 	Table            sync.Map      //For emulating table creation | ||||
| 	batch            leveldb.Batch //Batch write | ||||
| 	writeFlushTicker *time.Ticker  //Ticker for flushing data into disk | ||||
| 	writeFlushStop   chan bool     //Stop channel for write flush ticker | ||||
| } | ||||
|  | ||||
| func NewDB(path string) (*DB, error) { | ||||
| 	//If the path is not a directory (e.g. /tmp/dbfile.db), convert the filename to directory | ||||
| 	if filepath.Ext(path) != "" { | ||||
| 		path = strings.ReplaceAll(path, ".", "_") | ||||
| 	} | ||||
|  | ||||
| 	db, err := leveldb.OpenFile(path, nil) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	thisDB := &DB{ | ||||
| 		db:    db, | ||||
| 		Table: sync.Map{}, | ||||
| 		batch: leveldb.Batch{}, | ||||
| 	} | ||||
|  | ||||
| 	//Create a ticker to flush data into disk every 1 seconds | ||||
| 	writeFlushTicker := time.NewTicker(1 * time.Second) | ||||
| 	writeFlushStop := make(chan bool) | ||||
| 	go func() { | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-writeFlushTicker.C: | ||||
| 				if thisDB.batch.Len() == 0 { | ||||
| 					//No flushing needed | ||||
| 					continue | ||||
| 				} | ||||
| 				err = db.Write(&thisDB.batch, nil) | ||||
| 				if err != nil { | ||||
| 					log.Println("[LevelDB] Failed to flush data into disk: ", err) | ||||
| 				} | ||||
| 				thisDB.batch.Reset() | ||||
| 			case <-writeFlushStop: | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	thisDB.writeFlushTicker = writeFlushTicker | ||||
| 	thisDB.writeFlushStop = writeFlushStop | ||||
|  | ||||
| 	return thisDB, nil | ||||
| } | ||||
|  | ||||
| func (d *DB) NewTable(tableName string) error { | ||||
| 	//Create a table entry in the sync.Map | ||||
| 	d.Table.Store(tableName, true) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (d *DB) TableExists(tableName string) bool { | ||||
| 	_, ok := d.Table.Load(tableName) | ||||
| 	return ok | ||||
| } | ||||
|  | ||||
| func (d *DB) DropTable(tableName string) error { | ||||
| 	d.Table.Delete(tableName) | ||||
| 	iter := d.db.NewIterator(nil, nil) | ||||
| 	defer iter.Release() | ||||
|  | ||||
| 	for iter.Next() { | ||||
| 		key := iter.Key() | ||||
| 		if filepath.Dir(string(key)) == tableName { | ||||
| 			err := d.db.Delete(key, nil) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (d *DB) Write(tableName string, key string, value interface{}) error { | ||||
| 	data, err := json.Marshal(value) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	d.batch.Put([]byte(filepath.ToSlash(filepath.Join(tableName, key))), data) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (d *DB) Read(tableName string, key string, assignee interface{}) error { | ||||
| 	data, err := d.db.Get([]byte(filepath.ToSlash(filepath.Join(tableName, key))), nil) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return json.Unmarshal(data, assignee) | ||||
| } | ||||
|  | ||||
| func (d *DB) KeyExists(tableName string, key string) bool { | ||||
| 	_, err := d.db.Get([]byte(filepath.ToSlash(filepath.Join(tableName, key))), nil) | ||||
| 	return err == nil | ||||
| } | ||||
|  | ||||
| func (d *DB) Delete(tableName string, key string) error { | ||||
| 	return d.db.Delete([]byte(filepath.ToSlash(filepath.Join(tableName, key))), nil) | ||||
| } | ||||
|  | ||||
| func (d *DB) ListTable(tableName string) ([][][]byte, error) { | ||||
| 	iter := d.db.NewIterator(util.BytesPrefix([]byte(tableName+"/")), nil) | ||||
| 	defer iter.Release() | ||||
|  | ||||
| 	var result [][][]byte | ||||
| 	for iter.Next() { | ||||
| 		key := iter.Key() | ||||
| 		//The key contains the table name as prefix. Trim it before returning | ||||
| 		value := iter.Value() | ||||
| 		result = append(result, [][]byte{[]byte(strings.TrimPrefix(string(key), tableName+"/")), value}) | ||||
| 	} | ||||
|  | ||||
| 	err := iter.Error() | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return result, nil | ||||
| } | ||||
|  | ||||
| func (d *DB) Close() { | ||||
| 	//Write the remaining data in batch back into disk | ||||
| 	d.writeFlushStop <- true | ||||
| 	d.writeFlushTicker.Stop() | ||||
| 	d.db.Write(&d.batch, nil) | ||||
| 	d.db.Close() | ||||
| } | ||||
							
								
								
									
										141
									
								
								example/plugins/ztnc/mod/database/dbleveldb/dbleveldb_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								example/plugins/ztnc/mod/database/dbleveldb/dbleveldb_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,141 @@ | ||||
| package dbleveldb_test | ||||
|  | ||||
| import ( | ||||
| 	"os" | ||||
| 	"testing" | ||||
|  | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database/dbleveldb" | ||||
| ) | ||||
|  | ||||
| func TestNewDB(t *testing.T) { | ||||
| 	path := "/tmp/testdb" | ||||
| 	defer os.RemoveAll(path) | ||||
|  | ||||
| 	db, err := dbleveldb.NewDB(path) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new DB: %v", err) | ||||
| 	} | ||||
| 	defer db.Close() | ||||
| } | ||||
|  | ||||
| func TestNewTable(t *testing.T) { | ||||
| 	path := "/tmp/testdb" | ||||
| 	defer os.RemoveAll(path) | ||||
|  | ||||
| 	db, err := dbleveldb.NewDB(path) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new DB: %v", err) | ||||
| 	} | ||||
| 	defer db.Close() | ||||
|  | ||||
| 	err = db.NewTable("testTable") | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new table: %v", err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestTableExists(t *testing.T) { | ||||
| 	path := "/tmp/testdb" | ||||
| 	defer os.RemoveAll(path) | ||||
|  | ||||
| 	db, err := dbleveldb.NewDB(path) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new DB: %v", err) | ||||
| 	} | ||||
| 	defer db.Close() | ||||
|  | ||||
| 	db.NewTable("testTable") | ||||
| 	if !db.TableExists("testTable") { | ||||
| 		t.Fatalf("Table should exist") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestDropTable(t *testing.T) { | ||||
| 	path := "/tmp/testdb" | ||||
| 	defer os.RemoveAll(path) | ||||
|  | ||||
| 	db, err := dbleveldb.NewDB(path) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new DB: %v", err) | ||||
| 	} | ||||
| 	defer db.Close() | ||||
|  | ||||
| 	db.NewTable("testTable") | ||||
| 	err = db.DropTable("testTable") | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to drop table: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	if db.TableExists("testTable") { | ||||
| 		t.Fatalf("Table should not exist") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestWriteAndRead(t *testing.T) { | ||||
| 	path := "/tmp/testdb" | ||||
| 	defer os.RemoveAll(path) | ||||
|  | ||||
| 	db, err := dbleveldb.NewDB(path) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new DB: %v", err) | ||||
| 	} | ||||
| 	defer db.Close() | ||||
|  | ||||
| 	db.NewTable("testTable") | ||||
| 	err = db.Write("testTable", "testKey", "testValue") | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to write to table: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	var value string | ||||
| 	err = db.Read("testTable", "testKey", &value) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to read from table: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	if value != "testValue" { | ||||
| 		t.Fatalf("Expected 'testValue', got '%v'", value) | ||||
| 	} | ||||
| } | ||||
| func TestListTable(t *testing.T) { | ||||
| 	path := "/tmp/testdb" | ||||
| 	defer os.RemoveAll(path) | ||||
|  | ||||
| 	db, err := dbleveldb.NewDB(path) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to create new DB: %v", err) | ||||
| 	} | ||||
| 	defer db.Close() | ||||
|  | ||||
| 	db.NewTable("testTable") | ||||
| 	err = db.Write("testTable", "testKey1", "testValue1") | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to write to table: %v", err) | ||||
| 	} | ||||
| 	err = db.Write("testTable", "testKey2", "testValue2") | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to write to table: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	result, err := db.ListTable("testTable") | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("Failed to list table: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	if len(result) != 2 { | ||||
| 		t.Fatalf("Expected 2 entries, got %v", len(result)) | ||||
| 	} | ||||
|  | ||||
| 	expected := map[string]string{ | ||||
| 		"testTable/testKey1": "\"testValue1\"", | ||||
| 		"testTable/testKey2": "\"testValue2\"", | ||||
| 	} | ||||
|  | ||||
| 	for _, entry := range result { | ||||
| 		key := string(entry[0]) | ||||
| 		value := string(entry[1]) | ||||
| 		if expected[key] != value { | ||||
| 			t.Fatalf("Expected value '%v' for key '%v', got '%v'", expected[key], key, value) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @@ -9,7 +9,7 @@ import ( | ||||
| 	"os/user" | ||||
| 	"strings" | ||||
| 
 | ||||
| 	"imuslab.com/zoraxy/mod/utils" | ||||
| 	"aroz.org/zoraxy/ztnc/mod/utils" | ||||
| ) | ||||
| 
 | ||||
| func readAuthTokenAsAdmin() (string, error) { | ||||
| @@ -5,15 +5,13 @@ package ganserv | ||||
| 
 | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"strings" | ||||
| 	"syscall" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"aroz.org/zoraxy/ztnc/mod/utils" | ||||
| 	"golang.org/x/sys/windows" | ||||
| 	"imuslab.com/zoraxy/mod/utils" | ||||
| ) | ||||
| 
 | ||||
| // Use admin permission to read auth token on Windows | ||||
| @@ -46,15 +44,6 @@ func readAuthTokenAsAdmin() (string, error) { | ||||
| 		return "", err | ||||
| 	} | ||||
| 
 | ||||
| 	log.Println("Please click agree to allow access to ZeroTier authtoken from ProgramData") | ||||
| 	retry := 0 | ||||
| 	time.Sleep(3 * time.Second) | ||||
| 	for !utils.FileExists("./conf/authtoken.secret") && retry < 10 { | ||||
| 		time.Sleep(3 * time.Second) | ||||
| 		log.Println("Waiting for ZeroTier authtoken extraction...") | ||||
| 		retry++ | ||||
| 	} | ||||
| 
 | ||||
| 	authKey, err := os.ReadFile("./conf/authtoken.secret") | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
| @@ -4,7 +4,7 @@ import ( | ||||
| 	"log" | ||||
| 	"net" | ||||
| 
 | ||||
| 	"imuslab.com/zoraxy/mod/database" | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database" | ||||
| ) | ||||
| 
 | ||||
| /* | ||||
| @@ -7,7 +7,7 @@ import ( | ||||
| 	"regexp" | ||||
| 	"strings" | ||||
| 
 | ||||
| 	"imuslab.com/zoraxy/mod/utils" | ||||
| 	"aroz.org/zoraxy/ztnc/mod/utils" | ||||
| ) | ||||
| 
 | ||||
| func (m *NetworkManager) HandleGetNodeID(w http.ResponseWriter, r *http.Request) { | ||||
| @@ -6,7 +6,7 @@ import ( | ||||
| 	"strconv" | ||||
| 	"testing" | ||||
| 
 | ||||
| 	"imuslab.com/zoraxy/mod/ganserv" | ||||
| 	"aroz.org/zoraxy/ztnc/mod/ganserv" | ||||
| ) | ||||
| 
 | ||||
| func TestGetRandomFreeIP(t *testing.T) { | ||||
							
								
								
									
										105
									
								
								example/plugins/ztnc/mod/utils/conv.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								example/plugins/ztnc/mod/utils/conv.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | ||||
| package utils | ||||
|  | ||||
| import ( | ||||
| 	"archive/zip" | ||||
| 	"io" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func StringToInt64(number string) (int64, error) { | ||||
| 	i, err := strconv.ParseInt(number, 10, 64) | ||||
| 	if err != nil { | ||||
| 		return -1, err | ||||
| 	} | ||||
| 	return i, nil | ||||
| } | ||||
|  | ||||
| func Int64ToString(number int64) string { | ||||
| 	convedNumber := strconv.FormatInt(number, 10) | ||||
| 	return convedNumber | ||||
| } | ||||
|  | ||||
| func ReplaceSpecialCharacters(filename string) string { | ||||
| 	replacements := map[string]string{ | ||||
| 		"#":  "%pound%", | ||||
| 		"&":  "%amp%", | ||||
| 		"{":  "%left_cur%", | ||||
| 		"}":  "%right_cur%", | ||||
| 		"\\": "%backslash%", | ||||
| 		"<":  "%left_ang%", | ||||
| 		">":  "%right_ang%", | ||||
| 		"*":  "%aster%", | ||||
| 		"?":  "%quest%", | ||||
| 		" ":  "%space%", | ||||
| 		"$":  "%dollar%", | ||||
| 		"!":  "%exclan%", | ||||
| 		"'":  "%sin_q%", | ||||
| 		"\"": "%dou_q%", | ||||
| 		":":  "%colon%", | ||||
| 		"@":  "%at%", | ||||
| 		"+":  "%plus%", | ||||
| 		"`":  "%backtick%", | ||||
| 		"|":  "%pipe%", | ||||
| 		"=":  "%equal%", | ||||
| 		".":  "_", | ||||
| 		"/":  "-", | ||||
| 	} | ||||
|  | ||||
| 	for char, replacement := range replacements { | ||||
| 		filename = strings.ReplaceAll(filename, char, replacement) | ||||
| 	} | ||||
|  | ||||
| 	return filename | ||||
| } | ||||
|  | ||||
| /* Zip File Handler */ | ||||
| // zipFiles compresses multiple files into a single zip archive file | ||||
| func ZipFiles(filename string, files ...string) error { | ||||
| 	newZipFile, err := os.Create(filename) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	defer newZipFile.Close() | ||||
|  | ||||
| 	zipWriter := zip.NewWriter(newZipFile) | ||||
| 	defer zipWriter.Close() | ||||
|  | ||||
| 	for _, file := range files { | ||||
| 		if err := addFileToZip(zipWriter, file); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // addFileToZip adds an individual file to a zip archive | ||||
| func addFileToZip(zipWriter *zip.Writer, filename string) error { | ||||
| 	fileToZip, err := os.Open(filename) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	defer fileToZip.Close() | ||||
|  | ||||
| 	info, err := fileToZip.Stat() | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	header, err := zip.FileInfoHeader(info) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	header.Name = filepath.Base(filename) | ||||
| 	header.Method = zip.Deflate | ||||
|  | ||||
| 	writer, err := zipWriter.CreateHeader(header) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	_, err = io.Copy(writer, fileToZip) | ||||
| 	return err | ||||
| } | ||||
							
								
								
									
										19
									
								
								example/plugins/ztnc/mod/utils/template.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								example/plugins/ztnc/mod/utils/template.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| package utils | ||||
|  | ||||
| import ( | ||||
| 	"net/http" | ||||
| ) | ||||
|  | ||||
| /* | ||||
| 	Web Template Generator | ||||
|  | ||||
| 	This is the main system core module that perform function similar to what PHP did. | ||||
| 	To replace part of the content of any file, use {{paramter}} to replace it. | ||||
|  | ||||
|  | ||||
| */ | ||||
|  | ||||
| func SendHTMLResponse(w http.ResponseWriter, msg string) { | ||||
| 	w.Header().Set("Content-Type", "text/html") | ||||
| 	w.Write([]byte(msg)) | ||||
| } | ||||
							
								
								
									
										202
									
								
								example/plugins/ztnc/mod/utils/utils.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										202
									
								
								example/plugins/ztnc/mod/utils/utils.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,202 @@ | ||||
| package utils | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"log" | ||||
| 	"net" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| /* | ||||
| 	Common | ||||
|  | ||||
| 	Some commonly used functions in ArozOS | ||||
|  | ||||
| */ | ||||
|  | ||||
| // Response related | ||||
| func SendTextResponse(w http.ResponseWriter, msg string) { | ||||
| 	w.Write([]byte(msg)) | ||||
| } | ||||
|  | ||||
| // Send JSON response, with an extra json header | ||||
| func SendJSONResponse(w http.ResponseWriter, json string) { | ||||
| 	w.Header().Set("Content-Type", "application/json") | ||||
| 	w.Write([]byte(json)) | ||||
| } | ||||
|  | ||||
| func SendErrorResponse(w http.ResponseWriter, errMsg string) { | ||||
| 	w.Header().Set("Content-Type", "application/json") | ||||
| 	w.Write([]byte("{\"error\":\"" + errMsg + "\"}")) | ||||
| } | ||||
|  | ||||
| func SendOK(w http.ResponseWriter) { | ||||
| 	w.Header().Set("Content-Type", "application/json") | ||||
| 	w.Write([]byte("\"OK\"")) | ||||
| } | ||||
|  | ||||
| // Get GET parameter | ||||
| func GetPara(r *http.Request, key string) (string, error) { | ||||
| 	// Get first value from the URL query | ||||
| 	value := r.URL.Query().Get(key) | ||||
| 	if len(value) == 0 { | ||||
| 		return "", errors.New("invalid " + key + " given") | ||||
| 	} | ||||
| 	return value, nil | ||||
| } | ||||
|  | ||||
| // Get GET paramter as boolean, accept 1 or true | ||||
| func GetBool(r *http.Request, key string) (bool, error) { | ||||
| 	x, err := GetPara(r, key) | ||||
| 	if err != nil { | ||||
| 		return false, err | ||||
| 	} | ||||
|  | ||||
| 	// Convert to lowercase and trim spaces just once to compare | ||||
| 	switch strings.ToLower(strings.TrimSpace(x)) { | ||||
| 	case "1", "true", "on": | ||||
| 		return true, nil | ||||
| 	case "0", "false", "off": | ||||
| 		return false, nil | ||||
| 	} | ||||
|  | ||||
| 	return false, errors.New("invalid boolean given") | ||||
| } | ||||
|  | ||||
| // Get POST parameter | ||||
| func PostPara(r *http.Request, key string) (string, error) { | ||||
| 	// Try to parse the form | ||||
| 	if err := r.ParseForm(); err != nil { | ||||
| 		return "", err | ||||
| 	} | ||||
| 	// Get first value from the form | ||||
| 	x := r.Form.Get(key) | ||||
| 	if len(x) == 0 { | ||||
| 		return "", errors.New("invalid " + key + " given") | ||||
| 	} | ||||
| 	return x, nil | ||||
| } | ||||
|  | ||||
| // Get POST paramter as boolean, accept 1 or true | ||||
| func PostBool(r *http.Request, key string) (bool, error) { | ||||
| 	x, err := PostPara(r, key) | ||||
| 	if err != nil { | ||||
| 		return false, err | ||||
| 	} | ||||
|  | ||||
| 	// Convert to lowercase and trim spaces just once to compare | ||||
| 	switch strings.ToLower(strings.TrimSpace(x)) { | ||||
| 	case "1", "true", "on": | ||||
| 		return true, nil | ||||
| 	case "0", "false", "off": | ||||
| 		return false, nil | ||||
| 	} | ||||
|  | ||||
| 	return false, errors.New("invalid boolean given") | ||||
| } | ||||
|  | ||||
| // Get POST paramter as int | ||||
| func PostInt(r *http.Request, key string) (int, error) { | ||||
| 	x, err := PostPara(r, key) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
|  | ||||
| 	x = strings.TrimSpace(x) | ||||
| 	rx, err := strconv.Atoi(x) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
|  | ||||
| 	return rx, nil | ||||
| } | ||||
|  | ||||
| func FileExists(filename string) bool { | ||||
| 	_, err := os.Stat(filename) | ||||
| 	if err == nil { | ||||
| 		// File exists | ||||
| 		return true | ||||
| 	} else if errors.Is(err, os.ErrNotExist) { | ||||
| 		// File does not exist | ||||
| 		return false | ||||
| 	} | ||||
| 	// Some other error | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| func IsDir(path string) bool { | ||||
| 	if !FileExists(path) { | ||||
| 		return false | ||||
| 	} | ||||
| 	fi, err := os.Stat(path) | ||||
| 	if err != nil { | ||||
| 		log.Fatal(err) | ||||
| 		return false | ||||
| 	} | ||||
| 	switch mode := fi.Mode(); { | ||||
| 	case mode.IsDir(): | ||||
| 		return true | ||||
| 	case mode.IsRegular(): | ||||
| 		return false | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| func TimeToString(targetTime time.Time) string { | ||||
| 	return targetTime.Format("2006-01-02 15:04:05") | ||||
| } | ||||
|  | ||||
| // Check if given string in a given slice | ||||
| func StringInArray(arr []string, str string) bool { | ||||
| 	for _, a := range arr { | ||||
| 		if a == str { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| func StringInArrayIgnoreCase(arr []string, str string) bool { | ||||
| 	smallArray := []string{} | ||||
| 	for _, item := range arr { | ||||
| 		smallArray = append(smallArray, strings.ToLower(item)) | ||||
| 	} | ||||
|  | ||||
| 	return StringInArray(smallArray, strings.ToLower(str)) | ||||
| } | ||||
|  | ||||
| // Validate if the listening address is correct | ||||
| func ValidateListeningAddress(address string) bool { | ||||
| 	// Check if the address starts with a colon, indicating it's just a port | ||||
| 	if strings.HasPrefix(address, ":") { | ||||
| 		return true | ||||
| 	} | ||||
|  | ||||
| 	// Split the address into host and port parts | ||||
| 	host, port, err := net.SplitHostPort(address) | ||||
| 	if err != nil { | ||||
| 		// Try to parse it as just a port | ||||
| 		if _, err := strconv.Atoi(address); err == nil { | ||||
| 			return false // It's just a port number | ||||
| 		} | ||||
| 		return false // It's an invalid address | ||||
| 	} | ||||
|  | ||||
| 	// Check if the port part is a valid number | ||||
| 	if _, err := strconv.Atoi(port); err != nil { | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	// Check if the host part is a valid IP address or empty (indicating any IP) | ||||
| 	if host != "" { | ||||
| 		if net.ParseIP(host) == nil { | ||||
| 			return false | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return true | ||||
| } | ||||
							
								
								
									
										19
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/README.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/README.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| # Zoraxy Plugin | ||||
|  | ||||
| ## Overview | ||||
| This module serves as a template for building your own plugins for the Zoraxy Reverse Proxy. By copying this module to your plugin mod folder, you can create a new plugin with the necessary structure and components. | ||||
|  | ||||
| ## Instructions | ||||
|  | ||||
| 1. **Copy the Module:** | ||||
|     - Copy the entire `zoraxy_plugin` module to your plugin mod folder. | ||||
|  | ||||
| 2. **Include the Structure:** | ||||
|     - Ensure that you maintain the directory structure and file organization as provided in this module. | ||||
|  | ||||
| 3. **Modify as Needed:** | ||||
|     - Customize the copied module to implement the desired functionality for your plugin. | ||||
|  | ||||
| ## Directory Structure | ||||
|  zoraxy_plugin: Handle -introspect and -configuration process required for plugin loading and startup | ||||
|  embed_webserver: Handle embeded web server routing and injecting csrf token to your plugin served UI pages | ||||
							
								
								
									
										145
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/dev_webserver.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/dev_webserver.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type PluginUiDebugRouter struct { | ||||
| 	PluginID         string //The ID of the plugin | ||||
| 	TargetDir        string //The directory where the UI files are stored | ||||
| 	HandlerPrefix    string //The prefix of the handler used to route this router, e.g. /ui | ||||
| 	EnableDebug      bool   //Enable debug mode | ||||
| 	terminateHandler func() //The handler to be called when the plugin is terminated | ||||
| } | ||||
|  | ||||
| // NewPluginFileSystemUIRouter creates a new PluginUiRouter with file system | ||||
| // The targetDir is the directory where the UI files are stored (e.g. ./www) | ||||
| // The handlerPrefix is the prefix of the handler used to route this router | ||||
| // The handlerPrefix should start with a slash (e.g. /ui) that matches the http.Handle path | ||||
| // All prefix should not end with a slash | ||||
| func NewPluginFileSystemUIRouter(pluginID string, targetDir string, handlerPrefix string) *PluginUiDebugRouter { | ||||
| 	//Make sure all prefix are in /prefix format | ||||
| 	if !strings.HasPrefix(handlerPrefix, "/") { | ||||
| 		handlerPrefix = "/" + handlerPrefix | ||||
| 	} | ||||
| 	handlerPrefix = strings.TrimSuffix(handlerPrefix, "/") | ||||
|  | ||||
| 	//Return the PluginUiRouter | ||||
| 	return &PluginUiDebugRouter{ | ||||
| 		PluginID:      pluginID, | ||||
| 		TargetDir:     targetDir, | ||||
| 		HandlerPrefix: handlerPrefix, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *PluginUiDebugRouter) populateCSRFToken(r *http.Request, fsHandler http.Handler) http.Handler { | ||||
| 	//Get the CSRF token from header | ||||
| 	csrfToken := r.Header.Get("X-Zoraxy-Csrf") | ||||
| 	if csrfToken == "" { | ||||
| 		csrfToken = "missing-csrf-token" | ||||
| 	} | ||||
|  | ||||
| 	//Return the middleware | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		// Check if the request is for an HTML file | ||||
| 		if strings.HasSuffix(r.URL.Path, ".html") { | ||||
| 			//Read the target file from file system | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetDir + "/" + targetFilePath | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			targetFileContent, err := os.ReadFile(targetFilePath) | ||||
| 			if err != nil { | ||||
| 				http.Error(w, "File not found", http.StatusNotFound) | ||||
| 				return | ||||
| 			} | ||||
| 			body := string(targetFileContent) | ||||
| 			body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 			w.Header().Set("Content-Type", "text/html") | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte(body)) | ||||
| 			return | ||||
| 		} else if strings.HasSuffix(r.URL.Path, "/") { | ||||
| 			//Check if the request is for a directory | ||||
| 			//Check if the directory has an index.html file | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetDir + "/" + targetFilePath + "index.html" | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			if _, err := os.Stat(targetFilePath); err == nil { | ||||
| 				//Serve the index.html file | ||||
| 				targetFileContent, err := os.ReadFile(targetFilePath) | ||||
| 				if err != nil { | ||||
| 					http.Error(w, "File not found", http.StatusNotFound) | ||||
| 					return | ||||
| 				} | ||||
| 				body := string(targetFileContent) | ||||
| 				body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 				w.Header().Set("Content-Type", "text/html") | ||||
| 				w.WriteHeader(http.StatusOK) | ||||
| 				w.Write([]byte(body)) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Call the next handler | ||||
| 		fsHandler.ServeHTTP(w, r) | ||||
| 	}) | ||||
|  | ||||
| } | ||||
|  | ||||
| // GetHttpHandler returns the http.Handler for the PluginUiRouter | ||||
| func (p *PluginUiDebugRouter) Handler() http.Handler { | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		//Remove the plugin UI handler path prefix | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Print("Request URL:", r.URL.Path, " rewriting to ") | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, p.HandlerPrefix) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		r.URL.Path = rewrittenURL | ||||
| 		r.RequestURI = rewrittenURL | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Println(r.URL.Path) | ||||
| 		} | ||||
|  | ||||
| 		//Serve the file from the file system | ||||
| 		fsHandler := http.FileServer(http.Dir(p.TargetDir)) | ||||
|  | ||||
| 		// Replace {{csrf_token}} with the actual CSRF token and serve the file | ||||
| 		p.populateCSRFToken(r, fsHandler).ServeHTTP(w, r) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // RegisterTerminateHandler registers the terminate handler for the PluginUiRouter | ||||
| // The terminate handler will be called when the plugin is terminated from Zoraxy plugin manager | ||||
| // if mux is nil, the handler will be registered to http.DefaultServeMux | ||||
| func (p *PluginUiDebugRouter) RegisterTerminateHandler(termFunc func(), mux *http.ServeMux) { | ||||
| 	p.terminateHandler = termFunc | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
| 	mux.HandleFunc(p.HandlerPrefix+"/term", func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.terminateHandler() | ||||
| 		w.WriteHeader(http.StatusOK) | ||||
| 		go func() { | ||||
| 			//Make sure the response is sent before the plugin is terminated | ||||
| 			time.Sleep(100 * time.Millisecond) | ||||
| 			os.Exit(0) | ||||
| 		}() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // Attach the file system UI handler to the target http.ServeMux | ||||
| func (p *PluginUiDebugRouter) AttachHandlerToMux(mux *http.ServeMux) { | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
|  | ||||
| 	p.HandlerPrefix = strings.TrimSuffix(p.HandlerPrefix, "/") | ||||
| 	mux.Handle(p.HandlerPrefix+"/", p.Handler()) | ||||
| } | ||||
							
								
								
									
										162
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/dynamic_router.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										162
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/dynamic_router.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,162 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| /* | ||||
|  | ||||
| 	Dynamic Path Handler | ||||
|  | ||||
| */ | ||||
|  | ||||
| type SniffResult int | ||||
|  | ||||
| const ( | ||||
| 	SniffResultAccpet SniffResult = iota // Forward the request to this plugin dynamic capture ingress | ||||
| 	SniffResultSkip                      // Skip this plugin and let the next plugin handle the request | ||||
| ) | ||||
|  | ||||
| type SniffHandler func(*DynamicSniffForwardRequest) SniffResult | ||||
|  | ||||
| /* | ||||
| RegisterDynamicSniffHandler registers a dynamic sniff handler for a path | ||||
| You can decide to accept or skip the request based on the request header and paths | ||||
| */ | ||||
| func (p *PathRouter) RegisterDynamicSniffHandler(sniff_ingress string, mux *http.ServeMux, handler SniffHandler) { | ||||
| 	if !strings.HasSuffix(sniff_ingress, "/") { | ||||
| 		sniff_ingress = sniff_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(sniff_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Println("Request captured by dynamic sniff path: " + r.RequestURI) | ||||
| 		} | ||||
|  | ||||
| 		// Decode the request payload | ||||
| 		jsonBytes, err := io.ReadAll(r.Body) | ||||
| 		if err != nil { | ||||
| 			if p.enableDebugPrint { | ||||
| 				fmt.Println("Error reading request body:", err) | ||||
| 			} | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
| 		payload, err := DecodeForwardRequestPayload(jsonBytes) | ||||
| 		if err != nil { | ||||
| 			if p.enableDebugPrint { | ||||
| 				fmt.Println("Error decoding request payload:", err) | ||||
| 				fmt.Print("Payload: ") | ||||
| 				fmt.Println(string(jsonBytes)) | ||||
| 			} | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		// Get the forwarded request UUID | ||||
| 		forwardUUID := r.Header.Get("X-Zoraxy-RequestID") | ||||
| 		payload.requestUUID = forwardUUID | ||||
| 		payload.rawRequest = r | ||||
|  | ||||
| 		sniffResult := handler(&payload) | ||||
| 		if sniffResult == SniffResultAccpet { | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte("OK")) | ||||
| 		} else { | ||||
| 			w.WriteHeader(http.StatusNotImplemented) | ||||
| 			w.Write([]byte("SKIP")) | ||||
| 		} | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| // RegisterDynamicCaptureHandle register the dynamic capture ingress path with a handler | ||||
| func (p *PathRouter) RegisterDynamicCaptureHandle(capture_ingress string, mux *http.ServeMux, handlefunc func(http.ResponseWriter, *http.Request)) { | ||||
| 	if !strings.HasSuffix(capture_ingress, "/") { | ||||
| 		capture_ingress = capture_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(capture_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Println("Request captured by dynamic capture path: " + r.RequestURI) | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, capture_ingress) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		if rewrittenURL == "" { | ||||
| 			rewrittenURL = "/" | ||||
| 		} | ||||
| 		if !strings.HasPrefix(rewrittenURL, "/") { | ||||
| 			rewrittenURL = "/" + rewrittenURL | ||||
| 		} | ||||
| 		r.RequestURI = rewrittenURL | ||||
|  | ||||
| 		handlefunc(w, r) | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| /* | ||||
| 	Sniffing and forwarding | ||||
|  | ||||
| 	The following functions are here to help with | ||||
| 	sniffing and forwarding requests to the dynamic | ||||
| 	router. | ||||
| */ | ||||
| // A custom request object to be used in the dynamic sniffing | ||||
| type DynamicSniffForwardRequest struct { | ||||
| 	Method     string              `json:"method"` | ||||
| 	Hostname   string              `json:"hostname"` | ||||
| 	URL        string              `json:"url"` | ||||
| 	Header     map[string][]string `json:"header"` | ||||
| 	RemoteAddr string              `json:"remote_addr"` | ||||
| 	Host       string              `json:"host"` | ||||
| 	RequestURI string              `json:"request_uri"` | ||||
| 	Proto      string              `json:"proto"` | ||||
| 	ProtoMajor int                 `json:"proto_major"` | ||||
| 	ProtoMinor int                 `json:"proto_minor"` | ||||
|  | ||||
| 	/* Internal use */ | ||||
| 	rawRequest  *http.Request `json:"-"` | ||||
| 	requestUUID string        `json:"-"` | ||||
| } | ||||
|  | ||||
| // GetForwardRequestPayload returns a DynamicSniffForwardRequest object from an http.Request object | ||||
| func EncodeForwardRequestPayload(r *http.Request) DynamicSniffForwardRequest { | ||||
| 	return DynamicSniffForwardRequest{ | ||||
| 		Method:     r.Method, | ||||
| 		Hostname:   r.Host, | ||||
| 		URL:        r.URL.String(), | ||||
| 		Header:     r.Header, | ||||
| 		RemoteAddr: r.RemoteAddr, | ||||
| 		Host:       r.Host, | ||||
| 		RequestURI: r.RequestURI, | ||||
| 		Proto:      r.Proto, | ||||
| 		ProtoMajor: r.ProtoMajor, | ||||
| 		ProtoMinor: r.ProtoMinor, | ||||
| 		rawRequest: r, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // DecodeForwardRequestPayload decodes JSON bytes into a DynamicSniffForwardRequest object | ||||
| func DecodeForwardRequestPayload(jsonBytes []byte) (DynamicSniffForwardRequest, error) { | ||||
| 	var payload DynamicSniffForwardRequest | ||||
| 	err := json.Unmarshal(jsonBytes, &payload) | ||||
| 	if err != nil { | ||||
| 		return DynamicSniffForwardRequest{}, err | ||||
| 	} | ||||
| 	return payload, nil | ||||
| } | ||||
|  | ||||
| // GetRequest returns the original http.Request object, for debugging purposes | ||||
| func (dsfr *DynamicSniffForwardRequest) GetRequest() *http.Request { | ||||
| 	return dsfr.rawRequest | ||||
| } | ||||
|  | ||||
| // GetRequestUUID returns the request UUID | ||||
| // if this UUID is empty string, that might indicate the request | ||||
| // is not coming from the dynamic router | ||||
| func (dsfr *DynamicSniffForwardRequest) GetRequestUUID() string { | ||||
| 	return dsfr.requestUUID | ||||
| } | ||||
							
								
								
									
										156
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/embed_webserver.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/embed_webserver.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	"fmt" | ||||
| 	"io/fs" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type PluginUiRouter struct { | ||||
| 	PluginID         string    //The ID of the plugin | ||||
| 	TargetFs         *embed.FS //The embed.FS where the UI files are stored | ||||
| 	TargetFsPrefix   string    //The prefix of the embed.FS where the UI files are stored, e.g. /web | ||||
| 	HandlerPrefix    string    //The prefix of the handler used to route this router, e.g. /ui | ||||
| 	EnableDebug      bool      //Enable debug mode | ||||
| 	terminateHandler func()    //The handler to be called when the plugin is terminated | ||||
| } | ||||
|  | ||||
| // NewPluginEmbedUIRouter creates a new PluginUiRouter with embed.FS | ||||
| // The targetFsPrefix is the prefix of the embed.FS where the UI files are stored | ||||
| // The targetFsPrefix should be relative to the root of the embed.FS | ||||
| // The targetFsPrefix should start with a slash (e.g. /web) that corresponds to the root folder of the embed.FS | ||||
| // The handlerPrefix is the prefix of the handler used to route this router | ||||
| // The handlerPrefix should start with a slash (e.g. /ui) that matches the http.Handle path | ||||
| // All prefix should not end with a slash | ||||
| func NewPluginEmbedUIRouter(pluginID string, targetFs *embed.FS, targetFsPrefix string, handlerPrefix string) *PluginUiRouter { | ||||
| 	//Make sure all prefix are in /prefix format | ||||
| 	if !strings.HasPrefix(targetFsPrefix, "/") { | ||||
| 		targetFsPrefix = "/" + targetFsPrefix | ||||
| 	} | ||||
| 	targetFsPrefix = strings.TrimSuffix(targetFsPrefix, "/") | ||||
|  | ||||
| 	if !strings.HasPrefix(handlerPrefix, "/") { | ||||
| 		handlerPrefix = "/" + handlerPrefix | ||||
| 	} | ||||
| 	handlerPrefix = strings.TrimSuffix(handlerPrefix, "/") | ||||
|  | ||||
| 	//Return the PluginUiRouter | ||||
| 	return &PluginUiRouter{ | ||||
| 		PluginID:       pluginID, | ||||
| 		TargetFs:       targetFs, | ||||
| 		TargetFsPrefix: targetFsPrefix, | ||||
| 		HandlerPrefix:  handlerPrefix, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *PluginUiRouter) populateCSRFToken(r *http.Request, fsHandler http.Handler) http.Handler { | ||||
| 	//Get the CSRF token from header | ||||
| 	csrfToken := r.Header.Get("X-Zoraxy-Csrf") | ||||
| 	if csrfToken == "" { | ||||
| 		csrfToken = "missing-csrf-token" | ||||
| 	} | ||||
|  | ||||
| 	//Return the middleware | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		// Check if the request is for an HTML file | ||||
| 		if strings.HasSuffix(r.URL.Path, ".html") { | ||||
| 			//Read the target file from embed.FS | ||||
| 			targetFilePath := strings.TrimPrefix(r.URL.Path, "/") | ||||
| 			targetFilePath = p.TargetFsPrefix + "/" + targetFilePath | ||||
| 			targetFilePath = strings.TrimPrefix(targetFilePath, "/") | ||||
| 			targetFileContent, err := fs.ReadFile(*p.TargetFs, targetFilePath) | ||||
| 			if err != nil { | ||||
| 				http.Error(w, "File not found", http.StatusNotFound) | ||||
| 				return | ||||
| 			} | ||||
| 			body := string(targetFileContent) | ||||
| 			body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 			w.Header().Set("Content-Type", "text/html") | ||||
| 			w.WriteHeader(http.StatusOK) | ||||
| 			w.Write([]byte(body)) | ||||
| 			return | ||||
| 		} else if strings.HasSuffix(r.URL.Path, "/") { | ||||
| 			// Check if the directory has an index.html file | ||||
| 			indexFilePath := strings.TrimPrefix(r.URL.Path, "/") + "index.html" | ||||
| 			indexFilePath = p.TargetFsPrefix + "/" + indexFilePath | ||||
| 			indexFilePath = strings.TrimPrefix(indexFilePath, "/") | ||||
| 			indexFileContent, err := fs.ReadFile(*p.TargetFs, indexFilePath) | ||||
| 			if err == nil { | ||||
| 				body := string(indexFileContent) | ||||
| 				body = strings.ReplaceAll(body, "{{.csrfToken}}", csrfToken) | ||||
| 				w.Header().Set("Content-Type", "text/html") | ||||
| 				w.WriteHeader(http.StatusOK) | ||||
| 				w.Write([]byte(body)) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		//Call the next handler | ||||
| 		fsHandler.ServeHTTP(w, r) | ||||
| 	}) | ||||
|  | ||||
| } | ||||
|  | ||||
| // GetHttpHandler returns the http.Handler for the PluginUiRouter | ||||
| func (p *PluginUiRouter) Handler() http.Handler { | ||||
| 	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		//Remove the plugin UI handler path prefix | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Print("Request URL:", r.URL.Path, " rewriting to ") | ||||
| 		} | ||||
|  | ||||
| 		rewrittenURL := r.RequestURI | ||||
| 		rewrittenURL = strings.TrimPrefix(rewrittenURL, p.HandlerPrefix) | ||||
| 		rewrittenURL = strings.ReplaceAll(rewrittenURL, "//", "/") | ||||
| 		r.URL, _ = url.Parse(rewrittenURL) | ||||
| 		r.RequestURI = rewrittenURL | ||||
| 		if p.EnableDebug { | ||||
| 			fmt.Println(r.URL.Path) | ||||
| 		} | ||||
|  | ||||
| 		//Serve the file from the embed.FS | ||||
| 		subFS, err := fs.Sub(*p.TargetFs, strings.TrimPrefix(p.TargetFsPrefix, "/")) | ||||
| 		if err != nil { | ||||
| 			fmt.Println(err.Error()) | ||||
| 			http.Error(w, "Internal Server Error", http.StatusInternalServerError) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		// Replace {{csrf_token}} with the actual CSRF token and serve the file | ||||
| 		p.populateCSRFToken(r, http.FileServer(http.FS(subFS))).ServeHTTP(w, r) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // RegisterTerminateHandler registers the terminate handler for the PluginUiRouter | ||||
| // The terminate handler will be called when the plugin is terminated from Zoraxy plugin manager | ||||
| // if mux is nil, the handler will be registered to http.DefaultServeMux | ||||
| func (p *PluginUiRouter) RegisterTerminateHandler(termFunc func(), mux *http.ServeMux) { | ||||
| 	p.terminateHandler = termFunc | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
| 	mux.HandleFunc(p.HandlerPrefix+"/term", func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.terminateHandler() | ||||
| 		w.WriteHeader(http.StatusOK) | ||||
| 		go func() { | ||||
| 			//Make sure the response is sent before the plugin is terminated | ||||
| 			time.Sleep(100 * time.Millisecond) | ||||
| 			os.Exit(0) | ||||
| 		}() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // Attach the embed UI handler to the target http.ServeMux | ||||
| func (p *PluginUiRouter) AttachHandlerToMux(mux *http.ServeMux) { | ||||
| 	if mux == nil { | ||||
| 		mux = http.DefaultServeMux | ||||
| 	} | ||||
|  | ||||
| 	p.HandlerPrefix = strings.TrimSuffix(p.HandlerPrefix, "/") | ||||
| 	mux.Handle(p.HandlerPrefix+"/", p.Handler()) | ||||
| } | ||||
							
								
								
									
										105
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/static_router.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/static_router.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type PathRouter struct { | ||||
| 	enableDebugPrint bool | ||||
| 	pathHandlers     map[string]http.Handler | ||||
| 	defaultHandler   http.Handler | ||||
| } | ||||
|  | ||||
| // NewPathRouter creates a new PathRouter | ||||
| func NewPathRouter() *PathRouter { | ||||
| 	return &PathRouter{ | ||||
| 		enableDebugPrint: false, | ||||
| 		pathHandlers:     make(map[string]http.Handler), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // RegisterPathHandler registers a handler for a path | ||||
| func (p *PathRouter) RegisterPathHandler(path string, handler http.Handler) { | ||||
| 	path = strings.TrimSuffix(path, "/") | ||||
| 	p.pathHandlers[path] = handler | ||||
| } | ||||
|  | ||||
| // RemovePathHandler removes a handler for a path | ||||
| func (p *PathRouter) RemovePathHandler(path string) { | ||||
| 	delete(p.pathHandlers, path) | ||||
| } | ||||
|  | ||||
| // SetDefaultHandler sets the default handler for the router | ||||
| // This handler will be called if no path handler is found | ||||
| func (p *PathRouter) SetDefaultHandler(handler http.Handler) { | ||||
| 	p.defaultHandler = handler | ||||
| } | ||||
|  | ||||
| // SetDebugPrintMode sets the debug print mode | ||||
| func (p *PathRouter) SetDebugPrintMode(enable bool) { | ||||
| 	p.enableDebugPrint = enable | ||||
| } | ||||
|  | ||||
| // StartStaticCapture starts the static capture ingress | ||||
| func (p *PathRouter) RegisterStaticCaptureHandle(capture_ingress string, mux *http.ServeMux) { | ||||
| 	if !strings.HasSuffix(capture_ingress, "/") { | ||||
| 		capture_ingress = capture_ingress + "/" | ||||
| 	} | ||||
| 	mux.Handle(capture_ingress, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||
| 		p.staticCaptureServeHTTP(w, r) | ||||
| 	})) | ||||
| } | ||||
|  | ||||
| // staticCaptureServeHTTP serves the static capture path using user defined handler | ||||
| func (p *PathRouter) staticCaptureServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
| 	capturePath := r.Header.Get("X-Zoraxy-Capture") | ||||
| 	if capturePath != "" { | ||||
| 		if p.enableDebugPrint { | ||||
| 			fmt.Printf("Using capture path: %s\n", capturePath) | ||||
| 		} | ||||
| 		originalURI := r.Header.Get("X-Zoraxy-Uri") | ||||
| 		r.URL.Path = originalURI | ||||
| 		if handler, ok := p.pathHandlers[capturePath]; ok { | ||||
| 			handler.ServeHTTP(w, r) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	p.defaultHandler.ServeHTTP(w, r) | ||||
| } | ||||
|  | ||||
| func (p *PathRouter) PrintRequestDebugMessage(r *http.Request) { | ||||
| 	if p.enableDebugPrint { | ||||
| 		fmt.Printf("Capture Request with path: %s \n\n**Request Headers** \n\n", r.URL.Path) | ||||
| 		keys := make([]string, 0, len(r.Header)) | ||||
| 		for key := range r.Header { | ||||
| 			keys = append(keys, key) | ||||
| 		} | ||||
| 		sort.Strings(keys) | ||||
| 		for _, key := range keys { | ||||
| 			for _, value := range r.Header[key] { | ||||
| 				fmt.Printf("%s: %s\n", key, value) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		fmt.Printf("\n\n**Request Details**\n\n") | ||||
| 		fmt.Printf("Method: %s\n", r.Method) | ||||
| 		fmt.Printf("URL: %s\n", r.URL.String()) | ||||
| 		fmt.Printf("Proto: %s\n", r.Proto) | ||||
| 		fmt.Printf("Host: %s\n", r.Host) | ||||
| 		fmt.Printf("RemoteAddr: %s\n", r.RemoteAddr) | ||||
| 		fmt.Printf("RequestURI: %s\n", r.RequestURI) | ||||
| 		fmt.Printf("ContentLength: %d\n", r.ContentLength) | ||||
| 		fmt.Printf("TransferEncoding: %v\n", r.TransferEncoding) | ||||
| 		fmt.Printf("Close: %v\n", r.Close) | ||||
| 		fmt.Printf("Form: %v\n", r.Form) | ||||
| 		fmt.Printf("PostForm: %v\n", r.PostForm) | ||||
| 		fmt.Printf("MultipartForm: %v\n", r.MultipartForm) | ||||
| 		fmt.Printf("Trailer: %v\n", r.Trailer) | ||||
| 		fmt.Printf("RemoteAddr: %s\n", r.RemoteAddr) | ||||
| 		fmt.Printf("RequestURI: %s\n", r.RequestURI) | ||||
|  | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										176
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/zoraxy_plugin.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								example/plugins/ztnc/mod/zoraxy_plugin/zoraxy_plugin.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | ||||
| package zoraxy_plugin | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| /* | ||||
| 	Plugins Includes.go | ||||
|  | ||||
| 	This file is copied from Zoraxy source code | ||||
| 	You can always find the latest version under mod/plugins/includes.go | ||||
| 	Usually this file are backward compatible | ||||
| */ | ||||
|  | ||||
| type PluginType int | ||||
|  | ||||
| const ( | ||||
| 	PluginType_Router    PluginType = 0 //Router Plugin, used for handling / routing / forwarding traffic | ||||
| 	PluginType_Utilities PluginType = 1 //Utilities Plugin, used for utilities like Zerotier or Static Web Server that do not require interception with the dpcore | ||||
| ) | ||||
|  | ||||
| type StaticCaptureRule struct { | ||||
| 	CapturePath string `json:"capture_path"` | ||||
| 	//To be expanded | ||||
| } | ||||
|  | ||||
| type ControlStatusCode int | ||||
|  | ||||
| const ( | ||||
| 	ControlStatusCode_CAPTURED  ControlStatusCode = 280 //Traffic captured by plugin, ask Zoraxy not to process the traffic | ||||
| 	ControlStatusCode_UNHANDLED ControlStatusCode = 284 //Traffic not handled by plugin, ask Zoraxy to process the traffic | ||||
| 	ControlStatusCode_ERROR     ControlStatusCode = 580 //Error occurred while processing the traffic, ask Zoraxy to process the traffic and log the error | ||||
| ) | ||||
|  | ||||
| type SubscriptionEvent struct { | ||||
| 	EventName   string `json:"event_name"` | ||||
| 	EventSource string `json:"event_source"` | ||||
| 	Payload     string `json:"payload"` //Payload of the event, can be empty | ||||
| } | ||||
|  | ||||
| type RuntimeConstantValue struct { | ||||
| 	ZoraxyVersion    string `json:"zoraxy_version"` | ||||
| 	ZoraxyUUID       string `json:"zoraxy_uuid"` | ||||
| 	DevelopmentBuild bool   `json:"development_build"` //Whether the Zoraxy is a development build or not | ||||
| } | ||||
|  | ||||
| /* | ||||
| IntroSpect Payload | ||||
|  | ||||
| When the plugin is initialized with -introspect flag, | ||||
| the plugin shell return this payload as JSON and exit | ||||
| */ | ||||
| type IntroSpect struct { | ||||
| 	/* Plugin metadata */ | ||||
| 	ID            string     `json:"id"`             //Unique ID of your plugin, recommended using your own domain in reverse like com.yourdomain.pluginname | ||||
| 	Name          string     `json:"name"`           //Name of your plugin | ||||
| 	Author        string     `json:"author"`         //Author name of your plugin | ||||
| 	AuthorContact string     `json:"author_contact"` //Author contact of your plugin, like email | ||||
| 	Description   string     `json:"description"`    //Description of your plugin | ||||
| 	URL           string     `json:"url"`            //URL of your plugin | ||||
| 	Type          PluginType `json:"type"`           //Type of your plugin, Router(0) or Utilities(1) | ||||
| 	VersionMajor  int        `json:"version_major"`  //Major version of your plugin | ||||
| 	VersionMinor  int        `json:"version_minor"`  //Minor version of your plugin | ||||
| 	VersionPatch  int        `json:"version_patch"`  //Patch version of your plugin | ||||
|  | ||||
| 	/* | ||||
|  | ||||
| 		Endpoint Settings | ||||
|  | ||||
| 	*/ | ||||
|  | ||||
| 	/* | ||||
| 		Static Capture Settings | ||||
|  | ||||
| 		Once plugin is enabled these rules always applies to the enabled HTTP Proxy rule | ||||
| 		This is faster than dynamic capture, but less flexible | ||||
| 	*/ | ||||
| 	StaticCapturePaths   []StaticCaptureRule `json:"static_capture_paths"`   //Static capture paths of your plugin, see Zoraxy documentation for more details | ||||
| 	StaticCaptureIngress string              `json:"static_capture_ingress"` //Static capture ingress path of your plugin (e.g. /s_handler) | ||||
|  | ||||
| 	/* | ||||
| 		Dynamic Capture Settings | ||||
|  | ||||
| 		Once plugin is enabled, these rules will be captured and forward to plugin sniff | ||||
| 		if the plugin sniff returns 280, the traffic will be captured | ||||
| 		otherwise, the traffic will be forwarded to the next plugin | ||||
| 		This is slower than static capture, but more flexible | ||||
| 	*/ | ||||
| 	DynamicCaptureSniff   string `json:"dynamic_capture_sniff"`   //Dynamic capture sniff path of your plugin (e.g. /d_sniff) | ||||
| 	DynamicCaptureIngress string `json:"dynamic_capture_ingress"` //Dynamic capture ingress path of your plugin (e.g. /d_handler) | ||||
|  | ||||
| 	/* UI Path for your plugin */ | ||||
| 	UIPath string `json:"ui_path"` //UI path of your plugin (e.g. /ui), will proxy the whole subpath tree to Zoraxy Web UI as plugin UI | ||||
|  | ||||
| 	/* Subscriptions Settings */ | ||||
| 	SubscriptionPath    string            `json:"subscription_path"`    //Subscription event path of your plugin (e.g. /notifyme), a POST request with SubscriptionEvent as body will be sent to this path when the event is triggered | ||||
| 	SubscriptionsEvents map[string]string `json:"subscriptions_events"` //Subscriptions events of your plugin, see Zoraxy documentation for more details | ||||
| } | ||||
|  | ||||
| /* | ||||
| ServeIntroSpect Function | ||||
|  | ||||
| This function will check if the plugin is initialized with -introspect flag, | ||||
| if so, it will print the intro spect and exit | ||||
|  | ||||
| Place this function at the beginning of your plugin main function | ||||
| */ | ||||
| func ServeIntroSpect(pluginSpect *IntroSpect) { | ||||
| 	if len(os.Args) > 1 && os.Args[1] == "-introspect" { | ||||
| 		//Print the intro spect and exit | ||||
| 		jsonData, _ := json.MarshalIndent(pluginSpect, "", " ") | ||||
| 		fmt.Println(string(jsonData)) | ||||
| 		os.Exit(0) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* | ||||
| ConfigureSpec Payload | ||||
|  | ||||
| Zoraxy will start your plugin with -configure flag, | ||||
| the plugin shell read this payload as JSON and configure itself | ||||
| by the supplied values like starting a web server at given port | ||||
| that listens to 127.0.0.1:port | ||||
| */ | ||||
| type ConfigureSpec struct { | ||||
| 	Port         int                  `json:"port"`          //Port to listen | ||||
| 	RuntimeConst RuntimeConstantValue `json:"runtime_const"` //Runtime constant values | ||||
| 	//To be expanded | ||||
| } | ||||
|  | ||||
| /* | ||||
| RecvExecuteConfigureSpec Function | ||||
|  | ||||
| This function will read the configure spec from Zoraxy | ||||
| and return the ConfigureSpec object | ||||
|  | ||||
| Place this function after ServeIntroSpect function in your plugin main function | ||||
| */ | ||||
| func RecvConfigureSpec() (*ConfigureSpec, error) { | ||||
| 	for i, arg := range os.Args { | ||||
| 		if strings.HasPrefix(arg, "-configure=") { | ||||
| 			var configSpec ConfigureSpec | ||||
| 			if err := json.Unmarshal([]byte(arg[11:]), &configSpec); err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			return &configSpec, nil | ||||
| 		} else if arg == "-configure" { | ||||
| 			var configSpec ConfigureSpec | ||||
| 			var nextArg string | ||||
| 			if len(os.Args) > i+1 { | ||||
| 				nextArg = os.Args[i+1] | ||||
| 				if err := json.Unmarshal([]byte(nextArg), &configSpec); err != nil { | ||||
| 					return nil, err | ||||
| 				} | ||||
| 			} else { | ||||
| 				return nil, fmt.Errorf("No port specified after -configure flag") | ||||
| 			} | ||||
| 			return &configSpec, nil | ||||
| 		} | ||||
| 	} | ||||
| 	return nil, fmt.Errorf("No -configure flag found") | ||||
| } | ||||
|  | ||||
| /* | ||||
| ServeAndRecvSpec Function | ||||
|  | ||||
| This function will serve the intro spect and return the configure spec | ||||
| See the ServeIntroSpect and RecvConfigureSpec for more details | ||||
| */ | ||||
| func ServeAndRecvSpec(pluginSpect *IntroSpect) (*ConfigureSpec, error) { | ||||
| 	ServeIntroSpect(pluginSpect) | ||||
| 	return RecvConfigureSpec() | ||||
| } | ||||
							
								
								
									
										69
									
								
								example/plugins/ztnc/start.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								example/plugins/ztnc/start.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
|  | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database" | ||||
| 	"aroz.org/zoraxy/ztnc/mod/database/dbinc" | ||||
| 	"aroz.org/zoraxy/ztnc/mod/ganserv" | ||||
| 	"aroz.org/zoraxy/ztnc/mod/utils" | ||||
| ) | ||||
|  | ||||
| func startGanNetworkController() error { | ||||
| 	fmt.Println("Starting ZeroTier Network Controller") | ||||
| 	//Create a new database | ||||
| 	var err error | ||||
| 	sysdb, err = database.NewDatabase(DB_FILE_PATH, dbinc.BackendBoltDB) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	//Initiate the GAN server manager | ||||
| 	usingZtAuthToken := "" | ||||
| 	ztAPIPort := 9993 | ||||
|  | ||||
| 	if utils.FileExists(AUTH_TOKEN_PATH) { | ||||
| 		authToken, err := os.ReadFile(AUTH_TOKEN_PATH) | ||||
| 		if err != nil { | ||||
| 			fmt.Println("Error reading auth config file:", err) | ||||
| 			return err | ||||
| 		} | ||||
| 		usingZtAuthToken = string(authToken) | ||||
| 		fmt.Println("Loaded ZeroTier Auth Token from file") | ||||
| 	} | ||||
|  | ||||
| 	if usingZtAuthToken == "" { | ||||
| 		usingZtAuthToken, err = ganserv.TryLoadorAskUserForAuthkey() | ||||
| 		if err != nil { | ||||
| 			fmt.Println("Error getting ZeroTier Auth Token:", err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	ganManager = ganserv.NewNetworkManager(&ganserv.NetworkManagerOptions{ | ||||
| 		AuthToken: usingZtAuthToken, | ||||
| 		ApiPort:   ztAPIPort, | ||||
| 		Database:  sysdb, | ||||
| 	}) | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func initApiEndpoints() { | ||||
| 	//UI_RELPATH must be the same as the one in the plugin intro spect | ||||
| 	// as Zoraxy plugin UI proxy will only forward the UI path to your plugin | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/network/info", ganManager.HandleGetNodeID) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/network/add", ganManager.HandleAddNetwork) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/network/remove", ganManager.HandleRemoveNetwork) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/network/list", ganManager.HandleListNetwork) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/network/name", ganManager.HandleNetworkNaming) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/network/setRange", ganManager.HandleSetRanges) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/network/join", ganManager.HandleServerJoinNetwork) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/network/leave", ganManager.HandleServerLeaveNetwork) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/members/list", ganManager.HandleMemberList) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/members/ip", ganManager.HandleMemberIP) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/members/name", ganManager.HandleMemberNaming) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/members/authorize", ganManager.HandleMemberAuthorization) | ||||
| 	http.HandleFunc(UI_RELPATH+"/api/gan/members/delete", ganManager.HandleMemberDelete) | ||||
| } | ||||
| @@ -1,3 +1,4 @@ | ||||
| <!-- This is being loaded in index.html as ajax --> | ||||
| <div class="standardContainer"> | ||||
|     <button onclick="exitToGanList();" class="ui large circular black icon button"><i class="angle left icon"></i></button> | ||||
|     <div style="max-width: 300px; margin-top: 1em;"> | ||||
| @@ -16,12 +17,12 @@ | ||||
|         <div class="field"> | ||||
|             <label>Network Name</label> | ||||
|             <input type="text" id="gaNetNameInput" placeholder=""> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="field"> | ||||
|           <label>Network Description</label> | ||||
|           <textarea id="gaNetDescInput" style="resize: none;"></textarea> | ||||
|           <button onclick="saveNameAndDesc(this);" class="ui basic right floated button" style="margin-top: 0.6em;"><i class="ui save icon"></i> Save</button> | ||||
|           <button onclick='$("#gannetDetailEdit").slideUp("fast");' class="ui basic right floated button" style="margin-top: 0.6em;"><i class="ui red remove icon"></i> Cancel</button> | ||||
|         <label>Network Description</label> | ||||
|         <textarea id="gaNetDescInput" style="resize: none;"></textarea> | ||||
|         <button onclick="saveNameAndDesc(this);" class="ui basic right floated button" style="margin-top: 0.6em;"><i class="ui save icon"></i> Save</button> | ||||
|         <button onclick='$("#gannetDetailEdit").slideUp("fast");' class="ui basic right floated button" style="margin-top: 0.6em;"><i class="ui red remove icon"></i> Cancel</button> | ||||
|         </div> | ||||
|         <br><br> | ||||
|     </div> | ||||
| @@ -215,7 +216,7 @@ | ||||
|         var cidr = $(".iprange.active").attr("cidr"); | ||||
| 
 | ||||
|         $.cjax({ | ||||
|             url: "/api/gan/network/setRange", | ||||
|             url: "./api/gan/network/setRange", | ||||
|             metohd: "POST", | ||||
|             data:{ | ||||
|                 netid: currentGANetID, | ||||
| @@ -225,9 +226,9 @@ | ||||
|             }, | ||||
|             success: function(data){ | ||||
|                 if (data.error != undefined){ | ||||
|                     msgbox(data.error, false, 5000) | ||||
|                     parent.msgbox(data.error, false, 5000) | ||||
|                 }else{ | ||||
|                     msgbox("Network Range Updated") | ||||
|                     parent.msgbox("Network Range Updated") | ||||
|                 } | ||||
|             } | ||||
|         }) | ||||
| @@ -241,7 +242,7 @@ | ||||
|             $(object).addClass("loading"); | ||||
|         } | ||||
|         $.cjax({ | ||||
|             url: "/api/gan/network/name", | ||||
|             url: "./api/gan/network/name", | ||||
|             method: "POST", | ||||
|             data: { | ||||
|                 netid: currentGANetID, | ||||
| @@ -252,7 +253,7 @@ | ||||
|                 initNetNameAndDesc(); | ||||
|                 if (object != undefined){ | ||||
|                     $(object).removeClass("loading"); | ||||
|                     msgbox("Network Metadata Updated"); | ||||
|                     parent.msgbox("Network Metadata Updated"); | ||||
|                 } | ||||
|                 $("#gannetDetailEdit").slideUp("fast"); | ||||
|             } | ||||
| @@ -261,9 +262,9 @@ | ||||
| 
 | ||||
|     function initNetNameAndDesc(){ | ||||
|         //Get the details of the net | ||||
|         $.get("/api/gan/network/name?netid=" + currentGANetID, function(data){ | ||||
|         $.get("./api/gan/network/name?netid=" + currentGANetID, function(data){ | ||||
|             if (data.error !== undefined){ | ||||
|                 msgbox(data.error, false, 6000); | ||||
|                 parent.msgbox(data.error, false, 6000); | ||||
|             }else{ | ||||
|                 $("#gaNetNameInput").val(data[0]); | ||||
|                 $(".ganetName").html(data[0]); | ||||
| @@ -274,10 +275,10 @@ | ||||
|     } | ||||
| 
 | ||||
|     function initNetDetails(){ | ||||
|          //Get the details of the net | ||||
|          $.get("/api/gan/network/list?netid=" + currentGANetID, function(data){ | ||||
|         //Get the details of the net | ||||
|         $.get("./api/gan/network/list?netid=" + currentGANetID, function(data){ | ||||
|             if (data.error !== undefined){ | ||||
|                 msgbox(data.error, false, 6000); | ||||
|                 parent.msgbox(data.error, false, 6000); | ||||
|             }else{ | ||||
|                 currentGaNetDetails = data; | ||||
|                 highlightCurrentGANetCIDR(); | ||||
| @@ -288,7 +289,7 @@ | ||||
|     //Handle delete IP from memeber | ||||
|     function deleteIpFromMemeber(memberid, ip){ | ||||
|         $.cjax({ | ||||
|             url: "/api/gan/members/ip", | ||||
|             url: "./api/gan/members/ip", | ||||
|             metohd: "POST", | ||||
|             data: { | ||||
|                 netid: currentGANetID,  | ||||
| @@ -298,9 +299,9 @@ | ||||
|             }, | ||||
|             success: function(data){ | ||||
|                 if (data.error != undefined){ | ||||
|                     msgbox(data.error, false, 5000); | ||||
|                     parent.msgbox(data.error, false, 5000); | ||||
|                 }else{ | ||||
|                     msgbox("IP removed from member " + memberid) | ||||
|                     parent.msgbox("IP removed from member " + memberid) | ||||
|                 } | ||||
|                 renderMemeberTable(); | ||||
|             } | ||||
| @@ -330,12 +331,12 @@ | ||||
|         } | ||||
| 
 | ||||
|         if (!isValidIPv4Address(newip)){ | ||||
|             msgbox(newip + " is not a valid IPv4 address", false, 5000) | ||||
|             parent.msgbox(newip + " is not a valid IPv4 address", false, 5000) | ||||
|             return | ||||
|         } | ||||
| 
 | ||||
|         $.cjax({ | ||||
|             url: "/api/gan/members/ip", | ||||
|             url: "./api/gan/members/ip", | ||||
|             metohd: "POST", | ||||
|             data: { | ||||
|                 netid: currentGANetID,  | ||||
| @@ -345,9 +346,9 @@ | ||||
|             }, | ||||
|             success: function(data){ | ||||
|                 if (data.error != undefined){ | ||||
|                     msgbox(data.error, false, 5000); | ||||
|                     parent.msgbox(data.error, false, 5000); | ||||
|                 }else{ | ||||
|                     msgbox("IP added to member " + memberid) | ||||
|                     parent.msgbox("IP added to member " + memberid) | ||||
|                 } | ||||
|                 renderMemeberTable(); | ||||
|             } | ||||
| @@ -357,7 +358,7 @@ | ||||
|     //Member table populate | ||||
|     function renderMemeberTable(forceUpdate = false) { | ||||
|         $.ajax({ | ||||
|             url: '/api/gan/members/list?netid=' + currentGANetID + '&detail=true', | ||||
|             url: './api/gan/members/list?netid=' + currentGANetID + '&detail=true', | ||||
|             type: 'GET', | ||||
|             success: function(data) { | ||||
|                 let tableBody = $('#networkMemeberTable'); | ||||
| @@ -462,7 +463,7 @@ | ||||
|             let addr = $(this).attr("addr"); | ||||
|             let targetDOM = $(this); | ||||
|             $.cjax({ | ||||
|                 url: "/api/gan/members/name", | ||||
|                 url: "./api/gan/members/name", | ||||
|                 method: "POST", | ||||
|                 data: { | ||||
|                     netid: currentGANetID, | ||||
| @@ -481,14 +482,14 @@ | ||||
| 
 | ||||
|     function renameMember(targetMemberAddr){ | ||||
|         if (targetMemberAddr == ""){ | ||||
|             msgbox("Member address cannot be empty", false, 5000) | ||||
|             parent.msgbox("Member address cannot be empty", false, 5000) | ||||
|             return | ||||
|         } | ||||
| 
 | ||||
|         let newname = prompt("Enter a easy manageable name for " + targetMemberAddr, ""); | ||||
|         if (newname != null && newname.trim() != "") { | ||||
|           $.cjax({ | ||||
|             url: "/api/gan/members/name", | ||||
|         $.cjax({ | ||||
|             url: "./api/gan/members/name", | ||||
|             method: "POST", | ||||
|             data: { | ||||
|                 netid: currentGANetID, | ||||
| @@ -497,13 +498,13 @@ | ||||
|             }, | ||||
|             success: function(data){ | ||||
|                 if (data.error != undefined){ | ||||
|                     msgbox(data.error, false, 6000); | ||||
|                     parent.msgbox(data.error, false, 6000); | ||||
|                 }else{ | ||||
|                     msgbox("Member Name Updated");  | ||||
|                     parent.msgbox("Member Name Updated");  | ||||
|                 } | ||||
|                 renderMemeberTable(true); | ||||
|             } | ||||
|           }) | ||||
|         }) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| @@ -554,7 +555,7 @@ | ||||
|         let targetMemberAddr = $(object).attr("addr"); | ||||
|         let isAuthed = object.checked; | ||||
|         $.cjax({ | ||||
|             url: "/api/gan/members/authorize", | ||||
|             url: "./api/gan/members/authorize", | ||||
|             method: "POST", | ||||
|             data: { | ||||
|                 netid:currentGANetID,  | ||||
| @@ -563,12 +564,12 @@ | ||||
|             }, | ||||
|             success: function(data){ | ||||
|                 if (data.error != undefined){ | ||||
|                     msgbox(data.error, false, 6000); | ||||
|                     parent.msgbox(data.error, false, 6000); | ||||
|                 }else{ | ||||
|                     if (isAuthed){ | ||||
|                         msgbox("Member Authorized"); | ||||
|                         parent.msgbox("Member Authorized"); | ||||
|                     }else{ | ||||
|                         msgbox("Member Deauthorized"); | ||||
|                         parent.msgbox("Member Deauthorized"); | ||||
|                     } | ||||
|                      | ||||
|                 } | ||||
| @@ -579,25 +580,26 @@ | ||||
|     } | ||||
| 
 | ||||
|     function handleMemberDelete(addr){ | ||||
|         if (confirm("Confirm delete member " + addr + " ?")){ | ||||
|             $.cjax({ | ||||
|                 url: "/api/gan/members/delete", | ||||
|                 method: "POST", | ||||
|                 data: { | ||||
|                     netid:currentGANetID,  | ||||
|                     memid: addr,  | ||||
|                 }, | ||||
|                 success: function(data){ | ||||
|                     if (data.error != undefined){ | ||||
|                         msgbox(data.error, false, 6000); | ||||
|                     }else{ | ||||
|                         msgbox("Member Deleted"); | ||||
|        parent.confirmBox("Confirm delete member " + addr + " ?", function(choice){ | ||||
|             if (choice){ | ||||
|                 $.cjax({ | ||||
|                     url: "./api/gan/members/delete", | ||||
|                     method: "POST", | ||||
|                     data: { | ||||
|                         netid:currentGANetID,  | ||||
|                         memid: addr,  | ||||
|                     }, | ||||
|                     success: function(data){ | ||||
|                         if (data.error != undefined){ | ||||
|                             parent.msgbox(data.error, false, 6000); | ||||
|                         }else{ | ||||
|                             parent.msgbox("Member Deleted"); | ||||
|                         } | ||||
|                         renderMemeberTable(true); | ||||
|                     } | ||||
|                     renderMemeberTable(true); | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|         | ||||
|                 }); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     //Add and remove this controller node to network as member | ||||
| @@ -606,7 +608,7 @@ | ||||
|         $(".addControllerToNetworkBtn").addClass("loading"); | ||||
|      | ||||
|         $.cjax({ | ||||
|             url: "/api/gan/network/join", | ||||
|             url: "./api/gan/network/join", | ||||
|             method: "POST", | ||||
|             data: { | ||||
|                 netid:currentGANetID,  | ||||
| @@ -615,13 +617,18 @@ | ||||
|                 $(".addControllerToNetworkBtn").removeClass("disabled"); | ||||
|                 $(".addControllerToNetworkBtn").removeClass("loading"); | ||||
|                 if (data.error != undefined){ | ||||
|                     msgbox(data.error, false, 6000); | ||||
|                     parent.msgbox(data.error, false, 6000); | ||||
|                 }else{ | ||||
|                     msgbox("Controller joint " + currentGANetID); | ||||
|                     parent.msgbox("Controller joint " + currentGANetID); | ||||
|                 } | ||||
|                 setTimeout(function(){ | ||||
|                     renderMemeberTable(true); | ||||
|                 }, 3000) | ||||
|             }, | ||||
|             error: function(){ | ||||
|                 $(".addControllerToNetworkBtn").removeClass("disabled"); | ||||
|                 $(".addControllerToNetworkBtn").removeClass("loading"); | ||||
| 
 | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
| @@ -631,16 +638,16 @@ | ||||
|         $(".removeControllerFromNetworkBtn").addClass("loading"); | ||||
|          | ||||
|         $.cjax({ | ||||
|             url: "/api/gan/network/leave", | ||||
|             url: "./api/gan/network/leave", | ||||
|             method: "POST", | ||||
|             data: { | ||||
|                 netid:currentGANetID,  | ||||
|             }, | ||||
|             success: function(data){ | ||||
|                 if (data.error != undefined){ | ||||
|                     msgbox(data.error, false, 6000); | ||||
|                     parent.msgbox(data.error, false, 6000); | ||||
|                 }else{ | ||||
|                     msgbox("Controller left " + currentGANetID); | ||||
|                     parent.msgbox("Controller left " + currentGANetID); | ||||
|                 } | ||||
|                 renderMemeberTable(true); | ||||
|                 $(".removeControllerFromNetworkBtn").removeClass("disabled"); | ||||
| @@ -669,19 +676,77 @@ | ||||
|          | ||||
|     } | ||||
| 
 | ||||
|     //Switch from other tabs back to this, exit to GAN list | ||||
|     tabSwitchEventBind["gan"] = function(){ | ||||
|         exitToGanList(); | ||||
|     } | ||||
| 
 | ||||
|     //Exit point | ||||
|     function exitToGanList(){ | ||||
|         $("#gan").load("./components/gan.html", function(){ | ||||
|             if (tabSwitchEventBind["gan"]){ | ||||
|                 tabSwitchEventBind["gan"](); | ||||
|             } | ||||
|         }); | ||||
|         location.href = "./index.html" | ||||
|     } | ||||
| 
 | ||||
|     //Debug functions | ||||
|     if (typeof(msgbox) == "undefined"){ | ||||
|         msgbox = function(msg, error=false, timeout=3000){ | ||||
|             console.log(msg); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     function ip2long (argIP) { | ||||
|         //  discuss at: https://locutus.io/php/ip2long/ | ||||
|         // original by: Waldo Malqui Silva (https://waldo.malqui.info) | ||||
|         // improved by: Victor | ||||
|         //  revised by: fearphage (https://my.opera.com/fearphage/) | ||||
|         //  revised by: Theriault (https://github.com/Theriault) | ||||
|         //    estarget: es2015 | ||||
|         //   example 1: ip2long('192.0.34.166') | ||||
|         //   returns 1: 3221234342 | ||||
|         //   example 2: ip2long('0.0xABCDEF') | ||||
|         //   returns 2: 11259375 | ||||
|         //   example 3: ip2long('255.255.255.256') | ||||
|         //   returns 3: false | ||||
|         let i = 0 | ||||
|         // PHP allows decimal, octal, and hexadecimal IP components. | ||||
|         // PHP allows between 1 (e.g. 127) to 4 (e.g 127.0.0.1) components. | ||||
|         const pattern = new RegExp([ | ||||
|             '^([1-9]\\d*|0[0-7]*|0x[\\da-f]+)', | ||||
|             '(?:\\.([1-9]\\d*|0[0-7]*|0x[\\da-f]+))?', | ||||
|             '(?:\\.([1-9]\\d*|0[0-7]*|0x[\\da-f]+))?', | ||||
|             '(?:\\.([1-9]\\d*|0[0-7]*|0x[\\da-f]+))?$' | ||||
|         ].join(''), 'i') | ||||
|         argIP = argIP.match(pattern) // Verify argIP format. | ||||
|         if (!argIP) { | ||||
|             // Invalid format. | ||||
|             return false | ||||
|         } | ||||
|         // Reuse argIP variable for component counter. | ||||
|         argIP[0] = 0 | ||||
|         for (i = 1; i < 5; i += 1) { | ||||
|             argIP[0] += !!((argIP[i] || '').length) | ||||
|             argIP[i] = parseInt(argIP[i]) || 0 | ||||
|         } | ||||
|         // Continue to use argIP for overflow values. | ||||
|         // PHP does not allow any component to overflow. | ||||
|         argIP.push(256, 256, 256, 256) | ||||
|         // Recalculate overflow of last component supplied to make up for missing components. | ||||
|         argIP[4 + argIP[0]] *= Math.pow(256, 4 - argIP[0]) | ||||
|         if (argIP[1] >= argIP[5] || | ||||
|             argIP[2] >= argIP[6] || | ||||
|             argIP[3] >= argIP[7] || | ||||
|             argIP[4] >= argIP[8]) { | ||||
|             return false | ||||
|         } | ||||
|         return argIP[1] * (argIP[0] === 1 || 16777216) + | ||||
|             argIP[2] * (argIP[0] <= 2 || 65536) + | ||||
|             argIP[3] * (argIP[0] <= 3 || 256) + | ||||
|             argIP[4] * 1 | ||||
|     } | ||||
| 
 | ||||
|     function long2ip (ip) { | ||||
|         //  discuss at: https://locutus.io/php/long2ip/ | ||||
|         // original by: Waldo Malqui Silva (https://fayr.us/waldo/) | ||||
|         //   example 1: long2ip( 3221234342 ) | ||||
|         //   returns 1: '192.0.34.166' | ||||
|         if (!isFinite(ip)) { | ||||
|             return false | ||||
|         } | ||||
|         return [ip >>> 24 & 0xFF, ip >>> 16 & 0xFF, ip >>> 8 & 0xFF, ip & 0xFF].join('.') | ||||
|     } | ||||
|      | ||||
| </script> | ||||
							
								
								
									
										267
									
								
								example/plugins/ztnc/web/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										267
									
								
								example/plugins/ztnc/web/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,267 @@ | ||||
| <html> | ||||
|     <head> | ||||
|         <meta name="apple-mobile-web-app-capable" content="yes" /> | ||||
|         <meta name="viewport" content="user-scalable=no, width=device-width, initial-scale=1, maximum-scale=1"/> | ||||
|         <meta charset="UTF-8"> | ||||
|         <meta name="theme-color" content="#4b75ff"> | ||||
|         <meta name="zoraxy.csrf.Token" content="{{.csrfToken}}"> | ||||
|         <link rel="icon" type="image/png" href="/favicon.png" /> | ||||
|         <title>Global Area Network | Zoraxy</title> | ||||
|         <link rel="stylesheet" href="/script/semantic/semantic.min.css"> | ||||
|         <script src="/script/jquery-3.6.0.min.js"></script> | ||||
|         <script src="/script/semantic/semantic.min.js"></script> | ||||
|         <script src="/script/tablesort.js"></script> | ||||
|         <script src="/script/countryCode.js"></script> | ||||
|         <script src="/script/chart.js"></script> | ||||
|         <script src="/script/utils.js"></script> | ||||
|         <link rel="stylesheet" href="/main.css"> | ||||
|         <style> | ||||
|             body{ | ||||
|                 background:none; | ||||
|             } | ||||
|         </style> | ||||
|     </head> | ||||
| <body> | ||||
|     <!-- Dark theme script must be included after body tag--> | ||||
|     <link rel="stylesheet" href="/darktheme.css"> | ||||
|     <script src="/script/darktheme.js"></script> | ||||
|     <div id="ganetWindow" class="standardContainer"> | ||||
|         <div class="ui basic segment"> | ||||
|             <h2>Global Area Network</h2> | ||||
|             <p>Virtual Network Hub that allows all networked devices to communicate as if they all reside in the same physical data center or cloud region</p> | ||||
|         </div> | ||||
|         <div class="gansnetworks"> | ||||
|             <div class="ganstats ui basic segment"> | ||||
|                 <div style="float: right; max-width: 300px; margin-top: 0.4em;"> | ||||
|                     <h1 class="ui header" style="text-align: right;"> | ||||
|                         <span class="ganControllerID"></span> | ||||
|                         <div class="sub header">Network Controller ID</div> | ||||
|                     </h1> | ||||
|                 </div> | ||||
|                 <div class="ui list"> | ||||
|                     <div class="item"> | ||||
|                         <i class="exchange icon"></i> | ||||
|                         <div class="content"> | ||||
|                         <div class="header" style="font-size: 1.2em;" id="ganetCount">0</div> | ||||
|                         <div class="description">Networks</div> | ||||
|                         </div> | ||||
|                     </div> | ||||
|                     <div class="item"> | ||||
|                         <i class="desktop icon"></i> | ||||
|                         <div class="content"> | ||||
|                         <div class="header" style="font-size: 1.2em;" id="ganodeCount">0</div> | ||||
|                         <div class="description" id="connectedNodes" count="0">Connected Nodes</div> | ||||
|                     </div> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|             <div class="ganlist"> | ||||
|                 <button class="ui basic orange button" onclick="addGANet();">Create New Network</button> | ||||
|                 <div class="ui divider"></div> | ||||
|                 <!--  | ||||
|                 <div class="ui icon input" style="margin-bottom: 1em;"> | ||||
|                     <input type="text" placeholder="Search a Network"> | ||||
|                     <i class="circular search link icon"></i> | ||||
|                 </div>--> | ||||
|                 <div style="width: 100%; overflow-x: auto;"> | ||||
|                     <table class="ui celled basic unstackable striped table"> | ||||
|                         <thead> | ||||
|                             <tr> | ||||
|                                 <th>Network ID</th> | ||||
|                                 <th>Name</th> | ||||
|                                 <th>Description</th> | ||||
|                                 <th>Subnet (Assign Range)</th> | ||||
|                                 <th>Nodes</th> | ||||
|                                 <th>Actions</th> | ||||
|                             </tr> | ||||
|                         </thead> | ||||
|                         <tbody id="GANetList"> | ||||
|                             <tr> | ||||
|                                 <td colspan="6"><i class="ui green circle check icon"></i> No Global Area Network Found on this host</td> | ||||
|                             </tr> | ||||
|                         </tbody> | ||||
|                     </table> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
|     <script> | ||||
|         /* | ||||
|             Network Management Functions | ||||
|         */ | ||||
|         function handleAddNetwork(){ | ||||
|             let networkName = $("#networkName").val().trim(); | ||||
|             if (networkName == ""){ | ||||
|                 parent.msgbox("Network name cannot be empty", false, 5000); | ||||
|                 return; | ||||
|             } | ||||
|      | ||||
|             //Add network with default settings | ||||
|             addGANet(networkName, "192.168.196.0/24"); | ||||
|             $("#networkName").val(""); | ||||
|         } | ||||
|      | ||||
|         function initGANetID(){ | ||||
|             $.get("./api/gan/network/info", function(data){ | ||||
|                 if (data.error !== undefined){ | ||||
|                     parent.msgbox(data.error, false, 5000) | ||||
|                 }else{ | ||||
|                     if (data != ""){ | ||||
|                         $(".ganControllerID").text(data); | ||||
|                     } | ||||
|                 } | ||||
|             }) | ||||
|         } | ||||
|      | ||||
|         function addGANet() { | ||||
|             $.cjax({ | ||||
|                 url: "./api/gan/network/add", | ||||
|                 type: "POST", | ||||
|                 dataType: "json", | ||||
|                 data: {}, | ||||
|                 success: function(response) { | ||||
|                     if (response.error != undefined){ | ||||
|                         parent.msgbox(response.error, false, 5000); | ||||
|                     }else{ | ||||
|                         parent.msgbox("Network added successfully"); | ||||
|                     } | ||||
|                     console.log("Network added successfully:", response); | ||||
|                     listGANet(); | ||||
|                 }, | ||||
|                 error: function(xhr, status, error) { | ||||
|                     console.log("Error adding network:", error); | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|      | ||||
|         function listGANet(){ | ||||
|             $("#connectedNodes").attr("count", "0"); | ||||
|      | ||||
|             $.get("./api/gan/network/list", function(data){ | ||||
|                 $("#GANetList").empty(); | ||||
|                 if (data.error != undefined){ | ||||
|                     console.log(data.error); | ||||
|                     parent.msgbox("Unable to load auth token for GANet", false, 5000); | ||||
|                     //token error or no zerotier found | ||||
|                     $(".gansnetworks").addClass("disabled"); | ||||
|                     $("#GANetList").append(`<tr> | ||||
|                         <td colspan="6"><i class="red times circle icon"></i> Auth token access error or not found</td> | ||||
|                     </tr>`); | ||||
|                     $(".ganControllerID").text('Access Denied'); | ||||
|                 }else{ | ||||
|                     var nodeCount = 0; | ||||
|                     data.forEach(function(gan){ | ||||
|                         $("#GANetList").append(`<tr class="ganetEntry" addr="${gan.nwid}"> | ||||
|                             <td><a href="#" onclick="event.preventDefault(); openGANetDetails('${gan.nwid}');">${gan.nwid}</a></td> | ||||
|                             <td>${gan.name}</td> | ||||
|                             <td class="gandesc" addr="${gan.nwid}"></td> | ||||
|                             <td class="ganetSubnet"></td> | ||||
|                             <td class="ganetNodes"></td> | ||||
|                             <td> | ||||
|                                 <button onclick="openGANetDetails('${gan.nwid}');" class="ui tiny basic icon button" title="Edit Network"><i class="edit icon"></i></button> | ||||
|                                 <button onclick="removeGANet('${gan.nwid}');" class="ui tiny basic icon button" title="Remove Network"><i class="red remove icon"></i></button> | ||||
|                             </td> | ||||
|                         </tr>`); | ||||
|      | ||||
|                         nodeCount += 0; | ||||
|                     }); | ||||
|      | ||||
|                     if (data.length == 0){ | ||||
|                         $("#GANetList").append(`<tr> | ||||
|                             <td colspan="6"><i class="ui green circle check icon"></i> No Global Area Network Found on this host</td> | ||||
|                         </tr>`); | ||||
|                     } | ||||
|      | ||||
|                     $("#ganodeCount").text(nodeCount); | ||||
|                     $("#ganetCount").text(data.length); | ||||
|      | ||||
|                     //Load description | ||||
|                     $(".gandesc").each(function(){ | ||||
|                         let addr = $(this).attr("addr"); | ||||
|                         let domEle = $(this); | ||||
|                         $.get("./api/gan/network/name?netid=" + addr, function(data){ | ||||
|                             $(domEle).text(data[1]); | ||||
|                         }); | ||||
|                     }); | ||||
|      | ||||
|                     $(".ganetEntry").each(function(){ | ||||
|                         let addr = $(this).attr("addr"); | ||||
|                         let subnetEle = $(this).find(".ganetSubnet"); | ||||
|                         let nodeEle = $(this).find(".ganetNodes"); | ||||
|      | ||||
|                         $.get("./api/gan/network/list?netid=" + addr, function(data){ | ||||
|                             if (data.routes != undefined && data.routes.length > 0){ | ||||
|                                  | ||||
|                                 if (data.ipAssignmentPools != undefined && data.ipAssignmentPools.length > 0){ | ||||
|                                     $(subnetEle).html(`${data.routes[0].target} <br> (${data.ipAssignmentPools[0].ipRangeStart} - ${data.ipAssignmentPools[0].ipRangeEnd})`); | ||||
|                                 }else{ | ||||
|                                     $(subnetEle).html(`${data.routes[0].target}<br>(Unassigned Range)`); | ||||
|                                 } | ||||
|                             }else{ | ||||
|                                 $(subnetEle).text("Unassigned"); | ||||
|                             } | ||||
|                             //console.log(data); | ||||
|                         }); | ||||
|      | ||||
|                         $.get("./api/gan/members/list?netid=" + addr, function(data){ | ||||
|                             $(nodeEle).text(data.length); | ||||
|                             let currentNodesCount = parseInt($("#connectedNodes").attr("count")); | ||||
|                             currentNodesCount += data.length; | ||||
|                             $("#connectedNodes").attr("count", currentNodesCount); | ||||
|                             $("#ganodeCount").text($("#connectedNodes").attr("count")); | ||||
|                         }) | ||||
|                     }); | ||||
|                 } | ||||
|             }) | ||||
|         } | ||||
|      | ||||
|         //Remove the given GANet | ||||
|         function removeGANet(netid){ | ||||
|             //Reusing Zoraxy confirm box | ||||
|             parent.confirmBox("Confirm remove " + netid + "?", function(choice){ | ||||
|                 if (choice == true){ | ||||
|                     $.cjax({ | ||||
|                         url: "./api/gan/network/remove", | ||||
|                         type: "POST", | ||||
|                         dataType: "json", | ||||
|                         data: { | ||||
|                             id: netid, | ||||
|                         }, | ||||
|                         success: function(data){ | ||||
|                             if (data.error != undefined){ | ||||
|                                 parent.msgbox(data.error, false, 5000); | ||||
|                             }else{ | ||||
|                                 parent.msgbox("Net " + netid + " removed"); | ||||
|                             } | ||||
|                             listGANet(); | ||||
|                         } | ||||
|                     }); | ||||
|                 } | ||||
|             }); | ||||
|              | ||||
|         } | ||||
|      | ||||
|         function openGANetDetails(netid){ | ||||
|             $("#ganetWindow").load("./details.html", function(){ | ||||
|                 setTimeout(function(){ | ||||
|                     initGanetDetails(netid); | ||||
|                 }); | ||||
|             }); | ||||
|             | ||||
|         } | ||||
|          | ||||
|         $(document).ready(function(){ | ||||
|             listGANet(); | ||||
|             initGANetID(); | ||||
|         }); | ||||
|  | ||||
|         if (typeof(msgbox) == "undefined"){ | ||||
|             msgbox = function(msg, error=false, timeout=3000){ | ||||
|                 console.log(msg); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         | ||||
|     </script> | ||||
| </body> | ||||
| </html> | ||||
| @@ -1,5 +1,5 @@ | ||||
| # PLATFORMS := darwin/amd64 darwin/arm64 freebsd/amd64 linux/386 linux/amd64 linux/arm linux/arm64 linux/mipsle windows/386 windows/amd64 windows/arm windows/arm64 | ||||
| PLATFORMS := linux/amd64 linux/386 linux/arm linux/arm64 linux/mipsle linux/riscv64 windows/amd64 | ||||
| PLATFORMS := linux/amd64 linux/386 linux/arm linux/arm64 linux/mipsle linux/riscv64 windows/amd64 freebsd/amd64 | ||||
| temp = $(subst /, ,$@) | ||||
| os = $(word 1, $(temp)) | ||||
| arch = $(word 2, $(temp)) | ||||
|   | ||||
| @@ -3,6 +3,7 @@ package main | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"net/http" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/google/uuid" | ||||
| @@ -545,3 +546,71 @@ func handleWhitelistEnable(w http.ResponseWriter, r *http.Request) { | ||||
| 		utils.SendOK(w) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func handleWhitelistAllowLoopback(w http.ResponseWriter, r *http.Request) { | ||||
| 	enable, _ := utils.PostPara(r, "enable") | ||||
| 	ruleID, err := utils.PostPara(r, "id") | ||||
| 	if err != nil { | ||||
| 		ruleID = "default" | ||||
| 	} | ||||
|  | ||||
| 	rule, err := accessController.GetAccessRuleByID(ruleID) | ||||
| 	if err != nil { | ||||
| 		utils.SendErrorResponse(w, err.Error()) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if enable == "" { | ||||
| 		//Return the current enabled state | ||||
| 		currentEnabled := rule.WhitelistAllowLocalAndLoopback | ||||
| 		js, _ := json.Marshal(currentEnabled) | ||||
| 		utils.SendJSONResponse(w, string(js)) | ||||
| 	} else { | ||||
| 		if enable == "true" { | ||||
| 			rule.ToggleAllowLoopback(true) | ||||
| 		} else if enable == "false" { | ||||
| 			rule.ToggleAllowLoopback(false) | ||||
| 		} else { | ||||
| 			utils.SendErrorResponse(w, "invalid enable state: only true and false is accepted") | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		utils.SendOK(w) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // List all quick ban ip address | ||||
| func handleListQuickBan(w http.ResponseWriter, r *http.Request) { | ||||
| 	currentSummary := statisticCollector.GetCurrentDailySummary() | ||||
| 	type quickBanEntry struct { | ||||
| 		IpAddr      string | ||||
| 		Count       int | ||||
| 		CountryCode string | ||||
| 	} | ||||
| 	result := []quickBanEntry{} | ||||
| 	currentSummary.RequestClientIp.Range(func(key, value interface{}) bool { | ||||
| 		ip := key.(string) | ||||
| 		count := value.(int) | ||||
| 		thisEntry := quickBanEntry{ | ||||
| 			IpAddr: ip, | ||||
| 			Count:  count, | ||||
| 		} | ||||
|  | ||||
| 		//Get the country code | ||||
| 		geoinfo, err := geodbStore.ResolveCountryCodeFromIP(ip) | ||||
| 		if err == nil { | ||||
| 			thisEntry.CountryCode = geoinfo.CountryIsoCode | ||||
| 		} | ||||
|  | ||||
| 		result = append(result, thisEntry) | ||||
| 		return true | ||||
| 	}) | ||||
|  | ||||
| 	//Sort result based on count | ||||
| 	sort.Slice(result, func(i, j int) bool { | ||||
| 		return result[i].Count > result[j].Count | ||||
| 	}) | ||||
|  | ||||
| 	js, _ := json.Marshal(result) | ||||
| 	utils.SendJSONResponse(w, string(js)) | ||||
| } | ||||
|   | ||||
							
								
								
									
										70
									
								
								src/api.go
									
									
									
									
									
								
							
							
						
						
									
										70
									
								
								src/api.go
									
									
									
									
									
								
							| @@ -2,6 +2,7 @@ package main | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"io/fs" | ||||
| 	"net/http" | ||||
| 	"net/http/pprof" | ||||
|  | ||||
| @@ -29,6 +30,7 @@ func RegisterHTTPProxyAPIs(authRouter *auth.RouterDef) { | ||||
| 	authRouter.HandleFunc("/api/proxy/status", ReverseProxyStatus) | ||||
| 	authRouter.HandleFunc("/api/proxy/toggle", ReverseProxyToggleRuleSet) | ||||
| 	authRouter.HandleFunc("/api/proxy/list", ReverseProxyList) | ||||
| 	authRouter.HandleFunc("/api/proxy/listTags", ReverseProxyListTags) | ||||
| 	authRouter.HandleFunc("/api/proxy/detail", ReverseProxyListDetail) | ||||
| 	authRouter.HandleFunc("/api/proxy/edit", ReverseProxyHandleEditEndpoint) | ||||
| 	authRouter.HandleFunc("/api/proxy/setAlias", ReverseProxyHandleAlias) | ||||
| @@ -78,9 +80,10 @@ func RegisterTLSAPIs(authRouter *auth.RouterDef) { | ||||
| 	authRouter.HandleFunc("/api/cert/delete", handleCertRemove) | ||||
| } | ||||
|  | ||||
| // Register the APIs for Authentication handlers like Authelia and OAUTH2 | ||||
| // Register the APIs for Authentication handlers like Forward Auth and OAUTH2 | ||||
| func RegisterAuthenticationHandlerAPIs(authRouter *auth.RouterDef) { | ||||
| 	authRouter.HandleFunc("/api/sso/Authelia", autheliaRouter.HandleSetAutheliaURLAndHTTPS) | ||||
| 	authRouter.HandleFunc("/api/sso/forward-auth", forwardAuthRouter.HandleAPIOptions) | ||||
| 	authRouter.HandleFunc("/api/sso/OAuth2", oauth2Router.HandleSetOAuth2Settings) | ||||
| } | ||||
|  | ||||
| // Register the APIs for redirection rules management functions | ||||
| @@ -88,6 +91,7 @@ func RegisterRedirectionAPIs(authRouter *auth.RouterDef) { | ||||
| 	authRouter.HandleFunc("/api/redirect/list", handleListRedirectionRules) | ||||
| 	authRouter.HandleFunc("/api/redirect/add", handleAddRedirectionRule) | ||||
| 	authRouter.HandleFunc("/api/redirect/delete", handleDeleteRedirectionRule) | ||||
| 	authRouter.HandleFunc("/api/redirect/edit", handleEditRedirectionRule) | ||||
| 	authRouter.HandleFunc("/api/redirect/regex", handleToggleRedirectRegexpSupport) | ||||
| } | ||||
|  | ||||
| @@ -113,6 +117,9 @@ func RegisterAccessRuleAPIs(authRouter *auth.RouterDef) { | ||||
| 	authRouter.HandleFunc("/api/whitelist/ip/add", handleIpWhitelistAdd) | ||||
| 	authRouter.HandleFunc("/api/whitelist/ip/remove", handleIpWhitelistRemove) | ||||
| 	authRouter.HandleFunc("/api/whitelist/enable", handleWhitelistEnable) | ||||
| 	authRouter.HandleFunc("/api/whitelist/allowLocal", handleWhitelistAllowLoopback) | ||||
| 	/* Quick Ban List */ | ||||
| 	authRouter.HandleFunc("/api/quickban/list", handleListQuickBan) | ||||
| } | ||||
|  | ||||
| // Register the APIs for path blocking rules management functions, WIP | ||||
| @@ -140,24 +147,6 @@ func RegisterStatisticalAPIs(authRouter *auth.RouterDef) { | ||||
| 	authRouter.HandleFunc("/api/utm/list", HandleUptimeMonitorListing) | ||||
| } | ||||
|  | ||||
| // Register the APIs for Global Area Network management functions, Will be moving to plugin soon | ||||
| func RegisterGANAPIs(authRouter *auth.RouterDef) { | ||||
| 	authRouter.HandleFunc("/api/gan/network/info", ganManager.HandleGetNodeID) | ||||
| 	authRouter.HandleFunc("/api/gan/network/add", ganManager.HandleAddNetwork) | ||||
| 	authRouter.HandleFunc("/api/gan/network/remove", ganManager.HandleRemoveNetwork) | ||||
| 	authRouter.HandleFunc("/api/gan/network/list", ganManager.HandleListNetwork) | ||||
| 	authRouter.HandleFunc("/api/gan/network/name", ganManager.HandleNetworkNaming) | ||||
| 	//authRouter.HandleFunc("/api/gan/network/detail", ganManager.HandleNetworkDetails) | ||||
| 	authRouter.HandleFunc("/api/gan/network/setRange", ganManager.HandleSetRanges) | ||||
| 	authRouter.HandleFunc("/api/gan/network/join", ganManager.HandleServerJoinNetwork) | ||||
| 	authRouter.HandleFunc("/api/gan/network/leave", ganManager.HandleServerLeaveNetwork) | ||||
| 	authRouter.HandleFunc("/api/gan/members/list", ganManager.HandleMemberList) | ||||
| 	authRouter.HandleFunc("/api/gan/members/ip", ganManager.HandleMemberIP) | ||||
| 	authRouter.HandleFunc("/api/gan/members/name", ganManager.HandleMemberNaming) | ||||
| 	authRouter.HandleFunc("/api/gan/members/authorize", ganManager.HandleMemberAuthorization) | ||||
| 	authRouter.HandleFunc("/api/gan/members/delete", ganManager.HandleMemberDelete) | ||||
| } | ||||
|  | ||||
| // Register the APIs for Stream (TCP / UDP) Proxy management functions | ||||
| func RegisterStreamProxyAPIs(authRouter *auth.RouterDef) { | ||||
| 	authRouter.HandleFunc("/api/streamprox/config/add", streamProxyManager.HandleAddProxyConfig) | ||||
| @@ -234,6 +223,28 @@ func RegisterNetworkUtilsAPIs(authRouter *auth.RouterDef) { | ||||
| 	authRouter.HandleFunc("/api/tools/fwdproxy/port", forwardProxy.HandlePort) | ||||
| } | ||||
|  | ||||
| func RegisterPluginAPIs(authRouter *auth.RouterDef) { | ||||
| 	authRouter.HandleFunc("/api/plugins/list", pluginManager.HandleListPlugins) | ||||
| 	authRouter.HandleFunc("/api/plugins/enable", pluginManager.HandleEnablePlugin) | ||||
| 	authRouter.HandleFunc("/api/plugins/disable", pluginManager.HandleDisablePlugin) | ||||
| 	authRouter.HandleFunc("/api/plugins/icon", pluginManager.HandleLoadPluginIcon) | ||||
| 	authRouter.HandleFunc("/api/plugins/info", pluginManager.HandlePluginInfo) | ||||
|  | ||||
| 	authRouter.HandleFunc("/api/plugins/groups/list", pluginManager.HandleListPluginGroups) | ||||
| 	authRouter.HandleFunc("/api/plugins/groups/add", pluginManager.HandleAddPluginToGroup) | ||||
| 	authRouter.HandleFunc("/api/plugins/groups/remove", pluginManager.HandleRemovePluginFromGroup) | ||||
| 	authRouter.HandleFunc("/api/plugins/groups/deleteTag", pluginManager.HandleRemovePluginGroup) | ||||
|  | ||||
| 	authRouter.HandleFunc("/api/plugins/store/list", pluginManager.HandleListDownloadablePlugins) | ||||
| 	authRouter.HandleFunc("/api/plugins/store/resync", pluginManager.HandleResyncPluginList) | ||||
| 	authRouter.HandleFunc("/api/plugins/store/install", pluginManager.HandleInstallPlugin) | ||||
| 	authRouter.HandleFunc("/api/plugins/store/uninstall", pluginManager.HandleUninstallPlugin) | ||||
|  | ||||
| 	// Developer options | ||||
| 	authRouter.HandleFunc("/api/plugins/developer/enableAutoReload", pluginManager.HandleEnableHotReload) | ||||
| 	authRouter.HandleFunc("/api/plugins/developer/setAutoReloadInterval", pluginManager.HandleSetHotReloadInterval) | ||||
| } | ||||
|  | ||||
| // Register the APIs for Auth functions, due to scoping issue some functions are defined here | ||||
| func RegisterAuthAPIs(requireAuth bool, targetMux *http.ServeMux) { | ||||
| 	targetMux.HandleFunc("/api/auth/login", authAgent.HandleLogin) | ||||
| @@ -315,13 +326,20 @@ func initAPIs(targetMux *http.ServeMux) { | ||||
| 		}, | ||||
| 	}) | ||||
|  | ||||
| 	//Register the standard web services urls | ||||
| 	fs := http.FileServer(http.FS(webres)) | ||||
| 	if DEVELOPMENT_BUILD { | ||||
| 		fs = http.FileServer(http.Dir("web/")) | ||||
| 	// Register the standard web services URLs | ||||
| 	var staticWebRes http.Handler | ||||
| 	if *development_build { | ||||
| 		staticWebRes = http.FileServer(http.Dir("web/")) | ||||
| 	} else { | ||||
| 		subFS, err := fs.Sub(webres, "web") | ||||
| 		if err != nil { | ||||
| 			panic("Failed to strip 'web/' from embedded resources: " + err.Error()) | ||||
| 		} | ||||
| 		staticWebRes = http.FileServer(http.FS(subFS)) | ||||
| 	} | ||||
|  | ||||
| 	//Add a layer of middleware for advance control | ||||
| 	advHandler := FSHandler(fs) | ||||
| 	advHandler := FSHandler(staticWebRes) | ||||
| 	targetMux.Handle("/", advHandler) | ||||
|  | ||||
| 	//Register the APIs | ||||
| @@ -333,12 +351,12 @@ func initAPIs(targetMux *http.ServeMux) { | ||||
| 	RegisterAccessRuleAPIs(authRouter) | ||||
| 	RegisterPathRuleAPIs(authRouter) | ||||
| 	RegisterStatisticalAPIs(authRouter) | ||||
| 	RegisterGANAPIs(authRouter) | ||||
| 	RegisterStreamProxyAPIs(authRouter) | ||||
| 	RegisterMDNSAPIs(authRouter) | ||||
| 	RegisterNetworkUtilsAPIs(authRouter) | ||||
| 	RegisterACMEAndAutoRenewerAPIs(authRouter) | ||||
| 	RegisterStaticWebServerAPIs(authRouter) | ||||
| 	RegisterPluginAPIs(authRouter) | ||||
|  | ||||
| 	//Account Reset | ||||
| 	targetMux.HandleFunc("/api/account/reset", HandleAdminAccountResetEmail) | ||||
|   | ||||
| @@ -54,6 +54,11 @@ func LoadReverseProxyConfig(configFilepath string) error { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	//Make sure the tags are not nil | ||||
| 	if thisConfigEndpoint.Tags == nil { | ||||
| 		thisConfigEndpoint.Tags = []string{} | ||||
| 	} | ||||
|  | ||||
| 	//Matching domain not set. Assume root | ||||
| 	if thisConfigEndpoint.RootOrMatchingDomain == "" { | ||||
| 		thisConfigEndpoint.RootOrMatchingDomain = "/" | ||||
| @@ -175,8 +180,8 @@ func ExportConfigAsZip(w http.ResponseWriter, r *http.Request) { | ||||
|  | ||||
| 	// Set the Content-Type header to indicate it's a zip file | ||||
| 	w.Header().Set("Content-Type", "application/zip") | ||||
| 	// Set the Content-Disposition header to specify the file name | ||||
| 	w.Header().Set("Content-Disposition", "attachment; filename=\"config.zip\"") | ||||
| 	// Set the Content-Disposition header to specify the file name, add timestamp to the filename | ||||
| 	w.Header().Set("Content-Disposition", "attachment; filename=\"zoraxy-config-"+time.Now().Format("2006-01-02-15-04-05")+".zip\"") | ||||
|  | ||||
| 	// Create a zip writer | ||||
| 	zipWriter := zip.NewWriter(w) | ||||
|   | ||||
							
								
								
									
										67
									
								
								src/def.go
									
									
									
									
									
								
							
							
						
						
									
										67
									
								
								src/def.go
									
									
									
									
									
								
							| @@ -13,23 +13,25 @@ import ( | ||||
| 	"net/http" | ||||
| 	"time" | ||||
|  | ||||
| 	"imuslab.com/zoraxy/mod/auth/sso/oauth2" | ||||
|  | ||||
| 	"imuslab.com/zoraxy/mod/access" | ||||
| 	"imuslab.com/zoraxy/mod/acme" | ||||
| 	"imuslab.com/zoraxy/mod/auth" | ||||
| 	"imuslab.com/zoraxy/mod/auth/sso/authelia" | ||||
| 	"imuslab.com/zoraxy/mod/auth/sso/forward" | ||||
| 	"imuslab.com/zoraxy/mod/database" | ||||
| 	"imuslab.com/zoraxy/mod/dockerux" | ||||
| 	"imuslab.com/zoraxy/mod/dynamicproxy/loadbalance" | ||||
| 	"imuslab.com/zoraxy/mod/dynamicproxy/redirection" | ||||
| 	"imuslab.com/zoraxy/mod/email" | ||||
| 	"imuslab.com/zoraxy/mod/forwardproxy" | ||||
| 	"imuslab.com/zoraxy/mod/ganserv" | ||||
| 	"imuslab.com/zoraxy/mod/geodb" | ||||
| 	"imuslab.com/zoraxy/mod/info/logger" | ||||
| 	"imuslab.com/zoraxy/mod/info/logviewer" | ||||
| 	"imuslab.com/zoraxy/mod/mdns" | ||||
| 	"imuslab.com/zoraxy/mod/netstat" | ||||
| 	"imuslab.com/zoraxy/mod/pathrule" | ||||
| 	"imuslab.com/zoraxy/mod/plugins" | ||||
| 	"imuslab.com/zoraxy/mod/sshprox" | ||||
| 	"imuslab.com/zoraxy/mod/statistic" | ||||
| 	"imuslab.com/zoraxy/mod/statistic/analytic" | ||||
| @@ -42,31 +44,33 @@ import ( | ||||
| const ( | ||||
| 	/* Build Constants */ | ||||
| 	SYSTEM_NAME       = "Zoraxy" | ||||
| 	SYSTEM_VERSION    = "3.1.6" | ||||
| 	DEVELOPMENT_BUILD = false /* Development: Set to false to use embedded web fs */ | ||||
| 	SYSTEM_VERSION    = "3.2.3" | ||||
| 	DEVELOPMENT_BUILD = false | ||||
|  | ||||
| 	/* System Constants */ | ||||
| 	TMP_FOLDER                 = "./tmp" | ||||
| 	WEBSERV_DEFAULT_PORT       = 5487 | ||||
| 	MDNS_HOSTNAME_PREFIX       = "zoraxy_" /* Follow by node UUID */ | ||||
| 	MDNS_IDENTIFY_DEVICE_TYPE  = "Network Gateway" | ||||
| 	MDNS_IDENTIFY_DOMAIN       = "zoraxy.aroz.org" | ||||
| 	MDNS_IDENTIFY_VENDOR       = "imuslab.com" | ||||
| 	MDNS_SCAN_TIMEOUT          = 30 /* Seconds */ | ||||
| 	MDNS_SCAN_UPDATE_INTERVAL  = 15 /* Minutes */ | ||||
| 	GEODB_CACHE_CLEAR_INTERVAL = 15 /* Minutes */ | ||||
| 	ACME_AUTORENEW_CONFIG_PATH = "./conf/acme_conf.json" | ||||
| 	CSRF_COOKIENAME            = "zoraxy_csrf" | ||||
| 	LOG_PREFIX                 = "zr" | ||||
| 	LOG_EXTENSION              = ".log" | ||||
| 	TMP_FOLDER                   = "./tmp" | ||||
| 	WEBSERV_DEFAULT_PORT         = 5487 | ||||
| 	MDNS_HOSTNAME_PREFIX         = "zoraxy_" /* Follow by node UUID */ | ||||
| 	MDNS_IDENTIFY_DEVICE_TYPE    = "Network Gateway" | ||||
| 	MDNS_IDENTIFY_DOMAIN         = "zoraxy.aroz.org" | ||||
| 	MDNS_IDENTIFY_VENDOR         = "imuslab.com" | ||||
| 	MDNS_SCAN_TIMEOUT            = 30 /* Seconds */ | ||||
| 	MDNS_SCAN_UPDATE_INTERVAL    = 15 /* Minutes */ | ||||
| 	GEODB_CACHE_CLEAR_INTERVAL   = 15 /* Minutes */ | ||||
| 	ACME_AUTORENEW_CONFIG_PATH   = "./conf/acme_conf.json" | ||||
| 	CSRF_COOKIENAME              = "zoraxy_csrf" | ||||
| 	LOG_PREFIX                   = "zr" | ||||
| 	LOG_EXTENSION                = ".log" | ||||
| 	STATISTIC_AUTO_SAVE_INTERVAL = 600 /* Seconds */ | ||||
|  | ||||
| 	/* Configuration Folder Storage Path Constants */ | ||||
| 	CONF_HTTP_PROXY   = "./conf/proxy" | ||||
| 	CONF_STREAM_PROXY = "./conf/streamproxy" | ||||
| 	CONF_CERT_STORE   = "./conf/certs" | ||||
| 	CONF_REDIRECTION  = "./conf/redirect" | ||||
| 	CONF_ACCESS_RULE  = "./conf/access" | ||||
| 	CONF_PATH_RULE    = "./conf/rules/pathrules" | ||||
| 	CONF_HTTP_PROXY    = "./conf/proxy" | ||||
| 	CONF_STREAM_PROXY  = "./conf/streamproxy" | ||||
| 	CONF_CERT_STORE    = "./conf/certs" | ||||
| 	CONF_REDIRECTION   = "./conf/redirect" | ||||
| 	CONF_ACCESS_RULE   = "./conf/access" | ||||
| 	CONF_PATH_RULE     = "./conf/rules/pathrules" | ||||
| 	CONF_PLUGIN_GROUPS = "./conf/plugin_groups.json" | ||||
| ) | ||||
|  | ||||
| /* System Startup Flags */ | ||||
| @@ -78,8 +82,6 @@ var ( | ||||
| 	allowSshLoopback           = flag.Bool("sshlb", false, "Allow loopback web ssh connection (DANGER)") | ||||
| 	allowMdnsScanning          = flag.Bool("mdns", true, "Enable mDNS scanner and transponder") | ||||
| 	mdnsName                   = flag.String("mdnsname", "", "mDNS name, leave empty to use default (zoraxy_{node-uuid}.local)") | ||||
| 	ztAuthToken                = flag.String("ztauth", "", "ZeroTier authtoken for the local node") | ||||
| 	ztAPIPort                  = flag.Int("ztport", 9993, "ZeroTier controller API port") | ||||
| 	runningInDocker            = flag.Bool("docker", false, "Run Zoraxy in docker compatibility mode") | ||||
| 	acmeAutoRenewInterval      = flag.Int("autorenew", 86400, "ACME auto TLS/SSL certificate renew check interval (seconds)") | ||||
| 	acmeCertAutoRenewDays      = flag.Int("earlyrenew", 30, "Number of days to early renew a soon expiring certificate (days)") | ||||
| @@ -87,15 +89,21 @@ var ( | ||||
| 	allowWebFileManager        = flag.Bool("webfm", true, "Enable web file manager for static web server root folder") | ||||
| 	enableAutoUpdate           = flag.Bool("cfgupgrade", true, "Enable auto config upgrade if breaking change is detected") | ||||
|  | ||||
| 	/* Default Configuration Flags */ | ||||
| 	defaultInboundPort          = flag.Int("default_inbound_port", 443, "Default web server listening port") | ||||
| 	defaultEnableInboundTraffic = flag.Bool("default_inbound_enabled", true, "If web server is enabled by default") | ||||
|  | ||||
| 	/* Path Configuration Flags */ | ||||
| 	//path_database  = flag.String("dbpath", "./sys.db", "Database path") | ||||
| 	//path_conf      = flag.String("conf", "./conf", "Configuration folder path") | ||||
| 	path_uuid      = flag.String("uuid", "./sys.uuid", "sys.uuid file path") | ||||
| 	path_logFile   = flag.String("log", "./log", "Log folder path") | ||||
| 	path_webserver = flag.String("webroot", "./www", "Static web server root folder. Only allow change in start paramters") | ||||
| 	path_plugin    = flag.String("plugin", "./plugins", "Plugin folder path") | ||||
|  | ||||
| 	/* Maintaince Function Flags */ | ||||
| 	geoDbUpdate = flag.Bool("update_geoip", false, "Download the latest GeoIP data and exit") | ||||
| 	/* Maintaince & Development Function Flags */ | ||||
| 	geoDbUpdate       = flag.Bool("update_geoip", false, "Download the latest GeoIP data and exit") | ||||
| 	development_build = flag.Bool("dev", false, "Use external web folder for UI development") | ||||
| ) | ||||
|  | ||||
| /* Global Variables and Handlers */ | ||||
| @@ -127,7 +135,6 @@ var ( | ||||
| 	statisticCollector *statistic.Collector      //Collecting statistic from visitors | ||||
| 	uptimeMonitor      *uptime.Monitor           //Uptime monitor service worker | ||||
| 	mdnsScanner        *mdns.MDNSHost            //mDNS discovery services | ||||
| 	ganManager         *ganserv.NetworkManager   //Global Area Network Manager | ||||
| 	webSshManager      *sshprox.Manager          //Web SSH connection service | ||||
| 	streamProxyManager *streamproxy.Manager      //Stream Proxy Manager for TCP / UDP forwarding | ||||
| 	acmeHandler        *acme.ACMEHandler         //Handler for ACME Certificate renew | ||||
| @@ -135,9 +142,11 @@ var ( | ||||
| 	staticWebServer    *webserv.WebServer        //Static web server for hosting simple stuffs | ||||
| 	forwardProxy       *forwardproxy.Handler     //HTTP Forward proxy, basically VPN for web browser | ||||
| 	loadBalancer       *loadbalance.RouteManager //Global scope loadbalancer, store the state of the lb routing | ||||
| 	pluginManager      *plugins.Manager          //Plugin manager for managing plugins | ||||
|  | ||||
| 	//Authentication Provider | ||||
| 	autheliaRouter *authelia.AutheliaRouter //Authelia router for Authelia authentication | ||||
| 	forwardAuthRouter *forward.AuthRouter  // Forward Auth router for Authelia/Authentik/etc authentication | ||||
| 	oauth2Router      *oauth2.OAuth2Router //OAuth2Router router for OAuth2Router authentication | ||||
|  | ||||
| 	//Helper modules | ||||
| 	EmailSender       *email.Sender         //Email sender that handle email sending | ||||
|   | ||||
							
								
								
									
										141
									
								
								src/go.mod
									
									
									
									
									
								
							
							
						
						
									
										141
									
								
								src/go.mod
									
									
									
									
									
								
							| @@ -5,56 +5,54 @@ go 1.22.0 | ||||
| toolchain go1.22.2 | ||||
|  | ||||
| require ( | ||||
| 	github.com/armon/go-radix v1.0.0 | ||||
| 	github.com/boltdb/bolt v1.3.1 | ||||
| 	github.com/docker/docker v27.0.0+incompatible | ||||
| 	github.com/go-acme/lego/v4 v4.19.2 | ||||
| 	github.com/go-acme/lego/v4 v4.21.0 | ||||
| 	github.com/go-ping/ping v1.1.0 | ||||
| 	github.com/go-session/session v3.1.2+incompatible | ||||
| 	github.com/google/uuid v1.6.0 | ||||
| 	github.com/gorilla/sessions v1.2.2 | ||||
| 	github.com/gorilla/websocket v1.5.1 | ||||
| 	github.com/grandcat/zeroconf v1.0.0 | ||||
| 	github.com/likexian/whois v1.15.1 | ||||
| 	github.com/microcosm-cc/bluemonday v1.0.26 | ||||
| 	golang.org/x/net v0.29.0 | ||||
| 	golang.org/x/sys v0.25.0 | ||||
| 	golang.org/x/text v0.18.0 | ||||
| 	github.com/monperrus/crawler-user-agents v1.1.0 | ||||
| 	github.com/shirou/gopsutil/v4 v4.25.1 | ||||
| 	github.com/syndtr/goleveldb v1.0.0 | ||||
| 	golang.org/x/net v0.33.0 | ||||
| 	golang.org/x/text v0.21.0 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	cloud.google.com/go/auth v0.9.3 // indirect | ||||
| 	cloud.google.com/go/auth/oauth2adapt v0.2.4 // indirect | ||||
| 	cloud.google.com/go/auth v0.13.0 // indirect | ||||
| 	cloud.google.com/go/auth/oauth2adapt v0.2.6 // indirect | ||||
| 	github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/resourcegraph/armresourcegraph v0.9.0 // indirect | ||||
| 	github.com/benbjohnson/clock v1.3.0 // indirect | ||||
| 	github.com/golang-jwt/jwt v3.2.2+incompatible // indirect | ||||
| 	github.com/ebitengine/purego v0.8.2 // indirect | ||||
| 	github.com/go-ole/go-ole v1.2.6 // indirect | ||||
| 	github.com/golang-jwt/jwt/v5 v5.2.1 // indirect | ||||
| 	github.com/golang/snappy v0.0.1 // indirect | ||||
| 	github.com/huaweicloud/huaweicloud-sdk-go-v3 v0.1.114 // indirect | ||||
| 	github.com/huaweicloud/huaweicloud-sdk-go-v3 v0.1.128 // indirect | ||||
| 	github.com/opentracing/opentracing-go v1.2.1-0.20220228012449-10b1cf09e00b // indirect | ||||
| 	github.com/peterhellberg/link v1.2.0 // indirect | ||||
| 	github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c // indirect | ||||
| 	github.com/shopspring/decimal v1.3.1 // indirect | ||||
| 	github.com/syndtr/goleveldb v1.0.0 // indirect | ||||
| 	github.com/tidwall/btree v0.0.0-20191029221954-400434d76274 // indirect | ||||
| 	github.com/tidwall/buntdb v1.1.2 // indirect | ||||
| 	github.com/tidwall/gjson v1.12.1 // indirect | ||||
| 	github.com/tidwall/grect v0.0.0-20161006141115-ba9a043346eb // indirect | ||||
| 	github.com/tidwall/match v1.1.1 // indirect | ||||
| 	github.com/tidwall/pretty v1.2.0 // indirect | ||||
| 	github.com/tidwall/rtree v0.0.0-20180113144539-6cd427091e0e // indirect | ||||
| 	github.com/tidwall/tinyqueue v0.0.0-20180302190814-1e39f5511563 // indirect | ||||
| 	github.com/tjfoc/gmsm v1.4.1 // indirect | ||||
| 	github.com/vultr/govultr/v3 v3.9.1 // indirect | ||||
| 	github.com/yusufpapurcu/wmi v1.2.4 // indirect | ||||
| 	go.mongodb.org/mongo-driver v1.12.0 // indirect | ||||
| 	golang.org/x/sys v0.28.0 // indirect | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	cloud.google.com/go/compute/metadata v0.5.1 // indirect | ||||
| 	cloud.google.com/go/compute/metadata v0.6.0 // indirect | ||||
| 	github.com/AdamSLevy/jsonrpc2/v14 v14.1.0 // indirect | ||||
| 	github.com/Azure/azure-sdk-for-go v68.0.0+incompatible // indirect | ||||
| 	github.com/Azure/azure-sdk-for-go/sdk/azcore v1.14.0 // indirect | ||||
| 	github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.7.0 // indirect | ||||
| 	github.com/Azure/azure-sdk-for-go/sdk/azcore v1.16.0 // indirect | ||||
| 	github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.8.0 // indirect | ||||
| 	github.com/Azure/azure-sdk-for-go/sdk/internal v1.10.0 // indirect | ||||
| 	github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/dns/armdns v1.2.0 // indirect | ||||
| 	github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/privatedns/armprivatedns v1.2.0 // indirect | ||||
| 	github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/privatedns/armprivatedns v1.3.0 // indirect | ||||
| 	github.com/Azure/go-autorest v14.2.0+incompatible // indirect | ||||
| 	github.com/Azure/go-autorest/autorest v0.11.29 // indirect | ||||
| 	github.com/Azure/go-autorest/autorest/adal v0.9.22 // indirect | ||||
| @@ -67,28 +65,28 @@ require ( | ||||
| 	github.com/AzureAD/microsoft-authentication-library-for-go v1.2.2 // indirect | ||||
| 	github.com/Microsoft/go-winio v0.4.14 // indirect | ||||
| 	github.com/OpenDNS/vegadns2client v0.0.0-20180418235048-a3fa4a771d87 // indirect | ||||
| 	github.com/aliyun/alibaba-cloud-sdk-go v1.63.15 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2 v1.30.5 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/config v1.27.33 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/credentials v1.17.32 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.13 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.17 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.17 // indirect | ||||
| 	github.com/aliyun/alibaba-cloud-sdk-go v1.63.72 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2 v1.32.7 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/config v1.28.7 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/credentials v1.17.48 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.22 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.26 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.26 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/internal/ini v1.8.1 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.4 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.19 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/lightsail v1.40.6 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/route53 v1.43.2 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/sso v1.22.7 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/ssooidc v1.26.7 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/sts v1.30.7 // indirect | ||||
| 	github.com/aws/smithy-go v1.20.4 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.12.1 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.12.7 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/lightsail v1.42.8 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/route53 v1.46.4 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/sso v1.24.8 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/ssooidc v1.28.7 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/service/sts v1.33.3 // indirect | ||||
| 	github.com/aws/smithy-go v1.22.1 // indirect | ||||
| 	github.com/aymerick/douceur v0.2.0 // indirect | ||||
| 	github.com/boombuler/barcode v1.0.1-0.20190219062509-6c824513bacc // indirect | ||||
| 	github.com/cenkalti/backoff v2.2.1+incompatible // indirect | ||||
| 	github.com/cenkalti/backoff/v4 v4.3.0 // indirect | ||||
| 	github.com/civo/civogo v0.3.11 // indirect | ||||
| 	github.com/cloudflare/cloudflare-go v0.104.0 // indirect | ||||
| 	github.com/cloudflare/cloudflare-go v0.112.0 // indirect | ||||
| 	github.com/containerd/log v0.1.0 // indirect | ||||
| 	github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect | ||||
| 	github.com/dimchansky/utfbom v1.1.1 // indirect | ||||
| @@ -98,25 +96,22 @@ require ( | ||||
| 	github.com/docker/go-units v0.5.0 // indirect | ||||
| 	github.com/fatih/structs v1.1.0 // indirect | ||||
| 	github.com/felixge/httpsnoop v1.0.4 // indirect | ||||
| 	github.com/fsnotify/fsnotify v1.7.0 // indirect | ||||
| 	github.com/fsnotify/fsnotify v1.8.0 // indirect | ||||
| 	github.com/ghodss/yaml v1.0.0 // indirect | ||||
| 	github.com/go-errors/errors v1.0.1 // indirect | ||||
| 	github.com/go-jose/go-jose/v4 v4.0.4 // indirect | ||||
| 	github.com/go-logr/logr v1.4.2 // indirect | ||||
| 	github.com/go-logr/stdr v1.2.2 // indirect | ||||
| 	github.com/go-oauth2/oauth2/v4 v4.5.2 | ||||
| 	github.com/go-resty/resty/v2 v2.13.1 // indirect | ||||
| 	github.com/go-viper/mapstructure/v2 v2.1.0 // indirect | ||||
| 	github.com/goccy/go-json v0.10.3 // indirect | ||||
| 	github.com/gofrs/uuid v4.4.0+incompatible | ||||
| 	github.com/go-resty/resty/v2 v2.16.2 // indirect | ||||
| 	github.com/go-viper/mapstructure/v2 v2.2.1 // indirect | ||||
| 	github.com/goccy/go-json v0.10.4 // indirect | ||||
| 	github.com/gogo/protobuf v1.3.2 // indirect | ||||
| 	github.com/golang-jwt/jwt/v4 v4.5.0 // indirect | ||||
| 	github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect | ||||
| 	github.com/golang-jwt/jwt/v4 v4.5.1 // indirect | ||||
| 	github.com/google/go-querystring v1.1.0 // indirect | ||||
| 	github.com/google/s2a-go v0.1.8 // indirect | ||||
| 	github.com/googleapis/enterprise-certificate-proxy v0.3.4 // indirect | ||||
| 	github.com/googleapis/gax-go/v2 v2.13.0 // indirect | ||||
| 	github.com/gophercloud/gophercloud v1.14.0 // indirect | ||||
| 	github.com/googleapis/gax-go/v2 v2.14.0 // indirect | ||||
| 	github.com/gophercloud/gophercloud v1.14.1 // indirect | ||||
| 	github.com/gorilla/csrf v1.7.2 | ||||
| 	github.com/gorilla/css v1.0.1 // indirect | ||||
| 	github.com/gorilla/securecookie v1.1.2 // indirect | ||||
| @@ -133,7 +128,7 @@ require ( | ||||
| 	github.com/kylelemons/godebug v1.1.0 // indirect | ||||
| 	github.com/labbsr0x/bindman-dns-webhook v1.0.2 // indirect | ||||
| 	github.com/labbsr0x/goh v1.0.1 // indirect | ||||
| 	github.com/linode/linodego v1.40.0 // indirect | ||||
| 	github.com/linode/linodego v1.44.0 // indirect | ||||
| 	github.com/liquidweb/liquidweb-cli v0.6.9 // indirect | ||||
| 	github.com/liquidweb/liquidweb-go v1.6.4 // indirect | ||||
| 	github.com/mattn/go-isatty v0.0.20 // indirect | ||||
| @@ -149,9 +144,9 @@ require ( | ||||
| 	github.com/namedotcom/go v0.0.0-20180403034216-08470befbe04 // indirect | ||||
| 	github.com/nrdcg/auroradns v1.1.0 // indirect | ||||
| 	github.com/nrdcg/bunny-go v0.0.0-20240207213615-dde5bf4577a3 // indirect | ||||
| 	github.com/nrdcg/desec v0.8.0 // indirect | ||||
| 	github.com/nrdcg/desec v0.10.0 // indirect | ||||
| 	github.com/nrdcg/dnspod-go v0.4.0 // indirect | ||||
| 	github.com/nrdcg/freemyip v0.2.0 // indirect | ||||
| 	github.com/nrdcg/freemyip v0.3.0 // indirect | ||||
| 	github.com/nrdcg/goinwx v0.10.0 // indirect | ||||
| 	github.com/nrdcg/mailinabox v0.2.0 // indirect | ||||
| 	github.com/nrdcg/namesilo v0.2.1 // indirect | ||||
| @@ -167,24 +162,22 @@ require ( | ||||
| 	github.com/pquerna/otp v1.4.0 // indirect | ||||
| 	github.com/sacloud/api-client-go v0.2.10 // indirect | ||||
| 	github.com/sacloud/go-http v0.1.8 // indirect | ||||
| 	github.com/sacloud/iaas-api-go v1.12.0 // indirect | ||||
| 	github.com/sacloud/iaas-api-go v1.14.0 // indirect | ||||
| 	github.com/sacloud/packages-go v0.0.10 // indirect | ||||
| 	github.com/scaleway/scaleway-sdk-go v1.0.0-beta.30 // indirect | ||||
| 	github.com/sirupsen/logrus v1.9.3 // indirect | ||||
| 	github.com/smartystreets/go-aws-auth v0.0.0-20180515143844-0c1422d1fdb9 // indirect | ||||
| 	github.com/softlayer/softlayer-go v1.1.5 // indirect | ||||
| 	github.com/softlayer/softlayer-go v1.1.7 // indirect | ||||
| 	github.com/softlayer/xmlrpc v0.0.0-20200409220501-5f089df7cb7e // indirect | ||||
| 	github.com/spf13/cast v1.6.0 // indirect | ||||
| 	github.com/stretchr/testify v1.9.0 // indirect | ||||
| 	github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/common v1.0.1002 // indirect | ||||
| 	github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/dnspod v1.0.1002 // indirect | ||||
| 	github.com/stretchr/testify v1.10.0 // indirect | ||||
| 	github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/common v1.0.1065 // indirect | ||||
| 	github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/dnspod v1.0.1065 // indirect | ||||
| 	github.com/transip/gotransip/v6 v6.26.0 // indirect | ||||
| 	github.com/ultradns/ultradns-go-sdk v1.7.0-20240913052650-970ca9a // indirect | ||||
| 	github.com/ultradns/ultradns-go-sdk v1.8.0-20241010134910-243eeec // indirect | ||||
| 	github.com/vinyldns/go-vinyldns v0.9.16 // indirect | ||||
| 	github.com/xlzd/gotp v0.1.0 | ||||
| 	github.com/yandex-cloud/go-genproto v0.0.0-20240911120709-1fa0cb6f47c2 // indirect | ||||
| 	github.com/yandex-cloud/go-sdk v0.0.0-20240911121212-e4e74d0d02f5 // indirect | ||||
| 	go.opencensus.io v0.24.0 // indirect | ||||
| 	github.com/yandex-cloud/go-genproto v0.0.0-20241220122821-aeb3b05efd1c // indirect | ||||
| 	github.com/yandex-cloud/go-sdk v0.0.0-20241220131134-2393e243c134 // indirect | ||||
| 	go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.54.0 // indirect | ||||
| 	go.opentelemetry.io/otel v1.29.0 // indirect | ||||
| 	go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.27.0 // indirect | ||||
| @@ -192,20 +185,20 @@ require ( | ||||
| 	go.opentelemetry.io/otel/sdk v1.28.0 // indirect | ||||
| 	go.opentelemetry.io/otel/trace v1.29.0 // indirect | ||||
| 	go.uber.org/ratelimit v0.3.0 // indirect | ||||
| 	golang.org/x/crypto v0.27.0 // indirect | ||||
| 	golang.org/x/mod v0.21.0 // indirect | ||||
| 	golang.org/x/oauth2 v0.23.0 // indirect | ||||
| 	golang.org/x/sync v0.8.0 // indirect | ||||
| 	golang.org/x/time v0.6.0 // indirect | ||||
| 	golang.org/x/tools v0.25.0 // indirect | ||||
| 	google.golang.org/api v0.197.0 // indirect | ||||
| 	google.golang.org/genproto v0.0.0-20240903143218-8af14fe29dc1 // indirect | ||||
| 	google.golang.org/genproto/googleapis/api v0.0.0-20240827150818-7e3bb234dfed // indirect | ||||
| 	google.golang.org/genproto/googleapis/rpc v0.0.0-20240903143218-8af14fe29dc1 // indirect | ||||
| 	google.golang.org/grpc v1.66.1 // indirect | ||||
| 	google.golang.org/protobuf v1.34.2 // indirect | ||||
| 	golang.org/x/crypto v0.31.0 // indirect | ||||
| 	golang.org/x/mod v0.22.0 // indirect | ||||
| 	golang.org/x/oauth2 v0.24.0 // indirect | ||||
| 	golang.org/x/sync v0.10.0 // indirect | ||||
| 	golang.org/x/time v0.8.0 // indirect | ||||
| 	golang.org/x/tools v0.28.0 // indirect | ||||
| 	google.golang.org/api v0.214.0 // indirect | ||||
| 	google.golang.org/genproto v0.0.0-20241021214115-324edc3d5d38 // indirect | ||||
| 	google.golang.org/genproto/googleapis/api v0.0.0-20241118233622-e639e219e697 // indirect | ||||
| 	google.golang.org/genproto/googleapis/rpc v0.0.0-20241209162323-e6fa225c2576 // indirect | ||||
| 	google.golang.org/grpc v1.67.1 // indirect | ||||
| 	google.golang.org/protobuf v1.35.2 // indirect | ||||
| 	gopkg.in/ini.v1 v1.67.0 // indirect | ||||
| 	gopkg.in/ns1/ns1-go.v2 v2.12.0 // indirect | ||||
| 	gopkg.in/ns1/ns1-go.v2 v2.13.0 // indirect | ||||
| 	gopkg.in/yaml.v2 v2.4.0 // indirect | ||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||
| 	gotest.tools/v3 v3.5.1 // indirect | ||||
|   | ||||
							
								
								
									
										399
									
								
								src/go.sum
									
									
									
									
									
								
							
							
						
						
									
										399
									
								
								src/go.sum
									
									
									
									
									
								
							| @@ -5,13 +5,13 @@ cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6A | ||||
| cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY= | ||||
| cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc= | ||||
| cloud.google.com/go v0.46.3/go.mod h1:a6bKKbmY7er1mI7TEI4lsAkts/mkhTSZK8w33B4RAg0= | ||||
| cloud.google.com/go/auth v0.9.3 h1:VOEUIAADkkLtyfr3BLa3R8Ed/j6w1jTBmARx+wb5w5U= | ||||
| cloud.google.com/go/auth v0.9.3/go.mod h1:7z6VY+7h3KUdRov5F1i8NDP5ZzWKYmEPO842BgCsmTk= | ||||
| cloud.google.com/go/auth/oauth2adapt v0.2.4 h1:0GWE/FUsXhf6C+jAkWgYm7X9tK8cuEIfy19DBn6B6bY= | ||||
| cloud.google.com/go/auth/oauth2adapt v0.2.4/go.mod h1:jC/jOpwFP6JBxhB3P5Rr0a9HLMC/Pe3eaL4NmdvqPtc= | ||||
| cloud.google.com/go/auth v0.13.0 h1:8Fu8TZy167JkW8Tj3q7dIkr2v4cndv41ouecJx0PAHs= | ||||
| cloud.google.com/go/auth v0.13.0/go.mod h1:COOjD9gwfKNKz+IIduatIhYJQIc0mG3H102r/EMxX6Q= | ||||
| cloud.google.com/go/auth/oauth2adapt v0.2.6 h1:V6a6XDu2lTwPZWOawrAa9HUK+DB2zfJyTuciBG5hFkU= | ||||
| cloud.google.com/go/auth/oauth2adapt v0.2.6/go.mod h1:AlmsELtlEBnaNTL7jCj8VQFLy6mbZv0s4Q7NGBeQ5E8= | ||||
| cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o= | ||||
| cloud.google.com/go/compute/metadata v0.5.1 h1:NM6oZeZNlYjiwYje+sYFjEpP0Q0zCan1bmQW/KmIrGs= | ||||
| cloud.google.com/go/compute/metadata v0.5.1/go.mod h1:C66sj2AluDcIqakBq/M8lw8/ybHgOZqin2obFxa/E5k= | ||||
| cloud.google.com/go/compute/metadata v0.6.0 h1:A6hENjEsCDtC1k8byVsgwvVcioamEHvZ4j01OwKxG9I= | ||||
| cloud.google.com/go/compute/metadata v0.6.0/go.mod h1:FjyFAW1MW0C203CEOMDTu3Dk1FlqW3Rga40jzHL4hfg= | ||||
| cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE= | ||||
| cloud.google.com/go/firestore v1.1.0/go.mod h1:ulACoGHTpvq5r8rxGJ4ddJZBZqakUQqClKRT5SZwBmk= | ||||
| cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I= | ||||
| @@ -21,22 +21,24 @@ github.com/AdamSLevy/jsonrpc2/v14 v14.1.0 h1:Dy3M9aegiI7d7PF1LUdjbVigJReo+QOceYs | ||||
| github.com/AdamSLevy/jsonrpc2/v14 v14.1.0/go.mod h1:ZakZtbCXxCz82NJvq7MoREtiQesnDfrtF6RFUGzQfLo= | ||||
| github.com/Azure/azure-sdk-for-go v68.0.0+incompatible h1:fcYLmCpyNYRnvJbPerq7U0hS+6+I79yEDJBqVNcqUzU= | ||||
| github.com/Azure/azure-sdk-for-go v68.0.0+incompatible/go.mod h1:9XXNKU+eRnpl9moKnB4QOLf1HestfXbmab5FXxiDBjc= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/azcore v1.14.0 h1:nyQWyZvwGTvunIMxi1Y9uXkcyr+I7TeNrr/foo4Kpk8= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/azcore v1.14.0/go.mod h1:l38EPgmsp71HHLq9j7De57JcKOWPyhrsW1Awm1JS6K0= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.7.0 h1:tfLQ34V6F7tVSwoTf/4lH5sE0o6eCJuNDTmH09nDpbc= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.7.0/go.mod h1:9kIvujWAA58nmPmWB1m23fyWic1kYZMxD9CxaWn4Qpg= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/azcore v1.16.0 h1:JZg6HRh6W6U4OLl6lk7BZ7BLisIzM9dG1R50zUk9C/M= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/azcore v1.16.0/go.mod h1:YL1xnZ6QejvQHWJrX/AvhFl4WW4rqHVoKspWNVwFk0M= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.8.0 h1:B/dfvscEQtew9dVuoxqxrUKKv8Ih2f55PydknDamU+g= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.8.0/go.mod h1:fiPSssYvltE08HJchL04dOy+RD4hgrjph0cwGGMntdI= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/azidentity/cache v0.3.0 h1:+m0M/LFxN43KvULkDNfdXOgrjtg6UYJPFBJyuEcRCAw= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/azidentity/cache v0.3.0/go.mod h1:PwOyop78lveYMRs6oCxjiVyBdyCgIYH6XHIVZO9/SFQ= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/internal v1.10.0 h1:ywEEhmNahHBihViHepv3xPBn1663uRv2t2q/ESv9seY= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/internal v1.10.0/go.mod h1:iZDifYGJTIgIIkYRNWPENUnqx6bJ2xnSDFI2tjwZNuY= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/dns/armdns v1.2.0 h1:lpOxwrQ919lCZoNCd69rVt8u1eLZuMORrGXqy8sNf3c= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/dns/armdns v1.2.0/go.mod h1:fSvRkb8d26z9dbL40Uf/OO6Vo9iExtZK3D0ulRV+8M0= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/internal/v2 v2.0.0 h1:PTFGRSlMKCQelWwxUyYVEUqseBJVemLyqWJjvMyt0do= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/internal/v2 v2.0.0/go.mod h1:LRr2FzBTQlONPPa5HREE5+RjSCTXl7BwOvYOaWTqCaI= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/privatedns/armprivatedns v1.2.0 h1:9Eih8XcEeQnFD0ntMlUDleKMzfeCeUfa+VbnDCI4AZs= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/privatedns/armprivatedns v1.2.0/go.mod h1:wGPyTi+aURdqPAGMZDQqnNs9IrShADF8w2WZb6bKeq0= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/internal/v3 v3.1.0 h1:2qsIIvxVT+uE6yrNldntJKlLRgxGbZ85kgtz5SNBhMw= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/internal/v3 v3.1.0/go.mod h1:AW8VEadnhw9xox+VaVd9sP7NjzOAnaZBLRH6Tq3cJ38= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/privatedns/armprivatedns v1.3.0 h1:yzrctSl9GMIQ5lHu7jc8olOsGjWDCsBpJhWqfGa/YIM= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/privatedns/armprivatedns v1.3.0/go.mod h1:GE4m0rnnfwLGX0Y9A9A25Zx5N/90jneT5ABevqzhuFQ= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/resourcegraph/armresourcegraph v0.9.0 h1:zLzoX5+W2l95UJoVwiyNS4dX8vHyQ6x2xRLoBBL9wMk= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/resourcegraph/armresourcegraph v0.9.0/go.mod h1:wVEOJfGTj0oPAUGA1JuRAvz/lxXQsWW16axmHPP47Bk= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/resources/armresources v1.1.1 h1:7CBQ+Ei8SP2c6ydQTGCCrS35bDxgTMfoP2miAwK++OU= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/resources/armresources v1.1.1/go.mod h1:c/wcGeGx5FUPbM/JltUYHZcKmigwyVLJlDq+4HdtXaw= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/resources/armresources v1.2.0 h1:Dd+RhdJn0OTtVGaeDLZpcumkIVCtA/3/Fo42+eoYvVM= | ||||
| github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/resources/armresources v1.2.0/go.mod h1:5kakwfW5CjC9KK+Q4wjXAg+ShuIm2mBMua0ZFj2C8PE= | ||||
| github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1 h1:UQHMgLO+TxOElx5B5HZ4hJQsoJ/PvUvKRhJHDQXO8P8= | ||||
| github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E= | ||||
| github.com/Azure/go-autorest v14.2.0+incompatible h1:V5VMDjClD3GiElqLWO7mz2MxNAK/vTfRHdAubSIPRgs= | ||||
| @@ -62,6 +64,8 @@ github.com/Azure/go-autorest/logger v0.2.1 h1:IG7i4p/mDa2Ce4TRyAO8IHnVhAVF3RFU+Z | ||||
| github.com/Azure/go-autorest/logger v0.2.1/go.mod h1:T9E3cAhj2VqvPOtCYAvby9aBXkZmbF5NWuPV8+WeEW8= | ||||
| github.com/Azure/go-autorest/tracing v0.6.0 h1:TYi4+3m5t6K48TGI9AUdb+IzbnSxvnvUMfuitfgcfuo= | ||||
| github.com/Azure/go-autorest/tracing v0.6.0/go.mod h1:+vhtPC754Xsa23ID7GlGsrdKBpUA79WCAKPPZVC2DeU= | ||||
| github.com/AzureAD/microsoft-authentication-extensions-for-go/cache v0.1.1 h1:WJTmL004Abzc5wDB5VtZG2PJk5ndYDgVacGqfirKxjM= | ||||
| github.com/AzureAD/microsoft-authentication-extensions-for-go/cache v0.1.1/go.mod h1:tCcJZ0uHAmvjsVYzEFivsRTN00oz5BEsRgQHu5JZ9WE= | ||||
| github.com/AzureAD/microsoft-authentication-library-for-go v1.2.2 h1:XHOnouVk1mxXfQidrMEnLlPk9UMeRtyBTnEFtxkV0kU= | ||||
| github.com/AzureAD/microsoft-authentication-library-for-go v1.2.2/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI= | ||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||
| @@ -72,49 +76,46 @@ github.com/Microsoft/go-winio v0.4.14/go.mod h1:qXqCSQ3Xa7+6tgxaGTIe4Kpcdsi+P8jB | ||||
| github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU= | ||||
| github.com/OpenDNS/vegadns2client v0.0.0-20180418235048-a3fa4a771d87 h1:xPMsUicZ3iosVPSIP7bW5EcGUzjiiMl1OYTe14y/R24= | ||||
| github.com/OpenDNS/vegadns2client v0.0.0-20180418235048-a3fa4a771d87/go.mod h1:iGLljf5n9GjT6kc0HBvyI1nOKnGQbNB66VzSNbK5iks= | ||||
| github.com/ajg/form v1.5.1 h1:t9c7v8JUKu/XxOGBU0yjNpaMloxGEJhUkqFRq0ibGeU= | ||||
| github.com/ajg/form v1.5.1/go.mod h1:uL1WgH+h2mgNtvBq0339dVnzXdBETtL2LeUXaIv25UY= | ||||
| github.com/ajstarks/svgo v0.0.0-20180226025133-644b8db467af/go.mod h1:K08gAheRH3/J6wwsYMMT4xOr94bZjxIelGM0+d/wbFw= | ||||
| github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= | ||||
| github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= | ||||
| github.com/aliyun/alibaba-cloud-sdk-go v1.63.15 h1:r2uwBUQhLhcPzaWz9tRJqc8MjYwHb+oF2+Q6467BF14= | ||||
| github.com/aliyun/alibaba-cloud-sdk-go v1.63.15/go.mod h1:SOSDHfe1kX91v3W5QiBsWSLqeLxImobbMX1mxrFHsVQ= | ||||
| github.com/andybalholm/brotli v1.0.4 h1:V7DdXeJtZscaqfNuAdSRuRFzuiKlHSC/Zh3zl9qY3JY= | ||||
| github.com/andybalholm/brotli v1.0.4/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig= | ||||
| github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY= | ||||
| github.com/aliyun/alibaba-cloud-sdk-go v1.63.72 h1:HvFZUzEbNvfe8F2Mg0wBGv90bPhWDxgVtDHR5zoBOU0= | ||||
| github.com/aliyun/alibaba-cloud-sdk-go v1.63.72/go.mod h1:SOSDHfe1kX91v3W5QiBsWSLqeLxImobbMX1mxrFHsVQ= | ||||
| github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o= | ||||
| github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY= | ||||
| github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= | ||||
| github.com/aws/aws-sdk-go-v2 v1.30.5 h1:mWSRTwQAb0aLE17dSzztCVJWI9+cRMgqebndjwDyK0g= | ||||
| github.com/aws/aws-sdk-go-v2 v1.30.5/go.mod h1:CT+ZPWXbYrci8chcARI3OmI/qgd+f6WtuLOoaIA8PR0= | ||||
| github.com/aws/aws-sdk-go-v2/config v1.27.33 h1:Nof9o/MsmH4oa0s2q9a0k7tMz5x/Yj5k06lDODWz3BU= | ||||
| github.com/aws/aws-sdk-go-v2/config v1.27.33/go.mod h1:kEqdYzRb8dd8Sy2pOdEbExTTF5v7ozEXX0McgPE7xks= | ||||
| github.com/aws/aws-sdk-go-v2/credentials v1.17.32 h1:7Cxhp/BnT2RcGy4VisJ9miUPecY+lyE9I8JvcZofn9I= | ||||
| github.com/aws/aws-sdk-go-v2/credentials v1.17.32/go.mod h1:P5/QMF3/DCHbXGEGkdbilXHsyTBX5D3HSwcrSc9p20I= | ||||
| github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.13 h1:pfQ2sqNpMVK6xz2RbqLEL0GH87JOwSxPV2rzm8Zsb74= | ||||
| github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.13/go.mod h1:NG7RXPUlqfsCLLFfi0+IpKN4sCB9D9fw/qTaSB+xRoU= | ||||
| github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.17 h1:pI7Bzt0BJtYA0N/JEC6B8fJ4RBrEMi1LBrkMdFYNSnQ= | ||||
| github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.17/go.mod h1:Dh5zzJYMtxfIjYW+/evjQ8uj2OyR/ve2KROHGHlSFqE= | ||||
| github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.17 h1:Mqr/V5gvrhA2gvgnF42Zh5iMiQNcOYthFYwCyrnuWlc= | ||||
| github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.17/go.mod h1:aLJpZlCmjE+V+KtN1q1uyZkfnUWpQGpbsn89XPKyzfU= | ||||
| github.com/armon/go-radix v1.0.0 h1:F4z6KzEeeQIMeLFa97iZU6vupzoecKdU5TX24SNppXI= | ||||
| github.com/armon/go-radix v1.0.0/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= | ||||
| github.com/aws/aws-sdk-go-v2 v1.32.7 h1:ky5o35oENWi0JYWUZkB7WYvVPP+bcRF5/Iq7JWSb5Rw= | ||||
| github.com/aws/aws-sdk-go-v2 v1.32.7/go.mod h1:P5WJBrYqqbWVaOxgH0X/FYYD47/nooaPOZPlQdmiN2U= | ||||
| github.com/aws/aws-sdk-go-v2/config v1.28.7 h1:GduUnoTXlhkgnxTD93g1nv4tVPILbdNQOzav+Wpg7AE= | ||||
| github.com/aws/aws-sdk-go-v2/config v1.28.7/go.mod h1:vZGX6GVkIE8uECSUHB6MWAUsd4ZcG2Yq/dMa4refR3M= | ||||
| github.com/aws/aws-sdk-go-v2/credentials v1.17.48 h1:IYdLD1qTJ0zanRavulofmqut4afs45mOWEI+MzZtTfQ= | ||||
| github.com/aws/aws-sdk-go-v2/credentials v1.17.48/go.mod h1:tOscxHN3CGmuX9idQ3+qbkzrjVIx32lqDSU1/0d/qXs= | ||||
| github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.22 h1:kqOrpojG71DxJm/KDPO+Z/y1phm1JlC8/iT+5XRmAn8= | ||||
| github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.22/go.mod h1:NtSFajXVVL8TA2QNngagVZmUtXciyrHOt7xgz4faS/M= | ||||
| github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.26 h1:I/5wmGMffY4happ8NOCuIUEWGUvvFp5NSeQcXl9RHcI= | ||||
| github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.26/go.mod h1:FR8f4turZtNy6baO0KJ5FJUmXH/cSkI9fOngs0yl6mA= | ||||
| github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.26 h1:zXFLuEuMMUOvEARXFUVJdfqZ4bvvSgdGRq/ATcrQxzM= | ||||
| github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.26/go.mod h1:3o2Wpy0bogG1kyOPrgkXA8pgIfEEv0+m19O9D5+W8y8= | ||||
| github.com/aws/aws-sdk-go-v2/internal/ini v1.8.1 h1:VaRN3TlFdd6KxX1x3ILT5ynH6HvKgqdiXoTxAF4HQcQ= | ||||
| github.com/aws/aws-sdk-go-v2/internal/ini v1.8.1/go.mod h1:FbtygfRFze9usAadmnGJNc8KsP346kEe+y2/oyhGAGc= | ||||
| github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.4 h1:KypMCbLPPHEmf9DgMGw51jMj77VfGPAN2Kv4cfhlfgI= | ||||
| github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.4/go.mod h1:Vz1JQXliGcQktFTN/LN6uGppAIRoLBR2bMvIMP0gOjc= | ||||
| github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.19 h1:rfprUlsdzgl7ZL2KlXiUAoJnI/VxfHCvDFr2QDFj6u4= | ||||
| github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.19/go.mod h1:SCWkEdRq8/7EK60NcvvQ6NXKuTcchAD4ROAsC37VEZE= | ||||
| github.com/aws/aws-sdk-go-v2/service/lightsail v1.40.6 h1:ea6TO3HgVeVTB2Ie1djyBFWBOc9CohpKbo/QZbGTCJQ= | ||||
| github.com/aws/aws-sdk-go-v2/service/lightsail v1.40.6/go.mod h1:D2TUTD3v6AWmE5LzdCXLWNFtoYbSf6IEjKh1ggbuVdw= | ||||
| github.com/aws/aws-sdk-go-v2/service/route53 v1.43.2 h1:957e1/SwXIfPi/0OUJkH9YnPZRe9G6Kisd/xUhF7AUE= | ||||
| github.com/aws/aws-sdk-go-v2/service/route53 v1.43.2/go.mod h1:343vcjcyOTuHTBBgUrOxPM36/jE96qLZnGL447ldrB0= | ||||
| github.com/aws/aws-sdk-go-v2/service/sso v1.22.7 h1:pIaGg+08llrP7Q5aiz9ICWbY8cqhTkyy+0SHvfzQpTc= | ||||
| github.com/aws/aws-sdk-go-v2/service/sso v1.22.7/go.mod h1:eEygMHnTKH/3kNp9Jr1n3PdejuSNcgwLe1dWgQtO0VQ= | ||||
| github.com/aws/aws-sdk-go-v2/service/ssooidc v1.26.7 h1:/Cfdu0XV3mONYKaOt1Gr0k1KvQzkzPyiKUdlWJqy+J4= | ||||
| github.com/aws/aws-sdk-go-v2/service/ssooidc v1.26.7/go.mod h1:bCbAxKDqNvkHxRaIMnyVPXPo+OaPRwvmgzMxbz1VKSA= | ||||
| github.com/aws/aws-sdk-go-v2/service/sts v1.30.7 h1:NKTa1eqZYw8tiHSRGpP0VtTdub/8KNk8sDkNPFaOKDE= | ||||
| github.com/aws/aws-sdk-go-v2/service/sts v1.30.7/go.mod h1:NXi1dIAGteSaRLqYgarlhP/Ij0cFT+qmCwiJqWh/U5o= | ||||
| github.com/aws/smithy-go v1.20.4 h1:2HK1zBdPgRbjFOHlfeQZfpC4r72MOb9bZkiFwggKO+4= | ||||
| github.com/aws/smithy-go v1.20.4/go.mod h1:irrKGvNn1InZwb2d7fkIRNucdfwR8R+Ts3wxYa/cJHg= | ||||
| github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.12.1 h1:iXtILhvDxB6kPvEXgsDhGaZCSC6LQET5ZHSdJozeI0Y= | ||||
| github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.12.1/go.mod h1:9nu0fVANtYiAePIBh2/pFUSwtJ402hLnp854CNoDOeE= | ||||
| github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.12.7 h1:8eUsivBQzZHqe/3FE+cqwfH+0p5Jo8PFM/QYQSmeZ+M= | ||||
| github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.12.7/go.mod h1:kLPQvGUmxn/fqiCrDeohwG33bq2pQpGeY62yRO6Nrh0= | ||||
| github.com/aws/aws-sdk-go-v2/service/lightsail v1.42.8 h1:+lmJoqxuUoPlSfGk5JYQQivd9YFjUvRZR6RPY+Wcx48= | ||||
| github.com/aws/aws-sdk-go-v2/service/lightsail v1.42.8/go.mod h1:Gg8/myP4+rgRi4+j9gQdbOEnMtwMAUUIeXo+nKCFVj8= | ||||
| github.com/aws/aws-sdk-go-v2/service/route53 v1.46.4 h1:0jMtawybbfpFEIMy4wvfyW2Z4YLr7mnuzT0fhR67Nrc= | ||||
| github.com/aws/aws-sdk-go-v2/service/route53 v1.46.4/go.mod h1:xlMODgumb0Pp8bzfpojqelDrf8SL9rb5ovwmwKJl+oU= | ||||
| github.com/aws/aws-sdk-go-v2/service/sso v1.24.8 h1:CvuUmnXI7ebaUAhbJcDy9YQx8wHR69eZ9I7q5hszt/g= | ||||
| github.com/aws/aws-sdk-go-v2/service/sso v1.24.8/go.mod h1:XDeGv1opzwm8ubxddF0cgqkZWsyOtw4lr6dxwmb6YQg= | ||||
| github.com/aws/aws-sdk-go-v2/service/ssooidc v1.28.7 h1:F2rBfNAL5UyswqoeWv9zs74N/NanhK16ydHW1pahX6E= | ||||
| github.com/aws/aws-sdk-go-v2/service/ssooidc v1.28.7/go.mod h1:JfyQ0g2JG8+Krq0EuZNnRwX0mU0HrwY/tG6JNfcqh4k= | ||||
| github.com/aws/aws-sdk-go-v2/service/sts v1.33.3 h1:Xgv/hyNgvLda/M9l9qxXc4UFSgppnRczLxlMs5Ae/QY= | ||||
| github.com/aws/aws-sdk-go-v2/service/sts v1.33.3/go.mod h1:5Gn+d+VaaRgsjewpMvGazt0WfcFO+Md4wLOuBfGR9Bc= | ||||
| github.com/aws/smithy-go v1.22.1 h1:/HPHZQ0g7f4eUeK6HKglFz8uwVfZKgoI25rb/J+dnro= | ||||
| github.com/aws/smithy-go v1.22.1/go.mod h1:irrKGvNn1InZwb2d7fkIRNucdfwR8R+Ts3wxYa/cJHg= | ||||
| github.com/aymerick/douceur v0.2.0 h1:Mv+mAeH1Q+n9Fr+oyamOlAkUNPWPlA8PPGR0QAaYuPk= | ||||
| github.com/aymerick/douceur v0.2.0/go.mod h1:wlT5vV2O3h55X9m7iVYN0TBM0NH/MmbLnd30/FjWUq4= | ||||
| github.com/benbjohnson/clock v1.3.0 h1:ip6w0uFQkncKQ979AypyG0ER7mqUSBdKLOgAle/AT8A= | ||||
| @@ -129,26 +130,24 @@ github.com/boltdb/bolt v1.3.1/go.mod h1:clJnj/oiGkjum5o1McbSZDSLxVThjynRyGBgiAx2 | ||||
| github.com/boombuler/barcode v1.0.1-0.20190219062509-6c824513bacc h1:biVzkmvwrH8WK8raXaxBx6fRVTlJILwEwQGL1I/ByEI= | ||||
| github.com/boombuler/barcode v1.0.1-0.20190219062509-6c824513bacc/go.mod h1:paBWMcWSl3LHKBqUq+rly7CNSldXjb2rDl3JlRe0mD8= | ||||
| github.com/c-bata/go-prompt v0.2.5/go.mod h1:vFnjEGDIIA/Lib7giyE4E9c50Lvl8j0S+7FVlAwDAVw= | ||||
| github.com/c2h5oh/datasize v0.0.0-20200112174442-28bbd4740fee/go.mod h1:S/7n9copUssQ56c7aAgHqftWO4LTf4xY6CGWt8Bc+3M= | ||||
| github.com/cenkalti/backoff v2.2.1+incompatible h1:tNowT99t7UNflLxfYYSlKYsBpXdEet03Pg2g16Swow4= | ||||
| github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= | ||||
| github.com/cenkalti/backoff/v4 v4.3.0 h1:MyRJ/UdXutAwSAT+s3wNd7MfTIcy71VQueUuFK343L8= | ||||
| github.com/cenkalti/backoff/v4 v4.3.0/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash v1.1.0 h1:a6HrQnmkObjyL+Gs60czilIUGqrzKutQD6XZog3p+ko= | ||||
| github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= | ||||
| github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= | ||||
| github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= | ||||
| github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= | ||||
| github.com/civo/civogo v0.3.11 h1:mON/fyrV946Sbk6paRtOSGsN+asCgCmHCgArf5xmGxM= | ||||
| github.com/civo/civogo v0.3.11/go.mod h1:7+GeeFwc4AYTULaEshpT2vIcl3Qq8HPoxA17viX3l6g= | ||||
| github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= | ||||
| github.com/cloudflare/cloudflare-go v0.104.0 h1:R/lB0dZupaZbOgibAH/BRrkFbZ6Acn/WsKg2iX2xXuY= | ||||
| github.com/cloudflare/cloudflare-go v0.104.0/go.mod h1:pfUQ4PIG4ISI0/Mmc21Bp86UnFU0ktmPf3iTgbSL+cM= | ||||
| github.com/cloudflare/cloudflare-go v0.112.0 h1:caFwqXdGJCl3rjVMgbPEn8iCYAg9JsRYV3dIVQE5d7g= | ||||
| github.com/cloudflare/cloudflare-go v0.112.0/go.mod h1:QB55kuJ5ZTeLNFcLJePfMuBilhu/LDKpLBmKFQIoSZ0= | ||||
| github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= | ||||
| github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= | ||||
| github.com/cncf/xds/go v0.0.0-20210312221358-fbca930ec8ed/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= | ||||
| github.com/cncf/xds/go v0.0.0-20210805033703-aa0b78936158/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= | ||||
| github.com/containerd/log v0.1.0 h1:TCJt7ioM2cr/tfR8GPbGf9/VRAX8D2B4PjzCpfX540I= | ||||
| github.com/containerd/log v0.1.0/go.mod h1:VRRf09a7mHDIRezVKTRCrOq78v577GXq3bSa3EhrzVo= | ||||
| github.com/coreos/bbolt v1.3.2/go.mod h1:iRUV2dpdMOn7Bo10OQBFzIJO9kkE559Wcmn+qkEiiKk= | ||||
| @@ -163,6 +162,8 @@ github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSs | ||||
| github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM= | ||||
| github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | ||||
| github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no= | ||||
| github.com/dimchansky/utfbom v1.1.1 h1:vV6w1AhK4VMnhBno/TPVCoK9U/LP0PkLCS9tbxHdi/U= | ||||
| github.com/dimchansky/utfbom v1.1.1/go.mod h1:SxdoEBH5qIqFocHMyGOXVAybYJdr71b1Q/j0mACtrfE= | ||||
| @@ -176,14 +177,11 @@ github.com/docker/go-connections v0.5.0 h1:USnMq7hx7gwdVZq1L49hLXaFtUdTADjXGp+uj | ||||
| github.com/docker/go-connections v0.5.0/go.mod h1:ov60Kzw0kKElRwhNs9UlUHAE/F9Fe6GLaXnqyDdmEXc= | ||||
| github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4= | ||||
| github.com/docker/go-units v0.5.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk= | ||||
| github.com/ebitengine/purego v0.8.2 h1:jPPGWs2sZ1UgOSgD2bClL0MJIqu58nOmIcBuXr62z1I= | ||||
| github.com/ebitengine/purego v0.8.2/go.mod h1:iIjxzd6CiRiOG0UyXP+V1+jWqUXVjPKLAI0mRfJZTmQ= | ||||
| github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= | ||||
| github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= | ||||
| github.com/envoyproxy/go-control-plane v0.9.9-0.20210512163311-63b5d3c536b0/go.mod h1:hliV/p42l8fGbc6Y9bQ70uLwIvmJyVE5k4iMKlh8wCQ= | ||||
| github.com/envoyproxy/go-control-plane v0.9.10-0.20210907150352-cf90f659a021/go.mod h1:AFq3mo9L8Lqqiid3OhADV3RfLJnjiw63cSpi+fDTRC0= | ||||
| github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= | ||||
| github.com/fasthttp-contrib/websocket v0.0.0-20160511215533-1f3b11f56072/go.mod h1:duJ4Jxv5lDcvg4QuQr0oowTf7dz4/CR8NtyCooz9HL8= | ||||
| github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= | ||||
| github.com/fatih/color v1.16.0 h1:zmkK9Ngbjj+K0yRhTVONQh1p/HknKYSlNT+vZCzyokM= | ||||
| github.com/fatih/color v1.16.0/go.mod h1:fL2Sau1YI5c0pdGEVCbKQbLXB6edEj1ZgiY4NijnWvE= | ||||
| @@ -197,14 +195,12 @@ github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7z | ||||
| github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | ||||
| github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= | ||||
| github.com/fsnotify/fsnotify v1.5.1/go.mod h1:T3375wBYaZdLLcVNkcVbzGHY7f1l/uK5T5Ai1i3InKU= | ||||
| github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA= | ||||
| github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM= | ||||
| github.com/gavv/httpexpect v2.0.0+incompatible h1:1X9kcRshkSKEjNJJxX9Y9mQ5BRfbxU5kORdjhlA1yX8= | ||||
| github.com/gavv/httpexpect v2.0.0+incompatible/go.mod h1:x+9tiU1YnrOvnB725RkpoLv1M62hOWzwo5OXotisrKc= | ||||
| github.com/fsnotify/fsnotify v1.8.0 h1:dAwr6QBTBZIkG8roQaJjGof0pp0EeF+tNV7YBP3F/8M= | ||||
| github.com/fsnotify/fsnotify v1.8.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0= | ||||
| github.com/ghodss/yaml v1.0.0 h1:wQHKEahhL6wmXdzwWG11gIVCkOv05bNOh+Rxn0yngAk= | ||||
| github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= | ||||
| github.com/go-acme/lego/v4 v4.19.2 h1:Y8hrmMvWETdqzzkRly7m98xtPJJivWFsgWi8fcvZo+Y= | ||||
| github.com/go-acme/lego/v4 v4.19.2/go.mod h1:wtDe3dDkmV4/oI2nydpNXSJpvV10J9RCyZ6MbYxNtlQ= | ||||
| github.com/go-acme/lego/v4 v4.21.0 h1:arEW+8o5p7VI8Bk1kr/PDlgD1DrxtTH1gJ4b7mehL8o= | ||||
| github.com/go-acme/lego/v4 v4.21.0/go.mod h1:HrSWzm3Ckj45Ie3i+p1zKVobbQoMOaGu9m4up0dUeDI= | ||||
| github.com/go-cmd/cmd v1.0.5/go.mod h1:y8q8qlK5wQibcw63djSl/ntiHUHXHGdCkPk0j4QeW4s= | ||||
| github.com/go-errors/errors v1.0.1 h1:LUHzmkK3GUKUrL/1gfBUxAHzcev3apQlezX/+O7ma6w= | ||||
| github.com/go-errors/errors v1.0.1/go.mod h1:f4zRHt4oKfwPJE5k8C9vpYG+aDHdBFUsgrm6/TyX73Q= | ||||
| @@ -219,47 +215,37 @@ github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY= | ||||
| github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY= | ||||
| github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag= | ||||
| github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE= | ||||
| github.com/go-oauth2/oauth2/v4 v4.5.2 h1:CuZhD3lhGuI6aNLyUbRHXsgG2RwGRBOuCBfd4WQKqBQ= | ||||
| github.com/go-oauth2/oauth2/v4 v4.5.2/go.mod h1:wk/2uLImWIa9VVQDgxz99H2GDbhmfi/9/Xr+GvkSUSQ= | ||||
| github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= | ||||
| github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= | ||||
| github.com/go-ping/ping v1.1.0 h1:3MCGhVX4fyEUuhsfwPrsEdQw6xspHkv5zHsiSoDFZYw= | ||||
| github.com/go-ping/ping v1.1.0/go.mod h1:xIFjORFzTxqIV/tDVGO4eDy/bLuSyawEeojSm3GfRGk= | ||||
| github.com/go-resty/resty/v2 v2.13.1 h1:x+LHXBI2nMB1vqndymf26quycC4aggYJ7DECYbiz03g= | ||||
| github.com/go-resty/resty/v2 v2.13.1/go.mod h1:GznXlLxkq6Nh4sU59rPmUw3VtgpO3aS96ORAI6Q7d+0= | ||||
| github.com/go-session/session v3.1.2+incompatible h1:yStchEObKg4nk2F7JGE7KoFIrA/1Y078peagMWcrncg= | ||||
| github.com/go-session/session v3.1.2+incompatible/go.mod h1:8B3iivBQjrz/JtC68Np2T1yBBLxTan3mn/3OM0CyRt0= | ||||
| github.com/go-resty/resty/v2 v2.16.2 h1:CpRqTjIzq/rweXUt9+GxzzQdlkqMdt8Lm/fuK/CAbAg= | ||||
| github.com/go-resty/resty/v2 v2.16.2/go.mod h1:0fHAoK7JoBy/Ch36N8VFeMsK7xQOHhvWaC3iOktwmIU= | ||||
| github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= | ||||
| github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= | ||||
| github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 h1:tfuBGBXKqDEevZMzYi5KSi8KkcZtzBcTgAUUtapy0OI= | ||||
| github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572/go.mod h1:9Pwr4B2jHnOSGXyyzV8ROjYa2ojvAY6HCGYYfMoC3Ls= | ||||
| github.com/go-viper/mapstructure/v2 v2.1.0 h1:gHnMa2Y/pIxElCH2GlZZ1lZSsn6XMtufpGyP1XxdC/w= | ||||
| github.com/go-viper/mapstructure/v2 v2.1.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM= | ||||
| github.com/go-viper/mapstructure/v2 v2.2.1 h1:ZAaOCxANMuZx5RCeg0mBdEZk7DZasvvZIxtHqx8aGss= | ||||
| github.com/go-viper/mapstructure/v2 v2.2.1/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM= | ||||
| github.com/gobs/pretty v0.0.0-20180724170744-09732c25a95b h1:/vQ+oYKu+JoyaMPDsv5FzwuL2wwWBgBbtj/YLCi4LuA= | ||||
| github.com/gobs/pretty v0.0.0-20180724170744-09732c25a95b/go.mod h1:Xo4aNUOrJnVruqWQJBtW6+bTBDTniY8yZum5rF3b5jw= | ||||
| github.com/goccy/go-json v0.10.3 h1:KZ5WoDbxAIgm2HNbYckL0se1fHD6rz5j4ywS6ebzDqA= | ||||
| github.com/goccy/go-json v0.10.3/go.mod h1:oq7eo15ShAhp70Anwd5lgX2pLfOS3QCiwU/PULtXL6M= | ||||
| github.com/gofrs/uuid v4.4.0+incompatible h1:3qXRTX8/NbyulANqlc0lchS1gqAVxRgsuW1YrTJupqA= | ||||
| github.com/gofrs/uuid v4.4.0+incompatible/go.mod h1:b2aQJv3Z4Fp6yNu3cdSllBxTCLRxnplIgP/c0N/04lM= | ||||
| github.com/goccy/go-json v0.10.4 h1:JSwxQzIqKfmFX1swYPpUThQZp/Ka4wzJdK0LWVytLPM= | ||||
| github.com/goccy/go-json v0.10.4/go.mod h1:oq7eo15ShAhp70Anwd5lgX2pLfOS3QCiwU/PULtXL6M= | ||||
| github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= | ||||
| github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4= | ||||
| github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= | ||||
| github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= | ||||
| github.com/goji/httpauth v0.0.0-20160601135302-2da839ab0f4d/go.mod h1:nnjvkQ9ptGaCkuDUx6wNykzzlUixGxvkme+H/lnzb+A= | ||||
| github.com/golang-jwt/jwt v3.2.1+incompatible/go.mod h1:8pz2t5EyA70fFQQSrl6XZXzqecmYZeUEB8OUGHkxJ+I= | ||||
| github.com/golang-jwt/jwt v3.2.2+incompatible h1:IfV12K8xAKAnZqdXVzCZ+TOjboZ2keLg81eXfW3O+oY= | ||||
| github.com/golang-jwt/jwt v3.2.2+incompatible/go.mod h1:8pz2t5EyA70fFQQSrl6XZXzqecmYZeUEB8OUGHkxJ+I= | ||||
| github.com/golang-jwt/jwt/v4 v4.0.0/go.mod h1:/xlHOz8bRuivTWchD4jCa+NbatV+wEUSzwAxVc6locg= | ||||
| github.com/golang-jwt/jwt/v4 v4.1.0/go.mod h1:/xlHOz8bRuivTWchD4jCa+NbatV+wEUSzwAxVc6locg= | ||||
| github.com/golang-jwt/jwt/v4 v4.2.0/go.mod h1:/xlHOz8bRuivTWchD4jCa+NbatV+wEUSzwAxVc6locg= | ||||
| github.com/golang-jwt/jwt/v4 v4.5.0 h1:7cYmW1XlMY7h7ii7UhUyChSgS5wUJEnm9uZVTGqOWzg= | ||||
| github.com/golang-jwt/jwt/v4 v4.5.0/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= | ||||
| github.com/golang-jwt/jwt/v4 v4.5.1 h1:JdqV9zKUdtaa9gdPlywC3aeoEsR681PlKC+4F5gQgeo= | ||||
| github.com/golang-jwt/jwt/v4 v4.5.1/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= | ||||
| github.com/golang-jwt/jwt/v5 v5.2.1 h1:OuVbFODueb089Lh128TAcimifWaLhJwVflnrgM17wHk= | ||||
| github.com/golang-jwt/jwt/v5 v5.2.1/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk= | ||||
| github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0/go.mod h1:E/TSTwGwJL78qG/PmXZO1EjYhfJinVAhrmmHX6Z8B9k= | ||||
| github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= | ||||
| github.com/golang/groupcache v0.0.0-20190129154638-5b532d6fd5ef/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= | ||||
| github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= | ||||
| github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da h1:oI5xCqsCo564l8iNU+DwB5epxmsaqB+rhGL0m5jtYqE= | ||||
| github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= | ||||
| github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= | ||||
| github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= | ||||
| github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= | ||||
| @@ -272,9 +258,7 @@ github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:x | ||||
| github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= | ||||
| github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= | ||||
| github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | ||||
| github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= | ||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= | ||||
| @@ -286,9 +270,7 @@ github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5a | ||||
| github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.7/go.mod h1:n+brtR0CgQNWTVd5ZUFpTBC8YFBDLK/h/bpaJ8/DtOE= | ||||
| @@ -309,8 +291,6 @@ github.com/google/pprof v0.0.0-20210407192527-94a9f03dee38/go.mod h1:kpwsk12EmLe | ||||
| github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= | ||||
| github.com/google/s2a-go v0.1.8 h1:zZDs9gcbt9ZPLV0ndSyQk6Kacx2g/X+SKYovpnz3SMM= | ||||
| github.com/google/s2a-go v0.1.8/go.mod h1:6iNWHTpQ+nfNRN5E00MSdfDwVesa8hhS32PhPO8deJA= | ||||
| github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/uuid v1.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | ||||
| @@ -319,11 +299,10 @@ github.com/googleapis/enterprise-certificate-proxy v0.3.4 h1:XYIDZApgAnrN1c855gT | ||||
| github.com/googleapis/enterprise-certificate-proxy v0.3.4/go.mod h1:YKe7cfqYXjKGpGvmSg28/fFvhNzinZQm8DGnaburhGA= | ||||
| github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= | ||||
| github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= | ||||
| github.com/googleapis/gax-go/v2 v2.13.0 h1:yitjD5f7jQHhyDsnhKEBU52NdvvdSeGzlAnDPT0hH1s= | ||||
| github.com/googleapis/gax-go/v2 v2.13.0/go.mod h1:Z/fvTZXF8/uw7Xu5GuslPw+bplx6SS338j1Is2S+B7A= | ||||
| github.com/gophercloud/gophercloud v1.14.0 h1:Bt9zQDhPrbd4qX7EILGmy+i7GP35cc+AAL2+wIJpUE8= | ||||
| github.com/gophercloud/gophercloud v1.14.0/go.mod h1:aAVqcocTSXh2vYFZ1JTvx4EQmfgzxRcNupUfxZbBNDM= | ||||
| github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1 h1:EGx4pi6eqNxGaHF6qqu48+N2wcFQ5qg5FXgOdqsJ5d8= | ||||
| github.com/googleapis/gax-go/v2 v2.14.0 h1:f+jMrjBPl+DL9nI4IQzLUxMq7XrAqFYB7hBPqMNIe8o= | ||||
| github.com/googleapis/gax-go/v2 v2.14.0/go.mod h1:lhBCnjdLrWRaPvLWhmc8IS24m9mr07qSYnHncrgo+zk= | ||||
| github.com/gophercloud/gophercloud v1.14.1 h1:DTCNaTVGl8/cFu58O1JwWgis9gtISAFONqpMKNg/Vpw= | ||||
| github.com/gophercloud/gophercloud v1.14.1/go.mod h1:aAVqcocTSXh2vYFZ1JTvx4EQmfgzxRcNupUfxZbBNDM= | ||||
| github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= | ||||
| github.com/gorilla/csrf v1.7.2 h1:oTUjx0vyf2T+wkrx09Trsev1TE+/EbDAeHtSTbtC2eI= | ||||
| github.com/gorilla/csrf v1.7.2/go.mod h1:F1Fj3KG23WYHE6gozCmBAezKookxbIvUJT+121wTuLk= | ||||
| @@ -341,9 +320,8 @@ github.com/grandcat/zeroconf v1.0.0 h1:uHhahLBKqwWBV6WZUDAT71044vwOTL+McW0mBJvo6 | ||||
| github.com/grandcat/zeroconf v1.0.0/go.mod h1:lTKmG1zh86XyCoUeIHSA4FJMBwCJiQmGfcP2PdzytEs= | ||||
| github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs= | ||||
| github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk= | ||||
| github.com/grpc-ecosystem/grpc-gateway v1.9.0 h1:bM6ZAFZmc/wPFaRDi0d5L7hGEZEx/2u+Tmr2evNHDiI= | ||||
| github.com/grpc-ecosystem/grpc-gateway v1.9.0/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY= | ||||
| github.com/grpc-ecosystem/grpc-gateway v1.16.0 h1:gmcG1KaJ57LophUzW0Hy8NmPhnMZb4M0+kPpLofRdBo= | ||||
| github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw= | ||||
| github.com/grpc-ecosystem/grpc-gateway/v2 v2.20.0 h1:bkypFPDjIYGfCYD5mRBvpqxfYX1YCS1PXdKYWi8FsN0= | ||||
| github.com/grpc-ecosystem/grpc-gateway/v2 v2.20.0/go.mod h1:P+Lt/0by1T8bfcF3z737NnSbmxQAppXMRziHUxPOC8k= | ||||
| github.com/hashicorp/consul/api v1.1.0/go.mod h1:VmuI/Lkw1nC05EYQWNKwWGbkg+FbDBtguAZLlVdkD9Q= | ||||
| @@ -376,13 +354,11 @@ github.com/hashicorp/mdns v1.0.0/go.mod h1:tL+uN++7HEJ6SQLQ2/p+z2pH24WQKWjBPkE0m | ||||
| github.com/hashicorp/memberlist v0.1.3/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2pPBoIllUwCN7I= | ||||
| github.com/hashicorp/serf v0.8.2/go.mod h1:6hOLApaqBFA1NXqRQAsxw9QxuDEvNxSQRwA/JwenrHc= | ||||
| github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||
| github.com/huaweicloud/huaweicloud-sdk-go-v3 v0.1.114 h1:X3E16S6AUZsQKhJIQ5kNnylnp0GtSy2YhIbxfvDavtU= | ||||
| github.com/huaweicloud/huaweicloud-sdk-go-v3 v0.1.114/go.mod h1:JWz2ujO9X3oU5wb6kXp+DpR2UuDj2SldDbX8T0FSuhI= | ||||
| github.com/huaweicloud/huaweicloud-sdk-go-v3 v0.1.128 h1:kQ2Agpfy7Ze1ajn9xCQG9G6T7XIbqv+FBDS/U98W9Mk= | ||||
| github.com/huaweicloud/huaweicloud-sdk-go-v3 v0.1.128/go.mod h1:JWz2ujO9X3oU5wb6kXp+DpR2UuDj2SldDbX8T0FSuhI= | ||||
| github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= | ||||
| github.com/iij/doapi v0.0.0-20190504054126-0bbf12d6d7df h1:MZf03xP9WdakyXhOWuAD5uPK3wHh96wCsqe3hCMKh8E= | ||||
| github.com/iij/doapi v0.0.0-20190504054126-0bbf12d6d7df/go.mod h1:QMZY7/J/KSQEhKWFeDesPjMj+wCHReeknARU3wqlyN4= | ||||
| github.com/imkira/go-interpol v1.1.0 h1:KIiKr0VSG2CUW1hl1jpiyuzuJeKUUpC8iM1AIE7N1Vk= | ||||
| github.com/imkira/go-interpol v1.1.0/go.mod h1:z0h2/2T3XF8kyEPpRgJ3kmNv+C43p+I/CoI+jC3w2iA= | ||||
| github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= | ||||
| github.com/infobloxopen/infoblox-go-client v1.1.1 h1:728A6LbLjptj/7kZjHyIxQnm768PWHfGFm0HH8FnbtU= | ||||
| github.com/infobloxopen/infoblox-go-client v1.1.1/go.mod h1:BXiw7S2b9qJoM8MS40vfgCNB2NLHGusk1DtO16BD9zI= | ||||
| @@ -399,19 +375,17 @@ github.com/json-iterator/go v1.1.7/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/u | ||||
| github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= | ||||
| github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= | ||||
| github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= | ||||
| github.com/jtolds/gls v4.20.0+incompatible h1:xdiiI2gbIgH/gLH7ADydsJ1uDOEzR8yvV7C0MuV77Wo= | ||||
| github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= | ||||
| github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= | ||||
| github.com/jung-kurt/gofpdf v1.0.3-0.20190309125859-24315acbbda5/go.mod h1:7Id9E/uU8ce6rXgefFLlgrJj/GYY22cpxn+r32jIOes= | ||||
| github.com/k0kubun/colorstring v0.0.0-20150214042306-9440f1994b88/go.mod h1:3w7q1U84EfirKl04SVQ/s7nPm1ZPhiXd34z40TNz36k= | ||||
| github.com/k0kubun/go-ansi v0.0.0-20180517002512-3bf9e2903213 h1:qGQQKEcAR99REcMpsXCp3lJ03zYT1PkRd3kQGPn9GVg= | ||||
| github.com/k0kubun/go-ansi v0.0.0-20180517002512-3bf9e2903213/go.mod h1:vNUNkEQ1e29fT/6vq2aBdFsgNPmy8qMdSay1npru+Sw= | ||||
| github.com/keybase/go-keychain v0.0.0-20231219164618-57a3676c3af6 h1:IsMZxCuZqKuao2vNdfD82fjjgPLfyHLpR41Z88viRWs= | ||||
| github.com/keybase/go-keychain v0.0.0-20231219164618-57a3676c3af6/go.mod h1:3VeWNIJaW+O5xpRQbPp0Ybqu1vJd/pm7s2F473HRrkw= | ||||
| github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvWXihfKN4Q= | ||||
| github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= | ||||
| github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= | ||||
| github.com/klauspost/compress v1.13.6/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk= | ||||
| github.com/klauspost/compress v1.15.0 h1:xqfchp4whNFxn5A4XFyyYtitiWI8Hy5EW59jEwcyL6U= | ||||
| github.com/klauspost/compress v1.15.0/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk= | ||||
| github.com/kolo/xmlrpc v0.0.0-20220921171641-a4b6fa1dd06b h1:udzkj9S/zlT5X367kqJis0QP7YMxobob6zhzq6Yre00= | ||||
| github.com/kolo/xmlrpc v0.0.0-20220921171641-a4b6fa1dd06b/go.mod h1:pcaDhQK0/NJZEvtCO0qQPPropqV0sJOJ6YW7X+9kRwM= | ||||
| github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= | ||||
| @@ -434,8 +408,8 @@ github.com/likexian/gokit v0.25.13 h1:p2Uw3+6fGG53CwdU2Dz0T6bOycdb2+bAFAa3ymwWVk | ||||
| github.com/likexian/gokit v0.25.13/go.mod h1:qQhEWFBEfqLCO3/vOEo2EDKd+EycekVtUK4tex+l2H4= | ||||
| github.com/likexian/whois v1.15.1 h1:6vTMI8n9s1eJdmcO4R9h1x99aQWIZZX1CD3am68gApU= | ||||
| github.com/likexian/whois v1.15.1/go.mod h1:/nxmQ6YXvLz+qTxC/QFtEJNAt0zLuRxJrKiWpBJX8X0= | ||||
| github.com/linode/linodego v1.40.0 h1:7ESY0PwK94hoggoCtIroT1Xk6b1flrFBNZ6KwqbTqlI= | ||||
| github.com/linode/linodego v1.40.0/go.mod h1:NsUw4l8QrLdIofRg1NYFBbW5ZERnmbZykVBszPZLORM= | ||||
| github.com/linode/linodego v1.44.0 h1:JZLLWzCAx3CmHSV9NmCoXisuqKtrmPhfY9MrgvaHMUY= | ||||
| github.com/linode/linodego v1.44.0/go.mod h1:umdoNOmtbqAdGQbmQnPFZ2YS4US+/mU/1bA7MjoKAvg= | ||||
| github.com/liquidweb/go-lwApi v0.0.0-20190605172801-52a4864d2738/go.mod h1:0sYF9rMXb0vlG+4SzdiGMXHheCZxjguMq+Zb4S2BfBs= | ||||
| github.com/liquidweb/liquidweb-cli v0.6.9 h1:acbIvdRauiwbxIsOCEMXGwF75aSJDbDiyAWPjVnwoYM= | ||||
| github.com/liquidweb/liquidweb-cli v0.6.9/go.mod h1:cE1uvQ+x24NGUL75D0QagOFCG8Wdvmwu8aL9TLmA/eQ= | ||||
| @@ -474,8 +448,9 @@ github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceT | ||||
| github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= | ||||
| github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= | ||||
| github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= | ||||
| github.com/mitchellh/go-testing-interface v1.0.0 h1:fzU/JVNcaqHQEcVFAKeR41fkiLdIPrefOvVG1VZ96U0= | ||||
| github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= | ||||
| github.com/mitchellh/go-testing-interface v1.14.1 h1:jrgshOhYAUVNMAJiKbEu7EqAwgJJ2JqpQmpLJOu07cU= | ||||
| github.com/mitchellh/go-testing-interface v1.14.1/go.mod h1:gfgS7OtZj6MA4U1UrDRp04twqAjfvlZyCfX3sDjEym8= | ||||
| github.com/mitchellh/gox v0.4.0/go.mod h1:Sd9lOJ0+aimLBi73mGofS1ycjY8lL3uZM3JPS42BGNg= | ||||
| github.com/mitchellh/iochan v1.0.0/go.mod h1:JwYml1nuB7xOzsp52dPpHFffvOCDupsG0QubkSMEySY= | ||||
| github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= | ||||
| @@ -494,11 +469,11 @@ github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lN | ||||
| github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= | ||||
| github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M= | ||||
| github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= | ||||
| github.com/monperrus/crawler-user-agents v1.1.0 h1:Xy8ZrhizT+y2FONWFFdKOP+3BhH97BDLuG7W/MswoGI= | ||||
| github.com/monperrus/crawler-user-agents v1.1.0/go.mod h1:GfRyKbsbxSrRxTPYnVi4U/0stQd6BcFCxDy6i6IxQ0M= | ||||
| github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe/go.mod h1:wL8QJuTMNUDYhXwkmfOly8iTdp5TEcJFWZD2D7SIkUc= | ||||
| github.com/morikuni/aec v1.0.0 h1:nP9CBfwrvYnBRgY6qfDQkygYDmYwOilePFkwzv4dU8A= | ||||
| github.com/morikuni/aec v1.0.0/go.mod h1:BbKIizmSmc5MMPqRYbxO4ZU0S0+P200+tUnFx7PXmsc= | ||||
| github.com/moul/http2curl v1.0.0 h1:dRMWoAtb+ePxMlLkrCbAqh4TlPHXvoGUSQ323/9Zahs= | ||||
| github.com/moul/http2curl v1.0.0/go.mod h1:8UbvGypXm98wA/IqH45anm5Y2Z6ep6O31QGOAZ3H0fQ= | ||||
| github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||
| github.com/namedotcom/go v0.0.0-20180403034216-08470befbe04 h1:o6uBwrhM5C8Ll3MAAxrQxRHEu7FkapwTuI2WmL1rw4g= | ||||
| github.com/namedotcom/go v0.0.0-20180403034216-08470befbe04/go.mod h1:5sN+Lt1CaY4wsPvgQH/jsuJi4XO2ssZbdsIizr4CVC8= | ||||
| @@ -507,12 +482,12 @@ github.com/nrdcg/auroradns v1.1.0 h1:KekGh8kmf2MNwqZVVYo/fw/ZONt8QMEmbMFOeljteWo | ||||
| github.com/nrdcg/auroradns v1.1.0/go.mod h1:O7tViUZbAcnykVnrGkXzIJTHoQCHcgalgAe6X1mzHfk= | ||||
| github.com/nrdcg/bunny-go v0.0.0-20240207213615-dde5bf4577a3 h1:ouZ2JWDl8IW5k1qugYbmpbmW8hn85Ig6buSMBRlz3KI= | ||||
| github.com/nrdcg/bunny-go v0.0.0-20240207213615-dde5bf4577a3/go.mod h1:ZwadWt7mVhMHMbAQ1w8IhDqtWO3eWqWq72W7trnaiE8= | ||||
| github.com/nrdcg/desec v0.8.0 h1:FJbRWUAluTCUi9nHFnhqPhLSIHiNnB9elZVWYgFtIqA= | ||||
| github.com/nrdcg/desec v0.8.0/go.mod h1:BsnYPtSlBttJL3Gyzv0kDH7zkk60obwThlnqiiKzn+o= | ||||
| github.com/nrdcg/desec v0.10.0 h1:qrEDiqnsvNU9QE7lXIXi/tIHAfyaFXKxF2/8/52O8uM= | ||||
| github.com/nrdcg/desec v0.10.0/go.mod h1:5+4vyhMRTs49V9CNoODF/HwT8Mwxv9DJ6j+7NekUnBs= | ||||
| github.com/nrdcg/dnspod-go v0.4.0 h1:c/jn1mLZNKF3/osJ6mz3QPxTudvPArXTjpkmYj0uK6U= | ||||
| github.com/nrdcg/dnspod-go v0.4.0/go.mod h1:vZSoFSFeQVm2gWLMkyX61LZ8HI3BaqtHZWgPTGKr6KQ= | ||||
| github.com/nrdcg/freemyip v0.2.0 h1:/GscavT4GVqAY13HExl5UyoB4wlchv6Cg5NYDGsUoJ8= | ||||
| github.com/nrdcg/freemyip v0.2.0/go.mod h1:HjF0Yz0lSb37HD2ihIyGz9esyGcxbCrrGFLPpKevbx4= | ||||
| github.com/nrdcg/freemyip v0.3.0 h1:0D2rXgvLwe2RRaVIjyUcQ4S26+cIS2iFwnhzDsEuuwc= | ||||
| github.com/nrdcg/freemyip v0.3.0/go.mod h1:c1PscDvA0ukBF0dwelU/IwOakNKnVxetpAQ863RMJoM= | ||||
| github.com/nrdcg/goinwx v0.10.0 h1:6W630bjDxQD6OuXKqrFRYVpTt0G/9GXXm3CeOrN0zJM= | ||||
| github.com/nrdcg/goinwx v0.10.0/go.mod h1:mnMSTi7CXBu2io4DzdOBoGFA1XclD0sEPWJaDhNgkA4= | ||||
| github.com/nrdcg/mailinabox v0.2.0 h1:IKq8mfKiVwNW2hQii/ng1dJ4yYMMv3HAP3fMFIq2CFk= | ||||
| @@ -532,7 +507,6 @@ github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn | ||||
| github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= | ||||
| github.com/onsi/ginkgo v1.13.0/go.mod h1:+REjRxOmWfHCjfv9TTWB1jD1Frx4XydAD3zm1lskyM0= | ||||
| github.com/onsi/ginkgo v1.16.4 h1:29JGrr5oVBm5ulCWet69zQkzWipVXIol6ygQUe/EzNc= | ||||
| github.com/onsi/ginkgo v1.16.4/go.mod h1:dX+/inL/fNMqNlz0e9LfyB9TswhZpCVdJM/Z6Vvnwo0= | ||||
| github.com/onsi/ginkgo/v2 v2.0.0/go.mod h1:vw5CSIxN1JObi/U8gcbwft7ZxR2dgaR70JSE3/PpL4c= | ||||
| @@ -558,6 +532,8 @@ github.com/ovh/go-ovh v1.6.0/go.mod h1:cTVDnl94z4tl8pP1uZ/8jlVxntjSIf09bNcQ5TJSC | ||||
| github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= | ||||
| github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic= | ||||
| github.com/pelletier/go-toml v1.8.1/go.mod h1:T2/BmBdy8dvIRq1a/8aqjN41wvWlN4lrapLU/GW4pbc= | ||||
| github.com/peterhellberg/link v1.2.0 h1:UA5pg3Gp/E0F2WdX7GERiNrPQrM1K6CVJUUWfHa4t6c= | ||||
| github.com/peterhellberg/link v1.2.0/go.mod h1:gYfAh+oJgQu2SrZHg5hROVRQe1ICoK0/HHJTcE0edxc= | ||||
| github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c h1:+mdjkGKdHQG3305AYmdv1U2eRNDiU2ErMBj1gwrq8eQ= | ||||
| github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c/go.mod h1:7rwL4CYBLnjLxUqIJNnCWiEdr3bn6IUYi15bNlnbCCU= | ||||
| github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| @@ -570,6 +546,8 @@ github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZN | ||||
| github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U= | ||||
| github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= | ||||
| github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c h1:ncq/mPwQF4JjgDlrVEn3C11VoGHZN7m8qihwgMEtzYw= | ||||
| github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c/go.mod h1:OmDBASR4679mdNQnz2pUhc2G8CO2JrUAVFDRBDP/hJE= | ||||
| github.com/pquerna/otp v1.4.0 h1:wZvl1TIVxKRThZIBiwOOHOGP/1+nZyWBil9Y2XNEDzg= | ||||
| github.com/pquerna/otp v1.4.0/go.mod h1:dkJfzwRKNiegxyNb54X/3fLwhCynbMspSyWKnvi1AEg= | ||||
| github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= | ||||
| @@ -588,8 +566,9 @@ github.com/prometheus/procfs v0.0.0-20190507164030-5867b95ac084/go.mod h1:TjEm7z | ||||
| github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= | ||||
| github.com/prometheus/procfs v0.0.3/go.mod h1:4A/X28fw3Fc593LaREMrKMqOKvUAntwMDaekg4FpcdQ= | ||||
| github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU= | ||||
| github.com/redis/go-redis/v9 v9.6.1 h1:HHDteefn6ZkTtY5fGUE8tj8uy85AHk6zP7CpzIAM0y4= | ||||
| github.com/redis/go-redis/v9 v9.6.1/go.mod h1:0C0c6ycQsdpVNQpxb1njEQIqkx5UcsM8FJCQLgE9+RA= | ||||
| github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg= | ||||
| github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= | ||||
| github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= | ||||
| github.com/rogpeppe/go-internal v1.12.0 h1:exVL4IDcn6na9z1rAb56Vxr+CgyK3nn3O+epU5NdKM8= | ||||
| github.com/rogpeppe/go-internal v1.12.0/go.mod h1:E+RYuTGaKKdloAfM02xzb0FW3Paa99yedzYV+kq4uf4= | ||||
| @@ -599,16 +578,15 @@ github.com/sacloud/api-client-go v0.2.10 h1:+rv3jDohD+pkdYwOTBiB+jZsM0xK3AxadXRz | ||||
| github.com/sacloud/api-client-go v0.2.10/go.mod h1:Jj3CTy2+O4bcMedVDXlbHuqqche85HEPuVXoQFhLaRc= | ||||
| github.com/sacloud/go-http v0.1.8 h1:ynreWA/vnM8G2ksbMlmefBHsXURKPz49qlPRqQ9IQdw= | ||||
| github.com/sacloud/go-http v0.1.8/go.mod h1:7TL7TN1fnPKHsMifIqURDkGujnKViCgEz5Ei/LQdFK8= | ||||
| github.com/sacloud/iaas-api-go v1.12.0 h1:kqXFn3HzCiawlX6hVJb1GVqcSJqcmiGHB4Zp14sxiI8= | ||||
| github.com/sacloud/iaas-api-go v1.12.0/go.mod h1:SZLXeWOdXk3WReIS557sbU1gkOgrE4rseIBQV1B3b7o= | ||||
| github.com/sacloud/iaas-api-go v1.14.0 h1:xjkFWqdo4ilTrKPNNYBNWR/CZ/kVRsJrdAHAad6J/AQ= | ||||
| github.com/sacloud/iaas-api-go v1.14.0/go.mod h1:C8os2Mnj0TOmMdSllwhaDWKMVG2ysFnpe69kyA4M3V0= | ||||
| github.com/sacloud/packages-go v0.0.10 h1:UiQGjy8LretewkRhsuna1TBM9Vz/l9FoYpQx+D+AOck= | ||||
| github.com/sacloud/packages-go v0.0.10/go.mod h1:f8QITBh9z4IZc4yE9j21Q8b0sXEMwRlRmhhjWeDVTYs= | ||||
| github.com/scaleway/scaleway-sdk-go v1.0.0-beta.30 h1:yoKAVkEVwAqbGbR8n87rHQ1dulL25rKloGadb3vm770= | ||||
| github.com/scaleway/scaleway-sdk-go v1.0.0-beta.30/go.mod h1:sH0u6fq6x4R5M7WxkoQFY/o7UaiItec0o1LinLCJNq8= | ||||
| github.com/sclevine/agouti v3.0.0+incompatible/go.mod h1:b4WX9W9L1sfQKXeJf1mUTLZKJ48R1S7H23Ji7oFO5Bw= | ||||
| github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= | ||||
| github.com/sergi/go-diff v1.1.0 h1:we8PVUC3FE2uYfodKH/nBHMSetSfHDR6scGdBi+erh0= | ||||
| github.com/sergi/go-diff v1.1.0/go.mod h1:STckp+ISIX8hZLjrqAeVduY0gWCT9IjLuqbuNXdaHfM= | ||||
| github.com/shirou/gopsutil/v4 v4.25.1 h1:QSWkTc+fu9LTAWfkZwZ6j8MSUk4A2LV7rbH0ZqmLjXs= | ||||
| github.com/shirou/gopsutil/v4 v4.25.1/go.mod h1:RoUCUpndaJFtT+2zsZzzmhvbfGoDCJ7nFXKJf8GqJbI= | ||||
| github.com/shopspring/decimal v1.3.1 h1:2Usl1nmF/WZucqkFZhnfFYxxxu8LG21F6nPQBE5gKV8= | ||||
| github.com/shopspring/decimal v1.3.1/go.mod h1:DKyhrW/HYNuLGql+MJL6WCR6knT2jwCFRcu2hWCYk4o= | ||||
| github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= | ||||
| @@ -622,12 +600,11 @@ github.com/smartystreets/assertions v1.0.1 h1:voD4ITNjPL5jjBfgR/r8fPIIBrliWrWHei | ||||
| github.com/smartystreets/assertions v1.0.1/go.mod h1:kHHU4qYBaI3q23Pp3VPrmWhuIUrLW/7eUrw0BU5VaoM= | ||||
| github.com/smartystreets/go-aws-auth v0.0.0-20180515143844-0c1422d1fdb9 h1:hp2CYQUINdZMHdvTdXtPOY2ainKl4IoMcpAXEf2xj3Q= | ||||
| github.com/smartystreets/go-aws-auth v0.0.0-20180515143844-0c1422d1fdb9/go.mod h1:SnhjPscd9TpLiy1LpzGSKh3bXCfxxXuqd9xmQJy3slM= | ||||
| github.com/smartystreets/goconvey v1.6.4 h1:fv0U8FUIMPNf1L9lnHLvLhgicrIVChEkdzIKYqbNC9s= | ||||
| github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA= | ||||
| github.com/smartystreets/gunit v1.0.4 h1:tpTjnuH7MLlqhoD21vRoMZbMIi5GmBsAJDFyF67GhZA= | ||||
| github.com/smartystreets/gunit v1.0.4/go.mod h1:EH5qMBab2UclzXUcpR8b93eHsIlp9u+pDQIRp5DZNzQ= | ||||
| github.com/softlayer/softlayer-go v1.1.5 h1:UFFtgKxiw0yIuUw93XBCFIiIMYR5eLgmm4a5DqMHXGg= | ||||
| github.com/softlayer/softlayer-go v1.1.5/go.mod h1:WeJrBLoTJcaT8nO1azeyHyNpo/fDLtbpbvh+pzts+Qw= | ||||
| github.com/softlayer/softlayer-go v1.1.7 h1:SgTL+pQZt1h+5QkAhVmHORM/7N9c1X0sljJhuOIHxWE= | ||||
| github.com/softlayer/softlayer-go v1.1.7/go.mod h1:WeJrBLoTJcaT8nO1azeyHyNpo/fDLtbpbvh+pzts+Qw= | ||||
| github.com/softlayer/xmlrpc v0.0.0-20200409220501-5f089df7cb7e h1:3OgWYFw7jxCZPcvAg+4R8A50GZ+CCkARF10lxu2qDsQ= | ||||
| github.com/softlayer/xmlrpc v0.0.0-20200409220501-5f089df7cb7e/go.mod h1:fKZCUVdirrxrBpwd9wb+lSoVixvpwAu8eHzbQB2tums= | ||||
| github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM= | ||||
| @@ -655,40 +632,19 @@ github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXf | ||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||
| github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= | ||||
| github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | ||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= | ||||
| github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= | ||||
| github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= | ||||
| github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= | ||||
| github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= | ||||
| github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA= | ||||
| github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= | ||||
| github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw= | ||||
| github.com/syndtr/goleveldb v1.0.0 h1:fBdIW9lB4Iz0n9khmH8w27SJ3QEJ7+IgjPEwGSZiFdE= | ||||
| github.com/syndtr/goleveldb v1.0.0/go.mod h1:ZVVdQEZoIme9iO1Ch2Jdy24qqXrMMOU6lpPAyBWyWuQ= | ||||
| github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/common v1.0.1002 h1:RE84sHFFx6t24DJvSnF9fS1DzBNv9OpctzHK3t7AY+I= | ||||
| github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/common v1.0.1002/go.mod h1:r5r4xbfxSaeR04b166HGsBa/R4U3SueirEUpXGuw+Q0= | ||||
| github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/dnspod v1.0.1002 h1:QwE0dRkAAbdf+eACnkNULgDn9ZKUJpPWRyXdqJolP5E= | ||||
| github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/dnspod v1.0.1002/go.mod h1:WdC0FYbqYhJwQ3kbqri6hVP5HAEp+rzX9FToItTAzUg= | ||||
| github.com/tidwall/btree v0.0.0-20191029221954-400434d76274 h1:G6Z6HvJuPjG6XfNGi/feOATzeJrfgTNJY+rGrHbA04E= | ||||
| github.com/tidwall/btree v0.0.0-20191029221954-400434d76274/go.mod h1:huei1BkDWJ3/sLXmO+bsCNELL+Bp2Kks9OLyQFkzvA8= | ||||
| github.com/tidwall/buntdb v1.1.2 h1:noCrqQXL9EKMtcdwJcmuVKSEjqu1ua99RHHgbLTEHRo= | ||||
| github.com/tidwall/buntdb v1.1.2/go.mod h1:xAzi36Hir4FarpSHyfuZ6JzPJdjRZ8QlLZSntE2mqlI= | ||||
| github.com/tidwall/gjson v1.3.4/go.mod h1:P256ACg0Mn+j1RXIDXoss50DeIABTYK1PULOJHhxOls= | ||||
| github.com/tidwall/gjson v1.12.1 h1:ikuZsLdhr8Ws0IdROXUS1Gi4v9Z4pGqpX/CvJkxvfpo= | ||||
| github.com/tidwall/gjson v1.12.1/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= | ||||
| github.com/tidwall/grect v0.0.0-20161006141115-ba9a043346eb h1:5NSYaAdrnblKByzd7XByQEJVT8+9v0W/tIY0Oo4OwrE= | ||||
| github.com/tidwall/grect v0.0.0-20161006141115-ba9a043346eb/go.mod h1:lKYYLFIr9OIgdgrtgkZ9zgRxRdvPYsExnYBsEAd8W5M= | ||||
| github.com/tidwall/match v1.0.1/go.mod h1:LujAq0jyVjBy028G1WhWfIzbpQfMO8bBZ6Tyb0+pL9E= | ||||
| github.com/tidwall/match v1.1.1 h1:+Ho715JplO36QYgwN9PGYNhgZvoUSc9X2c80KVTi+GA= | ||||
| github.com/tidwall/match v1.1.1/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM= | ||||
| github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk= | ||||
| github.com/tidwall/pretty v1.2.0 h1:RWIZEg2iJ8/g6fDDYzMpobmaoGh5OLl4AXtGUGPcqCs= | ||||
| github.com/tidwall/pretty v1.2.0/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU= | ||||
| github.com/tidwall/rtree v0.0.0-20180113144539-6cd427091e0e h1:+NL1GDIUOKxVfbp2KoJQD9cTQ6dyP2co9q4yzmT9FZo= | ||||
| github.com/tidwall/rtree v0.0.0-20180113144539-6cd427091e0e/go.mod h1:/h+UnNGt0IhNNJLkGikcdcJqm66zGD/uJGMRxK/9+Ao= | ||||
| github.com/tidwall/tinyqueue v0.0.0-20180302190814-1e39f5511563 h1:Otn9S136ELckZ3KKDyCkxapfufrqDqwmGjcHfAyXRrE= | ||||
| github.com/tidwall/tinyqueue v0.0.0-20180302190814-1e39f5511563/go.mod h1:mLqSmt7Dv/CNneF2wfcChfN1rvapyQr01LGKnKex0DQ= | ||||
| github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/common v1.0.1065 h1:krcqtAmexnHHBm/4ge4tr2b1cn/a7JGBESVGoZYXQAE= | ||||
| github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/common v1.0.1065/go.mod h1:r5r4xbfxSaeR04b166HGsBa/R4U3SueirEUpXGuw+Q0= | ||||
| github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/dnspod v1.0.1065 h1:aEFtLD1ceyeljQXB1S2BjN0zjTkf0X3XmpuxFIiC29w= | ||||
| github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/dnspod v1.0.1065/go.mod h1:HWvwy09hFSMXrj9SMvVRWV4U7rZO3l+WuogyNuxiT3M= | ||||
| github.com/tjfoc/gmsm v1.4.1 h1:aMe1GlZb+0bLjn+cKTPEvvn9oUEBlJitaZiiBwsbgho= | ||||
| github.com/tjfoc/gmsm v1.4.1/go.mod h1:j4INPkHWMrhJb38G+J6W4Tw0AbuN8Thu3PbdVYhVcTE= | ||||
| github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= | ||||
| @@ -698,13 +654,8 @@ github.com/uber/jaeger-client-go v2.30.0+incompatible h1:D6wyKGCecFaSRUpo8lCVbaO | ||||
| github.com/uber/jaeger-client-go v2.30.0+incompatible/go.mod h1:WVhlPFC8FDjOFMMWRy2pZqQJSXxYSwNYOkTr/Z6d3Kk= | ||||
| github.com/uber/jaeger-lib v2.4.1+incompatible h1:td4jdvLcExb4cBISKIpHuGoVXh+dVKhn2Um6rjCsSsg= | ||||
| github.com/uber/jaeger-lib v2.4.1+incompatible/go.mod h1:ComeNDZlWwrWnDv8aPp0Ba6+uUTzImX/AauajbLI56U= | ||||
| github.com/ultradns/ultradns-go-sdk v1.7.0-20240913052650-970ca9a h1:R6IR+Vj/RnGZLnX8PpPQsbbQthctO7Ah2q4tj5eoe2o= | ||||
| github.com/ultradns/ultradns-go-sdk v1.7.0-20240913052650-970ca9a/go.mod h1:BZr7Qs3ku1ckpqed8tCRSqTlp8NAeZfAVpfx4OzXMss= | ||||
| github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw= | ||||
| github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= | ||||
| github.com/valyala/fasthttp v1.34.0 h1:d3AAQJ2DRcxJYHm7OXNXtXt2as1vMDfxeIcFvhmGGm4= | ||||
| github.com/valyala/fasthttp v1.34.0/go.mod h1:epZA5N+7pY6ZaEKRmstzOuYJx9HI8DI1oaCGZpdH4h0= | ||||
| github.com/valyala/tcplisten v1.0.0/go.mod h1:T0xQ8SeCZGxckz9qRXTfG43PvQ/mcWh7FwZEA7Ioqkc= | ||||
| github.com/ultradns/ultradns-go-sdk v1.8.0-20241010134910-243eeec h1:2s/ghQ8wKE+UzD/hf3P4Gd1j0JI9ncbxv+nsypPoUYI= | ||||
| github.com/ultradns/ultradns-go-sdk v1.8.0-20241010134910-243eeec/go.mod h1:BZr7Qs3ku1ckpqed8tCRSqTlp8NAeZfAVpfx4OzXMss= | ||||
| github.com/vinyldns/go-vinyldns v0.9.16 h1:GZJStDkcCk1F1AcRc64LuuMh+ENL8pHA0CVd4ulRMcQ= | ||||
| github.com/vinyldns/go-vinyldns v0.9.16/go.mod h1:5qIJOdmzAnatKjurI+Tl4uTus7GJKJxb+zitufjHs3Q= | ||||
| github.com/vultr/govultr/v3 v3.9.1 h1:uxSIb8Miel7tqTs3ee+z3t+JelZikwqBBsZzCOPBy/8= | ||||
| @@ -712,38 +663,23 @@ github.com/vultr/govultr/v3 v3.9.1/go.mod h1:Rd8ebpXm7jxH3MDmhnEs+zrlYW212ouhx+H | ||||
| github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= | ||||
| github.com/xdg-go/scram v1.1.2/go.mod h1:RT/sEzTbU5y00aCK8UOx6R7YryM0iF1N2MOmC3kKLN4= | ||||
| github.com/xdg-go/stringprep v1.0.4/go.mod h1:mPGuuIYwz7CmR2bT9j4GbQqutWS1zV24gijq1dTyGkM= | ||||
| github.com/xeipuuv/gojsonpointer v0.0.0-20180127040702-4e3ac2762d5f h1:J9EGpcZtP0E/raorCMxlFGSTBrsSlaDGf3jU/qvAE2c= | ||||
| github.com/xeipuuv/gojsonpointer v0.0.0-20180127040702-4e3ac2762d5f/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU= | ||||
| github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 h1:EzJWgHovont7NscjpAxXsDA8S8BMYve8Y5+7cuRE7R0= | ||||
| github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415/go.mod h1:GwrjFmJcFw6At/Gs6z4yjiIwzuJ1/+UwLxMQDVQXShQ= | ||||
| github.com/xeipuuv/gojsonschema v1.2.0 h1:LhYJRs+L4fBtjZUfuSZIKGeVu0QRy8e5Xi7D17UxZ74= | ||||
| github.com/xeipuuv/gojsonschema v1.2.0/go.mod h1:anYRn/JVcOK2ZgGU+IjEV4nwlhoK5sQluxsYJ78Id3Y= | ||||
| github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU= | ||||
| github.com/xlzd/gotp v0.1.0 h1:37blvlKCh38s+fkem+fFh7sMnceltoIEBYTVXyoa5Po= | ||||
| github.com/xlzd/gotp v0.1.0/go.mod h1:ndLJ3JKzi3xLmUProq4LLxCuECL93dG9WASNLpHz8qg= | ||||
| github.com/yalp/jsonpath v0.0.0-20180802001716-5cc68e5049a0 h1:6fRhSjgLCkTD3JnJxvaJ4Sj+TYblw757bqYgZaOq5ZY= | ||||
| github.com/yalp/jsonpath v0.0.0-20180802001716-5cc68e5049a0/go.mod h1:/LWChgwKmvncFJFHJ7Gvn9wZArjbV5/FppcK2fKk/tI= | ||||
| github.com/yandex-cloud/go-genproto v0.0.0-20240911120709-1fa0cb6f47c2 h1:WgeEP+8WizCQyccJNHOMLONq23qVAzYHtyg5qTdUWmg= | ||||
| github.com/yandex-cloud/go-genproto v0.0.0-20240911120709-1fa0cb6f47c2/go.mod h1:HEUYX/p8966tMUHHT+TsS0hF/Ca/NYwqprC5WXSDMfE= | ||||
| github.com/yandex-cloud/go-sdk v0.0.0-20240911121212-e4e74d0d02f5 h1:Q4LvUMF4kzaGtopoIdXReL9/qGtmzOewBhF3dQvuHMU= | ||||
| github.com/yandex-cloud/go-sdk v0.0.0-20240911121212-e4e74d0d02f5/go.mod h1:9dt2V80cfJGRZA+5SKP3Ky+R/DxH02XfKObi2Uy2uPc= | ||||
| github.com/yandex-cloud/go-genproto v0.0.0-20241220122821-aeb3b05efd1c h1:Rnr+lDYXVkP+3eT8/d68iq4G/UeIhyCQk+HKa8toTvg= | ||||
| github.com/yandex-cloud/go-genproto v0.0.0-20241220122821-aeb3b05efd1c/go.mod h1:0LDD/IZLIUIV4iPH+YcF+jysO3jkSvADFGm4dCAuwQo= | ||||
| github.com/yandex-cloud/go-sdk v0.0.0-20241220131134-2393e243c134 h1:qmpz0Kvr9GAng8LAhRcKIpY71CEAcL3EBkftVlsP5Cw= | ||||
| github.com/yandex-cloud/go-sdk v0.0.0-20241220131134-2393e243c134/go.mod h1:KgZCJrxdhdw/sKhTQ/M3S9WOLri2PCnBlc4C3s+PfKY= | ||||
| github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d/go.mod h1:rHwXgn7JulP+udvsHwJoVG1YGAP6VLg4y9I5dyZdqmA= | ||||
| github.com/yudai/gojsondiff v1.0.0 h1:27cbfqXLVEJ1o8I6v3y9lg8Ydm53EKqHXAOMxEGlCOA= | ||||
| github.com/yudai/gojsondiff v1.0.0/go.mod h1:AY32+k2cwILAkW1fbgxQ5mUmMiZFgLIV+FBNExI05xg= | ||||
| github.com/yudai/golcs v0.0.0-20170316035057-ecda9a501e82 h1:BHyfKlQyqbsFN5p3IfnEUduWvb9is428/nNb5L3U01M= | ||||
| github.com/yudai/golcs v0.0.0-20170316035057-ecda9a501e82/go.mod h1:lgjkn3NuSvDfVJdfcVVdX+jpBxNmX4rDAzaS45IcYoM= | ||||
| github.com/yudai/pp v2.0.1+incompatible/go.mod h1:PuxR/8QJ7cyCkFp/aUDS+JY727OFEZkTdatxwunjIkc= | ||||
| github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||
| github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||
| github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= | ||||
| github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= | ||||
| github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo0= | ||||
| github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0= | ||||
| go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU= | ||||
| go.mongodb.org/mongo-driver v1.12.0 h1:aPx33jmn/rQuJXPQLZQ8NtfPQG8CaqgLThFtqRb0PiE= | ||||
| go.mongodb.org/mongo-driver v1.12.0/go.mod h1:AZkxhPnFJUoH7kZlFkVKucV20K387miPfm7oimrSmK0= | ||||
| go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= | ||||
| go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= | ||||
| go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0= | ||||
| go.opencensus.io v0.24.0/go.mod h1:vNK8G9p7aAivkbmorf4v+7Hgx+Zs0yY+0fOtgBfjQKo= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.54.0 h1:TT4fX+nBOA/+LUkobKGW1ydGcn+G3vRw9+g5HwCphpk= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.54.0/go.mod h1:L7UH0GbB0p47T4Rri3uHjbpCFYrVrwc1I25QhNPiGK8= | ||||
| go.opentelemetry.io/otel v1.29.0 h1:PdomN/Al4q/lN6iBJEN3AwPvUiHPMlt93c8bqTG5Llw= | ||||
| @@ -758,7 +694,6 @@ go.opentelemetry.io/otel/sdk v1.28.0 h1:b9d7hIry8yZsgtbmM0DKyPWMMUMlK9NEKuIG4aBq | ||||
| go.opentelemetry.io/otel/sdk v1.28.0/go.mod h1:oYj7ClPUA7Iw3m+r7GeEjz0qckQRJK2B8zjcZEfu7Pg= | ||||
| go.opentelemetry.io/otel/trace v1.29.0 h1:J/8ZNK4XgR7a21DZUAsbF8pZ5Jcw1VhACmnYt39JTi4= | ||||
| go.opentelemetry.io/otel/trace v1.29.0/go.mod h1:eHl3w0sp3paPkYstJOmAimxhiFXPg+MMTlEh3nsQgWQ= | ||||
| go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI= | ||||
| go.opentelemetry.io/proto/otlp v1.2.0 h1:pVeZGk7nXDC9O2hncA6nHldxEjm6LByfA2aN8IOkz94= | ||||
| go.opentelemetry.io/proto/otlp v1.2.0/go.mod h1:gGpR8txAl5M03pDhMC79G6SdqNV26naRm/KDsgaHD8A= | ||||
| go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= | ||||
| @@ -779,7 +714,6 @@ golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPh | ||||
| golang.org/x/crypto v0.0.0-20201012173705-84dcc777aaee/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20201016220609-9e8e0b390897/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||
| golang.org/x/crypto v0.0.0-20220214200702-86341886e292/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||
| golang.org/x/crypto v0.0.0-20220622213112-05595931fe9d/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||
| golang.org/x/crypto v0.0.0-20220722155217-630584e8d5aa/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||
| golang.org/x/crypto v0.0.0-20220829220503-c86fa9a7ed90/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||
| @@ -787,8 +721,8 @@ golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58 | ||||
| golang.org/x/crypto v0.17.0/go.mod h1:gCAAfMLgwOJRpTjQ2zCCt2OcSfYMTeZVSRtQlPC7Nq4= | ||||
| golang.org/x/crypto v0.19.0/go.mod h1:Iy9bg/ha4yyC70EfRS8jz+B6ybOBKMaSxLj6P6oBDfU= | ||||
| golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8= | ||||
| golang.org/x/crypto v0.27.0 h1:GXm2NjJrPaiv/h1tb2UH8QfgC/hOf/+z0p6PT8o1w7A= | ||||
| golang.org/x/crypto v0.27.0/go.mod h1:1Xngt8kV6Dvbssa53Ziq6Eqn0HqbZi5Z6R0ZpwQzt70= | ||||
| golang.org/x/crypto v0.31.0 h1:ihbySMvVjLAeSH1IbfcRTkD/iNscyz8rGzjF/E5hV6U= | ||||
| golang.org/x/crypto v0.31.0/go.mod h1:kDsLvtWBEx7MV9tJOj9bnXsPbxwJQ6csT/x4KIN4Ssk= | ||||
| golang.org/x/exp v0.0.0-20180321215751-8460e604b9de/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20180807140117-3d87b88a115f/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| @@ -817,8 +751,8 @@ golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= | ||||
| golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= | ||||
| golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | ||||
| golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= | ||||
| golang.org/x/mod v0.21.0 h1:vvrHzRwRfVKSiLrG+d4FMl/Qi4ukBCE6kZlTUkDYRT0= | ||||
| golang.org/x/mod v0.21.0/go.mod h1:6SkKJ3Xj0I0BrPOZoBy3bdMptDDU9oJrpohJ3eWZ1fY= | ||||
| golang.org/x/mod v0.22.0 h1:D4nJWe9zXqHOmWqj4VMOJhvzj7bEZg4wEYa759z1pH4= | ||||
| golang.org/x/mod v0.22.0/go.mod h1:6SkKJ3Xj0I0BrPOZoBy3bdMptDDU9oJrpohJ3eWZ1fY= | ||||
| golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| @@ -839,10 +773,8 @@ golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLL | ||||
| golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= | ||||
| golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= | ||||
| golang.org/x/net v0.0.0-20201010224723-4f7140c49acb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc= | ||||
| golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= | ||||
| @@ -854,15 +786,13 @@ golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug | ||||
| golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= | ||||
| golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= | ||||
| golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44= | ||||
| golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM= | ||||
| golang.org/x/net v0.29.0 h1:5ORfpBpCs4HzDYoodCDBbwHzdR5UrLBZ3sOnUJmFoHo= | ||||
| golang.org/x/net v0.29.0/go.mod h1:gLkgy8jTGERgjzMic6DS9+SP0ajcu6Xu3Orq/SpETg0= | ||||
| golang.org/x/net v0.33.0 h1:74SYHlV8BIgHIFC/LrYkOGIwL19eTYXQ5wc6TBuO36I= | ||||
| golang.org/x/net v0.33.0/go.mod h1:HXLR5J+9DxmrqMwG9qjGCxZ+zKXxBru04zlTvWlWuN4= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||
| golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/oauth2 v0.23.0 h1:PbgcYx2W7i4LvjJWEbf0ngHV6qJYr86PkAV3bXdLEbs= | ||||
| golang.org/x/oauth2 v0.23.0/go.mod h1:XYTD2NtWslqkgxebSiOHnXEap4TF09sJSc7H1sXbhtI= | ||||
| golang.org/x/oauth2 v0.24.0 h1:KTBBxWqUa0ykRPLtV69rRto9TLXcqYkeswu48x/gvNE= | ||||
| golang.org/x/oauth2 v0.24.0/go.mod h1:XYTD2NtWslqkgxebSiOHnXEap4TF09sJSc7H1sXbhtI= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| @@ -873,8 +803,8 @@ golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJ | ||||
| golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ= | ||||
| golang.org/x/sync v0.8.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||
| golang.org/x/sync v0.10.0 h1:3NQrjDixjgGwUOCaF8w2+VYHv0Ve/vGYSbdkTa98gmQ= | ||||
| golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||
| golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| @@ -893,6 +823,7 @@ golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7w | ||||
| golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190801041406-cbf593c0f2f3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190924154521-2837fb4f24fe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191008105621-543471e840be/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| @@ -901,12 +832,12 @@ golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7w | ||||
| golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200519105757-fe76b779f299/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200909081042-eff7692f9009/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200918174421-af09f7315aff/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201110211018-35f3e6cf4a65/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201204225414-ed752295db88/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| @@ -916,7 +847,6 @@ golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBc | ||||
| golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220209214540-3681064d5158/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220227234510-4e6760a101f9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| @@ -927,8 +857,8 @@ golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.15.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/sys v0.25.0 h1:r+8e+loiHxRqhXVl6ML1nO3l1+oFoWbnlu2Ehimmi34= | ||||
| golang.org/x/sys v0.25.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/sys v0.28.0 h1:Fksou7UEQUWlKvIdsqzJmUmCX3cZuD2+P3XyyzwMhlA= | ||||
| golang.org/x/sys v0.28.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||
| golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= | ||||
| @@ -936,14 +866,13 @@ golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo= | ||||
| golang.org/x/term v0.15.0/go.mod h1:BDl952bC7+uMoWR75FIrCDx79TPU9oHkTZ9yRbYOrX0= | ||||
| golang.org/x/term v0.17.0/go.mod h1:lLRBjIVuehSbZlaOtGMbcMncT+aqLLLmKrsjNrUguwk= | ||||
| golang.org/x/term v0.20.0/go.mod h1:8UkIAJTvZgivsXaD6/pH6U9ecQzZ45awqEOzuCvwpFY= | ||||
| golang.org/x/term v0.24.0 h1:Mh5cbb+Zk2hqqXNO7S1iTjEphVL+jb8ZWaqh/g+JWkM= | ||||
| golang.org/x/term v0.24.0/go.mod h1:lOBK/LVxemqiMij05LGJ0tzNr8xlmwBRJ81PX6wVLH8= | ||||
| golang.org/x/term v0.27.0 h1:WP60Sv1nlK1T6SupCHbXzSaN0b9wUmsPoRS9b61A23Q= | ||||
| golang.org/x/term v0.27.0/go.mod h1:iMsnZpn0cago0GOrHO2+Y7u7JPn5AylBrcoWkElMTSM= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | ||||
| golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= | ||||
| @@ -951,14 +880,13 @@ golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= | ||||
| golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= | ||||
| golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= | ||||
| golang.org/x/text v0.15.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= | ||||
| golang.org/x/text v0.18.0 h1:XvMDiNzPAl0jr17s6W9lcaIhGUfUORdGCNsuLmPG224= | ||||
| golang.org/x/text v0.18.0/go.mod h1:BuEKDfySbSR4drPmRPG/7iBdf8hvFMuRexcpahXilzY= | ||||
| golang.org/x/text v0.21.0 h1:zyQAAkrwaneQ066sspRyJaG9VNi/YJ1NfzcGB3hZ/qo= | ||||
| golang.org/x/text v0.21.0/go.mod h1:4IBbMaMmOPCJ8SecivzSH54+73PCFmPWxNTLm+vZkEQ= | ||||
| golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.3.0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.5.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | ||||
| golang.org/x/time v0.6.0 h1:eTDhh4ZXt5Qf0augr54TN6suAUudPcawVZeIAPU7D4U= | ||||
| golang.org/x/time v0.6.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | ||||
| golang.org/x/time v0.8.0 h1:9i3RxcPv3PZnitoVGMPDKZSq1xW1gK1Xy3ArNOGZfEg= | ||||
| golang.org/x/time v0.8.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | ||||
| golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20180525024113-a5b4c53f6e8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| @@ -987,8 +915,8 @@ golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4f | ||||
| golang.org/x/tools v0.1.6-0.20210726203631-07bc1bf47fb2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= | ||||
| golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= | ||||
| golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= | ||||
| golang.org/x/tools v0.25.0 h1:oFU9pkj/iJgs+0DT+VMHrx+oBKs/LJMV+Uvg78sl+fE= | ||||
| golang.org/x/tools v0.25.0/go.mod h1:/vtpO8WL1N9cQC3FN5zPqb//fRXskFHbLKk4OW1Q7rg= | ||||
| golang.org/x/tools v0.28.0 h1:WuB6qZ4RPCQo5aP3WdKZS7i595EdWqWR8vqJTlwTVK8= | ||||
| golang.org/x/tools v0.28.0/go.mod h1:dcIOrVd3mfQKTgrDVQHqCPMWy6lnhfhtX3hLXYVLfRw= | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| @@ -1002,13 +930,12 @@ google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E | ||||
| google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= | ||||
| google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= | ||||
| google.golang.org/api v0.13.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= | ||||
| google.golang.org/api v0.197.0 h1:x6CwqQLsFiA5JKAiGyGBjc2bNtHtLddhJCE2IKuhhcQ= | ||||
| google.golang.org/api v0.197.0/go.mod h1:AuOuo20GoQ331nq7DquGHlU6d+2wN2fZ8O0ta60nRNw= | ||||
| google.golang.org/api v0.214.0 h1:h2Gkq07OYi6kusGOaT/9rnNljuXmqPnaig7WGPmKbwA= | ||||
| google.golang.org/api v0.214.0/go.mod h1:bYPpLG8AyeMWwDU6NXoB00xC0DFkikVvd5MfwoxjLqE= | ||||
| google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= | ||||
| google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= | ||||
| google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= | ||||
| google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | ||||
| google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| @@ -1018,43 +945,31 @@ google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98 | ||||
| google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | ||||
| google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8= | ||||
| google.golang.org/genproto v0.0.0-20191108220845-16a3f7862a1a/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= | ||||
| google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= | ||||
| google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= | ||||
| google.golang.org/genproto v0.0.0-20211021150943-2b146023228c/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= | ||||
| google.golang.org/genproto v0.0.0-20240903143218-8af14fe29dc1 h1:BulPr26Jqjnd4eYDVe+YvyR7Yc2vJGkO5/0UxD0/jZU= | ||||
| google.golang.org/genproto v0.0.0-20240903143218-8af14fe29dc1/go.mod h1:hL97c3SYopEHblzpxRL4lSs523++l8DYxGM1FQiYmb4= | ||||
| google.golang.org/genproto/googleapis/api v0.0.0-20240827150818-7e3bb234dfed h1:3RgNmBoI9MZhsj3QxC+AP/qQhNwpCLOvYDYYsFrhFt0= | ||||
| google.golang.org/genproto/googleapis/api v0.0.0-20240827150818-7e3bb234dfed/go.mod h1:OCdP9MfskevB/rbYvHTsXTtKC+3bHWajPdoKgjcYkfo= | ||||
| google.golang.org/genproto/googleapis/rpc v0.0.0-20240903143218-8af14fe29dc1 h1:pPJltXNxVzT4pK9yD8vR9X75DaWYYmLGMsEvBfFQZzQ= | ||||
| google.golang.org/genproto/googleapis/rpc v0.0.0-20240903143218-8af14fe29dc1/go.mod h1:UqMtugtsSgubUsoxbuAoiCXvqvErP7Gf0so0mK9tHxU= | ||||
| google.golang.org/genproto v0.0.0-20241021214115-324edc3d5d38 h1:Q3nlH8iSQSRUwOskjbcSMcF2jiYMNiQYZ0c2KEJLKKU= | ||||
| google.golang.org/genproto v0.0.0-20241021214115-324edc3d5d38/go.mod h1:xBI+tzfqGGN2JBeSebfKXFSdBpWVQ7sLW40PTupVRm4= | ||||
| google.golang.org/genproto/googleapis/api v0.0.0-20241118233622-e639e219e697 h1:pgr/4QbFyktUv9CtQ/Fq4gzEE6/Xs7iCXbktaGzLHbQ= | ||||
| google.golang.org/genproto/googleapis/api v0.0.0-20241118233622-e639e219e697/go.mod h1:+D9ySVjN8nY8YCVjc5O7PZDIdZporIDY3KaGfJunh88= | ||||
| google.golang.org/genproto/googleapis/rpc v0.0.0-20241209162323-e6fa225c2576 h1:8ZmaLZE4XWrtU3MyClkYqqtl6Oegr3235h7jxsDyqCY= | ||||
| google.golang.org/genproto/googleapis/rpc v0.0.0-20241209162323-e6fa225c2576/go.mod h1:5uTbfoYQed2U9p3KIj2/Zzm02PYhndfdmML0qC3q3FU= | ||||
| google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= | ||||
| google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= | ||||
| google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= | ||||
| google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= | ||||
| google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= | ||||
| google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | ||||
| google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= | ||||
| google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0= | ||||
| google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= | ||||
| google.golang.org/grpc v1.36.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= | ||||
| google.golang.org/grpc v1.40.0/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= | ||||
| google.golang.org/grpc v1.41.0/go.mod h1:U3l9uK9J0sini8mHphKoXyaqDA/8VyGnDee1zzIUK6k= | ||||
| google.golang.org/grpc v1.66.1 h1:hO5qAXR19+/Z44hmvIM4dQFMSYX9XcWsByfoxutBpAM= | ||||
| google.golang.org/grpc v1.66.1/go.mod h1:s3/l6xSSCURdVfAnL+TqCNMyTDAGN6+lZeVxnZR128Y= | ||||
| google.golang.org/grpc v1.67.1 h1:zWnc1Vrcno+lHZCOofnIMvycFcc0QRGIzm9dhnDX68E= | ||||
| google.golang.org/grpc v1.67.1/go.mod h1:1gLDyUQU7CTLJI90u3nXZ9ekeghjeM7pTDZlqFNg2AA= | ||||
| google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= | ||||
| google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= | ||||
| google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= | ||||
| google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= | ||||
| google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= | ||||
| google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| google.golang.org/protobuf v1.34.2 h1:6xV6lTsCfpGD21XK49h7MhtcApnLqkfYgPcdHftf6hg= | ||||
| google.golang.org/protobuf v1.34.2/go.mod h1:qYOHts0dSfpeUzUFpOMr/WGzszTmLH+DiWniOlNbLDw= | ||||
| google.golang.org/protobuf v1.35.2 h1:8Ar7bF+apOIoThw1EdZl0p1oWvMqTHmpA2fRTyZO8io= | ||||
| google.golang.org/protobuf v1.35.2/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE= | ||||
| gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| @@ -1068,22 +983,20 @@ gopkg.in/ini.v1 v1.51.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= | ||||
| gopkg.in/ini.v1 v1.62.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= | ||||
| gopkg.in/ini.v1 v1.67.0 h1:Dgnx+6+nfE+IfzjUEISNeydPJh9AXNNsWbGP9KzCsOA= | ||||
| gopkg.in/ini.v1 v1.67.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= | ||||
| gopkg.in/ns1/ns1-go.v2 v2.12.0 h1:cqdqQoTx17JmTusfxh5m3e2b36jfUzFAZedv89pFX18= | ||||
| gopkg.in/ns1/ns1-go.v2 v2.12.0/go.mod h1:pfaU0vECVP7DIOr453z03HXS6dFJpXdNRwOyRzwmPSc= | ||||
| gopkg.in/ns1/ns1-go.v2 v2.13.0 h1:I5NNqI9Bi1SGK92TVkOvLTwux5LNrix/99H2datVh48= | ||||
| gopkg.in/ns1/ns1-go.v2 v2.13.0/go.mod h1:pfaU0vECVP7DIOr453z03HXS6dFJpXdNRwOyRzwmPSc= | ||||
| gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | ||||
| gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74= | ||||
| gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= | ||||
| gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200615113413-eeeca48fe776/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| gopkg.in/yaml.v3 v3.0.0/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||
|   | ||||
| @@ -50,7 +50,7 @@ import ( | ||||
| /* SIGTERM handler, do shutdown sequences before closing */ | ||||
| func SetupCloseHandler() { | ||||
| 	c := make(chan os.Signal, 2) | ||||
| 	signal.Notify(c, os.Interrupt, syscall.SIGTERM) | ||||
| 	signal.Notify(c, os.Interrupt, syscall.SIGTERM, syscall.SIGINT) | ||||
| 	go func() { | ||||
| 		<-c | ||||
| 		ShutdownSeq() | ||||
|   | ||||
| @@ -94,7 +94,7 @@ func NewAccessController(options *Options) (*Controller, error) { | ||||
| 		thisAccessRule := AccessRule{} | ||||
| 		err = json.Unmarshal(configContent, &thisAccessRule) | ||||
| 		if err != nil { | ||||
| 			options.Logger.PrintAndLog("Access", "Unable to parse config "+filepath.Base(configFile), err) | ||||
| 			options.Logger.PrintAndLog("access", "Unable to parse config "+filepath.Base(configFile), err) | ||||
| 			continue | ||||
| 		} | ||||
| 		thisAccessRule.parent = &thisController | ||||
| @@ -102,6 +102,19 @@ func NewAccessController(options *Options) (*Controller, error) { | ||||
| 	} | ||||
| 	thisController.ProxyAccessRule = &ProxyAccessRules | ||||
|  | ||||
| 	//Start the public ip ticker | ||||
| 	if options.PublicIpCheckInterval <= 0 { | ||||
| 		options.PublicIpCheckInterval = 12 * 60 * 60 //12 hours | ||||
| 	} | ||||
| 	thisController.ServerPublicIP = "127.0.0.1" | ||||
| 	go func() { | ||||
| 		err = thisController.UpdatePublicIP() | ||||
| 		if err != nil { | ||||
| 			options.Logger.PrintAndLog("access", "Unable to update public IP address", err) | ||||
| 		} | ||||
|  | ||||
| 		thisController.StartPublicIPUpdater() | ||||
| 	}() | ||||
| 	return &thisController, nil | ||||
| } | ||||
|  | ||||
| @@ -147,11 +160,7 @@ func (c *Controller) ListAllAccessRules() []*AccessRule { | ||||
| // Check if an access rule exists given the rule id | ||||
| func (c *Controller) AccessRuleExists(ruleID string) bool { | ||||
| 	r, _ := c.GetAccessRuleByID(ruleID) | ||||
| 	if r != nil { | ||||
| 		//An access rule with identical ID exists | ||||
| 		return true | ||||
| 	} | ||||
| 	return false | ||||
| 	return r != nil | ||||
| } | ||||
|  | ||||
| // Add a new access rule to runtime and save it to file | ||||
| @@ -219,3 +228,7 @@ func (c *Controller) RemoveAccessRuleByID(ruleID string) error { | ||||
| 	//Remove it | ||||
| 	return c.DeleteAccessRuleByID(ruleID) | ||||
| } | ||||
|  | ||||
| func (c *Controller) Close() { | ||||
| 	c.StopPublicIPUpdater() | ||||
| } | ||||
|   | ||||
| @@ -25,18 +25,24 @@ func (s *AccessRule) AllowConnectionAccess(conn net.Conn) bool { | ||||
| 	return true | ||||
| } | ||||
|  | ||||
| // Toggle black list | ||||
| // Toggle blacklist | ||||
| func (s *AccessRule) ToggleBlacklist(enabled bool) { | ||||
| 	s.BlacklistEnabled = enabled | ||||
| 	s.SaveChanges() | ||||
| } | ||||
|  | ||||
| // Toggel white list | ||||
| // Toggel whitelist | ||||
| func (s *AccessRule) ToggleWhitelist(enabled bool) { | ||||
| 	s.WhitelistEnabled = enabled | ||||
| 	s.SaveChanges() | ||||
| } | ||||
|  | ||||
| // Toggle whitelist loopback | ||||
| func (s *AccessRule) ToggleAllowLoopback(enabled bool) { | ||||
| 	s.WhitelistAllowLocalAndLoopback = enabled | ||||
| 	s.SaveChanges() | ||||
| } | ||||
|  | ||||
| /* | ||||
| Check if a IP address is blacklisted, in either country or IP blacklist | ||||
| IsBlacklisted default return is false (allow access) | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user