一、功能描述:

客户端通过访问外网服务器上指定端口,间接访问自已本地的内网服务。

二、原理图如下:

三、实现代码如下:

server.go代码:

package main;

import (
"net"
"fmt"
"flag"
"os"
) type MidServer struct {
//客户端监听
clientLis *net.TCPListener;
//后端服务连接
transferLis *net.TCPListener;
//所有通道
channels map[int]*Channel;
//当前通道ID
curChannelId int;
} type Channel struct {
//通道ID
id int;
//客户端连接
client net.Conn;
//后端服务连接
transfer net.Conn;
//客户端接收消息
clientRecvMsg chan []byte;
//后端服务发送消息
transferSendMsg chan []byte;
} //创建一个服务器
func New() *MidServer {
return &MidServer{
channels: make(map[int]*Channel),
curChannelId: 0,
};
} //启动服务
func (m *MidServer) Start(clientPort int, transferPort int) error {
addr, err := net.ResolveTCPAddr("tcp", fmt.Sprintf(":%d", clientPort));
if err != nil {
return err;
}
m.clientLis, err = net.ListenTCP("tcp", addr);
if err != nil {
return err;
}
addr, err = net.ResolveTCPAddr("tcp", fmt.Sprintf(":%d", transferPort));
if err != nil {
return err;
}
m.transferLis, err = net.ListenTCP("tcp", addr);
if err != nil {
return err;
}
go m.AcceptLoop();
return nil;
} //关闭服务
func (m *MidServer) Stop() {
m.clientLis.Close();
m.transferLis.Close();
//循环关闭通道连接
for _, v := range m.channels {
v.client.Close();
v.transfer.Close();
}
} //删除通道
func (m *MidServer) DelChannel(id int) {
chs := m.channels;
delete(chs, id);
m.channels = chs;
} //处理连接
func (m *MidServer) AcceptLoop() {
transfer, err := m.transferLis.Accept();
if err != nil {
return;
}
for {
//获取连接
client, err := m.clientLis.Accept();
if err != nil {
continue;
} //创建一个通道
ch := &Channel{
id: m.curChannelId,
client: client,
transfer: transfer,
clientRecvMsg: make(chan []byte),
transferSendMsg: make(chan []byte),
};
m.curChannelId++; //把通道加入channels中
chs := m.channels;
chs[ch.id] = ch;
m.channels = chs; //启一个goroutine处理客户端消息
go m.ClientMsgLoop(ch);
//启一个goroutine处理后端服务消息
go m.TransferMsgLoop(ch);
go m.MsgLoop(ch);
}
} //处理客户端消息
func (m *MidServer) ClientMsgLoop(ch *Channel) {
defer func() {
fmt.Println("ClientMsgLoop exit");
}();
for {
select {
case data, isClose := <-ch.transferSendMsg:
{
//判断channel是否关闭,如果是则返回
if !isClose {
return;
}
_, err := ch.client.Write(data);
if err != nil {
return;
}
}
}
}
} //处理后端服务消息
func (m *MidServer) TransferMsgLoop(ch *Channel) {
defer func() {
fmt.Println("TransferMsgLoop exit");
}();
for {
select {
case data, isClose := <-ch.clientRecvMsg:
{
//判断channel是否关闭,如果是则返回
if !isClose {
return;
}
_, err := ch.transfer.Write(data);
if err != nil {
return;
}
}
}
}
} //客户端与后端服务消息处理
func (m *MidServer) MsgLoop(ch *Channel) {
defer func() {
//关闭channel,好让ClientMsgLoop与TransferMsgLoop退出
close(ch.clientRecvMsg);
close(ch.transferSendMsg);
m.DelChannel(ch.id);
fmt.Println("MsgLoop exit");
}();
buf := make([]byte, 1024);
for {
n, err := ch.client.Read(buf);
if err != nil {
return;
}
ch.clientRecvMsg <- buf[:n];
n, err = ch.transfer.Read(buf);
if err != nil {
return;
}
ch.transferSendMsg <- buf[:n];
}
} func main() {
//参数解析
localPort := flag.Int("localPort", 8080, "客户端访问端口");
remotePort := flag.Int("remotePort", 8888, "服务访问端口");
flag.Parse();
if flag.NFlag() != 2 {
flag.PrintDefaults();
os.Exit(1);
} ms := New();
//启动服务
ms.Start(*localPort, *remotePort);
//循环
select {};
}

client.go代码:

package main;

import (
"net"
"fmt"
"flag"
"os"
) func handler(r net.Conn, localPort int) {
buf := make([]byte, 1024);
for {
//先从远程读数据
n, err := r.Read(buf);
if err != nil {
continue;
}
data := buf[:n];
//建立与本地80服务的连接
local, err := net.Dial("tcp", fmt.Sprintf(":%d", localPort));
if err != nil {
continue;
}
//向80服务写数据
n, err = local.Write(data);
if err != nil {
continue;
}
//读取80服务返回的数据
n, err = local.Read(buf);
//关闭80服务,因为本地80服务是http服务,不是持久连接
//一个请求结束,就会自动断开。所以在for循环里我们要不断Dial,然后关闭。
local.Close();
if err != nil {
continue;
}
data = buf[:n];
//向远程写数据
n, err = r.Write(data);
if err != nil {
continue;
}
}
} func main() {
//参数解析
host := flag.String("host", "127.0.0.1", "服务器地址");
remotePort := flag.Int("remotePort", 8888, "服务器端口");
localPort := flag.Int("localPort", 80, "本地端口");
flag.Parse();
if flag.NFlag() != 3 {
flag.PrintDefaults();
os.Exit(1);
}
//建立与服务器的连接
remote, err := net.Dial("tcp", fmt.Sprintf("%s:%d", *host, *remotePort));
if err != nil {
fmt.Println(err);
}
go handler(remote, *localPort); select {};
}

