Данная статья написана для начинающих, тех кто на начальном уровне знает Python и немного разбирается в АСУ ТП. Задача достаточно распространенная, надо взять данные со старого, со своей спецификой оборудования и перевести ее в такой вид, что бы ее можно было легко достать (MQTT сервер) и обрабатывать (SCADA или любое ПО, которое умеет работать с MQTT).

Зачем это вообще нужно? да просто ходить до весоизмерительного модуля далеко, а контролировать его надо, да и необходима база в которой будут записаны его показания, такие как общее количество кг взвешенного им и общее количество нафасованных мешков и про хотелки производственников не забываем)

Мы имеем:

  • "Прибор весоизмерительный ЦЕНТА АД-К", связь с внешним миром через RS-485 с специфическим протоколом верхнего уровня немного похожий на MODBUS (Используется символьный протокол, с ASCII-HEX представлением данных. Числовые данные передаются в виде последовательности ASCII символов шестнадцатеричной записи числа, с выравниванием до чётного числа знакомест, так пишет разработчик данного прибора)

  • Одноплатный компьютер Raspberry Pi, в особом представлении не нуждается

  • Сервер, любой ПК с характеристиками, как минимум офисного ПК начального уровня, главное, побольше ОЗУ

Сама схема простая:
По RS-485 весоизмерительного прибора берем данные,
Переводим в читаемый вид,
Отправляем на MQTT сервер.
Эх, если бы все было так легко и непринужденно... но так не бывает, нас ожидают куча подводных камней, ограничения и т.д., все как обычно)

Что бы взять данные с весоизмерительного модуля надо понять, как он вообще общается с внешним миром, хорошо что по запросу в http://www.centa.ru/about.htm нам предоставили ПО для снятия данных и описание работы протокола с перечнем запросов. По хорошему, можно было обойтись и этой программой, но для этого пришлось бы тянуть кабель для подключения, сама ПО неудобная, связь с БД ограничена. Другой выход, это преобразовать сигнал RS-485 и отправлять его по имеющимся Ethernet, но опять же, танцы с бубном с настройками конвертации (помним, что верхний уровень, это своя версия MODBUS протокола), сами блоки конвертации стоят достаточно дорого, а смогут ли они нормально передать сигнал, да и кто на такой достаточно мутный проект (инициатива самого автора, который ранее подобного не делал) даст денег, так что для начала, надо собрать работающий прототип на том что есть в личных запасах

(1) Снифаем COM порт между весовым модулем и ПК

Берем ноут, ставим "Serial Port Monitor" (почему его? Все очень просто, он не занимает COM порт, можно спокойно снифать трафик и не мудрить с кабелями или программно эмулировать кучу COM портов), подключаем преобразователь интерфейса USB to RS-485 (народ, не жалейте денег на нормальный преобразователь, он должен быть с гальванической развязкой а не поделкой за 200 рублей с алика, пожалейте USB порт Вашего ПК или ноута), подключаемся к весоизмерительному прибору, запускаем "Serial Port Monitor" (вводим номер COM порта, далее все настройки скорости, четности и т.д. можно посмотреть в логах данной ПО), запускаем ПО от "centa" для снятия показаний и смотрим что нам показывает "Serial Port Monitor". А показывает он нам корректные запросы с адресом, настройками COM порта и кодировкой используемой в протоколе, далее нам эта информация потребуется для конвертации ответов на запросы к весоизмерительному модулю

(2) Пишем скрипт для переброски данных с весового модуля на "MQTT" сервер

В шаге (1) мы поняли как именно общается между собой ПК и весоизмерительный модуль. Теперь нам надо создать скрипт, так как автор знает немного python, то на нем и будем писать скрипт, так же не забываем, что надо написать и MQTT клиент (шаг (3)) который на сервер будет отправлять нужные нам данные. Еще одна загвостка, все это должно работать на "Raspberry Pi" под ее величием "Raspberry Pi OS" основанным на "Debian", так что все что мы пишем в консоле, обязательно приправляем незаменимой командой "sudo", звучит пафосно, но без "sudo", как и без синей изоленты в электрике, электронике ничего не работает)

Прежде чем дать текст скрипта, нам надо настроить среду, начнем с удаленного рабочего стола, "Xrdp" не хотит нормально работать, требуются танцы с бубном, очень жаль что с новыми ОС она криво работает, если кто то знает как ее заставить работать, напишите пожалуйста. Мы будем использовать "VNC", он уже установлен на "Raspberry Pi", его надо включить ( Menu > Preference > Raspberry configuration > Interfaces > Enable VNC ), а на другой стороне установить "VNC-viewer", она бесплатна. Далее, нам надо установить библиотеку "paho-mqtt", идем на "https://pypi.org/project/paho-mqtt/" смотрим, как это чудо устанавливается и вспоминаем про "sudo", без этого волшебного слова в консоле у меня криво встал "paho-mqtt", он виден в pip но корректно не запускается, по этому устанавливаем так:

