python实现MC协议(SLMP 3E帧)的TCP服务端(篇二)

python实现MC协议(SLMP 3E帧)的TCP服务端是一件稍微麻烦点的事情。它不像modbusTCP那样,可以使用现成的pymodbus模块去实现。但是,我们可以根据协议帧进行组包,自己去实现帧的格式,而这一切可以基于socket模块。本文为第二篇。

二、读写保持寄存器的完整交互包

# 客户端发送(读) -》
50 00 00 FF FF 03 00 0C 00 10 00 01 04 00 00 00 00 00 A8 05 00
# 《- 服务端应答
D0 00 00 FF FF 03 00 0C 00 00 00 73 00 00 00 00 00 00 00 00 00
# 客户端发送(写) -》
50 00 00 FF FF 03 00 16 00 10 00 01 14 00 00 0A 00 00 A8 05 00 4E 47 00 00 00 00 00 00 00 00
# 《- 服务端应答
D0 00 00 FF FF 03 00 02 00 00 00

1、分析交互包

基于上述交互包,我们查阅官方文档发现交互包使用的是二进制代码。那么,二进制代码与ASCII代码有什么区别呢?

SLMP(Seamless Message Protocol)3E帧有两种表示方式:二进制格式和ASCII格式。它们的区别在于数据的传输方式和呈现形式。

(1)二进制格式

在二进制格式中,SLMP 3E帧中的各个字段(如帧头、副帧头、命令码、数据等)以二进制形式直接编码和传输。数据在网络中以原始的二进制位模式传输,这种方式效率较高,适用于网络传输。二进制格式通常用于实际的网络通信中,数据以二进制流的形式在网络上传输。

(2)ASCII格式

在ASCII格式中,SLMP 3E帧中的各个字段被转换成ASCII字符表示。数据以ASCII码的文本形式进行传输,每个字节被转换为两个ASCII字符(通常是十六进制表示)。ASCII格式通常用于调试和人机界面中,方便人们查看和理解数据。

总的来说,二进制格式适用于机器之间的网络通信,而ASCII格式适用于人机交互和调试过程中的数据显示。选择哪种格式取决于具体的应用场景和需求。

因此,本文实现的是二进制格式,如果你会实现二进制格式,那么你也能实现ASCII格式。

2、读写保持寄存器的请求处理

(1)表头

客户端的两个请求,相同部分都为50 00 00 FF FF 03 00,我们姑且称之为表头。

(2)读/写长度(协议帧的长度)

0C 00是固定长度(读的时候报文都是这么长)与16 00 根据实际长度变化,表示后面数据的长度,例如前者,应该以00 0C来看长度,表示后面有12个00那样的长度。

(3)固定值

10 00

(4)读/写指令

01 04 / 01 14

(5)读/写寄存器地址

00 00 00 00 00 A8 05 00 /  00 00 0A 00 00 A8 05 00,其中写的0A 00代表从第10个保持寄存器,05表示读写5个寄存器

3、读写保持寄存器的响应处理

(1)表头

客户端的两个请求,相同部分都为D0 00 00 FF FF 03 00,我们姑且称之为表头。

(2)长度(协议帧的长度)

读:0C 00根据实际长度变化,写:02 00 可以不变化。

(3)固定值

00 00

(4)读/写响应

响应实际读到的数据 / 无

4、程序设计

根据上述内容,实现了一个定制MC服务器,能够处理保持寄存器的读写请求,给出正确的响应。

import socket
import struct

# 创建一个TCP/IP套接字
server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# 绑定套接字到特定地址和端口
server_address = ('192.168.1.188', 12345)  # 服务器地址和端口
server_socket.bind(server_address)

# 监听连接
server_socket.listen(1)

print('等待客户端连接...')
connection, client_address = server_socket.accept()

print('客户端已连接:', client_address)

def request_verdict(req_bytes_frame):  # req_bytes_frame是字节数据b'\x02\x00\x08\x00\x00\x00\x00\x00\x10\x00\x01\x01\x02\x03\x04\x03'
    command = req_bytes_frame.hex()[22:26]  # 转成16进制字符串好数据处理
    if command in ["0104", "0401"]:  # 判断读写
        return False # 读
    elif command in ["0114", "1401"]:
        return True  # 写
    else:
        raise ValueError("读写指令错误!")

def write_response_frame(req_bytes_frame):
    response = "D00000FFFF030002000000"  # 写成功则返回这一串数据
    content = req_bytes_frame.hex()[42:]  # 看一下客户端想写的内容
    print("客户端想要写入的内容:", bytes.fromhex(content).decode())
    return bytes().fromhex(response)

