Python中socket经ssl加密后server开多线程

        前几天手撸Python socket代码,撸完之后经过ssl加密,确保数据的安全,外加server端开启多线程保证一个客户端连接有一个线程来服务客户端,走了不少的弯路,网上的信息啥的要么有ssl没有服务端的多线程,要不只有多线程没有加ssl加密,对于新手做这种需求还是有些困难的,这里,经过我!李帅帅的实践得出各种版本的代码以及最终终结版的代码,话说妇联4都他娘的药终结了,还有几天就上映了,说票挺贵的,这他娘的看个屁,大不了过半个月在啃,不知道黑寡妇最后咋样了,啧啧,挺好的一个姑娘,那脸,那腰,那身材,那充满对李帅帅爱意恒生的眼神,咳咳,不扯了不扯了,直接上代码自己看

1.没有ssl加密没有线程前的server端

server.py
import ssl
import threading

class ListenServer(object):

    PORT = 8000  # 监听的端口
    # IP = "127.0.0.1"
    IP = "127.0.0.1"

    def __init__(self):  # 初始化
        # 生成SSL上下文
        # 切记!!!!!加密一定要在跟客户端连接前进行加密,否则没有意义!!!!!!
        self.context = ssl.SSLContext(ssl.PROTOCOL_SSLv23)
        # 加载服务器所用证书和私钥
        self.context.load_cert_chain('cert/server.crt', 'cert/server.key')
        # 开启socket,
        try:
            #                           套接字:ipv4           TCP协议
            self.SOCK = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
            # 绑定ip以及端口号
            self.SOCK.bind((self.IP, self.PORT))
            print("监听iP: %s; 监听Port: %s" % (self.IP, self.PORT))
            # 监听数,也可以理解服务端一次性处理多少的客户端的连接请求
            self.SOCK.listen(100)
        except Exception as e:
            print("socket errer!: %s" % e)
        else:
            print("success SOCKET!")

    def run_pro(self, act):
        '''
            程序的主要函数
            处理连接客户端
        '''

        print("server running...\r\n")
        # act = active.Active()
        with self.context.wrap_socket(self.SOCK, server_side=True) as ssock:
            while True:
                client_socket, client_ip = ssock.accept()  # 获取客户端

                # p = Process(target=process, args=(
                # client_socket, client_ip))  # 开始新进程
                p = threading.Thread(target=act,args=(client_socket, client_ip))
                p.start()
                # p.join() #线程会等待


if __name__ == "__main__":
    s = ListenServer()
    s.run_pro()
View Code

没有ssl加密没有线程前的client端

import socket
import ssl

class client_ssl:
    def send_hello(self,):
        # 生成SSL上下文
        context = ssl.SSLContext(ssl.PROTOCOL_SSLv23)
        # 加载信任根证书
        context.load_verify_locations('cert/ca.crt')
        print("现在证书啥的都加载完成了")
        # 与服务端建立socket连接
        with socket.create_connection(('127.0.0.1', 8000)) as sock:
            print("准备连接啦")
            # 将socket打包成SSL socket
            # 一定要注意的是这里的server_hostname不是指服务端IP,而是指服务端证书中设置的CN,我这里正好设置成127.0.1而已
            with context.wrap_socket(sock, server_hostname='127.0.0.1') as ssock:

                # 向服务端发送信息
                print("向服务端发送信息")
                msg = "do i connect with server ?".encode("utf-8")
                ssock.send(msg)
                # 接收服务端返回的信息
                msg = ssock.recv(1024).decode("utf-8")
                print("receive msg from server :" , msg)
                ssock.close()

if __name__ == "__main__":
    client = client_ssl()
    client.send_hello()
View Code

2.没有ssl加密,只在服务端开启多线程的server

from socketserver import BaseRequestHandler,ThreadingTCPServer
import threading

class Hander(BaseRequestHandler):
    def handle(self):
        '''
        实现并发的效果就是重写父类的handle方法(直接写逻辑,连接准备listen()等都干好了)
        :return:
        '''
        address = self.client_address
        print(address, "客户端连接了!!")
        while 1:
            # 接受客户端的数据
            data = self.request.recv(1024)
            # 判断连接与否
            if len(data) > 0:
                print("客户端", address, data.decode("utf-8"))
                cur_thread = threading.current_thread()
                self.request.sendall('response'.encode("utf-8"))
            else:
                print("关闭连接")
                break

if __name__ == '__main__':
    HOST = '192.168.0.177'
    PORT = 8000
    ADDR = (HOST, PORT)
    server = ThreadingTCPServer(ADDR,Hander)
    print("listening")
    server.serve_forever()
View Code

3.没有加线程的server端

#!C:\Python3.6.5\python.exe
# -*- coding: gbk -*-

import socket
import ssl
import threading


