欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

python實(shí)現(xiàn)代理服務(wù)功能實(shí)例

 更新時(shí)間:2013年11月15日 14:43:27   作者:  
本文分析了代理服務(wù)的原理,并使用PYTHON實(shí)現(xiàn)了一個(gè)簡(jiǎn)單的代理服務(wù)功能供大家參考使用
代理服務(wù)原理很簡(jiǎn)單,就拿瀏覽器與web服務(wù)器來(lái)說。無(wú)非是A瀏覽器
發(fā)request給B代理,B代理再把request把送給C web服務(wù),然后C的reponse->B->A。
要寫web代理服務(wù)就要先了解下http協(xié)議,當(dāng)然并不要多深入,除非要實(shí)現(xiàn)強(qiáng)大的功能:修改XX信息、
負(fù)載均衡等。http請(qǐng)求由三部分組成:請(qǐng)求行、消息報(bào)頭、請(qǐng)求正文;
詳細(xì)的網(wǎng)上有,想了解可以看看。下面是一個(gè)正常的GET請(qǐng)求頭(Cookie部分本人沒截屏,使用的系統(tǒng)w7):



可以看到首行:GET是請(qǐng)求方法, /是路徑,在后面是協(xié)議版本;第二行以后是請(qǐng)求報(bào)頭,都是鍵值對(duì)形式;
GET方法沒有正文。post有正文,除此之外,請(qǐng)求方法頭部基本一致,每一行結(jié)尾都是\r\n。
基本的請(qǐng)求方法,如下:

GET        請(qǐng)求獲取Request-URI所標(biāo)識(shí)的資源
POST      在Request-URI所標(biāo)識(shí)的資源后附加新的數(shù)據(jù)
HEAD      請(qǐng)求獲取由Request-URI所標(biāo)識(shí)的資源的響應(yīng)消息報(bào)頭
PUT         請(qǐng)求服務(wù)器存儲(chǔ)一個(gè)資源,并用Request-URI作為其標(biāo)識(shí)
DELETE   請(qǐng)求服務(wù)器刪除Request-URI所標(biāo)識(shí)的資源
TRACE     請(qǐng)求服務(wù)器回送收到的請(qǐng)求信息,主要用于測(cè)試或診斷
CONNECT 保留將來(lái)使用
OPTIONS  請(qǐng)求查詢服務(wù)器的性能,或者查詢與資源相關(guān)的選項(xiàng)和需求
但是使用代理后,從代理服務(wù)上得到的請(qǐng)求如下:


與第一張圖片對(duì)比一下,有什么不同......第一行的資源路徑不對(duì)。當(dāng)瀏覽器上設(shè)置代理請(qǐng)求時(shí)把整個(gè)url都作為資源路徑了,所以我們要把域名刪掉,然后代理服務(wù)器在把修改后的請(qǐng)求發(fā)送給目標(biāo)
web服務(wù)器。就這么簡(jiǎn)單,當(dāng)然CONNECT方法特別,要特別對(duì)待,所以先說其他方法。
基本的思路:
1、代理服務(wù)器運(yùn)行監(jiān)聽,當(dāng)有客戶端瀏覽器請(qǐng)求到來(lái)時(shí)通過accept()獲得client句柄(或者叫描述符);
2、利用client描述符接收瀏覽器發(fā)來(lái)的request,分離出第一行為了修改第一行和獲得method,
要去掉的的部分,除去http://的部分用targetHost表示吧。
3、通過第2步能夠獲得方法method、request和targetHost,這一步可以根據(jù)不同的method做不同的處理,
由于GET、POET、PUT、DELETE等除了CONNECT處理基本一致,所以處理首行,比如:
復(fù)制代碼 代碼如下:

GET http://www.a.com/ HTTP/1.1
替換為
GET / HTTP/1.1

此時(shí)targetHost也就是紅色的部分,默認(rèn)的請(qǐng)求80端口,此時(shí)port為80;如果targetHost中有端口(比如www.a.com:8081),
就要分理處端口,此時(shí)port為8081。然后根據(jù)targetHost和port連接到目標(biāo)服務(wù)器target了,實(shí)現(xiàn)代碼如下:
復(fù)制代碼 代碼如下:

