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的更多相关文章

  1. appium的几点总结

    原文链接:http://blog.csdn.net/bear_w/article/details/50345283 1. 建立session时常用命令: DesiredCapabilities cap ...

  2. github之git基本命令介绍的简单整理

    git 格式: git [--version] [--exec-path[=<path>]] [--html-path] [--info-path] [-p|--paginate|--no ...

  3. Linux防火墙配置(iptables, firewalld)

    netfilter和底层实现 iptables firealld Linux中的防火墙 RHEL中有几种防火墙共存: iptables firewalld ip6tables ebtables 这些软 ...

  4. 我的第一个GitHub仓库

    GitHub 仓库地址 https://github.com/FBean/test.git GitHub 常用命令 add--Add file contents to the index bisect ...

  5. 创建 Web 前端开发环境

    Web 前端开发涉及多种工具,这里将常用工具的安装和配置进行说明,提供了详细的说明,为后继的开发创建一个坚实的基础. 本文介绍的工具有:NodeJS, NPM, Bower, Git 和 Grunt. ...

  6. Consul 启动命令

    服务端: nohup consul agent -server -bootstrap-expect 1 -config-dir /etc/consul.d/ -data-dir /var/opt/co ...

  7. Git的常用命令的使用方法和解释

    我们常用的git命令: add        Add file contents to the index(将文件添加到暂存区)  用法: 保存某个文件到暂缓区:git add 文件名 保存当前路径的 ...

  8. Httptunnel教程

    hts --forward-port localhost:22 80 htc -F 6789 192.168.10.1:80

  9. Git - Tutorial官方【转】

    转自:http://www.vogella.com/tutorials/Git/article.html#git_rename_branch Lars Vogel Version 5.8 Copyri ...

随机推荐

  1. iframe不起作用?你可能碰到它了。

    有一个需求要在iframe里显示一个网站,但设置iframe的src后,iframe并没有起作用.然后打开控制台,发现错误如下: , 对其搜索找到了答案:https://stackoverflow.c ...

  2. Linux的chkconfig命令详解

    chkconfig命令主要用来更新(启动或停止)和查询系统服务的运行级信息.谨记chkconfig不是立即自动禁止或激活一个服务,它只是简单的改变了符号连接. 使用语法: chkconfig [--a ...

  3. CSS的display:table

    好久都没有写博客了,似乎总是觉得少了些什么-- 刚好最近在工作中遇到了一个新的东西display:table,这个也是css的布局的一种,而且又是display的,之前已经写过了display的fle ...

  4. var $this = $(this)是什么意思?

    var $this = $(this) 声明一个变量,$this 是变量名,加$说明是jquery对象. 给声明的变量赋值,赋的值是将this元素转换为jQuery对象.

  5. Redis案例——商品秒杀,购物车

    秒杀案例: <?php header("content-type:text/html;charset=utf-8"); $redis = new redis(); $resu ...

  6. ScalaPB(4): 通用跨系统protobuf数据,sbt设置

    我们知道,在集群环境节点之间进行交换的数据必须经过序列化/反序列化处理过程,而在这方面protobuf是一个比较高效.易用的模式.用户首先在.proto文件中用IDL来定义系统中各种需要进行交换的数据 ...

  7. js相关

    记录工作中用到的js实用方法: 1. 判断包含:indexOf "https".indexOf("http") 包含结果为0,不包含结果为-1: 通常用(&qu ...

  8. JS 数据类型、赋值、深拷贝和浅拷贝

    js 数据类型 六种 基本数据类型: Boolean. 布尔值,true 和 false. null. 一个表明 null 值的特殊关键字. JavaScript 是大小写敏感的,因此 null 与 ...

  9. python装饰器小计

    1.装饰器:本质是函数,是用来给其他函数添加附加扩展功能的函数,其返回值是一个函数(函数指针) 2.装饰器作用:不改变函数源代码和函数调用方式的前提下添加函数的附加功能. 3.装饰器储备知识点: A. ...

  10. python 关于操作文件的相关模块(os,sys,shutil,subprocess,configparser)

    一:os模块 os模块提供了许多允许你程序与操作系统直接交互的功能 os.getcwd() 获取当前工作目录,即当前python脚本工作的目录路径 os.chdir("dirname&quo ...