| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  | # Copyright 2015-2017 Espressif Systems (Shanghai) PTE LTD | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  | # you may not use this file except in compliance with the License. | 
					
						
							|  |  |  | # You may obtain a copy of the License at | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #     http:#www.apache.org/licenses/LICENSE-2.0 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  | # distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  | # See the License for the specific language governing permissions and | 
					
						
							|  |  |  | # limitations under the License. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | DUT provides 3 major groups of features: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | * DUT port feature, provide basic open/close/read/write features | 
					
						
							|  |  |  | * DUT tools, provide extra methods to control the device, like download and start app | 
					
						
							|  |  |  | * DUT expect method, provide features for users to check DUT outputs | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The current design of DUT have 3 classes for one DUT: BaseDUT, DUTPort, DUTTool. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | * BaseDUT class: | 
					
						
							|  |  |  |     * defines methods DUT port and DUT tool need to overwrite | 
					
						
							|  |  |  |     * provide the expect methods and some other methods based on DUTPort | 
					
						
							|  |  |  | * DUTPort class: | 
					
						
							|  |  |  |     * inherent from BaseDUT class | 
					
						
							|  |  |  |     * implements the port features by overwriting port methods defined in BaseDUT | 
					
						
							|  |  |  | * DUTTool class: | 
					
						
							|  |  |  |     * inherent from one of the DUTPort class | 
					
						
							|  |  |  |     * implements the tools features by overwriting tool methods defined in BaseDUT | 
					
						
							|  |  |  |     * could add some new methods provided by the tool | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This module implements the BaseDUT class and one of the port class SerialDUT. | 
					
						
							|  |  |  | User should implement their DUTTool classes. | 
					
						
							|  |  |  | If they using different port then need to implement their DUTPort class as well. | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import time | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | import threading | 
					
						
							|  |  |  | import copy | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | import functools | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import serial | 
					
						
							|  |  |  | from serial.tools import list_ports | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-08 12:27:57 +08:00
										 |  |  | import Utility | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  | if sys.version_info[0] == 2: | 
					
						
							|  |  |  |     import Queue as _queue | 
					
						
							|  |  |  | else: | 
					
						
							|  |  |  |     import queue as _queue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ExpectTimeout(ValueError): | 
					
						
							|  |  |  |     """ timeout for expect method """ | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class UnsupportedExpectItem(ValueError): | 
					
						
							|  |  |  |     """ expect item not supported by the expect method """ | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _expect_lock(func): | 
					
						
							|  |  |  |     @functools.wraps(func) | 
					
						
							|  |  |  |     def handler(self, *args, **kwargs): | 
					
						
							|  |  |  |         with self.expect_lock: | 
					
						
							|  |  |  |             ret = func(self, *args, **kwargs) | 
					
						
							|  |  |  |         return ret | 
					
						
							|  |  |  |     return handler | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-08 12:27:57 +08:00
										 |  |  | def _decode_data(data): | 
					
						
							|  |  |  |     """ for python3, if the data is bytes, then decode it to string """ | 
					
						
							|  |  |  |     if isinstance(data, bytes): | 
					
						
							|  |  |  |         # convert bytes to string | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             data = data.decode("utf-8", "ignore") | 
					
						
							|  |  |  |         except UnicodeDecodeError: | 
					
						
							|  |  |  |             data = data.decode("iso8859-1", ) | 
					
						
							|  |  |  |     return data | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-16 22:16:03 +08:00
										 |  |  | def _pattern_to_string(pattern): | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         ret = "RegEx: " + pattern.pattern | 
					
						
							|  |  |  |     except AttributeError: | 
					
						
							|  |  |  |         ret = pattern | 
					
						
							|  |  |  |     return ret | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  | class _DataCache(_queue.Queue): | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     Data cache based on Queue. Allow users to process data cache based on bytes instead of Queue." | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, maxsize=0): | 
					
						
							|  |  |  |         _queue.Queue.__init__(self, maxsize=maxsize) | 
					
						
							|  |  |  |         self.data_cache = str() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-16 22:16:03 +08:00
										 |  |  |     def _move_from_queue_to_cache(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         move all of the available data in the queue to cache | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :return: True if moved any item from queue to data cache, else False | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         ret = False | 
					
						
							|  |  |  |         while True: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 self.data_cache += _decode_data(self.get(0)) | 
					
						
							|  |  |  |                 ret = True | 
					
						
							|  |  |  |             except _queue.Empty: | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  |         return ret | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 17:33:39 +08:00
										 |  |  |     def get_data(self, timeout=0.0): | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         get a copy of data from cache. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param timeout: timeout for waiting new queue item | 
					
						
							|  |  |  |         :return: copy of data cache | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # make sure timeout is non-negative | 
					
						
							|  |  |  |         if timeout < 0: | 
					
						
							|  |  |  |             timeout = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-16 22:16:03 +08:00
										 |  |  |         ret = self._move_from_queue_to_cache() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not ret: | 
					
						
							|  |  |  |             # we only wait for new data if we can't provide a new data_cache | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 data = self.get(timeout=timeout) | 
					
						
							|  |  |  |                 self.data_cache += _decode_data(data) | 
					
						
							|  |  |  |             except _queue.Empty: | 
					
						
							|  |  |  |                 # don't do anything when on update for cache | 
					
						
							|  |  |  |                 pass | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |         return copy.deepcopy(self.data_cache) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def flush(self, index=0xFFFFFFFF): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         flush data from cache. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param index: if < 0 then don't do flush, otherwise flush data before index | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # first add data in queue to cache | 
					
						
							|  |  |  |         self.get_data() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if index > 0: | 
					
						
							|  |  |  |             self.data_cache = self.data_cache[index:] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _RecvThread(threading.Thread): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-08 12:27:57 +08:00
										 |  |  |     PERFORMANCE_PATTERN = re.compile(r"\[Performance]\[(\w+)]: ([^\r\n]+)\r?\n") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |     def __init__(self, read, data_cache): | 
					
						
							|  |  |  |         super(_RecvThread, self).__init__() | 
					
						
							|  |  |  |         self.exit_event = threading.Event() | 
					
						
							|  |  |  |         self.setDaemon(True) | 
					
						
							|  |  |  |         self.read = read | 
					
						
							|  |  |  |         self.data_cache = data_cache | 
					
						
							| 
									
										
										
										
											2017-11-08 12:27:57 +08:00
										 |  |  |         # cache the last line of recv data for collecting performance | 
					
						
							|  |  |  |         self._line_cache = str() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def collect_performance(self, data): | 
					
						
							|  |  |  |         """ collect performance """ | 
					
						
							|  |  |  |         if data: | 
					
						
							|  |  |  |             decoded_data = _decode_data(data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             matches = self.PERFORMANCE_PATTERN.findall(self._line_cache + decoded_data) | 
					
						
							|  |  |  |             for match in matches: | 
					
						
							|  |  |  |                 Utility.console_log("[Performance][{}]: {}".format(match[0], match[1]), | 
					
						
							|  |  |  |                                     color="orange") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # cache incomplete line to later process | 
					
						
							|  |  |  |             lines = decoded_data.splitlines(True) | 
					
						
							|  |  |  |             last_line = lines[-1] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if last_line[-1] != "\n": | 
					
						
							|  |  |  |                 if len(lines) == 1: | 
					
						
							|  |  |  |                     # only one line and the line is not finished, then append this to cache | 
					
						
							|  |  |  |                     self._line_cache += lines[-1] | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     # more than one line and not finished, replace line cache | 
					
						
							|  |  |  |                     self._line_cache = lines[-1] | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 # line finishes, flush cache | 
					
						
							|  |  |  |                 self._line_cache = str() | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def run(self): | 
					
						
							|  |  |  |         while not self.exit_event.isSet(): | 
					
						
							|  |  |  |             data = self.read(1000) | 
					
						
							|  |  |  |             if data: | 
					
						
							|  |  |  |                 self.data_cache.put(data) | 
					
						
							| 
									
										
										
										
											2017-11-08 12:27:57 +08:00
										 |  |  |                 self.collect_performance(data) | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def exit(self): | 
					
						
							|  |  |  |         self.exit_event.set() | 
					
						
							|  |  |  |         self.join() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class BaseDUT(object): | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     :param name: application defined name for port | 
					
						
							|  |  |  |     :param port: comport name, used to create DUT port | 
					
						
							|  |  |  |     :param log_file: log file name | 
					
						
							|  |  |  |     :param app: test app instance | 
					
						
							|  |  |  |     :param kwargs: extra args for DUT to create ports | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DEFAULT_EXPECT_TIMEOUT = 5 | 
					
						
							| 
									
										
										
										
											2018-06-15 17:33:39 +08:00
										 |  |  |     MAX_EXPECT_FAILURES_TO_SAVED = 10 | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name, port, log_file, app, **kwargs): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.expect_lock = threading.Lock() | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.port = port | 
					
						
							|  |  |  |         self.log_file = log_file | 
					
						
							|  |  |  |         self.app = app | 
					
						
							|  |  |  |         self.data_cache = _DataCache() | 
					
						
							|  |  |  |         self.receive_thread = None | 
					
						
							| 
									
										
										
										
											2018-06-15 17:33:39 +08:00
										 |  |  |         self.expect_failures = [] | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |         # open and start during init | 
					
						
							|  |  |  |         self.open() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __str__(self): | 
					
						
							|  |  |  |         return "DUT({}: {})".format(self.name, str(self.port)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 17:33:39 +08:00
										 |  |  |     def _save_expect_failure(self, pattern, data, start_time): | 
					
						
							|  |  |  |         self.expect_failures.insert(0, {"pattern": pattern, "data": data, | 
					
						
							|  |  |  |                                         "start": start_time, "end": time.time()}) | 
					
						
							|  |  |  |         self.expect_failures = self.expect_failures[:self.MAX_EXPECT_FAILURES_TO_SAVED] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |     # define for methods need to be overwritten by Port | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def list_available_ports(cls): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         list all available ports. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subclass (port) must overwrite this method. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :return: list of available comports | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _port_open(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         open the port. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subclass (port) must overwrite this method. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _port_read(self, size=1): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         read form port. This method should not blocking for long time, otherwise receive thread can not exit. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subclass (port) must overwrite this method. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param size: max size to read. | 
					
						
							|  |  |  |         :return: read data. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _port_write(self, data): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         write to port. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subclass (port) must overwrite this method. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param data: data to write | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _port_close(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         close port. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subclass (port) must overwrite this method. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # methods that need to be overwritten by Tool | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def confirm_dut(cls, port, app, **kwargs): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         confirm if it's a DUT, usually used by auto detecting DUT in by Env config. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subclass (tool) must overwrite this method. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param port: comport | 
					
						
							|  |  |  |         :param app: app instance | 
					
						
							|  |  |  |         :return: True or False | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def start_app(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         usually after we got DUT, we need to do some extra works to let App start. | 
					
						
							|  |  |  |         For example, we need to reset->download->reset to let IDF application start on DUT. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subclass (tool) must overwrite this method. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # methods that features raw port methods | 
					
						
							|  |  |  |     def open(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         open port and create thread to receive data. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         self._port_open() | 
					
						
							|  |  |  |         self.receive_thread = _RecvThread(self._port_read, self.data_cache) | 
					
						
							|  |  |  |         self.receive_thread.start() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def close(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         close receive thread and then close port. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if self.receive_thread: | 
					
						
							|  |  |  |             self.receive_thread.exit() | 
					
						
							|  |  |  |         self._port_close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def write(self, data, eol="\r\n", flush=True): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         :param data: data | 
					
						
							|  |  |  |         :param eol: end of line pattern. | 
					
						
							|  |  |  |         :param flush: if need to flush received data cache before write data. | 
					
						
							|  |  |  |                       usually we need to flush data before write, | 
					
						
							|  |  |  |                       make sure processing outputs generated by wrote. | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # do flush before write | 
					
						
							|  |  |  |         if flush: | 
					
						
							|  |  |  |             self.data_cache.flush() | 
					
						
							|  |  |  |         # do write if cache | 
					
						
							| 
									
										
										
										
											2018-01-31 16:57:40 +08:00
										 |  |  |         if data is not None: | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |             self._port_write(data + eol if eol else data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @_expect_lock | 
					
						
							|  |  |  |     def read(self, size=0xFFFFFFFF): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         read(size=0xFFFFFFFF) | 
					
						
							|  |  |  |         read raw data. NOT suggested to use this method. | 
					
						
							|  |  |  |         Only use it if expect method doesn't meet your requirement. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param size: read size. default read all data | 
					
						
							|  |  |  |         :return: read data | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         data = self.data_cache.get_data(0)[:size] | 
					
						
							|  |  |  |         self.data_cache.flush(size) | 
					
						
							|  |  |  |         return data | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # expect related methods | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							|  |  |  |     def _expect_str(data, pattern): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         protected method. check if string is matched in data cache. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param data: data to process | 
					
						
							|  |  |  |         :param pattern: string | 
					
						
							|  |  |  |         :return: pattern if match succeed otherwise None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         index = data.find(pattern) | 
					
						
							|  |  |  |         if index != -1: | 
					
						
							|  |  |  |             ret = pattern | 
					
						
							|  |  |  |             index += len(pattern) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             ret = None | 
					
						
							|  |  |  |         return ret, index | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							|  |  |  |     def _expect_re(data, pattern): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         protected method. check if re pattern is matched in data cache | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param data: data to process | 
					
						
							|  |  |  |         :param pattern: compiled RegEx pattern | 
					
						
							|  |  |  |         :return: match groups if match succeed otherwise None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         ret = None | 
					
						
							|  |  |  |         match = pattern.search(data) | 
					
						
							|  |  |  |         if match: | 
					
						
							|  |  |  |             ret = match.groups() | 
					
						
							|  |  |  |             index = match.end() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             index = -1 | 
					
						
							|  |  |  |         return ret, index | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     EXPECT_METHOD = [ | 
					
						
							|  |  |  |         [type(re.compile("")), "_expect_re"], | 
					
						
							|  |  |  |         [str, "_expect_str"], | 
					
						
							|  |  |  |     ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_expect_method(self, pattern): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         protected method. get expect method according to pattern type. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param pattern: expect pattern, string or compiled RegEx | 
					
						
							|  |  |  |         :return: ``_expect_str`` or ``_expect_re`` | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         for expect_method in self.EXPECT_METHOD: | 
					
						
							|  |  |  |             if isinstance(pattern, expect_method[0]): | 
					
						
							|  |  |  |                 method = expect_method[1] | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             raise UnsupportedExpectItem() | 
					
						
							|  |  |  |         return self.__getattribute__(method) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @_expect_lock | 
					
						
							|  |  |  |     def expect(self, pattern, timeout=DEFAULT_EXPECT_TIMEOUT): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         expect(pattern, timeout=DEFAULT_EXPECT_TIMEOUT) | 
					
						
							|  |  |  |         expect received data on DUT match the pattern. will raise exception when expect timeout. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :raise ExpectTimeout: failed to find the pattern before timeout | 
					
						
							|  |  |  |         :raise UnsupportedExpectItem: pattern is not string or compiled RegEx | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param pattern: string or compiled RegEx(string pattern) | 
					
						
							|  |  |  |         :param timeout: timeout for expect | 
					
						
							|  |  |  |         :return: string if pattern is string; matched groups if pattern is RegEx | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         method = self._get_expect_method(pattern) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # non-blocking get data for first time | 
					
						
							|  |  |  |         data = self.data_cache.get_data(0) | 
					
						
							|  |  |  |         start_time = time.time() | 
					
						
							|  |  |  |         while True: | 
					
						
							|  |  |  |             ret, index = method(data, pattern) | 
					
						
							|  |  |  |             if ret is not None or time.time() - start_time > timeout: | 
					
						
							|  |  |  |                 self.data_cache.flush(index) | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  |             # wait for new data from cache | 
					
						
							|  |  |  |             data = self.data_cache.get_data(time.time() + timeout - start_time) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ret is None: | 
					
						
							| 
									
										
										
										
											2018-06-15 17:33:39 +08:00
										 |  |  |             pattern = _pattern_to_string(pattern) | 
					
						
							|  |  |  |             self._save_expect_failure(pattern, data, start_time) | 
					
						
							|  |  |  |             raise ExpectTimeout(self.name + ": " + pattern) | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |         return ret | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _expect_multi(self, expect_all, expect_item_list, timeout): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         protected method. internal logical for expect multi. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param expect_all: True or False, expect all items in the list or any in the list | 
					
						
							|  |  |  |         :param expect_item_list: expect item list | 
					
						
							|  |  |  |         :param timeout: timeout | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         def process_expected_item(item_raw): | 
					
						
							|  |  |  |             # convert item raw data to standard dict | 
					
						
							|  |  |  |             item = { | 
					
						
							|  |  |  |                 "pattern": item_raw[0] if isinstance(item_raw, tuple) else item_raw, | 
					
						
							|  |  |  |                 "method": self._get_expect_method(item_raw[0] if isinstance(item_raw, tuple) | 
					
						
							|  |  |  |                                                   else item_raw), | 
					
						
							|  |  |  |                 "callback": item_raw[1] if isinstance(item_raw, tuple) else None, | 
					
						
							|  |  |  |                 "index": -1, | 
					
						
							|  |  |  |                 "ret": None, | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return item | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         expect_items = [process_expected_item(x) for x in expect_item_list] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # non-blocking get data for first time | 
					
						
							|  |  |  |         data = self.data_cache.get_data(0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         start_time = time.time() | 
					
						
							|  |  |  |         matched_expect_items = list() | 
					
						
							|  |  |  |         while True: | 
					
						
							|  |  |  |             for expect_item in expect_items: | 
					
						
							|  |  |  |                 if expect_item not in matched_expect_items: | 
					
						
							|  |  |  |                     # exclude those already matched | 
					
						
							|  |  |  |                     expect_item["ret"], expect_item["index"] = \ | 
					
						
							|  |  |  |                         expect_item["method"](data, expect_item["pattern"]) | 
					
						
							|  |  |  |                     if expect_item["ret"] is not None: | 
					
						
							|  |  |  |                         # match succeed for one item | 
					
						
							|  |  |  |                         matched_expect_items.append(expect_item) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # if expect all, then all items need to be matched, | 
					
						
							|  |  |  |             # else only one item need to matched | 
					
						
							|  |  |  |             if expect_all: | 
					
						
							| 
									
										
										
										
											2018-01-16 22:16:03 +08:00
										 |  |  |                 match_succeed = len(matched_expect_items) == len(expect_items) | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 match_succeed = True if matched_expect_items else False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if time.time() - start_time > timeout or match_succeed: | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 data = self.data_cache.get_data(time.time() + timeout - start_time) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if match_succeed: | 
					
						
							|  |  |  |             # do callback and flush matched data cache | 
					
						
							|  |  |  |             slice_index = -1 | 
					
						
							|  |  |  |             for expect_item in matched_expect_items: | 
					
						
							|  |  |  |                 # trigger callback | 
					
						
							|  |  |  |                 if expect_item["callback"]: | 
					
						
							|  |  |  |                     expect_item["callback"](expect_item["ret"]) | 
					
						
							|  |  |  |                 slice_index = max(slice_index, expect_item["index"]) | 
					
						
							|  |  |  |             # flush already matched data | 
					
						
							|  |  |  |             self.data_cache.flush(slice_index) | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2018-06-15 17:33:39 +08:00
										 |  |  |             pattern = str([_pattern_to_string(x["pattern"]) for x in expect_items]) | 
					
						
							|  |  |  |             self._save_expect_failure(pattern, data, start_time) | 
					
						
							|  |  |  |             raise ExpectTimeout(self.name + ": " + pattern) | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @_expect_lock | 
					
						
							|  |  |  |     def expect_any(self, *expect_items, **timeout): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         expect_any(*expect_items, timeout=DEFAULT_TIMEOUT) | 
					
						
							|  |  |  |         expect any of the patterns. | 
					
						
							|  |  |  |         will call callback (if provided) if pattern match succeed and then return. | 
					
						
							|  |  |  |         will pass match result to the callback. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :raise ExpectTimeout: failed to match any one of the expect items before timeout | 
					
						
							|  |  |  |         :raise UnsupportedExpectItem: pattern in expect_item is not string or compiled RegEx | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :arg expect_items: one or more expect items. | 
					
						
							|  |  |  |                            string, compiled RegEx pattern or (string or RegEx(string pattern), callback) | 
					
						
							|  |  |  |         :keyword timeout: timeout for expect | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # to be compatible with python2 | 
					
						
							|  |  |  |         # in python3 we can write f(self, *expect_items, timeout=DEFAULT_TIMEOUT) | 
					
						
							|  |  |  |         if "timeout" not in timeout: | 
					
						
							|  |  |  |             timeout["timeout"] = self.DEFAULT_EXPECT_TIMEOUT | 
					
						
							|  |  |  |         return self._expect_multi(False, expect_items, **timeout) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @_expect_lock | 
					
						
							|  |  |  |     def expect_all(self, *expect_items, **timeout): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         expect_all(*expect_items, timeout=DEFAULT_TIMEOUT) | 
					
						
							|  |  |  |         expect all of the patterns. | 
					
						
							|  |  |  |         will call callback (if provided) if all pattern match succeed and then return. | 
					
						
							|  |  |  |         will pass match result to the callback. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :raise ExpectTimeout: failed to match all of the expect items before timeout | 
					
						
							|  |  |  |         :raise UnsupportedExpectItem: pattern in expect_item is not string or compiled RegEx | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :arg expect_items: one or more expect items. | 
					
						
							|  |  |  |                            string, compiled RegEx pattern or (string or RegEx(string pattern), callback) | 
					
						
							|  |  |  |         :keyword timeout: timeout for expect | 
					
						
							|  |  |  |         :return: None | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # to be compatible with python2 | 
					
						
							|  |  |  |         # in python3 we can write f(self, *expect_items, timeout=DEFAULT_TIMEOUT) | 
					
						
							|  |  |  |         if "timeout" not in timeout: | 
					
						
							|  |  |  |             timeout["timeout"] = self.DEFAULT_EXPECT_TIMEOUT | 
					
						
							|  |  |  |         return self._expect_multi(True, expect_items, **timeout) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 17:33:39 +08:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def _format_ts(ts): | 
					
						
							|  |  |  |         return "{}:{}".format(time.strftime("%m-%d %H:%M:%S", time.localtime(ts)), str(ts % 1)[2:5]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def print_debug_info(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Print debug info of current DUT. Currently we will print debug info for expect failures. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Utility.console_log("DUT debug info for DUT: {}:".format(self.name), color="orange") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for failure in self.expect_failures: | 
					
						
							|  |  |  |             Utility.console_log("\t[pattern]: {}\r\n\t[data]: {}\r\n\t[time]: {} - {}\r\n" | 
					
						
							|  |  |  |                                 .format(failure["pattern"], failure["data"], | 
					
						
							|  |  |  |                                         self._format_ts(failure["start"]), self._format_ts(failure["end"])), | 
					
						
							|  |  |  |                                 color="orange") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class SerialDUT(BaseDUT): | 
					
						
							|  |  |  |     """ serial with logging received data feature """ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DEFAULT_UART_CONFIG = { | 
					
						
							|  |  |  |         "baudrate": 115200, | 
					
						
							|  |  |  |         "bytesize": serial.EIGHTBITS, | 
					
						
							|  |  |  |         "parity": serial.PARITY_NONE, | 
					
						
							|  |  |  |         "stopbits": serial.STOPBITS_ONE, | 
					
						
							|  |  |  |         "timeout": 0.05, | 
					
						
							|  |  |  |         "xonxoff": False, | 
					
						
							|  |  |  |         "rtscts": False, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name, port, log_file, app, **kwargs): | 
					
						
							|  |  |  |         self.port_inst = None | 
					
						
							|  |  |  |         self.serial_configs = self.DEFAULT_UART_CONFIG.copy() | 
					
						
							|  |  |  |         self.serial_configs.update(kwargs) | 
					
						
							|  |  |  |         super(SerialDUT, self).__init__(name, port, log_file, app, **kwargs) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 17:33:39 +08:00
										 |  |  |     def _format_data(self, data): | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         format data for logging. do decode and add timestamp. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :param data: raw data from read | 
					
						
							|  |  |  |         :return: formatted data (str) | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2018-06-15 17:33:39 +08:00
										 |  |  |         timestamp = "[{}]".format(self._format_ts(time.time())) | 
					
						
							| 
									
										
										
										
											2018-01-31 16:57:40 +08:00
										 |  |  |         formatted_data = timestamp.encode() + b"\r\n" + data + b"\r\n" | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |         return formatted_data | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _port_open(self): | 
					
						
							|  |  |  |         self.port_inst = serial.Serial(self.port, **self.serial_configs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _port_close(self): | 
					
						
							|  |  |  |         self.port_inst.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _port_read(self, size=1): | 
					
						
							|  |  |  |         data = self.port_inst.read(size) | 
					
						
							|  |  |  |         if data: | 
					
						
							| 
									
										
										
										
											2018-01-31 16:57:40 +08:00
										 |  |  |             with open(self.log_file, "ab+") as _log_file: | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |                 _log_file.write(self._format_data(data)) | 
					
						
							|  |  |  |         return data | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _port_write(self, data): | 
					
						
							| 
									
										
										
										
											2018-01-31 16:57:40 +08:00
										 |  |  |         if isinstance(data, str): | 
					
						
							|  |  |  |             data = data.encode() | 
					
						
							| 
									
										
										
										
											2017-10-10 10:44:55 +08:00
										 |  |  |         self.port_inst.write(data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def list_available_ports(cls): | 
					
						
							|  |  |  |         return [x.device for x in list_ports.comports()] |