S7SvrSim 是一个西门子 S7
系列的PLC
通讯模拟器。
之所以有这样一个小项目,主要是为了缓解上位机开发人员想随时测试和S7
系列PLC通讯的尴尬:
- 没有特定型号的物理
PLC
;或者有相关型号的PLC
,却没带在手边 - 没钱购买博图,又不屑于使用盗版
- 有钱购买博图,但是这个软件超大,还会自动安装了一堆WinCC/SQL Server之类服务。偏偏自己对电脑还有洁癖。
- 有钱买了博图,然后废了老大劲安装好了,却不一定有权限拿到电气部门的
PLC
程序。 - 拿到了电气部门的程序,发现他们的程序还没写好,跑不起来
- ...
回过头来想,上位机开发人员真的需要物理PLC或者博图吗?我们只需要一个西门子 PLC
S7通讯模拟器而已!我们甚至不需要模拟完整的PLC
程序,我们仅仅只需要根据和电气部门的通讯规约操作点表!对,只需要模拟通讯。
当然,如果上面所列的场景不适合你,那么你没必要用这个程序 :)
我不打算、也没能力做一个完整的软PLC
。像画各种梯形图、执行ST程序这种事,压根不是这个项目的目标。
由于前端是使用 WPF
开发的,故目前只能在Windows上使用(后期可能会考虑做成跨平台的,但是暂时没有这个动力)。
使用该工具最简单的方法是使用ClickOnce
技术来安装,只需打开如下URL
即可(需要Windows操作系统):
https://assets.chiyiqian.net/S7SvrSim/S7SvrSim.application
(有个别同事使用最新的Win11
打开ClickOnce
有问题,所以不再提供ClickOnce方式安装)
请到本项目的Release下,直接下载对应的zip包即可。
启动程序后,在 DB配置 界面中:
- 设置好要监听的IP地址,
- 配置一个或者多个
DB
的号码、大小 - 点击 "启动" 按钮
- 本质上,这是S7 Server通信模拟器,可以简单把它当成采用S7通信时上位机所连的PLC。
- 提供DB、MB操作接口。
S7SvcSim 支持单步操作和基于Python
脚本的批量操作。其中,单步操作主要是为了方便手动调试。
更多的时候,为了减少重复工作,我们会编写一些Python
脚本来实现自动化测试。
# -*- coding: UTF-8 -*-
import time
import shell
DB_INDEX= 200
OFFSET_FLAGS_BASE = 1082
SIZEOF_DEV_BELT_MSG = 56
FUNC = shell.accept_input_int("请选择皮带线: 1-belt1; 2-belt2; 3-belt3; 4-belt4")
if FUNC < 1 and FUNC > 4 :
shell.show_message_box("请选择正确的皮带线!")
else:
# 计算地址偏移
OFFSET_FLAGS = OFFSET_FLAGS_BASE + (FUNC - 1) * SIZEOF_DEV_BELT_MSG
OFFSET_VEC_NUMBER = OFFSET_FLAGS + 1
# 让用户输入来料信息
vector_num = shell.accept_input_int("incoming vector number")
DB.WriteByte(DB_INDEX, OFFSET_VEC_NUMBER , vector_num)
DB.WriteBit(DB_INDEX, OFFSET_FLAGS, 0, True)
# 3s后清除信号
time.sleep(3)
DB.WriteByte(DB_INDEX, OFFSET_VEC_NUMBER , 0)
DB.WriteBit(DB_INDEX, OFFSET_FLAGS, 0, False)
注:这里的
shell
是用户自定义的模块。
有了这些测试脚本,我们只需要在合适的时间点导入这些脚本即可自动完成动作模拟。(更多示例参考 docs/sample scripts
文件夹)
底层用的是IronPython
,由于到目前为之,IronPython 3.x
尚未成熟,故这里IronPython
的版本是2.7.11
。
这意味着:
只能使用 Python2.x 的语法和特性如果要写中文,应该指定文件编码# -*- coding: UTF-8 -*-
从0.4.0开始,使用IronPython 3.4.1
然后通过在PyEngine/SearchPaths
界面中,点击 "选择路径",来导入自定义的检索路径:
点击 "DB配置" 里的 停止、"启动" 会清空当前模拟器中的数据。
用户可以自定义一个Python
脚本,然后在程序运行后导入。一旦导入,该脚本就会被立刻执行。
为了操作模拟器,我向Python
暴露了一个预定义的S7
对象(类型为IS7ServerService
),用于对当前正在运行的 S7 PLC模拟器 进行操作。
通过这些API方法,我们可以通过编写Python
脚本来动态执行一系列操作,比如:
DB.WriteString(200, 2000, 40, "abcdefghijklmn")
DB.WriteBit(200, 2060, 0, True)
DB.WriteShort(200, 2062, 1)
Python可以使用的DB
其实是一个IS7DataBlockService
接口对象:
public interface IS7DataBlockService
{
bool ReadBit(int dbNumber, int offset, byte bit);
void WriteBit(int dbNumber, int offset, byte bit, bool flag);
byte ReadByte(int dbNumber, int pos);
void WriteByte(int dbNumber, int pos, byte value);
short ReadShort(int dbNumber, int pos);
void WriteShort(int dbNumber, int pos, short value);
uint ReadUInt32(int dbNumber, int pos);
void WriteUInt32(int dbNumber, int pos, uint value);
ulong ReadULong(int dbNumber, int pos);
void WriteULong(int dbNumber, int pos, ulong value);
float ReadReal(int dbNumber, int pos);
void WriteReal(int dbNumber, int pos, float real);
double ReadLReal(int dbNumber, int pos);
void WriteLReal(int dbNumber, int pos, double real);
string ReadString(int dbNumber, int offset);
void WriteString(int dbNumber, int offset, int maxlen, string str);
}
而MB
则是一个IS7MBService
:
public interface IS7MBService
{
bool ReadBit(int offset, byte bit);
void WriteBit(int offset, byte bit, bool flag);
byte ReadByte(int pos);
void WriteByte(int pos, byte value);
short ReadShort(int pos);
void WriteShort(int pos, short value);
uint ReadUInt32(int pos);
void WriteUInt32(int pos, uint value);
ulong ReadULong(int pos);
void WriteULong(int pos, ulong value);
float ReadReal(int pos);
void WriteReal(int pos, float real);
double ReadLReal(int pos);
void WriteLReal(int pos, double real);
string ReadString(int offset);
void WriteString(int offset, int maxlen, string str);
}
还暴露一个Logger
对象,用于追加日志:
void LogInfo(string content);
void LogError(string content);
-
由于#44602 这个问题,目前没法使用。2021-01-16:#44602 号问题已在 VS16.8.4 中修复,参见VS Developer CommunityClickOnce
发布 -
目前由于IronPython
的限制,发布时无法生成单个文件。参见IronPython #762。故发布项目时请不要使用-p:PublishSingleFile=true
。