def read_response_frame(req_bytes_frame, res_data):
    header = "D00000FFFF03000C000000"  # 读的响应头
    nums = req_bytes_frame.hex()[38:42]  # 获取客户端想要读的寄存器个数
    act_nums_hex = nums[2:] + nums[:2]  # 涉及大端序和小端序,需要转一下
    act_nums = int(act_nums_hex, 16)  # 得到实际数量
    res_data_hex = ''.join([hex(ord(c))[2:].zfill(2) for c in res_data])  # 将要返回的数据转成16进制字符串
    response = header + res_data_hex + '0'*(act_nums*2*2-len(res_data_hex))  # 根据请求数量返回对应的内容
    return bytes().fromhex(response)

try:
    while True:
        # 接收客户端请求
        request = connection.recv(1024)
        print("001:", request)
        if request:
            flag = request_verdict(request)
            if flag:  # 响应写
                response = write_response_frame(request)
                print("002:",response)
            else:  # 响应读
                response = read_response_frame(request, "start")
                print("003:",response)
            connection.sendall(response)
finally:
    # 清理连接
    connection.close()

三、关于MC协议的整体实现

通过“二”,我们实现了一个基于MC协议的保持寄存器的读写服务器,但并没有像pymodbus这种现成模块那样完整实现,这里探讨一下还可以做的事。

1、SLMP 3E帧实现步骤

实现SLMP(Seamless Message Protocol) 3E帧协议涉及到网络通信、数据处理、错误处理等多个步骤。以下是实现SLMP 3E帧的一般步骤:

(1)建立TCP连接:

在服务端,监听指定端口(通常是4999)。
在客户端,连接到服务端的IP地址和端口。
(2)接收请求:

服务端接收客户端发送的SLMP 3E帧请求。
解析SLMP 3E帧,获取命令码、子命令码、数据等信息。
(3)处理请求:

根据SLMP 3E帧中的命令码和数据,执行相应的操作,如读取、写入、控制等。
处理请求可能涉及到对PLC或其他设备进行读写操作,具体实现根据设备和应用需求而定。
(4)生成响应:

根据请求处理的结果,生成SLMP 3E帧的响应数据。
设置响应帧的命令码、子命令码、数据等。
(5)发送响应:

将生成的SLMP 3E帧响应数据发送回客户端。
(6)错误处理:

在处理请求和生成响应的过程中,可能出现各种错误,如无效命令、数据不合法等。
针对不同的错误情况,生成相应的错误响应帧。
(7)关闭连接:

当通信结束或出现错误时,关闭TCP连接,释放资源。
请注意,SLMP 3E帧协议具体的实现步骤和数据格式可能因具体设备和应用而有所不同。在实际开发中,需要参考设备文档和SLMP协议规范来进行具体的实现。

2、SLMP协议规范

查阅三菱PLC关于SLMP的文档!

SLMP的详细规范通常由设备厂商提供,以便开发者能够正确地使用该协议与设备进行通信。规范文件通常包含SLMP协议的命令码、数据格式、通信流程、错误处理等方面的详细信息。

3、为什么会有SLMP协议

SLMP(Seamless Message Protocol)协议是为了在自动化领域(例如工业自动化、制造业、机器人技术等)中实现设备之间的无缝通信而设计的。它提供了一种标准化的通信协议,使不同厂商、不同类型的设备能够在同一个网络上进行通信,实现设备的互联互通。

在现代工业自动化系统中,通常涉及到各种各样的设备,这些设备由不同的厂商制造,可能使用不同的通信协议和数据格式。为了实现这些设备之间的互联互通,需要一种通用的、标准化的通信协议。SLMP就是为了满足这种需求而被开发出来的。

SLMP协议的设计目标包括:

标准化通信: 提供一种通用的通信协议,使得不同厂商生产的设备可以在同一个网络上进行通信。

灵活性: 允许不同类型的数据(例如状态信息、控制命令等)通过同一个协议进行传输。

高效性: 设计为高效的通信协议,以满足工业自动化系统对实时性和响应速度的要求。

易用性: 设计为易于实现和配置,使得工程师能够方便地将SLMP协议集成到他们的设备和系统中。

综上所述,SLMP协议的存在使得不同类型的自动化设备能够方便地相互通信,实现了工业自动化系统的互操作性和灵活性。

