client-gui.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. import numpy as np
  2. from matplotlib.figure import Figure
  3. import socket
  4. import msgpack
  5. from PySide6.QtWidgets import QApplication, QMainWindow, QWidget, QVBoxLayout, QPushButton, QHBoxLayout, QMessageBox
  6. from PySide6.QtWidgets import QDialog, QFormLayout, QLineEdit, QLabel, QComboBox, QSpinBox, QToolButton, QFileDialog
  7. from PySide6.QtCore import QTimer
  8. from matplotlib.backends.backend_qt5agg import FigureCanvasQTAgg as FigureCanvas
  9. from matplotlib.backends.backend_qt5agg import NavigationToolbar2QT as NavigationToolbar
  10. from PySide6.QtWidgets import QSizePolicy
  11. import json
  12. class MplCanvas(FigureCanvas):
  13. def __init__(self, parent=None, width=5, height=4, dpi=100):
  14. self.fig = Figure(figsize=(width, height), dpi=dpi, facecolor='grey')
  15. super().__init__(self.fig)
  16. self.setParent(parent)
  17. self.axes = self.fig.add_subplot(111)
  18. # Построим простой график
  19. self.axes.plot([0, 1, 2, 3], [10, 1, 20, 3])
  20. self.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Expanding)
  21. class SDRClient(QMainWindow):
  22. def __init__(self):
  23. super().__init__()
  24. # Параметры подключения к серверу
  25. self.server_ip = '127.0.0.1'
  26. self.server_port = 5003
  27. self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
  28. # Параметры для отрисовки
  29. self.channel_data = {
  30. 'time': np.array([], dtype=np.int64),
  31. 'channel_g': np.array([], dtype=np.int16),
  32. 'channel_rf': np.array([], dtype=np.int16),
  33. 'channel_adc_ttl': np.array([], dtype=np.int16),
  34. 'channel_rf_ttl': np.array([], dtype=np.int16)
  35. }
  36. self.wavetypes = ["default"]
  37. # Настройка окна
  38. self.setWindowTitle("SDR Client")
  39. self.setGeometry(100, 100, 1280, 720)
  40. self.central_widget = QWidget()
  41. self.setCentralWidget(self.central_widget)
  42. self.layout = QHBoxLayout()
  43. self.plot_layout = QVBoxLayout()
  44. self.menu_layout = QVBoxLayout()
  45. self.layout.addLayout(self.menu_layout)
  46. self.layout.addLayout(self.plot_layout)
  47. self.central_widget.setLayout(self.layout)
  48. self.command_layout = QHBoxLayout()
  49. self.plot_layout.addLayout(self.command_layout)
  50. self.add_event_button = QPushButton("Add Event")
  51. self.command_layout.addWidget(self.add_event_button)
  52. self.add_rfwave_button = QPushButton("Add RF Wave")
  53. self.command_layout.addWidget(self.add_rfwave_button)
  54. self.run_event_list_button = QPushButton("Run Event List")
  55. self.command_layout.addWidget(self.run_event_list_button)
  56. self.clear_event_list_button = QPushButton("Clear Event List")
  57. self.command_layout.addWidget(self.clear_event_list_button)
  58. self.plot_canvas = MplCanvas(self, width=5, height=4, dpi=100)
  59. #self.plot_canvas.setSizePolicy(self.plot_widget.sizePolicy())
  60. self.plot_layout.addWidget(self.plot_canvas, stretch=1)
  61. toolbar = NavigationToolbar(self.plot_canvas, self)
  62. self.plot_layout.addWidget(toolbar, stretch=1)
  63. self.connect_button = QPushButton("Connect to SDR")
  64. self.connect_button.setMinimumSize(200, 50)
  65. self.disconnect_button = QPushButton("Disconnect")
  66. self.disconnect_button.setMinimumSize(200, 50)
  67. self.settings_button = QPushButton("Settings")
  68. self.settings_button.setMinimumSize(200, 50)
  69. self.recv_sim_data_button = QPushButton("Receive Sim Data")
  70. self.recv_sim_data_button.setMinimumSize(200, 50)
  71. self.menu_layout.addWidget(self.connect_button)
  72. self.menu_layout.addWidget(self.disconnect_button)
  73. self.menu_layout.addWidget(self.settings_button)
  74. self.menu_layout.addWidget(self.recv_sim_data_button)
  75. self.connect_button.clicked.connect(self.connect_to_server)
  76. self.disconnect_button.clicked.connect(self.disconnect_from_server)
  77. self.add_event_button.clicked.connect(self.add_event_dialog)
  78. self.add_rfwave_button.clicked.connect(self.add_wavetype_dialog)
  79. self.run_event_list_button.clicked.connect(self.run_event_list)
  80. self.clear_event_list_button.clicked.connect(self.clear_event_list)
  81. self.recv_sim_data_button.clicked.connect(self.receive_sim_data)
  82. def connect_to_server(self):
  83. try:
  84. self.sock.connect((self.server_ip, self.server_port))
  85. print(f"Connected to SDR server at {self.server_ip}:{self.server_port}")
  86. QMessageBox.information(self, "Connection Successful", f"Successfully connected to SDR server at {self.server_ip}:{self.server_port}")
  87. except Exception as e:
  88. print(f"Failed to connect to server: {e}")
  89. QMessageBox.critical(self, "Connection Error", f"Failed to connect to server: {e}")
  90. self.sock = None
  91. def disconnect_from_server(self):
  92. if self.sock:
  93. self.sock.close()
  94. QMessageBox.information(self, "Disconnected", "Successfully disconnected from SDR server")
  95. print("Disconnected from SDR server")
  96. def add_event_dialog(self):
  97. #if not self.sock:
  98. # QMessageBox.warning(self, "Not Connected", "Please connect to the SDR server first")
  99. # return False
  100. add_event_dialog = QDialog(self)
  101. add_event_dialog.setWindowTitle("Add Event")
  102. layout = QFormLayout()
  103. labels = ["Start Time (ticks)", "Duration (ticks)", "Waveform Type", "time_front_rf_ttl (ticks)", "time_back_rf_ttl (ticks)", "time_front_adc_ttl (ticks)", "time_back_adc_ttl (ticks)", "time_start_rf (ticks)", "time_end_rf (ticks)", "rf_freq (Hz)"]
  104. self.event_inputs = {}
  105. for label in labels:
  106. if label == "Waveform Type":
  107. combo = QComboBox()
  108. combo.addItems(self.wavetypes)
  109. layout.addRow(QLabel(label), combo)
  110. self.event_inputs[label] = combo
  111. else:
  112. line_edit = QLineEdit()
  113. layout.addRow(QLabel(label), line_edit)
  114. self.event_inputs[label] = line_edit
  115. submit_button = QPushButton("Submit")
  116. layout.addWidget(submit_button)
  117. load_event_list_button = QPushButton("Load Event List")
  118. layout.addWidget(load_event_list_button)
  119. submit_button.clicked.connect(self.submit_event)
  120. load_event_list_button.clicked.connect(self.load_event_list)
  121. add_event_dialog.setLayout(layout)
  122. add_event_dialog.exec()
  123. def submit_event(self):
  124. try:
  125. msg = {
  126. "magic": 0xAA,
  127. "cmd": 0x10,
  128. "time": int(self.event_inputs["Start Time (ticks)"].text()),
  129. "duration": int(self.event_inputs["Duration (ticks)"].text()),
  130. "wave": self.event_inputs["Waveform Type"].currentText(),
  131. "time_front_rf_trigger": int(self.event_inputs["time_front_rf_ttl (ticks)"].text()),
  132. "time_back_rf_trigger": int(self.event_inputs["time_back_rf_ttl (ticks)"].text()),
  133. "time_front_adc_trigger": int(self.event_inputs["time_front_adc_ttl (ticks)"].text()),
  134. "time_back_adc_trigger": int(self.event_inputs["time_back_adc_ttl (ticks)"].text()),
  135. "rf_start": int(self.event_inputs["time_start_rf (ticks)"].text()),
  136. "rf_end": int(self.event_inputs["time_end_rf (ticks)"].text()),
  137. "carrier_freq": float(self.event_inputs["rf_freq (Hz)"].text())
  138. }
  139. msg = msgpack.packb(msg, use_bin_type=True)
  140. self.sock.sendall(msg)
  141. raw_resp = self.sock.recv(4096)
  142. if not raw_resp:
  143. raise Exception("No response from server")
  144. resp = msgpack.unpackb(raw_resp, raw=False)
  145. if resp["cmd"] != 0xFE or resp["code"] != 0x00000010:
  146. raise Exception(f"Server error: {resp['message']}")
  147. print("Event added successfully")
  148. QMessageBox.information(self, "Success", "Event added successfully")
  149. except Exception as e:
  150. print(f"Failed to add event: {e}")
  151. QMessageBox.critical(self, "Error", f"Failed to add event: {e}")
  152. def load_event_list(self):
  153. file_dialog = QFileDialog(self)
  154. file_dialog.setNameFilter("JSON Files (*.json)")
  155. if file_dialog.exec():
  156. selected_file = file_dialog.selectedFiles()[0]
  157. try:
  158. with open(selected_file, 'r') as f:
  159. event_list = json.load(f)
  160. for event_data in event_list:
  161. msg = {
  162. "magic": 0xAA,
  163. "cmd": 0x10,
  164. "time": event_data["time"],
  165. "duration": event_data["duration"],
  166. "wave": event_data["wave"],
  167. "time_front_rf_trigger": event_data["time_front_rf_trigger"],
  168. "time_back_rf_trigger": event_data["time_back_rf_trigger"],
  169. "time_front_adc_trigger": event_data["time_front_adc_trigger"],
  170. "time_back_adc_trigger": event_data["time_back_adc_trigger"],
  171. "rf_start": event_data["rf_start"],
  172. "rf_end": event_data["rf_end"],
  173. "carrier_freq": event_data["carrier_freq"]
  174. }
  175. packed_msg = msgpack.packb(msg, use_bin_type=True)
  176. self.sock.sendall(packed_msg)
  177. resp = self.sock.recv(4096)
  178. if not resp:
  179. raise Exception("No response from server")
  180. unpacked_resp = msgpack.unpackb(resp, raw=False)
  181. if unpacked_resp["cmd"] != 0xFE and unpacked_resp["code"] != 0x00000010:
  182. raise Exception(f"Server error: {unpacked_resp['message']}")
  183. print("Event list loaded successfully")
  184. QMessageBox.information(self, "Success", "Event list loaded successfully")
  185. self.add_event_dialog.close()
  186. except Exception as e:
  187. print(f"Failed to load event list: {e}")
  188. QMessageBox.critical(self, "Error", f"Failed to load event list: {e}")
  189. def add_wavetype_dialog(self):
  190. add_wave_dialog = QDialog(self)
  191. add_wave_dialog.setWindowTitle("Add Waveform Type")
  192. layout = QFormLayout()
  193. name_input = QLineEdit()
  194. layout.addRow(QLabel("Waveform Name"), name_input)
  195. self.file_input = QLineEdit()
  196. file_dialog_button = QPushButton("Browse")
  197. file_dialog_button.clicked.connect(lambda: self.open_wave_file_dialog(self.file_input))
  198. layout.addRow(QLabel("Waveform File"), self.file_input)
  199. layout.addWidget(file_dialog_button)
  200. submit_button = QPushButton("Submit")
  201. layout.addWidget(submit_button)
  202. submit_button.clicked.connect(lambda: self.submit_wavetype(name_input.text(), self.file_input.text(), add_wave_dialog))
  203. add_wave_dialog.setLayout(layout)
  204. add_wave_dialog.exec()
  205. def open_wave_file_dialog(self, file_input):
  206. file_dialog = QFileDialog(self)
  207. file_dialog.setNameFilter("All Files (*);;Binary Files (*.bin)")
  208. if file_dialog.exec():
  209. selected_file = file_dialog.selectedFiles()[0]
  210. file_input.setText(selected_file)
  211. def submit_wavetype(self, name, file_path, dialog):
  212. if not name:
  213. QMessageBox.warning(self, "Input Error", "Please enter a waveform name")
  214. return
  215. if name in self.wavetypes:
  216. QMessageBox.warning(self, "Input Error", "Waveform name already exists")
  217. return
  218. header_msg = {
  219. "magic": 0xAA,
  220. "cmd": 0x20,
  221. "wavetype": name,
  222. "total_length": 0, # Placeholder, will be updated after loading the waveform data
  223. "total_packets": 0, # Placeholder, will be updated after loading the waveform data
  224. }
  225. with open(file_path, 'rb') as f:
  226. # Load rows point from .bin file (first 2 bytes is Q-point, next 2 bytes is I-point)
  227. total_wavedata_q = np.fromfile(f, dtype=np.int16, count=-1, offset=0)
  228. total_wavedata_i = np.fromfile(f, dtype=np.int16, count=-1, offset=2)
  229. if len(total_wavedata_q) != len(total_wavedata_i):
  230. QMessageBox.warning(self, "Input Error", "In-phase and quadrature wave data must have the same length")
  231. return
  232. packet_length = 512 # Define packet length for sending waveform data
  233. header_msg["total_length"] = len(total_wavedata_q)
  234. header_msg["total_packets"] = len(total_wavedata_q) // packet_length + (1 if len(total_wavedata_q) % packet_length > 0 else 0)
  235. try:
  236. packed_header = msgpack.packb(header_msg, use_bin_type=True)
  237. self.sock.sendall(packed_header)
  238. raw_resp = self.sock.recv(4096)
  239. if not raw_resp:
  240. raise Exception("No response from server")
  241. resp = msgpack.unpackb(raw_resp, raw=False)
  242. if resp["cmd"] != 0xFE or resp["code"] != 0x00000001:
  243. raise Exception(f"Server error: {resp['message']}")
  244. except Exception as e:
  245. print(f"Failed to send waveform header: {e}")
  246. QMessageBox.critical(self, "Error", f"Failed to send waveform header: {e}")
  247. return
  248. try:
  249. for i in range(header_msg["total_packets"]):
  250. packet_length = min(512, len(total_wavedata_q) - i * 512)
  251. start_idx = i * 512
  252. end_idx = min((i + 1) * 512, len(total_wavedata_q))
  253. packet_data = {
  254. "magic": 0xAA,
  255. "cmd": 0x2C,
  256. "packet_index": i,
  257. "packet_length": packet_length,
  258. "wavedata_q": total_wavedata_q[start_idx:end_idx].tobytes(),
  259. "wavedata_i": total_wavedata_i[start_idx:end_idx].tobytes()
  260. }
  261. packed_packet = msgpack.packb(packet_data, use_bin_type=True)
  262. self.sock.sendall(packed_packet)
  263. raw_resp = self.sock.recv(4096)
  264. if not raw_resp:
  265. raise Exception("No response from server")
  266. resp = msgpack.unpackb(raw_resp, raw=False)
  267. if resp["cmd"] != 0xFE or resp["code"] != 0x00000002:
  268. raise Exception(f"Server error: {resp['message']}")
  269. self.sock.sendall(b"ACK") # Send ACK to server to indicate ready for next packet
  270. raw_resp = self.sock.recv(4096)
  271. if not raw_resp:
  272. raise Exception("No response from server after sending all packets")
  273. resp = msgpack.unpackb(raw_resp, raw=False)
  274. if resp["cmd"] != 0xFE or resp["code"] != 0x00000003:
  275. raise Exception(f"Server error after sending all packets: {resp['message']}")
  276. except Exception as e:
  277. print(f"Failed to send waveform data: {e}")
  278. QMessageBox.critical(self, "Error", f"Failed to send waveform data: {e}")
  279. return
  280. self.wavetypes.append(name)
  281. print(f"Added new waveform type: {name}")
  282. QMessageBox.information(self, "Success", f"Added new waveform type: {name}")
  283. dialog.close()
  284. def run_event_list(self):
  285. msg = {
  286. "magic": 0xAA,
  287. "cmd": 0x30,
  288. "interpolation_method": "linear" # Placeholder, can be extended to allow user selection of interpolation method
  289. }
  290. try:
  291. packed_msg = msgpack.packb(msg, use_bin_type=True)
  292. self.sock.sendall(packed_msg)
  293. print("Sent run event list command to server")
  294. except Exception as e:
  295. print(f"Failed to send run event list command: {e}")
  296. QMessageBox.critical(self, "Error", f"Failed to send run event list command: {e}")
  297. return
  298. raw_resp = self.sock.recv(4096)
  299. if not raw_resp:
  300. print("No response from server after sending run event list command")
  301. QMessageBox.warning(self, "No Response", "No response from server after sending run event list command")
  302. return
  303. resp = msgpack.unpackb(raw_resp, raw=False)
  304. if resp["cmd"] != 0xFE or resp["code"] != 0x00000032:
  305. print(f"Server error after sending run event list command: {resp['message']}")
  306. QMessageBox.warning(self, "Server Error", f"Server error after sending run event list command: {resp['message']}")
  307. return
  308. QMessageBox.information(self, "Success", "Run event list command sent to server")
  309. def clear_event_list(self):
  310. msg = {
  311. "magic": 0xAA,
  312. "cmd": 0x60
  313. }
  314. try:
  315. packed_msg = msgpack.packb(msg, use_bin_type=True)
  316. self.sock.sendall(packed_msg)
  317. print("Sent clear event list command to server")
  318. except Exception as e:
  319. print(f"Failed to send clear event list command: {e}")
  320. QMessageBox.critical(self, "Error", f"Failed to send clear event list command: {e}")
  321. return
  322. raw_resp = self.sock.recv(4096)
  323. if not raw_resp:
  324. print("No response from server after sending clear event list command")
  325. QMessageBox.warning(self, "No Response", "No response from server after sending clear event list command")
  326. return
  327. resp = msgpack.unpackb(raw_resp, raw=False)
  328. if resp["cmd"] != 0xFE or resp["code"] != 0x00000040:
  329. print(f"Server error after sending clear event list command: {resp['message']}")
  330. QMessageBox.warning(self, "Server Error", f"Server error after sending clear event list command: {resp['message']}")
  331. return
  332. QMessageBox.information(self, "Success", "Clear event list command sent to server")
  333. def plot_sim_data(self):
  334. self.plot_canvas.axes.clear()
  335. self.plot_canvas.axes.plot(self.channel_data["time"], self.channel_data["channel_g"], label="Gradient")
  336. self.plot_canvas.axes.plot(self.channel_data["time"], self.channel_data["channel_rf"], label="RF")
  337. self.plot_canvas.axes.plot(self.channel_data["time"], self.channel_data["channel_adc_ttl"], label="ADC TTL")
  338. self.plot_canvas.axes.plot(self.channel_data["time"], self.channel_data["channel_rf_ttl"], label="RF TTL")
  339. self.plot_canvas.axes.set_xlabel("Time (ticks)")
  340. self.plot_canvas.axes.set_ylabel("Signal Value")
  341. self.plot_canvas.axes.set_title("Simulation Data from SDR Server")
  342. self.plot_canvas.axes.legend()
  343. self.plot_canvas.draw()
  344. def receive_sim_data(self):
  345. msg = {
  346. "magic": 0xAA,
  347. "cmd": 0x40
  348. }
  349. try:
  350. packed_msg = msgpack.packb(msg, use_bin_type=True)
  351. self.sock.sendall(packed_msg)
  352. print("Sent request to receive simulation data from server")
  353. except Exception as e:
  354. print(f"Failed to send request for simulation data: {e}")
  355. QMessageBox.critical(self, "Error", f"Failed to send request for simulation data: {e}")
  356. return
  357. try:
  358. raw_resp = self.sock.recv(4096)
  359. if not raw_resp:
  360. raise Exception("No response from server after requesting simulation data")
  361. resp = msgpack.unpackb(raw_resp, raw=False)
  362. if resp["cmd"] != 0xFE or resp["code"] != 0x00000043:
  363. raise Exception(f"Server error after requesting simulation data: {resp['message']}")
  364. except Exception as e:
  365. print(f"Failed to receive simulation data header: {e}")
  366. QMessageBox.critical(self, "Error", f"Failed to receive simulation data header: {e}")
  367. return
  368. print(f"Server is ready to send simulation data: {resp['message']}")
  369. self.sock.sendall(b"ACK") # Send ACK to server to indicate ready to receive simulation data
  370. while True:
  371. try:
  372. raw_packet = self.sock.recv(8192)
  373. if not raw_packet:
  374. print("No more data from server, finished receiving simulation data")
  375. break
  376. packet = msgpack.unpackb(raw_packet, raw=False)
  377. if packet["cmd"] == 0xFE and packet["code"] == 0x00000044:
  378. print(f"Received end of simulation data message from server: {self.channel_data}") # Debug print to check received data
  379. self.plot_sim_data()
  380. QMessageBox.information(self, "Simulation Data Received", "All simulation data packets received from server")
  381. break
  382. if packet["cmd"] != 0x33:
  383. print(f"Received non-simulation data message from server: {packet}")
  384. continue
  385. channel = packet["channel"]
  386. packet_index = packet["packet_index"]
  387. total_packets = packet["total_packets"]
  388. data_length = packet["data_length"]
  389. print(f"Received simulation data packet {packet_index}/{total_packets} for channel {channel} with length {data_length} bytes")
  390. if channel not in self.channel_data:
  391. print(f"Unknown channel {channel} in received simulation data packet")
  392. continue
  393. points = None
  394. if channel == "time":
  395. points = np.frombuffer(packet["data"], dtype=np.int64)
  396. else:
  397. # Append received data to channel data
  398. points = np.frombuffer(packet["data"], dtype=np.int16)
  399. print(f"Packet data preview (first 16 points): {points[:16]}") # Debug print to check packet data
  400. self.channel_data[channel] = np.append(self.channel_data[channel], points)
  401. # Send ACK to server to indicate ready for next packet
  402. self.sock.sendall(b"ACK")
  403. except Exception as e:
  404. print(f"Failed to receive or process simulation data packet: {e}")
  405. QMessageBox.critical(self, "Error", f"Failed to receive or process simulation data packet: {e}")
  406. return
  407. app = QApplication([])
  408. client = SDRClient()
  409. client.show()
  410. #plt.ion() # Enable interactive mode for plotting
  411. #plt.show()
  412. app.exec()