forwardport--源码笔记--注释
package main
import (
"encoding/json"
"flag"
"fmt"
"io"
"log"
"net"
"net/http"
"os"
"os/signal"
"runtime"
"sync"
"syscall"
"time"
)
var restApiServer = flag.String("restApi", "", "listen addr for restapi") //监听rest full api 地址 ip+端口 默认值 0.0.0.0:8000 参见参数判断
var auth = flag.String("auth", "taven123", "restApi Password") //身份认证 默认rest api 是 taven123
var gLocalConn net.Listener //监听地址
var clientSMap map[string]net.Conn
var forwardInfo string
func main() {
clientSMap = make(map[string]net.Conn) //key 为 ip value 为 ip+端口创建的链接
//解析传入的参数
flag.Parse()
if *restApiServer == "" { //默认监听地址设置
*restApiServer = "0.0.0.0:8000" //默认值
}
go StartHttpServer(*restApiServer)//开启http 监听
log.Println("restApiServer:", *restApiServer)
fmt.Println("------------启动成功------------")
//开启线程同步锁
var w sync.WaitGroup
w.Add(2) //默认监听两个链接
//开一个并发线程,接收退出信号
go func() {
c := make(chan os.Signal, 1) //信号量通道
signal.Notify(c, os.Interrupt, syscall.SIGTERM) //监听系统信号
n := 0
f := func() {
<-c
n++
if n > 2 {
log.Println("force shutdown")
os.Exit(-1)
}
log.Println("received signal,shutdown")
closeAllConn() //关闭对应客户端ip 对应的链接
}
f()
go func() {
for {
f()
}
}()
//执行完成一次,Done() 等同于 Add(-1),计数不为0,则阻塞
w.Done()
}()
loop := func() {
w.Done()
}
loop()
w.Wait()
fmt.Println("------------程序执行完成------------")
}
func StartHttpServer(addr string) { //开启服务器端的信息监听
http.HandleFunc("/ServerSummary", ServerSummary) //获取服务状态
http.HandleFunc("/ForwardWork", ForwardWork) //端口映射服务
//
err := http.ListenAndServe(addr, http.DefaultServeMux)
if err != nil {
fmt.Println("ListenAndServe error: ", err.Error())
}
}
func ServerSummary(rw http.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 //请求rest服务
obj["server_Time"] = time.Now()
obj["clients_Count"] = len(clientSMap) //客户端数量
var clist []string
for cId, _ := range clientSMap { //迭代客户端来链接map
clist = append(clist, cId)
}
obj["clients_List"] = clist //客户端列表
obj["forwardInfo"] = forwardInfo //转向服务器信息
res, err := json.Marshal(obj) //生成json数据
if err != nil {
log.Println("json marshal:", err)
return
}
rw.Header().Add("Content-Type", "application/json;charset=utf-8") //添加响应数据头 数据格式
_, err = rw.Write(res)
if err != nil {
log.Println("write err:", err)
}
return
}
func ForwardWork(rw http.ResponseWriter, req *http.Request) {
req.ParseForm() //解析请求参数
obj := make(map[string]interface{}) //存储状态信息
obj["code"] = 0
obj["msg"] = ""
paramAuth, hasAuth := req.Form["auth"] //获取参数 密码
if !hasAuth { //情况一 不存在密码
log.Println("request no auth")
obj["code"] = 1
obj["msg"] = "request no auth"
responseResult(obj, rw) //json数据格式 响应数据
return
}
if paramAuth[0] != *auth { //情况二 存在密码 但是密码不匹配
log.Println("request auth failed")
obj["code"] = 1
obj["msg"] = "request auth failed"
responseResult(obj, rw)//json数据格式 响应数据
return
}
paramStatus, hasStatus := req.Form["status"] //status:如果是开启转发,则为1,如果是关闭转发,则为0
if !hasStatus { //没有开启转发 直接返回
return
}
log.Println("param_status:", paramStatus)
if paramStatus[0] == "1" { //开启转发
//启动服务
paramFromAddr, hasFromAddr := req.Form["fromAddr"] //要用来在A机器上监听的一个端口,用来给客户端连接 =====服务器
paramToAddr, hasToAddr := req.Form["toAddr"] //把fromAddr端口的数据转发到哪个IP的端口上 ===转发到的服务器
if gLocalConn != nil { //目标服务器是否存在链接 不存在 直接拒接
gLocalConn.Close() //释放资源
}
if hasFromAddr && hasToAddr { //目标服务器 和 转发到服务器存在
go forwardPort(paramFromAddr[0], paramToAddr[0])
}
}
if paramStatus[0] == "0" { //转发状态为0 时 直接释放资源
//关闭服务
closeAllConn()
forwardInfo = ""
}
responseResult(obj, rw)// json格式数据响应客户端
return
}
func responseResult(data map[string]interface{}, rw http.ResponseWriter) {// json格式数据响应客户端
res, err := json.Marshal(data) //生成json数据格式
if err != nil {
log.Println("json marshal:", err)
return
}
rw.Header().Add("Content-Type", "application/json;charset=utf-8") //响应头数据格式设置
_, err = rw.Write(res) //响应写入数据
if err != nil {
log.Println("write err:", err)
}
}
func closeAllConn() {// 关闭连接
for cId, conn := range clientSMap {
log.Println("clientMap id:", cId)
conn.Close() //关闭连接 conn
delete(clientSMap, cId) //从客户端map中删除对应的数据
}
if gLocalConn != nil { //关闭元数据链接【服务端】
gLocalConn.Close()
log.Println("Listener Close")
} else {
gLocalConn = nil
log.Println("Listener set to nil", gLocalConn)
}
}
func forwardPort(sourcePort string, targetPort string) { //转发服务
fmt.Println("sourcePort:", sourcePort, "targetPort:", targetPort)
localConn, err := net.Listen("tcp", sourcePort) //目标服务器 监听服务器
if err != nil {
fmt.Println(err.Error())
return
}
gLocalConn = localConn
fmt.Println("服务启动成功,服务地址:", sourcePort)
forwardInfo = fmt.Sprintf("%s - %s", sourcePort, targetPort)
for {
fmt.Println("Ready to Accept ...")
sourceConn, err := gLocalConn.Accept() //获取监听服务链接
if err != nil {
log.Println("server err:", err.Error())
break
}
//log.Println("client", sc.id, "create session", sessionId)
id := sourceConn.RemoteAddr().String() //获取客户端ip地址
clientSMap[id] = sourceConn
fmt.Println("conn.RemoteAddr().String() :", id)
//targetPort := "172.16.128.83:22"
targetConn, err := net.DialTimeout("tcp", targetPort, 30*time.Second)//获取转发服务链接
//开启协程 支持从客户端 到服务端 以及 从 服务端到客户端数据 读写
go func() {
_, err = io.Copy(targetConn, sourceConn) //从sourceConn读取数据 写入到targetConn中
if err != nil {
//log.Fatalf("io.Copy 1 failed: %v", err)
fmt.Println("io.Copy 1 failed:", err.Error())
}
}()
go func() {
_, err = io.Copy(sourceConn, targetConn)//从targetConn读取数据 写入到sourceConn中
if err != nil {
//log.Fatalf("io.Copy 2 failed: %v", err)
fmt.Println("io.Copy 2 failed:", err.Error())
}
}()
}
//
log.Println("forwardPort end.")
}
forwardport--源码笔记--注释的更多相关文章
- AsyncTask源码笔记
AsyncTask源码笔记 AsyncTask在注释中建议只用来做短时间的异步操作,也就是只有几秒的操作:如果是长时间的操作,建议还是使用java.util.concurrent包中的工具类,例如Ex ...
- Java Arrays 源码 笔记
Arrays.java是Java中用来操作数组的类.使用这个工具类可以减少平常很多的工作量.了解其实现,可以避免一些错误的用法. 它提供的操作包括: 排序 sort 查找 binarySearch() ...
- Tomcat8源码笔记(六)连接器Connector分析
根据 Tomcat8源码笔记(五)组件Container分析 前文分析,StandardService的初始化重心由 StandardEngine转移到了Connector的初始化,本篇记录下Conn ...
- Golang构建HTTP服务(一)--- net/http库源码笔记
搭建一个简单的Go Web服务器 Go语言标准库 - net/http 在学习Go语言有一个很好的起点,Go语言官方文档很详细,今天我们学习的Go Web服务器的搭建就需要用到Go语言官方提供的标准库 ...
- Zepto源码笔记(一)
最近在研究Zepto的源码,这是第一篇分析,欢迎大家继续关注,第一次写源码笔记,希望大家多指点指点,第一篇文章由于首次分析原因不会有太多干货,希望后面的文章能成为各位大大心目中的干货. Zepto是一 ...
- robotlegs2.0框架实例源码带注释
robotlegs2.0框架实例源码带注释 Robotlegs2的Starling扩展 有个老外写了robotleges2的starling扩展,地址是 https://github.com/brea ...
- redis源码笔记(一) —— 从redis的启动到command的分发
本作品采用知识共享署名 4.0 国际许可协议进行许可.转载联系作者并保留声明头部与原文链接https://luzeshu.com/blog/redis1 本博客同步在http://www.cnblog ...
- Tomcat8源码笔记(八)明白Tomcat怎么部署webapps下项目
以前没想过这么个问题:Tomcat怎么处理webapps下项目,并且我访问浏览器ip: port/项目名/请求路径,以SSM为例,Tomcat怎么就能将请求找到项目呢,项目还是个文件夹类型的? Tom ...
- Tomcat8源码笔记(七)组件启动Server Service Engine Host启动
一.Tomcat启动的入口 Tomcat初始化简单流程前面博客介绍了一遍,组件除了StandardHost都有博客,欢迎大家指文中错误.Tomcat启动类是Bootstrap,而启动容器启动入口位于 ...
- Tomcat8源码笔记(五)组件Container分析
Tomcat8源码笔记(四)Server和Service初始化 介绍过Tomcat中Service的初始化 最先初始化就是Container,而Container初始化过程是咋样的? 说到Contai ...
随机推荐
- 使用bootstrap table 插件固定表头时 表头与表格内容无法对齐
在使用bootstrap table开发后台管理系统,表格利用bootstrap-table插件来实现,使用bootstrap-table过程中,会出现表头错位的情况 表头对不齐效果: 解决的方法: ...
- java并发包分析之———AQS框架
一.什么是同步器 多线程并发的执行,之间通过某种 共享 状态来同步,只有当状态满足 xxxx 条件,才能触发线程执行 xxxx . 这个共同的语义可以称之为同步器.可以认为以上所有的锁机制都可以基 ...
- 安装VirtualBox后 不能选择64bit的系统
之前在台式机上安装VirtualBox,一切OK,能够安装64位的任何版本iso包今天在hp笔记本上安装,安装VirtualBox完毕后,只能选择32位的iso版本. 而我目前只有一个linux64b ...
- 精彩源于起点——2018年潍坊市首次青少年Python编程公开课
有一种语言叫计算机语言 I want to talk with Computer 春遇到冬,有了岁月 天遇到地,有了永恒 我们拥有的, 不止是长大, 还有那份长大的悲欢经历. 未来会有很多可能, 但一 ...
- SignUtil
最近接的新项目 加密比较多 我就记录下. SignUtil是jnewsdk-mer-1.0.0.jar com.jnewsdk.util中的一个工具类.由于我没有百度到对应的信息.所以我只能看源码 ...
- 关于图数据库查询语言:Cypher
Neo4j Cypher Refcard:http://neo4j.com/docs/cypher-refcard/current/Neo4j发布开源图查询语言openCypher:http://ww ...
- 0基础一分钟入门Python
这篇文章面向所有想学python的小伙伴(甚至你从没听过编程),这篇文章将会带你以最快的速度入门python.赶快上车,时间来不及了... 一,下载和安装python 1.下载: 1.1 python ...
- java之Spring(AOP)前奏-动态代理设计模式(上)
我们常常会遇到这样的事,项目经理让你为一个功能类再加一个功能A,然后你加班为这个类加上了功能A: 过了两天又来了新需求,再在A功能后面加上一个新功能B,你加班写好了这个功能B,加在了A后面:又过 了几 ...
- 程序员DD 《Spring boot教程系列》补充
最近在跟着程序员DD的Spring boot教程系列学习Spring boot,由于年代原因,Spring boot已经发生了一些变化,所以在这里进行一些补充. 补充的知识大多来自评论区,百度,Sta ...
- CRM客户关系管理系统(九)
第九章.filter_horizontal优化和kingadmin删除功能 9.1.filter_horizontal优化 (1)添加Choose ALL 和Remove ALL table_obj_ ...