gpt4 book ai didi

java - 为 SSL 套接字上的每个请求创建新进程给出 "TypeError: Cannot serialize socket object",但对普通/非 SSL 套接字做同样的工作

转载 作者:太空宇宙 更新时间:2023-11-03 14:32:59 24 4
gpt4 key购买 nike

我正在尝试在 python 服务器和 java 客户端中使用使用 java keytool 生成的 key 和证书。我创建了 key 和 keystore 、导出证书、将证书添加到信任库、将 keystore 转换为标准 pkcs 格式,然后从 pkcs 中提取 key 和证书以在 python 服务器中使用。 (前三步来自 here,后三步来自 here)。这个问题有点跟进this question生成 key 和证书的详细步骤可以在that question中找到。 .

我的 SSL 服务器看起来像这样

server.py

import socket
import multiprocessing as mup
import ssl

def worker_ssl(data_socket, client_address):
print("Inside worker")
#some processing

def start_server_ssl():

socketObj = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server_address = ("127.0.0.1", 6000)
socketObj.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
socketObj.bind(server_address)
socketObj.listen(10)

ssl_socket = ssl.wrap_socket(socketObj,
server_side = True,
certfile="cert.pem",
keyfile="key.pem")

while True:
print("Waiting For Connections .........\n")
try:
data_socket, client_address = ssl_socket.accept()

process = mup.Process(target=worker_ssl, args=(data_socket, client_address))
process.daemon = True
process.start()

except socket.error as msg:
print (" [ERROR] : %s " % msg)
continue

socketObj.close()
ssl_socket.shutdown(socket.SHUT_RDWR)
ssl_socket.close()

if __name__ == '__main__':
start_server_ssl()

SSL 客户端看起来像这样:

Client4Py.java

import java.io.BufferedWriter;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.nio.charset.StandardCharsets;
import java.security.KeyStore;

import javax.net.ssl.KeyManagerFactory;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLSocket;
import javax.net.ssl.SSLSocketFactory;
import javax.net.ssl.TrustManager;
import javax.net.ssl.TrustManagerFactory;

public class Client4Py {
static KeyStore ks;
static KeyManagerFactory kmf;
static TrustManagerFactory tmf;
static SSLContext sc;
static TrustManager[] trustManagers;

static {
try {
ks = KeyStore.getInstance("JKS");
ks.load(new FileInputStream("D:\\javasslstores\\truststore.jks"), "passwd123".toCharArray());

kmf = KeyManagerFactory.getInstance("SunX509");
kmf.init(ks, "passwd123".toCharArray());

tmf = TrustManagerFactory.getInstance("SunX509");
tmf.init(ks);

sc = SSLContext.getInstance("TLS");

sc.init(null, tmf.getTrustManagers(), null);
} catch (Exception e) {
System.out.println(e.getMessage());
System.out.println(e.getStackTrace());
}
}

public static void main(String[] args) throws IOException {
SSLSocketFactory ssf = sc.getSocketFactory();
SSLSocket socket = (SSLSocket) ssf.createSocket("127.0.0.1", 6000);
socket.startHandshake();

PrintWriter out = new PrintWriter(new BufferedWriter(new OutputStreamWriter(socket.getOutputStream(),StandardCharsets.UTF_8)));
//PrintWriter out = new PrintWriter(new BufferedWriter(new OutputStreamWriter(socket.getOutputStream())));

out.println("<<startMessage>>");
out.println("Message from Client4Py");
out.println("<<endMessage>>");
out.flush();

if (out.checkError())
System.out.println(
"SSLSocketClient: java.io.PrintWriter error");

out.close();
socket.close();
}
}

先运行server再运行client后,服务器控制台输出如下:

1    Waiting For Connections .........
2
3 Traceback (most recent call last):
4 File "D:\workspaces\workspace6\PythonServer\server.py", line 40, in <module>
5 start_server_ssl()
6 File "D:\workspaces\workspace6\PythonServer\server.py", line 29, in start_server_ssl
7 process.start()
8 File "D:\Programs\python\python-3.6.6-amd64\lib\multiprocessing\process.py", line 105, in start
9 self._popen = self._Popen(self)
10 File "D:\Programs\python\python-3.6.6-amd64\lib\multiprocessing\context.py", line 223, in _Popen
11 return _default_context.get_context().Process._Popen(process_obj)
12 File "D:\Programs\python\python-3.6.6-amd64\lib\multiprocessing\context.py", line 322, in _Popen
13 return Popen(process_obj)
14 File "D:\Programs\python\python-3.6.6-amd64\lib\multiprocessing\popen_spawn_win32.py", line 65, in __init__
15 reduction.dump(process_obj, to_child)
16 File "D:\Programs\python\python-3.6.6-amd64\lib\multiprocessing\reduction.py", line 60, in dump
17 ForkingPickler(file, protocol).dump(obj)
18 File "D:\Programs\python\python-3.6.6-amd64\lib\socket.py", line 185, in __getstate__
19 raise TypeError("Cannot serialize socket object")
20 TypeError: Cannot serialize socket object
21 Traceback (most recent call last):
22 File "<string>", line 1, in <module>
23 File "D:\Programs\python\python-3.6.6-amd64\lib\multiprocessing\spawn.py", line 99, in spawn_main
24 new_handle = reduction.steal_handle(parent_pid, pipe_handle)
25 File "D:\Programs\python\python-3.6.6-amd64\lib\multiprocessing\reduction.py", line 87, in steal_handle
26 _winapi.DUPLICATE_SAME_ACCESS | _winapi.DUPLICATE_CLOSE_SOURCE)
27 PermissionError: [WinError 5] Access is denied

可以看到第20行,有TypeError: Cannot serialize socket object

代码在删除所有 SSL 内容后开始工作

当我评论对 wrap_socket() 的调用时,将 ssl_socket.accept() 替换为 socketObject.accept() 并评论 ssl_socket.shutdown()close(),代码开始工作。它根据需要从 worker() 创建新进程,并结束在控制台上打印 Inside worker。这是修改后的非 SSL 服务器:

import socket
import multiprocessing as mup
# import ssl

def worker_ssl(data_socket, client_address):
print("Inside worker")
#some processing

def start_server_ssl():

socketObj = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server_address = ("127.0.0.1", 6000)
socketObj.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
socketObj.bind(server_address)
socketObj.listen(10)

# ssl_socket = ssl.wrap_socket(socketObj,
# server_side = True,
# certfile="cert.pem",
# keyfile="key.pem")

while True:
print("Waiting For Connections .........\n")
try:
data_socket, client_address = socketObj.accept()

process = mup.Process(target=worker_ssl, args=(data_socket, client_address))
process.daemon = True
process.start()

except socket.error as msg:
print (" [ERROR] : %s " % msg)
continue

socketObj.close()
# ssl_socket.shutdown(socket.SHUT_RDWR)
# ssl_socket.close()

if __name__ == '__main__':
start_server_ssl()

最佳答案

我通过在子进程中用 SSLContext 明确地接受套接字来解决这个问题。我创建 SSLContext 多处理所需的一切都没有序列化问题。

更新:以下是服务器 sudo 代码,说明了上面的解释:

def client_connection_handler(client_socket, client_address, ssl_settings):
context = ssl.SSLContext()
context.load_cert_chain(certfile=ssl_settings['certfile'], keyfile=ssl_settings['keyfile'],)
context.verify_mode = ssl_settings["verify_mode"]
context.load_verify_locations(cafile=ssl_settings["cafile"])
client_secured_socket = context.wrap_socket(client_socket, server_side=True)
#
# send and receive data
#
client_secured_socket.close()


def server():
server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server_socket.bind((hostname, port))
server_socket.listen(5)
while True:
# Waiting for client connection
client_socket, client_address = server_socket.accept()
concurrent_connections = len(multiprocessing.active_children())
if max_concurrent_clients > concurrent_connections:
p = multiprocessing.Process(target=client_connection_handler, args=(client_socket, client_address, ssl_settings))
p.daemon = True
p.start()
continue
# Max client connection has been reached
client_socket.close()

关于java - 为 SSL 套接字上的每个请求创建新进程给出 "TypeError: Cannot serialize socket object",但对普通/非 SSL 套接字做同样的工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50910579/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com