| 
									
										
										
										
											2024-06-28 03:59:15 -07:00
										 |  |  | import base64 | 
					
						
							| 
									
										
										
										
											2024-06-24 08:05:47 -07:00
										 |  |  | import os | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-28 03:59:15 -07:00
										 |  |  | from cryptography.hazmat.backends import default_backend | 
					
						
							|  |  |  | from cryptography.hazmat.primitives.ciphers import Cipher, algorithms, modes | 
					
						
							| 
									
										
										
										
											2024-06-28 09:49:43 -07:00
										 |  |  | from meshtastic.mesh_pb2 import MeshPacket, Data, HardwareModel | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  | from meshtastic.portnums_pb2 import PortNum | 
					
						
							|  |  |  | from prometheus_client import CollectorRegistry, Counter, Histogram, Gauge | 
					
						
							| 
									
										
										
										
											2024-06-28 09:49:43 -07:00
										 |  |  | from psycopg_pool import ConnectionPool | 
					
						
							| 
									
										
										
										
											2024-06-23 12:15:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-24 11:44:10 -07:00
										 |  |  | from exporter.registry import ProcessorRegistry, ClientDetails | 
					
						
							| 
									
										
										
										
											2024-06-23 12:15:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MessageProcessor: | 
					
						
							| 
									
										
										
										
											2024-06-28 09:49:43 -07:00
										 |  |  |     def __init__(self, registry: CollectorRegistry, db_pool: ConnectionPool): | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         self.rx_rssi_gauge = None | 
					
						
							|  |  |  |         self.channel_counter = None | 
					
						
							|  |  |  |         self.packet_id_counter = None | 
					
						
							|  |  |  |         self.hop_start_gauge = None | 
					
						
							|  |  |  |         self.via_mqtt_counter = None | 
					
						
							|  |  |  |         self.want_ack_counter = None | 
					
						
							|  |  |  |         self.hop_limit_counter = None | 
					
						
							|  |  |  |         self.rx_snr_gauge = None | 
					
						
							|  |  |  |         self.rx_time_histogram = None | 
					
						
							|  |  |  |         self.total_packets_counter = None | 
					
						
							|  |  |  |         self.destination_message_type_counter = None | 
					
						
							|  |  |  |         self.source_message_type_counter = None | 
					
						
							| 
									
										
										
										
											2024-06-23 12:15:31 -07:00
										 |  |  |         self.registry = registry | 
					
						
							| 
									
										
										
										
											2024-06-28 09:49:43 -07:00
										 |  |  |         self.db_pool = db_pool | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         self.init_metrics() | 
					
						
							| 
									
										
										
										
											2024-06-24 11:44:10 -07:00
										 |  |  |         self.processor_registry = ProcessorRegistry() | 
					
						
							| 
									
										
										
										
											2024-06-23 12:15:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |     def init_metrics(self): | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |         common_labels = [ | 
					
						
							|  |  |  |             'source_id', 'source_short_name', 'source_long_name', 'source_hardware_model', 'source_role', | 
					
						
							|  |  |  |             'destination_id', 'destination_short_name', 'destination_long_name', 'destination_hardware_model', | 
					
						
							|  |  |  |             'destination_role' | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         self.source_message_type_counter = Counter( | 
					
						
							|  |  |  |             'mesh_packet_source_types', | 
					
						
							|  |  |  |             'Types of mesh packets processed by source', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels + ['portnum'], | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Destination-related counters | 
					
						
							|  |  |  |         self.destination_message_type_counter = Counter( | 
					
						
							|  |  |  |             'mesh_packet_destination_types', | 
					
						
							|  |  |  |             'Types of mesh packets processed by destination', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels + ['portnum'], | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Counters for the total number of packets | 
					
						
							|  |  |  |         self.total_packets_counter = Counter( | 
					
						
							|  |  |  |             'mesh_packet_total', | 
					
						
							|  |  |  |             'Total number of mesh packets processed', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Histogram for the rx_time (time in seconds) | 
					
						
							|  |  |  |         self.rx_time_histogram = Histogram( | 
					
						
							|  |  |  |             'mesh_packet_rx_time', | 
					
						
							|  |  |  |             'Receive time of mesh packets (seconds since 1970)', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Gauge for the rx_snr (signal-to-noise ratio) | 
					
						
							|  |  |  |         self.rx_snr_gauge = Gauge( | 
					
						
							|  |  |  |             'mesh_packet_rx_snr', | 
					
						
							|  |  |  |             'Receive SNR of mesh packets', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Counter for hop_limit | 
					
						
							|  |  |  |         self.hop_limit_counter = Counter( | 
					
						
							|  |  |  |             'mesh_packet_hop_limit', | 
					
						
							|  |  |  |             'Hop limit of mesh packets', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Counter for want_ack (occurrences of want_ack set to true) | 
					
						
							|  |  |  |         self.want_ack_counter = Counter( | 
					
						
							|  |  |  |             'mesh_packet_want_ack', | 
					
						
							|  |  |  |             'Occurrences of want ACK for mesh packets', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Counter for via_mqtt (occurrences of via_mqtt set to true) | 
					
						
							|  |  |  |         self.via_mqtt_counter = Counter( | 
					
						
							|  |  |  |             'mesh_packet_via_mqtt', | 
					
						
							|  |  |  |             'Occurrences of mesh packets sent via MQTT', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Gauge for hop_start | 
					
						
							|  |  |  |         self.hop_start_gauge = Gauge( | 
					
						
							|  |  |  |             'mesh_packet_hop_start', | 
					
						
							|  |  |  |             'Hop start of mesh packets', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Counter for unique packet IDs | 
					
						
							|  |  |  |         self.packet_id_counter = Counter( | 
					
						
							|  |  |  |             'mesh_packet_ids', | 
					
						
							|  |  |  |             'Unique IDs for mesh packets', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels + ['packet_id'], | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Counter for the channel used | 
					
						
							|  |  |  |         self.channel_counter = Counter( | 
					
						
							|  |  |  |             'mesh_packet_channel', | 
					
						
							|  |  |  |             'Channel used for mesh packets', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels + ['channel'], | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         # Gauge for the rx_rssi (received signal strength indicator) | 
					
						
							|  |  |  |         self.rx_rssi_gauge = Gauge( | 
					
						
							|  |  |  |             'mesh_packet_rx_rssi', | 
					
						
							|  |  |  |             'Receive RSSI of mesh packets', | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             registry=self.registry | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-23 12:15:31 -07:00
										 |  |  |     def process(self, mesh_packet: MeshPacket): | 
					
						
							| 
									
										
										
										
											2024-06-28 03:59:15 -07:00
										 |  |  |         if getattr(mesh_packet, 'encrypted'): | 
					
						
							|  |  |  |             key_bytes = base64.b64decode(os.getenv('MQTT_SERVER_KEY', '1PG7OiApB1nwvP+rz05pAQ==').encode('ascii')) | 
					
						
							|  |  |  |             nonce_packet_id = getattr(mesh_packet, "id").to_bytes(8, "little") | 
					
						
							|  |  |  |             nonce_from_node = getattr(mesh_packet, "from").to_bytes(8, "little") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Put both parts into a single byte array. | 
					
						
							|  |  |  |             nonce = nonce_packet_id + nonce_from_node | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             cipher = Cipher(algorithms.AES(key_bytes), modes.CTR(nonce), backend=default_backend()) | 
					
						
							|  |  |  |             decryptor = cipher.decryptor() | 
					
						
							|  |  |  |             decrypted_bytes = decryptor.update(getattr(mesh_packet, "encrypted")) + decryptor.finalize() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             data = Data() | 
					
						
							|  |  |  |             data.ParseFromString(decrypted_bytes) | 
					
						
							|  |  |  |             mesh_packet.decoded.CopyFrom(data) | 
					
						
							| 
									
										
										
										
											2024-06-24 08:05:47 -07:00
										 |  |  |         port_num = int(mesh_packet.decoded.portnum) | 
					
						
							| 
									
										
										
										
											2024-06-23 12:15:31 -07:00
										 |  |  |         payload = mesh_packet.decoded.payload | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-28 09:49:43 -07:00
										 |  |  |         source_node_id = getattr(mesh_packet, 'from') | 
					
						
							|  |  |  |         source_client_details = self._get_client_details(source_node_id) | 
					
						
							| 
									
										
										
										
											2024-06-24 08:05:47 -07:00
										 |  |  |         if os.getenv('MESH_HIDE_SOURCE_DATA', 'false') == 'true': | 
					
						
							| 
									
										
										
										
											2024-06-24 12:56:49 -07:00
										 |  |  |             source_client_details = ClientDetails(node_id=source_client_details.node_id, short_name='Hidden', | 
					
						
							| 
									
										
										
										
											2024-06-24 11:44:10 -07:00
										 |  |  |                                                   long_name='Hidden') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-28 09:49:43 -07:00
										 |  |  |         destination_node_id = getattr(mesh_packet, 'to') | 
					
						
							|  |  |  |         destination_client_details = self._get_client_details(destination_node_id) | 
					
						
							| 
									
										
										
										
											2024-06-24 08:05:47 -07:00
										 |  |  |         if os.getenv('MESH_HIDE_DESTINATION_DATA', 'false') == 'true': | 
					
						
							| 
									
										
										
										
											2024-06-24 12:56:49 -07:00
										 |  |  |             destination_client_details = ClientDetails(node_id=destination_client_details.node_id, short_name='Hidden', | 
					
						
							| 
									
										
										
										
											2024-06-24 11:44:10 -07:00
										 |  |  |                                                        long_name='Hidden') | 
					
						
							| 
									
										
										
										
											2024-06-24 08:05:47 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if port_num in map(int, os.getenv('FILTERED_PORTS', '1').split(',')):  # Filter out ports | 
					
						
							|  |  |  |             return None  # Ignore this packet | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         self.process_simple_packet_details(destination_client_details, mesh_packet, port_num, source_client_details) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-28 09:49:43 -07:00
										 |  |  |         processor = ProcessorRegistry.get_processor(port_num)(self.registry, self.db_pool) | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         processor.process(payload, client_details=source_client_details) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_port_name_from_portnum(self, port_num): | 
					
						
							| 
									
										
										
										
											2024-06-25 23:33:39 -07:00
										 |  |  |         descriptor = PortNum.DESCRIPTOR | 
					
						
							|  |  |  |         for enum_value in descriptor.values: | 
					
						
							|  |  |  |             if enum_value.number == port_num: | 
					
						
							|  |  |  |                 return enum_value.name | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         return 'UNKNOWN_PORT' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def process_simple_packet_details(self, destination_client_details, mesh_packet, port_num, source_client_details): | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |         common_labels = { | 
					
						
							|  |  |  |             'source_id': source_client_details.node_id, | 
					
						
							|  |  |  |             'source_short_name': source_client_details.short_name, | 
					
						
							|  |  |  |             'source_long_name': source_client_details.long_name, | 
					
						
							|  |  |  |             'source_hardware_model': source_client_details.hardware_model, | 
					
						
							|  |  |  |             'source_role': source_client_details.role, | 
					
						
							|  |  |  |             'destination_id': destination_client_details.node_id, | 
					
						
							|  |  |  |             'destination_short_name': destination_client_details.short_name, | 
					
						
							|  |  |  |             'destination_long_name': destination_client_details.long_name, | 
					
						
							|  |  |  |             'destination_hardware_model': destination_client_details.hardware_model, | 
					
						
							|  |  |  |             'destination_role': destination_client_details.role, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         self.source_message_type_counter.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             **common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             portnum=self.get_port_name_from_portnum(port_num) | 
					
						
							|  |  |  |         ).inc() | 
					
						
							| 
									
										
										
										
											2024-06-24 08:05:47 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         self.destination_message_type_counter.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             **common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             portnum=self.get_port_name_from_portnum(port_num) | 
					
						
							| 
									
										
										
										
											2024-06-23 13:00:33 -07:00
										 |  |  |         ).inc() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         self.total_packets_counter.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             **common_labels | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         ).inc() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.rx_time_histogram.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             **common_labels | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         ).observe(mesh_packet.rx_time) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.rx_snr_gauge.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             **common_labels | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         ).set(mesh_packet.rx_snr) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.hop_limit_counter.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             **common_labels | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         ).inc(mesh_packet.hop_limit) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if mesh_packet.want_ack: | 
					
						
							|  |  |  |             self.want_ack_counter.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |                 **common_labels | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             ).inc() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if mesh_packet.via_mqtt: | 
					
						
							|  |  |  |             self.via_mqtt_counter.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |                 **common_labels | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             ).inc() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.hop_start_gauge.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             **common_labels | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         ).set(mesh_packet.hop_start) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.packet_id_counter.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             **common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             packet_id=mesh_packet.id | 
					
						
							|  |  |  |         ).inc() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Increment the channel counter | 
					
						
							|  |  |  |         self.channel_counter.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             **common_labels, | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |             channel=mesh_packet.channel | 
					
						
							|  |  |  |         ).inc() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Set the rx_rssi in the gauge | 
					
						
							|  |  |  |         self.rx_rssi_gauge.labels( | 
					
						
							| 
									
										
										
										
											2024-06-30 09:20:25 -07:00
										 |  |  |             **common_labels | 
					
						
							| 
									
										
										
										
											2024-06-25 12:39:27 -07:00
										 |  |  |         ).set(mesh_packet.rx_rssi) | 
					
						
							| 
									
										
										
										
											2024-06-23 13:00:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-28 09:49:43 -07:00
										 |  |  |     def _get_client_details(self, node_id: int) -> ClientDetails: | 
					
						
							|  |  |  |         node_id_str = str(node_id)  # Convert the integer to a string | 
					
						
							|  |  |  |         with self.db_pool.connection() as conn: | 
					
						
							|  |  |  |             with conn.cursor() as cur: | 
					
						
							|  |  |  |                 # First, try to select the existing record | 
					
						
							|  |  |  |                 cur.execute("""
 | 
					
						
							|  |  |  |                     SELECT node_id, short_name, long_name, hardware_model, role  | 
					
						
							|  |  |  |                     FROM client_details  | 
					
						
							|  |  |  |                     WHERE node_id = %s; | 
					
						
							|  |  |  |                 """, (node_id_str,))
 | 
					
						
							|  |  |  |                 result = cur.fetchone() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if not result: | 
					
						
							|  |  |  |                     # If the client is not found, insert a new record | 
					
						
							|  |  |  |                     cur.execute("""
 | 
					
						
							|  |  |  |                         INSERT INTO client_details (node_id, short_name, long_name, hardware_model, role) | 
					
						
							|  |  |  |                         VALUES (%s, %s, %s, %s, %s) | 
					
						
							|  |  |  |                         RETURNING node_id, short_name, long_name, hardware_model, role; | 
					
						
							|  |  |  |                     """, (node_id_str, 'Unknown', 'Unknown', HardwareModel.UNSET, None))
 | 
					
						
							|  |  |  |                     conn.commit() | 
					
						
							|  |  |  |                     result = cur.fetchone() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # At this point, we should always have a result, either from SELECT or INSERT | 
					
						
							|  |  |  |         return ClientDetails( | 
					
						
							|  |  |  |             node_id=result[0], | 
					
						
							|  |  |  |             short_name=result[1], | 
					
						
							|  |  |  |             long_name=result[2], | 
					
						
							|  |  |  |             hardware_model=result[3], | 
					
						
							|  |  |  |             role=result[4] | 
					
						
							|  |  |  |         ) |