def getTargetInfo(self,host): #處理targetHost獲得網(wǎng)址和端口,作為返回值。
        port=0
        site=None
        if ':' in host:
            tmp=host.split(':')
            site=tmp[0]
            port=int(tmp[1])
        else:
            site=host
            port=80
        return site,port
def commonMethod(self,request): #處理除CONNECT以外的方法
        tmp=self.targetHost.split('/')
        net=tmp[0]+'//'+tmp[2]
        request=request.replace(net,'') #替換掉首行不必要的部分
        targetAddr=self.getTargetInfo(tmp[2]) #調(diào)用上面的函數(shù)
        try:
            (fam,_,_,_,addr)=socket.getaddrinfo(targetAddr[0],targetAddr[1])[0]
        except Exception as e:
            print e
            return
        self.target=socket.socket(fam)
        self.target.connect(addr) #連接到目標(biāo)web服務(wù)

4、這一步就好辦了,根據(jù)第三步處理后的request就可以self.target.send(request)發(fā)送給web服務(wù)器了。
5、這一步web服務(wù)器的reponse反響通過代理服務(wù)直接轉(zhuǎn)發(fā)給客戶端就行了,本人用了非阻塞select,可以試試epoll。
基本步驟就是這樣,使用的方法函數(shù)可以改進(jìn),比如主函數(shù)部分使用的多線程或者多進(jìn)程,怎樣選擇......
但是思路差不多都是這樣啦。想測(cè)試的話,chrome安裝SwitchySharp插件,設(shè)置一下,代理端口8083;
firefox插件autoproxy。
 對(duì)于connect的處理還在解決中(如果有博友幫助就更好了),所以現(xiàn)在這個(gè)代理程序不支持https協(xié)議。
代理服務(wù)可以獲得http協(xié)議的所有信息,想了解學(xué)習(xí)http,利用代理服務(wù)器是個(gè)不錯(cuò)的方法。
下面附上代碼
復(fù)制代碼 代碼如下:

#-*- coding: UTF-8 -*-
import socket,select
import sys
import thread
from multiprocessing import Process
class Proxy:
    def __init__(self,soc):
        self.client,_=soc.accept()
        self.target=None
        self.request_url=None
        self.BUFSIZE=4096
        self.method=None
        self.targetHost=None
    def getClientRequest(self):
        request=self.client.recv(self.BUFSIZE)
        if not request:
            return None
        cn=request.find('\n')
        firstLine=request[:cn]
        print firstLine[:len(firstLine)-9]
        line=firstLine.split()
        self.method=line[0]
        self.targetHost=line[1]
        return request
    def commonMethod(self,request):
        tmp=self.targetHost.split('/')
        net=tmp[0]+'//'+tmp[2]
        request=request.replace(net,'')
        targetAddr=self.getTargetInfo(tmp[2])
        try:
            (fam,_,_,_,addr)=socket.getaddrinfo(targetAddr[0],targetAddr[1])[0]
        except Exception as e:
            print e
            return
        self.target=socket.socket(fam)
        self.target.connect(addr)
        self.target.send(request)
        self.nonblocking()
    def connectMethod(self,request): #對(duì)于CONNECT處理可以添加在這里
        pass
    def run(self):
        request=self.getClientRequest()
        if request:
            if self.method in ['GET','POST','PUT',"DELETE",'HAVE']:
                self.commonMethod(request)
            elif self.method=='CONNECT':
                self.connectMethod(request)
    def nonblocking(self):
        inputs=[self.client,self.target]
        while True:
            readable,writeable,errs=select.select(inputs,[],inputs,3)
            if errs:
                break
            for soc in readable:
                data=soc.recv(self.BUFSIZE)
                if data:
                    if soc is self.client:
                        self.target.send(data)
                    elif soc is self.target:
                        self.client.send(data)
                else:
                    break
        self.client.close()
        self.target.close()
    def getTargetInfo(self,host):
        port=0
        site=None
        if ':' in host:
            tmp=host.split(':')
            site=tmp[0]
            port=int(tmp[1])
        else:
            site=host
            port=80
        return site,port
if __name__=='__main__':     
    host = '127.0.0.1'
    port = 8083
    backlog = 5
    server = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
    server.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)
    server.bind((host,port))
    server.listen(5)
    while True:
        thread.start_new_thread(Proxy(server).run,())
        # p=Process(target=Proxy(server).run, args=()) #多進(jìn)程
        # p.start()

