symetrie-hexapod 2023.1.0__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.
@@ -0,0 +1,415 @@
1
+ import logging
2
+ import socket
3
+ import threading
4
+ import time
5
+ from telnetlib import Telnet
6
+ from time import sleep
7
+ from typing import List
8
+
9
+ import paramiko
10
+
11
+ from egse.decorators import timer
12
+ from egse.device import DeviceTransport
13
+ from egse.settings import Settings
14
+ from egse.system import wait_until
15
+
16
+ logger = logging.getLogger(__name__)
17
+ ctrl_settings = Settings.load("ZONDA Controller")
18
+
19
+ RETURN_CODES = {
20
+ 0: "Success.",
21
+ -1: "Undefined error.",
22
+ -10: "Wrong value for parameter at index 0.",
23
+ -11: "Wrong value for parameter at index 1.",
24
+ -12: "Wrong value for parameter at index 2.",
25
+ -13: "Wrong value for parameter at index 3.",
26
+ -14: "Wrong value for parameter at index 4.",
27
+ -15: "Wrong value for parameter at index 5.",
28
+ -16: "Wrong value for parameter at index 6.",
29
+ -17: "Wrong value for parameter at index 7.",
30
+ -18: "Wrong value for parameter at index 8.",
31
+ -19: "Wrong value for parameter at index 9.",
32
+ -20: "Wrong value for parameter at index 10.",
33
+ -21: "Wrong value for parameter at index 11.",
34
+ -22: "Wrong value for parameter at index 12.",
35
+ -23: "Wrong value for parameter at index 13.",
36
+ -24: "Wrong value for parameter at index 14.",
37
+ -25: "Wrong value for parameter at index 15.",
38
+ -26: "Wrong value for parameter at index 16.",
39
+ -27: "Wrong value for parameter at index 17.",
40
+ -28: "Wrong value for parameter at index 18.",
41
+ -29: "Wrong value for parameter at index 19.",
42
+ -30: "Unknown command number.",
43
+ -31: "This configuration command is a 'get' only type.",
44
+ -32: "This configuration command is a 'set' only type.",
45
+ -33: "The axis number do not correspond to an axis defined on the controller.",
46
+ -34: "A stop task is running.",
47
+ -35: "All motors need to be control on.",
48
+ -36: "All motors need to be control off.",
49
+ -37: "Emergency stop is pressed.",
50
+ -38: "A motion task is running.",
51
+ -39: "A home task is running.",
52
+ -40: "Requested move is not feasible.",
53
+ -41: "Power supply of limit switches is off.",
54
+ -42: "Power supply of encoders is off.",
55
+ -43: "A fatal error is present. This type of error needs a controller restart to be removed.",
56
+ -44: "An error is present, error reset is required.",
57
+ -45: "Home is not completed.",
58
+ -46: "Software option not available (can be linked to hardware configuration).",
59
+ -47: "Virtual home: file was created on another controller (different MAC address).",
60
+ -48: "Virtual home: some positions read in file are out of software limits.",
61
+ -49: "Virtual home: file data were stored while hexapod was moving.",
62
+ -50: "Virtual home: no data available.",
63
+ -51: "Command has been rejected because another action is running.",
64
+ -52: "Timeout waiting for home complete status.",
65
+ -53: "Timeout waiting for control on status.",
66
+ -54: "Timeout on motion program start.",
67
+ -55: "Timeout on home task start.",
68
+ -56: "Timeout on virtual home write file task.",
69
+ -57: "Timeout on virtual home delete file task.",
70
+ -58: "Timeout on virtual home read file task.",
71
+ -59: "Timeout on disk access verification task.",
72
+ -60: "Configuration file: save process failed.",
73
+ -61: "Configuration file: loaded file is empty.",
74
+ -62: "Configuration file: loaded data are corrupted.",
75
+ -63: "No access to the memory disk.",
76
+ -64: "File does not exist.",
77
+ -65: "Folder access failed.",
78
+ -66: "Creation of folder tree on the memory disk failed.",
79
+ -67: "Generation or write of the checksum failed.",
80
+ -68: "File read: no data or wrong data size.",
81
+ -69: "File read: no checksum.",
82
+ -70: "File read: incorrect checksum.",
83
+ -71: "File write: failed.",
84
+ -72: "File open: failed.",
85
+ -73: "File delete: failed.",
86
+ -74: "Get MAC address failed.",
87
+ -75: "NaN (Not a Number) or infinite value found.",
88
+ -76: "The coordinate system transformations are not initialized.",
89
+ -77: "A kinematic error is present.",
90
+ -78: "The motor phase process failed (phase search or phase set from position offset).",
91
+ -79: "The motor phase is not found.",
92
+ -80: "Timeout waiting for control off status.",
93
+ -81: "The requested kinematic mode (number) is not defined for the machine.",
94
+ -82: "Timeout waiting for phase found status.",
95
+ -1000: "Internal error: 'RET_Dev_CfS_NaNReturned'.",
96
+ -1001: "Internal error: 'RET_Dev_CfS_FctNotAvailableInKernel'.",
97
+ -1002: "Internal error: 'RET_Dev_CfS_UndefinedCfSType'.",
98
+ -1003: "Internal error: 'RET_Dev_CfS_FIO_UndefinedFioType'.",
99
+ -1004: "Internal error: 'RET_Dev_CfS_FIO_HomeFile_UndefinedAction'.",
100
+ -1005: "Internal error: 'RET_Dev_UndefinedEnumValue'.",
101
+ -1006: "Internal error: 'RET_Dev_LdataCmdStatusIsNegative'.",
102
+ -1007: "Internal error: 'RET_Dev_NumMotorsInCoord_Sup_DEF_aGrQ_SIZE'.",
103
+ -1008: "Internal error: 'RET_Dev_NumMotorsInCoord_WrongNumber'.",
104
+ -1009: "Internal error: 'RET_String_StrCat_DestSizeReached'.",
105
+ -1010: "Internal error: 'RET_String_LengthOverStringSize'.",
106
+ -1011: "Internal error: 'RET_String_AllCharShouldIntBetween_0_255'.",
107
+ -1012: "Internal error: 'RET_String_StrCpy_DestSizeReached'.",
108
+ -1013: "Internal error: 'RET_ErrAction_HomeReset'.",
109
+ -1014: "Internal error: 'RET_Home_StopReceivedWhileRunning'.",
110
+ -1015: "Internal error: 'RET_UndefinedKinAssembly'.",
111
+ -1016: "Internal error: 'RET_WrongPmcConfig'.",
112
+ }
113
+
114
+
115
+ class ZondaError(Exception):
116
+ pass
117
+
118
+
119
+ class ZondaSSHInterface(object):
120
+ def __init__(self):
121
+ self.client = None
122
+ self.gpascii_client = None
123
+ self.connected = False
124
+ self.ssh_output = None
125
+ self.ssh_error = None
126
+ self.verbose = False
127
+ self.ip = ctrl_settings.IP
128
+
129
+ self.semaphore = threading.Semaphore()
130
+
131
+ self.CommandReturns = RETURN_CODES
132
+
133
+ # Etablish the SSH connection with the controller and open gpascii
134
+ def connect(self, ip):
135
+ try:
136
+ # Paramiko.SSHClient can be used to make connections to the remote server and
137
+ # transfer files
138
+ logger.info("Establishing ssh connection with {}...".format(ip))
139
+ self.client = paramiko.SSHClient()
140
+ # Parsing an instance of the AutoAddPolicy to set_missing_host_key_policy() changes
141
+ # it to allow any host.
142
+ self.client.set_missing_host_key_policy(paramiko.AutoAddPolicy())
143
+ self.client.connect(hostname=ip, port=22, username="root", password="deltatau",
144
+ timeout=5,
145
+ allow_agent=False, look_for_keys=False)
146
+ logger.warning("Connected to the server: {}".format(ip))
147
+ self.connected = True
148
+ except paramiko.AuthenticationException:
149
+ logger.warning("Authentication failed, please verify your credentials.")
150
+ return False
151
+ except paramiko.SSHException as sshException:
152
+ logger.warning("Could not establish SSH connection: %s" % sshException)
153
+ return False
154
+ except socket.timeout as e:
155
+ logger.warning("Connection timed out.")
156
+ return False
157
+ except Exception as e:
158
+ logger.warning("Exception in connecting to the server.")
159
+ logger.warning("Python says:", e)
160
+ self.client.close()
161
+ return False
162
+
163
+ self.gpascii_client = self.client.invoke_shell(term="vt100")
164
+ self.sendcommand("gpascii -2", 0.5)
165
+ self.sendcommand("echo7")
166
+
167
+ def is_connected(self):
168
+ return self.connected
169
+
170
+ # send a command through gpascii
171
+ def sendcommand(self, command, wait=0):
172
+ self.gpascii_client.send(command + "\r\n")
173
+ if self.verbose == True:
174
+ pass # print("Command send: {}".format(command))
175
+
176
+ if wait > 0:
177
+ sleep(wait)
178
+
179
+ # wait for response from gpascii
180
+ step = 0.1
181
+ time = 0
182
+
183
+ while True:
184
+ if self.gpascii_client.recv_ready():
185
+ break
186
+ if time > 2.0:
187
+ logger.warning("Command response timed out!")
188
+ return ""
189
+ sleep(step)
190
+ time += step
191
+
192
+ # get and clean response
193
+ response = self.gpascii_client.recv(2048)
194
+ response = response.decode()
195
+ response = response.replace(command, "")
196
+ response = response.replace("\r", "")
197
+ # response = response.replace("\n"," ")
198
+ response = response.replace("\x06", "")
199
+
200
+ # remove empty elements
201
+ cleaned = ""
202
+ lines = response.split("\n")
203
+ for line in lines:
204
+ if line == "" or line == " ":
205
+ lines.remove(line)
206
+ else:
207
+ cleaned += line + "\n"
208
+
209
+ # if self.verbose is True:
210
+ # print("Command rcv: {}".format(cleaned))
211
+ return cleaned
212
+
213
+ def cmd_decode(self, name, arguments=list()):
214
+ command = ""
215
+
216
+ if "JOG" in name:
217
+ command += "c_ax={} ".format(arguments[0])
218
+ arguments.pop(0)
219
+
220
+ # set cfg part
221
+ cfg = 1
222
+ if "?" in name:
223
+ cfg = 0
224
+ name = name.replace("?", "")
225
+ if "CFG_" in name:
226
+ command += "c_cfg={} ".format(cfg)
227
+
228
+ # create the parameters part of a command using the given arguments
229
+ for index in range(0, len(arguments)):
230
+ command += "c_par({})={} ".format(index, arguments[index])
231
+
232
+ # finish command line
233
+ command += "c_cmd=C_{}".format(name)
234
+
235
+ # logger.debug("command sent: ", command)
236
+ return command
237
+
238
+ def cmd_check(self):
239
+ # sends "c_cmd" and returns the state of the command that has been previously sent.
240
+ Timer = 0
241
+ last = time.process_time()
242
+
243
+ # For a maximum of 5 sec
244
+ while Timer < 5:
245
+
246
+ # wait and count
247
+ time.sleep(0.1)
248
+ Timer += time.process_time() - last
249
+ last = time.process_time()
250
+
251
+ # ask for command state
252
+ response = self.sendcommand("c_cmd")
253
+ response.replace("\n", "")
254
+
255
+ if "=" in response:
256
+ # get value
257
+ elements = response.split("=", 1)
258
+ element = elements[1]
259
+
260
+ # exit if command is completed
261
+ if int(element) <= 0:
262
+ response = element
263
+
264
+ code = int(response)
265
+
266
+ if code < 0:
267
+ message = self.CommandReturns[code]
268
+ logger.warning("Command Error return: {} : {}".format(code, message))
269
+ elif code == 0:
270
+ message = "Command successful: 0, execution time {} sec".format(Timer)
271
+ # logger.info(message)
272
+ return [code, message]
273
+ logger.warning("Error: Order timed out !")
274
+ return [-1, "Error: Order timed out !"]
275
+
276
+ def p_check(self, number=20):
277
+ # ask the value of the command parameters
278
+ command = "c_par(0)"
279
+ if number > 1:
280
+ command += ",{},1".format(number)
281
+
282
+ answer = self.sendcommand(command)
283
+ return answer
284
+
285
+ def disconnect(self):
286
+ try:
287
+ logger.info("Closing ssh connection with {}...")
288
+ self.client.close()
289
+ self.gpascii_client.close()
290
+ self.connected = False
291
+ logger.warning("...disconnected from ssh".format(self.ip))
292
+ except Exception as e_exc:
293
+ raise ZondaError(f"Could not close socket to ") from e_exc
294
+
295
+
296
+ class ZondaTelnetInterface(DeviceTransport):
297
+ """
298
+ The ZONDA Hexapod device interface based on the telnet protocol.
299
+ """
300
+
301
+ TELNET_TIMEOUT = 0.5
302
+
303
+ def __init__(self):
304
+ self.telnet = Telnet()
305
+ self._is_connected = False
306
+
307
+ def connect(self, hostname: str):
308
+ self.telnet.open(hostname, 23)
309
+ self.telnet.read_until(b"login: ", timeout=self.TELNET_TIMEOUT)
310
+ self.telnet.write(b"root\r\n")
311
+ self.telnet.read_until(b"Password: ", timeout=self.TELNET_TIMEOUT)
312
+ self.telnet.write(b"deltatau\r\n")
313
+ self.telnet.read_until(b"ppmac# ", timeout=self.TELNET_TIMEOUT)
314
+ self.telnet.write(b"gpascii -2\r\n")
315
+ self.telnet.write(b"echo7\r\n")
316
+ self.telnet.read_until(b'\x06', timeout=self.TELNET_TIMEOUT)
317
+ self.telnet.read_very_eager()
318
+ self._is_connected = True
319
+
320
+ def is_connected(self):
321
+ return self._is_connected
322
+
323
+ def disconnect(self):
324
+ self.telnet.read_very_eager()
325
+ self.telnet.close()
326
+ self._is_connected = False
327
+
328
+ def check_command_status(self):
329
+
330
+ if wait_until(lambda: self.trans('c_cmd')[0] == '0', interval=0.01):
331
+ rc = int(self.trans('c_cmd')[0])
332
+ logger.warning(f"Command check: {RETURN_CODES[rc]} [{rc}]")
333
+ else:
334
+ rc = 0
335
+
336
+ return rc, RETURN_CODES[rc]
337
+
338
+ def get_pars(self, count: int = 20):
339
+ # ask the value of the command parameters
340
+ command = "c_par(0)"
341
+ if count > 1:
342
+ command += f",{count},1"
343
+
344
+ return self.trans(command)
345
+
346
+ def trans(self, cmd: str) -> List:
347
+ self.write(cmd)
348
+ response = self.read()
349
+ if response and response[0] == cmd:
350
+ return response[1:]
351
+ else:
352
+ return response
353
+
354
+ def read(self) -> List:
355
+ response = self.telnet.read_until(b'\x06\r\n', timeout=self.TELNET_TIMEOUT)
356
+ response = response.decode()
357
+ parts = response.split("\r\n")
358
+ if len(parts) == 1:
359
+ return []
360
+ if parts[-2] == '\x06':
361
+ return parts[:-2]
362
+ logger.warning("Expected ACK at the end of the response")
363
+ return parts
364
+
365
+ def write(self, cmd: str):
366
+ self.telnet.write(cmd.encode() + b"\r\n")
367
+
368
+
369
+ def decode_command(cmd_name: str, *args):
370
+
371
+ args = list(args)
372
+ cmd_name = cmd_name.upper()
373
+
374
+ full_command = ""
375
+
376
+ if "JOG" in cmd_name:
377
+ arg = args.pop(0)
378
+ full_command += f"c_ax={arg} "
379
+
380
+ # set cfg part
381
+
382
+ if "?" in cmd_name:
383
+ cmd_name = cmd_name.replace("?", "")
384
+ cfg = 0
385
+ else:
386
+ cfg = 1
387
+
388
+ if "CFG_" in cmd_name:
389
+ full_command += f"c_cfg={cfg} "
390
+
391
+ # create the parameters part of a full_command using the given arguments
392
+
393
+ for index, arg in enumerate(args):
394
+ full_command += f"c_par({index})={arg} "
395
+
396
+ # finish full_command line
397
+
398
+ full_command += f"c_cmd=C_{cmd_name}"
399
+
400
+ return full_command
401
+
402
+
403
+ if __name__ == "__main__":
404
+ import rich
405
+
406
+ zonda = ZondaTelnetInterface()
407
+ zonda.connect("192.168.56.10")
408
+ rich.print(zonda.trans("s_hexa,50,1"))
409
+ rich.print(zonda.trans("s_ax_1,6,1"))
410
+ rich.print(zonda.trans("s_pos_ax_1,6,1"))
411
+ rich.print(zonda.read())
412
+ zonda.disconnect()
413
+
414
+ print(decode_command("CFG_LIMIT?", 0))
415
+ print(decode_command("CFG_LIMIT", 1, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12))
@@ -0,0 +1,119 @@
1
+ import logging
2
+
3
+ from egse.command import ClientServerCommand
4
+ from egse.control import ControlServer
5
+ from egse.hexapod.symetrie.zonda import ZondaController
6
+ from egse.hexapod.symetrie.zonda import ZondaInterface
7
+ from egse.hexapod.symetrie.zonda import ZondaSimulator
8
+ from egse.hk import read_conversion_dict, convert_hk_names
9
+ from egse.metrics import define_metrics
10
+ from egse.protocol import CommandProtocol
11
+ from egse.settings import Settings
12
+ from egse.system import format_datetime
13
+ from egse.zmq_ser import bind_address
14
+
15
+ logger = logging.getLogger(__name__)
16
+
17
+ ctrl_settings = Settings.load("Hexapod ZONDA Control Server")
18
+ zonda_settings = Settings.load(filename="zonda.yaml")
19
+
20
+
21
+ class ZondaCommand(ClientServerCommand):
22
+ pass
23
+
24
+
25
+ class ZondaProtocol(CommandProtocol):
26
+ def __init__(self, control_server: ControlServer):
27
+ super().__init__()
28
+ self.control_server = control_server
29
+
30
+ self.hk_conversion_table = read_conversion_dict(self.control_server.get_storage_mnemonic(), use_site=True)
31
+
32
+ if Settings.simulation_mode():
33
+ self.hexapod = ZondaSimulator()
34
+ else:
35
+ self.hexapod = ZondaController()
36
+
37
+ self.hexapod.connect()
38
+
39
+ self.load_commands(zonda_settings.Commands, ZondaCommand, ZondaInterface)
40
+
41
+ self.build_device_method_lookup_table(self.hexapod)
42
+
43
+ self.metrics = define_metrics("ZONDA")
44
+
45
+ def get_bind_address(self):
46
+ return bind_address(
47
+ self.control_server.get_communication_protocol(),
48
+ self.control_server.get_commanding_port(),
49
+ )
50
+
51
+ def get_status(self):
52
+
53
+ status = super().get_status()
54
+
55
+ mach_positions = self.hexapod.get_machine_positions()
56
+ user_positions = self.hexapod.get_user_positions()
57
+ actuator_length = self.hexapod.get_actuator_length()
58
+
59
+ status.update({"mach": mach_positions, "user": user_positions, "alength": actuator_length})
60
+
61
+ return status
62
+
63
+ def get_housekeeping(self) -> dict:
64
+
65
+ result = dict()
66
+ result["timestamp"] = format_datetime()
67
+
68
+ mach_positions = self.hexapod.get_machine_positions()
69
+ user_positions = self.hexapod.get_user_positions()
70
+ actuator_length = self.hexapod.get_actuator_length()
71
+ actuator_temperature = self.hexapod.get_temperature()
72
+
73
+ # TODO If you change these names, please, also change them in egse.fov.fov_hk!
74
+ for idx, key in enumerate(
75
+ ["user_t_x", "user_t_y", "user_t_z", "user_r_x", "user_r_y", "user_r_z"]
76
+ ):
77
+ result[key] = user_positions[idx]
78
+
79
+ for idx, key in enumerate(
80
+ ["mach_t_x", "mach_t_y", "mach_t_z", "mach_r_x", "mach_r_y", "mach_r_z"]
81
+ ):
82
+ result[key] = mach_positions[idx]
83
+
84
+ for idx, key in enumerate(
85
+ ["alen_t_x", "alen_t_y", "alen_t_z", "alen_r_x", "alen_r_y", "alen_r_z"]
86
+ ):
87
+ result[key] = actuator_length[idx]
88
+
89
+ for idx, key in enumerate(
90
+ ["atemp_1", "atemp_2", "atemp_3", "atemp_4", "atemp_5", "atemp_6"]
91
+ ):
92
+ result[key] = actuator_temperature[idx]
93
+
94
+ # # TODO:
95
+ # # the get_general_state() method should be refactored as to return a dict instead of a
96
+ # # list. Also, we might want to rethink the usefulness of returning the tuple,
97
+ # # it the first return value ever used?
98
+ #
99
+ # _, _ = self.hexapod.get_general_state()
100
+ #
101
+ result["Homing done"] = self.hexapod.is_homing_done()
102
+ result["In position"] = self.hexapod.is_in_position()
103
+
104
+ hk_dict = convert_hk_names(result, self.hk_conversion_table)
105
+
106
+ for key, value in hk_dict.items():
107
+ if key != "timestamp":
108
+ self.metrics[key].set(value)
109
+
110
+ return hk_dict
111
+
112
+ def is_connected(self):
113
+ # FIXME(rik): There must be another way to check if the socket is still alive...
114
+ # This will send way too many VERSION requests to the controllers.
115
+ # According to SO [https://stackoverflow.com/a/15175067] the best way
116
+ # to check for a connection drop / close is to handle the exceptions
117
+ # properly.... so, no polling for connections by sending it a simple
118
+ # command.
119
+ return self.hexapod.is_connected()