diff --git a/Tools/autotest/common.py b/Tools/autotest/common.py index 94bfa2f8bcdacf..387146e8a93407 100644 --- a/Tools/autotest/common.py +++ b/Tools/autotest/common.py @@ -28,6 +28,7 @@ import tempfile import threading import enum +from pathlib import Path from MAVProxy.modules.lib import mp_util from MAVProxy.modules.lib import mp_elevation @@ -3450,34 +3451,13 @@ def HIGH_LATENCY2_links(self): if ex is not None: raise ex - def LogDownload(self): - '''Test Onboard Log Download''' - if self.is_tracker(): - # tracker starts armed, which is annoying - return - self.progress("Ensuring we have contents we care about") - self.set_parameter("LOG_FILE_DSRMROT", 1) - self.set_parameter("LOG_DISARMED", 0) - self.reboot_sitl() - original_log_list = self.log_list() - for i in range(0, 10): - self.wait_ready_to_arm() - self.arm_vehicle() - self.delay_sim_time(1) - self.disarm_vehicle() - new_log_list = self.log_list() - new_log_count = len(new_log_list) - len(original_log_list) - if new_log_count != 10: - raise NotAchievedException("Expected exactly 10 new logs got %u (%s) to (%s)" % - (new_log_count, original_log_list, new_log_list)) - self.progress("Directory contents: %s" % str(new_log_list)) - + def download_full_log_list(self, print_logs=True): tstart = self.get_sim_time() self.mav.mav.log_request_list_send(self.sysid_thismav(), 1, # target component 0, - 0xff) - logs = [] + 0xffff) + logs = {} last_id = None num_logs = None while True: @@ -3487,10 +3467,11 @@ def LogDownload(self): m = self.mav.recv_match(type='LOG_ENTRY', blocking=True, timeout=1) - self.progress("Received (%s)" % str(m)) + if print_logs: + self.progress("Received (%s)" % str(m)) if m is None: continue - logs.append(m) + logs[m.id] = m if last_id is None: if m.num_logs == 0: # caller to guarantee this works: @@ -3517,7 +3498,127 @@ def LogDownload(self): timeout=2) if m is not None: raise NotAchievedException("Received extra LOG_ENTRY?!") + return logs + + def test_log_download2(self): + """Test log wrapping.""" + if self.is_tracker(): + # tracker starts armed, which is annoying + return + self.progress("Ensuring we have contents we care about") + self.set_parameter("LOG_FILE_DSRMROT", 1) + self.set_parameter("LOG_DISARMED", 0) + self.reboot_sitl() + logspath = Path("logs") + + def create_num_logs(num_logs, logsdir, clear_logsdir=True): + if clear_logsdir: + shutil.rmtree(logsdir, ignore_errors=True) + logsdir.mkdir() + lastlogfile_path = logsdir / Path("LASTLOG.TXT") + self.progress(f"Add LASTLOG.TXT file with counter at {num_logs}") + with open(lastlogfile_path, 'w') as lastlogfile: + lastlogfile.write(f"{num_logs}\n") + self.progress(f"Create fakelogs from 1 to {num_logs} on logs directory") + for ii in range(1, num_logs + 1): + new_log = logsdir / Path(f"{str(ii).zfill(8)}.BIN") + with open(new_log, 'w+') as logfile: + logfile.write(f"I AM LOG {ii}\n") + logfile.write(f'1' * ii) + + def verify_logs(test_log_num): + try: + wrap = False + offset = 0 + max_logs_num = int(self.get_parameter("LOG_MAX_FILES")) + if test_log_num > max_logs_num: + wrap = True + offset = test_log_num - max_logs_num + test_log_num = max_logs_num + logs_dict = self.download_full_log_list(print_logs=False) + if len(logs_dict) != test_log_num: + raise NotAchievedException(f"Didn't get the full log list, expect {test_log_num} got {len(logs_dict)}") + self.progress("Checking logs size are matching") + start_log = offset if wrap else 1 + for ii in range(start_log, test_log_num + 1 - offset): + log_i = logspath / Path(f"{str(ii + offset).zfill(8)}.BIN") + if logs_dict[ii].size != log_i.stat().st_size: + logs_dict = self.download_full_log_list(print_logs=False) + if logs_dict[ii].size != log_i.stat().st_size: # sometimes we don't have finish writing the log, so get it again prevent failure + raise NotAchievedException(f"Log{ii} size mismatch : {logs_dict[ii].size} vs {log_i.stat().st_size}") + if wrap: + self.progress("Checking wrapped logs size are matching") + for ii in range(1, offset): + log_i = logspath / Path(f"{str(ii).zfill(8)}.BIN") + if logs_dict[test_log_num + 1 - offset + ii].size != log_i.stat().st_size: + self.progress(f"{logs_dict[test_log_num + 1 - offset + ii]}") + raise NotAchievedException(f"Log{test_log_num + 1 - offset + ii} size mismatch : {logs_dict[test_log_num + 1 - offset + ii].size} vs {log_i.stat().st_size}") + except NotAchievedException as e: + shutil.rmtree(logspath, ignore_errors=True) + logspath.mkdir() + with open(logspath / Path("LASTLOG.TXT"), 'w') as lastlogfile: + lastlogfile.write(f"1\n") + raise e + + def add_and_verify_log(test_log_num): + self.wait_ready_to_arm() + self.arm_vehicle() + self.delay_sim_time(1) + self.disarm_vehicle() + self.delay_sim_time(10) + verify_logs(test_log_num + 1) + + def create_and_verify_logs(test_log_num, clear_logsdir=True): + self.progress(f"Test {test_log_num} logs") + create_num_logs(test_log_num, logspath, clear_logsdir) + self.reboot_sitl() + verify_logs(test_log_num) + self.start_subsubtest("Adding one more log") + add_and_verify_log(test_log_num) + + self.start_subtest("Checking log list match with filesystem info") + create_and_verify_logs(500) + create_and_verify_logs(10) + create_and_verify_logs(1) + + self.start_subtest("Change LOG_MAX_FILES and Checking log list match with filesystem info") + self.set_parameter("LOG_MAX_FILES", 250) + create_and_verify_logs(250) + self.set_parameter("LOG_MAX_FILES", 1) + create_and_verify_logs(1) + + self.start_subtest("Change LOG_MAX_FILES, don't clear old logs and Checking log list match with filesystem info") + self.set_parameter("LOG_MAX_FILES", 500) + create_and_verify_logs(500) + self.set_parameter("LOG_MAX_FILES", 250) + create_and_verify_logs(250, clear_logsdir=False) + + # cleanup + shutil.rmtree(logspath, ignore_errors=True) + + def test_log_download(self): + """Test Onboard Log Download.""" + if self.is_tracker(): + # tracker starts armed, which is annoying + return + self.progress("Ensuring we have contents we care about") + self.set_parameter("LOG_FILE_DSRMROT", 1) + self.set_parameter("LOG_DISARMED", 0) + self.reboot_sitl() + original_log_list = self.log_list() + for i in range(0, 10): + self.wait_ready_to_arm() + self.arm_vehicle() + self.delay_sim_time(1) + self.disarm_vehicle() + new_log_list = self.log_list() + new_log_count = len(new_log_list) - len(original_log_list) + if new_log_count != 10: + raise NotAchievedException("Expected exactly 10 new logs got %u (%s) to (%s)" % + (new_log_count, original_log_list, new_log_list)) + self.progress("Directory contents: %s" % str(new_log_list)) + self.download_full_log_list() log_id = 5 ofs = 6 count = 2 @@ -13438,6 +13539,7 @@ def tests(self): self.Logging, self.GetCapabilities, self.InitialMode, + self.test_log_download2, ] def post_tests_announcements(self):