class WSGIServer(object):
    def __init__(self, port):
        """初始化对象"""
        # 生成SSL上下文
        self.context = ssl.SSLContext(ssl.PROTOCOL_SSLv23)  # 指定ssl版本
        # 加载信任根证书
        self.context.load_cert_chain(certfile="cert/ca.crt", keyfile="cert/ca.key")
        # self.context.load_verify_locations('cert.pem')  # server端的证书
        # self.context.load_verify_locations('key.pem')  # server端的
        print("现在证书啥的都加载完成了")
        # 创建套接字
        print(self.context)
        self.tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        # 解决程序端口占用问题
        self.tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
        # 绑定本地ip地址
        self.tcp_server_socket.bind(("127.0.0.1", port))
        # 将套接字变为监听套接字,最大连接数量为100
        self.tcp_server_socket.listen(100)
        print("最大连接数是100昂")

    def run_forever(self):
        """设备连接"""
        print("等待设备的链接ing。。。")
        # 1.等待设备连接(通过ip地址和端口建立tcp连接)
        #   如果有设备连接,则会生成用于设备和服务器通讯的套接字:new_socket
        #   会获取到设备的ip地址和端口
        print("等待生产新的套接字。。。")

        print("哎呀,等到了!!!")

        with self.context.wrap_socket(self.tcp_server_socket, server_side=True) as ssock:
            # print("这一步有错")
            while 1:
                new_socket, client_addr = ssock.accept()
                print("设备{0}已连接".format(client_addr))


    def service_machine(self, new_socket, client_addr):
        """业务处理"""
        while 1:
            # 3.接收设备发送的数据,单次最大1024字节,按‘gbk’格式解码
            receive_data = new_socket.recv(1024).decode("gbk")
            # 4.如果设备发送的数据不为空
            if receive_data:
                # 4.1 打印接收的数据,这里可以将设备发送的数据写入到文件中
                # 获取设备的ID信息
                print(receive_data)
                if receive_data[0:6] == "report":
                    response = "SET OK:" + receive_data
                else:
                    receive_data = receive_data[6:].split(",")[0]
                    # 拼接响应数据
                    response = "alarm=" + receive_data + ",Switch:clear"
                print(response)
                # 4.2 返回原数据作为应答,按‘utf-8’格式编码
                new_socket.send(response.encode("utf-8"))
            # 5.当设备断开连接时,会收到空的字节数据,判断设备已断开连接
            else:
                print('设备{0}断开连接...'.format(client_addr))
                break

        # 关闭套接字
        new_socket.close()


def main(port):
    """创建一个WEB服务器"""
    wsgi_server = WSGIServer(port)
    print("服务器已开启")
    wsgi_server.run_forever()


if __name__ == '__main__':
    port = 8000  # 指定端口
    main(port)
View Code

4.ssl+多线程的server端!!!!!!

#!C:\Python3.6.5\python.exe
# -*- coding: gbk -*-

import socket
import ssl
import threading


class WSGIServer(object):
    def __init__(self, port):
        """初始化对象"""
        # 生成SSL上下文
        self.context = ssl.SSLContext(ssl.PROTOCOL_SSLv23)  # 指定ssl版本
        # 加载信任根证书
        self.context.load_cert_chain(certfile="cert/ca.crt", keyfile="cert/ca.key")
        # self.context.load_verify_locations('cert.pem')  # server端的证书
        # self.context.load_verify_locations('key.pem')  # server端的
        print("现在证书啥的都加载完成了")
        # 创建套接字
        print(self.context)
        self.tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        # 解决程序端口占用问题
        self.tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
        # 绑定本地ip地址
        self.tcp_server_socket.bind(("127.0.0.1", port))
        # 将套接字变为监听套接字,最大连接数量为100
        self.tcp_server_socket.listen(100)
        print("最大连接数是100昂")

    def run_forever(self):
        """设备连接"""
        print("等待设备的链接ing。。。")
        # 1.等待设备连接(通过ip地址和端口建立tcp连接)
        #   如果有设备连接,则会生成用于设备和服务器通讯的套接字:new_socket
        #   会获取到设备的ip地址和端口
        print("等待生产新的套接字。。。")

        print("哎呀,等到了!!!")

        with self.context.wrap_socket(self.tcp_server_socket, server_side=True) as ssock:
            # print("这一步有错")
            while 1:
                new_socket, client_addr = ssock.accept()
                print("设备{0}已连接".format(client_addr))

                #     # 2.创建线程处理设备的需求
                t1 = threading.Thread(target=self.service_machine, args=(new_socket, client_addr))
                t1.start()

    def service_machine(self, new_socket, client_addr):
        """业务处理"""
        while 1:
            # 3.接收设备发送的数据,单次最大1024字节,按‘gbk’格式解码
            receive_data = new_socket.recv(1024).decode("gbk")
            # 4.如果设备发送的数据不为空
            if receive_data:
                # 4.1 打印接收的数据,这里可以将设备发送的数据写入到文件中
                # 获取设备的ID信息
                print(receive_data)
                if receive_data[0:6] == "report":
                    response = "SET OK:" + receive_data
                else:
                    receive_data = receive_data[6:].split(",")[0]
                    # 拼接响应数据
                    response = "alarm=" + receive_data + ",Switch:clear"
                print(response)
                # 4.2 返回原数据作为应答,按‘utf-8’格式编码
                new_socket.send(response.encode("utf-8"))
            # 5.当设备断开连接时,会收到空的字节数据,判断设备已断开连接
            else:
                print('设备{0}断开连接...'.format(client_addr))
                break

        # 关闭套接字
        new_socket.close()


def main(port):
    """创建一个WEB服务器"""
    wsgi_server = WSGIServer(port)
    print("服务器已开启")
    wsgi_server.run_forever()


if __name__ == '__main__':
    port = 8000  # 指定端口
    main(port)
View Code

 

转载于:https://www.cnblogs.com/lzqrkn/p/10737154.html

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Python 使用 socket 模块来实现 socket 通信。 服务端代码示例: ```python import socket # 创建 socket 对象 server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 绑定端口号 server_socket.bind(('127.0.0.1', 65432)) # 监听连接 server_socket.listen() # 接受客户端连接 client_socket, address = server_socket.accept() print('Connected by', address) # 接收客户端发送的数据 data = client_socket.recv(1024) print('Received:', data.decode()) # 发送数据给客户端 client_socket.sendall(b'Hello, World!') # 关闭 socket client_socket.close() server_socket.close() ``` 客户端代码示例: ```python import socket # 创建 socket 对象 client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 连接服务端 client_socket.connect(('127.0.0.1', 65432)) # 发送数据给服务端 client_socket.sendall(b'Hello, Server!') # 接收服务端发送的数据 data = client_socket.recv(1024) print('Received:', data.decode()) # 关闭 socket client_socket.close() ``` 这是一个简单的 socket 通信的示例,在实际应用还有很多细节需要考虑,比如错误处理、数据传输的安全性等。 ### 回答2: Python通过内置的socket模块来实现网络通信。要使用socket进行通信,通常需要以下步骤: 1. 创建一个socket对象: import socket sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) 在创建socket对象时,需要指定地址族(IPv4或IPv6)和socket类型(TCP或UDP)。上面的示例,创建了一个IPv4的TCP socket对象。 2. 绑定地址和端口: server_address = ('localhost', 8000) # 指定服务器地址和端口 sock.bind(server_address) 这里指定了服务器的地址和端口号,将socket对象与指定的地址和端口进行绑定。 3. 监听连接请求: sock.listen(5) 通过listen方法,让socket对象始监听来自客户端的连接请求。这里的参数5表示最大连接数为5,可以根据需要进行调整。 4. 接受连接请求: while True: client_socket, client_address = sock.accept() # 处理连接请求,client_socket表示与客户端通信的socket对象 通过accept方法,接受客户端的连接请求,并返回与客户端通信的socket对象和客户端地址。 5. 发送和接收数据: data = client_socket.recv(1024) # 接收客户端发送的数据 client_socket.sendall(b'Hello, World!') # 发送数据给客户端 使用客户端和服务器端的socket对象来进行数据的接收和发送。接收数据使用recv方法,参数表示一次接收的最大字节数。发送数据使用sendall方法,参数为要发送的数据。 6. 关闭连接: client_socket.close() sock.close() 在通信结束后,需要关闭socket对象,释放系统资源。 通过以上步骤,可以实现Pythonsocket通信。可以根据具体的需求对代码进行扩展和优化。 ### 回答3: Python实现Socket通信可以通过`socket`模块来实现。 首先,需要导入`socket`模块: ```python import socket ``` 创建一个Socket对象,并指定通信类型(如TCP或UDP)和地址族(如IPv4或IPv6): ```python sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) ``` 根据需要,可以选择设置一些Socket选项,如超时时间: ```python sock.settimeout(5) ``` 接下来,可以进行Socket的绑定操作,指定要监听的IP地址和端口号: ```python sock.bind(("127.0.0.1", 8080)) ``` 如果是TCP通信,需要将Socket设置为监听模式: ```python sock.listen(1) ``` 如果是UDP通信,可以直接发送数据: ```python sock.sendto(data, address) ``` 如果是TCP通信,需要先接受客户端的连接请求,然后进行数据传输: ```python client_sock, client_address = sock.accept() data = client_sock.recv(1024) ``` 最后,记得在通信结束后关闭Socket连接: ```python sock.close() ``` 以上是Python实现Socket通信的基本步骤。通过Socket,可以实现不同设备间的数据传输和通信。在实际应用,可以根据具体需求进行更复杂的操作,如使用多线程处理并发请求、使用SSL加密通信等。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值