Compare commits
615 Commits
2.6.4
...
273cae2a98
Author | SHA1 | Date | |
---|---|---|---|
![]() |
273cae2a98 | ||
![]() |
6b3b89f7bf | ||
![]() |
2d611a559a | ||
![]() |
6c5eba01c2 | ||
![]() |
f641797d10 | ||
![]() |
f92ff068f3 | ||
![]() |
b59ac47c8c | ||
![]() |
8030f3d62a | ||
![]() |
f8f623e3e4 | ||
![]() |
061839756c | ||
![]() |
1dcaa0c257 | ||
![]() |
ffd3909964 | ||
![]() |
3ddccdffce | ||
![]() |
929d4cc82a | ||
![]() |
4f1cd8a571 | ||
![]() |
f6b3656bb1 | ||
![]() |
74a816216e | ||
![]() |
4a093cf096 | ||
![]() |
68f9fccf3a | ||
![]() |
f276040ad0 | ||
![]() |
2f40593daf | ||
![]() |
0b6dbd49bb | ||
![]() |
eb07917c14 | ||
![]() |
217bc48001 | ||
![]() |
38cfab4a09 | ||
![]() |
217e5e90ff | ||
![]() |
4a37a989a0 | ||
![]() |
eb540b774d | ||
![]() |
26d03f9ad4 | ||
![]() |
31ba4f20ae | ||
![]() |
650d61ba24 | ||
![]() |
6d0c0be8c2 | ||
![]() |
366a44a992 | ||
![]() |
7164b74d4a | ||
![]() |
b01a21f318 | ||
![]() |
809e1fa815 | ||
![]() |
c7b5e0994e | ||
![]() |
1f8684481a | ||
![]() |
0e74ff69c3 | ||
![]() |
f0fa71c5b4 | ||
![]() |
8cb47e19fa | ||
![]() |
475650de0d | ||
![]() |
b19867865c | ||
![]() |
df636c9f76 | ||
![]() |
e6b2cf09d7 | ||
![]() |
bf0df928c7 | ||
![]() |
eec6cec0db | ||
![]() |
0215171646 | ||
![]() |
cd822ed904 | ||
![]() |
a1df5a1060 | ||
![]() |
54c130165a | ||
![]() |
ddb1a8773e | ||
![]() |
29daa4402d | ||
![]() |
a85bf82c3e | ||
![]() |
c56e317bfd | ||
![]() |
b4c2f6bf13 | ||
![]() |
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 | ||
![]() |
d1e5581eea | ||
![]() |
be5797c8a5 | ||
![]() |
ebd316a7f1 | ||
![]() |
84aec4387a | ||
![]() |
30dfb9cb65 | ||
![]() |
0b1768ab5b | ||
![]() |
ad4721820b | ||
![]() |
1d4c275db3 | ||
![]() |
b3ad97743c | ||
![]() |
1a6a87e79b | ||
![]() |
749fd4b7af | ||
![]() |
85422c0a74 | ||
![]() |
73999c1ae9 | ||
![]() |
0ad84b3415 | ||
![]() |
64b6769695 | ||
![]() |
e72b2f9e09 | ||
![]() |
992dd231f2 | ||
![]() |
49555c1191 | ||
![]() |
2fca458bd0 | ||
![]() |
2423d0fb3a | ||
![]() |
bb0f55018c | ||
![]() |
9e95d84627 | ||
![]() |
e73841786b | ||
![]() |
d5449c947a | ||
![]() |
8ff51044bb | ||
![]() |
cc08c704de | ||
![]() |
2f1a6b5ba4 | ||
![]() |
4d163fe80f | ||
![]() |
24371ed22e | ||
![]() |
12358d3522 | ||
![]() |
c39af1ff8e | ||
![]() |
6bf944e13c | ||
![]() |
b653b805b8 | ||
![]() |
eb91865b70 | ||
![]() |
57e72a8a90 | ||
![]() |
4dbf110edc | ||
![]() |
1eefa99b72 | ||
![]() |
e6b2d458f7 | ||
![]() |
4a4483e09d | ||
![]() |
4485d1f811 | ||
![]() |
0eb0696670 | ||
![]() |
9fca2354c6 | ||
![]() |
e56b045689 | ||
![]() |
763ccb4d60 | ||
![]() |
4d4492069d | ||
![]() |
f3591aa171 | ||
![]() |
2dcf578cbe | ||
![]() |
23a5c6ceb0 | ||
![]() |
015889851a | ||
![]() |
093ed9c212 | ||
![]() |
0af8c67346 | ||
![]() |
c5170bcb94 | ||
![]() |
cd48388c02 | ||
![]() |
373845f8fd | ||
![]() |
293a527ffc | ||
![]() |
e4facbc7b6 | ||
![]() |
1c79fa4e96 | ||
![]() |
6515eb99e3 | ||
![]() |
ec5c24b9b8 | ||
![]() |
df88084375 | ||
![]() |
74017baecf | ||
![]() |
294d504ee6 | ||
![]() |
477429900e | ||
![]() |
2e9bc77a5d | ||
![]() |
ed178d857a | ||
![]() |
4cf5d29692 | ||
![]() |
634e9c9855 | ||
![]() |
e79a70b7ac | ||
![]() |
779115d06b | ||
![]() |
9cb315ea67 | ||
![]() |
43ba00ec8d | ||
![]() |
4577fb1f2f | ||
![]() |
f877bf9eda | ||
![]() |
363b9b6d94 | ||
![]() |
c5ca68868b | ||
![]() |
f927bb539a | ||
![]() |
5f64b622b5 | ||
![]() |
9a371f5bcb | ||
![]() |
172c5afa60 | ||
![]() |
f98e04a9fc | ||
![]() |
99295cad86 | ||
![]() |
95d0a98576 | ||
![]() |
00bfa262cb | ||
![]() |
528be69fe0 | ||
![]() |
6923f0d200 | ||
![]() |
7255b62e31 | ||
![]() |
cf14d12c31 | ||
![]() |
90cf26306a | ||
![]() |
cab2f4e63a | ||
![]() |
75d773887c | ||
![]() |
a944c3ff36 | ||
![]() |
465f332dfc | ||
![]() |
dfda3fe94b | ||
![]() |
5c56da1180 | ||
![]() |
3392013a5c | ||
![]() |
8b4c601d50 | ||
![]() |
3a2eaf8766 | ||
![]() |
a45092a449 | ||
![]() |
d5315e5b8e | ||
![]() |
31cc1a69a1 | ||
![]() |
d348cbf48b | ||
![]() |
f6339868ac | ||
![]() |
af10f2a644 | ||
![]() |
3b247c31da | ||
![]() |
d74e8badb9 | ||
![]() |
b40131d212 | ||
![]() |
563a12c860 | ||
![]() |
8b2c3b7e03 | ||
![]() |
608cc0c523 | ||
![]() |
b558bcbfcf | ||
![]() |
9ea3fa2542 | ||
![]() |
01f68c5ef5 | ||
![]() |
a7f89086d4 | ||
![]() |
a5ef6456c6 | ||
![]() |
87659b43bd | ||
![]() |
ddbecf7b68 | ||
![]() |
1b3a9de378 | ||
![]() |
6dd62f509d | ||
![]() |
d5cc6a6859 | ||
![]() |
1d965da7d0 | ||
![]() |
3567c70bab | ||
![]() |
0a734e0bd3 | ||
![]() |
f4fa92635c | ||
![]() |
7d5151bb00 | ||
![]() |
54475e4b99 | ||
![]() |
6ac16caf37 | ||
![]() |
97502db607 | ||
![]() |
0747cf4b0f | ||
![]() |
94483acc92 | ||
![]() |
7626857c02 | ||
![]() |
0f772a715b | ||
![]() |
fd1439f746 | ||
![]() |
ca37bfbfa6 | ||
![]() |
c1e16d55ab | ||
![]() |
f595da92a1 | ||
![]() |
8a8ec1cb0b | ||
![]() |
e53c3cf3c4 | ||
![]() |
d17de5c200 | ||
![]() |
97ff48ee70 | ||
![]() |
d64b1174af | ||
![]() |
bec363abab | ||
![]() |
0dddd1f9e3 | ||
![]() |
6bfcb2e1f5 | ||
![]() |
02ff288280 | ||
![]() |
b1c5bc2963 | ||
![]() |
d3dbbf9052 | ||
![]() |
f4a5c905e7 | ||
![]() |
245379e91f | ||
![]() |
955a2232df | ||
![]() |
7eb7ae7ced | ||
![]() |
3aa0f2d914 | ||
![]() |
39b0c8c674 | ||
![]() |
bddeae8365 | ||
![]() |
8e0e9531e7 | ||
![]() |
6ff22865e0 | ||
![]() |
0828fd1958 | ||
![]() |
82f84470f7 | ||
![]() |
cf9a05f130 | ||
![]() |
301072db90 | ||
![]() |
cfcd10d64f | ||
![]() |
c85760c73a | ||
![]() |
b7bb918aa3 | ||
![]() |
962f3e0566 | ||
![]() |
0bcf2b2ae3 | ||
![]() |
6bfeb8cf3d | ||
![]() |
33def66386 | ||
![]() |
cb469f28d2 | ||
![]() |
8239f4cb53 | ||
![]() |
e410b92e34 | ||
![]() |
aca6e44b35 | ||
![]() |
2aa35cbe6d | ||
![]() |
745a54605f | ||
![]() |
e3b61868a1 | ||
![]() |
764b1944be | ||
![]() |
100cd727fc | ||
![]() |
7e62fef879 | ||
![]() |
1a4a55721f | ||
![]() |
bb9deccff6 | ||
![]() |
a18413dd03 | ||
![]() |
2cd1b1de3c | ||
![]() |
3a2db63d61 | ||
![]() |
123d3bcf3f | ||
![]() |
3ec1d9c888 | ||
![]() |
5785261c7e | ||
![]() |
89e60649e5 | ||
![]() |
5423b82858 | ||
![]() |
57135a867e | ||
![]() |
547855f30f | ||
![]() |
05b477e90a | ||
![]() |
3519c7841c | ||
![]() |
e7b4054248 | ||
![]() |
973d0b3372 | ||
![]() |
704980d4f8 | ||
![]() |
03974163d4 | ||
![]() |
dfb81513b1 | ||
![]() |
b604c66a2f | ||
![]() |
dd84864dd4 | ||
![]() |
443cd961d2 | ||
![]() |
10048150bb | ||
![]() |
85f9b297c4 | ||
![]() |
07e524a007 | ||
![]() |
25c7e8ac1a | ||
![]() |
49babbd60f | ||
![]() |
fa11422748 | ||
![]() |
bb1b161ae2 | ||
![]() |
9545343151 | ||
![]() |
61e4d45430 | ||
![]() |
6026c4fd53 | ||
![]() |
e3f8c99ed3 | ||
![]() |
fc88dfe72e | ||
![]() |
d43322f7a5 | ||
![]() |
83536a83f7 | ||
![]() |
1183b0ed55 | ||
![]() |
b00e302f6d | ||
![]() |
deddb17803 | ||
![]() |
aa96d831e1 | ||
![]() |
c6f7f37aaf | ||
![]() |
63f12dedcf | ||
![]() |
136d1ecafb | ||
![]() |
7193defad7 | ||
![]() |
cf4c57298e | ||
![]() |
d82a531a41 | ||
![]() |
7694e317f7 | ||
![]() |
ed4945ab7e | ||
![]() |
ce8741bfc8 | ||
![]() |
7a3db09811 | ||
![]() |
e73f9b47d3 | ||
![]() |
c248dacccf | ||
![]() |
d596d6b843 | ||
![]() |
6feb2d105d | ||
![]() |
3a26a5b4d3 | ||
![]() |
2cdd5654ed | ||
![]() |
a0d362df4e | ||
![]() |
334c1ab131 | ||
![]() |
08d52024ab | ||
![]() |
a3e16594e8 | ||
![]() |
cced07ba2d | ||
![]() |
2003992d75 | ||
![]() |
71423d98b1 | ||
![]() |
8ca716c59f | ||
![]() |
fe48a9a0c3 | ||
![]() |
ec973eb3bc | ||
![]() |
7b69b5fa63 | ||
![]() |
ce4f46cb50 | ||
![]() |
3454a9b975 | ||
![]() |
55bc939a37 | ||
![]() |
1d63b679dc | ||
![]() |
3df96350a3 | ||
![]() |
34fab7b3d0 | ||
![]() |
46817d0664 | ||
![]() |
1db2ca61fa | ||
![]() |
0b601406de | ||
![]() |
b4c771cdee | ||
![]() |
a486d42351 | ||
![]() |
90c2199a1b | ||
![]() |
161c61fac7 | ||
![]() |
5ffacb1d06 | ||
![]() |
75ebd0ffbe | ||
![]() |
dc069f3c57 | ||
![]() |
e1b512f78f | ||
![]() |
8854a38f49 | ||
![]() |
7583a4628c | ||
![]() |
73c0ea0896 | ||
![]() |
7dad7c7305 | ||
![]() |
faa95b4e21 | ||
![]() |
cb0e13976d | ||
![]() |
ccd8dcff56 | ||
![]() |
750656fd7f | ||
![]() |
d9f515fdba | ||
![]() |
176249a7d9 | ||
![]() |
e2a449a7bc | ||
![]() |
a9695e969e | ||
![]() |
7ba997dfc2 | ||
![]() |
d00117e878 | ||
![]() |
43a84a3f1c | ||
![]() |
e24f31bdef | ||
![]() |
fc9240fbac | ||
![]() |
e0f5431215 | ||
![]() |
de658a3c6c | ||
![]() |
73276b1918 | ||
![]() |
abdb7d4d75 | ||
![]() |
72299ace15 | ||
![]() |
4d6c79f51b | ||
![]() |
2c045f4f40 | ||
![]() |
b8cf046ca6 | ||
![]() |
026dd6b89d | ||
![]() |
5805fe6ed2 | ||
![]() |
3c78211800 | ||
![]() |
8e648a8e1f | ||
![]() |
a000893dd1 | ||
![]() |
db88bfb752 | ||
![]() |
05297d854b | ||
![]() |
0d7bce4d30 | ||
![]() |
8db95dddc6 | ||
![]() |
05daeded37 | ||
![]() |
8ce6471be5 | ||
![]() |
e242c9288f | ||
![]() |
c55a29e7cf | ||
![]() |
6af047430c | ||
![]() |
200c924acd | ||
![]() |
9b2168466c | ||
![]() |
7ae48bf370 | ||
![]() |
ee3d76fb96 | ||
![]() |
40d192524b | ||
![]() |
c659e05005 | ||
![]() |
676a45c222 | ||
![]() |
1da0761b13 | ||
![]() |
32939874f2 | ||
![]() |
43a4bf389a | ||
![]() |
33c7c5fa00 | ||
![]() |
216b53f224 | ||
![]() |
059b0a2e1c | ||
![]() |
3ab952f168 | ||
![]() |
4f676d6770 | ||
![]() |
e980bc847b | ||
![]() |
174efc9080 | ||
![]() |
3228789375 | ||
![]() |
36e461795a | ||
![]() |
d6e7641364 | ||
![]() |
15cebd6e06 | ||
![]() |
e9a074d4d1 | ||
![]() |
4b7fd39e57 | ||
![]() |
fa005f1327 | ||
![]() |
c7a9f40baa | ||
![]() |
d5b9726158 | ||
![]() |
f659e66cf7 | ||
![]() |
801bdbf298 | ||
![]() |
09da93cfb3 | ||
![]() |
70ace02e80 | ||
![]() |
1f758e953d | ||
![]() |
ffad2cab81 | ||
![]() |
dbb10644de | ||
![]() |
4848392185 | ||
![]() |
956f4ac30f | ||
![]() |
c09ff28fd5 | ||
![]() |
20cf290d37 | ||
![]() |
4ca0fcc6d1 | ||
![]() |
ce4ce72820 | ||
![]() |
e363d55899 | ||
![]() |
172479e4fb | ||
![]() |
156fa5dace | ||
![]() |
4d40e0aa38 | ||
![]() |
045e66b631 | ||
![]() |
62e60d78de | ||
![]() |
23bdaa1517 | ||
![]() |
50f222cced | ||
![]() |
640e1adf96 | ||
![]() |
d4bb84180c | ||
![]() |
bda47fc36b | ||
![]() |
fd6ba56143 | ||
![]() |
b63a0fc246 | ||
![]() |
ed92cccf0e | ||
![]() |
95892802fd | ||
![]() |
8a5004e828 | ||
![]() |
c6c523e005 | ||
![]() |
a692ec818d | ||
![]() |
c65f780613 | ||
![]() |
507c2ab468 | ||
![]() |
1180da8d11 | ||
![]() |
b1a14872c3 | ||
![]() |
df9deb3fbb | ||
![]() |
83f574e3ab | ||
![]() |
60837f307d | ||
![]() |
50d5dedabe | ||
![]() |
f15c774c70 | ||
![]() |
069f4805f6 | ||
![]() |
eb98624a6a | ||
![]() |
6a0c7cf499 | ||
![]() |
73ab9ca778 | ||
![]() |
9f9e0750e1 | ||
![]() |
5664965491 | ||
![]() |
db4016e79f | ||
![]() |
f84c4370cf | ||
![]() |
b39cb6391b | ||
![]() |
4f7f60188f | ||
![]() |
dce58343db | ||
![]() |
415838ad39 | ||
![]() |
ce0b1a7585 | ||
![]() |
9369237229 | ||
![]() |
352995e852 | ||
![]() |
a3d55a3274 | ||
![]() |
70adadf129 | ||
![]() |
d42ac8a146 | ||
![]() |
f304ff8862 | ||
![]() |
7d91e02dc9 | ||
![]() |
dae510ae0a | ||
![]() |
cd382a78a5 | ||
![]() |
987de4a7be | ||
![]() |
52d3b2f8c2 | ||
![]() |
5038429a70 | ||
![]() |
2acbf0f3f5 | ||
![]() |
aed703e260 | ||
![]() |
5ece7c0da4 | ||
![]() |
7eda6ba501 | ||
![]() |
2da5ef048f | ||
![]() |
6c48939316 | ||
![]() |
544894bbba | ||
![]() |
153d056bdf | ||
![]() |
12c1118af9 | ||
![]() |
67ba143999 | ||
![]() |
0a8a821394 | ||
![]() |
36b17ce4cf | ||
![]() |
519372069f | ||
![]() |
2f14d6f271 | ||
![]() |
44ac7144ec | ||
![]() |
741d3f8de1 | ||
![]() |
23eca5afae | ||
![]() |
050fab9481 | ||
![]() |
3fc92bac27 | ||
![]() |
594f75da97 | ||
![]() |
3fbf246fb4 | ||
![]() |
828af6263d | ||
![]() |
ab42cec31f |
7
.github/ISSUE_TEMPLATE/bug_report.md
vendored
@@ -28,11 +28,16 @@ If applicable, add screenshots to help explain your problem.
|
|||||||
- Browser [e.g. chrome, safari]
|
- Browser [e.g. chrome, safari]
|
||||||
- Version [e.g. 22]
|
- 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]
|
- Arch: [e.g. arm64]
|
||||||
- Device: [e.g. Bananapi R2 PRO]
|
- Device: [e.g. Bananapi R2 PRO]
|
||||||
- OS: [e.g. Armbian]
|
- OS: [e.g. Armbian]
|
||||||
- Version [e.g. 23.02 Bullseye ]
|
- Version [e.g. 23.02 Bullseye ]
|
||||||
|
- 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**
|
**Additional context**
|
||||||
Add any other context about the problem here.
|
Add any other context about the problem here.
|
||||||
|
25
.github/ISSUE_TEMPLATE/help-needed.md
vendored
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
---
|
||||||
|
name: Help Needed
|
||||||
|
about: Something went wrong but I don't know why
|
||||||
|
title: "[HELP]"
|
||||||
|
labels: help wanted
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**What happened?**
|
||||||
|
A clear and concise description of what the problem is. Ex. I tried to create a proxy rule but it doesn't work. When I connects to my domain, I see [...]
|
||||||
|
|
||||||
|
**Describe what have you tried**
|
||||||
|
A clear and concise description of what you expect to see and what you have tried to debug it.
|
||||||
|
|
||||||
|
**Describe the networking setup you are using**
|
||||||
|
Here are some example, commonly asked questions from our maintainers:
|
||||||
|
- Are you using the docker build of Zoraxy? [yes (with docker setup & networking config attach) /no]
|
||||||
|
- Your Zoraxy version? [e.g. 3.0.4]
|
||||||
|
- Are you using Cloudflare? [yes/no]
|
||||||
|
- Are your system hosted under a NAT router? [e.g. yes, with subnet is e.g. 192.168.0.0/24 and include port forwarding config if any]
|
||||||
|
- DNS record (if any)
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Add any other context or screenshots about the feature request here.
|
60
.github/workflows/docker.yml
vendored
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
name: Build and push Docker image
|
||||||
|
|
||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [ released, prereleased ]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
setup-build-push:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
ref: ${{ github.event.release.tag_name }}
|
||||||
|
|
||||||
|
- name: Set up QEMU
|
||||||
|
uses: docker/setup-qemu-action@v3
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Login to Docker Hub
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
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/src/
|
||||||
|
|
||||||
|
- name: Build and push Docker image (Release)
|
||||||
|
if: "!github.event.release.prerelease"
|
||||||
|
uses: docker/build-push-action@v6
|
||||||
|
with:
|
||||||
|
context: ./docker
|
||||||
|
push: true
|
||||||
|
platforms: linux/amd64,linux/arm64
|
||||||
|
tags: |
|
||||||
|
zoraxydocker/zoraxy:latest
|
||||||
|
zoraxydocker/zoraxy:${{ github.event.release.tag_name }}
|
||||||
|
cache-from: type=gha
|
||||||
|
cache-to: type=gha,mode=max
|
||||||
|
|
||||||
|
- name: Build and push Docker image (Prerelease)
|
||||||
|
if: "github.event.release.prerelease"
|
||||||
|
uses: docker/build-push-action@v6
|
||||||
|
with:
|
||||||
|
context: ./docker
|
||||||
|
push: true
|
||||||
|
platforms: linux/amd64,linux/arm64
|
||||||
|
tags: |
|
||||||
|
zoraxydocker/zoraxy:${{ github.event.release.tag_name }}
|
||||||
|
cache-from: type=gha
|
||||||
|
cache-to: type=gha,mode=max
|
||||||
|
|
29
.gitignore
vendored
@@ -29,3 +29,32 @@ src/Zoraxy_*_*
|
|||||||
src/certs/*
|
src/certs/*
|
||||||
src/rules/*
|
src/rules/*
|
||||||
src/README.md
|
src/README.md
|
||||||
|
docker/ContainerTester.sh
|
||||||
|
docker/docker-compose.yaml
|
||||||
|
src/mod/acme/test/stackoverflow.pem
|
||||||
|
/tools/dns_challenge_update/code-gen/acmedns
|
||||||
|
/tools/dns_challenge_update/code-gen/lego
|
||||||
|
src/tmp/localhost.key
|
||||||
|
src/tmp/localhost.pem
|
||||||
|
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
|
||||||
|
/src/dist
|
||||||
|
356
CHANGELOG.md
@@ -1,19 +1,367 @@
|
|||||||
|
# v3.2.4 28 Jun 2025
|
||||||
|
|
||||||
|
A big release since v3.1.9. Versions from 3.2.0 to 3.2.3 were prereleases.
|
||||||
|
|
||||||
|
|
||||||
|
+ Added Authentik support by [JokerQyou](https://github.com/tobychui/zoraxy/commits?author=JokerQyou)
|
||||||
|
+ Added pluginsystem and moved GAN and Zerotier to plugins
|
||||||
|
+ Add loopback detection [#573](https://github.com/tobychui/zoraxy/issues/573)
|
||||||
|
+ Fixed Dark theme not working with Advanced Option accordion [#591](https://github.com/tobychui/zoraxy/issues/591)
|
||||||
|
+ Update logger to include UserAgent by [Raithmir](https://github.com/Raithmir)
|
||||||
|
+ Fixed memory usage in UI [#600](https://github.com/tobychui/zoraxy/issues/600)
|
||||||
|
+ Added docker-compose.yml by [SamuelPalubaCZ](https://github.com/tobychui/zoraxy/commits?author=SamuelPalubaCZ)
|
||||||
|
+ Added more statistics for proxy hosts [#201](https://github.com/tobychui/zoraxy/issues/201) and [#608](https://github.com/tobychui/zoraxy/issues/608)
|
||||||
|
+ Fixed origin field in logs [#618](https://github.com/tobychui/zoraxy/issues/618)
|
||||||
|
+ Added FreeBSD support by Andreas Burri
|
||||||
|
+ Fixed HTTP proxy redirect [#626](https://github.com/tobychui/zoraxy/issues/626)
|
||||||
|
+ Fixed proxy handling #629](https://github.com/tobychui/zoraxy/issues/629)
|
||||||
|
+ Move Scope ID handling into CIDR check by [Nirostar](https://github.com/tobychui/zoraxy/commits?author=Nirostar)
|
||||||
|
+ Prevent the browser from filling the saved Zoraxy login account by [WHFo](https://github.com/tobychui/zoraxy/commits?author=WHFo)
|
||||||
|
+ Added port number and http proto to http proxy list link
|
||||||
|
+ Fixed headers for authelia by [james-d-elliott](https://github.com/tobychui/zoraxy/commits?author=james-d-elliott)
|
||||||
|
+ Refactored docker container list and UI improvements by [eyerrock](https://github.com/tobychui/zoraxy/commits?author=eyerrock)
|
||||||
|
+ Refactored Dockerfile by [PassiveLemon](https://github.com/tobychui/zoraxy/commits?author=PassiveLemon)
|
||||||
|
+ Added new HTTP proxy UI
|
||||||
|
+ Added inbound host name edit function
|
||||||
|
+ Added static web server option to disable listen to all interface
|
||||||
|
+ Merged SSO implementations (Oauth2) [#649](https://github.com/tobychui/zoraxy/pull/649)
|
||||||
|
+ Merged forward-auth optimization [#692(https://github.com/tobychui/zoraxy/pull/692)
|
||||||
|
+ Optimized SSO UI
|
||||||
|
+ Refactored docker image workflows by [PassiveLemon](https://github.com/tobychui/zoraxy/commits?author=PassiveLemon)
|
||||||
|
+ Added disable chunked transfer encoding checkbox (for upstreams that uses legacy HTTP implementations)
|
||||||
|
+ Bug fixes [#694](https://github.com/tobychui/zoraxy/issues/694), [#659](https://github.com/tobychui/zoraxy/issues/659) by [jemmy1794](https://github.com/tobychui/zoraxy/commits?author=jemmy1794), [#695](https://github.com/tobychui/zoraxy/issues/695)
|
||||||
|
|
||||||
|
# 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)
|
||||||
|
+ Fixed ACME table too wide css bug [#422](https://github.com/tobychui/zoraxy/issues/422)
|
||||||
|
+ Fixed HSTS toggle button bug [#415](https://github.com/tobychui/zoraxy/issues/415)
|
||||||
|
+ Fixed slow GeoIP resolve mode concurrent r/w bug [#401](https://github.com/tobychui/zoraxy/issues/401)
|
||||||
|
+ Added close connection as default site option [#430](https://github.com/tobychui/zoraxy/issues/430)
|
||||||
|
+ Added experimental authelia support [#384](https://github.com/tobychui/zoraxy/issues/384)
|
||||||
|
+ Added custom header support to websocket [#426](https://github.com/tobychui/zoraxy/issues/426)
|
||||||
|
+ Added levelDB as database implementation (not currently used)
|
||||||
|
+ Added external GeoIP db loading support
|
||||||
|
+ Restructured a lot of modules
|
||||||
|
|
||||||
|
# v3.1.4 24 Nov 2024
|
||||||
|
|
||||||
|
+ **Added Dark Theme Mode** [#390](https://github.com/tobychui/zoraxy/issues/390) [#82](https://github.com/tobychui/zoraxy/issues/82)
|
||||||
|
+ Added an auto sniffer for self-signed certificates
|
||||||
|
+ Added robots.txt to the project
|
||||||
|
+ Introduced an EU wrapper in the front-end for automatic registration of 26 countries [#378](https://github.com/tobychui/zoraxy/issues/378)
|
||||||
|
+ Moved all hard-coded values to a dedicated def.go file
|
||||||
|
+ Fixed a panic issue occurring on unsupported platform exits
|
||||||
|
+ Integrated fixes for SSH proxy and Docker snippet updates [#330](https://github.com/tobychui/zoraxy/issues/330) [#348](https://github.com/tobychui/zoraxy/issues/348)
|
||||||
|
+ **Changed the default listening port to 443 and enable TLS by default**
|
||||||
|
+ Optimized GeoIP database slow-search mode CPU usage
|
||||||
|
|
||||||
|
|
||||||
|
# v3.1.3 12 Nov 2024
|
||||||
|
|
||||||
|
+ Fixed a critical security bug [CVE-2024-52010](https://github.com/advisories/GHSA-7hpf-g48v-hw3j)
|
||||||
|
|
||||||
|
# v3.1.2 03 Nov 2024
|
||||||
|
|
||||||
|
+ Added auto start port 80 listener on acme certificate generator
|
||||||
|
+ Added polling interval and propagation timeout option in ACME module [#300](https://github.com/tobychui/zoraxy/issues/300)
|
||||||
|
+ Added support for custom header variables [#318](https://github.com/tobychui/zoraxy/issues/318)
|
||||||
|
+ Added support for X-Remote-User
|
||||||
|
+ Added port scanner [#342](https://github.com/tobychui/zoraxy/issues/342)
|
||||||
|
+ Optimized code base for stream proxy and config file storage [#320](https://github.com/tobychui/zoraxy/issues/320)
|
||||||
|
+ Removed sorting on cert list
|
||||||
|
+ Fixed request certificate button bug
|
||||||
|
+ Fixed cert auto renew logic [#316](https://github.com/tobychui/zoraxy/issues/316)
|
||||||
|
+ Fixed unable to remove new stream proxy bug
|
||||||
|
+ Fixed many other minor bugs [#328](https://github.com/tobychui/zoraxy/issues/328) [#297](https://github.com/tobychui/zoraxy/issues/297)
|
||||||
|
+ Added more code to SSO system (disabled in release)
|
||||||
|
|
||||||
|
|
||||||
|
# v3.1.1. 09 Sep 2024
|
||||||
|
|
||||||
|
+ Updated country name in access list [#287](https://github.com/tobychui/zoraxy/issues/287)
|
||||||
|
+ Added tour for basic operations
|
||||||
|
+ Updated acme log to system wide logger implementation
|
||||||
|
+ Fixed path traversal in file manager [#274](https://github.com/tobychui/zoraxy/issues/274)
|
||||||
|
+ Removed Proxmox debug code
|
||||||
|
+ Fixed trie tree implementations
|
||||||
|
|
||||||
|
**Thanks to all contributors**
|
||||||
|
|
||||||
|
+ Fix existing containers list in docker popup [7brend7](https://github.com/tobychui/zoraxy/issues?q=is%3Apr+author%3A7brend7)
|
||||||
|
+ Fix network I/O chart not rendering [JokerQyou](https://github.com/tobychui/zoraxy/issues?q=is%3Apr+author%3AJokerQyou)
|
||||||
|
+ Fix typo remvoeClass to removeClass [Aahmadsyamim](https://github.com/tobychui/zoraxy/issues?q=is%3Apr+author%3Aahmadsyamim)
|
||||||
|
+ Updated weighted random upstream implementation [bouroo](https://github.com/tobychui/zoraxy/issues?q=is%3Apr+author%3Abouroo)
|
||||||
|
|
||||||
|
# v3.1.0 31 Jul 2024
|
||||||
|
|
||||||
|
+ Updated log viewer with filter and auto refresh [#243](https://github.com/tobychui/zoraxy/issues/243)
|
||||||
|
+ Fixed csrf vulnerability [#267](https://github.com/tobychui/zoraxy/issues/267)
|
||||||
|
+ Fixed promox issue
|
||||||
|
+ Fixed status code bug in upstream log [#254](https://github.com/tobychui/zoraxy/issues/254)
|
||||||
|
+ Added host overwrite and hop-by-hop header remover
|
||||||
|
+ Added early renew days settings [#256](https://github.com/tobychui/zoraxy/issues/256)
|
||||||
|
+ Updated make file to force no CGO in cicd process
|
||||||
|
+ Fixed bug in updater
|
||||||
|
+ Fixed wildcard certificate renew bug [#249](https://github.com/tobychui/zoraxy/issues/249)
|
||||||
|
+ Added certificate download function [#227](https://github.com/tobychui/zoraxy/issues/227)
|
||||||
|
|
||||||
|
# v3.0.9 16 Jul 2024
|
||||||
|
|
||||||
|
+ Added certificate download [#227](https://github.com/tobychui/zoraxy/issues/227)
|
||||||
|
+ Updated netcup timeout value [#231](https://github.com/tobychui/zoraxy/issues/231)
|
||||||
|
+ Updated geoip db
|
||||||
|
+ Removed debug print from log viewer
|
||||||
|
+ Upgraded netstat log printing to new log formatter
|
||||||
|
+ Improved update module implementation
|
||||||
|
|
||||||
|
# v3.0.8 15 Jul 2024
|
||||||
|
|
||||||
|
+ Added apache style logging mechanism (and build-in log viewer) [#218](https://github.com/tobychui/zoraxy/issues/218)
|
||||||
|
+ Fixed keep alive flushing issues [#235](https://github.com/tobychui/zoraxy/issues/235)
|
||||||
|
+ Added multi-upstream supports [#100](https://github.com/tobychui/zoraxy/issues/100)
|
||||||
|
+ Added stick session load balancer
|
||||||
|
+ Added weighted random load balancer
|
||||||
|
+ Added domain cleaning logic to domain / IP input fields
|
||||||
|
+ Added HSTS "include subdomain" auto injector
|
||||||
|
+ Added work-in-progress SSO / Oauth Server UI
|
||||||
|
+ Fixed uptime monitor not updating on proxy rule change bug
|
||||||
|
+ Optimized UI for create new proxy rule
|
||||||
|
+ Removed service expose proxy feature
|
||||||
|
|
||||||
|
# v3.0.7 20 Jun 2024
|
||||||
|
|
||||||
|
+ Fixed redirection enable bug [#199](https://github.com/tobychui/zoraxy/issues/199)
|
||||||
|
+ Fixed header tool user agent rewrite sequence
|
||||||
|
+ Optimized rate limit UI
|
||||||
|
+ Added HSTS and Permission Policy Editor [#163](https://github.com/tobychui/zoraxy/issues/163)
|
||||||
|
+ Docker UX optimization start parameter `-docker`
|
||||||
|
+ Docker container selector implementation for conditional compilations for Windows
|
||||||
|
|
||||||
|
From contributors:
|
||||||
|
|
||||||
|
+ Add Rate Limits Limits to Zoraxy fixes [185](https://github.com/tobychui/zoraxy/issues/185) by [Kirari04](https://github.com/Kirari04)
|
||||||
|
+ Add docker containers list to set rule by [7brend7](https://github.com/7brend7) [PR202](https://github.com/tobychui/zoraxy/pull/202)
|
||||||
|
|
||||||
|
# v3.0.6 10 Jun 2024
|
||||||
|
|
||||||
|
+ Added fastly_client_ip to X-Real-IP auto rewrite
|
||||||
|
+ Added atomic accumulator to TCP proxy
|
||||||
|
+ Added white logo for future dark theme
|
||||||
|
+ Added multi selection for white / blacklist [#176](https://github.com/tobychui/zoraxy/issues/176)
|
||||||
|
+ Moved custom header rewrite to dpcore
|
||||||
|
+ Restructure dpcore header rewrite sequence
|
||||||
|
+ Added advance custom header settings (zoraxy to upstream and zoraxy to downstream mode)
|
||||||
|
+ Added header remove feature
|
||||||
|
+ Removed password requirement for SMTP [#162](https://github.com/tobychui/zoraxy/issues/162) [#80](https://github.com/tobychui/zoraxy/issues/80)
|
||||||
|
+ Restructured TCP proxy into Stream Proxy (Support both TCP and UDP) [#147](https://github.com/tobychui/zoraxy/issues/147)
|
||||||
|
+ Added stream proxy auto start [#169](https://github.com/tobychui/zoraxy/issues/169)
|
||||||
|
+ Optimized UX for reminding user to click Apply after port change
|
||||||
|
+ Added version number to footer [#160](https://github.com/tobychui/zoraxy/issues/160)
|
||||||
|
|
||||||
|
From contributors:
|
||||||
|
|
||||||
|
+ Fixed missing / unnecessary error check [PR187](https://github.com/tobychui/zoraxy/pull/187) by [Kirari04](https://github.com/Kirari04)
|
||||||
|
|
||||||
|
# v3.0.5 May 26 2024
|
||||||
|
|
||||||
|
|
||||||
|
+ Optimized uptime monitor error message [#121](https://github.com/tobychui/zoraxy/issues/121)
|
||||||
|
+ Optimized detection logic for internal proxy target and header rewrite condition for HTTP_HOST [#164](https://github.com/tobychui/zoraxy/issues/164)
|
||||||
|
+ Fixed ovh DNS challenge provider form generator bug [#161](https://github.com/tobychui/zoraxy/issues/161)
|
||||||
|
+ Added permission policy module (not enabled)
|
||||||
|
+ Added single-use cookiejar to uptime monitor request client to handle cookie issues on some poorly written back-end server [#149](https://github.com/tobychui/zoraxy/issues/149)
|
||||||
|
|
||||||
|
|
||||||
|
# v3.0.4 May 18 2024
|
||||||
|
|
||||||
|
## This release tidied up the contribution by [Teifun2](https://github.com/Teifun2) and added a new way to generate DNS challenge based certificate (e.g. wildcards) from Let's Encrypt without changing any environment variables. This also fixes a few previous ACME module EAB settings bug related to concurrent save.
|
||||||
|
|
||||||
|
You can find the DNS challenge settings under TLS / SSL > ACME snippet > Generate New Certificate > (Check the "Use a DNS Challenge" checkbox)
|
||||||
|
|
||||||
|
+ Optimized DNS challenge implementation [thanks to Teifun2](https://github.com/Teifun2) / Issues [#49](https://github.com/tobychui/zoraxy/issues/49) [#79](https://github.com/tobychui/zoraxy/issues/79)
|
||||||
|
+ Removed dependencies on environment variable write and keep all data contained
|
||||||
|
+ Fixed panic on loading certificate generated by Zoraxy v2
|
||||||
|
+ Added automatic form generator for DNS challenge / providers
|
||||||
|
+ Added CA name default value
|
||||||
|
+ Added code generator for acmedns module (storing the DNS challenge provider contents extracted from lego)
|
||||||
|
+ Fixed ACME snippet "Obtain Certificate" concurrent issues in save EAB and DNS credentials
|
||||||
|
|
||||||
|
|
||||||
|
# v3.0.3 Apr 30 2024
|
||||||
|
## Breaking Change
|
||||||
|
|
||||||
|
For users using SMTP with older versions, you might need to update the settings by moving the domains (the part after @ in the username and domain setup field) into the username field.
|
||||||
|
|
||||||
|
+ Updated SMTP UI for non email login username [#129](https://github.com/tobychui/zoraxy/issues/129)
|
||||||
|
+ Fixed ACME cert store reload after cert request [#126](https://github.com/tobychui/zoraxy/issues/126)
|
||||||
|
+ Fixed default rule not applying to default site when default site is set to proxy target [#130](https://github.com/tobychui/zoraxy/issues/130)
|
||||||
|
+ Fixed blacklist-ip not working with CIDR bug
|
||||||
|
+ Fixed minor vdir bug in tailing slash detection and redirect logic
|
||||||
|
+ Added custom mdns name support (-mdnsname flag)
|
||||||
|
+ Added LAN tag in statistic [#131](https://github.com/tobychui/zoraxy/issues/131)
|
||||||
|
|
||||||
|
|
||||||
|
# v3.0.2 Apr 24 2024
|
||||||
|
|
||||||
|
+ Added alias for HTTP proxy host names [#76](https://github.com/tobychui/zoraxy/issues/76)
|
||||||
|
+ Added separator support for create new proxy rules (use "," to add alias when creating new proxy rule)
|
||||||
|
+ Added HTTP proxy host based access rules [#69](https://github.com/tobychui/zoraxy/issues/69)
|
||||||
|
+ Added EAD Configuration for ACME (by [yeungalan](https://github.com/yeungalan)) [#45](https://github.com/tobychui/zoraxy/issues/45)
|
||||||
|
+ Fixed bug for bypassGlobalTLS endpoint do not support basic-auth
|
||||||
|
+ Fixed panic due to empty domain field in json config [#120](https://github.com/tobychui/zoraxy/issues/120)
|
||||||
|
+ Removed dependencies on management panel css for online font files
|
||||||
|
|
||||||
|
# v3.0.1 Apr 04 2024
|
||||||
|
|
||||||
|
## Bugfixupdate for big release of V3, read update notes from V3 if you are still on V2
|
||||||
|
|
||||||
|
+ Added regex support for redirect (slow, don't use it unless you really needs it) [#42](https://github.com/tobychui/zoraxy/issues/42)
|
||||||
|
+ Added new dpcore implementations for faster proxy speed
|
||||||
|
+ Added support for CF-Connecting-IP to X-Real-IP auto rewrite [#114](https://github.com/tobychui/zoraxy/issues/114)
|
||||||
|
+ Added enable / disable of HTTP proxy rules in runtime via slider [#108](https://github.com/tobychui/zoraxy/issues/108)
|
||||||
|
+ Added better 404 page
|
||||||
|
+ Added option to bypass websocket origin check [#107](https://github.com/tobychui/zoraxy/issues/107)
|
||||||
|
+ Updated project homepage design
|
||||||
|
+ Fixed recursive port detection logic
|
||||||
|
+ Fixed UserAgent in resp bug
|
||||||
|
+ Updated minimum required Go version to v1.22 (Notes: Windows 7 support is dropped) [#112](https://github.com/tobychui/zoraxy/issues/112)
|
||||||
|
|
||||||
|
|
||||||
|
# v3.0.0 Feb 18 2024
|
||||||
|
|
||||||
|
## IMPORTANT: V3 is a big rewrite and it is incompatible with V2! There is NO migration, if you want to stay on V2, please use V2 branch!
|
||||||
|
|
||||||
|
+ Added comments for whitelist [#97](https://github.com/tobychui/zoraxy/issues/97)
|
||||||
|
+ Added force-renew for certificates [#92](https://github.com/tobychui/zoraxy/issues/92)
|
||||||
|
+ Added automatic cert pick for multi-host certs (SNI)
|
||||||
|
+ Renamed .crt to .pem for cert store
|
||||||
|
+ Added best-fit selection for wildcard matching rules
|
||||||
|
+ Added x-proxy-by header / Added X-real-Ip header [#93](https://github.com/tobychui/zoraxy/issues/93)
|
||||||
|
+ Added Development Mode (Cache-Control: no-store)
|
||||||
|
+ Updated utm timeout to 10 seconds instead of 90
|
||||||
|
+ Added "Add controller as member" feature to Global Area Network editor
|
||||||
|
+ Added custom header
|
||||||
|
+ Deprecated aroz subservice support
|
||||||
|
+ Updated visuals, improving logical structure, less depressing colors [#95](https://github.com/tobychui/zoraxy/issues/95)
|
||||||
|
+ Added virtual directory into host routing object (each host now got its own sets of virtual directories)
|
||||||
|
+ Added support for wildcard host names (e.g. *.example.com)
|
||||||
|
+ Added best-fit selection for wildcard matching rules (e.g. *.a.example.com > *.example.com in routing)
|
||||||
|
+ Generalized root and hosts routing struct (no more conversion between runtime & save record object
|
||||||
|
+ Added "Default Site" to replace "Proxy Root" interface
|
||||||
|
+ Added Redirect & 404 page for "Default Site"
|
||||||
|
|
||||||
|
|
||||||
|
# v2.6.8 Nov 25 2023
|
||||||
|
|
||||||
|
+ Added opt-out for subdomains for global TLS settings: See [release notes](https://github.com/tobychui/zoraxy/releases/tag/2.6.8)
|
||||||
|
+ Optimized subdomain / vdir editing interface
|
||||||
|
+ Added system-wide logger (Work in progress)
|
||||||
|
+ Fixed issue for uptime monitor bug [#77](https://github.com/tobychui/zoraxy/issues/77)
|
||||||
|
+ Changed default static web port to 5487 (prevent already in use)
|
||||||
|
+ Added automatic HTTP/2 to TLS mode
|
||||||
|
+ Bug fix for webserver autostart [67](https://github.com/tobychui/zoraxy/issues/67)
|
||||||
|
|
||||||
|
# v2.6.7 Sep 26 2023
|
||||||
|
|
||||||
|
+ Added Static Web Server function [#56](https://github.com/tobychui/zoraxy/issues/56)
|
||||||
|
+ Web Directory Manager (see static webserver tab)
|
||||||
|
+ Added static web server and black / whitelist template [#38](https://github.com/tobychui/zoraxy/issues/38)
|
||||||
|
+ Added default / preferred CA features for ACME [#47](https://github.com/tobychui/zoraxy/issues/47)
|
||||||
|
+ Optimized TLS/SSL page and added dedicated section for ACME related features
|
||||||
|
+ Bugfixes [#61](https://github.com/tobychui/zoraxy/issues/61) [#58](https://github.com/tobychui/zoraxy/issues/58)
|
||||||
|
|
||||||
|
# v2.6.6 Aug 30 2023
|
||||||
|
|
||||||
|
+ Added basic auth editor custom exception rules
|
||||||
|
+ Fixed redirection bug under another reverse proxy and Apache location headers [#39](https://github.com/tobychui/zoraxy/issues/39)
|
||||||
|
+ Optimized memory usage (from 1.2GB to 61MB for low speed geoip lookup) [#52](https://github.com/tobychui/zoraxy/issues/52)
|
||||||
|
+ Added unset subdomain custom redirection feature [#46](https://github.com/tobychui/zoraxy/issues/46)
|
||||||
|
+ Fixed potential security issue in satori/go.uuid [#55](https://github.com/tobychui/zoraxy/issues/55)
|
||||||
|
+ Added custom ACME feature in backend, thx [@daluntw](https://github.com/daluntw)
|
||||||
|
+ Added bypass TLS check for custom acme server, thx [@daluntw](https://github.com/daluntw)
|
||||||
|
+ Introduce new start parameter `-fastgeoip=true`: see [release notes](https://github.com/tobychui/zoraxy/releases/tag/2.6.6)
|
||||||
|
|
||||||
|
# v2.6.5.1 Jul 26 2023
|
||||||
|
|
||||||
|
+ Patch on memory leaking for Windows netstat module (do not effect any of the previous non Windows builds)
|
||||||
|
+ Fixed potential memory leak in ACME handler logic
|
||||||
|
+ Added "Do you want to get a TLS certificate for this subdomain?" dialogue when a new subdomain proxy rule is created
|
||||||
|
|
||||||
|
# v2.6.5 Jul 19 2023
|
||||||
|
|
||||||
|
+ Added Import / Export-Feature
|
||||||
|
+ Moved configuration files to a separate folder [#26](https://github.com/tobychui/zoraxy/issues/26)
|
||||||
|
+ Added auto-renew with ACME [#6](https://github.com/tobychui/zoraxy/issues/6)
|
||||||
|
+ Fixed Whitelistbug [#18](https://github.com/tobychui/zoraxy/issues/18)
|
||||||
|
+ Added Whois
|
||||||
|
|
||||||
|
# v2.6.4 Jun 15 2023
|
||||||
|
|
||||||
|
+ Added force TLS v1.2 above toggle
|
||||||
|
+ Added trace route
|
||||||
|
+ Added ICMP ping
|
||||||
|
+ Added special routing rules module for up-coming ACME integration
|
||||||
|
+ Fixed IPv6 check bug in black/whitelist
|
||||||
|
+ Optimized UI for TCP Proxy
|
||||||
|
|
||||||
# v2.6.3 Jun 8 2023
|
# v2.6.3 Jun 8 2023
|
||||||
|
|
||||||
+ Added X-Forwarded-Proto for automatic proxy detector
|
+ Added X-Forwarded-Proto for automatic proxy detector
|
||||||
+ Split blacklist and whitelist from geodb script file
|
+ Split blacklist and whitelist from geodb script file
|
||||||
+ Optimized compile binary size
|
+ Optimized compile binary size
|
||||||
+ Added access control to TCP proxy
|
+ Added access control to TCP proxy
|
||||||
+ Added "invalid config detect" in up time monitor for isse #7
|
+ Added "invalid config detect" in up time monitor for issue [#7](https://github.com/tobychui/zoraxy/issues/7)
|
||||||
+ Fixed minor bugs in advance stats panel
|
+ Fixed minor bugs in advance stats panel
|
||||||
+ Reduced file size of embedded materials
|
+ Reduced file size of embedded materials
|
||||||
|
|
||||||
# v2.6.2 Jun 4 2023
|
# v2.6.2 Jun 4 2023
|
||||||
|
|
||||||
+ Added advance stats operation tab
|
+ Added advance stats operation tab
|
||||||
+ Added statistic reset #13
|
+ Added statistic reset [#13](https://github.com/tobychui/zoraxy/issues/13)
|
||||||
+ Added statistic export to csv and json (please use json)
|
+ Added statistic export to csv and json (please use json)
|
||||||
+ Make subdomain clickable (not vdir) #12
|
+ Make subdomain clickable (not vdir) [#12](https://github.com/tobychui/zoraxy/issues/12)
|
||||||
+ Added TCP Proxy
|
+ Added TCP Proxy
|
||||||
+ Updates SMTP setup UI to make it more straight forward to setup
|
+ Updates SMTP setup UI to make it more straight forward to setup
|
||||||
|
|
||||||
@@ -31,6 +379,6 @@
|
|||||||
+ Basic auth
|
+ Basic auth
|
||||||
+ Support TLS verification skip (for self signed certs)
|
+ Support TLS verification skip (for self signed certs)
|
||||||
+ Added trend analysis
|
+ Added trend analysis
|
||||||
+ Added referer and file type analysis
|
+ Added referrer and file type analysis
|
||||||
+ Added cert expire day display
|
+ Added cert expire day display
|
||||||
+ Moved subdomain proxy logic to dpcore
|
+ Moved subdomain proxy logic to dpcore
|
||||||
|
203
README.md
@@ -2,36 +2,61 @@
|
|||||||
|
|
||||||
# Zoraxy
|
# Zoraxy
|
||||||
|
|
||||||
General purpose request (reverse) proxy and forwarding tool for low power devices. Now written in Go!
|
A general purpose HTTP reverse proxy and forwarding tool. Now written in Go!
|
||||||
|
|
||||||
### Features
|
### Features
|
||||||
|
|
||||||
- Simple to use interface with detail in-system instructions
|
- Simple to use interface with detail in-system instructions
|
||||||
- Reverse Proxy
|
- Reverse Proxy (HTTP/2)
|
||||||
|
- Virtual Directory
|
||||||
- Subdomain Reverse Proxy
|
- WebSocket Proxy (automatic, no set-up needed)
|
||||||
|
- Basic Auth
|
||||||
- Virtual Directory Reverse Proxy
|
- Alias Hostnames
|
||||||
|
- Custom Headers
|
||||||
|
- Load Balancing
|
||||||
- Redirection Rules
|
- Redirection Rules
|
||||||
- TLS / SSL setup and deploy
|
- TLS / SSL setup and deploy
|
||||||
- Blacklist by country or IP address (single IP, CIDR or wildcard for beginners)
|
- ACME features like auto-renew to serve your sites in http**s**
|
||||||
- Global Area Network Controller Web UI (ZeroTier not included)
|
- SNI support (and SAN certs)
|
||||||
|
- DNS Challenge for Let's Encrypt and [these DNS providers](https://go-acme.github.io/lego/dns/)
|
||||||
|
- Blacklist / Whitelist by country or IP address (single IP, CIDR or wildcard for beginners)
|
||||||
|
- Stream Proxy (TCP & UDP)
|
||||||
- Integrated Up-time Monitor
|
- Integrated Up-time Monitor
|
||||||
- Web-SSH Terminal
|
- Web-SSH Terminal
|
||||||
|
- Plugin System
|
||||||
- Utilities
|
- Utilities
|
||||||
|
|
||||||
- CIDR IP converters
|
- CIDR IP converters
|
||||||
- mDNS Scanner
|
- mDNS Scanner
|
||||||
|
- Wake-On-Lan
|
||||||
|
- Debug Forward Proxy
|
||||||
- IP Scanner
|
- IP Scanner
|
||||||
|
- Port Scanner
|
||||||
- Others
|
- Others
|
||||||
- Basic single-admin management mode
|
- Basic single-admin management mode
|
||||||
- External permission management system for easy system integration
|
- External permission management system for easy system integration
|
||||||
- SMTP config for password reset
|
- SMTP config for password reset
|
||||||
|
- Dark Theme Mode
|
||||||
|
|
||||||
|
## Downloads
|
||||||
|
|
||||||
|
[Windows](https://github.com/tobychui/zoraxy/releases/latest/download/zoraxy_windows_amd64.exe)
|
||||||
|
/ [Linux (amd64)](https://github.com/tobychui/zoraxy/releases/latest/download/zoraxy_linux_amd64)
|
||||||
|
/ [Linux (arm64)](https://github.com/tobychui/zoraxy/releases/latest/download/zoraxy_linux_arm64)
|
||||||
|
|
||||||
|
For other systems or architectures, please see [Releases](https://github.com/tobychui/zoraxy/releases/latest/)
|
||||||
|
|
||||||
|
## Getting Started
|
||||||
|
|
||||||
|
[Installing Zoraxy Reverse Proxy: Your Gateway to Efficient Web Routing](https://geekscircuit.com/installing-zoraxy-reverse-proxy-your-gateway-to-efficient-web-routing/)
|
||||||
|
|
||||||
|
Thank you for the well written and easy to follow tutorial by Reddit user [itsvmn](https://www.reddit.com/user/itsvmn/)!
|
||||||
|
If you have no background in setting up reverse proxy or web routing, you should check this out before you start setting up your Zoraxy.
|
||||||
|
|
||||||
## Build from Source
|
## Build from Source
|
||||||
Require Go 1.20 or above
|
|
||||||
|
|
||||||
```
|
Requires Go 1.23 or higher
|
||||||
|
|
||||||
|
```bash
|
||||||
git clone https://github.com/tobychui/zoraxy
|
git clone https://github.com/tobychui/zoraxy
|
||||||
cd ./zoraxy/src/
|
cd ./zoraxy/src/
|
||||||
go mod tidy
|
go mod tidy
|
||||||
@@ -42,11 +67,11 @@ sudo ./zoraxy -port=:8000
|
|||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
Zoraxy provide basic authentication system for standalone mode. To use it in standalone mode, follow the instruction below for your desired deployment platform.
|
Zoraxy provides basic authentication system for standalone mode. To use it in standalone mode, follow the instructions below for your desired deployment platform.
|
||||||
|
|
||||||
### Standalone Mode
|
### Standalone Mode
|
||||||
|
|
||||||
Standalone mode is the default mode for Zoraxy. This allow single account to manage your reverse proxy server just like a home router. This mode is suitable for new owners for homelab or makers start growing their web services into multiple servers.
|
Standalone mode is the default mode for Zoraxy. This allows a single account to manage your reverse proxy server just like a basic home router. This mode is suitable for new owners to homelabs or makers starting growing their web services into multiple servers. A full "Getting Started" guide can be found [here](https://github.com/tobychui/zoraxy/wiki/Getting-Started).
|
||||||
|
|
||||||
#### Linux
|
#### Linux
|
||||||
|
|
||||||
@@ -60,68 +85,81 @@ Download the binary executable and double click the binary file to start it.
|
|||||||
|
|
||||||
#### Raspberry Pi
|
#### Raspberry Pi
|
||||||
|
|
||||||
The installation method is same as Linux. If you are using Raspberry Pi 4 or newer models, pick the arm64 release. For older version of Pis, use the arm (armv6) version instead.
|
The installation method is same as Linux. If you are using a Raspberry Pi 4 or newer models, pick the arm64 release. For older version of Pis, use the arm (armv6) version instead.
|
||||||
|
|
||||||
#### Other ARM SBCs or Android phone with Termux
|
#### Other ARM SBCs or Android phone with Termux
|
||||||
|
|
||||||
The installation method is same as Linux. For other ARM SBCs, please refer to your SBC's CPU architecture and pick the one that is suitable for your device.
|
The installation method is same as Linux. For other ARM SBCs, please refer to your SBC's CPU architecture and pick the one that is suitable for your device.
|
||||||
|
|
||||||
#### Docker
|
#### Docker
|
||||||
Thanks for cyb3rdoc and PassiveLemon for providing support over the Docker installation. You can check out their repo over here.
|
|
||||||
|
|
||||||
[https://github.com/cyb3rdoc/zoraxy-docker](https://github.com/cyb3rdoc/zoraxy-docker)
|
See the [/docker](https://github.com/tobychui/zoraxy/tree/main/docker) folder for more details.
|
||||||
|
|
||||||
[https://github.com/PassiveLemon/zoraxy-docker](https://github.com/PassiveLemon/zoraxy-docker)
|
### Start Parameters
|
||||||
|
|
||||||
|
```
|
||||||
|
Usage of zoraxy:
|
||||||
|
-autorenew int
|
||||||
|
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)
|
||||||
|
-dev
|
||||||
|
Use external web folder for UI development
|
||||||
|
-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 change in start paramters (default "./www")
|
||||||
|
```
|
||||||
|
|
||||||
### External Permission Management Mode
|
### External Permission Management Mode
|
||||||
|
|
||||||
If you already have a up-stream reverse proxy server in place with permission management, you can use Zoraxy in noauth mode. To enable noauth mode, start Zoraxy with the following flag
|
If you already have an upstream reverse proxy server in place with permission management, you can use Zoraxy in noauth mode. To enable noauth mode, start Zoraxy with the following flag:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
./zoraxy -noauth=true
|
./zoraxy -noauth=true
|
||||||
```
|
```
|
||||||
|
|
||||||
*Note: For security reaons, you should only enable no-auth if you are running Zoraxy in a trusted environment or with another authentication management proxy in front.*
|
> [!WARNING]
|
||||||
|
> For security reasons, you should only enable no-auth if you are running Zoraxy in a trusted environment or with another authentication management proxy in front.*
|
||||||
#### Use with ArozOS
|
|
||||||
|
|
||||||
[ArozOS ](https://arozos.com)subservice is a build in permission managed reverse proxy server. To use zoraxy with arozos, connect to your arozos host via ssh and use the following command to install zoraxy
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# cd into your arozos subservice folder. Sometime it is under ~/arozos/src/subservice
|
|
||||||
cd ~/arozos/subservices
|
|
||||||
mkdir zoraxy
|
|
||||||
cd ./zoraxy
|
|
||||||
|
|
||||||
# Download the release binary from Github release
|
|
||||||
wget {binary executable link from release page}
|
|
||||||
|
|
||||||
# Set permission. Change this if required
|
|
||||||
sudo chmod 775 -R ./
|
|
||||||
|
|
||||||
# Start zoraxy to see if the downloaded arch is correct.
|
|
||||||
./zoraxy
|
|
||||||
|
|
||||||
# After the unzip done, press Ctrl + C to kill it
|
|
||||||
# Rename it to valid arozos subservice binary format
|
|
||||||
mv ./zoraxy zoraxy_linux_amd64
|
|
||||||
|
|
||||||
# If you are using SBCs with different CPU arch, use the following names
|
|
||||||
# mv ./zoraxy zoraxy_linux_arm
|
|
||||||
# mv ./zoraxy zoraxy_linux_arm64
|
|
||||||
|
|
||||||
# Restart arozos
|
|
||||||
sudo systemctl restart arozos
|
|
||||||
```
|
|
||||||
|
|
||||||
To start the module, go to System Settings > Modules > Subservice and enable it in the menu. You should be able to see a new module named "Zoraxy" pop up in the start menu.
|
|
||||||
|
|
||||||
## Screenshots
|
## Screenshots
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
More screenshots on the wikipage [Screenshots](https://github.com/tobychui/zoraxy/wiki/Screenshots)!
|
More screenshots on the wikipage [Screenshots](https://github.com/tobychui/zoraxy/wiki/Screenshots)!
|
||||||
|
|
||||||
## FAQ
|
## FAQ
|
||||||
@@ -130,38 +168,57 @@ There is a wikipage with [Frequently-Asked-Questions](https://github.com/tobychu
|
|||||||
|
|
||||||
## Global Area Network Controller
|
## Global Area Network Controller
|
||||||
|
|
||||||
This project also compatible with [ZeroTier](https://www.zerotier.com/). However, due to licensing issues, ZeroTier is not included in the binary.
|
Moved to official plugin repo, see [ztnc](https://github.com/aroz-online/zoraxy-official-plugins/tree/main/src/ztnc) plugin
|
||||||
|
|
||||||
Assuming you already have a valid license, to use Zoraxy with ZeroTier, install ZeroTier on your host and then run Zoraxy in sudo mode (or Run As Administrator if you are on Windows). The program will automatically grab the authtoken at correct location in your host.
|
## Web SSH
|
||||||
|
|
||||||
If you prefer not to run Zoraxy in sudo mode or you have some weird installation profile, you can also pass in the ZeroTier auth token using the following flags
|
Web SSH currently only supports Linux based OSes. The following platforms are supported:
|
||||||
|
|
||||||
```
|
|
||||||
./zoraxy -ztauth="your_zerotier_authtoken" -ztport=9993
|
|
||||||
```
|
|
||||||
|
|
||||||
The ZeroTier auth token can usually be found at ```/var/lib/zerotier-one/authtoken.secret``` or ```C:\ProgramData\ZeroTier\One\authtoken.secret```.
|
|
||||||
|
|
||||||
This allows you to have infinite number of network members in your Global Area Network controller. For more technical details, see [here](https://docs.zerotier.com/self-hosting/network-controllers/).
|
|
||||||
|
|
||||||
## Web.SSH
|
|
||||||
|
|
||||||
Web SSH currently only support Linux based OS. The following platforms are supported
|
|
||||||
|
|
||||||
- linux/amd64
|
- linux/amd64
|
||||||
- linux/arm64
|
- linux/arm64
|
||||||
- linux/armv6 (experimental)
|
- linux/armv6 (experimental)
|
||||||
- linux/386 (experimental)
|
- linux/386 (experimental)
|
||||||
|
|
||||||
### Loopback Connection
|
### Loopback Connection
|
||||||
|
|
||||||
Loopback web ssh connection, by default, is disabled. This means that if you are trying to connect to address like 127.0.0.1 or localhost, the system will reject your connection due to security issues. To enable loopback for testing or development purpose, use the following flags to override the loopback checking.
|
Loopback web SSH connections, by default, are disabled. This means that if you are trying to connect to an address like 127.0.0.1 or localhost, the system will reject your connection for security reasons. To enable loopback for testing or development purpose, use the following flags to override the loopback checking:
|
||||||
|
|
||||||
```
|
```bash
|
||||||
./zoraxy -sshlb=true
|
./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.
|
||||||
|
|
||||||
|
- Forward Auth [@james-d-elliott](https://github.com/james-d-elliott)
|
||||||
|
|
||||||
|
- (Legacy) Authelia Support added by [@7brend7](https://github.com/7brend7)
|
||||||
|
|
||||||
|
- (Legacy) Authentik Support added by [@JokerQyou](https://github.com/JokerQyou)
|
||||||
|
|
||||||
|
|
||||||
|
- Docker Container List by [@eyerrock](https://github.com/eyerrock)
|
||||||
|
|
||||||
|
- Stream Proxy [@jemmy1794](https://github.com/jemmy1794)
|
||||||
|
|
||||||
|
- Change Log [@Morethanevil](https://github.com/Morethanevil)
|
||||||
|
|
||||||
|
### Looking for Maintainer
|
||||||
|
|
||||||
|
- ACME DNS Challenge Module
|
||||||
|
- Logging (including analysis & attack prevention) Module
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
- [tobychui (Primary author)](https://paypal.me/tobychui)
|
||||||
|
- [PassiveLemon (Docker compatibility maintainer)](https://github.com/PassiveLemon)
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
This project is open source under AGPL. I open source this project so everyone can check for security issues and benefit all users. **If your plans to use this project in commercial environment which violate the AGPL terms, please contact toby@imuslab.com for an alternative commercial license.**
|
This project is open-sourced under AGPL. I open-sourced this project so everyone can check for security issues and benefit all users. **This software is intended to be free of charge. If you have acquired this software from a third-party seller, the authors of this repository bears no responsibility for any technical difficulties assistance or support.**
|
||||||
|
|
||||||
|
95
docker/Dockerfile
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
## Build Zoraxy
|
||||||
|
FROM docker.io/golang:alpine AS build-zoraxy
|
||||||
|
|
||||||
|
RUN mkdir -p /opt/zoraxy/source/ &&\
|
||||||
|
mkdir -p /usr/local/bin/
|
||||||
|
|
||||||
|
# If you build it yourself, you will need to add the src directory into the docker directory.
|
||||||
|
COPY ./src/ /opt/zoraxy/source/
|
||||||
|
|
||||||
|
WORKDIR /opt/zoraxy/source/
|
||||||
|
|
||||||
|
RUN go mod tidy &&\
|
||||||
|
go build -o /usr/local/bin/zoraxy &&\
|
||||||
|
chmod 755 /usr/local/bin/zoraxy
|
||||||
|
|
||||||
|
|
||||||
|
## 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 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-*/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
|
||||||
|
|
||||||
|
|
||||||
|
## 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 --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/
|
||||||
|
|
||||||
|
ENV ZEROTIER="false"
|
||||||
|
|
||||||
|
ENV AUTORENEW="86400"
|
||||||
|
ENV CFGUPGRADE="true"
|
||||||
|
ENV DB="auto"
|
||||||
|
ENV DOCKER="true"
|
||||||
|
ENV EARLYRENEW="30"
|
||||||
|
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"
|
||||||
|
|
||||||
|
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
|
||||||
|
|
137
docker/README.md
Normal file
@@ -0,0 +1,137 @@
|
|||||||
|
# Zoraxy Docker
|
||||||
|
|
||||||
|
[](https://hub.docker.com/r/zoraxydocker/zoraxy)
|
||||||
|
[](https://hub.docker.com/r/zoraxydocker/zoraxy)
|
||||||
|
[](https://hub.docker.com/r/zoraxydocker/zoraxy)
|
||||||
|
[](https://hub.docker.com/r/zoraxydocker/zoraxy)
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
If you are attempting to access your service from outside your network, make sure to forward ports 80 and 443 to the Zoraxy host to allow web traffic. If you know how to do this, great! If not, find the manufacturer of your router and search on how to do that. There are too many to be listed here. Read more about it from [whatismyip](https://www.whatismyip.com/port-forwarding/).
|
||||||
|
|
||||||
|
In the examples below, make sure to update `/path/to/zoraxy/config/`. If a path is not provided, a Docker volume will be created at the location but it is recommended to store the data at a defined host location or a named Docker volume.
|
||||||
|
|
||||||
|
Once setup, access the webui at `http://<host-ip>:8000` to configure Zoraxy. Change the port in the URL if you changed the management port.
|
||||||
|
|
||||||
|
### Docker Run
|
||||||
|
|
||||||
|
```
|
||||||
|
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" \
|
||||||
|
zoraxydocker/zoraxy:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
### Docker Compose
|
||||||
|
|
||||||
|
```yml
|
||||||
|
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"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Ports
|
||||||
|
|
||||||
|
| Port | Details |
|
||||||
|
|:-|:-|
|
||||||
|
| `80` | HTTP traffic. |
|
||||||
|
| `443` | HTTPS traffic. |
|
||||||
|
| `8000` | Management interface. Can be changed with the `PORT` env. |
|
||||||
|
|
||||||
|
### Volumes
|
||||||
|
|
||||||
|
| 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).
|
||||||
|
|
||||||
|
| Variable | Default | Details |
|
||||||
|
|:-|:-|:-|
|
||||||
|
| `AUTORENEW` | `86400` (Integer) | ACME auto TLS/SSL certificate renew check interval. |
|
||||||
|
| `CFGUPGRADE` | `true` (Boolean) | Enable auto config upgrade if breaking change is detected. |
|
||||||
|
| `DB` | `auto` (String) | Database backend to use (leveldb, boltdb, auto) Note that fsdb will be used on unsupported platforms like RISCV (default "auto"). |
|
||||||
|
| `DOCKER` | `true` (Boolean) | Run Zoraxy in docker compatibility mode. |
|
||||||
|
| `EARLYRENEW` | `30` (Integer) | Number of days to early renew a soon expiring certificate. |
|
||||||
|
| `FASTGEOIP` | `false` (Boolean) | Enable high speed geoip lookup, require 1GB extra memory (Not recommend for low end devices). |
|
||||||
|
| `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. |
|
||||||
|
| `VERSION` | `false` (Boolean) | Show version of this server. |
|
||||||
|
| `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. |
|
||||||
|
|
||||||
|
> [!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/` 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
@@ -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
@@ -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"
|
55
docker/entrypoint.sh
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
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
|
||||||
|
}
|
||||||
|
|
||||||
|
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 &
|
||||||
|
zerotierpid=$!
|
||||||
|
echo "ZeroTier daemon started."
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Starting Zoraxy..."
|
||||||
|
zoraxy \
|
||||||
|
-autorenew="$AUTORENEW" \
|
||||||
|
-cfgupgrade="$CFGUPGRADE" \
|
||||||
|
-db="$DB" \
|
||||||
|
-docker="$DOCKER" \
|
||||||
|
-earlyrenew="$EARLYRENEW" \
|
||||||
|
-fastgeoip="$FASTGEOIP" \
|
||||||
|
-mdns="$MDNS" \
|
||||||
|
-mdnsname="$MDNSNAME" \
|
||||||
|
-noauth="$NOAUTH" \
|
||||||
|
-plugin="$PLUGIN" \
|
||||||
|
-port=:"$PORT" \
|
||||||
|
-sshlb="$SSHLB" \
|
||||||
|
-update_geoip="$UPDATE_GEOIP" \
|
||||||
|
-version="$VERSION" \
|
||||||
|
-webfm="$WEBFM" \
|
||||||
|
-webroot="$WEBROOT" \
|
||||||
|
&
|
||||||
|
|
||||||
|
zoraxypid=$!
|
||||||
|
wait "$zoraxypid"
|
||||||
|
wait "$zerotierpid"
|
||||||
|
|
@@ -1 +1 @@
|
|||||||
zoraxy.arozos.com
|
zoraxy.aroz.org
|
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
@@ -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/bg.png
Before Width: | Height: | Size: 4.5 MiB |
BIN
docs/img/bg2.png
Before Width: | Height: | Size: 9.4 MiB |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="m772-635-43-100-104-46 104-45 43-95 43 95 104 45-104 46-43 100Zm0 595-43-96-104-45 104-45 43-101 43 101 104 45-104 45-43 96ZM333-194l-92-197-201-90 201-90 92-196 93 196 200 90-200 90-93 197Zm0-148 48-96 98-43-98-43-48-96-47 96-99 43 99 43 47 96Zm0-139Z"/></svg>
|
<svg class="item-icon" xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="m772-635-43-100-104-46 104-45 43-95 43 95 104 45-104 46-43 100Zm0 595-43-96-104-45 104-45 43-101 43 101 104 45-104 45-43 96ZM333-194l-92-197-201-90 201-90 92-196 93 196 200 90-200 90-93 197Zm0-148 48-96 98-43-98-43-48-96-47 96-99 43 99 43 47 96Zm0-139Z"/></svg>
|
Before Width: | Height: | Size: 358 B After Width: | Height: | Size: 377 B |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M280-453h400v-60H280v60ZM480-80q-82 0-155-31.5t-127.5-86Q143-252 111.5-325T80-480q0-83 31.5-156t86-127Q252-817 325-848.5T480-880q83 0 156 31.5T763-763q54 54 85.5 127T880-480q0 82-31.5 155T763-197.5q-54 54.5-127 86T480-80Zm0-60q142 0 241-99.5T820-480q0-142-99-241t-241-99q-141 0-240.5 99T140-480q0 141 99.5 240.5T480-140Zm0-340Z"/></svg>
|
<svg fill="#ff7a7a" xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M280-453h400v-60H280v60ZM480-80q-82 0-155-31.5t-127.5-86Q143-252 111.5-325T80-480q0-83 31.5-156t86-127Q252-817 325-848.5T480-880q83 0 156 31.5T763-763q54 54 85.5 127T880-480q0 82-31.5 155T763-197.5q-54 54.5-127 86T480-80Zm0-60q142 0 241-99.5T820-480q0-142-99-241t-241-99q-141 0-240.5 99T140-480q0 141 99.5 240.5T480-140Zm0-340Z"/></svg>
|
Before Width: | Height: | Size: 433 B After Width: | Height: | Size: 448 B |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M320-242 80-482l242-242 43 43-199 199 197 197-43 43Zm318 2-43-43 199-199-197-197 43-43 240 240-242 242Z"/></svg>
|
<svg class="item-icon" xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M320-242 80-482l242-242 43 43-199 199 197 197-43 43Zm318 2-43-43 199-199-197-197 43-43 240 240-242 242Z"/></svg>
|
Before Width: | Height: | Size: 209 B After Width: | Height: | Size: 227 B |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M120-80v-270h120v-160h210v-100H330v-270h300v270H510v100h210v160h120v270H540v-270h120v-100H300v100h120v270H120Zm270-590h180v-150H390v150ZM180-140h180v-150H180v150Zm420 0h180v-150H600v150ZM480-670ZM360-290Zm240 0Z"/></svg>
|
<svg fill="#919191" xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M120-80v-270h120v-160h210v-100H330v-270h300v270H510v100h210v160h120v270H540v-270h120v-100H300v100h120v270H120Zm270-590h180v-150H390v150ZM180-140h180v-150H180v150Zm420 0h180v-150H600v150ZM480-670ZM360-290Zm240 0Z"/></svg>
|
Before Width: | Height: | Size: 317 B After Width: | Height: | Size: 332 B |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M220-180h150v-250h220v250h150v-390L480-765 220-570v390Zm-60 60v-480l320-240 320 240v480H530v-250H430v250H160Zm320-353Z"/></svg>
|
<svg class="item-icon" xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M220-180h150v-250h220v250h150v-390L480-765 220-570v390Zm-60 60v-480l320-240 320 240v480H530v-250H430v250H160Zm320-353Z"/></svg>
|
Before Width: | Height: | Size: 224 B After Width: | Height: | Size: 242 B |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M356-120H180q-24 0-42-18t-18-42v-176q44-5 75.5-34.5T227-463q0-43-31.5-72.5T120-570v-176q0-24 18-42t42-18h177q11-40 39.5-67t68.5-27q40 0 68.5 27t39.5 67h173q24 0 42 18t18 42v173q40 11 65.5 41.5T897-461q0 40-25.5 67T806-356v176q0 24-18 42t-42 18H570q-5-48-35.5-77.5T463-227q-41 0-71.5 29.5T356-120Zm-176-60h130q25-61 69.888-84 44.888-23 83-23T546-264q45 23 70 84h130v-235h45q20 0 33-13t13-33q0-20-13-33t-33-13h-45v-239H511v-48q0-20-13-33t-33-13q-20 0-33 13t-13 33v48H180v130q48.15 17.817 77.575 59.686Q287-514.445 287-462.777 287-412 257.5-370T180-310v130Zm329-330Z"/></svg>
|
<svg class="item-icon" xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M356-120H180q-24 0-42-18t-18-42v-176q44-5 75.5-34.5T227-463q0-43-31.5-72.5T120-570v-176q0-24 18-42t42-18h177q11-40 39.5-67t68.5-27q40 0 68.5 27t39.5 67h173q24 0 42 18t18 42v173q40 11 65.5 41.5T897-461q0 40-25.5 67T806-356v176q0 24-18 42t-42 18H570q-5-48-35.5-77.5T463-227q-41 0-71.5 29.5T356-120Zm-176-60h130q25-61 69.888-84 44.888-23 83-23T546-264q45 23 70 84h130v-235h45q20 0 33-13t13-33q0-20-13-33t-33-13h-45v-239H511v-48q0-20-13-33t-33-13q-20 0-33 13t-13 33v48H180v130q48.15 17.817 77.575 59.686Q287-514.445 287-462.777 287-412 257.5-370T180-310v130Zm329-330Z"/></svg>
|
Before Width: | Height: | Size: 669 B After Width: | Height: | Size: 688 B |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M273-160 80-353l193-193 42 42-121 121h316v60H194l121 121-42 42Zm414-254-42-42 121-121H450v-60h316L645-758l42-42 193 193-193 193Z"/></svg>
|
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48" fill="#fcba03"><path d="M273-160 80-353l193-193 42 42-121 121h316v60H194l121 121-42 42Zm414-254-42-42 121-121H450v-60h316L645-758l42-42 193 193-193 193Z"/></svg>
|
Before Width: | Height: | Size: 234 B After Width: | Height: | Size: 249 B |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M700-160v-410H275l153 153-42 43-226-226 226-226 42 42-153 154h485v470h-60Z"/></svg>
|
<svg xmlns="http://www.w3.org/2000/svg" fill="#0388fc" height="48" viewBox="0 -960 960 960" width="48"><path d="M700-160v-410H275l153 153-42 43-226-226 226-226 42 42-153 154h485v470h-60Z"/></svg>
|
Before Width: | Height: | Size: 180 B After Width: | Height: | Size: 195 B |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M197-197q-54-54-85.5-126.5T80-480q0-84 31.5-156.5T197-763l43 43q-46 46-73 107.5T140-480q0 71 26.5 132T240-240l-43 43Zm113-113q-32-32-51-75.5T240-480q0-51 19-94.5t51-75.5l43 43q-24 24-38.5 56.5T300-480q0 38 14 70t39 57l-43 43Zm170-90q-33 0-56.5-23.5T400-480q0-33 23.5-56.5T480-560q33 0 56.5 23.5T560-480q0 33-23.5 56.5T480-400Zm170 90-43-43q24-24 38.5-56.5T660-480q0-38-14-70t-39-57l43-43q32 32 51 75.5t19 94.5q0 50-19 93.5T650-310Zm113 113-43-43q46-46 73-107.5T820-480q0-71-26.5-132T720-720l43-43q54 55 85.5 127.5T880-480q0 83-31.5 155.5T763-197Z"/></svg>
|
<svg fill="#83f2c4" xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M197-197q-54-54-85.5-126.5T80-480q0-84 31.5-156.5T197-763l43 43q-46 46-73 107.5T140-480q0 71 26.5 132T240-240l-43 43Zm113-113q-32-32-51-75.5T240-480q0-51 19-94.5t51-75.5l43 43q-24 24-38.5 56.5T300-480q0 38 14 70t39 57l-43 43Zm170-90q-33 0-56.5-23.5T400-480q0-33 23.5-56.5T480-560q33 0 56.5 23.5T560-480q0 33-23.5 56.5T480-400Zm170 90-43-43q24-24 38.5-56.5T660-480q0-38-14-70t-39-57l43-43q32 32 51 75.5t19 94.5q0 50-19 93.5T650-310Zm113 113-43-43q46-46 73-107.5T820-480q0-71-26.5-132T720-720l43-43q54 55 85.5 127.5T880-480q0 83-31.5 155.5T763-197Z"/></svg>
|
Before Width: | Height: | Size: 652 B After Width: | Height: | Size: 667 B |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M345-377h391L609-548 506-413l-68-87-93 123Zm-85 177q-24 0-42-18t-18-42v-560q0-24 18-42t42-18h560q24 0 42 18t18 42v560q0 24-18 42t-42 18H260Zm0-60h560v-560H260v560ZM140-80q-24 0-42-18t-18-42v-620h60v620h620v60H140Zm120-740v560-560Z"/></svg>
|
<svg class="item-icon" xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M345-377h391L609-548 506-413l-68-87-93 123Zm-85 177q-24 0-42-18t-18-42v-560q0-24 18-42t42-18h560q24 0 42 18t18 42v560q0 24-18 42t-42 18H260Zm0-60h560v-560H260v560ZM140-80q-24 0-42-18t-18-42v-620h60v620h620v60H140Zm120-740v560-560Z"/></svg>
|
Before Width: | Height: | Size: 336 B After Width: | Height: | Size: 355 B |
@@ -1 +1 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M109.912-150Q81-150 60.5-170.589 40-191.177 40-220.089 40-249 60.494-269.5t49.273-20.5q5.233 0 10.233.5 5 .5 13 2.5l200-200q-2-8-2.5-13t-.5-10.233q0-28.779 20.589-49.273Q371.177-580 400.089-580 429-580 449.5-559.366t20.5 49.61Q470-508 467-487l110 110q8-2 13-2.5t10-.5q5 0 10 .5t13 2.5l160-160q-2-8-2.5-13t-.5-10.233q0-28.779 20.589-49.273Q821.177-630 850.089-630 879-630 899.5-609.411q20.5 20.588 20.5 49.5Q920-531 899.506-510.5T850.233-490Q845-490 840-490.5q-5-.5-13-2.5L667-333q2 8 2.5 13t.5 10.233q0 28.779-20.589 49.273Q628.823-240 599.911-240 571-240 550.5-260.494T530-309.767q0-5.233.5-10.233.5-5 2.5-13L423-443q-8 2-13 2.5t-10.25.5q-1.75 0-22.75-3L177-243q2 8 2.5 13t.5 10.233q0 28.779-20.589 49.273Q138.823-150 109.912-150ZM160-592l-20.253-43.747L96-656l43.747-20.253L160-720l20.253 43.747L224-656l-43.747 20.253L160-592Zm440-51-30.717-66.283L503-740l66.283-30.717L600-837l30.717 66.283L697-740l-66.283 30.717L600-643Z"/></svg>
|
<svg fill="#edf230" xmlns="http://www.w3.org/2000/svg" height="48" viewBox="0 -960 960 960" width="48"><path d="M109.912-150Q81-150 60.5-170.589 40-191.177 40-220.089 40-249 60.494-269.5t49.273-20.5q5.233 0 10.233.5 5 .5 13 2.5l200-200q-2-8-2.5-13t-.5-10.233q0-28.779 20.589-49.273Q371.177-580 400.089-580 429-580 449.5-559.366t20.5 49.61Q470-508 467-487l110 110q8-2 13-2.5t10-.5q5 0 10 .5t13 2.5l160-160q-2-8-2.5-13t-.5-10.233q0-28.779 20.589-49.273Q821.177-630 850.089-630 879-630 899.5-609.411q20.5 20.588 20.5 49.5Q920-531 899.506-510.5T850.233-490Q845-490 840-490.5q-5-.5-13-2.5L667-333q2 8 2.5 13t.5 10.233q0 28.779-20.589 49.273Q628.823-240 599.911-240 571-240 550.5-260.494T530-309.767q0-5.233.5-10.233.5-5 2.5-13L423-443q-8 2-13 2.5t-10.25.5q-1.75 0-22.75-3L177-243q2 8 2.5 13t.5 10.233q0 28.779-20.589 49.273Q138.823-150 109.912-150ZM160-592l-20.253-43.747L96-656l43.747-20.253L160-720l20.253 43.747L224-656l-43.747 20.253L160-592Zm440-51-30.717-66.283L503-740l66.283-30.717L600-837l30.717 66.283L697-740l-66.283 30.717L600-643Z"/></svg>
|
Before Width: | Height: | Size: 1.0 KiB After Width: | Height: | Size: 1.0 KiB |
BIN
docs/img/logo_white.png
Normal file
After Width: | Height: | Size: 6.5 KiB |
BIN
docs/img/preview-mobile.png
Normal file
After Width: | Height: | Size: 140 KiB |
BIN
docs/img/preview-mobile.psd
Normal file
BIN
docs/img/preview-pc.png
Normal file
After Width: | Height: | Size: 194 KiB |
BIN
docs/img/screenshots/1.png
Normal file
After Width: | Height: | Size: 202 KiB |
Before Width: | Height: | Size: 42 KiB |
BIN
docs/img/screenshots/10.png
Normal file
After Width: | Height: | Size: 120 KiB |
Before Width: | Height: | Size: 62 KiB |
BIN
docs/img/screenshots/2.png
Normal file
After Width: | Height: | Size: 146 KiB |
Before Width: | Height: | Size: 32 KiB |
BIN
docs/img/screenshots/3.png
Normal file
After Width: | Height: | Size: 88 KiB |
Before Width: | Height: | Size: 28 KiB |
BIN
docs/img/screenshots/4.png
Normal file
After Width: | Height: | Size: 203 KiB |
Before Width: | Height: | Size: 41 KiB |
BIN
docs/img/screenshots/5.png
Normal file
After Width: | Height: | Size: 123 KiB |
Before Width: | Height: | Size: 46 KiB |
BIN
docs/img/screenshots/6.png
Normal file
After Width: | Height: | Size: 194 KiB |
Before Width: | Height: | Size: 48 KiB |
BIN
docs/img/screenshots/7.png
Normal file
After Width: | Height: | Size: 152 KiB |
Before Width: | Height: | Size: 55 KiB |
BIN
docs/img/screenshots/8.png
Normal file
After Width: | Height: | Size: 185 KiB |
Before Width: | Height: | Size: 68 KiB |
BIN
docs/img/screenshots/9.png
Normal file
After Width: | Height: | Size: 867 KiB |
Before Width: | Height: | Size: 153 KiB |
908
docs/index.html
@@ -1,375 +1,599 @@
|
|||||||
<!DOCTYPE html>
|
<!doctype html>
|
||||||
<html lang="en">
|
<html>
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
<meta content="width=device-width, initial-scale=1.0" name="viewport">
|
<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="Reverse Proxy, Open Source, Aroz, Go, OS, NAS, Cloud" name="keywords">
|
||||||
<meta content="A reverse proxy server and cluster network gateway for noobs" name="description">
|
<meta content="Simplify your self-hosted services with Zoraxy, the ultimate homelab networking toolbox" name="description">
|
||||||
<meta name="author" content="tobychui">
|
<meta name="author" content="tobychui">
|
||||||
|
|
||||||
<!-- HTML Meta Tags -->
|
<!-- HTML Meta Tags -->
|
||||||
<title>Cluster Proxy Gateway | Zoraxy</title>
|
<title>Homelab Gateway | Zoraxy</title>
|
||||||
<meta name="description" content="A reverse proxy server and cluster network gateway for noobs">
|
<meta name="description" content="Simplify your self-hosted services with Zoraxy, the ultimate homelab networking toolbox">
|
||||||
|
|
||||||
<!-- Facebook Meta Tags -->
|
<!-- Facebook Meta Tags -->
|
||||||
<meta property="og:url" content="https://zoraxy.arozos.com/">
|
<meta property="og:url" content="https://zoraxy.aroz.org/">
|
||||||
<meta property="og:type" content="website">
|
<meta property="og:type" content="website">
|
||||||
<meta property="og:title" content="Cluster Proxy Gateway | Zoraxy">
|
<meta property="og:title" content="Hello Zoraxy">
|
||||||
<meta property="og:description" content="A reverse proxy server and cluster network gateway for noobs">
|
<meta property="og:description" content="Simplify your self-hosted services with Zoraxy, the ultimate homelab networking toolbox">
|
||||||
<meta property="og:image" content="https://zoraxy.arozos.com/img/og.png">
|
<meta property="og:image" content="https://zoraxy.aroz.org/img/og.png">
|
||||||
|
|
||||||
<!-- Twitter Meta Tags -->
|
<!-- Twitter Meta Tags -->
|
||||||
<meta name="twitter:card" content="summary_large_image">
|
<meta name="twitter:card" content="summary_large_image">
|
||||||
<meta property="twitter:domain" content="arozos.com">
|
<meta property="twitter:domain" content="os.aroz.org">
|
||||||
<meta property="twitter:url" content="https://zoraxy.arozos.com/">
|
<meta property="twitter:url" content="https://zoraxy.aroz.org/">
|
||||||
<meta name="twitter:title" content="Cluster Proxy Gateway | Zoraxy">
|
<meta name="twitter:title" content="Hello Zoraxy">
|
||||||
<meta name="twitter:description" content="A reverse proxy server and cluster network gateway for noobs">
|
<meta name="twitter:description" content="Simplify your self-hosted services with Zoraxy, the ultimate homelab networking toolbox">
|
||||||
<meta name="twitter:image" content="https://zoraxy.arozos.com/img/og.png">
|
<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 -->
|
<!-- Favicons -->
|
||||||
<link href="favicon.png" rel="icon">
|
<link href="favicon.png" rel="icon">
|
||||||
|
<link href="img/apple-touch-icon.png" rel="apple-touch-icon">
|
||||||
|
|
||||||
<!-- Google Fonts -->
|
<!-- Google Fonts -->
|
||||||
<link rel="preconnect" href="https://fonts.googleapis.com">
|
<link rel="preconnect" href="https://fonts.googleapis.com">
|
||||||
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
|
<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 href="https://fonts.googleapis.com/css2?family=Source+Sans+Pro:wght@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=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 -->
|
<!-- AOS.js-->
|
||||||
<link href="style.css" rel="stylesheet">
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/aos/2.3.4/aos.js" integrity="sha512-A7AYk1fGKX6S2SsHywmPkrnzTZHrgiVT7GcQkLGDe2ev0aWb8zejytzS8wjo7PGEXKqJOrjQ4oORtnimIRZBtw==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>
|
||||||
<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" />
|
||||||
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>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div class="main section">
|
<div id="backToTopBtn" class="ui big icon button" onclick="backToTop();">
|
||||||
<div class="left-menu">
|
<i class="ui arrow up icon"></i>
|
||||||
<div class="iconWrapper">
|
</div>
|
||||||
<a href="index.html"><img class="ui fluid image" src="img/icon.png"></a>
|
<button id="rwdmenubtn" class="ui black big icon button"><i class="ui bars icon"></i></button>
|
||||||
</div>
|
<div id="mainmenu" class="ui segment">
|
||||||
<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="dot-container">
|
|
||||||
<div class="dot"></div>
|
|
||||||
<div class="dot"></div>
|
|
||||||
<div class="dot"></div>
|
|
||||||
<div class="dot"></div>
|
|
||||||
</div>
|
|
||||||
<div class="headbanner"></div>
|
|
||||||
<div id="home" class="herotext">
|
|
||||||
<div class="ui basic segment">
|
|
||||||
<div class="bannerHeaderWrapper">
|
|
||||||
<h1 class="bannerHeader">Zoraxy</h1>
|
|
||||||
<p class="bannerSubheader">All in one homelab network routing solution</p>
|
|
||||||
</div>
|
|
||||||
<br><br>
|
|
||||||
<a class="ui black big button" href="#features">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>
|
|
||||||
|
|
||||||
<!-- 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, noobs 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 of the 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. Allow 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>Integrated with Gotty Web SSH terminal, allow 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 insights 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 discovering service, 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.webp" target="_blank"><img src="img/screenshots/1.webp" class="ui fluid image screenshot"></a>
|
|
||||||
</div>
|
|
||||||
<div class="column">
|
|
||||||
<a href="img/screenshots/2.webp" target="_blank"><img src="img/screenshots/2.webp" class="ui fluid image screenshot"></a>
|
|
||||||
</div>
|
|
||||||
<div class="column">
|
|
||||||
<a href="img/screenshots/3.webp" target="_blank"><img src="img/screenshots/3.webp" class="ui fluid image screenshot"></a>
|
|
||||||
</div>
|
|
||||||
<div class="column">
|
|
||||||
<a href="img/screenshots/4.webp" target="_blank"><img src="img/screenshots/4.webp" class="ui fluid image screenshot"></a>
|
|
||||||
</div>
|
|
||||||
<div class="column">
|
|
||||||
<a href="img/screenshots/5.webp" target="_blank"><img src="img/screenshots/5.webp" class="ui fluid image screenshot"></a>
|
|
||||||
</div>
|
|
||||||
<div class="column">
|
|
||||||
<a href="img/screenshots/6.webp" target="_blank"><img src="img/screenshots/6.webp" class="ui fluid image screenshot"></a>
|
|
||||||
</div>
|
|
||||||
<div class="column">
|
|
||||||
<a href="img/screenshots/7.webp" target="_blank"><img src="img/screenshots/7.webp" class="ui fluid image screenshot"></a>
|
|
||||||
</div>
|
|
||||||
<div class="column">
|
|
||||||
<a href="img/screenshots/8.webp" target="_blank"><img src="img/screenshots/8.webp" class="ui fluid image screenshot"></a>
|
|
||||||
</div>
|
|
||||||
<div class="column">
|
|
||||||
<a href="img/screenshots/9.webp" target="_blank"><img src="img/screenshots/9.webp" class="ui fluid image screenshot"></a>
|
|
||||||
</div>
|
|
||||||
<div class="column">
|
|
||||||
<a href="img/screenshots/10.webp" target="_blank"><img src="img/screenshots/10.webp" 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">
|
<div class="ui container">
|
||||||
<p style="color: #3a3a3a">CopyRight Zoraxy Project and its authors © 2021 - <span class="year"></span></p>
|
<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>
|
||||||
|
</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>
|
||||||
|
<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>
|
||||||
|
<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 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>
|
||||||
|
<div class="ui three column stackable grid">
|
||||||
|
<div class="column">
|
||||||
|
<img class="ui fluid image screenshot" src="img/screenshots/1.png">
|
||||||
|
</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><br>
|
<br><br><br>
|
||||||
</div>
|
|
||||||
</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="https://zoraxy.aroz.org/plugins/html/" 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="https://github.com/aroz-online/zoraxy-official-plugins" target="_blank">Offical Plugin List</a></div>
|
||||||
|
<div class="item"><a href="https://zoraxy.aroz.org/plugins/html/" 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>
|
<script>
|
||||||
$(".year").text(new Date().getFullYear() );
|
AOS.init();
|
||||||
|
$(".year").text(new Date().getFullYear());
|
||||||
|
|
||||||
$(".menu-item").on("click", function(){
|
// Function to open the modal with the clicked image using jQuery
|
||||||
$(".menu-item.active").removeClass("active");
|
function openModal(src) {
|
||||||
$(this).addClass("active");
|
// Remove the old modal if it exists
|
||||||
});
|
$('#imageModal').remove();
|
||||||
|
|
||||||
$(".right-content").on("scroll", function() {
|
const modal = $('<div style="display:none;">', { id: 'imageModal' }).css({
|
||||||
var scrollPos = $(".right-content").scrollTop();
|
position: 'fixed',
|
||||||
if (scrollPos < 10){
|
top: '0',
|
||||||
//Reaching the top
|
left: '0',
|
||||||
$('.menu-item.active').removeClass("active");
|
width: '100%',
|
||||||
$($('.menu-item')[0]).addClass('active');
|
height: '100%',
|
||||||
return;
|
backgroundColor: 'rgba(0, 0, 0, 0.8)',
|
||||||
}else if ($(".right-content")[0].scrollHeight == $(".right-content").scrollTop() + window.innerHeight ){
|
display: 'flex',
|
||||||
//Reaching the bottom
|
alignItems: 'center',
|
||||||
$('.menu-item.active').removeClass("active");
|
justifyContent: 'center',
|
||||||
$($('.menu-item').get().reverse()[0]).addClass('active');
|
zIndex: '1000'
|
||||||
return
|
});
|
||||||
|
|
||||||
|
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>
|
</script>
|
||||||
|
<!-- Locales -->
|
||||||
|
<script src="main.js" defer></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
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
@@ -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
@@ -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 = "";
|
||||||
|
});
|
||||||
|
}
|
BIN
docs/plugins/assets/banner.png
Normal file
After Width: | Height: | Size: 194 KiB |
BIN
docs/plugins/assets/banner.psd
Normal file
BIN
docs/plugins/assets/logo.png
Normal file
After Width: | Height: | Size: 4.2 KiB |
BIN
docs/plugins/assets/logo_white.png
Normal file
After Width: | Height: | Size: 4.1 KiB |
51
docs/plugins/assets/theme.js
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
/* Things to do before body loads */
|
||||||
|
function restoreDarkMode(){
|
||||||
|
if (localStorage.getItem("darkMode") === "enabled") {
|
||||||
|
$("html").addClass("is-dark");
|
||||||
|
$("html").removeClass("is-white");
|
||||||
|
} else {
|
||||||
|
$("html").removeClass("is-dark");
|
||||||
|
$("html").addClass("is-white");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
restoreDarkMode();
|
||||||
|
|
||||||
|
function updateElementToTheme(isDarkTheme=false){
|
||||||
|
if (!isDarkTheme){
|
||||||
|
let whiteSrc = $("#sysicon").attr("white_src");
|
||||||
|
$("#sysicon").attr("src", whiteSrc);
|
||||||
|
$("#darkModeToggle").html(`<span class="ts-icon is-sun-icon"></span>`);
|
||||||
|
|
||||||
|
// Update the rendering text color in the garphs
|
||||||
|
if (typeof(changeScaleTextColor) != "undefined"){
|
||||||
|
changeScaleTextColor("black");
|
||||||
|
}
|
||||||
|
|
||||||
|
}else{
|
||||||
|
let darkSrc = $("#sysicon").attr("dark_src");
|
||||||
|
$("#sysicon").attr("src", darkSrc);
|
||||||
|
$("#darkModeToggle").html(`<span class="ts-icon is-moon-icon"></span>`);
|
||||||
|
|
||||||
|
// Update the rendering text color in the garphs
|
||||||
|
if (typeof(changeScaleTextColor) != "undefined"){
|
||||||
|
changeScaleTextColor("white");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Things to do after body loads */
|
||||||
|
$(document).ready(function(){
|
||||||
|
$("#darkModeToggle").on("click", function() {
|
||||||
|
$("html").toggleClass("is-dark");
|
||||||
|
$("html").toggleClass("is-white");
|
||||||
|
if ($("html").hasClass("is-dark")) {
|
||||||
|
localStorage.setItem("darkMode", "enabled");
|
||||||
|
updateElementToTheme(true);
|
||||||
|
} else {
|
||||||
|
localStorage.setItem("darkMode", "disabled");
|
||||||
|
updateElementToTheme(false);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
updateElementToTheme(localStorage.getItem("darkMode") === "enabled");
|
||||||
|
});
|
280
docs/plugins/build.go
Normal file
@@ -0,0 +1,280 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"io/fs"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/gomarkdown/markdown"
|
||||||
|
"github.com/gomarkdown/markdown/html"
|
||||||
|
"github.com/gomarkdown/markdown/parser"
|
||||||
|
"github.com/yosssi/gohtml"
|
||||||
|
)
|
||||||
|
|
||||||
|
type FileInfo struct {
|
||||||
|
Filename string `json:"filename"`
|
||||||
|
Title string `json:"title"`
|
||||||
|
Type string `json:"type"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func build() {
|
||||||
|
rootDir := "./docs"
|
||||||
|
outputFile := "./index.json"
|
||||||
|
|
||||||
|
type Folder struct {
|
||||||
|
Title string `json:"title"`
|
||||||
|
Path string `json:"path"`
|
||||||
|
Type string `json:"type"`
|
||||||
|
Files []interface{} `json:"files,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
var buildTree func(path string, d fs.DirEntry) interface{}
|
||||||
|
buildTree = func(path string, d fs.DirEntry) interface{} {
|
||||||
|
relativePath, _ := filepath.Rel(rootDir, path)
|
||||||
|
relativePath = filepath.ToSlash(relativePath)
|
||||||
|
var title string
|
||||||
|
if d.IsDir() {
|
||||||
|
title = filepath.Base(relativePath)
|
||||||
|
} else {
|
||||||
|
title = strings.TrimSuffix(filepath.Base(relativePath), filepath.Ext(relativePath))
|
||||||
|
}
|
||||||
|
|
||||||
|
//Strip the leader numbers from the title, e.g. 1. Introduction -> Introduction
|
||||||
|
if strings.Contains(title, ".") {
|
||||||
|
parts := strings.SplitN(title, ".", 2)
|
||||||
|
if len(parts) > 1 {
|
||||||
|
title = strings.TrimSpace(parts[1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Remove leading numbers and dots
|
||||||
|
title = strings.TrimLeft(title, "0123456789. ")
|
||||||
|
// Remove leading spaces
|
||||||
|
title = strings.TrimLeft(title, " ")
|
||||||
|
|
||||||
|
if d.IsDir() {
|
||||||
|
folder := Folder{
|
||||||
|
Title: title,
|
||||||
|
Path: relativePath,
|
||||||
|
Type: "folder",
|
||||||
|
}
|
||||||
|
|
||||||
|
entries, err := os.ReadDir(path)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, entry := range entries {
|
||||||
|
if entry.Name() == "img" || entry.Name() == "assets" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if strings.Contains(filepath.ToSlash(filepath.Join(relativePath, entry.Name())), "/img/") || strings.Contains(filepath.ToSlash(filepath.Join(relativePath, entry.Name())), "/assets/") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
child := buildTree(filepath.Join(path, entry.Name()), entry)
|
||||||
|
if child != nil {
|
||||||
|
folder.Files = append(folder.Files, child)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return folder
|
||||||
|
} else {
|
||||||
|
ext := filepath.Ext(relativePath)
|
||||||
|
if ext != ".md" && ext != ".html" && ext != ".txt" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return FileInfo{
|
||||||
|
Filename: relativePath,
|
||||||
|
Title: title,
|
||||||
|
Type: "file",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
rootInfo, err := os.Stat(rootDir)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
rootFolder := buildTree(rootDir, fs.FileInfoToDirEntry(rootInfo))
|
||||||
|
jsonData, err := json.MarshalIndent(rootFolder, "", " ")
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
/* For debug purposes, print the JSON structure */
|
||||||
|
err = os.WriteFile(outputFile, jsonData, 0644)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
/* For each file in the folder structure, convert markdown to HTML */
|
||||||
|
htmlOutputDir := "./html"
|
||||||
|
os.RemoveAll(htmlOutputDir) // Clear previous HTML output
|
||||||
|
err = os.MkdirAll(htmlOutputDir, os.ModePerm)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var processFiles func(interface{})
|
||||||
|
processFiles = func(node interface{}) {
|
||||||
|
switch n := node.(type) {
|
||||||
|
case FileInfo:
|
||||||
|
if filepath.Ext(n.Filename) == ".md" {
|
||||||
|
inputPath := filepath.Join(rootDir, n.Filename)
|
||||||
|
outputPath := filepath.Join(htmlOutputDir, strings.TrimSuffix(n.Filename, ".md")+".html")
|
||||||
|
|
||||||
|
// Ensure the output directory exists
|
||||||
|
err := os.MkdirAll(filepath.Dir(outputPath), os.ModePerm)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read the markdown file
|
||||||
|
mdContent, err := os.ReadFile(inputPath)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Convert markdown to HTML
|
||||||
|
docContent := mdToHTML(mdContent)
|
||||||
|
docContent, err = optimizeCss(docContent)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load the HTML template
|
||||||
|
templateBytes, err := os.ReadFile("template/documents.html")
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate the side menu HTML
|
||||||
|
sideMenuHTML, err := generateSideMenu(string(jsonData), n.Title)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
templateBody := string(templateBytes)
|
||||||
|
// Replace placeholders in the template
|
||||||
|
htmlContent := strings.ReplaceAll(templateBody, "{{title}}", n.Title+" | Zoraxy Documentation")
|
||||||
|
htmlContent = strings.ReplaceAll(htmlContent, "{{content}}", string(docContent))
|
||||||
|
htmlContent = strings.ReplaceAll(htmlContent, "{{sideMenu}}", sideMenuHTML)
|
||||||
|
htmlContent = strings.ReplaceAll(htmlContent, "{{root_url}}", *root_url)
|
||||||
|
//Add more if needed
|
||||||
|
|
||||||
|
//Beautify the HTML content
|
||||||
|
htmlContent = gohtml.Format(htmlContent)
|
||||||
|
|
||||||
|
// Write the HTML file
|
||||||
|
err = os.WriteFile(outputPath, []byte(htmlContent), 0644)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
//Check if the .md file directory have an ./img or ./assets folder. If yes, copy the contents to the output directory
|
||||||
|
imgDir := filepath.Join(rootDir, filepath.Dir(n.Filename), "img")
|
||||||
|
assetsDir := filepath.Join(rootDir, filepath.Dir(n.Filename), "assets")
|
||||||
|
if _, err := os.Stat(imgDir); !os.IsNotExist(err) {
|
||||||
|
err = filepath.Walk(imgDir, func(srcPath string, info os.FileInfo, err error) error {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
relPath, err := filepath.Rel(imgDir, srcPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
destPath := filepath.Join(filepath.Dir(outputPath), "img", relPath)
|
||||||
|
if info.IsDir() {
|
||||||
|
return os.MkdirAll(destPath, os.ModePerm)
|
||||||
|
}
|
||||||
|
data, err := os.ReadFile(srcPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return os.WriteFile(destPath, data, 0644)
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if _, err := os.Stat(assetsDir); !os.IsNotExist(err) {
|
||||||
|
err = filepath.Walk(assetsDir, func(srcPath string, info os.FileInfo, err error) error {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
relPath, err := filepath.Rel(assetsDir, srcPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
destPath := filepath.Join(filepath.Dir(outputPath), "assets", relPath)
|
||||||
|
if info.IsDir() {
|
||||||
|
return os.MkdirAll(destPath, os.ModePerm)
|
||||||
|
}
|
||||||
|
data, err := os.ReadFile(srcPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return os.WriteFile(destPath, data, 0644)
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Println("Generated HTML:", outputPath)
|
||||||
|
}
|
||||||
|
case Folder:
|
||||||
|
for _, child := range n.Files {
|
||||||
|
processFiles(child)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
processFiles(rootFolder)
|
||||||
|
copyOtherRes()
|
||||||
|
}
|
||||||
|
|
||||||
|
func copyOtherRes() {
|
||||||
|
srcDir := "./assets"
|
||||||
|
destDir := "./html/assets"
|
||||||
|
|
||||||
|
err := filepath.Walk(srcDir, func(srcPath string, info os.FileInfo, err error) error {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
relPath, err := filepath.Rel(srcDir, srcPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
destPath := filepath.Join(destDir, relPath)
|
||||||
|
if info.IsDir() {
|
||||||
|
return os.MkdirAll(destPath, os.ModePerm)
|
||||||
|
}
|
||||||
|
data, err := os.ReadFile(srcPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return os.WriteFile(destPath, data, 0644)
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func mdToHTML(md []byte) []byte {
|
||||||
|
// create markdown parser with extensions
|
||||||
|
extensions := parser.CommonExtensions | parser.AutoHeadingIDs | parser.NoEmptyLineBeforeBlock
|
||||||
|
p := parser.NewWithExtensions(extensions)
|
||||||
|
doc := p.Parse(md)
|
||||||
|
|
||||||
|
// create HTML renderer with extensions
|
||||||
|
htmlFlags := html.CommonFlags | html.HrefTargetBlank
|
||||||
|
opts := html.RendererOptions{
|
||||||
|
Flags: htmlFlags,
|
||||||
|
}
|
||||||
|
renderer := html.NewRenderer(opts)
|
||||||
|
|
||||||
|
return markdown.Render(doc, renderer)
|
||||||
|
}
|
4
docs/plugins/build.sh
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Run the Go program with the specified arguments
|
||||||
|
./docs.exe -m=build -root=plugins/html/
|
130
docs/plugins/cssOptimizer.go
Normal file
@@ -0,0 +1,130 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/PuerkitoBio/goquery"
|
||||||
|
)
|
||||||
|
|
||||||
|
func optimizeCss(htmlContent []byte) ([]byte, error) {
|
||||||
|
doc, err := goquery.NewDocumentFromReader(strings.NewReader(string(htmlContent)))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
originalHTMLContent := string(htmlContent)
|
||||||
|
// Replace img elements
|
||||||
|
|
||||||
|
doc.Find("img").Each(func(i int, s *goquery.Selection) {
|
||||||
|
//For each of the image element, replace the parent from p to div
|
||||||
|
originalParent, err := s.Parent().Html()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Error getting parent HTML:", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
src, exists := s.Attr("src")
|
||||||
|
if !exists {
|
||||||
|
fmt.Println("No src attribute found for img element")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
encodedSrc := (&url.URL{Path: src}).String()
|
||||||
|
|
||||||
|
//Patch the bug in the parser that converts " />" to "/>"
|
||||||
|
originalParent = strings.ReplaceAll(originalParent, "/>", " />")
|
||||||
|
fmt.Println("<div class=\"ts-image is-rounded\"><img src=\"./" + encodedSrc + "\"></div>")
|
||||||
|
//Replace the img with ts-image
|
||||||
|
originalHTMLContent = strings.Replace(originalHTMLContent, originalParent, "<div class=\"ts-image is-rounded\" style=\"max-width: 800px\">"+originalParent+"</div>", 1)
|
||||||
|
})
|
||||||
|
|
||||||
|
// Add "ts-text" class to each p element
|
||||||
|
doc.Find("p").Each(func(i int, s *goquery.Selection) {
|
||||||
|
class, exists := s.Attr("class")
|
||||||
|
var newClass string
|
||||||
|
if exists {
|
||||||
|
newClass = fmt.Sprintf("%s ts-text", class)
|
||||||
|
} else {
|
||||||
|
newClass = "ts-text"
|
||||||
|
}
|
||||||
|
|
||||||
|
originalParagraph, _ := s.Html()
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, originalParagraph, fmt.Sprintf("<p class=\"%s\">%s</p>", newClass, originalParagraph))
|
||||||
|
})
|
||||||
|
|
||||||
|
//Replace hr with ts-divider
|
||||||
|
// Replace hr elements outside of code blocks
|
||||||
|
doc.Find("hr").Each(func(i int, s *goquery.Selection) {
|
||||||
|
parent := s.Parent()
|
||||||
|
if parent.Is("code") {
|
||||||
|
// Skip <hr> inside <code> blocks
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Replace <hr> with <div class="ts-divider"></div>
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "<hr>", "<div class=\"ts-divider has-top-spaced-large\"></div>")
|
||||||
|
})
|
||||||
|
|
||||||
|
// Add ts-table to all table elements
|
||||||
|
doc.Find("table").Each(func(i int, s *goquery.Selection) {
|
||||||
|
class, exists := s.Attr("class")
|
||||||
|
var newClass string
|
||||||
|
if exists {
|
||||||
|
newClass = fmt.Sprintf("%s ts-table", class)
|
||||||
|
} else {
|
||||||
|
newClass = "ts-table"
|
||||||
|
}
|
||||||
|
|
||||||
|
originalTable, _ := s.Html()
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, originalTable, fmt.Sprintf("<table class=\"%s\">%s</table>", newClass, originalTable))
|
||||||
|
})
|
||||||
|
|
||||||
|
// Replace <ul> <ol> and <li>
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "<ul>", "<div class=\"ts-list is-unordered\">")
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "</ul>", "</div>")
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "<ol>", "<div class=\"ts-list is-ordered\">")
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "</ol>", "</div>")
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "<li>", "<div class=\"item\">")
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "</li>", "</div>")
|
||||||
|
|
||||||
|
// Replace <strong> with <span class="ts-text is-heavy"></span>
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "<strong>", "<span class=\"ts-text is-heavy\">")
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "</strong>", "</span>")
|
||||||
|
|
||||||
|
// Replace <code> without class with <span class="ts-text is-code">
|
||||||
|
for {
|
||||||
|
startIndex := strings.Index(originalHTMLContent, "<code>")
|
||||||
|
if startIndex == -1 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
endIndex := strings.Index(originalHTMLContent[startIndex+6:], "</code>")
|
||||||
|
if endIndex == -1 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
endIndex += startIndex + 6
|
||||||
|
|
||||||
|
codeSegment := originalHTMLContent[startIndex : endIndex+7] // Include </code>
|
||||||
|
if !strings.Contains(codeSegment, "class=") {
|
||||||
|
replacement := strings.Replace(codeSegment, "<code>", "<span class=\"ts-text is-code\">", 1)
|
||||||
|
replacement = strings.Replace(replacement, "</code>", "</span>", 1)
|
||||||
|
originalHTMLContent = strings.Replace(originalHTMLContent, codeSegment, replacement, 1)
|
||||||
|
} else {
|
||||||
|
// Skip if <code> already has a class
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Replace blockquote to <div class="ts-quote">
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "<blockquote>", "<div class=\"ts-quote\">")
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "</blockquote>", "</div>")
|
||||||
|
|
||||||
|
/*
|
||||||
|
originalHTMLContent = strings.ReplaceAll(originalHTMLContent, "language-xml", "")
|
||||||
|
// Remove class attribute from <code> inside <pre>
|
||||||
|
re := regexp.MustCompile(`<pre><code class="[^"]*">`)
|
||||||
|
originalHTMLContent = re.ReplaceAllString(originalHTMLContent, "<pre><code>")
|
||||||
|
*/
|
||||||
|
return []byte(originalHTMLContent), err
|
||||||
|
}
|
12
docs/plugins/dev.sh
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
#/bin/bash
|
||||||
|
go build
|
||||||
|
# Run the Go program with the specified arguments
|
||||||
|
./docs.exe -m=build
|
||||||
|
|
||||||
|
echo "Running docs in development mode..."
|
||||||
|
./docs.exe
|
||||||
|
|
||||||
|
# After the docs web server mode terminate, rebuild it with root = plugins/html/
|
||||||
|
./docs.exe -m=build -root=plugins/html/
|
||||||
|
|
||||||
|
# The doc should always be ready to push to release branch
|
1
docs/plugins/diagrams/dynamic_capture.drawio
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<mxfile host="Electron" modified="2025-05-26T13:15:22.444Z" agent="Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) draw.io/12.4.2 Chrome/78.0.3904.130 Electron/7.1.4 Safari/537.36" etag="PlPASUA5DkVflfK71JS_" version="12.4.2" type="device" pages="1"><diagram id="fpBU8wsgIdeXEMTqvbYu" name="Page-1">5VrRcps4FP0az2Qf1gMIAX5MnCbNbHcn00xn277syCBjpRhRIcdOvn4lkG2Q1CZxMSGpHzxwBZI49xxd3QsjMF1uLhkqFn/TBGcjz0k2I3A+8rzQn4h/abivDRAGtSFlJKlN7t5wQx6wMjrKuiIJLlsXckozToq2MaZ5jmPesiHG6Lp92Zxm7VELlGLDcBOjzLT+SxK+qK2RF+7t7zFJF9uR3UA98BJtL1ZPUi5QQtcNE3g3AlNGKa+PlpspziR2W1zq+y5+0LqbGMM5f8oNX+6Xk9ldzsLJZXobx5+vspvwT9XLHcpW6oG/UoY292rK/H6LA6OrPMGyK2cEztYLwvFNgWLZuhaOF7YFX2bizBWH5tS242DG8aZhUlO9xHSJORPDOtvWiYJN8caN1Pl67wV3C+2i4QFf2ZByfLrreo+NOFDwPAMqz4DqOlulJH9xqHxvaFABAyoDJJwnp1Ke4izOUFmSuI0L3hD+WUI4hursS6PlfKPQrU7u1UmCykWF+0+BLemKxfhxTeCktTKY8DfghRZ0tzaGM8TJXXs9sUGuRrimRMx4510Qtr3ru5rX6udRdzX1r3W0m5DqKPC0jjhiKeZGRxUDdo99OCn8V02KYFCk8MPJGEbO/ue3XCtCwbjR6LiHMUaM8qKMcYPHKdP3Yuv5sA01sCy2FseD4FiLrRv9urB+CMxA+O4Fjh3151Na6wj2S+kBMtoPoR2SRxh9PEJPhhwpaoK8rkgRTva/txgotsM1GHOVc0bLQqaEOneELnibLSgjaS6pJJyNmTBI9RCRBJ6qhiVJEnn7GcMleUCzqivJmUI+UvWQ8GwEzyVxRHcrTss6k5W9M8qFB6nsJ+hGskDb8e9yzAanQgundLd0lxuZydGbi0F66ADOoRtxLZgBfUd/bLWY6dlNTuZzkqcS7TwpdiMPRDYioexGNy5o6yYKLKGuV+E8ISvqf0PQRgn4lg0BtIB0vNKLuW3C43Q8km65KCV3X5ytnazq7Y3Ybl1o4B70Ss7QgH2aEWxZG8oFKuThapmdxpw20f2AZji7FlApLc8o53QpLshkwxmKv6UVwac0k/eJ3sC8+lk8xKnGdrriGcnxdFf8dTqSgFZzsSR5oM8tsfcb5HhawdfXV5Mnh1fdeT2neJ6ZvnwqSs4wWr740h44bWwGUCo2t+7PZvbxkr2BSCOAPSRxPx+k75QOmG+mRD9EqGUzrlz/BgK+C7SV6qXTOPCEFze9L1nQa7PdVnC17EaPV3AFv0Gy68H2XtTX0Ty04OpHPa8jZoJ1QdkasaQC6XsFH0rE+iA/LBhSRtHIf2FH7y6idmLn2yq9vWYYoINS79ClZChgcujGVqsbQb0AdewXuGZIPkFxjAs5x3ktqj+GJJvOykaOthiGPZaN/rq6/e/7x3+uPn0oNmw9+fhwPvvaxQcWc5JljZw7gThKfGEXmQr9hhstkTcDwb5wranKAu1QYhbs6uWGJjs9F+pOdlZfmwHs5EpEJ2f79ZGMYitcyhkvUJ5kJE8HpcJ98DIU9yTy7IKX5l1LdexowcvqGNi5COdRjOPYJsJZBGW0fg0iDMeaow4t5ATaB0vweFtHq3/NovMJ0YR3W71fHKbugm505080d9o2jTZCHaA7cbr/HrZ24/6jYvDufw==</diagram></mxfile>
|
BIN
docs/plugins/diagrams/dynamic_capture.png
Normal file
After Width: | Height: | Size: 190 KiB |
1
docs/plugins/diagrams/plugin_workflow.drawio
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<mxfile host="Electron" modified="2025-05-25T13:00:38.901Z" agent="Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) draw.io/12.4.2 Chrome/78.0.3904.130 Electron/7.1.4 Safari/537.36" etag="lsRaVkYmhHZrtifUUFnd" version="12.4.2" type="device" pages="1"><diagram id="aAlAxoldjFgNxtltmMrS" name="Page-1">7Vpbc5s4FP41nuk+lEGI62PtdDfbzLaZOpnN7suOahSsXYy8Qm7s/PpKRoCRcEwTcNykfvCgIyHQdy76zhEjOFmsf2NoOf+DxjgdOXa8HsGzkeMEbiT+pWBTCDzPLwQJI3EhArVgSu6xEtpKuiIxzhsDOaUpJ8umcEazDM94Q4YYo3fNYbc0bT51iRJsCKYzlJrSP0nM54U0dIJafo5JMi+fDHy14AUqB6uV5HMU07sdEXw/ghNGKS+uFusJTiV2JS7Ffb/u6a1ejOGMd7nBTa5BHF3Azx/t65sPk4uLm9m/bx2lja8oXakVq7flmxICRldZjOUs9giO7+aE4+kSzWTvndC5kM35IhUtIC5vSZpOaEqZaGc0E4PGMcrn29tlv3ocZhyv9y4EVPAIs8J0gTnbiCGlTXnFHcqinEgBfFfrx3fVmPmObkI1DimTSKqZa9TEhQLue0AMfngQYdgRROANhqL3w6MIgq4ogqFQBAaIf1OG1punQdkDVKCMgyVW0MSqGrOLlTuYwRlQXaarhGRC9pmuOGbPjhmEp4YZNDC7XuacYbR4drC84NTAcg8HNJzF7yRLEa1ZivKczJq4tAQsHBuM5SBEOxB4LQiUMoZTxMnX5vRtsKgnXFIiHly7eKhpINCQzemKzbC6a5eqHJjIi7SJOGIJ5sZEWy1Vy3684jrsRC9KcTCMLN+L6l/Qjx4PzHtstXbgui9Krb4j4Pft6gf60erD0x5bqX1z7x52IqDTwhZyDVrUDgfLUKKnW/6pGzuImnuGYz86aMHmRDpXH9iggUnhn92iYahZdOR1smjPG8iggUne77d5zj/LLYe3fs84o9OlLAlJzuen4o3GXwSl9xN59WYqtCi7LhFD4pUw+8UAWcDFm0iilCSZdBABm0gP4FiCSmYofac6FiSO5e1jhnNyj75sp5IqWkpr2YLgjUfemdSKmG7FaV5UvOTsjHLhAFTOA/tRW6WTKuS3qK3NJ+FQgQiYCcSLi0RQY69GgtGZBushzT9yJDITGCLdKldu9ebD9NNH2U+Q+J9enX26vjopNxJEvp8NPdJKjm2pZdhiUc5gftR33XaATQJ63WiPsVP3V+Gxnx5txOLZ5kYak20BLyolf0lYLeBUgrO1ArpobXZbl5iR7SajhKcewUAYWk7UVCaMLOdxYQyGwArsffmCsBLLtr26W6MMA8e40iL2EokJzW5JsmJYkonXSiT0AAhBi2sHxyQSTg9E4tW6dlhXZLSY7drSG3d+fj8+b2uPAbYV7ji9toyhfT40bKc8JLGVOx/04FzwoP+wdrzUcuLU3dPbNujmFt4oPu3SnEj23tKMT9UrlynkU3ze1cgngFZL+hC1beiONVTe55iVjOrQ5qfq9qgOgvAEVAdNMlaft+H/VziX8eMcZXFKsuSnHqU2mhzMdVt2XXhkLb6GBD5yrZ1Di0grCLqO37q3fXcdXTvddWF41G0Qmun9+dXV5RY/5Y4vjsgaoTH0u8XFwQ7UzQ3txSvBDXQ22HFzGuygvs86QdUo8gjvQBbRNRg6dhEOHlhGURR6ttNGPZzpX4p1jYuufWCigeOiax7AVC6ZL2mW41PyyZ6Kd26wB/RG8e6oTtlnhj+QU57KEYOrnwx0/9JGC8WBLIUYtbpjed4DjOT0PO9tX5wEajrwK/o+gPOJZv3NeaG4+sN9+P4b</diagram></mxfile>
|
BIN
docs/plugins/diagrams/plugin_workflow.png
Normal file
After Width: | Height: | Size: 194 KiB |
1
docs/plugins/diagrams/static_capture.drawio
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<mxfile host="Electron" modified="2025-05-26T13:04:13.815Z" agent="Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) draw.io/12.4.2 Chrome/78.0.3904.130 Electron/7.1.4 Safari/537.36" etag="gzmRxnqbWDJ6DOZzUBFr" version="12.4.2" type="device" pages="1"><diagram id="fpBU8wsgIdeXEMTqvbYu" name="Page-1">5VrRjqM2FP2aSNOHRoAxhMeZzO52pW010qjqbt884AFawMg4k2S/vjaYALa3k0kDpJk8RHABY849x9fHsADrfPeJojL5lUQ4WzhWtFuA+4XjBBbk/yKwbwKe7zSBmKZRE7K7wGP6HcugJaObNMLV4ERGSMbSchgMSVHgkA1iiFKyHZ72TLLhXUsUYy3wGKJMj/6RRixpoivH7+K/4DRO2jvbXtAcyVF7snySKkER2fZC4MMCrCkhrNnKd2ucCexaXJrrPv7g6KFjFBfsmAu+7fPg6aWgfvAp/isMv37OHv2fZSsvKNvIB/6TULTbyy6zfYsDJZsiwqIpawHutknK8GOJQnF0yxPPYwnLM75n803ZKKYM737YW/uAAecOJjlmlN/Wai8IJGySN/ZK7m+7LNgttEkvA66MIZn4+NB0hw3fkPC8ASpHg+oh28RpMTtUrnNpUAENKg0kXES3Qp58L8xQVaXhEBe8S9lXAeESyr1vvSP3O4luvbOXOxGqkhr3DlgcafJWYOW9Ihsa4tdEosPfgxca0G1jFGeIpS/Dbpggl3d4ICnv4CG7wB9m17WVrDXdl1f19a80dOhQOxI7SkMM0RgzraGaAYfHPp0U7nWRwpuTFK4fLOHK6n7uILW8FCx7By37NMbwu8zKGNt7nTJTD7aOC4dQA8Nga0g88MYabO3VfxfWsbqYi++OZ5lRfzullYbgtJS+QEa7PjRD8gqjxyN08L+qFA1jLrpS+EH3u8ZC0d6ux5jPBaOkKoUlVLnDpcKGbEFZGheCSjy3mPKAEFTKTeCtPJCnUSQuv6O4Sr+jp7opwZlSPFL9kPBuAe8FcXhzG0aqxsmK1ilhPINEtOOdR7JAmfEfPGaPU76BU2pazueNdHN0dTVILR3AOnUirhQzoM7ox1aLbs++pJXoIHmuU8ZRCvlGiEq2objOAUuqS9IRd5jnmc2tVkMhrTy99pk4NZ6SjrBJ088QwJCwrmGGAA0gjbcWo8+j8DJeLkRaPqIoN6zKTM3WEYb5w0Ax2zDva7CvsxQXeo2tElSKzU2e3YaM9NH9gp5w9sChklp+IoyRnJ+QiQN3KPw7rgm+Jpm4jrcGnuufIUOMKGwnG5alBV4fVoOtM0lAWYQxuD4w5RzZeQemT1kBdtXR5Oh6qyZvYs/n6H7m97JiFKN89qHds4bYXMDasT6XfzOzJ3N/c0nDgxO4un+/ydQeD+ivqng7KVfLblmn/goKvg2UkWpuXweOeJMz+ZAFnSHbTSuwhtnoeCuw4B24XwcOVwldFc2jRxXF/brBxOOIbrBufiOig8I0dO5XeN6fZh9Uep4XnslUwNUSDP2ca1rxNS35jjfOwOtXUDAUEFTBPH46qzSkvgsZW0Cmdxit6b62gmwrWFuzK0W34FenFLXWwFOdn7pkC9WiNbZUdJd+c+FVZqQ3FNAwSQsm1c05Xyr2TKW19FuPafaVx4rtiE9OmrI9n+dUUnpyAVNmgFB9kTL2h0i6k7zhqilJUeH5tXgO8akrZ4ZFnUltpPsODJJWa/xzyWM1sTx0z39l8nCGL/1gO904vzz4bveBd5Og7it58OEf</diagram></mxfile>
|
BIN
docs/plugins/diagrams/static_capture.png
Normal file
After Width: | Height: | Size: 171 KiB |
@@ -0,0 +1,43 @@
|
|||||||
|
# What is Zoraxy Plugin?
|
||||||
|
|
||||||
|
Last Update: 25/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Zoraxy Plugin is a powerful extension feature designed to enhance the functionality of the Zoraxy system. It provides additional features and capabilities that are not part of the core system, allowing users to customize their experience and optimize performance. The plugin is built to be modular and flexible, enabling users to tailor their Zoraxy environment to meet specific needs.
|
||||||
|
|
||||||
|
Zoraxy plugins are distributed as binaries, and developers have the flexibility to choose whether to open source them or not **as the plugin library and interface are open source under the LGPL license**.
|
||||||
|
|
||||||
|
There are two primary types of plugins:
|
||||||
|
- **Router plugins**: Involved with connections from HTTP proxy rules.
|
||||||
|
- **Utility plugins**: Provide user interfaces for various network features that operate independently of the Zoraxy core.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## How plugins are distributed & installed
|
||||||
|
Zoraxy plugins are distributed as platform-dependent binaries, tailored to specific operating systems and CPU architectures. These binaries follow a naming convention that includes the operating system, CPU architecture, and plugin name, such as `linux_amd64_foobar`, `windows_amd64_foobar.exe`, or `linux_arm64_foobar`.
|
||||||
|
|
||||||
|
To manually install a plugin for testing, place the binary file into the `/plugins/{plugin_name}/` folder within your Zoraxy installation directory.
|
||||||
|
|
||||||
|
> **Warning:** The binary name inside the folder must match the plugin folder name. For example, the binary should be named `foobar` (or `foobar.exe` on Windows) if placed in the `/plugins/foobar/` folder. Avoid using names like `foobar_plugin.exe`.
|
||||||
|
|
||||||
|
For distribution, a plugin store system is used. The plugin store architecture is similar to the one built into the Arduino IDE, with a manager URL (a JSON file) listing all the plugins supported by that store. See the documentation section for more details on how to implement your own plugin store.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Plugin vs Pull Request
|
||||||
|
The Zoraxy plugin was introduced to address specific use cases that enhance its functionality. It serves as an extension to the core Zoraxy system, providing additional features and capabilities while maintaining the integrity of the core system.
|
||||||
|
|
||||||
|
- Designed to handle features that are challenging to integrate directly into the Zoraxy core.
|
||||||
|
- Caters to scenarios where certain features are only applicable in limited situations, avoiding unnecessary resource consumption for other users.
|
||||||
|
- Allows for frequent updates to specific code components without impacting the core's stability or causing downtime.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### When should you add a core PR or a plugin?
|
||||||
|
In certain situations, implementing a feature as a plugin is more reasonable than directly integrating it into the Zoraxy core:
|
||||||
|
|
||||||
|
- **Core PR**: If the feature is relevant to most users and enhances Zoraxy's core functionality, consider submitting a core Pull Request (PR).
|
||||||
|
- **Plugin**: If the feature is targeted at a smaller user base or requires additional dependencies that not all users need, it should be developed as a plugin.
|
||||||
|
|
||||||
|
The decision depends on the feature's general relevance and its impact on core stability. Plugins offer flexibility without burdening the core.
|
62
docs/plugins/docs/1. Introduction/2. Getting Started.md
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
# Getting Started
|
||||||
|
|
||||||
|
Last Update: 25/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
To start developing plugins, you will need the following installed on your computer
|
||||||
|
|
||||||
|
1. The source code of Zoraxy
|
||||||
|
2. Go compiler
|
||||||
|
3. VSCode (recommended, or any editor of your choice)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Step 1: Start Zoraxy at least once
|
||||||
|
|
||||||
|
If you have just cloned Zoraxy from the Github repo, use the following to build and run it once.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd src/
|
||||||
|
go mod tidy
|
||||||
|
go build
|
||||||
|
sudo ./zoraxy
|
||||||
|
```
|
||||||
|
|
||||||
|
This would allow Zoraxy to generate all the required folder structure on startup.
|
||||||
|
|
||||||
|
After the startup process completes, you would see a folder named "plugins" in the working directory of Zoraxy.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Steps 2: Prepare the development environment for Zoraxy Plugin
|
||||||
|
|
||||||
|
Next, you will need to think of a name for your plugin. Lets name our new plugin "Lapwing".
|
||||||
|
|
||||||
|
**Notes: Plugin name described in Introspect (will discuss this in later sessions) can contains space, but the folder and compiled binary filename must not contains space and special characters for platform compatibilities reasons.**
|
||||||
|
|
||||||
|
Follow the steps below to create the folder structure
|
||||||
|
|
||||||
|
### 2.1 Create Plugin Folder
|
||||||
|
|
||||||
|
Create a folder with your plugin name in the `plugins` folder. After creating the folder, you would have something like `plugins/Lapwing/`.
|
||||||
|
|
||||||
|
### 2.2 Locate and copy Zoraxy Plugin library
|
||||||
|
|
||||||
|
Locate the Zoraxy plugin library from the Zoraxy source code. You can find the `zoraxy_plugin` Go module under `src/mod/plugins/zoraxy_plugin`.
|
||||||
|
|
||||||
|
Copy the `zoraxy_plugin` folder from the Zoraxy source code mod folder into the your plugin's mod folder. Let assume you use the same mod folder name as Zoraxy as `mod`, then your copied library path should be `plugins/Lapwing/mod/zoraxy_plugin`.
|
||||||
|
|
||||||
|
### 2.3 Prepare Go Project structure
|
||||||
|
|
||||||
|
Create the `main.go` file for your plugin. In the example above, it would be located at `plugins/Lapwing/main.go`.
|
||||||
|
|
||||||
|
Use `go mod init yourdomain.com/foo/plugin_name` to initiate your plugin. By default the `go.mod` file will be automatically generated by the go compiler. Assuming you are developing Lapwing with its source located on Github, this command would be `go mod init github.com/your_user_name/Lapwing`.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Steps 3: Open plugin folder in IDE
|
||||||
|
|
||||||
|
Now open your preferred IDE or text editor and use your plugin folder as the project folder
|
||||||
|
|
||||||
|
Now, you are ready to start developing Zoraxy plugin!
|
26
docs/plugins/docs/1. Introduction/3. Installing Plugin.md
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
# Installing Plugin
|
||||||
|
|
||||||
|
Last Update: 25/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Install via Plugin Store
|
||||||
|
|
||||||
|
(Work in progress)
|
||||||
|
|
||||||
|
### Manual Install
|
||||||
|
|
||||||
|
The plugin shall be placed inside the `plugins/{{plugin_name}}/` directory where the binary executable name must be matching with the plugin name.
|
||||||
|
|
||||||
|
If you are on Linux, also make sure Zoraxy have the execution permission of the plugin. You can use the following command to enable execution of the plugin binary on Linux with the current user (Assume Zoraxy is run by the current user)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd ./plugins/{{plugin_name}}/
|
||||||
|
chmod +x ./{{plugin_name}}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Sometime plugins might come with additional assets other than the binary file. If that is the case, extract all of the plugins content into the folder with the plugin's name.
|
||||||
|
|
||||||
|
After the folder structure is ready, restart Zoraxy. If you are using systemd for Zoraxy, use `sudo systemctl restart zoraxy` to restart Zoraxy via systemd service.
|
40
docs/plugins/docs/1. Introduction/4. Enable Plugins.md
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
# Enable Plugins
|
||||||
|
|
||||||
|
Last Update: 25/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
To enable and assign a plugin to a certain HTTP Proxy Rule, you will need to do the following steps
|
||||||
|
|
||||||
|
## 1. Create a tag for your HTTP Proxy Rules
|
||||||
|
|
||||||
|
Let say you want to enable debugger on some of your HTTP Proxy Rules. You can do that by first creating a tag in the tag editor. In the example below, we will be using the tag "debug". After adding the tag to the HTTP Proxy rule, you will see something like this.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. Enable Plugin
|
||||||
|
|
||||||
|
Click on the "Enable" button on the plugin which you want to enable
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. Assign Plugin to HTTP Proxy Rule
|
||||||
|
|
||||||
|
Finally, select the tag that you just created in the dropdown menu
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Afterward, you will see the plugin is attached to the target tag
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
It means the plugin is enabled on the HTTP proxy rule
|
||||||
|
|
13
docs/plugins/docs/1. Introduction/5. Viewing Plugin Info.md
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# Viewing Plugin Info
|
||||||
|
|
||||||
|
To view plugin information, you can click on the (i) icon in the plugin list.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Next, a side menu will pop up from the side. Here ,you can see the current Plugin information and runtime values including Working directories and runtime assigned port.
|
||||||
|
|
||||||
|
If you are a developer (which you probably is considering you are reading this doc), you can click on the "developer insight" dropdown to show the capture paths registered by this plugin for debug purposes.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|

|
After Width: | Height: | Size: 39 KiB |
After Width: | Height: | Size: 31 KiB |
After Width: | Height: | Size: 63 KiB |
After Width: | Height: | Size: 24 KiB |
After Width: | Height: | Size: 47 KiB |
After Width: | Height: | Size: 6.9 KiB |
17
docs/plugins/docs/2. Architecture/1. Plugin Architecture.md
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
# Plugin Architecture
|
||||||
|
|
||||||
|
Last Update: 25/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
The Zoraxy Plugin uses a 3 steps approach to get information from plugin, setup the plugin and forward request to plugin. The name of the steps are partially referred from dbus designs as followings.
|
||||||
|
|
||||||
|
1. Introspect
|
||||||
|
2. Configure
|
||||||
|
3. Forwarding
|
||||||
|
|
||||||
|
The overall flow looks like this.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
This design make sure that the Zoraxy plugins do not depends on platform dependent implementations that uses, for example, unix socket. This also avoided protocol that require complex conversion to and from HTTP request (data structure) like gRPC, while making sure the plugin can be cross compile into different CPU architecture or OS environment with little to no effect on its performance.
|
101
docs/plugins/docs/2. Architecture/2. Introspect.md
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
# Introspect
|
||||||
|
|
||||||
|
Last Update: 25/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Introspect, similar to the one in dbus design, is used to get the information from plugin when Zoraxy starts (or manually triggered in development mode or force reload plugin list).
|
||||||
|
|
||||||
|
**This is a pre-defined structure where the plugin must provide to Zoraxy** when the plugin is being started with the `-introspect` flag.
|
||||||
|
|
||||||
|
The introspect structure is defined under the `zoraxy_plugin` library, where both Zoraxy and plugin should use. As of writing, the structure of introspect is like this.
|
||||||
|
|
||||||
|
```go
|
||||||
|
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
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
The introspect provide Zoraxy the required information to start the plugin and how to interact with it. For more details on what those capture settings are for, see "Capture Mode" section.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Introspect Manual Triggering
|
||||||
|
|
||||||
|
To manually test if the introspect return is correct, you can try using the `-introspect` flag on any Zoraxy plugin. You should be able to see an output like so.
|
||||||
|
|
||||||
|
```json
|
||||||
|
$ ./debugger -introspect
|
||||||
|
{
|
||||||
|
"id": "org.aroz.zoraxy.debugger",
|
||||||
|
"name": "Plugin Debugger",
|
||||||
|
"author": "aroz.org",
|
||||||
|
"author_contact": "https://aroz.org",
|
||||||
|
"description": "A debugger for Zoraxy \u003c-\u003e plugin communication pipeline",
|
||||||
|
"url": "https://zoraxy.aroz.org",
|
||||||
|
"type": 0,
|
||||||
|
"version_major": 1,
|
||||||
|
"version_minor": 0,
|
||||||
|
"version_patch": 0,
|
||||||
|
"static_capture_paths": [
|
||||||
|
{
|
||||||
|
"capture_path": "/test_a"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"capture_path": "/test_b"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"static_capture_ingress": "/s_capture",
|
||||||
|
"dynamic_capture_sniff": "/d_sniff",
|
||||||
|
"dynamic_capture_ingress": "/d_capture",
|
||||||
|
"ui_path": "/debug",
|
||||||
|
"subscription_path": "",
|
||||||
|
"subscriptions_events": null
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
18
docs/plugins/docs/2. Architecture/3. Configure.md
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# Configure
|
||||||
|
|
||||||
|
Configure or Configure Spec is the `exec` call where Zoraxy start the plugin. The configure spec JSON structure is defined in `zoraxy_plugin` library.
|
||||||
|
|
||||||
|
As the time of writing, the `ConfigureSpec` only contains information on some basic info.
|
||||||
|
|
||||||
|
```go
|
||||||
|
type ConfigureSpec struct {
|
||||||
|
Port int `json:"port"` //Port to listen
|
||||||
|
RuntimeConst RuntimeConstantValue `json:"runtime_const"` //Runtime constant values
|
||||||
|
//To be expanded
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
The `ConfigureSpec` struct will be parsed to JSON and pass to your plugin via the `-configure=(json payload here)`.
|
||||||
|
|
||||||
|
In your plugin, you can use the `zoraxy_plugin` library to parse it or parse it manually (if you are developing a plugin with other languages).
|
58
docs/plugins/docs/2. Architecture/4. Capture Modes.md
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
# Capture Modes
|
||||||
|
|
||||||
|
As you can see in the Introspect section, there are two types of capture mode in Zoraxy plugin API.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
- Static Capture Mode
|
||||||
|
- Dynamic Capture Mode
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
**Notes: When this document mention the term "endpoint", it means a particular sub-path on the plugin side. For example `/capture` or `/sniff`. In actual implementation, this can be a `http.HandleFunc` or `http.Handle` depends on the plugin implementation.**
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Static Capture Mode
|
||||||
|
|
||||||
|
Static Capture Mode register a static path to Zoraxy, when the plugin is enabled on a certain HTTP proxy rule, all request that matches the static capture registered paths are forwarded to the plugin without asking first. The overall process is shown in the diagram below.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
The main benefit of static capture mode is that the capture paths are stored in radix tree. That means it takes O(logn) time to resolve the path and forward the request. Hence, **this mode is generally faster** if your plugin always listens to a few certain paths for extended functionalities.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Dynamic Capture Mode
|
||||||
|
|
||||||
|
Dynamic Capture Mode register two endpoints to Zoraxy.
|
||||||
|
|
||||||
|
1. DynamicCaptureSniff - The sniffing endpoint where Zoraxy will first ask if the plugin want to handle this request
|
||||||
|
2. DynamicCaptureIngress - The handling endpoint, where if the plugin reply the sniffing with "YES", Zoraxy forward the incoming request to this plugin at this defined endpoint.
|
||||||
|
|
||||||
|
The whole process will takes a few request exchange between plugin and Zoraxy core. Since both of them are communicating via the loopback interface, speed should not be too big of a concern here.
|
||||||
|
|
||||||
|
The request handling flow is shown in the diagram below.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Once Zoraxy receive a request from a client that matches one of the HTTP Proxy Rule, Zoraxy will forward the request header to all the plugins that matches the following criteria
|
||||||
|
|
||||||
|
1. The plugin is assigned to a tag that is currently attached to the given HTTP Proxy that the request is coming through
|
||||||
|
2. The plugin is enabled and running
|
||||||
|
3. The plugin has registered its dynamic capture sniffing endpoint in Introspect
|
||||||
|
|
||||||
|
Then the plugin `/sniff` endpoint will receive some basic header information about the request, and response with `SniffResultAccpet` or `SniffResultSkip` to accept or reject handling such request. The response are defined in `zoraxy_plugin` as a public type where you can access with `zoraxy_plugin.SniffresultAccept` and `zoraxy_plugin.SniffResultSkip` respectively.
|
||||||
|
|
||||||
|
Note that this shall only be used if static capture mode cannot satisfy your needs in implementation the feature you want, as **dynamic capture is way slower than static capture mode**.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Mixing Capture Modes
|
||||||
|
|
||||||
|
It is possible for you to mix both Static and Capture modes if that is what you want. A few thing you need to know about mixing both mode in single plugin
|
||||||
|
|
||||||
|
1. Static capture mode has higher priority than dynamic capture mode across all plugins. That means if you have a request that matches Plugin A's static capture path and Plugin B's dynamic capture, the request will be first handled by Plugin A
|
||||||
|
2. The same plugin can register both static and dynamic capture modes. Similar to item (1), if the request has already captured by your static capture path, Zoraxy will not proceed and forward the request header to your dynamic sniffing endpoint.
|
||||||
|
3. In case there is a collision in static capture paths between two plugins, the longest one will have priority. For example, if Plugin A registered `/foo` and Plugin B registered `/foo/bar`, when a request to `/foo/bar/teacat` enter Zoraxy, Plugin B is used for handling such request.
|
||||||
|
|
23
docs/plugins/docs/2. Architecture/5. Plugin UI.md
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
# Plugin UI
|
||||||
|
|
||||||
|
Last Update: 25/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
A plugin can optionally expose a Web UI interface for user configuration.
|
||||||
|
|
||||||
|
**A plugin must provide a UI, as it is part of the control mechanism of the plugin life cycle. (i.e. Zoraxy use the plugin UI HTTP server to communicate with the plugin for control signals)** As plugin installed via plugin store provides limited ways for a user to configure the plugin, the plugin web UI will be the best way for user to setup your plugin.
|
||||||
|
|
||||||
|
## Plugin Web UI Access
|
||||||
|
|
||||||
|
If a plugin provide a Web UI endpoint for Zoraxy during the introspect process, a new item will be shown in the Plugins section on Zoraxy side menu. Below is an example of the Web UI of UPnP Port Forwarder plugin.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## Front-end Developer Notes
|
||||||
|
|
||||||
|
The Web UI is implemented as a reverse proxy and embed in an iframe. So you do not need to handle CORS issues with the web UI (as it will be proxy internally by Zoraxy as exposed as something like a virtual directory mounted website).
|
||||||
|
|
||||||
|
However, the plugin web UI is exposed via the path `/plugin.ui/{{plugin_uuid}}/`, for example, `/plugin.ui/org.aroz.zoraxy.plugins.upnp/`. **When developing the plugin web UI, do not use absolute path for any resources used in the HTML file**, unless you are trying to re-use Zoraxy components like css or image elements stored in Zoraxy embedded web file system (e.g. `/img/logo.svg`).
|
15
docs/plugins/docs/2. Architecture/6. Compile a Plugin.md
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
# Compile a Plugin
|
||||||
|
|
||||||
|
A plugin is basically a go program with a HTTP Server / Listener. The steps required to build a plugin is identical as building a ordinary go program.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Assuming you are currently inside the root folder of your plugin
|
||||||
|
go mod tidy
|
||||||
|
go build
|
||||||
|
|
||||||
|
# Validate if the plugin is correctly build using -introspect flag
|
||||||
|
./{{your_plugin_name}} -introspect
|
||||||
|
|
||||||
|
# You should see your plugin information printed to STDOUT as JSON string
|
||||||
|
```
|
||||||
|
|
After Width: | Height: | Size: 194 KiB |
After Width: | Height: | Size: 190 KiB |
After Width: | Height: | Size: 171 KiB |
After Width: | Height: | Size: 70 KiB |
329
docs/plugins/docs/3. Basic Examples/1. Hello World.md
Normal file
@@ -0,0 +1,329 @@
|
|||||||
|
# Hello World!
|
||||||
|
|
||||||
|
Last Update: 25/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Let start with a really simple Hello World plugin. This only function of this plugin is to print "Hello World" in the plugin web UI.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. Name your plugin
|
||||||
|
|
||||||
|
First things first, give your plugin a name. In this example, we are using the name "helloworld".
|
||||||
|
|
||||||
|
**Plugin name cannot contain space or special characters**, so you must use a file name that satisfies the requirement. Dont worry, the plugin file name is not the same as the plugin display name in the introspect.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. Create the plugin folder
|
||||||
|
|
||||||
|
If your zoraxy root folder do not contains a folder named "plugins", it might implies that your Zoraxy is freshly clone from Github. **You will need to build and run it once to start working on your plugin**, so if you have a newly cloned source code of Zoraxy, do the followings.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git clone https://github.com/tobychui/zoraxy
|
||||||
|
cd src
|
||||||
|
go mod tidy
|
||||||
|
go build
|
||||||
|
sudo ./zoraxy
|
||||||
|
```
|
||||||
|
|
||||||
|
Afterward, create a plugin folder under your Zoraxy development environment that is exactly matching your plugin name. In the above example, the folder name should be "helloworld".
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Assume you are already inside the src/ folder
|
||||||
|
mkdir helloworld
|
||||||
|
cd ./helloworld
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. Create a go project
|
||||||
|
|
||||||
|
Similar to any Go project, you can start by creating a `main.go` file. Next, you would want to let the go compiler knows your plugin name so when generating a binary file, it knows what to name it. This can be done via using the `go mod init` command.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
touch main.go
|
||||||
|
go mod init example.com/zoraxy/helloworld
|
||||||
|
ls
|
||||||
|
# After you are done, you should see the followings
|
||||||
|
# go.mod main.go
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. Copy the Zoraxy plugin lib from Zoraxy source code
|
||||||
|
|
||||||
|
Locate the Zoraxy plugin library from the Zoraxy source code. You can find the `zoraxy_plugin` Go module under `src/mod/plugins/zoraxy_plugin`
|
||||||
|
|
||||||
|
Copy the `zoraxy_plugin` folder from the Zoraxy source code mod folder into the your plugin’s mod folder. Let assume you use the same mod folder name as Zoraxy as `mod`, then your copied library path should be `plugins/helloworld/mod/zoraxy_plugin`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mkdir ./mod
|
||||||
|
cp -r "mod/plugins/zoraxy_plugin" ./mod/
|
||||||
|
ls ./mod/zoraxy_plugin/
|
||||||
|
# You should see something like this (might be different in future versions)
|
||||||
|
# dev_webserver.go dynamic_router.go embed_webserver.go README.txt static_router.go zoraxy_plugin.go
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 5. Create a web resources folder
|
||||||
|
|
||||||
|
Lets create a www folder and put all our web resources, we need to create an `index.html` file as our plugin web ui homepage. This can be done by creating a HTML file in the www folder.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Assuming you are currently in the src/plugins/helloworld/ folder
|
||||||
|
mkdir www
|
||||||
|
cd www
|
||||||
|
touch index.html
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
And here is an example `index.html` file that uses the Zoraxy internal resources like css and dark theme toggle mechanism. That csrf token template is not straightly needed in this example as helloworld plugin do not make any POST request to Zoraxy webmin interface, but it might come in handy later.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<!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>
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 6. Creating a handler for Introspect
|
||||||
|
|
||||||
|
To create a handler for introspect, you can first start your plugin with a few constants.
|
||||||
|
|
||||||
|
1. Plugin ID, this must be unique. You can use a domain you own like `com.example.helloworld`
|
||||||
|
2. UI Path, for now we uses "/" as this plugin do not have any other endpoints, so we can use the whole root just for web UI
|
||||||
|
3. Web root, for trimming off from the embedded web folder so when user can visit your `index.html` by accessing `/` instead of needing to navigate to `/www`
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
After you have defined these constant, we can use `plugin.ServeAndRecvSpec` function to handle the handshake between Zoraxy and your plugin.
|
||||||
|
|
||||||
|
```go
|
||||||
|
const (
|
||||||
|
PLUGIN_ID = "com.example.helloworld"
|
||||||
|
UI_PATH = "/"
|
||||||
|
WEB_ROOT = "/www"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main(){
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
**Notes: If some post processing is needed between Introspect and Configure, you can use two seperate function to handle the first start and the second starting of your plugin. The "separated version" of `ServeAndRecvSpec` is defined as ` ServeIntroSpect(pluginSpect *IntroSpect) ` and `RecvConfigureSpec() (*ConfigureSpec, error)`. See `zoraxy_plugin.go` for more information.**
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 7. Creating a web server from embedded web fs
|
||||||
|
|
||||||
|
After that, we need to create a web server to serve our plugin UI to Zoraxy via HTTP. This can be done via the `http.FileServer` but for simplicity and ease of upgrade, the Zoraxy plugin library provided an easy to use embedded web FS server API for plugin developers.
|
||||||
|
|
||||||
|
To use the Zoraxy plugin embedded web server, you first need to embed your web fs into Zoraxy as such.
|
||||||
|
|
||||||
|
```go
|
||||||
|
import (
|
||||||
|
_ "embed"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
plugin "example.com/zoraxy/helloworld/mod/zoraxy_plugin"
|
||||||
|
)
|
||||||
|
|
||||||
|
//go:embed www/*
|
||||||
|
var content embed.FS
|
||||||
|
```
|
||||||
|
|
||||||
|
Then call to the `NewPluginEmbedUIRouter` to create a new UI router from the embedded Fs.
|
||||||
|
|
||||||
|
```go
|
||||||
|
// 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)
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Here is the tricky part. since not all platform support cross process signaling, Zoraxy plugin uses HTTP request to request a plugin to shutdown. The `embedWebRouter` object has a function named `RegisterTerminateHandler` where you can easily use this function to register actions that needed to be done before shutdown.
|
||||||
|
|
||||||
|
```go
|
||||||
|
embedWebRouter.RegisterTerminateHandler(func() {
|
||||||
|
// Do cleanup here if needed
|
||||||
|
fmt.Println("Hello World Plugin Exited")
|
||||||
|
}, nil)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Notes: This is a blocking function. That is why Zoraxy has a build-in timeout context where if the terminate request takes more than 3 seconds, the plugin process will be treated as "freezed" and forcefully terminated. So please make sure the terminate handler complete its shutdown procedures within 3 seconds.**
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 8. Register & Serve the Web UI
|
||||||
|
|
||||||
|
After you have created a embedded web router, you can register it to the UI PATH as follows.
|
||||||
|
|
||||||
|
```go
|
||||||
|
// 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)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
As this is just the standard golang net/http package, you can of course add more Function Handlers to it based on your needs. There are something that you need to know about adding API endpoints, we will discuss this in the later sections.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 9. Build and Test
|
||||||
|
|
||||||
|
After saving the `main.go` file, you can now build your plugin with `go build`. It should generate the plugin in your platform architecture and OS. If you are on Linux, it will be `helloworld` and if you are on Windows, it will be `helloworld.exe`.
|
||||||
|
|
||||||
|
After you are done, restart Zoraxy and enable your plugin in the Plugin List. Now you can test and debug your plugin with your HTTP Proxy Rules. All the STDOUT and STDERR of your plugin will be forwarded to the STDOUT of Zoraxy as well as the log file.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
**Tips**
|
||||||
|
|
||||||
|
You can also enable the Developer Option - Plugin Auto Reload function if you are too lazy to restart Zoraxy everytime the plugin binary changed.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 10. Full Code
|
||||||
|
|
||||||
|
This is the full code of the helloworld plugin main.go file.
|
||||||
|
|
||||||
|
```go
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
493
docs/plugins/docs/3. Basic Examples/2. RESTful Example.md
Normal file
@@ -0,0 +1,493 @@
|
|||||||
|
# RESTful API Call in Web UI
|
||||||
|
Last Update: 29/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
When developing a UI for your plugin, sometime you might need to make RESTFUL API calls to your plugin backend for setting up something or getting latest information from your plugin. In this example, I will show you how to create a plugin with RESTful api call capabilities with the embedded web server and the custom `cjax` function.
|
||||||
|
|
||||||
|
**Notes: This example assumes you have basic understanding on how to use jQuery `ajax` request.**
|
||||||
|
|
||||||
|
Lets get started!
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. Create the plugin folder structures
|
||||||
|
|
||||||
|
This step is identical to the Hello World example, where you create a plugin folder with the required go project structure in the folder. Please refer to the Hello World example section 1 to 5 for details.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. Create Introspect
|
||||||
|
|
||||||
|
This is quite similar to the Hello World example as well, but we are changing some of the IDs to match what we want to do in this plugin.
|
||||||
|
|
||||||
|
```go
|
||||||
|
runtimeCfg, err := plugin.ServeAndRecvSpec(&plugin.IntroSpect{
|
||||||
|
ID: "com.example.restful-example",
|
||||||
|
Name: "Restful Example",
|
||||||
|
Author: "foobar",
|
||||||
|
AuthorContact: "admin@example.com",
|
||||||
|
Description: "A simple demo for making RESTful API calls in 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)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. Create an embedded web server with handlers
|
||||||
|
|
||||||
|
In this step, we create a basic embedded web file handler similar to the Hello World example, however, we will need to add a `http.HandleFunc` to the plugin so our front-end can request and communicate with the backend.
|
||||||
|
|
||||||
|
```go
|
||||||
|
embedWebRouter := plugin.NewPluginEmbedUIRouter(PLUGIN_ID, &content, WEB_ROOT, UI_PATH)
|
||||||
|
embedWebRouter.RegisterTerminateHandler(func() {
|
||||||
|
fmt.Println("Restful-example Exited")
|
||||||
|
}, nil)
|
||||||
|
|
||||||
|
//Register a simple API endpoint that will echo the request body
|
||||||
|
// Since we are using the default http.ServeMux, we can register the handler directly with the last
|
||||||
|
// parameter as nil
|
||||||
|
embedWebRouter.HandleFunc("/api/echo", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
//Some handler code here
|
||||||
|
}, nil)
|
||||||
|
```
|
||||||
|
|
||||||
|
The `embedWebRouter.HandleFunc` last parameter is the `http.Mux`, where if you have multiple web server listening interface, you can fill in different Mux based on your implementation. On of the examples is that, when you are developing a static web server plugin, where you need a dedicated HTTP listening endpoint that is not the one Zoraxy assigned to your plugin, you need to create two http.Mux and assign one of them for Zoraxy plugin UI purpose.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. Modify the front-end HTML file to make request to backend
|
||||||
|
|
||||||
|
To make a RESTFUL API to your plugin, **you must use relative path in your request URL**.
|
||||||
|
|
||||||
|
Absolute path that start with `/` is only use for accessing Zoraxy resouces. For example, when you access `/img/logo.svg`, Zoraxy webmin HTTP router will return the logo of Zoraxy for you instead of `/plugins/your_plugin_name/{your_web_root}/img/logo.svg`.
|
||||||
|
|
||||||
|
### Making GET request
|
||||||
|
|
||||||
|
Making GET request is similar to what you would do in ordinary web development, but only limited to relative paths like `./api/foo/bar` instead. Here is an example on a front-end and back-end implementation of a simple "echo" API.
|
||||||
|
|
||||||
|
The API logic is simple: when you make a GET request to the API with `?name=foobar`, it returns `Hello foobar`. Here is the backend implementation in your plugin code.
|
||||||
|
|
||||||
|
```go
|
||||||
|
embedWebRouter.HandleFunc("/api/echo", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
// This is a simple echo API that will return the request body as response
|
||||||
|
name := r.URL.Query().Get("name")
|
||||||
|
if name == "" {
|
||||||
|
http.Error(w, "Missing 'name' query parameter", http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
response := map[string]string{"message": fmt.Sprintf("Hello %s", name)}
|
||||||
|
if err := json.NewEncoder(w).Encode(response); err != nil {
|
||||||
|
http.Error(w, "Failed to encode response", http.StatusInternalServerError)
|
||||||
|
}
|
||||||
|
}, nil)
|
||||||
|
```
|
||||||
|
|
||||||
|
And here is the front-end code in your HTML file
|
||||||
|
|
||||||
|
```html
|
||||||
|
<!-- The example below show how HTTP GET is used with Zoraxy plugin -->
|
||||||
|
<h3>Echo Test (HTTP GET)</h3>
|
||||||
|
<div class="ui form">
|
||||||
|
<div class="field">
|
||||||
|
<label for="nameInput">Enter your name:</label>
|
||||||
|
<input type="text" id="nameInput" placeholder="Your name">
|
||||||
|
</div>
|
||||||
|
<button class="ui button primary" id="sendRequestButton">Send Request</button>
|
||||||
|
<div class="ui message" id="responseMessage" style="display: none;"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
document.getElementById('sendRequestButton').addEventListener('click', function() {
|
||||||
|
const name = document.getElementById('nameInput').value;
|
||||||
|
if (name.trim() === "") {
|
||||||
|
alert("Please enter a name.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Note the relative path is used here!
|
||||||
|
// GET do not require CSRF token, so you can use $.ajax directly
|
||||||
|
// or $.cjax (defined in /script/utils.js) to make GET request
|
||||||
|
$.ajax({
|
||||||
|
url: `./api/echo`,
|
||||||
|
type: 'GET',
|
||||||
|
data: { name: name },
|
||||||
|
success: function(data) {
|
||||||
|
console.log('Response:', data.message);
|
||||||
|
$('#responseMessage').text(data.message).show();
|
||||||
|
},
|
||||||
|
error: function(xhr, status, error) {
|
||||||
|
console.error('Error:', error);
|
||||||
|
$('#responseMessage').text('An error occurred while processing your request.').show();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Making POST request
|
||||||
|
|
||||||
|
Making POST request is also similar to GET request, except when making the request, you will need pass the CSRF-Token with the payload. This is required due to security reasons (See [#267](https://github.com/tobychui/zoraxy/issues/267) for more details).
|
||||||
|
|
||||||
|
Since the CSRF validation is done by Zoraxy, your plugin backend code can be implemented just like an ordinary handler. Here is an example POST handling function that receive a FORM POST and print it in an HTML response.
|
||||||
|
|
||||||
|
```go
|
||||||
|
embedWebRouter.HandleFunc("/api/post", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.Method != http.MethodPost {
|
||||||
|
http.Error(w, "Invalid request method", http.StatusMethodNotAllowed)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := r.ParseForm(); err != nil {
|
||||||
|
http.Error(w, "Failed to parse form data", http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for key, values := range r.PostForm {
|
||||||
|
for _, value := range values {
|
||||||
|
// Generate a simple HTML response
|
||||||
|
w.Header().Set("Content-Type", "text/html")
|
||||||
|
fmt.Fprintf(w, "%s: %s<br>", key, value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, nil)
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
For the front-end, you will need to use the `$.cjax` function implemented in Zoraxy `utils.js` file. You can include this file by adding these two lines to your HTML file.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<script src="/script/jquery-3.6.0.min.js"></script>
|
||||||
|
<script src="/script/utils.js"></script>
|
||||||
|
<!- More lines here -->
|
||||||
|
<!-- The example below shows how form post can be used in plugin -->
|
||||||
|
<h3>Form Post Test (HTTP POST)</h3>
|
||||||
|
<div class="ui form">
|
||||||
|
<div class="field">
|
||||||
|
<label for="postNameInput">Name:</label>
|
||||||
|
<input type="text" id="postNameInput" placeholder="Your name">
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<label for="postAgeInput">Age:</label>
|
||||||
|
<input type="number" id="postAgeInput" placeholder="Your age">
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<label>Gender:</label>
|
||||||
|
<div class="ui checkbox">
|
||||||
|
<input type="checkbox" id="genderMale" name="gender" value="Male">
|
||||||
|
<label for="genderMale">Male</label>
|
||||||
|
</div>
|
||||||
|
<div class="ui checkbox">
|
||||||
|
<input type="checkbox" id="genderFemale" name="gender" value="Female">
|
||||||
|
<label for="genderFemale">Female</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button class="ui button primary" id="postRequestButton">Send</button>
|
||||||
|
<div class="ui message" id="postResponseMessage" style="display: none;"></div>
|
||||||
|
</div>
|
||||||
|
```
|
||||||
|
|
||||||
|
After that, you can call to the `$.cjax` function just like what you would usually do with the `$ajax` function.
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// .cjax (defined in /script/utils.js) is used to make POST request with CSRF token support
|
||||||
|
// alternatively you can use $.ajax with CSRF token in headers
|
||||||
|
// the header is named "X-CSRF-Token" and the value is taken from the head
|
||||||
|
// meta tag content (i.e. <meta name="zoraxy.csrf.Token" content="{{.csrfToken}}">)
|
||||||
|
$.cjax({
|
||||||
|
url: './api/post',
|
||||||
|
type: 'POST',
|
||||||
|
data: { name: name, age: age, gender: gender },
|
||||||
|
success: function(data) {
|
||||||
|
console.log('Response:', data);
|
||||||
|
$('#postResponseMessage').html(data).show();
|
||||||
|
},
|
||||||
|
error: function(xhr, status, error) {
|
||||||
|
console.error('Error:', error);
|
||||||
|
$('#postResponseMessage').text('An error occurred while processing your request.').show();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
### POST Request with Vanilla JS
|
||||||
|
|
||||||
|
It is possible to make POST request with Vanilla JS. Note that you will need to populate the csrf-token field yourself to make the request pass through the plugin UI request router in Zoraxy. Here is a basic example on how it could be done.
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
fetch('./api/post', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
'X-CSRF-Token': csrfToken // Include the CSRF token in the headers
|
||||||
|
},
|
||||||
|
body: JSON.stringify({{your_data_here}})
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 5. Full Code
|
||||||
|
|
||||||
|
Here is the full code of the RESTFUL example for reference.
|
||||||
|
|
||||||
|
Front-end (`plugins/restful-example/www/index.html`)
|
||||||
|
|
||||||
|
```html
|
||||||
|
<!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>RESTful Example</title>
|
||||||
|
<style>
|
||||||
|
body {
|
||||||
|
background-color: var(--theme_bg_primary);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<!-- Dark theme script must be included after body tag-->
|
||||||
|
<link rel="stylesheet" href="/darktheme.css">
|
||||||
|
<script src="/script/darktheme.js"></script>
|
||||||
|
<br>
|
||||||
|
<div class="standardContainer">
|
||||||
|
<div class="ui container">
|
||||||
|
<h1>RESTFul API Example</h1>
|
||||||
|
<!-- The example below show how HTTP GET is used with Zoraxy plugin -->
|
||||||
|
<h3>Echo Test (HTTP GET)</h3>
|
||||||
|
<div class="ui form">
|
||||||
|
<div class="field">
|
||||||
|
<label for="nameInput">Enter your name:</label>
|
||||||
|
<input type="text" id="nameInput" placeholder="Your name">
|
||||||
|
</div>
|
||||||
|
<button class="ui button primary" id="sendRequestButton">Send Request</button>
|
||||||
|
<div class="ui message" id="responseMessage" style="display: none;"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
document.getElementById('sendRequestButton').addEventListener('click', function() {
|
||||||
|
const name = document.getElementById('nameInput').value;
|
||||||
|
if (name.trim() === "") {
|
||||||
|
alert("Please enter a name.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Note the relative path is used here!
|
||||||
|
// GET do not require CSRF token, so you can use $.ajax directly
|
||||||
|
// or $.cjax (defined in /script/utils.js) to make GET request
|
||||||
|
$.ajax({
|
||||||
|
url: `./api/echo`,
|
||||||
|
type: 'GET',
|
||||||
|
data: { name: name },
|
||||||
|
success: function(data) {
|
||||||
|
console.log('Response:', data.message);
|
||||||
|
$('#responseMessage').text(data.message).show();
|
||||||
|
},
|
||||||
|
error: function(xhr, status, error) {
|
||||||
|
console.error('Error:', error);
|
||||||
|
$('#responseMessage').text('An error occurred while processing your request.').show();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
<!-- The example below shows how form post can be used in plugin -->
|
||||||
|
<h3>Form Post Test (HTTP POST)</h3>
|
||||||
|
<div class="ui form">
|
||||||
|
<div class="field">
|
||||||
|
<label for="postNameInput">Name:</label>
|
||||||
|
<input type="text" id="postNameInput" placeholder="Your name">
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<label for="postAgeInput">Age:</label>
|
||||||
|
<input type="number" id="postAgeInput" placeholder="Your age">
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<label>Gender:</label>
|
||||||
|
<div class="ui checkbox">
|
||||||
|
<input type="checkbox" id="genderMale" name="gender" value="Male">
|
||||||
|
<label for="genderMale">Male</label>
|
||||||
|
</div>
|
||||||
|
<div class="ui checkbox">
|
||||||
|
<input type="checkbox" id="genderFemale" name="gender" value="Female">
|
||||||
|
<label for="genderFemale">Female</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button class="ui button primary" id="postRequestButton">Send</button>
|
||||||
|
<div class="ui message" id="postResponseMessage" style="display: none;"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
document.getElementById('postRequestButton').addEventListener('click', function() {
|
||||||
|
const name = document.getElementById('postNameInput').value;
|
||||||
|
const age = document.getElementById('postAgeInput').value;
|
||||||
|
const genderMale = document.getElementById('genderMale').checked;
|
||||||
|
const genderFemale = document.getElementById('genderFemale').checked;
|
||||||
|
|
||||||
|
if (name.trim() === "" || age.trim() === "" || (!genderMale && !genderFemale)) {
|
||||||
|
alert("Please fill out all fields.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const gender = genderMale ? "Male" : "Female";
|
||||||
|
|
||||||
|
// .cjax (defined in /script/utils.js) is used to make POST request with CSRF token support
|
||||||
|
// alternatively you can use $.ajax with CSRF token in headers
|
||||||
|
// the header is named "X-CSRF-Token" and the value is taken from the head
|
||||||
|
// meta tag content (i.e. <meta name="zoraxy.csrf.Token" content="{{.csrfToken}}">)
|
||||||
|
$.cjax({
|
||||||
|
url: './api/post',
|
||||||
|
type: 'POST',
|
||||||
|
data: { name: name, age: age, gender: gender },
|
||||||
|
success: function(data) {
|
||||||
|
console.log('Response:', data);
|
||||||
|
$('#postResponseMessage').html(data).show();
|
||||||
|
},
|
||||||
|
error: function(xhr, status, error) {
|
||||||
|
console.error('Error:', error);
|
||||||
|
$('#postResponseMessage').text('An error occurred while processing your request.').show();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Backend (`plugins/restful-example/main.go`)
|
||||||
|
|
||||||
|
```go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"embed"
|
||||||
|
_ "embed"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
plugin "example.com/zoraxy/restful-example/mod/zoraxy_plugin"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
PLUGIN_ID = "com.example.restful-example"
|
||||||
|
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.restful-example",
|
||||||
|
Name: "Restful Example",
|
||||||
|
Author: "foobar",
|
||||||
|
AuthorContact: "admin@example.com",
|
||||||
|
Description: "A simple demo for making RESTful API calls in 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("Restful-example Exited")
|
||||||
|
}, nil)
|
||||||
|
|
||||||
|
//Register a simple API endpoint that will echo the request body
|
||||||
|
// Since we are using the default http.ServeMux, we can register the handler directly with the last
|
||||||
|
// parameter as nil
|
||||||
|
embedWebRouter.HandleFunc("/api/echo", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
// This is a simple echo API that will return the request body as response
|
||||||
|
name := r.URL.Query().Get("name")
|
||||||
|
if name == "" {
|
||||||
|
http.Error(w, "Missing 'name' query parameter", http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
response := map[string]string{"message": fmt.Sprintf("Hello %s", name)}
|
||||||
|
if err := json.NewEncoder(w).Encode(response); err != nil {
|
||||||
|
http.Error(w, "Failed to encode response", http.StatusInternalServerError)
|
||||||
|
}
|
||||||
|
}, nil)
|
||||||
|
|
||||||
|
// Here is another example of a POST API endpoint that will echo the form data
|
||||||
|
// This will handle POST requests to /api/post and return the form data as response
|
||||||
|
embedWebRouter.HandleFunc("/api/post", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.Method != http.MethodPost {
|
||||||
|
http.Error(w, "Invalid request method", http.StatusMethodNotAllowed)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := r.ParseForm(); err != nil {
|
||||||
|
http.Error(w, "Failed to parse form data", http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for key, values := range r.PostForm {
|
||||||
|
for _, value := range values {
|
||||||
|
// Generate a simple HTML response
|
||||||
|
w.Header().Set("Content-Type", "text/html")
|
||||||
|
fmt.Fprintf(w, "%s: %s<br>", key, value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, nil)
|
||||||
|
|
||||||
|
// Serve the restful-example page in the www folder
|
||||||
|
http.Handle(UI_PATH, embedWebRouter.Handler())
|
||||||
|
fmt.Println("Restful-example 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)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
What you should expect to see if everything is correctly loaded and working in Zoray
|
||||||
|
|
||||||
|

|
263
docs/plugins/docs/3. Basic Examples/3. Static Capture Example.md
Normal file
@@ -0,0 +1,263 @@
|
|||||||
|
# Static Capture Example
|
||||||
|
Last Update: 29/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
This example demonstrates how to use static capture in Zoraxy plugins. Static capture allows you to define specific paths that will be intercepted by your plugin, enabling custom handling of requests to those paths.
|
||||||
|
|
||||||
|
**Notes: This example assumes you have already read Hello World example.**
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. Create the plugin folder structure
|
||||||
|
|
||||||
|
Follow the same steps as the Hello World example to set up the plugin folder structure. Refer to the Hello World example sections 1 to 5 for details.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. Define Introspect
|
||||||
|
|
||||||
|
The introspect configuration specifies the static capture paths and ingress for your plugin.
|
||||||
|
|
||||||
|
```go
|
||||||
|
runtimeCfg, err := plugin.ServeAndRecvSpec(&plugin.IntroSpect{
|
||||||
|
ID: "org.aroz.zoraxy.static-capture-example",
|
||||||
|
Name: "Static Capture Example",
|
||||||
|
Author: "aroz.org",
|
||||||
|
AuthorContact: "https://aroz.org",
|
||||||
|
Description: "An example for showing how static capture works in Zoraxy.",
|
||||||
|
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",
|
||||||
|
UIPath: UI_PATH,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Note the `StaticCapturePaths`. These are the paths that you want to capture in your plugin. These paths will be registered to Zoraxy and when a user have request that matches these paths (including subpaths), the request will get forwarded to your plugin. In this example, we are intercepting the `/test_a` and `test_b` sub-path.
|
||||||
|
|
||||||
|
We also defined a new value named `StaticCaptureIngress`. This is to tell Zoraxy that "if you receive requests that matches the above Static capture paths, please forward the request to this endpoint". In this example, this plugin asked Zoraxy to forward th HTTP traffic to `/s_capture` if anything is matched.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. Register Static Capture Handlers
|
||||||
|
|
||||||
|
Static capture handlers are used to process requests to the defined paths. Similar to ordinary http.HandleFunc, you can register `http.HandleFunc` as follows.
|
||||||
|
|
||||||
|
```go
|
||||||
|
pathRouter := plugin.NewPathRouter()
|
||||||
|
|
||||||
|
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) {
|
||||||
|
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)
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
The `SetDefaultHandler` is used to handle exceptions where a request is forwarded to your plugin but it cannot be handled by any of your registered path handlers. This is usually an implementation bug on the plugin side and you can add some help message or debug log to this function if needed.
|
||||||
|
|
||||||
|
The `RegisterStaticCaptureHandle` is used to register the static capture ingress endpoint, so Zoraxy knows where to forward the HTTP request when it thinks your plugin shall be the one handling the request. In this example, `/s_capture` is used for static capture endpoint.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. Implement Handlers
|
||||||
|
|
||||||
|
Here are examples of handlers for the captured paths:
|
||||||
|
|
||||||
|
### Handler for `/test_a`
|
||||||
|
|
||||||
|
```go
|
||||||
|
func HandleCaptureA(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.Header().Set("Content-Type", "text/html")
|
||||||
|
w.Write([]byte("This request is captured by A handler!<br>Request URI: " + r.URL.String()))
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Handler for `/test_b`
|
||||||
|
|
||||||
|
```go
|
||||||
|
func HandleCaptureB(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.Header().Set("Content-Type", "text/html")
|
||||||
|
w.Write([]byte("This request is captured by the B handler!<br>Request URI: " + r.URL.String()))
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
When the user request any HTTP Proxy Rule with the matching path, these two handlers will response to the request and return the hardcoded string above. Again, this is just for demonstration purpose and you should implement your functions here.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 5. Render Debug UI
|
||||||
|
|
||||||
|
The debug UI provides a simple interface for testing and inspecting requests.
|
||||||
|
|
||||||
|
```go
|
||||||
|
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")
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This is technically not related to static capturing, but it is really helpful to have a UI to help with printing debug information. You can access the page rendered by this function in the Zoraxy plugin menu. This should be replaced with the embedded web fs used in the Hello world example after the development is completed.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## 6. Full Code
|
||||||
|
|
||||||
|
Here is the complete code for the static capture example:
|
||||||
|
|
||||||
|
```go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"sort"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
plugin "example.com/zoraxy/static-capture-example/mod/zoraxy_plugin"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
PLUGIN_ID = "org.aroz.zoraxy.static-capture-example"
|
||||||
|
UI_PATH = "/ui"
|
||||||
|
STATIC_CAPTURE_INGRESS = "/s_capture"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
runtimeCfg, err := plugin.ServeAndRecvSpec(&plugin.IntroSpect{
|
||||||
|
ID: PLUGIN_ID,
|
||||||
|
Name: "Static Capture Example",
|
||||||
|
Author: "aroz.org",
|
||||||
|
AuthorContact: "https://aroz.org",
|
||||||
|
Description: "An example for showing how static capture works in Zoraxy.",
|
||||||
|
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: STATIC_CAPTURE_INGRESS,
|
||||||
|
UIPath: UI_PATH,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
pathRouter := plugin.NewPathRouter()
|
||||||
|
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) {
|
||||||
|
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)
|
||||||
|
|
||||||
|
http.HandleFunc(UI_PATH+"/", RenderDebugUI)
|
||||||
|
fmt.Println("Static path capture example started at http://127.0.0.1:" + strconv.Itoa(runtimeCfg.Port))
|
||||||
|
http.ListenAndServe("127.0.0.1:"+strconv.Itoa(runtimeCfg.Port), nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func HandleCaptureA(w http.ResponseWriter, r *http.Request) {
|
||||||
|
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) {
|
||||||
|
w.Header().Set("Content-Type", "text/html")
|
||||||
|
w.Write([]byte("This request is captured by the B handler!<br>Request URI: " + r.URL.String()))
|
||||||
|
}
|
||||||
|
|
||||||
|
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")
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 7. Expected Output
|
||||||
|
|
||||||
|
To enable the plugin, add the plugin to one of the tags and assign the tag to your HTTP Proxy Rule. Here is an example of assigning the plugin to the "debug" tag and assign it to a localhost loopback HTTP proxy rule.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
When the plugin is running, requests to `/test_a` and `/test_b` will be intercepted by their respective handlers. **Requests to other paths will not pass through your plugin and will be handled by the default upstream server set by the HTTP proxy Rule.**
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Example terminal output for requesting `/test_a`:
|
||||||
|
|
||||||
|
```
|
||||||
|
This request is captured by A handler!
|
||||||
|
Request URI: /test_a
|
||||||
|
```
|
||||||
|
|
||||||
|
Example output for requesting `/test_b`:
|
||||||
|
```
|
||||||
|
This request is captured by the B handler!
|
||||||
|
Request URI: /test_b
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Enjoy exploring static capture in Zoraxy!
|
||||||
|
|
@@ -0,0 +1,352 @@
|
|||||||
|
# Dynamic Capture Example
|
||||||
|
Last Update: 29/05/2025
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
|
||||||
|
This example demonstrates how to use dynamic capture in Zoraxy plugins. Dynamic capture allows you to intercept requests based on real-time conditions, so you can program your plugin in a way that it can decided if it want to handle the request or not.
|
||||||
|
|
||||||
|
**Notes: This example assumes you have already read Hello World and Stataic Capture Example.**
|
||||||
|
|
||||||
|
Lets dive in!
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. Create the plugin folder structure
|
||||||
|
|
||||||
|
Follow the same steps as the Hello World example to set up the plugin folder structure. Refer to the Hello World example sections 1 to 5 for details.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. Define Introspect
|
||||||
|
|
||||||
|
The introspect configuration specifies the dynamic capture sniff and ingress paths for your plugin.
|
||||||
|
|
||||||
|
```go
|
||||||
|
runtimeCfg, err := plugin.ServeAndRecvSpec(&plugin.IntroSpect{
|
||||||
|
ID: "org.aroz.zoraxy.dynamic-capture-example",
|
||||||
|
Name: "Dynamic Capture Example",
|
||||||
|
Author: "aroz.org",
|
||||||
|
AuthorContact: "https://aroz.org",
|
||||||
|
Description: "This is an example plugin for Zoraxy that demonstrates how to use dynamic captures.",
|
||||||
|
URL: "https://zoraxy.aroz.org",
|
||||||
|
Type: plugin.PluginType_Router,
|
||||||
|
VersionMajor: 1,
|
||||||
|
VersionMinor: 0,
|
||||||
|
VersionPatch: 0,
|
||||||
|
|
||||||
|
DynamicCaptureSniff: "/d_sniff",
|
||||||
|
DynamicCaptureIngress: "/d_capture",
|
||||||
|
|
||||||
|
UIPath: UI_PATH,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Note the `DynamicCaptureSniff` and `DynamicCaptureIngress`. These paths define the sniffing and capturing behavior for dynamic requests. The sniff path is used to evaluate whether a request should be intercepted, while the ingress path handles the intercepted requests.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. Register Dynamic Capture Handlers
|
||||||
|
|
||||||
|
Dynamic capture handlers are used to process requests that match specific conditions.
|
||||||
|
|
||||||
|
```go
|
||||||
|
pathRouter := plugin.NewPathRouter()
|
||||||
|
pathRouter.SetDebugPrintMode(true)
|
||||||
|
|
||||||
|
pathRouter.RegisterDynamicSniffHandler("/d_sniff", http.DefaultServeMux, func(dsfr *plugin.DynamicSniffForwardRequest) plugin.SniffResult {
|
||||||
|
if strings.HasPrefix(dsfr.RequestURI, "/foobar") {
|
||||||
|
fmt.Println("Accepting request with UUID: " + dsfr.GetRequestUUID())
|
||||||
|
return plugin.SniffResultAccpet
|
||||||
|
}
|
||||||
|
fmt.Println("Skipping request with UUID: " + dsfr.GetRequestUUID())
|
||||||
|
return plugin.SniffResultSkip
|
||||||
|
})
|
||||||
|
|
||||||
|
pathRouter.RegisterDynamicCaptureHandle("/d_capture", http.DefaultServeMux, func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte("Welcome to the dynamic capture handler!\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)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
The `RegisterDynamicSniffHandler` evaluates incoming requests, while the `RegisterDynamicCaptureHandle` processes the intercepted requests.
|
||||||
|
|
||||||
|
### Sniffing Logic
|
||||||
|
|
||||||
|
If a module registered a dynamic capture path, Zoraxy will forward the request headers as `DynamicSniffForwardRequest` (`dsfr`) object to all the plugins that is assigned to this tag. And in each of the plugins, a dedicated logic will take in the object and "think" if they want to handle the request. You can get the following information from the dsfr object by directly accessing the members of it.
|
||||||
|
|
||||||
|
```go
|
||||||
|
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"`
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
You can also use the `GetRequest()` function to get the `*http.Request` object or `GetRequestUUID()` to get a `string` value that is a UUID corresponding to this request for later matching with the incoming, forwarded request.
|
||||||
|
|
||||||
|
**Note that since all request will pass through the sniffing function in your plugin, do not implement any blocking logic in your sniffing function, otherwise this will slow down all traffic going through the HTTP proxy rule with the plugin enabled.**
|
||||||
|
|
||||||
|
In the sniffing stage, you can choose to either return `ControlStatusCode_CAPTURED`, where Zoraxy will forward the request to your plugin `DynamicCaptureIngress` endpoint, or `ControlStatusCode_UNHANDLED`, where Zoraxy will pass on the request to the next dynamic handling plugin or if there are no more plugins to handle the routing, to the upstream server.
|
||||||
|
|
||||||
|
### Capture Handling
|
||||||
|
|
||||||
|
The capture handling is where Zoraxy formally forward you the HTTP request the client is requesting. In this situation, you must response the request by properly handling the ` http.Request` by writing to the `http.ResponseWriter`.
|
||||||
|
|
||||||
|
If there is a need to match the sniffing to the capture handling logic (Let say you want to design your plugin to run some kind of pre-processing before the actual request came in), you can use the `X-Zoraxy-Requestid` header in the HTTP request. This is the same UUID as the one you get from `dsfr.GetRequestUUID()` in the sniffing stage if they are the same request object on Zoraxy side.
|
||||||
|
|
||||||
|
The http request that Zoraxy forwards to the plugin capture handling endpoint contains header like these.
|
||||||
|
|
||||||
|
```html
|
||||||
|
Request URI: /foobar/test
|
||||||
|
Request Method: GET
|
||||||
|
Request Headers:
|
||||||
|
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
|
||||||
|
Accept-Encoding: gzip, deflate, br, zstd
|
||||||
|
(more fileds)
|
||||||
|
X-Forwarded-For: 127.0.0.1
|
||||||
|
X-Forwarded-Proto: https
|
||||||
|
X-Real-Ip: 127.0.0.1
|
||||||
|
X-Zoraxy-Requestid: d00619b8-f39e-4c04-acd8-c3a6f55b1566
|
||||||
|
```
|
||||||
|
|
||||||
|
You can extract the `X-Zoraxy-Requestid` value from the request header and do your matching for implementing your function if needed.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. Render Debug UI
|
||||||
|
|
||||||
|
This UI is used help validate the management Web UI is correctly shown in Zoraxy webmin interface. You should implement the required management interface for your plugin here.
|
||||||
|
|
||||||
|
```go
|
||||||
|
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")
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 5. Full Code
|
||||||
|
|
||||||
|
Here is the complete code for the dynamic capture example:
|
||||||
|
|
||||||
|
```go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"sort"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
plugin "example.com/zoraxy/dynamic-capture-example/mod/zoraxy_plugin"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
PLUGIN_ID = "org.aroz.zoraxy.dynamic-capture-example"
|
||||||
|
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.dynamic-capture-example",
|
||||||
|
Name: "Dynamic Capture Example",
|
||||||
|
Author: "aroz.org",
|
||||||
|
AuthorContact: "https://aroz.org",
|
||||||
|
Description: "This is an example plugin for Zoraxy that demonstrates how to use dynamic captures.",
|
||||||
|
URL: "https://zoraxy.aroz.org",
|
||||||
|
Type: plugin.PluginType_Router,
|
||||||
|
VersionMajor: 1,
|
||||||
|
VersionMinor: 0,
|
||||||
|
VersionPatch: 0,
|
||||||
|
|
||||||
|
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)
|
||||||
|
|
||||||
|
/*
|
||||||
|
Dynamic Captures
|
||||||
|
*/
|
||||||
|
pathRouter.RegisterDynamicSniffHandler("/d_sniff", http.DefaultServeMux, func(dsfr *plugin.DynamicSniffForwardRequest) plugin.SniffResult {
|
||||||
|
//In this example, we want to capture all URI
|
||||||
|
//that start with /foobar and forward it to the dynamic capture handler
|
||||||
|
if strings.HasPrefix(dsfr.RequestURI, "/foobar") {
|
||||||
|
reqUUID := dsfr.GetRequestUUID()
|
||||||
|
fmt.Println("Accepting request with UUID: " + reqUUID)
|
||||||
|
|
||||||
|
// Print all the values of the request
|
||||||
|
fmt.Println("Method:", dsfr.Method)
|
||||||
|
fmt.Println("Hostname:", dsfr.Hostname)
|
||||||
|
fmt.Println("URL:", dsfr.URL)
|
||||||
|
fmt.Println("Header:")
|
||||||
|
for key, values := range dsfr.Header {
|
||||||
|
for _, value := range values {
|
||||||
|
fmt.Printf(" %s: %s\n", key, value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fmt.Println("RemoteAddr:", dsfr.RemoteAddr)
|
||||||
|
fmt.Println("Host:", dsfr.Host)
|
||||||
|
fmt.Println("RequestURI:", dsfr.RequestURI)
|
||||||
|
fmt.Println("Proto:", dsfr.Proto)
|
||||||
|
fmt.Println("ProtoMajor:", dsfr.ProtoMajor)
|
||||||
|
fmt.Println("ProtoMinor:", dsfr.ProtoMinor)
|
||||||
|
|
||||||
|
// We want to handle this request, reply with aSniffResultAccept
|
||||||
|
return plugin.SniffResultAccpet
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the request URI does not match, we skip this request
|
||||||
|
fmt.Println("Skipping request with UUID: " + dsfr.GetRequestUUID())
|
||||||
|
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("Dynamic capture example started at http://127.0.0.1:" + strconv.Itoa(runtimeCfg.Port))
|
||||||
|
http.ListenAndServe("127.0.0.1:"+strconv.Itoa(runtimeCfg.Port), nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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")
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 6. Expected Output
|
||||||
|
|
||||||
|
To enable the plugin, add the plugin to one of the tags and assign the tag to your HTTP Proxy Rule. Here is an example of assigning the plugin to the "debug" tag and assigning it to a localhost loopback HTTP proxy rule.
|
||||||
|
|
||||||
|
When the plugin is running, requests matching the sniff conditions will be intercepted and processed by the dynamic capture handler.
|
||||||
|
|
||||||
|
If everything is correctly setup, you should see the following page when requesting any URL with prefix `(your_HTTP_proxy_rule_hostname)/foobar`
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Example terminal output for requesting `/foobar/*`:
|
||||||
|
|
||||||
|
```html
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Request captured by dynamic sniff path: /d_sniff/
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Header:
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Method: GET
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Hostname: a.localhost
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] URL: /foobar/test
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Accepting request with UUID: 8c916c58-0d6a-4d11-a2f0-f29d3d984509
|
||||||
|
[2025-05-30 20:44:26.143165] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Sec-Fetch-Dest: document
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Accept-Encoding: gzip, deflate, br, zstd
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Accept-Language: zh-TW,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Cache-Control: max-age=0
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Sec-Fetch-User: ?1
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Upgrade-Insecure-Requests: 1
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Safari/537.36 Edg/136.0.0.0
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Priority: u=0, i
|
||||||
|
[2025-05-30 20:44:26.143149] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Sec-Ch-Ua-Mobile: ?0
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Sec-Ch-Ua: "Chromium";v="136", "Microsoft Edge";v="136", "Not.A/Brand";v="99"
|
||||||
|
[2025-05-30 20:44:26.143165] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Sec-Ch-Ua-Platform: "Windows"
|
||||||
|
[2025-05-30 20:44:26.142645] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Sec-Fetch-Site: none
|
||||||
|
[2025-05-30 20:44:26.143165] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Sec-Fetch-Mode: navigate
|
||||||
|
[2025-05-30 20:44:26.143165] [plugin-manager] [system:info] [Dynamic Capture Example:22964] RemoteAddr: [::1]:54522
|
||||||
|
[2025-05-30 20:44:26.143165] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Host: a.localhost
|
||||||
|
[2025-05-30 20:44:26.143165] [plugin-manager] [system:info] [Dynamic Capture Example:22964] RequestURI: /foobar/test
|
||||||
|
[2025-05-30 20:44:26.143165] [plugin-manager] [system:info] [Dynamic Capture Example:22964] Proto: HTTP/2.0
|
||||||
|
[2025-05-30 20:44:26.143165] [plugin-manager] [system:info] [Dynamic Capture Example:22964] ProtoMajor: 2
|
||||||
|
[2025-05-30 20:44:26.143165] [plugin-manager] [system:info] [Dynamic Capture Example:22964] ProtoMinor: 0
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Now you know how to develop a plugin in Zoraxy that handles special routings!
|
After Width: | Height: | Size: 38 KiB |
After Width: | Height: | Size: 37 KiB |
After Width: | Height: | Size: 58 KiB |
After Width: | Height: | Size: 62 KiB |
After Width: | Height: | Size: 31 KiB |