sudo pip install paho-mqtt

Далее маленькое отступление, я не программист, только учусь, код не оптимизирован, так как там будет куча правок, добавление функционала, это рабочий прототип, но конструктивная критика приветствуется)

Далее текст скрипта:
import serial
import time
#подгружаем библиотеку для работы с QMTT
import paho.mqtt.publish as publish


#настройки на COM порт
with serial.Serial() as ser:
    ser.baudrate = 19200
    #ser.port = 'COM5'
    ser.port = '/dev/ttyUSB0'
    bytesize=8
    parity='N'
    stopbits=1
    timeout=1
    write_timeout=1


    #готовим запрос (надо hex преобразовать в byte)
values = bytearray([
    int('0x23', 16),    #[0:7] -запрос веса
    int('0x30', 16),
    int('0x31', 16),
    int('0x31', 16),
    int('0x30', 16),
    int('0x34', 16),
    int('0x30', 16),
    0,
    int('0x23', 16),    #[8:15] -запрос колличество мешков
    int('0x30', 16),
    int('0x31', 16),
    int('0x31', 16),
    int('0x30', 16),
    int('0x34', 16),
    int('0x34', 16),
    ])

podpiski_values =["fasovka/error","fasovka/obhii_ves","fasovka/koll_mehkov"]

#--------------------------------------------
#включение COM порт
#через обработчика ошибок
#подпрограмма для отправки сообщений QMTT
def QMTT_read(a,b):
    #дописать сообщения
    try:
        publish.single( a,  payload = b, hostname="192.168.20.213",auth = {'username':"pi", 'password':"1"})#
        #publish.single( a,  payload = b, hostname="192.168.137.1")#
        
    except:
        print('нет связи с сервером QMTT')
#--------------------------------------------
def COM_write(c,d):
    print(values[c:d])
    ser.write(values[c:d] + b'\r\n')
#--------------------------------------------
def COM_read():
    #запрос или эмулятор вывода
    #Далее интересная конструкция:
    #если на прямую читать ser.readline(), то если порт пустой
    #система будет ждать сообщения и дальше скрипт выполняться не будет
    #По этому мы читаем количество байт в буфере порта, если 
    #что то есть, значит весовой модуль что то отправил 
    x = ser.inWaiting() #сколько байт в входном буфере
    print('в буфере - ',x)
    if x > 0: #если в буфере более ноля байт, то можно читать readline, иначе нет
        line = ser.readline()# read a '\n' terminated line

        print(type(line))
        print(line)
        #---line = b'@01(PAR40(\x91\xe7\xa5\xe2\xe7\x8e\xa1\xe9\x82\xa5\xe1\xa01=0x017E34E0))\r\n'
        #настраиваем поиск
        nathalo=line.find(b'=0x')
        konets=line.find(b'))')
        #запихиваем срез
        srez_znathenia = line[nathalo+3:konets]
        print(srez_znathenia)

        #выводим на экран 
        print(int("".join(srez_znathenia.decode('cp866')),16))
        #отправляем QMTT значение
        return int("".join(srez_znathenia.decode('cp866')),16)
        
    else:
        print('во входном буфере 0')
        #отправляем QMTT что 0
        return int(0)

    

#-------------------------------------------- 
#открываем порт
try:
    ser.open()
except serial.SerialException:
    print('ошибка при запуске COM')
    QMTT_read(podpiski_values[0],"no_COM_port")
else: 
    #нам надо отправить 2 запроса
    for number in [0,1]:
    #запрос с переносом строки
        print(number)
        if number == 0:
            COM_write(0,7)
        elif number == 1:
            COM_write(8,15)
        else:
            print('ошибка цикла')
        #ответ же не сразу предет, поспим немного
        time.sleep(1)
        
        #проверяем на наличие соединения
        try:
            COM_otvet = COM_read()
            print(COM_otvet)
        except serial.SerialException:
            print('ошибка при запуске COM')
            QMTT_read(podpiski_values[0],"no_COM_port")
        else:
        #проверяем на наличие байт в порте
            if COM_otvet > 0: #если в буфере более ноля байт, то можно читать readline, иначе нет
                QMTT_read(podpiski_values[1+number],COM_otvet)
                QMTT_read(podpiski_values[0],"---")
            else:
                QMTT_read(podpiski_values[0],"no_values_RS485")
        #порт и так сильно потрудилась, пускай поспит
        time.sleep(1)
    ser.close()  



Корректно укажите ip адрес, настройки и номер COM

