DLMS-SPODES-client 0.19.27__py3-none-any.whl → 0.19.29__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- DLMS_SPODES_client/services.py +47 -54
- DLMS_SPODES_client/task.py +37 -0
- {dlms_spodes_client-0.19.27.dist-info → dlms_spodes_client-0.19.29.dist-info}/METADATA +1 -1
- {dlms_spodes_client-0.19.27.dist-info → dlms_spodes_client-0.19.29.dist-info}/RECORD +7 -7
- {dlms_spodes_client-0.19.27.dist-info → dlms_spodes_client-0.19.29.dist-info}/WHEEL +0 -0
- {dlms_spodes_client-0.19.27.dist-info → dlms_spodes_client-0.19.29.dist-info}/entry_points.txt +0 -0
- {dlms_spodes_client-0.19.27.dist-info → dlms_spodes_client-0.19.29.dist-info}/top_level.txt +0 -0
DLMS_SPODES_client/services.py
CHANGED
|
@@ -11,19 +11,19 @@ _h: dict[str, [str]] = dict(settings.from_csv.header_names)
|
|
|
11
11
|
class IpAddress:
|
|
12
12
|
__value: list[int]
|
|
13
13
|
|
|
14
|
-
def __init__(self, value: str =
|
|
14
|
+
def __init__(self, value: str = "127.0.0.1"):
|
|
15
15
|
self.__value = list()
|
|
16
|
-
for el1 in value.split(
|
|
16
|
+
for el1 in value.split("."):
|
|
17
17
|
if el1.isdigit():
|
|
18
18
|
el = int(el1)
|
|
19
19
|
if 0 <= el <= 255:
|
|
20
20
|
self.__value.append(el)
|
|
21
21
|
else:
|
|
22
|
-
raise ValueError(F
|
|
22
|
+
raise ValueError(F"Wrong digit in value: {el}, must be 0..255")
|
|
23
23
|
else:
|
|
24
|
-
raise ValueError(F
|
|
24
|
+
raise ValueError(F"Value is not digit: {el1}")
|
|
25
25
|
if len(self.__value) != 4:
|
|
26
|
-
raise ValueError(F
|
|
26
|
+
raise ValueError(F"Length of Ip address {value} must be 4, got {len(self.__value)}")
|
|
27
27
|
|
|
28
28
|
@classmethod
|
|
29
29
|
def is_valid(cls, value: str) -> bool:
|
|
@@ -34,64 +34,57 @@ class IpAddress:
|
|
|
34
34
|
return False
|
|
35
35
|
|
|
36
36
|
def __str__(self):
|
|
37
|
-
return
|
|
37
|
+
return ".".join(map(str, self.__value))
|
|
38
38
|
|
|
39
39
|
|
|
40
40
|
def get_client_from_csv(
|
|
41
41
|
file_name: str,
|
|
42
|
-
id_factory: IDFactory
|
|
42
|
+
id_factory: IDFactory,
|
|
43
43
|
universal: bool = False
|
|
44
44
|
) -> list[Client]:
|
|
45
45
|
"""file in utf-8 format"""
|
|
46
46
|
da: str
|
|
47
|
-
with open(file_name,
|
|
48
|
-
|
|
49
|
-
dialect = sniffer.sniff(csv_file.readline(1024))
|
|
47
|
+
with open(file_name, "r", encoding="utf-8-sig") as csv_file:
|
|
48
|
+
dialect = csv.Sniffer().sniff(csv_file.readline(1024))
|
|
50
49
|
csv_file.seek(0)
|
|
51
50
|
reader = csv.reader(csv_file, dialect=dialect)
|
|
52
51
|
first_row: list[str] = next(reader)
|
|
53
52
|
if any(map(IpAddress.is_valid, first_row)): # search ip_address in first row
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
and da.isdigit()
|
|
93
|
-
):
|
|
94
|
-
c.com_profile.parameters.device_address = int(da)
|
|
95
|
-
if name := i.get("name"):
|
|
96
|
-
c.name = name
|
|
97
|
-
return res
|
|
53
|
+
raise ValueError("Table header not found")
|
|
54
|
+
# header is exist search column by name
|
|
55
|
+
field_names: list[str] = []
|
|
56
|
+
for index, cell in enumerate(first_row):
|
|
57
|
+
for column in _h:
|
|
58
|
+
if any(map(cell.lower().startswith, _h[column])):
|
|
59
|
+
field_names.append(_h[column][0])
|
|
60
|
+
break
|
|
61
|
+
else:
|
|
62
|
+
field_names.append(F"unknown{index}")
|
|
63
|
+
if all(map(lambda name: name in field_names, ("ip",))):
|
|
64
|
+
csv_file.seek(0)
|
|
65
|
+
reader = csv.DictReader(csv_file, fieldnames=field_names, dialect=dialect)
|
|
66
|
+
next(reader) # skeep header
|
|
67
|
+
res: list[Client] = []
|
|
68
|
+
for i in reader:
|
|
69
|
+
if IpAddress.is_valid(i["ip"]):
|
|
70
|
+
res.append(c := Client(
|
|
71
|
+
media=Network(
|
|
72
|
+
host=i.get("ip") or "127.0.0.1",
|
|
73
|
+
port=i.get("port") or "8888",
|
|
74
|
+
to_recv=float(i.get("timeout") or 120.0)
|
|
75
|
+
),
|
|
76
|
+
SAP=int(i.get("sap") or 0x30),
|
|
77
|
+
secret=bytes(i.get("secret", "0000000000000000"), "utf-8"),
|
|
78
|
+
m_id=int(i.get("m_id") or 2),
|
|
79
|
+
id_=id_factory.create(),
|
|
80
|
+
universal=universal
|
|
81
|
+
))
|
|
82
|
+
if (
|
|
83
|
+
(da := i.get("da"))
|
|
84
|
+
and da.isdigit()
|
|
85
|
+
):
|
|
86
|
+
c.com_profile.parameters.device_address = int(da)
|
|
87
|
+
if name := i.get("name"):
|
|
88
|
+
c.name = name
|
|
89
|
+
return res
|
|
90
|
+
raise ValueError("not find at least one client")
|
DLMS_SPODES_client/task.py
CHANGED
|
@@ -1240,6 +1240,43 @@ class Write2(SimpleCopy, OK):
|
|
|
1240
1240
|
return result.OK
|
|
1241
1241
|
|
|
1242
1242
|
|
|
1243
|
+
@dataclass(frozen=True)
|
|
1244
|
+
class SetBits(SimpleCopy, OK):
|
|
1245
|
+
"""set bits by pattern"""
|
|
1246
|
+
par: Parameter
|
|
1247
|
+
pattern: dict[int, int]
|
|
1248
|
+
msg: str = "set bits"
|
|
1249
|
+
|
|
1250
|
+
async def exchange(self, c: Client) -> result.Ok | result.Error:
|
|
1251
|
+
if isinstance(res_obj := c.objects.par2obj(self.par), result.Error):
|
|
1252
|
+
return res_obj
|
|
1253
|
+
if isinstance(res_read := await Par2Data[cdt.Digital | cdt.BitString](self.par.attr).exchange(c), result.Error):
|
|
1254
|
+
return res_read
|
|
1255
|
+
data = res_read.value
|
|
1256
|
+
for el in self.par.elements():
|
|
1257
|
+
data = data[el]
|
|
1258
|
+
if isinstance(data, cdt.Digital):
|
|
1259
|
+
data_int = int(data)
|
|
1260
|
+
for pos, val in self.pattern.items():
|
|
1261
|
+
data_int = (data_int | (1 << pos)) if val else (data_int & ~(1 << pos))
|
|
1262
|
+
data.set(data_int)
|
|
1263
|
+
elif isinstance(data, cdt.BitString):
|
|
1264
|
+
for pos, val in self.pattern.items():
|
|
1265
|
+
try:
|
|
1266
|
+
data[pos] = val
|
|
1267
|
+
except IndexError as e:
|
|
1268
|
+
return result.Error.from_e(ValueError(f"can't set bit {pos} in {self.par}"))
|
|
1269
|
+
else:
|
|
1270
|
+
return result.Error.from_e(TypeError(f"{self.par} not available SetBits"))
|
|
1271
|
+
c.get_set_request_normal(
|
|
1272
|
+
obj=res_obj.value,
|
|
1273
|
+
attr_index=self.par.i,
|
|
1274
|
+
value=res_read.value.encoding)
|
|
1275
|
+
if isinstance(res_pdu := await c.read_data_block(), result.Error):
|
|
1276
|
+
return res_pdu
|
|
1277
|
+
return result.OK
|
|
1278
|
+
|
|
1279
|
+
|
|
1243
1280
|
@dataclass
|
|
1244
1281
|
@deprecated("use <WriteTranscript>")
|
|
1245
1282
|
class WriteParValue(SimpleCopy, OK):
|
|
@@ -3,10 +3,10 @@ DLMS_SPODES_client/__init__.py,sha256=6wphXvqkodng7h4nKNmkfldbaxf--IDVGfT0yNbas-
|
|
|
3
3
|
DLMS_SPODES_client/client.py,sha256=HnsbECdBGzrlyT0RzNGweIBms5LKRNP4uUMSmaMEPIA,112148
|
|
4
4
|
DLMS_SPODES_client/logger.py,sha256=zAbihLloMU99w8Sw3djQ0cwItzyGq0Fz8DI9_suazv4,1913
|
|
5
5
|
DLMS_SPODES_client/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
6
|
-
DLMS_SPODES_client/services.py,sha256=
|
|
6
|
+
DLMS_SPODES_client/services.py,sha256=g2Rv2j7mlIcNpnqxRZrKgSpy7k6pnaJHAkZ3JJUhK3s,3418
|
|
7
7
|
DLMS_SPODES_client/session.py,sha256=nPzXujpmGSTFFvhyZRjgH_RLX1DS9moRddUEZuf-QEE,12760
|
|
8
8
|
DLMS_SPODES_client/settings.py,sha256=6mitGe9UYeEgL61sf933MJ-S5N-ReoxvXqiI3agBGYE,1623
|
|
9
|
-
DLMS_SPODES_client/task.py,sha256=
|
|
9
|
+
DLMS_SPODES_client/task.py,sha256=QnxVp4txhCgFqVw9Aj9gGwaZjlt0iPR2WkXvgk_hDPQ,79822
|
|
10
10
|
DLMS_SPODES_client/gurux_common/enums/TraceLevel.py,sha256=Ne0Rn3c9ACqQjmde_ksbiQxIUv6nXsPQRnhkGwIv3QI,521
|
|
11
11
|
DLMS_SPODES_client/gurux_common/enums/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
12
12
|
DLMS_SPODES_client/gurux_dlms/AesGcmParameter.py,sha256=HJt0uvxtkqKEkvfiqXTMNsiayN15AgPJa9_iMSSFZsQ,1429
|
|
@@ -54,8 +54,8 @@ DLMS_SPODES_client/gurux_dlms/enums/Task.py,sha256=chuOL6-IMxBvABUZtoFcaYaQQB4GZ
|
|
|
54
54
|
DLMS_SPODES_client/gurux_dlms/enums/VdeStateError.py,sha256=qT87LMbIYEs3TYPIp3N-dR2Tcg9KhKyiELwhVl5U-tw,233
|
|
55
55
|
DLMS_SPODES_client/gurux_dlms/enums/__init__.py,sha256=F_sgGwNmmdpbKvP1klJQUNiLXxU2BtZ-LgEI9e6xP8g,1314
|
|
56
56
|
DLMS_SPODES_client/gurux_dlms/internal/_GXCommon.py,sha256=7D9EYcfiZxwbk8sfpHv7s2nYqrbmGf-Tbwv2T-gqmgk,53226
|
|
57
|
-
dlms_spodes_client-0.19.
|
|
58
|
-
dlms_spodes_client-0.19.
|
|
59
|
-
dlms_spodes_client-0.19.
|
|
60
|
-
dlms_spodes_client-0.19.
|
|
61
|
-
dlms_spodes_client-0.19.
|
|
57
|
+
dlms_spodes_client-0.19.29.dist-info/METADATA,sha256=p7a8aAHAJ7xkK0xxcKBujHgCy6TEjXBKjkF1wYwoxrA,1062
|
|
58
|
+
dlms_spodes_client-0.19.29.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
|
|
59
|
+
dlms_spodes_client-0.19.29.dist-info/entry_points.txt,sha256=Z6UTeQjjCf2k1Y3Bjs0s7yr-UYSWb-TvJMuG2K2MApw,70
|
|
60
|
+
dlms_spodes_client-0.19.29.dist-info/top_level.txt,sha256=rh_3Uig5bc6J_lKni01btol7dX_IgIJulNtGjGehmBE,19
|
|
61
|
+
dlms_spodes_client-0.19.29.dist-info/RECORD,,
|
|
File without changes
|
{dlms_spodes_client-0.19.27.dist-info → dlms_spodes_client-0.19.29.dist-info}/entry_points.txt
RENAMED
|
File without changes
|
|
File without changes
|