4、我是否可以自定义一套协议

你完全可以自定义一套通信协议,以满足特定需求或者应用场景。自定义通信协议通常涉及到以下几个步骤:

确定通信需求: 首先,你需要明确通信双方之间需要传输哪些数据,以及数据的格式和类型。确定通信的数据结构、命令类型、错误处理机制等。

选择传输方式: 确定通信采用的传输方式,可以是基于串口的通信(例如RS-232、RS-485)、基于网络的通信(例如TCP/IP、UDP)、无线通信(例如Wi-Fi、蓝牙)等。

制定协议规范: 定义协议的数据帧格式,包括帧头、帧尾、校验码等信息。确保通信双方遵循相同的协议规范。

实现协议解析和封装: 在通信的发送端和接收端分别实现协议的封装和解析逻辑。封装就是将待发送的数据按照协议格式组织成数据帧,解析则是在接收端将接收到的数据帧按照协议格式解析成可处理的数据。

添加错误处理和安全性: 考虑数据传输中可能出现的错误情况,设计相应的错误处理机制,确保数据的完整性和可靠性。如果通信需要保密性,可以考虑加密通信数据。

测试和验证: 在实际环境中进行测试和验证,确保自定义协议能够正常工作,并且满足通信需求。

请注意,自定义协议需要考虑通信的稳定性、可靠性和安全性。在设计过程中,建议参考现有通信协议的设计经验,以及相关领域的最佳实践。同时,文档化自定义协议的规范,以便未来的维护和扩展。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/118096.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Kotlin 进阶函数式编程技巧

Kotlin 进阶函数式编程技巧 Kotlin 简介 软件开发环境不断变化,要求开发人员不仅适应,更要进化。Kotlin 以其简洁的语法和强大的功能迅速成为许多人进化过程中的信赖伙伴。虽然 Kotlin 的初始吸引力可能是它的简洁语法和与 Java 的互操作性&#xff0c…

每天一点python——day61

#第61天 #字符串的驻留机制字符串:python中基本数据类型,是一个不可变的序列【目前我们学了两个:元组、字符串】 可以使用单引号,双引号,三引号来定义#定义字符串 apython#用单引号,双引号,三引…

Redis原理到常用语法基础图文讲解

在初期,已经讲述了Redis安装问题。现在正式进入Redis的入门阶段 系统架构的演进 传统单机架构 一台机器运行应用程序、数据库服务器 现在大部分公司的产品都是这种单机架构。因为现在计算机硬件发展速度很快,哪怕只有一台主机,性能也很高…

智慧城市排水系统,管网水位监测仪怎么监测

地下排水管网应用于城市的多个环境之中,比如排放雨水,污水或者是地表水等,总之是在维护城市的安全运行,并且保护城市地下生命线处于正常状态。但是一旦排水系统面对各种极端天气,便有可能会突发安全事故,导…

如何再kali中下载iwebsec靶场

这个靶场有三种搭建方法: 第一种是在线靶场:http://www.iwebsec.com:81/ 第二种是虚拟机版本的,直接下载到本地搭建 官网地址下载:http://www.iwebsec.com/ 而第三种就是利用docker搭建这个靶场,我这里是用kali进行…

部署kubevirt教程

前提条件 已安装:kubernetes集群、kubectl、docker apt install -y qemu-kvm libvirt virt-install bridge-utils 【所有节点全部安装】 virt-host-validate qemu部署kubevirt 下载kubevirt-cr.yaml和kubevirt-operator.yaml 先执行: Kubectl apply …

网络安全(黑客)-0基础小白自学

1.网络安全是什么 网络安全可以基于攻击和防御视角来分类,我们经常听到的 “红队”、“渗透测试” 等就是研究攻击技术,而“蓝队”、“安全运营”、“安全运维”则研究防御技术。 2.网络安全市场 一、是市场需求量高; 二、则是发展相对成熟…

Go Gin中间件

Gin是一个用Go语言编写的Web框架,它提供了一种简单的方式来创建HTTP路由和处理HTTP请求。中间件是Gin框架中的一个重要概念,它可以用来处理HTTP请求和响应,或者在处理请求之前和之后执行一些操作。 以下是关于Gin中间件开发的一些基本信息&am…

数据库系统原理与实践 笔记 #7

文章目录 数据库系统原理与实践 笔记 #7数据库设计和E-R模型(续)转换为关系模式具有简单属性的实体集的表示复合属性多值属性联系集的表示模式的冗余—合并 实体-联系设计问题设计问题联系属性的布局 扩展的E-R特性特化概化属性继承特化/概化的设计约束聚集E-R图表示方法总结E-…