四、测试

1、先把server.go上传到外网服务器上,安装GO环境,并编译,然后运行server

> ./server -localPort 8080 -remotePort 8888

2、在本地编译client.go,运行client

> client.exe -host 外网服务器IP -localPort 80 -remotePort 8888

3、浏览器访问外网服务器8080端口

当我浏览器访问时,外网服务器的server会打印两次MsgLoop exit,这是因为谷歌浏览器会多一个favicon.ico请求,不知道其他浏览器会不会。

注意,上面的server.go和client.go代码不排除会有BUG,代码仅供参考,切勿用于生产环境。

golang 简单的实现内 网 穿 透,用户访问本地服务。的更多相关文章

  1. Pandorabox固件路由器上申请Let's Encrypt证书,为内网里的多个web服务提供SSL支持

    对于家中宽带有公网IP的用户,有时我们需要将路由器内部网络的某些web服务通过端口转发暴露到外网(例如NAS远程访问),但HTTP是明文传输,有被监听的风险:如果在NAS上使用自签名证书,再端口转发, ...

  2. [转]Zen Cart官网屏蔽中国用户访问的真正原因

    近需要到 zen cart 的官方网站查询一些资料,却发现无法访问!在网上搜索一番以后,原来如此. Zen Cart官网屏蔽中国用户访问的真正原因 作者:[鹏程万里] 日期:2011-03-26 准备 ...

  3. 内网穿透神器(ngrok)服务端部署【分享一台自己的ngrok服务器】【多平台】

    Ngrok为何物 “ngrok 是一个反向代理,通过在公共的端点和本地运行的 Web 服务器之间建立一个安全的通道.ngrok 可捕获和分析所有通道上的流量,便于后期分析和重放.”这是百度百科上给Ng ...

  4. 内网穿透神器ngrok——将本地项目驾到外网

    相信做Web开发的同学们,经常会遇到需要将本地部署的Web应用能够让公网环境直接访问到的情况,例如微信应用调试.支付宝接口调试等.这个时候,一个叫ngrok的神器可能会帮到你,它提供了一个能够在公网安 ...

  5. zookeeper两台内网服务器彼此调不到服务的问题。

    Start NettyClient /172.20.11.52 connect to the server /172.20.11.52:20881, dubbo version: 2.5.3, cur ...

  6. 内网中让其他人访问我电脑上的asp.net应用程序

    打开防火墙,高级配置,新建入站规则,选择“端口”,下一步,填写特定本地端口,然后都是点击下一步,命名规则,到此,内网中的其他人可以访问你的程序了

  7. 本地Linux虚拟机内网穿透,服务器文件下载到本地磁盘

    本地Linux虚拟内网穿透 把服务器文件下载到本地磁盘 https://natapp.cn/ 1.注册账户点击免费隧道  

  8. 使用内网映射工具Holer将本地的Web应用映射到公网上访问

    Holer exposes local servers behind NATs and firewalls to the public internet over secure tunnels. Su ...

  9. linux 能访问内网,但不能访问外网?解决方案

    用iptables就可以了 iptables -F iptables -t nat -F iptables -A INPUT -s -d -j ACCEPT iptables -A INPUT -d ...

随机推荐

  1. 用tomcat插件 在Eclipse 中配置Tomcat项目

    1.安装Tomcat在Eclipse中的插件(使可以在eclipse中启动Tomcat) 2.培植eclipse中tomcat属性, window-->preferences 对话框中Tomca ...

  2. 【369】列表/字典的分拆, unpacking

    参考: python--参数列表的分拆 参考: List Comprehensions 当你要传递的参数已经是一个列表,调用的函数却接受分开一个个的参数,这个时候可以考虑参数列表拆分: 可以使用* 操 ...

  3. 基于Java SE集合的图书管理系统

    图书管理系统一.需求说明1.功能:登录,注册,忘记密码,管理员管理,图书管理.2.管理员管理:管理员的增删改查.3.图书管理:图书的增删改查.4.管理员属性包括:id,姓名,性别,年龄,家庭住址,手机 ...

  4. 初识Swift中的值和引用,循坏引用、代理的注意点

    1.0 在Swift中分有值类型和引用类型 Int .String . 结构体和枚举都属于值类型, 将值类型传递给方法是,将在内存中创建其副本,并传递这个副本:这样我们就可以随心所欲修改它,而不用担心 ...

  5. EF 安装框架

    在NuGet中安装ef框架 命令:Install-package EntityFramework

  6. Applese的回文串-dfs

    链接:https://ac.nowcoder.com/acm/contest/330/I来源:牛客网 题目描述 自从 Applese 学会了字符串之后,精通各种字符串算法,比如……判断一个字符串是不是 ...

  7. 在webpack构建的项目中使用vue

    一.复习在普通网页中使用vue1.使用script引入vue2.在index中创建 id为app的容器3.通过new vue得到vm实例二.在webpack中尝试使用vue://注意 : 在webpa ...

  8. 移动端调起qq聊天

    <div class="item item-right" style='width:3rem;padding-left:0rem;'>QQ:<a target=& ...

  9. 安装 protoc 的各种坑

    首先下载 protoc 2.6.1   https://github.com/google/protobuf/releases/download/v2.6.1/protobuf-2.6.1.tar.g ...

  10. js正则表达式子校验

    //正则表达式校验new RegExp(/^[1-9]\d{4,8}$/,"g").test(1234);//执行一个字符串所表达的方法 eval(this['字符串']) 正则表 ...