micropython-stubber 1.20.4__py3-none-any.whl → 1.20.6__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.
Files changed (152) hide show
  1. {micropython_stubber-1.20.4.dist-info → micropython_stubber-1.20.6.dist-info}/LICENSE +30 -30
  2. {micropython_stubber-1.20.4.dist-info → micropython_stubber-1.20.6.dist-info}/METADATA +4 -4
  3. micropython_stubber-1.20.6.dist-info/RECORD +159 -0
  4. mpflash/README.md +184 -184
  5. mpflash/libusb_flash.ipynb +203 -203
  6. mpflash/mpflash/add_firmware.py +98 -98
  7. mpflash/mpflash/ask_input.py +236 -226
  8. mpflash/mpflash/bootloader/__init__.py +37 -0
  9. mpflash/mpflash/bootloader/manual.py +102 -0
  10. mpflash/mpflash/bootloader/micropython.py +10 -0
  11. mpflash/mpflash/bootloader/touch1200.py +45 -0
  12. mpflash/mpflash/cli_download.py +129 -128
  13. mpflash/mpflash/cli_flash.py +219 -212
  14. mpflash/mpflash/cli_group.py +98 -92
  15. mpflash/mpflash/cli_list.py +81 -77
  16. mpflash/mpflash/cli_main.py +41 -38
  17. mpflash/mpflash/common.py +164 -151
  18. mpflash/mpflash/config.py +47 -31
  19. mpflash/mpflash/connected.py +74 -74
  20. mpflash/mpflash/download.py +360 -361
  21. mpflash/mpflash/downloaded.py +129 -129
  22. mpflash/mpflash/errors.py +9 -5
  23. mpflash/mpflash/flash.py +52 -69
  24. mpflash/mpflash/flash_esp.py +59 -59
  25. mpflash/mpflash/flash_stm32.py +24 -24
  26. mpflash/mpflash/flash_stm32_cube.py +111 -111
  27. mpflash/mpflash/flash_stm32_dfu.py +101 -101
  28. mpflash/mpflash/flash_uf2.py +67 -67
  29. mpflash/mpflash/flash_uf2_boardid.py +15 -15
  30. mpflash/mpflash/flash_uf2_linux.py +123 -123
  31. mpflash/mpflash/flash_uf2_macos.py +34 -37
  32. mpflash/mpflash/flash_uf2_windows.py +34 -34
  33. mpflash/mpflash/list.py +89 -89
  34. mpflash/mpflash/logger.py +41 -41
  35. mpflash/mpflash/mpboard_id/__init__.py +93 -93
  36. mpflash/mpflash/mpboard_id/add_boards.py +255 -255
  37. mpflash/mpflash/mpboard_id/board.py +37 -37
  38. mpflash/mpflash/mpboard_id/board_id.py +86 -86
  39. mpflash/mpflash/mpboard_id/store.py +43 -43
  40. mpflash/mpflash/mpremoteboard/__init__.py +221 -221
  41. mpflash/mpflash/mpremoteboard/mpy_fw_info.py +141 -141
  42. mpflash/mpflash/mpremoteboard/runner.py +140 -140
  43. mpflash/mpflash/uf2disk.py +12 -12
  44. mpflash/mpflash/vendor/basicgit.py +288 -288
  45. mpflash/mpflash/vendor/click_aliases.py +91 -0
  46. mpflash/mpflash/vendor/dfu.py +165 -165
  47. mpflash/mpflash/vendor/pydfu.py +605 -605
  48. mpflash/mpflash/vendor/readme.md +2 -2
  49. mpflash/mpflash/vendor/versions.py +119 -117
  50. mpflash/mpflash/worklist.py +170 -170
  51. mpflash/poetry.lock +1588 -1623
  52. mpflash/pyproject.toml +60 -60
  53. mpflash/stm32_udev_rules.md +62 -62
  54. stubber/__init__.py +3 -3
  55. stubber/basicgit.py +294 -288
  56. stubber/board/board_info.csv +193 -193
  57. stubber/board/boot.py +34 -34
  58. stubber/board/createstubs.py +986 -987
  59. stubber/board/createstubs_db.py +825 -826
  60. stubber/board/createstubs_db_min.py +331 -331
  61. stubber/board/createstubs_db_mpy.mpy +0 -0
  62. stubber/board/createstubs_lvgl.py +741 -741
  63. stubber/board/createstubs_lvgl_min.py +741 -741
  64. stubber/board/createstubs_mem.py +766 -767
  65. stubber/board/createstubs_mem_min.py +306 -306
  66. stubber/board/createstubs_mem_mpy.mpy +0 -0
  67. stubber/board/createstubs_min.py +294 -294
  68. stubber/board/createstubs_mpy.mpy +0 -0
  69. stubber/board/fw_info.py +141 -141
  70. stubber/board/info.py +183 -183
  71. stubber/board/main.py +19 -19
  72. stubber/board/modulelist.txt +247 -247
  73. stubber/board/pyrightconfig.json +34 -34
  74. stubber/bulk/mcu_stubber.py +454 -455
  75. stubber/codemod/_partials/__init__.py +48 -50
  76. stubber/codemod/_partials/db_main.py +147 -147
  77. stubber/codemod/_partials/lvgl_main.py +77 -77
  78. stubber/codemod/_partials/modules_reader.py +80 -80
  79. stubber/codemod/add_comment.py +53 -53
  80. stubber/codemod/add_method.py +65 -65
  81. stubber/codemod/board.py +317 -317
  82. stubber/codemod/enrich.py +145 -145
  83. stubber/codemod/merge_docstub.py +284 -284
  84. stubber/codemod/modify_list.py +54 -54
  85. stubber/codemod/utils.py +57 -57
  86. stubber/commands/build_cmd.py +94 -94
  87. stubber/commands/cli.py +51 -51
  88. stubber/commands/clone_cmd.py +66 -66
  89. stubber/commands/config_cmd.py +29 -29
  90. stubber/commands/enrich_folder_cmd.py +70 -70
  91. stubber/commands/get_core_cmd.py +69 -69
  92. stubber/commands/get_docstubs_cmd.py +87 -87
  93. stubber/commands/get_frozen_cmd.py +112 -112
  94. stubber/commands/get_mcu_cmd.py +56 -56
  95. stubber/commands/merge_cmd.py +66 -66
  96. stubber/commands/publish_cmd.py +119 -119
  97. stubber/commands/stub_cmd.py +30 -30
  98. stubber/commands/switch_cmd.py +54 -54
  99. stubber/commands/variants_cmd.py +48 -48
  100. stubber/cst_transformer.py +178 -178
  101. stubber/data/board_info.csv +193 -193
  102. stubber/data/board_info.json +1729 -1729
  103. stubber/data/micropython_tags.csv +15 -15
  104. stubber/data/requirements-core-micropython.txt +38 -38
  105. stubber/data/requirements-core-pycopy.txt +39 -39
  106. stubber/downloader.py +36 -36
  107. stubber/freeze/common.py +68 -68
  108. stubber/freeze/freeze_folder.py +69 -69
  109. stubber/freeze/freeze_manifest_2.py +113 -113
  110. stubber/freeze/get_frozen.py +127 -127
  111. stubber/get_cpython.py +101 -101
  112. stubber/get_lobo.py +59 -59
  113. stubber/minify.py +418 -418
  114. stubber/publish/bump.py +86 -86
  115. stubber/publish/candidates.py +262 -283
  116. stubber/publish/database.py +18 -18
  117. stubber/publish/defaults.py +45 -45
  118. stubber/publish/enums.py +24 -30
  119. stubber/publish/helpers.py +29 -29
  120. stubber/publish/merge_docstubs.py +130 -130
  121. stubber/publish/missing_class_methods.py +49 -49
  122. stubber/publish/package.py +146 -177
  123. stubber/publish/pathnames.py +51 -51
  124. stubber/publish/publish.py +120 -121
  125. stubber/publish/pypi.py +38 -38
  126. stubber/publish/stubpackage.py +1029 -1029
  127. stubber/rst/__init__.py +9 -9
  128. stubber/rst/classsort.py +77 -77
  129. stubber/rst/lookup.py +530 -530
  130. stubber/rst/output_dict.py +401 -401
  131. stubber/rst/reader.py +822 -823
  132. stubber/rst/report_return.py +69 -69
  133. stubber/rst/rst_utils.py +540 -540
  134. stubber/stubber.py +38 -38
  135. stubber/stubs_from_docs.py +90 -90
  136. stubber/tools/manifestfile.py +610 -610
  137. stubber/tools/readme.md +5 -5
  138. stubber/update_fallback.py +117 -117
  139. stubber/update_module_list.py +123 -125
  140. stubber/utils/__init__.py +5 -5
  141. stubber/utils/config.py +127 -127
  142. stubber/utils/makeversionhdr.py +54 -54
  143. stubber/utils/manifest.py +92 -92
  144. stubber/utils/post.py +79 -79
  145. stubber/utils/repos.py +157 -154
  146. stubber/utils/stubmaker.py +139 -139
  147. stubber/utils/typed_config_toml.py +77 -77
  148. stubber/utils/versions.py +128 -120
  149. stubber/variants.py +106 -106
  150. micropython_stubber-1.20.4.dist-info/RECORD +0 -154
  151. {micropython_stubber-1.20.4.dist-info → micropython_stubber-1.20.6.dist-info}/WHEEL +0 -0
  152. {micropython_stubber-1.20.4.dist-info → micropython_stubber-1.20.6.dist-info}/entry_points.txt +0 -0