Так же в "paho" можно сообщения отправлять не подключаясь и отключаясь для каждого сообщения, а нормально подключится, но для нас это избыточно так как сообщений мало и они будут настроены на работу раз в минуту

Данный скрипт нужно запускать через определенное время, я для себя решил, что запускаться должен 1 раз в 1 минуту, наша цепочка будет такая: мы создадим исполняемый файл "/home/rpi/comand_COM_MQTT_bat.run"
делаем его исполняемым "chmod ugo+x /home/rpi/comand_COM_MQTT_bat.run"
запишем ф файл comand_COM_MQTT_bat.run следующее:
#!/bin/bash
sudo python3 /home/rpi/COM_MQTT.py

Затем мы должны настроить планировщик задач:
crontab -e
в нем добавляем строку:
/1 * * * * /home/rpi/comand_COM_MQTT_bat.run

Как видно из настройки, мы запускаем наш comand_COM_MQTT_bat.run раз в минуту

(3) Теперь нам надо установить и настроить "MQTT" сервер (у меня рабочая среда "Win10", настройки ниже проверенны на ней), устанавливаем "https://mosquitto.org/download/" для Вашей ОС, далее:

Простыня настроек

Для облегчения создаем testMQTT.cmd файл (командный файл) со следующим текстом:

start "QMTT_Server" /d "C:\Program Files\mosquitto\" mosquitto.exe -c "C:\Program Files\mosquitto\mosquitto.conf" -v
TIMEOUT /T 2
start "QMTT_podpiska" /d "C:\Program Files\mosquitto\" mosquitto_sub.exe -h 127.0.0.1 -t fasovka/# -u pi -P 1 --debug

При повторном запуске 1 строчка будет ругаться так как сервер уже запущен, не обращайте внимание, нам нужна 3 строчка, она покажет сообщения для отладки

Данным командным файлом мы в разных процессах запускаем сам сервер (строка 1), после перерыва в 2 секунды (строка 2) запускаем клиента который смотрит все топики fasovka/# поступающие на сервер "MQTT" (строка 3).

Не спешим запускать testMQTT.cmd, сначала настроим "MQTT" сервер

Настройка mosquitto.conf:

pid_file C:\mosquitto\mosquitto.pid
persistence true
persistence_location C:\mosquitto
log_dest file C:\mosquitto\mosquitto.log
#include_dir /etc/mosquitto/conf.d
password_file C:\mosquitto\passwd
listener 1883
persistence_file mosquitto.db
log_dest syslog
log_dest stdout
log_dest topic
log_type error
log_type warning
log_type notice
log_type information
connection_messages true
log_timestamp true
allow_anonymous false

К сожалению я не успел поэкспериментировать с настройками, единственное я точно могу сказать что с данными настройками сервер будет принимать не только локальные сообщения, но и удаленные

Создаем файл пароля:
"C:\Program Files\mosquitto\mosquitto_passwd.exe" -c -b C:\mosquitto\passwd pi 1
логин - pi
пароль - 1

Запускаем ранее созданный testMQTT.cmd файл

Проверяем что все работает, для этого в командной строке вводим:
"C:\Program Files\mosquitto\mosquitto_pub -h localhost -t "fasovka/error" -m "Privet" -u "pi" -P "1"
тем самым мы в топик "fasovka/error" отправили сообщение "Privet"

Более детально про настройки сервера, пароли, настройка mosquitto.conf можно прочитать здесь:
https://mosquitto.org/man/mosquitto-8.html
https://mosquitto.org/man/mosquitto_passwd-1.html
https://mosquitto.org/man/mosquitto-conf-5.html
Понимаю, что документация на английском языке, я сам английский не знаю, но через яндекс или гугл переводчик общий смысл понятен

Заключение

Для меня, как новичка, это реальный опыт воплощения проекта в жизнь, что бы это реализовать я пользовался собственным железом так как не был уверен в успехе, но теперь данный проект проверен и будет расширяться. Дальнейшие планы: подключить "MasterSCADA" к "MQTT" серверу и на ней обрабатывать и отображать информацию в том числе и от других источников (аварии на котельной, вентиляция и т.д.), так же хочу приделать "telegram" бота что бы он мог в чат отправлять оповещения, само собой что бы не плодить обработчиков, всем будет управлять "MasterSCADA".