聚会娱乐喝酒游戏小程序源码系统 可开流量主 带完整的搭建教程

今天罗峰来给大家分享一款聚会娱乐喝酒游戏小程序源码系统 。在聚会娱乐活动中,喝酒游戏是一种非常受欢迎的活动方式。但是,往往由于缺乏有效的组织和规则,导致游戏的进行不够顺畅,甚至出现混乱的情况。因此,开发一款能…

Java-Hbase介绍

1.1. 概念 base 是分布式、面向列的开源数据库(其实准确的说是面向列族)。HDFS 为 Hbase 提供可靠的 底层数据存储服务,MapReduce 为 Hbase 提供高性能的计算能力,Zookeeper 为 Hbase 提供 稳定服务和 Failover 机制&#xff0c…

云计算实战项目之---学之思在线考试系统

简介: 学之思开源考试系统是一款 java vue 的前后端分离的考试系统。主要优点是开发、部署简单快捷、界面设计友好、代码结构清晰。支持web端和微信小程序,能覆盖到pc机和手机等设备。 支持多种部署方式:集成部署、前后端分离部署、docker部…

一文深入了解 CPU 的型号、代际架构与微架构

在 10 月 16 号的时候,Intel 正式发布了第 14 代的酷睿处理器。但还有很多同学看不懂这种发布会上发布的各种 CPU 参数。借着这个时机,给大家深入地讲讲 CPU 的型号规则、代际架构与微架构方面的知识。 CPU 在整个计算机硬件中、技术体系中都算是最最重…

Wpf 使用 Prism 实战开发Day04

一.菜单导航实现 1.首先创建出所有的页面(View)及对应的页面逻辑处理类(ViewModel) IndexView(首页)-----------------IndexViewModelToDoView(待办事项)------------ToDoViewModelMemoView(忘备录)--------------MemoViewModelSettingsView(设置)--------------SettingsViewMo…

防逆流系统中防逆流电表的正确安装位置-安科瑞黄安南

随着光伏行业的发展,部分地区村级变压器及工业用电变压器容量与光伏项目的装机容量处于饱和。电网公司要求对后建的光伏并网系统为不可逆流发电系统,指光伏并网系统所发生的电由本地负载消耗,多余的电不允许通过低压配电变压器向上级电网逆向…

【漏洞复现】Apache_HTTP_2.4.50_路径穿越漏洞(CVE-2021-42013)

感谢互联网提供分享知识与智慧,在法治的社会里,请遵守有关法律法规 文章目录 1.1、漏洞描述1.2、漏洞等级1.3、影响版本1.4、漏洞复现1、基础环境2、漏洞扫描3、漏洞验证方式一 curl方式二 bp抓捕 1.5、修复建议 说明内容漏洞编号CVE-2021-42013漏洞名称…

【GEE】6、在 Google 地球引擎中构建各种遥感指数

1简介 在本模块中,我们将讨论以下概念: 如何在 GEE 中重命名图像的波段。如何使用已有的遥感指数。如何使用波段数学生成自己的遥感指数。 一个田地已经灌溉的年数的卫星图像。灌溉水最可能的来源是奥加拉拉含水层。图片来自科罗拉多州霍利奥克附近。资料…

设计模式之迭代器模式

什么是迭代器模式 迭代器模式(Iterator pattern)是一种对象行为型设计模式,它提供了一种方法来顺序访问聚合对象中的元素,而又不暴露该对象的内部表示,同时也可以将迭代逻辑与聚合对象的实现分离,增强了代码…

信驰达RF-DG-52PAS CC2652P Zigbee 3.0 USB Dongle烧录指南

一、使用前准备 RF-DG-52PAS是信驰达科技基于美国 TI CC2652P和CP2102为核心设计的Zigbee 3.0 USB Dongle,可烧录 Z-Stack 3.x.0协调器固件,可以直接连接到计算机或树莓派,通过ZHA或 Zigbee2MQTT连接到 Home Assistant或其他开源物联网平台。还可以烧录…

uniapp小程序刮刮乐抽奖

使用canvas画布画出刮刮乐要被刮的图片&#xff0c;使用移动清除画布。 当前代码封装为刮刮乐的组件&#xff1b; vue代码&#xff1a; <template><view class"page" v-if"merchantInfo.cdn_static"><image class"bg" :src&q…