@@ -1,221 +1,221 @@
1
- """
2
- Module to run mpremote commands, and retry on failure or timeout
3
- """
4
-
5
- import sys
6
- import time
7
- from pathlib import Path
8
- from typing import List, Optional, Union
9
-
10
- import serial.tools.list_ports
11
- from loguru import logger as log
12
- from rich.progress import track
13
- from tenacity import retry, stop_after_attempt, wait_fixed
14
-
15
- from mpflash.errors import MPFlashError
16
- from mpflash.mpboard_id.board_id import find_board_id_by_description
17
- from mpflash.mpremoteboard.runner import run
18
-
19
- ###############################################################################################
20
- # TODO : make this a bit nicer
21
- HERE = Path(__file__).parent
22
-
23
- OK = 0
24
- ERROR = -1
25
- RETRIES = 3
26
- ###############################################################################################
27
-
28
-
29
- class MPRemoteBoard:
30
- """Class to run mpremote commands"""
31
-
32
- def __init__(self, serialport: str = "", update: bool = False):
33
- """
34
- Initialize MPRemoteBoard object.
35
-
36
- Parameters:
37
- - serialport (str): The serial port to connect to. Default is an empty string.
38
- - update (bool): Whether to update the MCU information. Default is False.
39
- """
40
- self.serialport = serialport
41
- self.firmware = {}
42
-
43
- self.connected = False
44
- self.path: Optional[Path] = None
45
- self.family = "unknown"
46
- self.description = ""
47
- self.version = ""
48
- self.port = ""
49
- self.board = ""
50
- self.cpu = ""
51
- self.arch = ""
52
- self.mpy = ""
53
- self.build = ""
54
- if update:
55
- self.get_mcu_info()
56
-
57
- def __str__(self):
58
- """
59
- Return a string representation of the MPRemoteBoard object.
60
-
61
- Returns:
62
- - str: The string representation of the object.
63
- """
64
- return f"MPRemoteBoard({self.serialport}, {self.family} {self.port}, {self.board}, {self.version})"
65
-
66
- @staticmethod
67
- def connected_boards(bluetooth: bool = False, description: bool = False) -> List[str]:
68
- # TODO: rename to connected_comports
69
- """
70
- Get a list of connected comports.
71
-
72
- Parameters:
73
- - bluetooth (bool): Whether to include Bluetooth ports. Default is False.
74
-
75
- Returns:
76
- - List[str]: A list of connected board ports.
77
- """
78
- comports = serial.tools.list_ports.comports()
79
-
80
- if not bluetooth:
81
- # filter out bluetooth ports
82
- comports = [p for p in comports if "bluetooth" not in p.description.lower()]
83
- comports = [p for p in comports if "BTHENUM" not in p.hwid]
84
- if description:
85
- output = [
86
- f"{p.device} {(p.manufacturer + ' ') if p.manufacturer and not p.description.startswith(p.manufacturer) else ''}{p.description}"
87
- for p in comports
88
- ]
89
- else:
90
- output = [p.device for p in comports]
91
-
92
- if sys.platform == "win32":
93
- # Windows sort of comports by number - but fallback to device name
94
- return sorted(output, key=lambda x: int(x.split()[0][3:]) if x.split()[0][3:].isdigit() else x)
95
- # sort by device name
96
- return sorted(output)
97
-
98
- @retry(stop=stop_after_attempt(RETRIES), wait=wait_fixed(1), reraise=True) # type: ignore ## retry_error_cls=ConnectionError,
99
- def get_mcu_info(self, timeout: int = 2):
100
- """
101
- Get MCU information from the connected board.
102
-
103
- Parameters:
104
- - timeout (int): The timeout value in seconds. Default is 2.
105
-
106
- Raises:
107
- - ConnectionError: If failed to get mcu_info for the serial port.
108
- """
109
- rc, result = self.run_command(
110
- ["run", str(HERE / "mpy_fw_info.py")],
111
- no_info=True,
112
- timeout=timeout,
113
- )
114
- if rc != OK:
115
- raise ConnectionError(f"Failed to get mcu_info for {self.serialport}")
116
- # Ok we have the info, now parse it
117
- s = result[0].strip()
118
- if s.startswith("{") and s.endswith("}"):
119
- info = eval(s)
120
- self.family = info["family"]
121
- self.version = info["version"]
122
- self.build = info["build"]
123
- self.port = info["port"]
124
- self.cpu = info["cpu"]
125
- self.arch = info["arch"]
126
- self.mpy = info["mpy"]
127
- self.description = descr = info["board"]
128
- pos = descr.rfind(" with")
129
- short_descr = descr[:pos].strip() if pos != -1 else ""
130
- if board_name := find_board_id_by_description(descr, short_descr, version=self.version):
131
- self.board = board_name
132
- else:
133
- self.board = "UNKNOWN_BOARD"
134
-
135
- def disconnect(self) -> bool:
136
- """
137
- Disconnect from a board.
138
-
139
- Returns:
140
- - bool: True if successfully disconnected, False otherwise.
141
- """
142
- if not self.connected:
143
- return True
144
- if not self.serialport:
145
- log.error("No port connected")
146
- self.connected = False
147
- return False
148
- log.info(f"Disconnecting from {self.serialport}")
149
- result = self.run_command(["disconnect"])[0] == OK
150
- self.connected = False
151
- return result
152
-
153
- @retry(stop=stop_after_attempt(RETRIES), wait=wait_fixed(2), reraise=True)
154
- def run_command(
155
- self,
156
- cmd: Union[str, List[str]],
157
- *,
158
- log_errors: bool = True,
159
- no_info: bool = False,
160
- timeout: int = 60,
161
- **kwargs,
162
- ):
163
- """
164
- Run mpremote with the given command.
165
-
166
- Parameters:
167
- - cmd (Union[str, List[str]]): The command to run, either a string or a list of strings.
168
- - log_errors (bool): Whether to log errors. Default is True.
169
- - no_info (bool): Whether to skip printing info. Default is False.
170
- - timeout (int): The timeout value in seconds. Default is 60.
171
-
172
- Returns:
173
- - bool: True if the command succeeded, False otherwise.
174
- """
175
- if isinstance(cmd, str):
176
- cmd = cmd.split(" ")
177
- prefix = [sys.executable, "-m", "mpremote"]
178
- if self.serialport:
179
- prefix += ["connect", self.serialport]
180
- # if connected add resume to keep state between commands
181
- if self.connected:
182
- prefix += ["resume"]
183
- cmd = prefix + cmd
184
- log.debug(" ".join(cmd))
185
- result = run(cmd, timeout, log_errors, no_info, **kwargs)
186
- self.connected = result[0] == OK
187
- return result
188
-
189
- @retry(stop=stop_after_attempt(RETRIES), wait=wait_fixed(1))
190
- def mip_install(self, name: str) -> bool:
191
- """
192
- Install a micropython package.
193
-
194
- Parameters:
195
- - name (str): The name of the package to install.
196
-
197
- Returns:
198
- - bool: True if the installation succeeded, False otherwise.
199
- """
200
- # install createstubs to the board
201
- cmd = ["mip", "install", name]
202
- result = self.run_command(cmd)[0] == OK
203
- self.connected = True
204
- return result
205
-
206
- def wait_for_restart(self, timeout: int = 10):
207
- """wait for the board to restart"""
208
- for _ in track(
209
- range(timeout),
210
- description="Waiting for the board to restart",
211
- transient=True,
212
- get_time=lambda: time.time(),
213
- show_speed=False,
214
- refresh_per_second=1,
215
- ):
216
- time.sleep(1)
217
- try:
218
- self.get_mcu_info()
219
- break
220
- except (ConnectionError, MPFlashError):
221
- pass
1
+ """
2
+ Module to run mpremote commands, and retry on failure or timeout
3
+ """
4
+
5
+ import sys
6
+ import time
7
+ from pathlib import Path
8
+ from typing import List, Optional, Union
9
+
10
+ import serial.tools.list_ports
11
+ from loguru import logger as log
12
+ from rich.progress import track
13
+ from tenacity import retry, stop_after_attempt, wait_fixed
14
+
15
+ from mpflash.errors import MPFlashError
16
+ from mpflash.mpboard_id.board_id import find_board_id_by_description
17
+ from mpflash.mpremoteboard.runner import run
18
+
19
+ ###############################################################################################
20
+ # TODO : make this a bit nicer
21
+ HERE = Path(__file__).parent
22
+
23
+ OK = 0
24
+ ERROR = -1
25
+ RETRIES = 3
26
+ ###############################################################################################
27
+
28
+
29
+ class MPRemoteBoard:
30
+ """Class to run mpremote commands"""
31
+
32
+ def __init__(self, serialport: str = "", update: bool = False):
33
+ """
34
+ Initialize MPRemoteBoard object.
35
+
36
+ Parameters:
37
+ - serialport (str): The serial port to connect to. Default is an empty string.
38
+ - update (bool): Whether to update the MCU information. Default is False.
39
+ """
40
+ self.serialport = serialport
41
+ self.firmware = {}
42
+
43
+ self.connected = False
44
+ self.path: Optional[Path] = None
45
+ self.family = "unknown"
46
+ self.description = ""
47
+ self.version = ""
48
+ self.port = ""
49
+ self.board = ""
50
+ self.cpu = ""
51
+ self.arch = ""
52
+ self.mpy = ""
53
+ self.build = ""
54
+ if update:
55
+ self.get_mcu_info()
56
+
57
+ def __str__(self):
58
+ """
59
+ Return a string representation of the MPRemoteBoard object.
60
+
61
+ Returns:
62
+ - str: The string representation of the object.
63
+ """
64
+ return f"MPRemoteBoard({self.serialport}, {self.family} {self.port}, {self.board}, {self.version})"
65
+
66
+ @staticmethod
67
+ def connected_boards(bluetooth: bool = False, description: bool = False) -> List[str]:
68
+ # TODO: rename to connected_comports
69
+ """
70
+ Get a list of connected comports.
71
+
72
+ Parameters:
73
+ - bluetooth (bool): Whether to include Bluetooth ports. Default is False.
74
+
75
+ Returns:
76
+ - List[str]: A list of connected board ports.
77
+ """
78
+ comports = serial.tools.list_ports.comports()
79
+
80
+ if not bluetooth:
81
+ # filter out bluetooth ports
82
+ comports = [p for p in comports if "bluetooth" not in p.description.lower()]
83
+ comports = [p for p in comports if "BTHENUM" not in p.hwid]
84
+ if description:
85
+ output = [
86
+ f"{p.device} {(p.manufacturer + ' ') if p.manufacturer and not p.description.startswith(p.manufacturer) else ''}{p.description}"
87
+ for p in comports
88
+ ]
89
+ else:
90
+ output = [p.device for p in comports]
91
+
92
+ if sys.platform == "win32":
93
+ # Windows sort of comports by number - but fallback to device name
94
+ return sorted(output, key=lambda x: int(x.split()[0][3:]) if x.split()[0][3:].isdigit() else x)
95
+ # sort by device name
96
+ return sorted(output)
97
+
98
+ @retry(stop=stop_after_attempt(RETRIES), wait=wait_fixed(1), reraise=True) # type: ignore ## retry_error_cls=ConnectionError,
99
+ def get_mcu_info(self, timeout: int = 2):
100
+ """
101
+ Get MCU information from the connected board.
102
+
103
+ Parameters:
104
+ - timeout (int): The timeout value in seconds. Default is 2.
105
+
106
+ Raises:
107
+ - ConnectionError: If failed to get mcu_info for the serial port.
108
+ """
109
+ rc, result = self.run_command(
110
+ ["run", str(HERE / "mpy_fw_info.py")],
111
+ no_info=True,
112
+ timeout=timeout,
113
+ )
114
+ if rc != OK:
115
+ raise ConnectionError(f"Failed to get mcu_info for {self.serialport}")
116
+ # Ok we have the info, now parse it
117
+ s = result[0].strip()
118
+ if s.startswith("{") and s.endswith("}"):
119
+ info = eval(s)
120
+ self.family = info["family"]
121
+ self.version = info["version"]
122
+ self.build = info["build"]
123
+ self.port = info["port"]
124
+ self.cpu = info["cpu"]
125
+ self.arch = info["arch"]
126
+ self.mpy = info["mpy"]
127
+ self.description = descr = info["board"]
128
+ pos = descr.rfind(" with")
129
+ short_descr = descr[:pos].strip() if pos != -1 else ""
130
+ if board_name := find_board_id_by_description(descr, short_descr, version=self.version):
131
+ self.board = board_name
132
+ else:
133
+ self.board = "UNKNOWN_BOARD"
134
+
135
+ def disconnect(self) -> bool:
136
+ """
137
+ Disconnect from a board.
138
+
139
+ Returns:
140
+ - bool: True if successfully disconnected, False otherwise.
141
+ """
142
+ if not self.connected:
143
+ return True
144
+ if not self.serialport:
145
+ log.error("No port connected")
146
+ self.connected = False
147
+ return False
148
+ log.info(f"Disconnecting from {self.serialport}")
149
+ result = self.run_command(["disconnect"])[0] == OK
150
+ self.connected = False
151
+ return result
152
+
153
+ @retry(stop=stop_after_attempt(RETRIES), wait=wait_fixed(2), reraise=True)
154
+ def run_command(
155
+ self,
156
+ cmd: Union[str, List[str]],
157
+ *,
158
+ log_errors: bool = True,
159
+ no_info: bool = False,
160
+ timeout: int = 60,
161
+ **kwargs,
162
+ ):
163
+ """
164
+ Run mpremote with the given command.
165
+
166
+ Parameters:
167
+ - cmd (Union[str, List[str]]): The command to run, either a string or a list of strings.
168
+ - log_errors (bool): Whether to log errors. Default is True.
169
+ - no_info (bool): Whether to skip printing info. Default is False.
170
+ - timeout (int): The timeout value in seconds. Default is 60.
171
+
172
+ Returns:
173
+ - bool: True if the command succeeded, False otherwise.
174
+ """
175
+ if isinstance(cmd, str):
176
+ cmd = cmd.split(" ")
177
+ prefix = [sys.executable, "-m", "mpremote"]
178
+ if self.serialport:
179
+ prefix += ["connect", self.serialport]
180
+ # if connected add resume to keep state between commands
181
+ if self.connected:
182
+ prefix += ["resume"]
183
+ cmd = prefix + cmd
184
+ log.debug(" ".join(cmd))
185
+ result = run(cmd, timeout, log_errors, no_info, **kwargs)
186
+ self.connected = result[0] == OK
187
+ return result
188
+
189
+ @retry(stop=stop_after_attempt(RETRIES), wait=wait_fixed(1))
190
+ def mip_install(self, name: str) -> bool:
191
+ """
192
+ Install a micropython package.
193
+
194
+ Parameters:
195
+ - name (str): The name of the package to install.
196
+
197
+ Returns:
198
+ - bool: True if the installation succeeded, False otherwise.
199
+ """
200
+ # install createstubs to the board
201
+ cmd = ["mip", "install", name]
202
+ result = self.run_command(cmd)[0] == OK
203
+ self.connected = True
204
+ return result
205
+
206
+ def wait_for_restart(self, timeout: int = 10):
207
+ """wait for the board to restart"""
208
+ for _ in track(
209
+ range(timeout),
210
+ description="Waiting for the board to restart",
211
+ transient=True,
212
+ get_time=lambda: time.time(),
213
+ show_speed=False,
214
+ refresh_per_second=1,
215
+ ):
216
+ time.sleep(1)
217
+ try:
218
+ self.get_mcu_info()
219
+ break
220
+ except (ConnectionError, MPFlashError):
221
+ pass