相關(guān)文章

  • 在django中,關(guān)于session的通用設(shè)置方法

    在django中,關(guān)于session的通用設(shè)置方法

    今天小編就為大家分享一篇在django中,關(guān)于session的通用設(shè)置方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧
    2019-08-08
  • Python?Pipeline處理數(shù)據(jù)工作原理探究

    Python?Pipeline處理數(shù)據(jù)工作原理探究

    如果你是一個(gè)Python開發(fā)者,你可能聽過"pipeline"這個(gè)術(shù)語(yǔ),但?pipeline?到底是什么,它又有什么用呢?在這篇文章中,我們將探討?Python?中的?pipeline?概念,它們是如何工作的,以及它們?nèi)绾螏椭憔帉懜逦?、更高效的代碼
    2024-01-01
  • python計(jì)算機(jī)視覺opencv卡號(hào)識(shí)別示例詳解

    python計(jì)算機(jī)視覺opencv卡號(hào)識(shí)別示例詳解

    這篇文章主要為大家介紹了python計(jì)算機(jī)視覺opencv卡號(hào)識(shí)別的實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下 希望能夠有所幫助,祝大家多多進(jìn)步
    2021-11-11
  • 詳解Python開發(fā)中如何使用Hook技巧

    詳解Python開發(fā)中如何使用Hook技巧

    這篇文章主要介紹了詳解Python開發(fā)中如何使用Hook技巧,詳細(xì)的介紹了Python Hook的用法和示例,有興趣的可以了解一下
    2017-11-11
  • python?中raise用法

    python?中raise用法

    這篇文章主要介紹了python?中raise用法,Python?允許我們?cè)诔绦蛑惺謩?dòng)設(shè)置異常,就是使用raise?語(yǔ)句來(lái)實(shí)現(xiàn),下面我們就來(lái)看看raise的具體用法,文章內(nèi)容介紹詳細(xì),具有一定的參考價(jià)值,需要的小伙伴可以參考一下
    2021-12-12
  • Python實(shí)現(xiàn)循環(huán)語(yǔ)句的方式分享

    Python實(shí)現(xiàn)循環(huán)語(yǔ)句的方式分享

    這篇文章主要為大家詳細(xì)介紹了Python中實(shí)現(xiàn)循環(huán)語(yǔ)句的常用方式,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價(jià)值,感興趣的小伙伴可以了解一下
    2023-05-05
  • 完美解決pyinstaller打包報(bào)錯(cuò)找不到依賴pypiwin32或pywin32-ctypes的錯(cuò)誤

    完美解決pyinstaller打包報(bào)錯(cuò)找不到依賴pypiwin32或pywin32-ctypes的錯(cuò)誤

    這篇文章主要介紹了完美解決pyinstaller打包報(bào)錯(cuò)找不到依賴pypiwin32或pywin32-ctypes的錯(cuò)誤,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-04-04
  • Python基于Tensor FLow的圖像處理操作詳解

    Python基于Tensor FLow的圖像處理操作詳解

    這篇文章主要介紹了Python基于Tensor FLow的圖像處理操作,結(jié)合實(shí)例形式分析了Python基于Tensor FLow操作圖像解碼、縮放、剪切、翻轉(zhuǎn)、調(diào)整對(duì)比度、明度、飽和度等相關(guān)操作技巧,需要的朋友可以參考下
    2020-01-01
  • python 彈窗提示警告框MessageBox的實(shí)例

    python 彈窗提示警告框MessageBox的實(shí)例

    今天小編就為大家分享一篇python 彈窗提示警告框MessageBox的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧
    2019-06-06
  • Python通過RabbitMQ服務(wù)器實(shí)現(xiàn)交換機(jī)功能的實(shí)例教程

    Python通過RabbitMQ服務(wù)器實(shí)現(xiàn)交換機(jī)功能的實(shí)例教程

    RabbitMQ是一個(gè)基于消息隊(duì)列的服務(wù)器程序,Python可以通過Pika庫(kù)來(lái)驅(qū)動(dòng)它,這里我們將來(lái)看一個(gè)Python通過RabbitMQ服務(wù)器實(shí)現(xiàn)交換機(jī)功能的實(shí)例教程:
    2016-06-06

最新評(píng)論