Комментарии (13)


  1. koreec
    22.01.2023 17:01
    +2

    Serial - to - LAN (Moxa uPort, например) работает абсолютно прозрачно, безо всяких конвертацией и вмешательства в данные. В Windows просто появляется виртуальный порт, ничем не отличимый от железного.

    У меня на одном uPort висит 8 устройств разных вендоров, каждое со своими причудами и разными протоколами, включая "типа MODBUS" вроде вашего. И все замечательно работает.


    1. Useroff
      22.01.2023 21:25
      +1

      To - LAN - это nPort, uPort - это USB. Что касаемо "ничем не отличимый от железного", могут быть нюансы. У нас, например, используются Advantech EKI в связке с железкой, которая отличает адрес от данных по 9-у биту. EKI не умеет в 9-битные посылки (и nPort, кстати, тоже), поэтому используется переключение режимов четности mark/space. При этом смена режима занимает время, достаточное для того, чтобы железка, приняв адрес и не дождавшись данных, отвалилась по тайм-ауту. На "железном" порту, разумеется, все работает нормально.


      1. Sergeant101
        23.01.2023 09:41

        Какие у вас чудные технологии. Прям мастерская по изобретению велосипедов.


        1. Useroff
          23.01.2023 10:06

          Вот такое оно, прикладное айти в непрофильных организациях.


        1. SemLab Автор
          23.01.2023 18:18

          До мастерской мне еще далековато...
          Если Вы утверждаете что в статье велосипед, то как бы Вы решили данную проблему с теми ресурсами и условиями которые описаны в статье не прибегая к велосипедам?
          Я только учусь, и для меня важно выслушать аргументированное мнение сообщества, собственно по этому я и опубликовал статью


      1. IliaIT
        23.01.2023 17:02

        Если чудная железка принимает запрос на адрес 9 бит, а весь остальной обмен идёт по 8битному формату, то это очень странная железка.

        Если же она в принципе работает по 9 битному протоколу (но вначале использует все 9 бит для определения адреса, а потом девятый это бит чётности), то ничто не мешает вам реализовать 8+1(программный бит чётности), а в первом пакете слать адрес, а в остальных добавлять к 8 битам бит чётности, используя всегда 9битный формат.

        ПС. так кстати фирма энергомера работает (электросчётчики), только у них всегда 8 бит = 7+1(программно чётность) для совместимости с другим оборудованием на линии.


        1. Useroff
          23.01.2023 18:01

          Чудная железка всегда использует посылку длиной 9 бит (ну и старт-стоп, естественно). Если 9-й бит в единице, то она считает младшие 8 бит адресом; если в нуле, то младшие биты считаются данными. Железок на шине может быть несколько, отвечать может только та, которая поймала свой адрес.

          Соответственно, "честный" бит чётности к данным добавлять нельзя - могут возбудиться другие железки на линии, посчитав чужие данные своим адресом.

          Ну а поскольку, как я уже говорил, наши адаптеры EKI не умеют в посылки с 9-битными данными, единственный способ программно управлять 9 битом - переключать режим чётности mark/space, что, внезапно, занимает время.


  1. starik-2005
    23.01.2023 10:38

    Сколько я протоколов работы с весами повидал - мульон. Например, такой: https://massa.ru/upload/iblock/ee6/pprotocol-3.doc1.pdf Не знаю, как бы справился с ним автор.


    1. SemLab Автор
      23.01.2023 18:23

      К сожалению... а скорее всего, к счастью, с таким протоколом не встречался, но, думаю алгоритм будет примерно как и в данной статье: снифаем, расшифровываем


      1. starik-2005
        23.01.2023 18:34
        +1

        Встречался у одного из вендоров с протоколом весов с контрольной суммой, для вычисления которой был приведен код на ассемблере. Т.е. без байта с контрольной суммой весы ничего не вернут. Вес закодирован в пакете несколькими байтами.

        Я это все к чему? Обычно, для этого всего есть документация, которую можно (даже нужно) читать. И метод научного тыка - такой себе метод. Да, если документация утеряна, не найдена и найдена быть не может, тогда смысл какой-то в этом всем есть. Но протокол весов очень разный - от просто спама в порт текущей массы у автовесов Тензора до весьма защищенных протоколов, которые, боюсь, только лишь снифером не просечь.


        1. koreec
          24.01.2023 04:35

          Иногда "документация" такая, что без сниффера никак. Пример из нашего опыта - немецкий блок питания для магнетронного напыления. В мануале потокол гордо именуется MODBUS. Внутри ASCII-HEX с контрольной суммой, закодированной по своему алгоритму. В пакете передаются 12 параметров, и целых, и float, и битовые маски. При этом в каком формате закодирован float в мануале написать забыли. Ну хоть алгоритм контрольной суммы описали,и на том спасибо.


          1. starik-2005
            24.01.2023 10:11
            -1

            При этом в каком формате закодирован float в мануале написать забыли.

            Так это вроде в школе проходят, не? https://neerc.ifmo.ru/wiki/index.php?title=Представление_вещественных_чисел


  1. hooperer
    23.01.2023 15:17

    мастерскада с телеграммом может в обе стороны.

    https://www.youtube.com/watch?v=2JnOssfqF6o

    ну или совсем просто

    https://www.youtube.com/watch?v=GWNryHOylDE