You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

382 lines
16 KiB
Python

2 years ago
import collections
2 years ago
import json
from utils.DBModels.DeviceModels import DeviceDB
2 years ago
from model.ProjectModel.AreaManage import Area
import numpy as np
2 years ago
from protocol.ModBus.ByteOrder import *
2 years ago
from protocol.ModBus.TCPMaster import *
2 years ago
import struct
2 years ago
2 years ago
#从站 "AI" "DI"可强制
#主站 "AO" "DO"可强制
2 years ago
class Device():
2 years ago
inputStartAddress = 0
inputEndAddress = 0
outputStartAddress = 0
outputEndAddress = 0
2 years ago
protocolType = None
masterOrSlave = None
2 years ago
deviceName = None
2 years ago
def __init__(self):
2 years ago
self.inputAreas = []
self.outputAreas = []
2 years ago
self.indexDict = collections.OrderedDict()
# 有序字典 键:总区域索引 值: [类型(输入 : 0, 或输出 : 1), 该类型的第几个区域索引]
2 years ago
2 years ago
def addArea(self, type, nums, bytes, order = 'ABCD'):
area = Area()
bytes = int(bytes)
area.type = type
area.order = order
2 years ago
area.bytes = bytes
area.length = self.getLength(nums, bytes)
2 years ago
area.nums = nums
2 years ago
area.currentValue = [0] * nums
2 years ago
if type in ["AI", "DI"]:
2 years ago
area.startAddress = 0 if not self.inputEndAddress else self.inputEndAddress
area.endAddress = area.startAddress + area.length
# print(area.startAddress, area.endAddress)
2 years ago
self.inputEndAddress = area.endAddress
area.addressList = np.arange(area.startAddress, area.endAddress + 1, area.bytes).tolist()
2 years ago
self.indexDict[len(self.indexDict.values())] = len(self.inputAreas)
2 years ago
self.inputAreas.append(area)
elif type in ["DO" , "AO"]:
2 years ago
area.startAddress = 0 if not self.outputEndAddress else self.outputEndAddress
area.endAddress = area.startAddress + area.length
2 years ago
self.outputEndAddress = area.endAddress
area.addressList = np.arange(area.startAddress, area.endAddress + 1, area.bytes).tolist()
2 years ago
self.indexDict[len(self.indexDict.values())] = len(self.outputAreas)
2 years ago
self.outputAreas.append(area)
2 years ago
# print(id(self), self.outputAreas)
2 years ago
# print(area.addressList, area.startAddress, area.endAddress, self.outputAreas)
2 years ago
def delArea(self, index, type):
if type in ["DI", "AI"]:
2 years ago
self.inputAreas.pop(self.indexDict[index])
2 years ago
elif type in ["AO", "DO"]:
2 years ago
self.outputAreas.pop(self.indexDict[index])
2 years ago
# self.recalculateAddress()
2 years ago
2 years ago
def recalculateAddress(self):
for inputOrOutput, areas in enumerate([self.inputAreas, self.outputAreas]):
2 years ago
endAddress = 0
2 years ago
for index, area in enumerate(areas):
if index == 0 and inputOrOutput == 0:
area.startAddress = self.inputStartAddress
elif index == 0 and inputOrOutput == 1:
area.startAddress = self.outputStartAddress
else:
2 years ago
area.startAddress = areas[index - 1].endAddress
2 years ago
area.endAddress = area.startAddress + area.length
area.addressList = np.arange(area.startAddress, area.endAddress + 1, area.bytes).tolist()
2 years ago
# print(area.addressList, area.startAddress, area.endAddress, self.deviceName)
2 years ago
endAddress = area.endAddress
else:
if inputOrOutput == 0:
self.inputEndAddress = endAddress
elif inputOrOutput == 1:
self.outputEndAddress = endAddress
2 years ago
# endAddress = 0
# print(self.deviceName, self.inputEndAddress)
2 years ago
2 years ago
def editArea(self, index, type, order, bytes):
2 years ago
if type in ["DI", "AI"]:
2 years ago
self.inputAreas[self.indexDict[index]].order = order
self.inputAreas[self.indexDict[index]].bytes = bytes
2 years ago
elif type in ["AO", "DO"]:
2 years ago
self.outputAreas[self.indexDict[index]].order = order
self.outputAreas[self.indexDict[index]].bytes = bytes
2 years ago
# self.recalculateAddress()
2 years ago
2 years ago
def getAreaValues(self, index):
2 years ago
if self.masterOrSlave == "从站":
2 years ago
return self.outputAreas[index].currentValue
else:
2 years ago
# print(self.inputAreas, index)
2 years ago
return self.inputAreas[index].currentValue
2 years ago
def getLength(self, nums, bytes):
2 years ago
length = int(nums) * int(bytes)
2 years ago
# length = length / 2
2 years ago
return length
2 years ago
2 years ago
2 years ago
2 years ago
@classmethod
2 years ago
def delAreas(self, deviceName, id):
2 years ago
jsonCon = json.loads(DeviceDB.getByName(deviceName=deviceName).areaJson)
2 years ago
jsonCon.pop(id)
2 years ago
if jsonCon == []:
areaJson = None
DeviceDB.update(areaJson=areaJson).where(DeviceDB.deviceName == deviceName).execute()
else:
2 years ago
for index, areajsonId in enumerate(jsonCon):
areajsonId["id"] = index + 1
2 years ago
areaJson = json.dumps(jsonCon)
2 years ago
2 years ago
DeviceDB.update(areaJson=areaJson).where(DeviceDB.deviceName == deviceName).execute()
2 years ago
@classmethod
2 years ago
def getAreaJson(self, deviceNames):
deviceName = deviceNames
jsonConsStr = DeviceDB.getByName(deviceName=deviceName).areaJson
if jsonConsStr is None:
return
else:
jsonCons = json.loads(jsonConsStr)
return jsonCons
2 years ago
2 years ago
class DevicesManange():
def __init__(self):
2 years ago
self.dpMasterDevices = collections.OrderedDict()
self.dpSlaveDevices = collections.OrderedDict() # 有序字典 (OrderedDict)
self.paMasterDevices = collections.OrderedDict()
self.paSlaveDevices = collections.OrderedDict()
2 years ago
self.dpSlaveModbus = TcpMaster(host = '192.168.2.10', port = 502)
self.paSlaveModbus = TcpMaster(host = '192.168.4.10', port = 502)
2 years ago
self.dpMasterModbus = TcpMaster(host = '192.168.0.40', port = 502)
2 years ago
self.paMasterModbus = TcpMaster(host = '192.168.3.10', port = 502)
2 years ago
2 years ago
2 years ago
def addDevice(self, proType, masterSlaveModel, deviceName):
2 years ago
device = Device()
device.type = proType
2 years ago
device.masterOrSlave = masterSlaveModel
2 years ago
device.deviceName = deviceName
if proType == "DP" and masterSlaveModel == "主站":
curProDict = self.dpMasterDevices
elif proType == "DP" and masterSlaveModel == "从站":
curProDict = self.dpSlaveDevices
elif proType == "PA" and masterSlaveModel == "主站":
curProDict = self.paMasterDevices
elif proType == "PA" and masterSlaveModel == "从站":
curProDict = self.paSlaveDevices
if len(curProDict) == 0:
2 years ago
device.inputStartAddress = 0
device.outputStartAddress = 0
2 years ago
else:
2 years ago
device.inputStartAddress = list(curProDict.values())[-1].inputEndAddress
device.outputStartAddress = list(curProDict.values())[-1].outputEndAddress
2 years ago
curProDict[deviceName] = device
2 years ago
2 years ago
def initDevices(self):
pass
2 years ago
2 years ago
def delDevice(self, deviceName):
2 years ago
for devicesDict in [self.paMasterDevices, self.paSlaveDevices, self.dpMasterDevices, self.dpSlaveDevices]:
if deviceName in devicesDict:
2 years ago
del devicesDict[deviceName]
2 years ago
self.recalculateAddress()
2 years ago
def recalculateAddress(self):
for devicesDict in [self.paMasterDevices, self.paSlaveDevices, self.dpMasterDevices, self.dpSlaveDevices]:
2 years ago
# print(len(devicesDict))
2 years ago
for index, (deviceName, device) in enumerate(devicesDict.items()):
2 years ago
if index == 0:
device.inputStartAddress = 0
device.outputStartAddress = 0
else:
inputAddress = list(devicesDict.values())[index - 1].inputEndAddress
outputAddress = list(devicesDict.values())[index - 1].outputEndAddress
2 years ago
device.inputStartAddress = 0 if inputAddress == 0 else inputAddress
device.outputStartAddress = 0 if outputAddress == 0 else outputAddress
2 years ago
# print(device.inputStartAddress, device.inputEndAddress, deviceName, '输入')
# print(device.outputStartAddress, device.outputEndAddress, deviceName, 'shuchu')
2 years ago
device.recalculateAddress()
2 years ago
# previousDevice = device
2 years ago
def getDevice(self, deviceName):
for devicesDict in [self.paMasterDevices, self.paSlaveDevices, self.dpMasterDevices, self.dpSlaveDevices]:
if deviceName in devicesDict:
return devicesDict[deviceName]
2 years ago
2 years ago
def writeAreas(self, deviceName, values):
2 years ago
# print(values)
2 years ago
bytes = b""
device = self.getDevice(deviceName)
2 years ago
# print(device, deviceName)
2 years ago
if device.type == "DP" and device.masterOrSlave == "主站":
curProDict = self.dpMasterDevices
areas = device.outputAreas
2 years ago
modbusM = self.dpMasterModbus
2 years ago
elif device.type == "DP" and device.masterOrSlave == "从站":
curProDict = self.dpSlaveDevices
areas = device.inputAreas
2 years ago
modbusM = self.dpSlaveModbus
2 years ago
elif device.type == "PA" and device.masterOrSlave == "主站":
areas = device.outputAreas
curProDict = self.paMasterDevices
2 years ago
modbusM = self.paMasterModbus
2 years ago
elif device.type == "PA" and device.masterOrSlave == "从站":
areas = device.inputAreas
curProDict = self.paSlaveDevices
2 years ago
modbusM = self.paSlaveModbus
2 years ago
# print(values)
2 years ago
for area, value in zip(areas, values):
area.forceValue = value
2 years ago
if area.type in ['DO', 'DI'] and device.type == 'PA':
area.forceValue = value[8:] + value[:8]
elif area.type in ['DO', 'DI'] and device.type == 'DP' and device.masterOrSlave == "主站":
2 years ago
area.forceValue = value[8:] + value[:8]
2 years ago
for device in curProDict.values():
forceAreas = device.outputAreas if device.masterOrSlave == "主站" else device.inputAreas
for area in forceAreas:
2 years ago
# print(area.forceValue)
2 years ago
if area.type in ["AI", "AO"]:
2 years ago
# print(area.forceValue)
2 years ago
byte = floatToBytes(area.forceValue, area.bytes, order = area.order)
2 years ago
elif area.type in ["DI", "DO"]:
2 years ago
# if device.type == "PA" and device.masterOrSlave == "从站" and len(area.forceValue) == 16:
2 years ago
byte = coilsToBytes([int(x) for x in area.forceValue], area.bytes)
bytes += byte
else:
if len(bytes) % 2 != 0:
bytes += struct.pack('B', 0)
2 years ago
# print(bytes)
2 years ago
# print(bytes)
2 years ago
values = struct.unpack('!' + 'H' * int(len(bytes) / 2), bytes)
modbusM.writeMultipleRegister(slaveId = 1, address = 0, outputValue = values)
# print(struct.unpack('>f', struct.pack('!HH', *values[:2])))
2 years ago
2 years ago
def readAreas(self):
2 years ago
for index, curProDict in enumerate([self.paMasterDevices, self.paSlaveDevices, self.dpMasterDevices, self.dpSlaveDevices]):
2 years ago
match index:
case 0:
2 years ago
areaType = 'input'
2 years ago
modbusM = self.paMasterModbus
case 1:
areaType = 'output'
2 years ago
modbusM = self.paSlaveModbus
case 2:
areaType = 'input'
2 years ago
modbusM = self.dpMasterModbus
case 3:
2 years ago
areaType = 'output'
modbusM = self.dpSlaveModbus
if len(list(curProDict.values())) == 0:
continue
inputEndAddress = max([x.inputEndAddress for x in list(curProDict.values())])
outputEndAddress = max([x.outputEndAddress for x in list(curProDict.values())])
2 years ago
inputAreaNums = sum([len(x.inputAreas) for x in list(curProDict.values())]) - 1
outputAreaNums = sum([len(x.outputAreas) for x in list(curProDict.values())]) - 1
2 years ago
bytesNums = inputEndAddress if areaType == 'input' else outputEndAddress
2 years ago
intNums = int(bytesNums / 2) if bytesNums % 2 == 0 else int(bytesNums / 2) + 1
2 years ago
if bytesNums == 0:
continue
intValues = modbusM.readHoldingRegisters(slaveId = 1, startAddress = 0, varNums = intNums)
bytesValues = struct.pack(f"!{'H' * len(intValues)}", *intValues)
2 years ago
# print(bytesValues, intNums, bytesNums)
# print(bytes)
2 years ago
for device in curProDict.values():
readAreas = device.inputAreas if areaType == 'input' else device.outputAreas
for area in readAreas:
2 years ago
# print(area.startAddress, area.endAddress)
2 years ago
if area.startAddress == 0:
bytes = bytesValues[area.startAddress:area.endAddress]
else:
2 years ago
bytes = bytesValues[area.startAddress:area.endAddress]
# print(area.startAddress, area.endAddress)
2 years ago
if area.type in ['AI', 'AO']:
for i in range(0, len(bytes), 4):
2 years ago
byte = bytes[i:i+4]
2 years ago
if len(byte) != 4:
2 years ago
continue
2 years ago
# print(round(struct.unpack('!f', reorderBytes(byte, area.order))[0], 4))
2 years ago
area.currentValue[i] = round(struct.unpack('!f', reorderBytes(byte, area.order))[0], 4)
2 years ago
# print(round(struct.unpack('!f', reorderBytes(byte, area.order))[0], 4))
2 years ago
elif area.type in ['DI', 'DO']:
2 years ago
if device.masterOrSlave == '主站' and device.type == 'DP':
bytes = bytes
else:
bytes = bytes[::-1]
2 years ago
area.currentValue = bytesToCoils(bytes)
# print(area.currentValue)
2 years ago
@classmethod
2 years ago
def addAreas(self, type, order, bytes, deviceName):
2 years ago
if DeviceDB.getByName(deviceName=deviceName).areaJson is None:
jsonCon = ([{
"id": 1,
"type": type,
2 years ago
"order": order,
2 years ago
"bytes": bytes,
2 years ago
2 years ago
}])
else:
jsonCon = json.loads(DeviceDB.getByName(deviceName=deviceName).areaJson)
id = jsonCon[-1]["id"] + 1
jsonCon.append({
"id": id,
"type": type,
2 years ago
"order": order,
2 years ago
"bytes": bytes,
})
areaJson = json.dumps(jsonCon)
DeviceDB.update(areaJson=areaJson).where(DeviceDB.deviceName == deviceName).execute()
2 years ago
2 years ago
@classmethod
def updataAreas(self, type, order, bytes, deviceName, index):
if DeviceDB.getByName(deviceName=deviceName) is None:
return False
else:
jsonCon = json.loads(DeviceDB.getByName(deviceName=deviceName).areaJson)
for area in jsonCon:
if index == area["id"]:
area["type"] = type
area["order"] = order
area["bytes"] = bytes
areaJson = json.dumps(jsonCon)
DeviceDB.update(areaJson=areaJson).where(DeviceDB.deviceName == deviceName).execute()
2 years ago
@classmethod
def getAreaID(self, deviceNames):
deviceName = deviceNames
jsonConsStr = DeviceDB.getByName(deviceName=deviceName).areaJson
if jsonConsStr is None:
2 years ago
return
else:
2 years ago
jsonCons = json.loads(jsonConsStr)
id = []
for jsonCon in jsonCons:
id.append(jsonCon["id"])
return id
2 years ago
2 years ago
@classmethod
def getChannelLength(self, deviceName):
number = 0
if DeviceDB.getByName(deviceName=deviceName).areaJson is None:
return number
else:
numbers = json.loads(DeviceDB.getByName(deviceName=deviceName).areaJson)
for i in numbers:
number += int(i['nums'])
return number
2 years ago
def editDevies(self):
pass
2 years ago
@classmethod
2 years ago
def getAllDevice(self):
# 查询所有设备
devices = DeviceDB.get_all()
if devices is 'error':
return
l = []
for x in devices:
l.append([x.deviceName, x.proType, x.masterSlaveModel, x.areaJson])
2 years ago
return l
2 years ago