• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

proxy.go源码阅读

原作者: [db:作者] 来自: [db:来源] 收藏 邀请
package main

import (
    "net"
    "time"
)

func initProxy() {

    pLog.Infof("Proxying %s -> %s\n", pConfig.Bind, pConfig.Backend)  //输出服务地址   后端服务地址列表  

    server, err := net.Listen("tcp", pConfig.Bind)  //建立tcp连接
    if err != nil {
        pLog.Fatal(err)
    }

    waitQueue := make(chan net.Conn, pConfig.WaitQueueLen)  //建立连接队列  队列默认等待队列长度---channel  net.Conn  ==控制最大排队队列
    availPools := make(chan bool, pConfig.MaxConn)  //建立最大连接数--channel  bool=== 控制最大连接数
    for i := 0; i < pConfig.MaxConn; i++ {  //
        availPools <- true
    }

    go loop(waitQueue, availPools)

    for {
        connection, err := server.Accept()  //等待获取下一次连接
        if err != nil {
            pLog.Error(err)
        } else {
            pLog.Infof("Received connection from %s.\n", connection.RemoteAddr())
            waitQueue <- connection
        }
    }
}

func loop(waitQueue chan net.Conn, availPools chan bool) {
    for connection := range waitQueue { //循环等待队列中 排队等待需要处理的数据--连接
        <-availPools  //从通道中 连接池中取出一个
        go func(connection net.Conn) {
            handleConnection(connection)
            availPools <- true  //使用结束放回连接池中  目的控制连接数量  通道阻塞特性
            pLog.Infof("Closed connection from %s.\n", connection.RemoteAddr())
        }(connection)
    }
}

func handleConnection(connection net.Conn) {
    defer connection.Close()

    bksvr, ok := getBackendSvr(connection)
    if !ok {
        return
    }
    remote, err := net.Dial("tcp", bksvr.svrStr)

    if err != nil {
        pLog.Error(err)
        bksvr.failTimes++
        return
    }

    //等待双向连接完成
    complete := make(chan bool, 2)
    oneSide := make(chan bool, 1)
    otherSide := make(chan bool, 1)
    go pass(connection, remote, complete, oneSide, otherSide)
    go pass(remote, connection, complete, otherSide, oneSide)
    <-complete
    <-complete
    remote.Close()
}

// copy Content two-way
func pass(from net.Conn, to net.Conn, complete chan bool, oneSide chan bool, otherSide chan bool) {
    var err error
    var read int
    bytes := make([]byte, 256)

    for {
        select {

        case <-otherSide:
            complete <- true
            return

        default:

            from.SetReadDeadline(time.Now().Add(time.Duration(pConfig.Timeout) * time.Second))
            read, err = from.Read(bytes)
            if err != nil {
                complete <- true
                oneSide <- true
                return
            }

            to.SetWriteDeadline(time.Now().Add(time.Duration(pConfig.Timeout) * time.Second))
            _, err = to.Write(bytes[:read])
            if err != nil {
                complete <- true
                oneSide <- true
                return
            }
        }
    }
}


鲜花

握手

雷人

路过

鸡蛋
该文章已有0人参与评论

请发表评论

全部评论

专题导读
上一篇:
Go Redis发布时间:2022-07-10
下一篇:
go学习笔记(1)gocommand发布时间:2022-07-10
热门推荐
热门话题
阅读排行榜

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap