package main

import (
    "net"
    "time"
)

// 初始化代理服务
func initProxy() {
    Log.Infof("Proxying %s -> %s\n", Config.Bind, Config.Backend)
    server, err := net.Listen("tcp", Config.Bind)
    if err != nil {
        Log.Fatal(err)
    }
    // 等待的队列长度
    waitQueue := make(chan net.Conn, Config.WaitQueueLen)
    // 最大并发连接
    connPools := make(chan bool, Config.MaxConn)
    for i := 0; i < Config.MaxConn; i++ {
        connPools <- true
    }
    // 等待连接处理
    go waitConn(waitQueue, connPools)
    // 接收连接并抛给管道处理
    for {
        conn, err := server.Accept()
        if err != nil {
            Log.Error(err)
            continue
        }
        Log.Infof("Received connection from %s.\n", conn.RemoteAddr())
        waitQueue <- conn
    }
}

// 连接数控制
func waitConn(waitQueue chan net.Conn, connPools chan bool) {
    for conn := range waitQueue {
        // 接收一个链接,连接池释放一个
        <-connPools
        go func(conn net.Conn) {
            handleConn(conn)
            // 链接处理完毕,增加
            connPools <- true
            Log.Infof("Closed connection from %s.\n", conn.RemoteAddr())
        }(conn)
    }
}

// 处理连接
func handleConn(conn net.Conn) {
    defer conn.Close()
    // 根据链接哈希选择机器
    proxySvr, ok := getBackendSvr(conn)
    if !ok {
        return
    }
    // 链接远程代理服务器
    remote, err := net.Dial("tcp", proxySvr.identify)
    if err != nil {
        Log.Error(err)
        proxySvr.failTimes++
        return
    }
    // 等待双向连接完成
    complete := make(chan bool, 2)
    oneSwitch := make(chan bool, 1)
    otherSwitch := make(chan bool, 1)
    // 将当前客户端链接发送的数据发送给远程被代理的服务器
    go transaction(conn, remote, complete, oneSwitch, otherSwitch)
    // 将远程服务返回的数据返回给客户端
    go transaction(remote, conn, complete, otherSwitch, oneSwitch)
    <-complete
    <-complete
    remote.Close()
}

// 数据交换传输(从from读数据,再写入to)
func transaction(from, to net.Conn, complete, oneSwitch, otherSwitch chan bool) {
    var err error
    var read int
    bytes := make([]byte, 1024)
    for {
        select {
        case <-otherSwitch:
            complete <- true
            return
        default:
            timeOutSec := time.Duration(Config.Timeout) * time.Second
            // 设置超时时间
            from.SetReadDeadline(time.Now().Add(timeOutSec))
            read, err = from.Read(bytes)
            if err != nil {
                complete <- true
                oneSwitch <- true
                return
            }
            // 设置超时时间
            to.SetWriteDeadline(time.Now().Add(timeOutSec))
            _, err = to.Write(bytes[:read])
            if err != nil {
                complete <- true
                oneSwitch <- true
                return
            }
        }
    }
}

