我有一個基本的客戶端服務器腳本在Python中使用套接字。服務器綁定到特定的端口並等待客戶端連接。當客戶端連接時,它們會顯示一個raw_input提示符,用於將輸入的命令發送到服務器上的一個子進程並將輸出返回給客戶端。 有時,當我執行來自客戶端的命令時,輸出將掛起,直到按[Enter]鍵時纔會顯示raw_input提示符。 起初我以爲這可能是一個緩衝問題,但它發生在我使用的命令用小的輸出,像「清除」或「LS」等Python客戶端 - 服務器腳本掛起,直到我按[輸入]
客戶端代碼:
import os, sys
import socket
from base64 import *
import time
try:
HOST = sys.argv[1]
PORT = int(sys.argv[2])
except IndexError:
print("You must specify a host IP address and port number!")
print("usage: ./handler_client.py 192.168.1.4 4444")
sys.exit()
socksize = 4096
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
try:
server.connect((HOST, PORT))
print("[+] Connection established!")
print("[+] Type ':help' to view commands.")
except:
print("[!] Connection error!")
sys.exit(2)
while True:
data = server.recv(socksize)
cmd = raw_input(data)
server.sendall(str(cmd))
server.close()
服務器代碼:
import os,sys
import socket
import time
from subprocess import Popen,PIPE,STDOUT,call
HOST = ''
PORT = 4444
socksize = 4096
activePID = []
conn = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
conn.bind((HOST, PORT))
conn.listen(5)
print("Listening on TCP port %s" % PORT)
def reaper():
while activePID:
pid,stat = os.waitpid(0, os.WNOHANG)
if not pid: break
activePID.remove(pid)
def handler(connection):
time.sleep(3)
while True:
cmd = connection.recv(socksize)
proc = Popen(cmd,
shell=True,
stdout=PIPE,
stderr=PIPE,
stdin=PIPE,
)
stdout, stderr = proc.communicate()
if cmd == ":killme":
connection.close()
sys.exit(0)
elif proc:
connection.send(stdout)
connection.send("\nshell => ")
connection.close()
os._exit(0)
def accept():
while 1:
global connection
connection, address = conn.accept()
print "[!] New connection!"
connection.send("\nshell => ")
reaper()
childPid = os.fork() # forks the incoming connection and sends to conn handler
if childPid == 0:
handler(connection)
else:
activePID.append(childPid)
accept()
問題是我在這裏錯過了什麼?看起來,客戶端應該能夠從打印輸出轉換到shell提示符,而不是毫無意義。有什麼想法嗎? – ohdae 2012-04-14 18:19:04
您可能想查看[argparse](http://docs.python.org/library/argparse.html#module-argparse)的命令行參數 - 當它與您當前的代碼一樣簡單時,並非真正需要,但如果它變得更復雜,那麼值得一看。 – 2012-04-14 18:21:53
整個代碼比實際的代碼複雜得多,這只是一個客戶端服務器代碼片段。我只是把它打亂了一點,所以我的例子不是5頁長。感謝您的建議,雖然:) – ohdae 2012-04-14 18:35:08