嘗試用最短的Python代碼來(lái)實(shí)現(xiàn)服務(wù)器和代理服務(wù)器
一個(gè)最簡(jiǎn)單的服務(wù)器
Python擁有這種單獨(dú)起一個(gè)服務(wù)器監(jiān)聽(tīng)端口的能力,用標(biāo)準(zhǔn)庫(kù)的wsgiref就行。
from wsgiref.simple_server import make_server
def simple_app(environ, start_response):
status = '200 OK'
response_headers = [('Content-type','text/plain')]
start_response(status, response_headers)
return ['Hello world!\n']
httpd = make_server('', 80, simple_app)
httpd.serve_forever()

50行代碼實(shí)現(xiàn)代理服務(wù)器
之前遇到一個(gè)場(chǎng)景是這樣的:
我在自己的電腦上需要用mongodb圖形客戶(hù)端,但是mongodb的服務(wù)器地址沒(méi)有對(duì)外網(wǎng)開(kāi)放,只能通過(guò)先登錄主機(jī)A,然后再?gòu)腁連接mongodb服務(wù)器B。
本來(lái)想通過(guò)ssh端口轉(zhuǎn)發(fā)的,但是我沒(méi)有從機(jī)器A連接ssh到B的權(quán)限。于是就自己用Python寫(xiě)一個(gè)。
原理很簡(jiǎn)單。
1.開(kāi)一個(gè)socket server監(jiān)聽(tīng)連接請(qǐng)求
2.每接受一個(gè)客戶(hù)端的連接請(qǐng)求,就往要轉(zhuǎn)發(fā)的地址建一條連接請(qǐng)求。即client->proxy->forward。proxy既是socket服務(wù)端(監(jiān)聽(tīng)client),也是socket客戶(hù)端(往forward請(qǐng)求)。
3.把client->proxy和proxy->forward這2條socket用字典給綁定起來(lái)。
4.通過(guò)這個(gè)映射的字典把send/recv到的數(shù)據(jù)原封不動(dòng)的傳遞
下面上代碼。
#coding=utf-8
import socket
import select
import sys
to_addr = ('xxx.xxx.xx.xxx', 10000)#轉(zhuǎn)發(fā)的地址
class Proxy:
def __init__(self, addr):
self.proxy = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
self.proxy.bind(addr)
self.proxy.listen(10)
self.inputs = [self.proxy]
self.route = {}
def serve_forever(self):
print 'proxy listen...'
while 1:
readable, _, _ = select.select(self.inputs, [], [])
for self.sock in readable:
if self.sock == self.proxy:
self.on_join()
else:
data = self.sock.recv(8096)
if not data:
self.on_quit()
else:
self.route[self.sock].send(data)
def on_join(self):
client, addr = self.proxy.accept()
print addr,'connect'
forward = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
forward.connect(to_addr)
self.inputs += [client, forward]
self.route[client] = forward
self.route[forward] = client
def on_quit(self):
for s in self.sock, self.route[self.sock]:
self.inputs.remove(s)
del self.route[s]
s.close()
if __name__ == '__main__':
try:
Proxy(('',12345)).serve_forever()#代理服務(wù)器監(jiān)聽(tīng)的地址
except KeyboardInterrupt:
sys.exit(1)
效果截圖如下。

相關(guān)文章
python 對(duì)字典按照value進(jìn)行排序的方法
這篇文章主要介紹了python 對(duì)字典按照value進(jìn)行排序的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-05-05
使用Python完成SAP客戶(hù)端的打開(kāi)和系統(tǒng)登陸功能
這篇文章主要介紹了使用Python完成SAP客戶(hù)端的打開(kāi)和系統(tǒng)登陸,涉及到SAP的打開(kāi)和登錄過(guò)程,本文重點(diǎn)介紹如何通過(guò)Python完成SAP應(yīng)用程序的打開(kāi)并進(jìn)入特定的”用戶(hù)指定系統(tǒng)“,需要的朋友可以參考下2022-05-05
對(duì)Python3 解析html的幾種操作方式小結(jié)
今天小編就為大家分享一篇對(duì)Python3 解析html的幾種操作方式小結(jié),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-02-02
Python 判斷是否為質(zhì)數(shù)或素?cái)?shù)的實(shí)例
下面小編就為大家?guī)?lái)一篇Python 判斷是否為質(zhì)數(shù)或素?cái)?shù)的實(shí)例。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-10-10
使用Anaconda創(chuàng)建Pytorch虛擬環(huán)境的排坑詳細(xì)教程
PyTorch是一個(gè)開(kāi)源的Python機(jī)器學(xué)習(xí)庫(kù),基于Torch,用于自然語(yǔ)言處理等應(yīng)用程序,下面這篇文章主要給大家介紹了關(guān)于使用Anaconda創(chuàng)建Pytorch虛擬環(huán)境的相關(guān)資料,需要的朋友可以參考下2022-12-12
python?kornia計(jì)算機(jī)視覺(jué)庫(kù)實(shí)現(xiàn)圖像變化
這篇文章主要為大家介紹了python?kornia計(jì)算機(jī)視覺(jué)庫(kù)實(shí)現(xiàn)圖像變化算法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01
Python批量查詢(xún)關(guān)鍵詞微信指數(shù)實(shí)例方法
在本篇文章中小編給大家整理的是關(guān)于Python批量查詢(xún)關(guān)鍵詞微信指數(shù)實(shí)例方法以及相關(guān)代碼,需要的朋友們可以跟著學(xué)習(xí)下。2019-06-06