proxy.go的更多相关文章

  1. 实现代理设置proxy

    用户在哪些情况下是需要设置网络代理呢? 1. 内网上不了外网,需要连接能上外网的内网电脑做代理,就能上外网:多个电脑共享上外网,就要用代理: 2.有些网页被封,通过国外的代理就能看到这被封的网站:3. ...

  2. Could not create SSL connection through proxy serve-svn

    RA layer request failedsvn: Unable to connect to a repository at URL xxxxxx 最后:Could not create SSL ...

  3. could not initialize proxy - no Session

    这是一个精典的问题:因为我们在hibernate里面load一个对象出来时,用到的是代理对象,也就是说当我们在执行load方法时并没有发sql语句,而是返回一个proxy对象.只有当们具体用到哪个ge ...

  4. ABP源码分析三十七:ABP.Web.Api Script Proxy API

    ABP提供Script Proxy WebApi为所有的Dynamic WebApi生成访问这些WebApi的JQuery代理,AngularJs代理以及TypeScriptor代理.这些个代理就是j ...

  5. Java设计模式之代理模式(Proxy)

    前言: 最近在研究Retrofit开源框架的时候,其主要核心代码是通过注解标示参数,动态代理模式实现具体接口,反射机制进行参数解析,最终实现发送请求.其实之前在学习Xutils源码的时候,Xutils ...

  6. 设计模式(十三)代理模式(Proxy Pattern)

    一.引言 在软件开发过程中,有些对象有时候会由于网络或其他的障碍,以至于不能够或者不能直接访问到这些对象,如果直接访问对象给系统带来不必要的复杂性,这时候可以在客户端和目标对象之间增加一层中间层,让代 ...

  7. 设计模式-代理模式(Proxy Model)

    文 / vincentzh 原文连接:http://www.cnblogs.com/vincentzh/p/5988145.html 目录 1.写在前面 2.概述 3.目的 4.结构组成 5.实现 5 ...

  8. 神秘代理-Proxy

    前言: 代理模式作为常见的设计模式之一,在项目开发中不可或缺.本文就尝试着揭开代理的神秘面纱,也欢迎各路人批评指正! 1.如何实现代理: [假设有个关于汽车移动(move)的计时需求]设计:Movea ...

  9. 12,13 Proxy和Reflect

    Proxy和Reflect Proxy(代理) Proxy用于修改某些操作的默认行为,等同于在语言层面做出修改,所以属于一种"元编程"(meta programming),即对编程 ...

  10. nova instance出错:"message": "Proxy error: 502 Read from server failed

    执行 $ nova resize instance1 时候出错: {, "details": " File \"/opt/stack/nova/nova/com ...

随机推荐

  1. XSS攻击过滤处理

    关于XSS攻击 XSS是一种经常出现在web应用中的计算机安全漏洞,它允许恶意web用户将代码植入到提供给其它用户使用的页面中. XSS漏洞的危害 网络钓鱼,包括盗取各类用户账号: 窃取用户cooki ...

  2. x&(x-1)

    x&(x-1)可以用来求出x是否为2幂次方数:当&的结果为0时,x原值是2幂次方数,否则就不是2幂次方数: x=x&(x-1)即把x从低位开始的第一个1改成0.如1000,把1 ...

  3. Hibernate的二级缓存策略

    Hibernate的二级缓存策略的一般过程如下: 1) 条件查询的时候,总是发出一条select * from table_name where …. (选择所有字段)这样的SQL语句查询数据库,一次 ...

  4. Odoo 学习【一】http & rpc

    HTTP Odoo 中http类中的Root是wsgi应用的入口主程序. 入口,wsgi_server调用如下: def application(environ, start_response): i ...

  5. Java EE的未来

    http://www.infoq.com/cn/articles/enterprise-Java-opinion 作为InfoQ下一年编辑关注点审核工作的一部分,我们挑选了Java作为深入探讨的主题. ...

  6. 解决Android SDK Manager更新失败问题

    from:http://www.ztyhome.com/android-sdk-update/ 问题描述: 使用SDK Manager更新时无法完成更新ADT时无法解析https://dl-ssl.g ...

  7. Effective Java 第三版——40. 始终使用Override注解

    Tips <Effective Java, Third Edition>一书英文版已经出版,这本书的第二版想必很多人都读过,号称Java四大名著之一,不过第二版2009年出版,到现在已经将 ...

  8. gitlab钩子搭建

    目标:在本地开发机上push代码到GitLab仓库时,通过钩子同步到测试服务器 准备工作GitLab 服务器一台测试服务器一台本地开发服务器一台 1.在gitlab上新建一个项目,名称test2.在本 ...

  9. JavaScript要点汇总——The Most Important

    关于JavaScript的基础变量,运算符的详解以及基本的分支循环嵌套已经在 JS基础变量及JS中的运算符 JS中的循环分支嵌套 说过了,今天我们所说的是做网页中最长用到的东西.内容不算少,要有耐心, ...

  10. 自定义ExtJS主题

    ExtJS提供的可以使用的主题包对于创建一个干净专业的程序来说已经很有创意了,然而,你可能还是会希望提供自己的一种设计方式或现在存在的企业设计方式. 从历史上来说,给程序美化就是指的给html标签提供 ...