Compare commits

...

338 commits
0.3 ... master

Author SHA1 Message Date
esrrhs
66d34ba031
Merge pull request #60 from esrrhs/esrrhs-patch-1
Update go.yml
2023-11-02 20:23:13 +08:00
esrrhs
a350629abc
Update go.yml 2023-11-02 20:22:58 +08:00
esrrhs
5d53ddeb7a update 2023-11-02 20:22:15 +08:00
esrrhs
933b646d98
Merge pull request #59 from esrrhs/dependabot/go_modules/golang.org/x/net-0.17.0
Bump golang.org/x/net from 0.8.0 to 0.17.0
2023-10-12 09:19:58 +08:00
dependabot[bot]
404ea744fe
Bump golang.org/x/net from 0.8.0 to 0.17.0
Bumps [golang.org/x/net](https://github.com/golang/net) from 0.8.0 to 0.17.0.
- [Commits](https://github.com/golang/net/compare/v0.8.0...v0.17.0)

---
updated-dependencies:
- dependency-name: golang.org/x/net
  dependency-type: direct:production
...

Signed-off-by: dependabot[bot] <support@github.com>
2023-10-11 23:35:00 +00:00
esrrhs
b30676729c
Merge pull request #58 from jiqing112/master
更新 main.go内的 -key 说明
2023-06-19 15:34:16 +08:00
jiqing112
6ab38e9d01
更新 main.go内的 -key 说明
更新 main.go内的 -key 说明
2023-06-19 14:40:06 +08:00
esrrhs
19d00b970e
Merge pull request #57 from jiqing112/patch-4
还是空格的错误,这次应该没有了
2023-04-01 16:11:44 +08:00
jiqing112
3d001c21f0
还是空格的错误,这次应该没有了 2023-04-01 15:48:50 +08:00
esrrhs
3c2bcf9bc9
Merge pull request #56 from jiqing112/patch-3
docker命令里的空格错误
2023-04-01 15:37:50 +08:00
jiqing112
a4d7ee02d9
docker命令里的空格错误 2023-04-01 14:53:46 +08:00
esrrhs
327cf680c7
Merge pull request #55 from jiqing112/patch-2
Update README.md
2023-04-01 08:11:05 +08:00
esrrhs
cab02d012b
Merge pull request #54 from jiqing112/patch-1
Update README.md
2023-04-01 08:10:35 +08:00
jiqing112
8824c495f5
Update README.md
补充关于 -key参数的正确使用。避免使用者误会 “-key” 参数是任意类型的字符串形式的密码
2023-03-31 13:53:43 +08:00
jiqing112
1c79a8db64
Update README.md
范例的命令有个位置打错了空格,会让参数报错,无法执行。
2023-03-31 13:38:40 +08:00
esrrhs
7af41a7723 update 2023-03-18 16:43:16 +08:00
esrrhs
aa216a7d72 change 2023-03-18 13:35:27 +08:00
esrrhs
520cdd7063 move dir 2023-03-18 13:32:35 +08:00
esrrhs
f8d9ed6b5e Merge branch 'master' of https://github.com/esrrhs/pingtunnel 2023-02-22 21:45:45 +08:00
esrrhs
e816385534 upadte 2023-02-22 21:45:34 +08:00
esrrhs
4a02bd6270
Update README.md 2023-02-22 21:42:13 +08:00
esrrhs
5163a571dc add 2023-02-22 21:41:13 +08:00
esrrhs
7187a57ceb
Create docker-image.yml 2023-02-22 21:36:59 +08:00
esrrhs
5f81102814 update 2023-02-22 21:36:00 +08:00
esrrhs
94a9dc1eed
Create go.yml 2023-02-22 21:32:12 +08:00
esrrhs
fe571e8e4e add 2023-02-22 21:30:11 +08:00
esrrhs
93ab736d55 add 2023-02-22 21:29:12 +08:00
esrrhs
a1a1f0462a delete 2022-10-06 21:57:54 +08:00
esrrhs
7970b38a68 delete 2022-10-06 21:56:15 +08:00
esrrhs
8d63ad5334
Update README.md 2022-10-06 21:51:40 +08:00
esrrhs
219755bd5c
Update README.md 2022-10-06 21:42:02 +08:00
esrrhs
622f631bbd
Update README.md 2022-07-05 11:58:31 +08:00
esrrhs
4422d7c1d0
Create docker-image.yml 2022-06-14 11:45:27 +08:00
esrrhs
cb246793e6
Update README.md 2022-01-27 15:16:25 +08:00
esrrhs
cd1be4debf
Update README.md 2022-01-27 15:16:04 +08:00
esrrhs
856300a446
Update README.md 2021-12-20 10:49:26 +08:00
esrrhs
ec19b924c8
Update README.md 2021-12-20 10:49:13 +08:00
esrrhs
ef8486c4eb
Update README.md 2021-11-16 12:20:44 +08:00
benderzhao
60e4971fdf Merge branch 'master' of https://github.com/esrrhs/pingtunnel 2021-11-16 12:19:54 +08:00
benderzhao
59dc4b4f79 add 2021-11-16 12:19:44 +08:00
esrrhs
4b22d08d0b
Update go.yml 2021-09-28 16:34:29 +08:00
benderzhao
31ebcc8bf8 add 2021-09-28 16:33:39 +08:00
zhao xin
3d75970db9
Delete qtrun.jpg 2021-07-05 11:05:48 +08:00
zhao xin
be7f7650a4
Update README_EN.md 2021-07-05 11:05:31 +08:00
zhao xin
dff22bc62c
Update README.md 2021-07-05 11:05:05 +08:00
zhao xin
f9bf4076f2
Update README_EN.md 2021-06-19 00:51:21 +08:00
zhao xin
e1d5c970e0
Update README.md 2021-06-19 00:50:53 +08:00
zhao xin
f3a608957a Set theme jekyll-theme-cayman 2021-05-07 12:19:39 +08:00
esrrhs
c7700e858f add 2021-04-22 13:24:02 +08:00
esrrhs
1632aa4389 add 2021-04-21 21:08:30 +08:00
esrrhs
86a7340189 Merge remote-tracking branch 'origin/master'
# Conflicts:
#	pack.sh
2021-04-21 21:06:42 +08:00
esrrhs
780f75ae7c add 2021-04-21 21:06:22 +08:00
zhao xin
2eab964f13
Update README_EN.md 2021-04-20 13:23:14 +08:00
zhao xin
dd70313cf5
Update README.md 2021-04-20 13:23:00 +08:00
esrrhs
d04d249538 add 2021-04-20 11:06:00 +08:00
esrrhs
174fe4d7a2 add 2021-04-19 22:22:45 +08:00
esrrhs
6e8c975859 Merge remote-tracking branch 'origin/master' 2021-04-19 22:05:22 +08:00
esrrhs
74f6108315 add 2021-04-19 22:04:37 +08:00
zhao xin
e746dc3616
Update README_EN.md 2021-04-01 11:50:28 +08:00
zhao xin
cd21d586ae
Update README.md 2021-04-01 11:49:07 +08:00
zhao xin
41ccdf6d7f
Update Dockerfile 2021-03-18 11:22:13 +08:00
zhao xin
9e8e9eb535
Merge pull request #51 from sajad-sadra/master
easy deploy with docker-compose
2021-02-24 11:04:44 +08:00
sajad sadrayieh
24fa2404dd a basic documentation for easy deploy 2021-02-23 16:46:12 +03:30
sajad sadrayieh
122143529e make configs reads from env file 2021-02-23 16:39:19 +03:30
sajad sadrayieh
bbee9243e8 [ADD] client compose file 2021-02-23 16:33:36 +03:30
sajad sadrayieh
847d475cc2 [ADD] server compose file 2021-02-23 16:30:21 +03:30
zhao xin
136db8d776
Merge pull request #48 from phanirithvij/master
fix some markdown issues in the readme
2020-12-23 10:44:42 +08:00
phanirithvij
7f8d5fa390 fix more 2020-12-23 08:14:27 +05:30
phanirithvij
8db60694b6 fix new issues 2020-12-23 08:08:49 +05:30
phanirithvij
49d1af71b4 fix some markdown issues in the readme
To improve coverage
2020-12-23 07:41:06 +05:30
zhao xin
6fc899c759
Update README_EN.md 2020-10-31 23:12:08 +08:00
zhao xin
7454c250dc
Update README.md 2020-10-31 23:11:29 +08:00
zhao xin
ad69639117
Update README.md 2020-10-31 23:11:15 +08:00
zhao xin
fd4c4ebb47
Update README_EN.md 2020-10-31 23:10:21 +08:00
zhao xin
db05e85500
Update README.md 2020-10-31 23:09:55 +08:00
zhao xin
6b7bc8e876
Update README_EN.md 2020-10-31 23:08:41 +08:00
zhao xin
395a890632
Update README.md 2020-10-31 23:07:13 +08:00
zhao xin
0d6b836173
Update README.md 2020-10-31 23:06:46 +08:00
zhao xin
1c51495b9a
Update README.md 2020-10-31 23:03:22 +08:00
zhao xin
76deebb4b0
Update README_EN.md 2020-10-14 20:21:46 +08:00
zhao xin
a3209179a0
Update README.md 2020-10-14 20:21:08 +08:00
zhao xin
f365e94395
Update README_EN.md 2020-09-30 17:39:13 +08:00
esrrhs
aa88696c37 add 2020-09-03 09:36:44 +08:00
esrrhs
e7b0c16282 add 2020-09-03 09:26:40 +08:00
zhao xin
8be6b166fc
Update README_EN.md 2020-07-06 17:25:21 +08:00
zhao xin
0bfccb395e
Update README.md 2020-07-06 17:24:59 +08:00
zhao xin
6c35c96929
Merge pull request #33 from honwen/master
Ensure static linked binary
2020-06-28 12:33:08 +08:00
hwchan
fc68e6449c Ensure static linked binary 2020-06-28 11:14:11 +08:00
zhao xin
b899f65b5a
Update README_EN.md 2020-06-27 16:22:06 +08:00
zhao xin
1ac3aabd08
Update README.md 2020-06-27 16:21:45 +08:00
esrrhs
a575babaa7 add 2020-05-31 20:05:06 +08:00
zhao xin
f2d1e1a5b3
Update README_EN.md 2020-05-25 11:21:15 +08:00
zhao xin
6fa90c210b
Update README.md 2020-05-25 11:20:13 +08:00
zhao xin
5a61d67923
Update README_EN.md 2020-05-20 15:13:01 +08:00
zhao xin
2de5b16e2e
Update README.md 2020-05-20 15:12:37 +08:00
zhao xin
29b3f73ada
Update README.md 2020-05-20 15:11:35 +08:00
zhao xin
112fb03249
Update README.md 2020-05-20 14:59:55 +08:00
esrrhs
a9848f254a add 2020-05-08 19:56:38 +08:00
zhao xin
de0b4c2730
Update README_EN.md 2020-05-07 18:16:51 +08:00
zhao xin
d038899851
Update README.md 2020-05-07 18:16:33 +08:00
zhao xin
0c6be88983
Update README_EN.md 2020-05-07 18:14:40 +08:00
zhao xin
f51245fda4
Delete test.png 2020-05-07 18:12:48 +08:00
zhao xin
f97c162f3f
Update README.md 2020-05-07 18:12:24 +08:00
esrrhs
fd8c39df04 add 2020-05-01 15:21:30 +08:00
esrrhs
b8b216c115 add 2020-05-01 13:23:35 +08:00
zhao xin
9fcaf84ac3
Update README.md 2020-04-30 10:35:45 +08:00
zhao xin
db7551731a
Update README_EN.md 2020-04-23 10:37:34 +08:00
zhao xin
c71b537898
Update README.md 2020-04-23 10:35:45 +08:00
zhao xin
7b90bc40a6
Update go.yml 2020-04-02 11:31:14 +08:00
esrrhs
5c313bff4e add
pingtunnel_windows64.zip
2020-03-30 10:04:42 +08:00
esrrhs
5b52521521 add 2020-03-07 16:40:15 +08:00
esrrhs
a797fd5cc7 add 2020-03-03 09:13:08 +08:00
zhao xin
a923fed6fc
Update README.md 2020-03-01 21:43:57 +08:00
esrrhs
1dc3fe11be Merge remote-tracking branch 'origin/master' 2020-03-01 09:28:13 +08:00
esrrhs
f988b2c483 add 2020-03-01 09:28:02 +08:00
esrrhs
0a18ded9cc add 2020-03-01 08:25:57 +08:00
esrrhs
dd04aa9b14 add 2020-03-01 00:34:45 +08:00
esrrhs
6913d315da add 2020-02-29 22:44:45 +08:00
zhao xin
b1f6acb659
Update README.md 2020-02-29 12:46:34 +08:00
zhao xin
3b6f4eebcf
Update README_EN.md 2020-02-28 16:13:52 +08:00
zhao xin
32ee72b850
Update README_EN.md 2020-02-28 16:12:18 +08:00
zhao xin
d7f3c68d89
Update README.md 2020-02-28 16:08:03 +08:00
esrrhs
9b26464d10 add 2020-02-27 21:40:09 +08:00
zhao xin
d9ebc5c9f6
Update README.md 2020-02-26 11:20:07 +08:00
zhao xin
aaff23309f
Update README.md 2020-02-25 20:55:19 +08:00
esrrhs
ba73a631a6 add 2020-02-23 16:52:28 +08:00
esrrhs
610b029dd9 Merge remote-tracking branch 'origin/master' 2020-02-23 16:48:12 +08:00
esrrhs
2d3ed37feb add 2020-02-23 16:47:41 +08:00
zhao xin
2734daddc6
Update README_EN.md 2020-02-21 12:57:44 +08:00
zhao xin
fc865f754d
Update README.md 2020-02-21 12:57:00 +08:00
zhao xin
2b1abdf783
Update README.md 2020-02-19 22:08:50 +08:00
zhao xin
408c939da4
Update README.md 2020-02-19 22:01:35 +08:00
zhao xin
0ad2ad5fca
Update README.md 2020-02-19 21:18:07 +08:00
zhao xin
9917c5bdcb Update Dockerfile 2020-02-10 13:14:11 +08:00
zhao xin
41e144ceca
Update README_EN.md 2020-02-01 18:45:15 +08:00
zhao xin
ff392de1f1
Update README.md 2020-02-01 18:44:42 +08:00
zhao xin
44a1a5853d
Update README.md 2020-01-29 18:04:40 +08:00
zhao xin
43f40eda1b
Update README_EN.md 2020-01-08 11:22:51 +08:00
zhao xin
1d67562804
Update README.md 2020-01-08 11:22:40 +08:00
esrrhs
b4f7e69f6b add 2020-01-08 10:49:48 +08:00
esrrhs
aa046d8a2f add 2020-01-08 10:42:49 +08:00
esrrhs
d4731262db add 2020-01-08 10:38:33 +08:00
esrrhs
852c0b1761 Merge remote-tracking branch 'origin/master' 2020-01-08 10:36:08 +08:00
esrrhs
a33439b013 add 2020-01-08 10:35:57 +08:00
esrrhs
ba20b02358 add 2020-01-08 10:33:48 +08:00
esrrhs
00dfd7246b add 2020-01-08 10:33:02 +08:00
esrrhs
c79b7a2cea add 2020-01-07 22:10:28 +08:00
esrrhs
e0c5fd7b2f add 2020-01-07 19:39:46 +08:00
esrrhs
c66a9189eb Merge remote-tracking branch 'origin/master' 2020-01-07 19:36:03 +08:00
esrrhs
0f3d847962 add 2020-01-07 19:34:23 +08:00
zhao xin
97866a3486
Update README_EN.md 2020-01-07 18:06:41 +08:00
zhao xin
52337521dd
Update README.md 2020-01-07 18:06:15 +08:00
zhao xin
74c53223b0
Update README_EN.md 2020-01-06 10:42:53 +08:00
zhao xin
8954af9416
Update README.md 2020-01-06 10:42:34 +08:00
zhao xin
e5831b152c
Update README.md 2020-01-04 20:02:45 +08:00
esrrhs
85c558a677 add 2020-01-04 20:01:26 +08:00
zhao xin
c5a6c80561
Update README.md 2020-01-04 18:55:03 +08:00
zhao xin
e16bff920e
Update README.md 2020-01-04 18:53:10 +08:00
zhao xin
7b9113cd38
Add files via upload 2020-01-04 18:52:47 +08:00
zhao xin
57b58f555d
Update README.md 2020-01-04 18:52:30 +08:00
zhao xin
5eb78f2802
Update README.md 2020-01-04 18:30:28 +08:00
zhao xin
abd6db0ac5
Update README.md 2020-01-03 09:03:49 +08:00
zhao xin
8ba207626e
Update README.md 2020-01-01 16:54:54 +08:00
zhao xin
9e0ae35149
Update README.md 2020-01-01 16:33:40 +08:00
zhao xin
321517df80
Update README.md 2020-01-01 16:25:02 +08:00
esrrhs
ddea3a66bc Merge remote-tracking branch 'origin/master' 2020-01-01 16:02:03 +08:00
esrrhs
fd4b189c6f add 2020-01-01 16:01:50 +08:00
zhao xin
fa445c7d89
Update README.md 2019-12-27 12:24:58 +08:00
zhao xin
371dd4baa0
Update Dockerfile 2019-12-27 11:37:14 +08:00
zhao xin
9e62422db6
Create Dockerfile 2019-12-27 11:03:03 +08:00
zhao xin
1a994d448a
Update go.yml 2019-12-27 10:54:11 +08:00
zhao xin
0d61b60eaa
Update go.yml 2019-12-27 10:51:20 +08:00
zhao xin
d11d7c4b87
Update README.md 2019-12-27 10:45:40 +08:00
zhao xin
4986c989e3 Set theme jekyll-theme-hacker 2019-12-05 14:45:26 +08:00
zhao xin
c2ff8fa632
Create go.yml 2019-11-14 12:08:18 +08:00
esrrhs
0b0a9cdc6a add 2019-11-13 16:54:55 +08:00
esrrhs
8651c222c4 add 2019-11-12 17:11:30 +08:00
zhao xin
3050373508
Update README.md 2019-11-05 21:15:41 +08:00
zhao xin
1b02df4a4d
Update README.md 2019-11-05 21:09:36 +08:00
zhao xin
69273a94cc
Update README.md 2019-11-05 11:07:28 +08:00
esrrhs
524fdb836d add 2019-11-05 09:52:33 +08:00
esrrhs
bf6270387e add 2019-11-05 09:23:52 +08:00
esrrhs
91de7cb8f0 add 2019-11-05 09:07:14 +08:00
esrrhs
cb7d489988 add 2019-11-04 21:56:07 +08:00
esrrhs
728c2705b2 add 2019-11-02 15:50:54 +08:00
esrrhs
a70fcdff74 add 2019-11-01 23:54:44 +08:00
esrrhs
afa564c81d add 2019-11-01 23:51:49 +08:00
esrrhs
33b164e63f add 2019-11-01 23:22:24 +08:00
esrrhs
8fb7712b54 add 2019-11-01 23:18:57 +08:00
esrrhs
599fb597e8 add 2019-11-01 23:13:59 +08:00
esrrhs
18fa88a798 Merge branch 'master' of https://github.com/esrrhs/pingtunnel 2019-11-01 23:13:05 +08:00
esrrhs
a0a5ef06fb add 2019-11-01 22:38:09 +08:00
zhao xin
97ac0742b6
Update README.md 2019-11-01 21:53:14 +08:00
zhao xin
9665e03f6f
Update README.md 2019-11-01 21:50:45 +08:00
esrrhs
ae593ddeb2 add 2019-11-01 21:12:37 +08:00
esrrhs
db35790150 add 2019-11-01 21:10:32 +08:00
esrrhs
d84320fffb add 2019-11-01 20:54:55 +08:00
esrrhs
5c0c08b7f3 add 2019-11-01 20:52:18 +08:00
esrrhs
1657f46784 add 2019-11-01 18:48:08 +08:00
esrrhs
30b1cd117b add 2019-11-01 18:44:15 +08:00
esrrhs
8d1d2bd6a2 add 2019-10-31 21:55:40 +08:00
esrrhs
0ba2aa2297 add 2019-10-31 21:48:32 +08:00
esrrhs
2a75a80a0e add 2019-10-31 21:18:42 +08:00
esrrhs
64f8ec24e6 add 2019-10-31 21:16:14 +08:00
esrrhs
831a5ee51d add 2019-10-31 21:05:48 +08:00
esrrhs
a71309c8cc add 2019-10-31 20:30:38 +08:00
zhao xin
d25f8499f6
Update README.md 2019-10-30 20:40:13 +08:00
zhao xin
08f0f61840
Update README.md 2019-10-30 20:39:56 +08:00
esrrhs
f323941276 add 2019-10-30 20:15:43 +08:00
esrrhs
160d6efc89 add 2019-10-30 19:45:19 +08:00
esrrhs
a4573f1540 add 2019-10-30 19:28:00 +08:00
esrrhs
a43e5c435f add 2019-10-30 19:14:27 +08:00
esrrhs
9b86a95dfc Merge remote-tracking branch 'origin/master' 2019-10-30 18:57:40 +08:00
esrrhs
c348cfb2e9 add 2019-10-30 18:57:26 +08:00
esrrhs
4bfa31eea1 add 2019-10-29 20:39:12 +08:00
esrrhs
dc91b48b34 add 2019-10-28 19:48:50 +08:00
esrrhs
4bb047d433 add 2019-10-28 19:45:12 +08:00
zhao xin
6c7341375b
Update README.md 2019-10-28 17:13:00 +08:00
esrrhs
225992d9eb Merge remote-tracking branch 'origin/master' 2019-10-28 16:55:54 +08:00
esrrhs
8b712a1a9d add 2019-10-28 16:55:27 +08:00
zhao xin
85e3f4d20a
Update README.md 2019-10-28 16:54:09 +08:00
esrrhs
28eee0c37d add 2019-10-28 16:36:32 +08:00
esrrhs
15c292777f add 2019-10-28 16:33:44 +08:00
esrrhs
e1fb19862c add 2019-10-28 12:29:56 +08:00
esrrhs
b7e82c053a add 2019-10-28 12:11:04 +08:00
esrrhs
feba1f555a Merge remote-tracking branch 'origin/master' 2019-10-28 11:58:25 +08:00
esrrhs
2b6ecc47ae add 2019-10-28 11:58:01 +08:00
zhao xin
1113420ed9
Update README.md 2019-10-27 20:27:52 +08:00
zhao xin
de8be340ff
Update README.md 2019-10-27 20:12:42 +08:00
zhao xin
c2c8ad1985
Update README.md 2019-10-27 20:12:05 +08:00
zhao xin
47c1d9117a
Update README.md 2019-10-27 20:10:30 +08:00
zhao xin
6c495b2cf8
Update README.md 2019-10-27 20:09:51 +08:00
zhao xin
16c7838109
Update README.md 2019-10-27 20:09:19 +08:00
zhao xin
78346d4c6c
Update README.md 2019-10-27 20:05:36 +08:00
esrrhs
1f8263a322 Merge remote-tracking branch 'origin/master' 2019-10-27 20:05:06 +08:00
esrrhs
48b27bfe00 add 2019-10-27 20:04:51 +08:00
zhao xin
ae787b421e
Update README.md 2019-10-27 20:03:55 +08:00
esrrhs
2495f2290f add 2019-10-27 20:00:15 +08:00
esrrhs
d69f971519 add 2019-10-27 19:41:56 +08:00
esrrhs
551c1437e5 add 2019-10-27 19:20:35 +08:00
esrrhs
013c6124d9 add 2019-10-27 19:06:55 +08:00
esrrhs
2dd984018c add 2019-10-27 18:35:00 +08:00
esrrhs
79a278a90f add 2019-10-27 18:16:57 +08:00
esrrhs
6ad90d18aa add 2019-10-27 17:25:05 +08:00
esrrhs
aed73279f7 add 2019-10-27 17:23:30 +08:00
esrrhs
f93b24d3d1 add 2019-10-27 17:17:47 +08:00
esrrhs
b67d5d372d add 2019-10-27 12:55:40 +08:00
esrrhs
37005c8ae8 add 2019-10-27 12:52:02 +08:00
esrrhs
d0897dd522 add 2019-10-27 12:39:55 +08:00
esrrhs
dc02a58284 add 2019-10-27 12:32:31 +08:00
esrrhs
7b88a879e1 add 2019-10-27 12:25:17 +08:00
esrrhs
63e557b566 add 2019-10-27 12:19:16 +08:00
esrrhs
2b37b02276 add 2019-10-26 22:02:10 +08:00
esrrhs
1222033a82 add 2019-10-26 21:54:13 +08:00
esrrhs
b1fd10de27 add 2019-10-26 21:40:52 +08:00
esrrhs
284dcccd18 add 2019-10-26 21:26:19 +08:00
esrrhs
bf39ee41b2 add 2019-10-26 21:22:14 +08:00
esrrhs
9e1d5d61b2 add 2019-10-26 21:01:22 +08:00
esrrhs
51bf1778c5 add 2019-10-26 21:00:15 +08:00
esrrhs
15e53f1f52 add 2019-10-26 20:45:58 +08:00
esrrhs
c90545eaf6 add 2019-10-26 20:39:41 +08:00
esrrhs
c923db7d8a add 2019-10-26 20:02:37 +08:00
esrrhs
82045f9e55 add 2019-10-26 20:01:51 +08:00
esrrhs
2ffba2224b add 2019-10-26 19:54:49 +08:00
esrrhs
a2f8819782 add 2019-10-26 18:33:22 +08:00
esrrhs
a2c2ab95e2 add 2019-10-26 18:12:55 +08:00
esrrhs
60cb43cb2e add 2019-10-26 17:51:38 +08:00
esrrhs
f97c2fb4d0 add 2019-10-26 17:39:41 +08:00
esrrhs
f06498e912 add 2019-10-26 17:24:28 +08:00
esrrhs
2ed5548a6f add 2019-10-26 17:08:06 +08:00
esrrhs
ba76acaf87 add 2019-10-26 17:05:08 +08:00
esrrhs
b15faae234 add 2019-10-26 12:07:20 +08:00
esrrhs
a13cd98e29 add 2019-10-26 12:01:30 +08:00
esrrhs
ff3daaed0f add 2019-10-26 09:39:36 +08:00
esrrhs
111b7397d8 add 2019-10-25 21:57:38 +08:00
esrrhs
b0c7bc5949 add 2019-10-25 21:48:10 +08:00
esrrhs
b88c2320f3 add 2019-10-25 21:42:02 +08:00
esrrhs
959dd136d9 add 2019-10-25 21:31:04 +08:00
esrrhs
d3479c08a1 add 2019-10-25 21:28:30 +08:00
esrrhs
120d949c59 add 2019-10-25 21:21:55 +08:00
esrrhs
d34ecbaca6 add 2019-10-25 21:19:00 +08:00
esrrhs
2fda6b845c add 2019-10-25 21:17:47 +08:00
esrrhs
f6e315ba37 add 2019-10-25 20:42:51 +08:00
esrrhs
f12978b35b add 2019-10-25 20:06:13 +08:00
esrrhs
5a8b3469ba add 2019-10-25 20:04:52 +08:00
esrrhs
f512f32ec1 add 2019-10-24 21:57:03 +08:00
esrrhs
0c41f5e7d3 add 2019-10-24 21:42:46 +08:00
esrrhs
a2f409453b add 2019-10-23 20:37:22 +08:00
esrrhs
14315b85ab add 2019-10-23 20:36:13 +08:00
esrrhs
a72544515e add 2019-10-22 21:27:21 +08:00
esrrhs
05b7ff19aa add 2019-10-22 21:04:25 +08:00
esrrhs
a40c5c8232 add 2019-10-22 10:14:00 +08:00
esrrhs
13b2cbca81 add 2019-10-22 09:39:20 +08:00
esrrhs
f275f81308 add 2019-10-20 16:27:03 +08:00
esrrhs
a3036dcc66 add 2019-10-19 22:36:17 +08:00
esrrhs
1402d824c6 add 2019-10-18 21:30:54 +08:00
esrrhs
65cd195ca8 add 2019-10-16 21:09:21 +08:00
esrrhs
28ca3dec46 update 2019-01-08 11:15:28 +08:00
esrrhs
8dddc24f69 Merge branch 'master' of https://github.com/esrrhs/pingtunnel 2019-01-08 11:14:35 +08:00
esrrhs
b1654595ce fix 2019-01-08 11:13:41 +08:00
esrrhs
6e9a274f19 add 2019-01-08 10:43:31 +08:00
esrrhs
da7cbc2ef8 Merge branch 'master' of https://github.com/esrrhs/pingtunnel 2019-01-08 10:32:42 +08:00
esrrhs
eaa01d0e8b fix 2019-01-08 10:32:26 +08:00
esrrhs
0117c80ec8 fix 2019-01-08 10:31:52 +08:00
zhao xin
5d26cd81ec
Update README.md 2018-12-23 13:19:18 +08:00
zhao xin
f9c29f8a8d
Add files via upload 2018-12-23 13:16:14 +08:00
zhao xin
73fe99044f
Update README.md 2018-12-23 13:09:33 +08:00
esrrhs
ed841bc0d2 update 2018-12-23 07:19:19 +08:00
esrrhs
a903e4427e add 2018-12-23 07:09:25 +08:00
esrrhs
e7a8030939 add 2018-12-23 07:07:25 +08:00
esrrhs
54655acee9 add 2018-12-23 07:02:57 +08:00
esrrhs
150c7ed74b test 2018-12-23 06:12:16 +08:00
esrrhs
51aba43633 add 2018-12-23 06:08:26 +08:00
esrrhs
8efea89357 add 2018-12-22 18:42:41 +08:00
esrrhs
e1cdec8c00 add 2018-12-22 18:36:29 +08:00
esrrhs
0481f15f07 add 2018-12-22 18:36:03 +08:00
esrrhs
f46cac4904 add 2018-12-22 18:29:55 +08:00
esrrhs
a442584b89 add 2018-12-22 18:20:58 +08:00
esrrhs
1ec83ad73d add 2018-12-22 17:58:07 +08:00
esrrhs
aa606c2ef1 add 2018-12-22 17:49:45 +08:00
esrrhs
9f8e88b863 add 2018-12-22 17:25:59 +08:00
esrrhs
b76965fe52 add 2018-12-22 16:42:45 +08:00
esrrhs
d72a7ff774 fix 2018-12-21 16:09:54 +08:00
esrrhs
74b9105fbf Merge branch 'master' of https://github.com/esrrhs/pingtunnel 2018-12-21 15:52:51 +08:00
esrrhs
75edf4c3c1 add 2018-12-21 15:52:34 +08:00
zhao xin
6ed16750c2
Update README.md 2018-12-20 15:09:16 +08:00
zhao xin
0c5fb5e451
Update README.md 2018-12-20 14:58:07 +08:00
esrrhs
e7fecd4e1c fix 2018-12-20 14:31:01 +08:00
esrrhs
13534e6e81 fix 2018-12-20 14:25:02 +08:00
esrrhs
b18a569dcd update 2018-12-20 14:22:28 +08:00
esrrhs
d7a5a8d9af fix 2018-12-20 14:18:08 +08:00
esrrhs
ab0935816d fix 2018-12-20 14:15:36 +08:00
esrrhs
3ed77ca749 Merge branch 'master' of https://github.com/esrrhs/pingtunnel 2018-12-20 14:12:40 +08:00
esrrhs
d809161ade update 2018-12-20 14:12:17 +08:00
zhao xin
baff3ac364
Update README.md 2018-12-19 16:49:14 +08:00
zhao xin
b4ef459e9f
Update README.md 2018-12-19 16:47:45 +08:00
zhao xin
38acfcdd5c
Update README.md 2018-12-19 16:47:04 +08:00
zhao xin
bc59369bfd
Add files via upload 2018-12-19 16:45:56 +08:00
23 changed files with 2059 additions and 447 deletions

34
.github/workflows/docker-image.yml vendored Normal file
View file

@ -0,0 +1,34 @@
name: Docker Image CI
on:
push:
branches: [ "master" ]
pull_request:
branches: [ "master" ]
jobs:
build:
runs-on: ubuntu-latest
steps:
-
name: Set up QEMU
uses: docker/setup-qemu-action@v1
-
name: Set up Docker Buildx
uses: docker/setup-buildx-action@v1
-
name: Login to DockerHub
uses: docker/login-action@v1
with:
username: ${{ secrets.DOCKERHUB_USERNAME }}
password: ${{ secrets.DOCKERHUB_TOKEN }}
-
name: Build and push
id: docker_build
uses: docker/build-push-action@v2
with:
push: true
tags: esrrhs/pingtunnel:latest

30
.github/workflows/go.yml vendored Normal file
View file

@ -0,0 +1,30 @@
# This workflow will build a golang project
# For more information see: https://docs.github.com/en/actions/automating-builds-and-tests/building-and-testing-go
name: Go
on:
push:
branches: [ "master" ]
pull_request:
branches: [ "master" ]
jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Set up Go
uses: actions/setup-go@v3
with:
go-version: 1.21
- name: Build
run: |
go mod tidy
go build -v ./...
- name: Test
run: go test -v ./...

6
.idea/vcs.xml generated
View file

@ -1,6 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="VcsDirectoryMappings">
<mapping directory="$PROJECT_DIR$" vcs="Git" />
</component>
</project>

13
Dockerfile Normal file
View file

@ -0,0 +1,13 @@
FROM golang AS build-env
WORKDIR /app
COPY go.* ./
RUN go mod download
COPY . ./
RUN go build -v -o pingtunnel
FROM debian
COPY --from=build-env /app/pingtunnel .
COPY GeoLite2-Country.mmdb .
WORKDIR ./

BIN
GeoLite2-Country.mmdb Normal file

Binary file not shown.

View file

@ -1,30 +1,78 @@
# pingtunnel
通过伪造ping把udp流量通过远程服务器转发到目的服务器上。用于突破某些运营商封锁UDP流量。
By forging ping, the udp traffic is forwarded to the destination server through the remote server. Used to break certain operators to block UDP traffic.
# Pingtunnel
Usage:
[<img src="https://img.shields.io/github/license/esrrhs/pingtunnel">](https://github.com/esrrhs/pingtunnel)
[<img src="https://img.shields.io/github/languages/top/esrrhs/pingtunnel">](https://github.com/esrrhs/pingtunnel)
[![Go Report Card](https://goreportcard.com/badge/github.com/esrrhs/pingtunnel)](https://goreportcard.com/report/github.com/esrrhs/pingtunnel)
[<img src="https://img.shields.io/github/v/release/esrrhs/pingtunnel">](https://github.com/esrrhs/pingtunnel/releases)
[<img src="https://img.shields.io/github/downloads/esrrhs/pingtunnel/total">](https://github.com/esrrhs/pingtunnel/releases)
[<img src="https://img.shields.io/docker/pulls/esrrhs/pingtunnel">](https://hub.docker.com/repository/docker/esrrhs/pingtunnel)
[<img src="https://img.shields.io/github/actions/workflow/status/esrrhs/pingtunnel/go.yml?branch=master">](https://github.com/esrrhs/pingtunnel/actions)
pingtunnel -type server
Pingtunnel is a tool that send TCP/UDP traffic over ICMP.
pingtunnel -type client -l LOCAL_IP:4455 -s SERVER_IP -t SERVER_IP:4455
## Note: This tool is only to be used for study and research, do not use it for illegal purposes
-type 服务器或者客户端
client or server
![image](network.jpg)
-l 本地的地址,发到这个端口的流量将转发到服务器
Local address, traffic sent to this port will be forwarded to the server
## Usage
-s 服务器的地址,流量将通过隧道转发到这个服务器
The address of the server, the traffic will be forwarded to this server through the tunnel
### Install server
-t 远端服务器转发的目的地址,流量将转发到这个地址
Destination address forwarded by the remote server, traffic will be forwarded to this address
- First prepare a server with a public IP, such as EC2 on AWS, assuming the domain name or public IP is www.yourserver.com
- Download the corresponding installation package from [releases](https://github.com/esrrhs/pingtunnel/releases), such as pingtunnel_linux64.zip, then decompress and execute with **root** privileges
- “-key” parameter is **int** type, only supports numbers between 0-2147483647
-timeout 本地记录连接超时的时间,单位是秒
The time when the local record connection timed out, in seconds
```
sudo wget (link of latest release)
sudo unzip pingtunnel_linux64.zip
sudo ./pingtunnel -type server
```
- (Optional) Disable system default ping
```
echo 1 > /proc/sys/net/ipv4/icmp_echo_ignore_all
```
### Install the client
- Download the corresponding installation package from [releases](https://github.com/esrrhs/pingtunnel/releases), such as pingtunnel_windows64.zip, and decompress it
- Then run with **administrator** privileges. The commands corresponding to different forwarding functions are as follows.
- If you see a log of ping pong, the connection is normal
- “-key” parameter is **int** type, only supports numbers between 0-2147483647
#### Forward sock5
```
pingtunnel.exe -type client -l :4455 -s www.yourserver.com -sock5 1
```
#### Forward tcp
```
pingtunnel.exe -type client -l :4455 -s www.yourserver.com -t www.yourserver.com:4455 -tcp 1
```
#### Forward udp
```
pingtunnel.exe -type client -l :4455 -s www.yourserver.com -t www.yourserver.com:4455
```
### Use Docker
It can also be started directly with docker, which is more convenient. Same parameters as above
- server:
```
docker run --name pingtunnel-server -d --privileged --network host --restart=always esrrhs/pingtunnel ./pingtunnel -type server -key 123456
```
- client:
```
docker run --name pingtunnel-client -d --restart=always -p 1080:1080 esrrhs/pingtunnel ./pingtunnel -type client -l :1080 -s www.yourserver.com -sock5 1 -key 123456
```
## Thanks for free JetBrains Open Source license
<img src="https://resources.jetbrains.com/storage/products/company/brand/logos/GoLand.png" height="200"/></a>
-sproto 客户端发送ping协议的协议默认是13
The protocol that the client sends the ping. The default is 13.
-rproto 客户端接收ping协议的协议默认是14
The protocol that the client receives the ping. The default is 14.

1
_config.yml Normal file
View file

@ -0,0 +1 @@
theme: jekyll-theme-cayman

806
client.go
View file

@ -1,17 +1,43 @@
package pingtunnel
import (
"fmt"
"github.com/esrrhs/gohome/common"
"github.com/esrrhs/gohome/frame"
"github.com/esrrhs/gohome/loggo"
"github.com/esrrhs/gohome/network"
"github.com/golang/protobuf/proto"
"golang.org/x/net/icmp"
"io"
"math"
"math/rand"
"net"
"sync"
"time"
)
func NewClient(addr string, server string, target string, timeout int, sproto int, rproto int) (*Client, error) {
const (
SEND_PROTO int = 8
RECV_PROTO int = 0
)
ipaddr, err := net.ResolveUDPAddr("udp", addr)
if err != nil {
return nil, err
func NewClient(addr string, server string, target string, timeout int, key int,
tcpmode int, tcpmode_buffersize int, tcpmode_maxwin int, tcpmode_resend_timems int, tcpmode_compress int,
tcpmode_stat int, open_sock5 int, maxconn int, sock5_filter *func(addr string) bool) (*Client, error) {
var ipaddr *net.UDPAddr
var tcpaddr *net.TCPAddr
var err error
if tcpmode > 0 {
tcpaddr, err = net.ResolveTCPAddr("tcp", addr)
if err != nil {
return nil, err
}
} else {
ipaddr, err = net.ResolveUDPAddr("udp", addr)
if err != nil {
return nil, err
}
}
ipaddrServer, err := net.ResolveIPAddr("ip", server)
@ -19,48 +45,93 @@ func NewClient(addr string, server string, target string, timeout int, sproto in
return nil, err
}
rand.Seed(time.Now().UnixNano())
return &Client{
ipaddr: ipaddr,
addr: addr,
ipaddrServer: ipaddrServer,
addrServer: server,
targetAddr: target,
timeout: timeout,
sproto: sproto,
rproto: rproto,
exit: false,
rtt: 0,
id: rand.Intn(math.MaxInt16),
ipaddr: ipaddr,
tcpaddr: tcpaddr,
addr: addr,
ipaddrServer: ipaddrServer,
addrServer: server,
targetAddr: target,
timeout: timeout,
key: key,
tcpmode: tcpmode,
tcpmode_buffersize: tcpmode_buffersize,
tcpmode_maxwin: tcpmode_maxwin,
tcpmode_resend_timems: tcpmode_resend_timems,
tcpmode_compress: tcpmode_compress,
tcpmode_stat: tcpmode_stat,
open_sock5: open_sock5,
maxconn: maxconn,
pongTime: time.Now(),
sock5_filter: sock5_filter,
}, nil
}
type Client struct {
timeout int
sproto int
rproto int
exit bool
rtt time.Duration
workResultLock sync.WaitGroup
maxconn int
ipaddr *net.UDPAddr
addr string
id int
sequence int
timeout int
sproto int
rproto int
key int
tcpmode int
tcpmode_buffersize int
tcpmode_maxwin int
tcpmode_resend_timems int
tcpmode_compress int
tcpmode_stat int
open_sock5 int
sock5_filter *func(addr string) bool
ipaddr *net.UDPAddr
tcpaddr *net.TCPAddr
addr string
ipaddrServer *net.IPAddr
addrServer string
targetAddr string
conn *icmp.PacketConn
listenConn *net.UDPConn
conn *icmp.PacketConn
listenConn *net.UDPConn
tcplistenConn *net.TCPListener
localAddrToConnMap map[string]*ClientConn
localIdToConnMap map[string]*ClientConn
localAddrToConnMap sync.Map
localIdToConnMap sync.Map
sendPacket uint64
recvPacket uint64
sendPacketSize uint64
recvPacketSize uint64
sendPacket uint64
recvPacket uint64
sendPacketSize uint64
recvPacketSize uint64
localAddrToConnMapSize int
localIdToConnMapSize int
recvcontrol chan int
pongTime time.Time
}
type ClientConn struct {
ipaddr *net.UDPAddr
id string
activeTime time.Time
close bool
exit bool
ipaddr *net.UDPAddr
tcpaddr *net.TCPAddr
id string
activeRecvTime time.Time
activeSendTime time.Time
close bool
fm *frame.FrameMgr
}
func (p *Client) Addr() string {
@ -83,145 +154,518 @@ func (p *Client) ServerAddr() string {
return p.addrServer
}
func (p *Client) Run() {
func (p *Client) RTT() time.Duration {
return p.rtt
}
func (p *Client) RecvPacketSize() uint64 {
return p.recvPacketSize
}
func (p *Client) SendPacketSize() uint64 {
return p.sendPacketSize
}
func (p *Client) RecvPacket() uint64 {
return p.recvPacket
}
func (p *Client) SendPacket() uint64 {
return p.sendPacket
}
func (p *Client) LocalIdToConnMapSize() int {
return p.localIdToConnMapSize
}
func (p *Client) LocalAddrToConnMapSize() int {
return p.localAddrToConnMapSize
}
func (p *Client) Run() error {
conn, err := icmp.ListenPacket("ip4:icmp", "")
if err != nil {
fmt.Printf("Error listening for ICMP packets: %s\n", err.Error())
return
loggo.Error("Error listening for ICMP packets: %s", err.Error())
return err
}
defer conn.Close()
p.conn = conn
listener, err := net.ListenUDP("udp", p.ipaddr)
if err != nil {
fmt.Printf("Error listening for udp packets: %s\n", err.Error())
return
if p.tcpmode > 0 {
tcplistenConn, err := net.ListenTCP("tcp", p.tcpaddr)
if err != nil {
loggo.Error("Error listening for tcp packets: %s", err.Error())
return err
}
p.tcplistenConn = tcplistenConn
} else {
listener, err := net.ListenUDP("udp", p.ipaddr)
if err != nil {
loggo.Error("Error listening for udp packets: %s", err.Error())
return err
}
p.listenConn = listener
}
defer listener.Close()
p.listenConn = listener
p.localAddrToConnMap = make(map[string]*ClientConn)
p.localIdToConnMap = make(map[string]*ClientConn)
go p.Accept()
if p.tcpmode > 0 {
go p.AcceptTcp()
} else {
go p.Accept()
}
recv := make(chan *Packet, 10000)
go recvICMP(*p.conn, recv)
p.recvcontrol = make(chan int, 1)
go recvICMP(&p.workResultLock, &p.exit, *p.conn, recv)
interval := time.NewTicker(time.Second)
defer interval.Stop()
go func() {
defer common.CrashLog()
intervalPing := time.NewTicker(time.Second * 10)
defer interval.Stop()
p.workResultLock.Add(1)
defer p.workResultLock.Done()
for {
select {
case <-interval.C:
for !p.exit {
p.checkTimeoutConn()
p.showNet()
case <-intervalPing.C:
p.ping()
case r := <-recv:
p.processPacket(r)
p.showNet()
time.Sleep(time.Second)
}
}()
go func() {
defer common.CrashLog()
p.workResultLock.Add(1)
defer p.workResultLock.Done()
for !p.exit {
p.updateServerAddr()
time.Sleep(time.Second)
}
}()
go func() {
defer common.CrashLog()
p.workResultLock.Add(1)
defer p.workResultLock.Done()
for !p.exit {
select {
case <-p.recvcontrol:
return
case r := <-recv:
p.processPacket(r)
}
}
}()
return nil
}
func (p *Client) Stop() {
p.exit = true
p.recvcontrol <- 1
p.workResultLock.Wait()
p.conn.Close()
if p.tcplistenConn != nil {
p.tcplistenConn.Close()
}
if p.listenConn != nil {
p.listenConn.Close()
}
}
func (p *Client) AcceptTcp() error {
defer common.CrashLog()
p.workResultLock.Add(1)
defer p.workResultLock.Done()
loggo.Info("client waiting local accept tcp")
for !p.exit {
p.tcplistenConn.SetDeadline(time.Now().Add(time.Millisecond * 1000))
conn, err := p.tcplistenConn.AcceptTCP()
if err != nil {
nerr, ok := err.(net.Error)
if !ok || !nerr.Timeout() {
loggo.Info("Error accept tcp %s", err)
continue
}
}
if conn != nil {
if p.open_sock5 > 0 {
go p.AcceptSock5Conn(conn)
} else {
go p.AcceptTcpConn(conn, p.targetAddr)
}
}
}
return nil
}
func (p *Client) AcceptTcpConn(conn *net.TCPConn, targetAddr string) {
defer common.CrashLog()
p.workResultLock.Add(1)
defer p.workResultLock.Done()
tcpsrcaddr := conn.RemoteAddr().(*net.TCPAddr)
if p.maxconn > 0 && p.localIdToConnMapSize >= p.maxconn {
loggo.Info("too many connections %d, client accept new local tcp fail %s", p.localIdToConnMapSize, tcpsrcaddr.String())
return
}
uuid := common.UniqueId()
fm := frame.NewFrameMgr(FRAME_MAX_SIZE, FRAME_MAX_ID, p.tcpmode_buffersize, p.tcpmode_maxwin, p.tcpmode_resend_timems, p.tcpmode_compress, p.tcpmode_stat)
now := time.Now()
clientConn := &ClientConn{exit: false, tcpaddr: tcpsrcaddr, id: uuid, activeRecvTime: now, activeSendTime: now, close: false,
fm: fm}
p.addClientConn(uuid, tcpsrcaddr.String(), clientConn)
loggo.Info("client accept new local tcp %s %s", uuid, tcpsrcaddr.String())
loggo.Info("start connect remote tcp %s %s", uuid, tcpsrcaddr.String())
clientConn.fm.Connect()
startConnectTime := common.GetNowUpdateInSecond()
for !p.exit && !clientConn.exit {
if clientConn.fm.IsConnected() {
break
}
clientConn.fm.Update()
sendlist := clientConn.fm.GetSendList()
for e := sendlist.Front(); e != nil; e = e.Next() {
f := e.Value.(*frame.Frame)
mb, _ := clientConn.fm.MarshalFrame(f)
p.sequence++
sendICMP(p.id, p.sequence, *p.conn, p.ipaddrServer, targetAddr, clientConn.id, (uint32)(MyMsg_DATA), mb,
SEND_PROTO, RECV_PROTO, p.key,
p.tcpmode, p.tcpmode_buffersize, p.tcpmode_maxwin, p.tcpmode_resend_timems, p.tcpmode_compress, p.tcpmode_stat,
p.timeout)
p.sendPacket++
p.sendPacketSize += (uint64)(len(mb))
}
time.Sleep(time.Millisecond * 10)
now := common.GetNowUpdateInSecond()
diffclose := now.Sub(startConnectTime)
if diffclose > time.Second*5 {
loggo.Info("can not connect remote tcp %s %s", uuid, tcpsrcaddr.String())
p.close(clientConn)
return
}
}
if !clientConn.exit {
loggo.Info("connected remote tcp %s %s", uuid, tcpsrcaddr.String())
}
bytes := make([]byte, 10240)
tcpActiveRecvTime := common.GetNowUpdateInSecond()
tcpActiveSendTime := common.GetNowUpdateInSecond()
for !p.exit && !clientConn.exit {
now := common.GetNowUpdateInSecond()
sleep := true
left := common.MinOfInt(clientConn.fm.GetSendBufferLeft(), len(bytes))
if left > 0 {
conn.SetReadDeadline(time.Now().Add(time.Millisecond * 1))
n, err := conn.Read(bytes[0:left])
if err != nil {
nerr, ok := err.(net.Error)
if !ok || !nerr.Timeout() {
loggo.Info("Error read tcp %s %s %s", uuid, tcpsrcaddr.String(), err)
clientConn.fm.Close()
break
}
}
if n > 0 {
sleep = false
clientConn.fm.WriteSendBuffer(bytes[:n])
tcpActiveRecvTime = now
}
}
clientConn.fm.Update()
sendlist := clientConn.fm.GetSendList()
if sendlist.Len() > 0 {
sleep = false
clientConn.activeSendTime = now
for e := sendlist.Front(); e != nil; e = e.Next() {
f := e.Value.(*frame.Frame)
mb, err := clientConn.fm.MarshalFrame(f)
if err != nil {
loggo.Error("Error tcp Marshal %s %s %s", uuid, tcpsrcaddr.String(), err)
continue
}
p.sequence++
sendICMP(p.id, p.sequence, *p.conn, p.ipaddrServer, targetAddr, clientConn.id, (uint32)(MyMsg_DATA), mb,
SEND_PROTO, RECV_PROTO, p.key,
p.tcpmode, 0, 0, 0, 0, 0,
0)
p.sendPacket++
p.sendPacketSize += (uint64)(len(mb))
}
}
if clientConn.fm.GetRecvBufferSize() > 0 {
sleep = false
rr := clientConn.fm.GetRecvReadLineBuffer()
conn.SetWriteDeadline(time.Now().Add(time.Millisecond * 1))
n, err := conn.Write(rr)
if err != nil {
nerr, ok := err.(net.Error)
if !ok || !nerr.Timeout() {
loggo.Info("Error write tcp %s %s %s", uuid, tcpsrcaddr.String(), err)
clientConn.fm.Close()
break
}
}
if n > 0 {
clientConn.fm.SkipRecvBuffer(n)
tcpActiveSendTime = now
}
}
if sleep {
time.Sleep(time.Millisecond * 10)
}
diffrecv := now.Sub(clientConn.activeRecvTime)
diffsend := now.Sub(clientConn.activeSendTime)
tcpdiffrecv := now.Sub(tcpActiveRecvTime)
tcpdiffsend := now.Sub(tcpActiveSendTime)
if diffrecv > time.Second*(time.Duration(p.timeout)) || diffsend > time.Second*(time.Duration(p.timeout)) ||
(tcpdiffrecv > time.Second*(time.Duration(p.timeout)) && tcpdiffsend > time.Second*(time.Duration(p.timeout))) {
loggo.Info("close inactive conn %s %s", clientConn.id, clientConn.tcpaddr.String())
clientConn.fm.Close()
break
}
if clientConn.fm.IsRemoteClosed() {
loggo.Info("closed by remote conn %s %s", clientConn.id, clientConn.tcpaddr.String())
clientConn.fm.Close()
break
}
}
clientConn.fm.Close()
startCloseTime := common.GetNowUpdateInSecond()
for !p.exit && !clientConn.exit {
now := common.GetNowUpdateInSecond()
clientConn.fm.Update()
sendlist := clientConn.fm.GetSendList()
for e := sendlist.Front(); e != nil; e = e.Next() {
f := e.Value.(*frame.Frame)
mb, _ := clientConn.fm.MarshalFrame(f)
p.sequence++
sendICMP(p.id, p.sequence, *p.conn, p.ipaddrServer, targetAddr, clientConn.id, (uint32)(MyMsg_DATA), mb,
SEND_PROTO, RECV_PROTO, p.key,
p.tcpmode, 0, 0, 0, 0, 0,
0)
p.sendPacket++
p.sendPacketSize += (uint64)(len(mb))
}
nodatarecv := true
if clientConn.fm.GetRecvBufferSize() > 0 {
rr := clientConn.fm.GetRecvReadLineBuffer()
conn.SetWriteDeadline(time.Now().Add(time.Millisecond * 100))
n, _ := conn.Write(rr)
if n > 0 {
clientConn.fm.SkipRecvBuffer(n)
nodatarecv = false
}
}
diffclose := now.Sub(startCloseTime)
if diffclose > time.Second*60 {
loggo.Info("close conn had timeout %s %s", clientConn.id, clientConn.tcpaddr.String())
break
}
remoteclosed := clientConn.fm.IsRemoteClosed()
if remoteclosed && nodatarecv {
loggo.Info("remote conn had closed %s %s", clientConn.id, clientConn.tcpaddr.String())
break
}
time.Sleep(time.Millisecond * 100)
}
loggo.Info("close tcp conn %s %s", clientConn.id, clientConn.tcpaddr.String())
conn.Close()
p.close(clientConn)
}
func (p *Client) Accept() error {
fmt.Println("client waiting local accept")
defer common.CrashLog()
p.workResultLock.Add(1)
defer p.workResultLock.Done()
loggo.Info("client waiting local accept udp")
bytes := make([]byte, 10240)
for {
for !p.exit {
p.listenConn.SetReadDeadline(time.Now().Add(time.Millisecond * 100))
n, srcaddr, err := p.listenConn.ReadFromUDP(bytes)
if err != nil {
if neterr, ok := err.(*net.OpError); ok {
if neterr.Timeout() {
// Read timeout
continue
} else {
fmt.Printf("Error read udp %s\n", err)
continue
}
nerr, ok := err.(net.Error)
if !ok || !nerr.Timeout() {
loggo.Info("Error read udp %s", err)
continue
}
}
now := time.Now()
clientConn := p.localAddrToConnMap[srcaddr.String()]
if clientConn == nil {
uuid := UniqueId()
clientConn = &ClientConn{ipaddr: srcaddr, id: uuid, activeTime: now, close: false}
p.localAddrToConnMap[srcaddr.String()] = clientConn
p.localIdToConnMap[uuid] = clientConn
fmt.Printf("client accept new local %s %s\n", uuid, srcaddr.String())
if n <= 0 {
continue
}
clientConn.activeTime = now
sendICMP(*p.conn, p.ipaddrServer, p.targetAddr, clientConn.id, (uint32)(DATA), bytes[:n], p.sproto, p.rproto)
now := common.GetNowUpdateInSecond()
clientConn := p.getClientConnByAddr(srcaddr.String())
if clientConn == nil {
if p.maxconn > 0 && p.localIdToConnMapSize >= p.maxconn {
loggo.Info("too many connections %d, client accept new local udp fail %s", p.localIdToConnMapSize, srcaddr.String())
continue
}
uuid := common.UniqueId()
clientConn = &ClientConn{exit: false, ipaddr: srcaddr, id: uuid, activeRecvTime: now, activeSendTime: now, close: false}
p.addClientConn(uuid, srcaddr.String(), clientConn)
loggo.Info("client accept new local udp %s %s", uuid, srcaddr.String())
}
clientConn.activeSendTime = now
sendICMP(p.id, p.sequence, *p.conn, p.ipaddrServer, p.targetAddr, clientConn.id, (uint32)(MyMsg_DATA), bytes[:n],
SEND_PROTO, RECV_PROTO, p.key,
p.tcpmode, 0, 0, 0, 0, 0,
p.timeout)
p.sequence++
p.sendPacket++
p.sendPacketSize += (uint64)(n)
}
return nil
}
func (p *Client) processPacket(packet *Packet) {
if packet.msgType == PING {
if packet.my.Rproto >= 0 {
return
}
if packet.my.Key != (int32)(p.key) {
return
}
if packet.echoId != p.id {
return
}
if packet.my.Type == (int32)(MyMsg_PING) {
t := time.Time{}
t.UnmarshalBinary(packet.data)
d := time.Now().Sub(t)
fmt.Printf("pong from %s %s\n", packet.src.String(), d.String())
sendICMP(*p.conn, packet.src, "", "", (uint32)(DATA), packet.data, packet.rproto, 0)
t.UnmarshalBinary(packet.my.Data)
now := time.Now()
d := now.Sub(t)
loggo.Info("pong from %s %s", packet.src.String(), d.String())
p.rtt = d
p.pongTime = now
return
}
//fmt.Printf("processPacket %s %s %d\n", packet.id, packet.src.String(), len(packet.data))
if packet.my.Type == (int32)(MyMsg_KICK) {
clientConn := p.getClientConnById(packet.my.Id)
if clientConn != nil {
p.close(clientConn)
loggo.Info("remote kick local %s", packet.my.Id)
}
return
}
clientConn := p.localIdToConnMap[packet.id]
loggo.Debug("processPacket %s %s %d", packet.my.Id, packet.src.String(), len(packet.my.Data))
clientConn := p.getClientConnById(packet.my.Id)
if clientConn == nil {
fmt.Printf("processPacket no conn %s \n", packet.id)
loggo.Debug("processPacket no conn %s ", packet.my.Id)
p.remoteError(packet.my.Id)
return
}
addr := clientConn.ipaddr
now := common.GetNowUpdateInSecond()
clientConn.activeRecvTime = now
now := time.Now()
clientConn.activeTime = now
if p.tcpmode > 0 {
f := &frame.Frame{}
err := proto.Unmarshal(packet.my.Data, f)
if err != nil {
loggo.Error("Unmarshal tcp Error %s", err)
return
}
_, err := p.listenConn.WriteToUDP(packet.data, addr)
if err != nil {
fmt.Printf("WriteToUDP Error read udp %s\n", err)
clientConn.close = true
return
clientConn.fm.OnRecvFrame(f)
} else {
if packet.my.Data == nil {
return
}
addr := clientConn.ipaddr
_, err := p.listenConn.WriteToUDP(packet.my.Data, addr)
if err != nil {
loggo.Info("WriteToUDP Error read udp %s", err)
clientConn.close = true
return
}
}
p.recvPacket++
p.recvPacketSize += (uint64)(len(packet.data))
p.recvPacketSize += (uint64)(len(packet.my.Data))
}
func (p *Client) Close(clientConn *ClientConn) {
if p.localIdToConnMap[clientConn.id] != nil {
delete(p.localIdToConnMap, clientConn.id)
delete(p.localAddrToConnMap, clientConn.ipaddr.String())
}
func (p *Client) close(clientConn *ClientConn) {
clientConn.exit = true
p.deleteClientConn(clientConn.id, clientConn.ipaddr.String())
p.deleteClientConn(clientConn.id, clientConn.tcpaddr.String())
}
func (p *Client) checkTimeoutConn() {
now := time.Now()
for _, conn := range p.localIdToConnMap {
diff := now.Sub(conn.activeTime)
if diff > time.Second*(time.Duration(p.timeout)) {
if p.tcpmode > 0 {
return
}
tmp := make(map[string]*ClientConn)
p.localIdToConnMap.Range(func(key, value interface{}) bool {
id := key.(string)
clientConn := value.(*ClientConn)
tmp[id] = clientConn
return true
})
now := common.GetNowUpdateInSecond()
for _, conn := range tmp {
diffrecv := now.Sub(conn.activeRecvTime)
diffsend := now.Sub(conn.activeSendTime)
if diffrecv > time.Second*(time.Duration(p.timeout)) || diffsend > time.Second*(time.Duration(p.timeout)) {
conn.close = true
}
}
for id, conn := range p.localIdToConnMap {
for id, conn := range tmp {
if conn.close {
fmt.Printf("close inactive conn %s %s\n", id, conn.ipaddr.String())
p.Close(conn)
loggo.Info("close inactive conn %s %s", id, conn.ipaddr.String())
p.close(conn)
}
}
}
@ -229,14 +673,158 @@ func (p *Client) checkTimeoutConn() {
func (p *Client) ping() {
now := time.Now()
b, _ := now.MarshalBinary()
sendICMP(*p.conn, p.ipaddrServer, p.targetAddr, "", (uint32)(PING), b, p.sproto, p.rproto)
fmt.Printf("ping %s %s %d %d\n", p.addrServer, now.String(), p.sproto, p.rproto)
sendICMP(p.id, p.sequence, *p.conn, p.ipaddrServer, "", "", (uint32)(MyMsg_PING), b,
SEND_PROTO, RECV_PROTO, p.key,
0, 0, 0, 0, 0, 0,
0)
loggo.Info("ping %s %s %d %d %d %d", p.addrServer, now.String(), p.sproto, p.rproto, p.id, p.sequence)
p.sequence++
if now.Sub(p.pongTime) > time.Second*3 {
p.rtt = 0
}
}
func (p *Client) showNet() {
fmt.Printf("send %dPacket/s %dKB/s recv %dPacket/s %dKB/s\n", p.sendPacket, p.sendPacketSize/1024, p.recvPacket, p.recvPacketSize/1024)
p.localAddrToConnMapSize = 0
p.localIdToConnMap.Range(func(key, value interface{}) bool {
p.localAddrToConnMapSize++
return true
})
p.localIdToConnMapSize = 0
p.localIdToConnMap.Range(func(key, value interface{}) bool {
p.localIdToConnMapSize++
return true
})
loggo.Info("send %dPacket/s %dKB/s recv %dPacket/s %dKB/s %d/%dConnections",
p.sendPacket, p.sendPacketSize/1024, p.recvPacket, p.recvPacketSize/1024, p.localAddrToConnMapSize, p.localIdToConnMapSize)
p.sendPacket = 0
p.recvPacket = 0
p.sendPacketSize = 0
p.recvPacketSize = 0
}
func (p *Client) AcceptSock5Conn(conn *net.TCPConn) {
defer common.CrashLog()
p.workResultLock.Add(1)
defer p.workResultLock.Done()
var err error = nil
if err = network.Sock5HandshakeBy(conn, "", ""); err != nil {
loggo.Error("socks handshake: %s", err)
conn.Close()
return
}
_, addr, err := network.Sock5GetRequest(conn)
if err != nil {
loggo.Error("error getting request: %s", err)
conn.Close()
return
}
// Sending connection established message immediately to client.
// This some round trip time for creating socks connection with the client.
// But if connection failed, the client will get connection reset error.
_, err = conn.Write([]byte{0x05, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x08, 0x43})
if err != nil {
loggo.Error("send connection confirmation: %s", err)
conn.Close()
return
}
loggo.Info("accept new sock5 conn: %s", addr)
if p.sock5_filter == nil {
p.AcceptTcpConn(conn, addr)
} else {
if (*p.sock5_filter)(addr) {
p.AcceptTcpConn(conn, addr)
return
}
p.AcceptDirectTcpConn(conn, addr)
}
}
func (p *Client) addClientConn(uuid string, addr string, clientConn *ClientConn) {
p.localAddrToConnMap.Store(addr, clientConn)
p.localIdToConnMap.Store(uuid, clientConn)
}
func (p *Client) getClientConnByAddr(addr string) *ClientConn {
ret, ok := p.localAddrToConnMap.Load(addr)
if !ok {
return nil
}
return ret.(*ClientConn)
}
func (p *Client) getClientConnById(uuid string) *ClientConn {
ret, ok := p.localIdToConnMap.Load(uuid)
if !ok {
return nil
}
return ret.(*ClientConn)
}
func (p *Client) deleteClientConn(uuid string, addr string) {
p.localIdToConnMap.Delete(uuid)
p.localAddrToConnMap.Delete(addr)
}
func (p *Client) remoteError(uuid string) {
sendICMP(p.id, p.sequence, *p.conn, p.ipaddrServer, "", uuid, (uint32)(MyMsg_KICK), []byte{},
SEND_PROTO, RECV_PROTO, p.key,
0, 0, 0, 0, 0, 0,
0)
}
func (p *Client) AcceptDirectTcpConn(conn *net.TCPConn, targetAddr string) {
defer common.CrashLog()
p.workResultLock.Add(1)
defer p.workResultLock.Done()
tcpsrcaddr := conn.RemoteAddr().(*net.TCPAddr)
loggo.Info("client accept new direct local tcp %s %s", tcpsrcaddr.String(), targetAddr)
tcpaddrTarget, err := net.ResolveTCPAddr("tcp", targetAddr)
if err != nil {
loggo.Info("direct local tcp ResolveTCPAddr fail: %s %s", targetAddr, err.Error())
return
}
targetconn, err := net.DialTCP("tcp", nil, tcpaddrTarget)
if err != nil {
loggo.Info("direct local tcp DialTCP fail: %s %s", targetAddr, err.Error())
return
}
go p.transfer(conn, targetconn, conn.RemoteAddr().String(), targetconn.RemoteAddr().String())
go p.transfer(targetconn, conn, targetconn.RemoteAddr().String(), conn.RemoteAddr().String())
loggo.Info("client accept new direct local tcp ok %s %s", tcpsrcaddr.String(), targetAddr)
}
func (p *Client) transfer(destination io.WriteCloser, source io.ReadCloser, dst string, src string) {
defer common.CrashLog()
defer destination.Close()
defer source.Close()
loggo.Info("client begin transfer from %s -> %s", src, dst)
io.Copy(destination, source)
loggo.Info("client end transfer from %s -> %s", src, dst)
}
func (p *Client) updateServerAddr() {
ipaddrServer, err := net.ResolveIPAddr("ip", p.addrServer)
if err != nil {
return
}
if p.ipaddrServer.String() != ipaddrServer.String() {
p.ipaddrServer = ipaddrServer
}
}

View file

@ -3,22 +3,66 @@ package main
import (
"flag"
"fmt"
"github.com/esrrhs/gohome/common"
"github.com/esrrhs/gohome/geoip"
"github.com/esrrhs/gohome/loggo"
"github.com/esrrhs/pingtunnel"
"net"
"net/http"
_ "net/http/pprof"
"strconv"
"time"
)
var usage = `
通过伪造ping把udp流量通过远程服务器转发到目的服务器上用于突破某些运营商封锁UDP流量
By forging ping, the udp traffic is forwarded to the destination server through the remote server. Used to break certain operators to block UDP traffic.
通过伪造ping把tcp/udp/sock5流量通过远程服务器转发到目的服务器上用于突破某些运营商封锁TCP/UDP流量
By forging ping, the tcp/udp/sock5 traffic is forwarded to the destination server through the remote server. Used to break certain operators to block TCP/UDP traffic.
Usage:
// server
pingtunnel -type server
// client, Forward udp
pingtunnel -type client -l LOCAL_IP:4455 -s SERVER_IP -t SERVER_IP:4455
// client, Forward tcp
pingtunnel -type client -l LOCAL_IP:4455 -s SERVER_IP -t SERVER_IP:4455 -tcp 1
// client, Forward sock5, implicitly open tcp, so no target server is needed
pingtunnel -type client -l LOCAL_IP:4455 -s SERVER_IP -sock5 1
-type 服务器或者客户端
client or server
服务器参数server param:
-key 设置的纯数字密码默认0, 参数为int类型范围从0-2147483647不可夹杂字母特殊符号
Set password, default 0
-nolog 不写日志文件只打印标准输出默认0
Do not write log files, only print standard output, default 0 is off
-noprint 不打印屏幕输出默认0
Do not print standard output, default 0 is off
-loglevel 日志文件等级默认info
log level, default is info
-maxconn 最大连接数默认0不受限制
the max num of connections, default 0 is no limit
-maxprt server最大处理线程数默认100
max process thread in server, default 100
-maxprb server最大处理线程buffer数默认1000
max process thread's buffer in server, default 1000
-conntt server发起连接到目标地址的超时时间默认1000ms
The timeout period for the server to initiate a connection to the destination address. The default is 1000ms.
客户端参数client param:
-l 本地的地址发到这个端口的流量将转发到服务器
Local address, traffic sent to this port will be forwarded to the server
@ -31,58 +75,195 @@ Usage:
-timeout 本地记录连接超时的时间单位是秒默认60s
The time when the local record connection timed out, in seconds, 60 seconds by default
-sproto 客户端发送ping协议的协议默认是13
The protocol that the client sends the ping. The default is 13.
-key 设置的密码默认0
Set password, default 0
-rproto 客户端接收ping协议的协议默认是14
The protocol that the client receives the ping. The default is 14.
-tcp 设置是否转发tcp默认0
Set the switch to forward tcp, the default is 0
-tcp_bs tcp的发送接收缓冲区大小默认1MB
Tcp send and receive buffer size, default 1MB
-tcp_mw tcp的最大窗口默认20000
The maximum window of tcp, the default is 20000
-tcp_rst tcp的超时发送时间默认400ms
Tcp timeout resend time, default 400ms
-tcp_gz 当数据包超过这个大小tcp将压缩数据0表示不压缩默认0
Tcp will compress data when the packet exceeds this size, 0 means no compression, default 0
-tcp_stat 打印tcp的监控默认0
Print tcp connection statistic, default 0 is off
-nolog 不写日志文件只打印标准输出默认0
Do not write log files, only print standard output, default 0 is off
-noprint 不打印屏幕输出默认0
Do not print standard output, default 0 is off
-loglevel 日志文件等级默认info
log level, default is info
-sock5 开启sock5转发默认0
Turn on sock5 forwarding, default 0 is off
-profile 在指定端口开启性能检测默认0不开启
Enable performance detection on the specified port. The default 0 is not enabled.
-s5filter sock5模式设置转发过滤默认全转发设置CN代表CN地区的直连不转发
Set the forwarding filter in the sock5 mode. The default is full forwarding. For example, setting the CN indicates that the Chinese address is not forwarded.
-s5ftfile sock5模式转发过滤的数据文件默认读取当前目录的GeoLite2-Country.mmdb
The data file in sock5 filter mode, the default reading of the current directory GeoLite2-Country.mmdb
`
func main() {
defer common.CrashLog()
t := flag.String("type", "", "client or server")
listen := flag.String("l", "", "listen addr")
target := flag.String("t", "", "target addr")
server := flag.String("s", "", "server addr")
timeout := flag.Int("timeout", 60, "conn timeout")
sproto := flag.Int("sproto", 13, "send ping proto")
rproto := flag.Int("rproto", 14, "recv ping proto")
key := flag.Int("key", 0, "key")
tcpmode := flag.Int("tcp", 0, "tcp mode")
tcpmode_buffersize := flag.Int("tcp_bs", 1*1024*1024, "tcp mode buffer size")
tcpmode_maxwin := flag.Int("tcp_mw", 20000, "tcp mode max win")
tcpmode_resend_timems := flag.Int("tcp_rst", 400, "tcp mode resend time ms")
tcpmode_compress := flag.Int("tcp_gz", 0, "tcp data compress")
nolog := flag.Int("nolog", 0, "write log file")
noprint := flag.Int("noprint", 0, "print stdout")
tcpmode_stat := flag.Int("tcp_stat", 0, "print tcp stat")
loglevel := flag.String("loglevel", "info", "log level")
open_sock5 := flag.Int("sock5", 0, "sock5 mode")
maxconn := flag.Int("maxconn", 0, "max num of connections")
max_process_thread := flag.Int("maxprt", 100, "max process thread in server")
max_process_buffer := flag.Int("maxprb", 1000, "max process thread's buffer in server")
profile := flag.Int("profile", 0, "open profile")
conntt := flag.Int("conntt", 1000, "the connect call's timeout")
s5filter := flag.String("s5filter", "", "sock5 filter")
s5ftfile := flag.String("s5ftfile", "GeoLite2-Country.mmdb", "sock5 filter file")
flag.Usage = func() {
fmt.Printf(usage)
}
flag.Parse()
if (*t != "client" && *t != "server") || (*t == "client" && (len(*listen) == 0 || len(*target) == 0 || len(*server) == 0)) {
if *t != "client" && *t != "server" {
flag.Usage()
return
}
if *t == "client" {
if len(*listen) == 0 || len(*server) == 0 {
flag.Usage()
return
}
if *open_sock5 == 0 && len(*target) == 0 {
flag.Usage()
return
}
if *open_sock5 != 0 {
*tcpmode = 1
}
}
if *tcpmode_maxwin*10 > pingtunnel.FRAME_MAX_ID {
fmt.Println("set tcp win to big, max = " + strconv.Itoa(pingtunnel.FRAME_MAX_ID/10))
return
}
fmt.Println("start...")
level := loggo.LEVEL_INFO
if loggo.NameToLevel(*loglevel) >= 0 {
level = loggo.NameToLevel(*loglevel)
}
loggo.Ini(loggo.Config{
Level: level,
Prefix: "pingtunnel",
MaxDay: 3,
NoLogFile: *nolog > 0,
NoPrint: *noprint > 0,
})
loggo.Info("start...")
loggo.Info("key %d", *key)
if *t == "server" {
s, err := pingtunnel.NewServer(*timeout)
s, err := pingtunnel.NewServer(*key, *maxconn, *max_process_thread, *max_process_buffer, *conntt)
if err != nil {
fmt.Printf("ERROR: %s\n", err.Error())
loggo.Error("ERROR: %s", err.Error())
return
}
fmt.Printf("Server start\n")
s.Run()
}
if *t == "client" {
fmt.Printf("type %s\n", *t)
fmt.Printf("listen %s\n", *listen)
fmt.Printf("server %s\n", *server)
fmt.Printf("target %s\n", *target)
c, err := pingtunnel.NewClient(*listen, *server, *target, *timeout, *sproto, *rproto)
loggo.Info("Server start")
err = s.Run()
if err != nil {
fmt.Printf("ERROR: %s\n", err.Error())
loggo.Error("Run ERROR: %s", err.Error())
return
}
fmt.Printf("Client Listen %s (%s) Server %s (%s) TargetPort %s:\n", c.Addr(), c.IPAddr(),
} else if *t == "client" {
loggo.Info("type %s", *t)
loggo.Info("listen %s", *listen)
loggo.Info("server %s", *server)
loggo.Info("target %s", *target)
if *tcpmode == 0 {
*tcpmode_buffersize = 0
*tcpmode_maxwin = 0
*tcpmode_resend_timems = 0
*tcpmode_compress = 0
*tcpmode_stat = 0
}
if len(*s5filter) > 0 {
err := geoip.Load(*s5ftfile)
if err != nil {
loggo.Error("Load Sock5 ip file ERROR: %s", err.Error())
return
}
}
filter := func(addr string) bool {
if len(*s5filter) <= 0 {
return true
}
taddr, err := net.ResolveTCPAddr("tcp", addr)
if err != nil {
return false
}
ret, err := geoip.GetCountryIsoCode(taddr.IP.String())
if err != nil {
return false
}
if len(ret) <= 0 {
return false
}
return ret != *s5filter
}
c, err := pingtunnel.NewClient(*listen, *server, *target, *timeout, *key,
*tcpmode, *tcpmode_buffersize, *tcpmode_maxwin, *tcpmode_resend_timems, *tcpmode_compress,
*tcpmode_stat, *open_sock5, *maxconn, &filter)
if err != nil {
loggo.Error("ERROR: %s", err.Error())
return
}
loggo.Info("Client Listen %s (%s) Server %s (%s) TargetPort %s:", c.Addr(), c.IPAddr(),
c.ServerAddr(), c.ServerIPAddr(), c.TargetAddr())
c.Run()
err = c.Run()
if err != nil {
loggo.Error("Run ERROR: %s", err.Error())
return
}
} else {
return
}
if *profile > 0 {
go http.ListenAndServe("0.0.0.0:"+strconv.Itoa(*profile), nil)
}
for {
time.Sleep(time.Hour)
}
}

2
docker-compose/.env Normal file
View file

@ -0,0 +1,2 @@
KEY=123456
SERVER=www.yourserver.com

16
docker-compose/Readme.md Normal file
View file

@ -0,0 +1,16 @@
Deploy with docker-compose
===========================
**First** edit `.env` file in this directory to your appropriate value.
**Then** run stack with these commands:
- in the server
```
docker-compose -f server.yml up -d
```
- in client machine
```
docker-compose -f client.yml up -d
```
**Now** use socks5 proxy at port `1080` of your client machine

View file

@ -0,0 +1,9 @@
version: "3.7"
services:
pingtunnelServer:
image: esrrhs/pingtunnel:latest
restart: always
ports:
- 1080:1080
command: "./pingtunnel -type client -l 0.0.0.0:1080 -s ${SERVER} -sock5 1 -key ${KEY}"

View file

@ -0,0 +1,8 @@
version: "3.7"
services:
pingtunnelServer:
image: esrrhs/pingtunnel:latest
restart: always
network_mode: host
command: "./pingtunnel -type server -key ${KEY}"

1
gen.bat Normal file
View file

@ -0,0 +1 @@
protoc --go_out=. *.proto

18
go.mod Normal file
View file

@ -0,0 +1,18 @@
module github.com/esrrhs/pingtunnel
go 1.18
require (
github.com/esrrhs/gohome v0.0.0-20231102120537-c519efbde976
github.com/golang/protobuf v1.5.3
golang.org/x/net v0.17.0
)
require (
github.com/OneOfOne/xxhash v1.2.8 // indirect
github.com/google/uuid v1.4.0 // indirect
github.com/oschwald/geoip2-golang v1.9.0 // indirect
github.com/oschwald/maxminddb-golang v1.12.0 // indirect
golang.org/x/sys v0.13.0 // indirect
google.golang.org/protobuf v1.31.0 // indirect
)

29
go.sum Normal file
View file

@ -0,0 +1,29 @@
github.com/OneOfOne/xxhash v1.2.8 h1:31czK/TI9sNkxIKfaUfGlU47BAxQ0ztGgd9vPyqimf8=
github.com/OneOfOne/xxhash v1.2.8/go.mod h1:eZbhyaAYD41SGSSsnmcpxVoRiQ/MPUTjUdIIOT9Um7Q=
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
github.com/esrrhs/gohome v0.0.0-20231102120537-c519efbde976 h1:av0d/lRou1Z5cxdSQFwtVcqJjokFI5pJyyr63iAuYis=
github.com/esrrhs/gohome v0.0.0-20231102120537-c519efbde976/go.mod h1:S5fYcOFy4nUPnkYg7D9hIp+SwBR9kCBiOYmWVW42Yhs=
github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg=
github.com/golang/protobuf v1.5.3/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY=
github.com/google/go-cmp v0.5.5 h1:Khx7svrCpmxxtHBq5j2mp/xVjsi8hQMfNLvJFAlrGgU=
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
github.com/google/uuid v1.4.0 h1:MtMxsa51/r9yyhkyLsVeVt0B+BGQZzpQiTQ4eHZ8bc4=
github.com/google/uuid v1.4.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
github.com/oschwald/geoip2-golang v1.9.0 h1:uvD3O6fXAXs+usU+UGExshpdP13GAqp4GBrzN7IgKZc=
github.com/oschwald/geoip2-golang v1.9.0/go.mod h1:BHK6TvDyATVQhKNbQBdrj9eAvuwOMi2zSFXizL3K81Y=
github.com/oschwald/maxminddb-golang v1.12.0 h1:9FnTOD0YOhP7DGxGsq4glzpGy5+w7pq50AS6wALUMYs=
github.com/oschwald/maxminddb-golang v1.12.0/go.mod h1:q0Nob5lTCqyQ8WT6FYgS1L7PXKVVbgiymefNwIjPzgY=
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk=
golang.org/x/net v0.17.0 h1:pVaXccu2ozPjCXewfr1S7xza/zcXTity9cCdXQYSjIM=
golang.org/x/net v0.17.0/go.mod h1:NxSsAGuq816PNPmqtQdLE42eU2Fs7NoRIZrHJAlaCOE=
golang.org/x/sys v0.13.0 h1:Af8nKPmuFypiUBjVoU9V20FiaFXOcuZI21p0ycVYYGE=
golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4=
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
google.golang.org/protobuf v1.31.0 h1:g0LDEJHgrBl9N9r17Ru3sqWhkIx2NB67okBHPwC7hs8=
google.golang.org/protobuf v1.31.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=

228
msg.pb.go Normal file
View file

@ -0,0 +1,228 @@
// Code generated by protoc-gen-go. DO NOT EDIT.
// source: msg.proto
package pingtunnel
import (
fmt "fmt"
proto "github.com/golang/protobuf/proto"
math "math"
)
// Reference imports to suppress errors if they are not otherwise used.
var _ = proto.Marshal
var _ = fmt.Errorf
var _ = math.Inf
// This is a compile-time assertion to ensure that this generated file
// is compatible with the proto package it is being compiled against.
// A compilation error at this line likely means your copy of the
// proto package needs to be updated.
const _ = proto.ProtoPackageIsVersion3 // please upgrade the proto package
type MyMsg_TYPE int32
const (
MyMsg_DATA MyMsg_TYPE = 0
MyMsg_PING MyMsg_TYPE = 1
MyMsg_KICK MyMsg_TYPE = 2
MyMsg_MAGIC MyMsg_TYPE = 57005
)
var MyMsg_TYPE_name = map[int32]string{
0: "DATA",
1: "PING",
2: "KICK",
57005: "MAGIC",
}
var MyMsg_TYPE_value = map[string]int32{
"DATA": 0,
"PING": 1,
"KICK": 2,
"MAGIC": 57005,
}
func (x MyMsg_TYPE) String() string {
return proto.EnumName(MyMsg_TYPE_name, int32(x))
}
func (MyMsg_TYPE) EnumDescriptor() ([]byte, []int) {
return fileDescriptor_c06e4cca6c2cc899, []int{0, 0}
}
type MyMsg struct {
Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"`
Type int32 `protobuf:"varint,2,opt,name=type,proto3" json:"type,omitempty"`
Target string `protobuf:"bytes,3,opt,name=target,proto3" json:"target,omitempty"`
Data []byte `protobuf:"bytes,4,opt,name=data,proto3" json:"data,omitempty"`
Rproto int32 `protobuf:"zigzag32,5,opt,name=rproto,proto3" json:"rproto,omitempty"`
Magic int32 `protobuf:"zigzag32,6,opt,name=magic,proto3" json:"magic,omitempty"`
Key int32 `protobuf:"zigzag32,7,opt,name=key,proto3" json:"key,omitempty"`
Timeout int32 `protobuf:"varint,8,opt,name=timeout,proto3" json:"timeout,omitempty"`
Tcpmode int32 `protobuf:"varint,9,opt,name=tcpmode,proto3" json:"tcpmode,omitempty"`
TcpmodeBuffersize int32 `protobuf:"varint,10,opt,name=tcpmode_buffersize,json=tcpmodeBuffersize,proto3" json:"tcpmode_buffersize,omitempty"`
TcpmodeMaxwin int32 `protobuf:"varint,11,opt,name=tcpmode_maxwin,json=tcpmodeMaxwin,proto3" json:"tcpmode_maxwin,omitempty"`
TcpmodeResendTimems int32 `protobuf:"varint,12,opt,name=tcpmode_resend_timems,json=tcpmodeResendTimems,proto3" json:"tcpmode_resend_timems,omitempty"`
TcpmodeCompress int32 `protobuf:"varint,13,opt,name=tcpmode_compress,json=tcpmodeCompress,proto3" json:"tcpmode_compress,omitempty"`
TcpmodeStat int32 `protobuf:"varint,14,opt,name=tcpmode_stat,json=tcpmodeStat,proto3" json:"tcpmode_stat,omitempty"`
XXX_NoUnkeyedLiteral struct{} `json:"-"`
XXX_unrecognized []byte `json:"-"`
XXX_sizecache int32 `json:"-"`
}
func (m *MyMsg) Reset() { *m = MyMsg{} }
func (m *MyMsg) String() string { return proto.CompactTextString(m) }
func (*MyMsg) ProtoMessage() {}
func (*MyMsg) Descriptor() ([]byte, []int) {
return fileDescriptor_c06e4cca6c2cc899, []int{0}
}
func (m *MyMsg) XXX_Unmarshal(b []byte) error {
return xxx_messageInfo_MyMsg.Unmarshal(m, b)
}
func (m *MyMsg) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) {
return xxx_messageInfo_MyMsg.Marshal(b, m, deterministic)
}
func (m *MyMsg) XXX_Merge(src proto.Message) {
xxx_messageInfo_MyMsg.Merge(m, src)
}
func (m *MyMsg) XXX_Size() int {
return xxx_messageInfo_MyMsg.Size(m)
}
func (m *MyMsg) XXX_DiscardUnknown() {
xxx_messageInfo_MyMsg.DiscardUnknown(m)
}
var xxx_messageInfo_MyMsg proto.InternalMessageInfo
func (m *MyMsg) GetId() string {
if m != nil {
return m.Id
}
return ""
}
func (m *MyMsg) GetType() int32 {
if m != nil {
return m.Type
}
return 0
}
func (m *MyMsg) GetTarget() string {
if m != nil {
return m.Target
}
return ""
}
func (m *MyMsg) GetData() []byte {
if m != nil {
return m.Data
}
return nil
}
func (m *MyMsg) GetRproto() int32 {
if m != nil {
return m.Rproto
}
return 0
}
func (m *MyMsg) GetMagic() int32 {
if m != nil {
return m.Magic
}
return 0
}
func (m *MyMsg) GetKey() int32 {
if m != nil {
return m.Key
}
return 0
}
func (m *MyMsg) GetTimeout() int32 {
if m != nil {
return m.Timeout
}
return 0
}
func (m *MyMsg) GetTcpmode() int32 {
if m != nil {
return m.Tcpmode
}
return 0
}
func (m *MyMsg) GetTcpmodeBuffersize() int32 {
if m != nil {
return m.TcpmodeBuffersize
}
return 0
}
func (m *MyMsg) GetTcpmodeMaxwin() int32 {
if m != nil {
return m.TcpmodeMaxwin
}
return 0
}
func (m *MyMsg) GetTcpmodeResendTimems() int32 {
if m != nil {
return m.TcpmodeResendTimems
}
return 0
}
func (m *MyMsg) GetTcpmodeCompress() int32 {
if m != nil {
return m.TcpmodeCompress
}
return 0
}
func (m *MyMsg) GetTcpmodeStat() int32 {
if m != nil {
return m.TcpmodeStat
}
return 0
}
func init() {
proto.RegisterEnum("MyMsg_TYPE", MyMsg_TYPE_name, MyMsg_TYPE_value)
proto.RegisterType((*MyMsg)(nil), "MyMsg")
}
func init() { proto.RegisterFile("msg.proto", fileDescriptor_c06e4cca6c2cc899) }
var fileDescriptor_c06e4cca6c2cc899 = []byte{
// 342 bytes of a gzipped FileDescriptorProto
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x3c, 0x90, 0xdb, 0x6a, 0xe2, 0x50,
0x14, 0x86, 0x27, 0x27, 0x0f, 0xcb, 0xe8, 0xc4, 0x35, 0x07, 0xd6, 0x65, 0x46, 0x18, 0xc8, 0x5c,
0xcc, 0xc0, 0xb4, 0x4f, 0xa0, 0xb6, 0x88, 0x48, 0x8a, 0xa4, 0xde, 0xb4, 0x37, 0x12, 0xcd, 0x36,
0x84, 0x36, 0x07, 0xb2, 0xb7, 0xb4, 0xf6, 0x9d, 0xfa, 0x08, 0x7d, 0x8d, 0x3e, 0x4f, 0xc9, 0x72,
0xa7, 0x77, 0xff, 0xff, 0x7f, 0x5f, 0xc8, 0x62, 0x43, 0x3f, 0x97, 0xe9, 0xbf, 0xaa, 0x2e, 0x55,
0x39, 0x79, 0xb7, 0xc0, 0x09, 0x4f, 0xa1, 0x4c, 0x71, 0x04, 0x66, 0x96, 0x90, 0xe1, 0x1b, 0x41,
0x3f, 0x32, 0xb3, 0x04, 0x11, 0x6c, 0x75, 0xaa, 0x04, 0x99, 0xbe, 0x11, 0x38, 0x11, 0x67, 0xfc,
0x09, 0x1d, 0x15, 0xd7, 0xa9, 0x50, 0x64, 0xb1, 0xa7, 0x5b, 0xe3, 0x26, 0xb1, 0x8a, 0xc9, 0xf6,
0x8d, 0xc0, 0x8d, 0x38, 0x37, 0x6e, 0xcd, 0xff, 0x20, 0xc7, 0x37, 0x82, 0x71, 0xa4, 0x1b, 0x7e,
0x07, 0x27, 0x8f, 0xd3, 0x6c, 0x4f, 0x1d, 0x9e, 0xcf, 0x05, 0x3d, 0xb0, 0x1e, 0xc4, 0x89, 0xba,
0xbc, 0x35, 0x11, 0x09, 0xba, 0x2a, 0xcb, 0x45, 0x79, 0x54, 0xd4, 0xe3, 0x13, 0xda, 0xca, 0x64,
0x5f, 0xe5, 0x65, 0x22, 0xa8, 0xaf, 0xc9, 0xb9, 0xe2, 0x5f, 0x40, 0x1d, 0xb7, 0xbb, 0xe3, 0xe1,
0x20, 0x6a, 0x99, 0xbd, 0x08, 0x02, 0x96, 0xc6, 0x9a, 0xcc, 0x3e, 0x01, 0xfe, 0x86, 0x51, 0xab,
0xe7, 0xf1, 0xf3, 0x53, 0x56, 0xd0, 0x80, 0xd5, 0xa1, 0x5e, 0x43, 0x1e, 0xf1, 0x02, 0x7e, 0xb4,
0x5a, 0x2d, 0xa4, 0x28, 0x92, 0x6d, 0x73, 0x49, 0x2e, 0xc9, 0x65, 0xfb, 0x9b, 0x86, 0x11, 0xb3,
0x0d, 0x23, 0xfc, 0x03, 0x5e, 0xfb, 0xcd, 0xbe, 0xcc, 0xab, 0x5a, 0x48, 0x49, 0x43, 0xd6, 0xbf,
0xea, 0x7d, 0xae, 0x67, 0xfc, 0x05, 0x6e, 0xab, 0x4a, 0x15, 0x2b, 0x1a, 0xb1, 0x36, 0xd0, 0xdb,
0xad, 0x8a, 0xd5, 0xe4, 0x3f, 0xd8, 0x9b, 0xbb, 0xf5, 0x35, 0xf6, 0xc0, 0xbe, 0x9a, 0x6e, 0xa6,
0xde, 0x97, 0x26, 0xad, 0x97, 0x37, 0x0b, 0xcf, 0x68, 0xd2, 0x6a, 0x39, 0x5f, 0x79, 0x26, 0x0e,
0xc0, 0x09, 0xa7, 0x8b, 0xe5, 0xdc, 0x7b, 0x7d, 0xb3, 0x66, 0xee, 0x3d, 0x54, 0x59, 0x91, 0xaa,
0x63, 0x51, 0x88, 0xc7, 0x5d, 0x87, 0xdf, 0xfe, 0xf2, 0x23, 0x00, 0x00, 0xff, 0xff, 0x59, 0xbc,
0x55, 0x76, 0xfa, 0x01, 0x00, 0x00,
}

26
msg.proto Normal file
View file

@ -0,0 +1,26 @@
syntax = "proto3";
option go_package = "pingtunnel";
message MyMsg {
enum TYPE {
DATA = 0;
PING = 1;
KICK = 2;
MAGIC = 0xdead;
}
string id = 1;
int32 type = 2;
string target = 3;
bytes data = 4;
sint32 rproto = 5;
sint32 magic = 6;
sint32 key = 7;
int32 timeout = 8;
int32 tcpmode = 9;
int32 tcpmode_buffersize = 10;
int32 tcpmode_maxwin = 11;
int32 tcpmode_resend_timems = 12;
int32 tcpmode_compress = 13;
int32 tcpmode_stat = 14;
}

BIN
network.jpg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

56
pack.sh Executable file
View file

@ -0,0 +1,56 @@
#! /bin/bash
#set -x
NAME="pingtunnel"
export GO111MODULE=on
#go tool dist list
build_list=$(go tool dist list)
rm pack -rf
rm pack.zip -f
mkdir pack
go mod tidy
for line in $build_list; do
os=$(echo "$line" | awk -F"/" '{print $1}')
arch=$(echo "$line" | awk -F"/" '{print $2}')
echo "os="$os" arch="$arch" start build"
if [ $os == "android" ]; then
continue
fi
if [ $os == "ios" ]; then
continue
fi
if [ $arch == "wasm" ]; then
continue
fi
CGO_ENABLED=0 GOOS=$os GOARCH=$arch go build -ldflags="-s -w"
if [ $? -ne 0 ]; then
echo "os="$os" arch="$arch" build fail"
exit 1
fi
if [ $os = "windows" ]; then
zip ${NAME}_"${os}"_"${arch}"".zip" $NAME".exe"
if [ $? -ne 0 ]; then
echo "os="$os" arch="$arch" zip fail"
exit 1
fi
mv ${NAME}_"${os}"_"${arch}"".zip" pack/
rm $NAME".exe" -f
else
zip ${NAME}_"${os}"_"${arch}"".zip" $NAME
if [ $? -ne 0 ]; then
echo "os="$os" arch="$arch" zip fail"
exit 1
fi
mv ${NAME}_"${os}"_"${arch}"".zip" pack/
rm $NAME -f
fi
echo "os="$os" arch="$arch" done build"
done
zip pack.zip pack/ -r
echo "all done"

View file

@ -1,225 +1,119 @@
package pingtunnel
import (
"crypto/md5"
"crypto/rand"
"encoding/base64"
"encoding/binary"
"encoding/hex"
"fmt"
"github.com/esrrhs/gohome/common"
"github.com/esrrhs/gohome/loggo"
"github.com/golang/protobuf/proto"
"golang.org/x/net/icmp"
"golang.org/x/net/ipv4"
"io"
"net"
"syscall"
"sync"
"time"
)
const (
DATA uint32 = 0x01010101
PING uint32 = 0x02020202
END uint32 = 0xAAAABBBB
)
// An Echo represents an ICMP echo request or reply message body.
type MyMsg struct {
TYPE uint32
ID string
TARGET string
Data []byte
RPROTO uint16
ENDTYPE uint32
}
// Len implements the Len method of MessageBody interface.
func (p *MyMsg) Len(proto int) int {
if p == nil {
return 0
}
return 4 + p.LenString(p.ID) + p.LenString(p.TARGET) + p.LenData(p.Data) + 2 + 4
}
func (p *MyMsg) LenString(s string) int {
return 2 + len(s)
}
func (p *MyMsg) LenData(data []byte) int {
return 2 + len(data)
}
// Marshal implements the Marshal method of MessageBody interface.
func (p *MyMsg) Marshal(proto int) ([]byte, error) {
b := make([]byte, p.Len(proto))
binary.BigEndian.PutUint32(b[:4], uint32(p.TYPE))
id := p.MarshalString(p.ID)
copy(b[4:], id)
target := p.MarshalString(p.TARGET)
copy(b[4+p.LenString(p.ID):], target)
data := p.MarshalData(p.Data)
copy(b[4+p.LenString(p.ID)+p.LenString(p.TARGET):], data)
binary.BigEndian.PutUint16(b[4+p.LenString(p.ID)+p.LenString(p.TARGET)+p.LenData(p.Data):], uint16(p.RPROTO))
binary.BigEndian.PutUint32(b[4+p.LenString(p.ID)+p.LenString(p.TARGET)+p.LenData(p.Data)+2:], uint32(p.ENDTYPE))
return b, nil
}
func (p *MyMsg) MarshalString(s string) []byte {
b := make([]byte, p.LenString(s))
binary.BigEndian.PutUint16(b[:2], uint16(len(s)))
copy(b[2:], []byte(s))
return b
}
func (p *MyMsg) MarshalData(data []byte) []byte {
b := make([]byte, p.LenData(data))
binary.BigEndian.PutUint16(b[:2], uint16(len(data)))
copy(b[2:], []byte(data))
return b
}
// Marshal implements the Marshal method of MessageBody interface.
func (p *MyMsg) Unmarshal(b []byte) error {
defer func() {
recover()
}()
p.TYPE = binary.BigEndian.Uint32(b[:4])
p.ID = p.UnmarshalString(b[4:])
p.TARGET = p.UnmarshalString(b[4+p.LenString(p.ID):])
p.Data = p.UnmarshalData(b[4+p.LenString(p.ID)+p.LenString(p.TARGET):])
p.RPROTO = binary.BigEndian.Uint16(b[4+p.LenString(p.ID)+p.LenString(p.TARGET)+p.LenData(p.Data):])
p.ENDTYPE = binary.BigEndian.Uint32(b[4+p.LenString(p.ID)+p.LenString(p.TARGET)+p.LenData(p.Data)+2:])
return nil
}
func (p *MyMsg) UnmarshalString(b []byte) string {
len := binary.BigEndian.Uint16(b[:2])
if len > 32 || len < 0 {
panic(nil)
}
data := make([]byte, len)
copy(data, b[2:])
return string(data)
}
func (p *MyMsg) UnmarshalData(b []byte) []byte {
len := binary.BigEndian.Uint16(b[:2])
if len > 2048 || len < 0 {
panic(nil)
}
data := make([]byte, len)
copy(data, b[2:])
return data
}
func sendICMP(conn icmp.PacketConn, server *net.IPAddr, target string, connId string, msgType uint32, data []byte, sproto int, rproto int) {
func sendICMP(id int, sequence int, conn icmp.PacketConn, server *net.IPAddr, target string,
connId string, msgType uint32, data []byte, sproto int, rproto int, key int,
tcpmode int, tcpmode_buffer_size int, tcpmode_maxwin int, tcpmode_resend_time int, tcpmode_compress int, tcpmode_stat int,
timeout int) {
m := &MyMsg{
ID: connId,
TYPE: msgType,
TARGET: target,
Data: data,
RPROTO: (uint16)(rproto),
ENDTYPE: END,
Id: connId,
Type: (int32)(msgType),
Target: target,
Data: data,
Rproto: (int32)(rproto),
Key: (int32)(key),
Tcpmode: (int32)(tcpmode),
TcpmodeBuffersize: (int32)(tcpmode_buffer_size),
TcpmodeMaxwin: (int32)(tcpmode_maxwin),
TcpmodeResendTimems: (int32)(tcpmode_resend_time),
TcpmodeCompress: (int32)(tcpmode_compress),
TcpmodeStat: (int32)(tcpmode_stat),
Timeout: (int32)(timeout),
Magic: (int32)(MyMsg_MAGIC),
}
mb, err := proto.Marshal(m)
if err != nil {
loggo.Error("sendICMP Marshal MyMsg error %s %s", server.String(), err)
return
}
body := &icmp.Echo{
ID: id,
Seq: sequence,
Data: mb,
}
msg := &icmp.Message{
Type: (ipv4.ICMPType)(sproto),
Code: 0,
Body: m,
Body: body,
}
bytes, err := msg.Marshal(nil)
if err != nil {
fmt.Printf("sendICMP Marshal error %s %s\n", server.String(), err)
loggo.Error("sendICMP Marshal error %s %s", server.String(), err)
return
}
for {
if _, err := conn.WriteTo(bytes, server); err != nil {
if neterr, ok := err.(*net.OpError); ok {
if neterr.Err == syscall.ENOBUFS {
continue
}
}
fmt.Printf("sendICMP WriteTo error %s %s\n", server.String(), err)
}
break
}
return
conn.WriteTo(bytes, server)
}
func recvICMP(conn icmp.PacketConn, recv chan<- *Packet) {
func recvICMP(workResultLock *sync.WaitGroup, exit *bool, conn icmp.PacketConn, recv chan<- *Packet) {
defer common.CrashLog()
(*workResultLock).Add(1)
defer (*workResultLock).Done()
bytes := make([]byte, 10240)
for {
for !*exit {
conn.SetReadDeadline(time.Now().Add(time.Millisecond * 100))
n, srcaddr, err := conn.ReadFrom(bytes)
if err != nil {
if neterr, ok := err.(*net.OpError); ok {
if neterr.Timeout() {
// Read timeout
continue
} else {
fmt.Printf("Error read icmp message %s\n", err)
continue
}
nerr, ok := err.(net.Error)
if !ok || !nerr.Timeout() {
loggo.Info("Error read icmp message %s", err)
continue
}
}
my := &MyMsg{
}
my.Unmarshal(bytes[4:n])
if (my.TYPE != (uint32)(DATA) && my.TYPE != (uint32)(PING)) || my.ENDTYPE != (uint32)(END) {
//fmt.Printf("processPacket diff type %s %d %d \n", my.ID, my.TYPE, my.ENDTYPE)
if n <= 0 {
continue
}
if my.Data == nil {
fmt.Printf("processPacket data nil %s\n", my.ID)
return
echoId := int(binary.BigEndian.Uint16(bytes[4:6]))
echoSeq := int(binary.BigEndian.Uint16(bytes[6:8]))
my := &MyMsg{}
err = proto.Unmarshal(bytes[8:n], my)
if err != nil {
loggo.Debug("Unmarshal MyMsg error: %s", err)
continue
}
recv <- &Packet{msgType: my.TYPE, data: my.Data, id: my.ID, target: my.TARGET, src: srcaddr.(*net.IPAddr), rproto: (int)(my.RPROTO)}
if my.Magic != (int32)(MyMsg_MAGIC) {
loggo.Debug("processPacket data invalid %s", my.Id)
continue
}
recv <- &Packet{my: my,
src: srcaddr.(*net.IPAddr),
echoId: echoId, echoSeq: echoSeq}
}
}
type Packet struct {
msgType uint32
data []byte
id string
target string
my *MyMsg
src *net.IPAddr
rproto int
echoId int
echoSeq int
}
func UniqueId() string {
b := make([]byte, 48)
if _, err := io.ReadFull(rand.Reader, b); err != nil {
return ""
}
return GetMd5String(base64.URLEncoding.EncodeToString(b))
}
func GetMd5String(s string) string {
h := md5.New()
h.Write([]byte(s))
return hex.EncodeToString(h.Sum(nil))
}
const (
FRAME_MAX_SIZE int = 888
FRAME_MAX_ID int = 1000000
)

View file

@ -2,27 +2,26 @@ package pingtunnel
import (
"fmt"
"github.com/esrrhs/pingtunnel"
"github.com/golang/protobuf/proto"
"testing"
)
func Test0001(test *testing.T) {
func Test0001(t *testing.T) {
my := &pingtunnel.MyMsg{
}
my.ID = "12345"
my.TARGET = "111:11"
my.TYPE = 12
my.Data = make([]byte, 3)
dst,_ := my.Marshal(0)
my := &MyMsg{}
my.Id = "12345"
my.Target = "111:11"
my.Type = 12
my.Data = make([]byte, 0)
dst, _ := proto.Marshal(my)
fmt.Println("dst = ", dst)
my1 := &MyMsg{}
proto.Unmarshal(dst, my1)
fmt.Println("my1 = ", my1)
fmt.Println("my1.Data = ", my1.Data)
my1 := &pingtunnel.MyMsg{
}
my1.Unmarshal(dst)
proto.Unmarshal(dst[0:4], my1)
fmt.Println("my1 = ", my1)
my1.Unmarshal(dst[0:4])
fmt.Println("my1 = ", my1)
}

625
server.go
View file

@ -1,177 +1,614 @@
package pingtunnel
import (
"fmt"
"github.com/esrrhs/gohome/common"
"github.com/esrrhs/gohome/frame"
"github.com/esrrhs/gohome/loggo"
"github.com/esrrhs/gohome/threadpool"
"github.com/golang/protobuf/proto"
"golang.org/x/net/icmp"
"net"
"sync"
"time"
)
func NewServer(timeout int) (*Server, error) {
return &Server{
timeout: timeout,
}, nil
func NewServer(key int, maxconn int, maxprocessthread int, maxprocessbuffer int, connecttmeout int) (*Server, error) {
s := &Server{
exit: false,
key: key,
maxconn: maxconn,
maxprocessthread: maxprocessthread,
maxprocessbuffer: maxprocessbuffer,
connecttmeout: connecttmeout,
}
if maxprocessthread > 0 {
s.processtp = threadpool.NewThreadPool(maxprocessthread, maxprocessbuffer, func(v interface{}) {
packet := v.(*Packet)
s.processDataPacket(packet)
})
}
return s, nil
}
type Server struct {
timeout int
exit bool
key int
workResultLock sync.WaitGroup
maxconn int
maxprocessthread int
maxprocessbuffer int
connecttmeout int
conn *icmp.PacketConn
localConnMap map[string]*ServerConn
localConnMap sync.Map
connErrorMap sync.Map
sendPacket uint64
recvPacket uint64
sendPacketSize uint64
recvPacketSize uint64
sendPacket uint64
recvPacket uint64
sendPacketSize uint64
recvPacketSize uint64
localConnMapSize int
processtp *threadpool.ThreadPool
recvcontrol chan int
}
type ServerConn struct {
ipaddrTarget *net.UDPAddr
conn *net.UDPConn
id string
activeTime time.Time
close bool
rproto int
exit bool
timeout int
ipaddrTarget *net.UDPAddr
conn *net.UDPConn
tcpaddrTarget *net.TCPAddr
tcpconn *net.TCPConn
id string
activeRecvTime time.Time
activeSendTime time.Time
close bool
rproto int
fm *frame.FrameMgr
tcpmode int
echoId int
echoSeq int
}
func (p *Server) Run() {
func (p *Server) Run() error {
conn, err := icmp.ListenPacket("ip4:icmp", "")
if err != nil {
fmt.Printf("Error listening for ICMP packets: %s\n", err.Error())
return
loggo.Error("Error listening for ICMP packets: %s", err.Error())
return err
}
p.conn = conn
p.localConnMap = make(map[string]*ServerConn)
recv := make(chan *Packet, 10000)
go recvICMP(*p.conn, recv)
p.recvcontrol = make(chan int, 1)
go recvICMP(&p.workResultLock, &p.exit, *p.conn, recv)
interval := time.NewTicker(time.Second)
defer interval.Stop()
go func() {
defer common.CrashLog()
for {
select {
case <-interval.C:
p.workResultLock.Add(1)
defer p.workResultLock.Done()
for !p.exit {
p.checkTimeoutConn()
p.showNet()
case r := <-recv:
p.processPacket(r)
p.updateConnError()
time.Sleep(time.Second)
}
}
}()
go func() {
defer common.CrashLog()
p.workResultLock.Add(1)
defer p.workResultLock.Done()
for !p.exit {
select {
case <-p.recvcontrol:
return
case r := <-recv:
p.processPacket(r)
}
}
}()
return nil
}
func (p *Server) Stop() {
p.exit = true
p.recvcontrol <- 1
p.workResultLock.Wait()
p.processtp.Stop()
p.conn.Close()
}
func (p *Server) processPacket(packet *Packet) {
if packet.msgType == PING {
if packet.my.Key != (int32)(p.key) {
return
}
if packet.my.Type == (int32)(MyMsg_PING) {
t := time.Time{}
t.UnmarshalBinary(packet.data)
fmt.Printf("ping from %s %s %d\n", packet.src.String(), t.String(), packet.rproto)
sendICMP(*p.conn, packet.src, "", "", (uint32)(PING), packet.data, packet.rproto, 0)
t.UnmarshalBinary(packet.my.Data)
loggo.Info("ping from %s %s %d %d %d", packet.src.String(), t.String(), packet.my.Rproto, packet.echoId, packet.echoSeq)
sendICMP(packet.echoId, packet.echoSeq, *p.conn, packet.src, "", "", (uint32)(MyMsg_PING), packet.my.Data,
(int)(packet.my.Rproto), -1, p.key,
0, 0, 0, 0, 0, 0,
0)
return
}
//fmt.Printf("processPacket %s %s %d\n", packet.id, packet.src.String(), len(packet.data))
now := time.Now()
id := packet.id
udpConn := p.localConnMap[id]
if udpConn == nil {
addr := packet.target
ipaddrTarget, err := net.ResolveUDPAddr("udp", addr)
if err != nil {
fmt.Printf("Error ResolveUDPAddr for udp addr: %s %s\n", addr, err.Error())
return
if packet.my.Type == (int32)(MyMsg_KICK) {
localConn := p.getServerConnById(packet.my.Id)
if localConn != nil {
p.close(localConn)
loggo.Info("remote kick local %s", packet.my.Id)
}
targetConn, err := net.DialUDP("udp", nil, ipaddrTarget)
if err != nil {
fmt.Printf("Error listening for udp packets: %s\n", err.Error())
return
}
udpConn = &ServerConn{conn: targetConn, ipaddrTarget: ipaddrTarget, id: id, activeTime: now, close: false, rproto: packet.rproto}
p.localConnMap[id] = udpConn
go p.Recv(udpConn, id, packet.src)
}
udpConn.activeTime = now
_, err := udpConn.conn.Write(packet.data)
if err != nil {
fmt.Printf("WriteToUDP Error %s\n", err)
udpConn.close = true
return
}
p.recvPacket++
p.recvPacketSize += (uint64)(len(packet.data))
if p.maxprocessthread > 0 {
p.processtp.AddJob((int)(common.HashString(packet.my.Id)), packet)
} else {
p.processDataPacket(packet)
}
}
func (p *Server) processDataPacketNewConn(id string, packet *Packet) *ServerConn {
now := common.GetNowUpdateInSecond()
loggo.Info("start add new connect %s %s", id, packet.my.Target)
if p.maxconn > 0 && p.localConnMapSize >= p.maxconn {
loggo.Info("too many connections %d, server connected target fail %s", p.localConnMapSize, packet.my.Target)
p.remoteError(packet.echoId, packet.echoSeq, id, (int)(packet.my.Rproto), packet.src)
return nil
}
addr := packet.my.Target
if p.isConnError(addr) {
loggo.Info("addr connect Error before: %s %s", id, addr)
p.remoteError(packet.echoId, packet.echoSeq, id, (int)(packet.my.Rproto), packet.src)
return nil
}
if packet.my.Tcpmode > 0 {
c, err := net.DialTimeout("tcp", addr, time.Millisecond*time.Duration(p.connecttmeout))
if err != nil {
loggo.Error("Error listening for tcp packets: %s %s", id, err.Error())
p.remoteError(packet.echoId, packet.echoSeq, id, (int)(packet.my.Rproto), packet.src)
p.addConnError(addr)
return nil
}
targetConn := c.(*net.TCPConn)
ipaddrTarget := targetConn.RemoteAddr().(*net.TCPAddr)
fm := frame.NewFrameMgr(FRAME_MAX_SIZE, FRAME_MAX_ID, (int)(packet.my.TcpmodeBuffersize), (int)(packet.my.TcpmodeMaxwin), (int)(packet.my.TcpmodeResendTimems), (int)(packet.my.TcpmodeCompress),
(int)(packet.my.TcpmodeStat))
localConn := &ServerConn{exit: false, timeout: (int)(packet.my.Timeout), tcpconn: targetConn, tcpaddrTarget: ipaddrTarget, id: id, activeRecvTime: now, activeSendTime: now, close: false,
rproto: (int)(packet.my.Rproto), fm: fm, tcpmode: (int)(packet.my.Tcpmode)}
p.addServerConn(id, localConn)
go p.RecvTCP(localConn, id, packet.src)
return localConn
} else {
c, err := net.DialTimeout("udp", addr, time.Millisecond*time.Duration(p.connecttmeout))
if err != nil {
loggo.Error("Error listening for udp packets: %s %s", id, err.Error())
p.remoteError(packet.echoId, packet.echoSeq, id, (int)(packet.my.Rproto), packet.src)
p.addConnError(addr)
return nil
}
targetConn := c.(*net.UDPConn)
ipaddrTarget := targetConn.RemoteAddr().(*net.UDPAddr)
localConn := &ServerConn{exit: false, timeout: (int)(packet.my.Timeout), conn: targetConn, ipaddrTarget: ipaddrTarget, id: id, activeRecvTime: now, activeSendTime: now, close: false,
rproto: (int)(packet.my.Rproto), tcpmode: (int)(packet.my.Tcpmode)}
p.addServerConn(id, localConn)
go p.Recv(localConn, id, packet.src)
return localConn
}
return nil
}
func (p *Server) processDataPacket(packet *Packet) {
loggo.Debug("processPacket %s %s %d", packet.my.Id, packet.src.String(), len(packet.my.Data))
now := common.GetNowUpdateInSecond()
id := packet.my.Id
localConn := p.getServerConnById(id)
if localConn == nil {
localConn = p.processDataPacketNewConn(id, packet)
if localConn == nil {
return
}
}
localConn.activeRecvTime = now
localConn.echoId = packet.echoId
localConn.echoSeq = packet.echoSeq
if packet.my.Type == (int32)(MyMsg_DATA) {
if packet.my.Tcpmode > 0 {
f := &frame.Frame{}
err := proto.Unmarshal(packet.my.Data, f)
if err != nil {
loggo.Error("Unmarshal tcp Error %s", err)
return
}
localConn.fm.OnRecvFrame(f)
} else {
if packet.my.Data == nil {
return
}
_, err := localConn.conn.Write(packet.my.Data)
if err != nil {
loggo.Info("WriteToUDP Error %s", err)
localConn.close = true
return
}
}
p.recvPacket++
p.recvPacketSize += (uint64)(len(packet.my.Data))
}
}
func (p *Server) RecvTCP(conn *ServerConn, id string, src *net.IPAddr) {
defer common.CrashLog()
p.workResultLock.Add(1)
defer p.workResultLock.Done()
loggo.Info("server waiting target response %s -> %s %s", conn.tcpaddrTarget.String(), conn.id, conn.tcpconn.LocalAddr().String())
loggo.Info("start wait remote connect tcp %s %s", conn.id, conn.tcpaddrTarget.String())
startConnectTime := common.GetNowUpdateInSecond()
for !p.exit && !conn.exit {
if conn.fm.IsConnected() {
break
}
conn.fm.Update()
sendlist := conn.fm.GetSendList()
for e := sendlist.Front(); e != nil; e = e.Next() {
f := e.Value.(*frame.Frame)
mb, _ := conn.fm.MarshalFrame(f)
sendICMP(conn.echoId, conn.echoSeq, *p.conn, src, "", id, (uint32)(MyMsg_DATA), mb,
conn.rproto, -1, p.key,
0, 0, 0, 0, 0, 0,
0)
p.sendPacket++
p.sendPacketSize += (uint64)(len(mb))
}
time.Sleep(time.Millisecond * 10)
now := common.GetNowUpdateInSecond()
diffclose := now.Sub(startConnectTime)
if diffclose > time.Second*5 {
loggo.Info("can not connect remote tcp %s %s", conn.id, conn.tcpaddrTarget.String())
p.close(conn)
p.remoteError(conn.echoId, conn.echoSeq, id, conn.rproto, src)
return
}
}
if !conn.exit {
loggo.Info("remote connected tcp %s %s", conn.id, conn.tcpaddrTarget.String())
}
bytes := make([]byte, 10240)
tcpActiveRecvTime := common.GetNowUpdateInSecond()
tcpActiveSendTime := common.GetNowUpdateInSecond()
for !p.exit && !conn.exit {
now := common.GetNowUpdateInSecond()
sleep := true
left := common.MinOfInt(conn.fm.GetSendBufferLeft(), len(bytes))
if left > 0 {
conn.tcpconn.SetReadDeadline(time.Now().Add(time.Millisecond * 1))
n, err := conn.tcpconn.Read(bytes[0:left])
if err != nil {
nerr, ok := err.(net.Error)
if !ok || !nerr.Timeout() {
loggo.Info("Error read tcp %s %s %s", conn.id, conn.tcpaddrTarget.String(), err)
conn.fm.Close()
break
}
}
if n > 0 {
sleep = false
conn.fm.WriteSendBuffer(bytes[:n])
tcpActiveRecvTime = now
}
}
conn.fm.Update()
sendlist := conn.fm.GetSendList()
if sendlist.Len() > 0 {
sleep = false
conn.activeSendTime = now
for e := sendlist.Front(); e != nil; e = e.Next() {
f := e.Value.(*frame.Frame)
mb, err := conn.fm.MarshalFrame(f)
if err != nil {
loggo.Error("Error tcp Marshal %s %s %s", conn.id, conn.tcpaddrTarget.String(), err)
continue
}
sendICMP(conn.echoId, conn.echoSeq, *p.conn, src, "", id, (uint32)(MyMsg_DATA), mb,
conn.rproto, -1, p.key,
0, 0, 0, 0, 0, 0,
0)
p.sendPacket++
p.sendPacketSize += (uint64)(len(mb))
}
}
if conn.fm.GetRecvBufferSize() > 0 {
sleep = false
rr := conn.fm.GetRecvReadLineBuffer()
conn.tcpconn.SetWriteDeadline(time.Now().Add(time.Millisecond * 1))
n, err := conn.tcpconn.Write(rr)
if err != nil {
nerr, ok := err.(net.Error)
if !ok || !nerr.Timeout() {
loggo.Info("Error write tcp %s %s %s", conn.id, conn.tcpaddrTarget.String(), err)
conn.fm.Close()
break
}
}
if n > 0 {
conn.fm.SkipRecvBuffer(n)
tcpActiveSendTime = now
}
}
if sleep {
time.Sleep(time.Millisecond * 10)
}
diffrecv := now.Sub(conn.activeRecvTime)
diffsend := now.Sub(conn.activeSendTime)
tcpdiffrecv := now.Sub(tcpActiveRecvTime)
tcpdiffsend := now.Sub(tcpActiveSendTime)
if diffrecv > time.Second*(time.Duration(conn.timeout)) || diffsend > time.Second*(time.Duration(conn.timeout)) ||
(tcpdiffrecv > time.Second*(time.Duration(conn.timeout)) && tcpdiffsend > time.Second*(time.Duration(conn.timeout))) {
loggo.Info("close inactive conn %s %s", conn.id, conn.tcpaddrTarget.String())
conn.fm.Close()
break
}
if conn.fm.IsRemoteClosed() {
loggo.Info("closed by remote conn %s %s", conn.id, conn.tcpaddrTarget.String())
conn.fm.Close()
break
}
}
conn.fm.Close()
startCloseTime := common.GetNowUpdateInSecond()
for !p.exit && !conn.exit {
now := common.GetNowUpdateInSecond()
conn.fm.Update()
sendlist := conn.fm.GetSendList()
for e := sendlist.Front(); e != nil; e = e.Next() {
f := e.Value.(*frame.Frame)
mb, _ := conn.fm.MarshalFrame(f)
sendICMP(conn.echoId, conn.echoSeq, *p.conn, src, "", id, (uint32)(MyMsg_DATA), mb,
conn.rproto, -1, p.key,
0, 0, 0, 0, 0, 0,
0)
p.sendPacket++
p.sendPacketSize += (uint64)(len(mb))
}
nodatarecv := true
if conn.fm.GetRecvBufferSize() > 0 {
rr := conn.fm.GetRecvReadLineBuffer()
conn.tcpconn.SetWriteDeadline(time.Now().Add(time.Millisecond * 100))
n, _ := conn.tcpconn.Write(rr)
if n > 0 {
conn.fm.SkipRecvBuffer(n)
nodatarecv = false
}
}
diffclose := now.Sub(startCloseTime)
if diffclose > time.Second*60 {
loggo.Info("close conn had timeout %s %s", conn.id, conn.tcpaddrTarget.String())
break
}
remoteclosed := conn.fm.IsRemoteClosed()
if remoteclosed && nodatarecv {
loggo.Info("remote conn had closed %s %s", conn.id, conn.tcpaddrTarget.String())
break
}
time.Sleep(time.Millisecond * 100)
}
time.Sleep(time.Second)
loggo.Info("close tcp conn %s %s", conn.id, conn.tcpaddrTarget.String())
p.close(conn)
}
func (p *Server) Recv(conn *ServerConn, id string, src *net.IPAddr) {
fmt.Printf("server waiting target response %s -> %s %s\n", conn.ipaddrTarget.String(), conn.id, conn.conn.LocalAddr().String())
defer common.CrashLog()
bytes := make([]byte, 10240)
p.workResultLock.Add(1)
defer p.workResultLock.Done()
loggo.Info("server waiting target response %s -> %s %s", conn.ipaddrTarget.String(), conn.id, conn.conn.LocalAddr().String())
bytes := make([]byte, 2000)
for !p.exit {
for {
conn.conn.SetReadDeadline(time.Now().Add(time.Millisecond * 100))
n, _, err := conn.conn.ReadFromUDP(bytes)
if err != nil {
if neterr, ok := err.(*net.OpError); ok {
if neterr.Timeout() {
// Read timeout
continue
} else {
fmt.Printf("ReadFromUDP Error read udp %s\n", err)
conn.close = true
return
}
nerr, ok := err.(net.Error)
if !ok || !nerr.Timeout() {
loggo.Info("ReadFromUDP Error read udp %s", err)
conn.close = true
return
}
}
now := time.Now()
conn.activeTime = now
now := common.GetNowUpdateInSecond()
conn.activeSendTime = now
sendICMP(*p.conn, src, "", id, (uint32)(DATA), bytes[:n], conn.rproto, 0)
sendICMP(conn.echoId, conn.echoSeq, *p.conn, src, "", id, (uint32)(MyMsg_DATA), bytes[:n],
conn.rproto, -1, p.key,
0, 0, 0, 0, 0, 0,
0)
p.sendPacket++
p.sendPacketSize += (uint64)(n)
}
}
func (p *Server) Close(conn *ServerConn) {
if p.localConnMap[conn.id] != nil {
conn.conn.Close()
delete(p.localConnMap, conn.id)
func (p *Server) close(conn *ServerConn) {
if p.getServerConnById(conn.id) != nil {
conn.exit = true
if conn.conn != nil {
conn.conn.Close()
}
if conn.tcpconn != nil {
conn.tcpconn.Close()
}
p.deleteServerConn(conn.id)
}
}
func (p *Server) checkTimeoutConn() {
now := time.Now()
for _, conn := range p.localConnMap {
diff := now.Sub(conn.activeTime)
if diff > time.Second*(time.Duration(p.timeout)) {
tmp := make(map[string]*ServerConn)
p.localConnMap.Range(func(key, value interface{}) bool {
id := key.(string)
serverConn := value.(*ServerConn)
tmp[id] = serverConn
return true
})
now := common.GetNowUpdateInSecond()
for _, conn := range tmp {
if conn.tcpmode > 0 {
continue
}
diffrecv := now.Sub(conn.activeRecvTime)
diffsend := now.Sub(conn.activeSendTime)
if diffrecv > time.Second*(time.Duration(conn.timeout)) || diffsend > time.Second*(time.Duration(conn.timeout)) {
conn.close = true
}
}
for id, conn := range p.localConnMap {
for id, conn := range tmp {
if conn.tcpmode > 0 {
continue
}
if conn.close {
fmt.Printf("close inactive conn %s %s\n", id, conn.ipaddrTarget.String())
p.Close(conn)
loggo.Info("close inactive conn %s %s", id, conn.ipaddrTarget.String())
p.close(conn)
}
}
}
func (p *Server) showNet() {
fmt.Printf("send %dPacket/s %dKB/s recv %dPacket/s %dKB/s\n", p.sendPacket, p.sendPacketSize/1024, p.recvPacket, p.recvPacketSize/1024)
p.localConnMapSize = 0
p.localConnMap.Range(func(key, value interface{}) bool {
p.localConnMapSize++
return true
})
loggo.Info("send %dPacket/s %dKB/s recv %dPacket/s %dKB/s %dConnections",
p.sendPacket, p.sendPacketSize/1024, p.recvPacket, p.recvPacketSize/1024, p.localConnMapSize)
p.sendPacket = 0
p.recvPacket = 0
p.sendPacketSize = 0
p.recvPacketSize = 0
}
func (p *Server) addServerConn(uuid string, serverConn *ServerConn) {
p.localConnMap.Store(uuid, serverConn)
}
func (p *Server) getServerConnById(uuid string) *ServerConn {
ret, ok := p.localConnMap.Load(uuid)
if !ok {
return nil
}
return ret.(*ServerConn)
}
func (p *Server) deleteServerConn(uuid string) {
p.localConnMap.Delete(uuid)
}
func (p *Server) remoteError(echoId int, echoSeq int, uuid string, rprpto int, src *net.IPAddr) {
sendICMP(echoId, echoSeq, *p.conn, src, "", uuid, (uint32)(MyMsg_KICK), []byte{},
rprpto, -1, p.key,
0, 0, 0, 0, 0, 0,
0)
}
func (p *Server) addConnError(addr string) {
_, ok := p.connErrorMap.Load(addr)
if !ok {
now := common.GetNowUpdateInSecond()
p.connErrorMap.Store(addr, now)
}
}
func (p *Server) isConnError(addr string) bool {
_, ok := p.connErrorMap.Load(addr)
return ok
}
func (p *Server) updateConnError() {
tmp := make(map[string]time.Time)
p.connErrorMap.Range(func(key, value interface{}) bool {
id := key.(string)
t := value.(time.Time)
tmp[id] = t
return true
})
now := common.GetNowUpdateInSecond()
for id, t := range tmp {
diff := now.Sub(t)
if diff > time.Second*5 {
p.connErrorMap.Delete(id)
}
}
}