forwardPort.go
packagemain
import(
"encoding/json"
"flag"
"fmt"
"io"
"log"
"net"
"net/http"
"os"
"os/signal"
"runtime"
"sync"
"syscall"
"time"
)
//监听服务地址 默认使用8000端口 服务端监听端口
varrestApiServer=flag.String("restApi","","listenaddrforrestapi")
//监听服务认证 密码
varauth=flag.String("auth","taven123","restApiPassword")
//本地监听器
vargLocalConnnet.Listener
//客户端与服务链接关系--映射
varclientSMapmap[string]net.Conn
//目标服务器和本地服务器 两个服务地址之间使用“-”链接
varforwardInfostring
funcmain(){
clientSMap=make(map[string]net.Conn)
//解析传入的参数
flag.Parse()
//默认使用8000端口 服务端监听端口
if*restApiServer==""{
*restApiServer="0.0.0.0:8000"
}
//开启监听服务
goStartHttpServer(*restApiServer)
log.Println("restApiServer:",*restApiServer)
fmt.Println("------------启动成功------------")
//开启线程同步锁
varwsync.WaitGroup
w.Add()
//开一个并发线程,接收退出信号
//下面代码 书写混乱 逻辑不清晰 但是实现了
gofunc(){
c:=make(chanos.Signal,)
signal.Notify(c,os.Interrupt,syscall.SIGTERM)
n:=
f:=func(){
<-c
n++
ifn>{
log.Println("forceshutdown")
os.Exit(-)
}
log.Println("receivedsignal,shutdown")
//释放相关资源
closeAllConn()
}
f()
gofunc(){
for{
f()
}
}()
//执行完成一次,Done()等同于Add(-1),计数不为0,则阻塞
w.Done()
}()
loop:=func(){
w.Done()
}
loop()
w.Wait()
fmt.Println("------------程序执行完成------------")
}
funcStartHttpServer(addrstring){
http.HandleFunc("/ServerSummary",ServerSummary)
http.HandleFunc("/ForwardWork",ForwardWork)
//
err:=http.ListenAndServe(addr,http.DefaultServeMux)
iferr!=nil{
fmt.Println("ListenAndServeerror:",err.Error())
}
}
funcServerSummary(rwhttp.ResponseWriter,req*http.Request){
log.Println("ServerSummary")
obj:=make(map[string]interface{})
obj["runtime_NumGoroutine"]=runtime.NumGoroutine()
obj["runtime_GOOS"]=runtime.GOOS
obj["runtime_GOARCH"]=runtime.GOARCH
obj["restApi_Addr"]=*restApiServer
obj["server_Time"]=time.Now()
obj["clients_Count"]=len(clientSMap)
varclist[]string
forcId,_:=rangeclientSMap{
clist=append(clist,cId)
}
obj["clients_List"]=clist
obj["forwardInfo"]=forwardInfo
res,err:=json.Marshal(obj)
iferr!=nil{
log.Println("jsonmarshal:",err)
return
}
rw.Header().Add("Content-Type","application/json;charset=utf-8")
_,err=rw.Write(res)
iferr!=nil{
log.Println("writeerr:",err)
}
return
}
funcForwardWork(rwhttp.ResponseWriter,req*http.Request){
req.ParseForm()
obj:=make(map[string]interface{})
obj["code"]=
obj["msg"]=""
paramAuth,hasAuth:=req.Form["auth"]
if!hasAuth{
log.Println("requestnoauth")
obj["code"]=
obj["msg"]="requestnoauth"
responseResult(obj,rw)
return
}
ifparamAuth[]!=*auth{
log.Println("requestauthfailed")
obj["code"]=
obj["msg"]="requestauthfailed"
responseResult(obj,rw)
return
}
paramStatus,hasStatus:=req.Form["status"]
if!hasStatus{
return
}
log.Println("param_status:",paramStatus)
ifparamStatus[]=="1"{
//启动服务
paramFromAddr,hasFromAddr:=req.Form["fromAddr"]
paramToAddr,hasToAddr:=req.Form["toAddr"]
ifgLocalConn!=nil{
gLocalConn.Close()
}
ifhasFromAddr&&hasToAddr{
goforwardPort(paramFromAddr[],paramToAddr[])
}
}
ifparamStatus[]=="0"{
//关闭服务
closeAllConn()
forwardInfo=""
}
responseResult(obj,rw)
return
}
funcresponseResult(datamap[string]interface{},rwhttp.ResponseWriter){
res,err:=json.Marshal(data)
iferr!=nil{
log.Println("jsonmarshal:",err)
return
}
rw.Header().Add("Content-Type","application/json;charset=utf-8")
_,err=rw.Write(res)
iferr!=nil{
log.Println("writeerr:",err)
}
}
funccloseAllConn(){
forcId,conn:=rangeclientSMap{
log.Println("clientMapid:",cId)
conn.Close()
delete(clientSMap,cId)
}
ifgLocalConn!=nil{
gLocalConn.Close()
log.Println("ListenerClose")
}else{
gLocalConn=nil
log.Println("Listenersettonil",gLocalConn)
}
}
//通读全部代码 发现这里存在未释放资源 targetPort 。准确说 需要先释放sourcePort 资源 再释放targetPort 资源
funcforwardPort(sourcePortstring,targetPortstring){
fmt.Println("sourcePort:",sourcePort,"targetPort:",targetPort)
localConn,err:=net.Listen("tcp",sourcePort)
iferr!=nil{
fmt.Println(err.Error())
return
}
gLocalConn=localConn
fmt.Println("服务启动成功,服务地址:",sourcePort)
forwardInfo=fmt.Sprintf("%s-%s",sourcePort,targetPort)
for{
fmt.Println("ReadytoAccept...")
sourceConn,err:=gLocalConn.Accept()
iferr!=nil{
log.Println("servererr:",err.Error())
break
}
//log.Println("client",sc.id,"createsession",sessionId)
id:=sourceConn.RemoteAddr().String()
clientSMap[id]=sourceConn
fmt.Println("conn.RemoteAddr().String():",id)
//targetPort:="172.16.128.83:22"
targetConn,err:=net.DialTimeout("tcp",targetPort,*time.Second)
gofunc(){
_,err=io.Copy(targetConn,sourceConn)
iferr!=nil{
//log.Fatalf("io.Copyfailed:%v",err)
fmt.Println("io.Copyfailed:",err.Error())
}
}()
gofunc(){
_,err=io.Copy(sourceConn,targetConn)
iferr!=nil{
//log.Fatalf("io.Copyfailed:%v",err)
fmt.Println("io.Copyfailed:",err.Error())
}
}()
}
//
log.Println("forwardPortend.")
}
<wiz_tmp_tag id="wiz-table-range-border" contenteditable="false" style="display: none;">
forwardPort.go的更多相关文章
- appium的几点总结
原文链接:http://blog.csdn.net/bear_w/article/details/50345283 1. 建立session时常用命令: DesiredCapabilities cap ...
- github之git基本命令介绍的简单整理
git 格式: git [--version] [--exec-path[=<path>]] [--html-path] [--info-path] [-p|--paginate|--no ...
- Linux防火墙配置(iptables, firewalld)
netfilter和底层实现 iptables firealld Linux中的防火墙 RHEL中有几种防火墙共存: iptables firewalld ip6tables ebtables 这些软 ...
- 我的第一个GitHub仓库
GitHub 仓库地址 https://github.com/FBean/test.git GitHub 常用命令 add--Add file contents to the index bisect ...
- 创建 Web 前端开发环境
Web 前端开发涉及多种工具,这里将常用工具的安装和配置进行说明,提供了详细的说明,为后继的开发创建一个坚实的基础. 本文介绍的工具有:NodeJS, NPM, Bower, Git 和 Grunt. ...
- Consul 启动命令
服务端: nohup consul agent -server -bootstrap-expect 1 -config-dir /etc/consul.d/ -data-dir /var/opt/co ...
- Git的常用命令的使用方法和解释
我们常用的git命令: add Add file contents to the index(将文件添加到暂存区) 用法: 保存某个文件到暂缓区:git add 文件名 保存当前路径的 ...
- Httptunnel教程
hts --forward-port localhost:22 80 htc -F 6789 192.168.10.1:80
- Git - Tutorial官方【转】
转自:http://www.vogella.com/tutorials/Git/article.html#git_rename_branch Lars Vogel Version 5.8 Copyri ...
随机推荐
- masm下几种常见函数调用方式
masm没有fastcall调用方式,其特点为: 1 第一个参数放入ecx,第二个参数放入edx: 2 如果有剩余参数则从右向左压栈: 3 被调用函数清理栈(平衡栈): 4 若有返回值放入eax: 5 ...
- Unity的资源管理
本篇文章翻译自Unity的官方文档,原文地址:https://docs.unity3d.com/Manual/BehindtheScenes.html Unity自动导入资源的同时为你管理这些资源产生 ...
- Nginx + IIS实现负载均衡 Session多站点共享
日子过得太索然无味了,研究了一下,所谓的负载均衡(主要是windows服务器IIS下的).先看看分析图:环境:linux服务器: centos 6.3windows服务器: windows serve ...
- JMM
1.JMM简介 i.内存模型概述 Java平台自动集成了线程以及多处理器技术,这种集成程度比Java以前诞生的计算机语言要厉害很多,该语言针对多种异构平台的平台独立性而使用的多线程技术支持也是具有开拓 ...
- java并发包分析之———ConcurrentSkipListMap
一.前言 concurrentHashMap与ConcurrentSkipListMap性能测试 在4线程1.6万数据的条件下,ConcurrentHashMap 存取速度是ConcurrentSki ...
- docker的安装和基础使用
Docker EE/Docker CE简介与版本规划 版本区别 Docker EE Docker EE由公司支持,可在经过认证的操作系统和云提供商中使用,并可运行来自Docker Store的.经过认 ...
- SignUtil
最近接的新项目 加密比较多 我就记录下. SignUtil是jnewsdk-mer-1.0.0.jar com.jnewsdk.util中的一个工具类.由于我没有百度到对应的信息.所以我只能看源码 ...
- 基于Python的数据分析:数据库索引效率探究
索引在数据库中是一个很特殊的存在,它的目的就是为了提高数据查询得效率.同样,它也有弊端,更新一个带索引的表的时间比更新一个没有带索引的时间更长.有得有失.我希望做一些研究测试,搞清楚索引对于我们使用数 ...
- Django模型层之字段查询参数及聚合函数
该系列教程系个人原创,并完整发布在个人官网刘江的博客和教程 所有转载本文者,需在顶部显著位置注明原作者及www.liujiangblog.com官网地址. 字段查询是指如何指定SQL WHERE子句的 ...
- Pascal Triangle
Description: Given numRows, generate the first numRows of Pascal's triangle. For example, given numR ...