You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							167 lines
						
					
					
						
							5.3 KiB
						
					
					
				
			
		
		
	
	
							167 lines
						
					
					
						
							5.3 KiB
						
					
					
				| #include <QDebug>
 | |
| #include <QEventLoop>
 | |
| 
 | |
| #include "catch2/catch.hpp"
 | |
| #include "selfdrive/common/util.h"
 | |
| #include "selfdrive/ui/replay/replay.h"
 | |
| #include "selfdrive/ui/replay/util.h"
 | |
| 
 | |
| const QString DEMO_ROUTE = "4cf7a6ad03080c90|2021-09-29--13-46-36";
 | |
| const std::string TEST_RLOG_URL = "https://commadataci.blob.core.windows.net/openpilotci/0c94aa1e1296d7c6/2021-05-05--19-48-37/0/rlog.bz2";
 | |
| const std::string TEST_RLOG_CHECKSUM = "5b966d4bb21a100a8c4e59195faeb741b975ccbe268211765efd1763d892bfb3";
 | |
| 
 | |
| TEST_CASE("httpMultiPartDownload") {
 | |
|   char filename[] = "/tmp/XXXXXX";
 | |
|   close(mkstemp(filename));
 | |
| 
 | |
|   const size_t chunk_size = 5 * 1024 * 1024;
 | |
|   std::string content;
 | |
|   SECTION("download to file") {
 | |
|     REQUIRE(httpDownload(TEST_RLOG_URL, filename, chunk_size));
 | |
|     content = util::read_file(filename);
 | |
|   }
 | |
|   SECTION("download to buffer") {
 | |
|     content = httpGet(TEST_RLOG_URL, chunk_size);
 | |
|     REQUIRE(!content.empty());
 | |
|   }
 | |
|   REQUIRE(content.size() == 9112651);
 | |
|   REQUIRE(sha256(content) == TEST_RLOG_CHECKSUM);
 | |
| }
 | |
| 
 | |
| int random_int(int min, int max) {
 | |
|   std::random_device dev;
 | |
|   std::mt19937 rng(dev());
 | |
|   std::uniform_int_distribution<std::mt19937::result_type> dist(min, max);
 | |
|   return dist(rng);
 | |
| }
 | |
| 
 | |
| TEST_CASE("FileReader") {
 | |
|   auto enable_local_cache = GENERATE(true, false);
 | |
|   std::string cache_file = cacheFilePath(TEST_RLOG_URL);
 | |
|   system(("rm " + cache_file + " -f").c_str());
 | |
| 
 | |
|   FileReader reader(enable_local_cache);
 | |
|   std::string content = reader.read(TEST_RLOG_URL);
 | |
|   REQUIRE(sha256(content) == TEST_RLOG_CHECKSUM);
 | |
|   if (enable_local_cache) {
 | |
|     REQUIRE(sha256(util::read_file(cache_file)) == TEST_RLOG_CHECKSUM);
 | |
|   } else {
 | |
|     REQUIRE(util::file_exists(cache_file) == false);
 | |
|   }
 | |
| }
 | |
| 
 | |
| TEST_CASE("LogReader") {
 | |
|   SECTION("corrupt log") {
 | |
|     FileReader reader(true);
 | |
|     std::string corrupt_content = reader.read(TEST_RLOG_URL);
 | |
|     corrupt_content.resize(corrupt_content.length() / 2);
 | |
|     LogReader log;
 | |
|     REQUIRE(log.load((std::byte *)corrupt_content.data(), corrupt_content.size()));
 | |
|     REQUIRE(log.events.size() > 0);
 | |
|   }
 | |
| }
 | |
| 
 | |
| TEST_CASE("Segment") {
 | |
|   auto flags = GENERATE(REPLAY_FLAG_DCAM | REPLAY_FLAG_ECAM, REPLAY_FLAG_QCAMERA);
 | |
|   Route demo_route(DEMO_ROUTE);
 | |
|   REQUIRE(demo_route.load());
 | |
|   REQUIRE(demo_route.segments().size() == 11);
 | |
| 
 | |
|   QEventLoop loop;
 | |
|   Segment segment(0, demo_route.at(0), flags);
 | |
|   QObject::connect(&segment, &Segment::loadFinished, [&]() {
 | |
|     REQUIRE(segment.isLoaded() == true);
 | |
|     REQUIRE(segment.log != nullptr);
 | |
|     REQUIRE(segment.frames[RoadCam] != nullptr);
 | |
|     if (flags & REPLAY_FLAG_DCAM) {
 | |
|       REQUIRE(segment.frames[DriverCam] != nullptr);
 | |
|     }
 | |
|     if (flags & REPLAY_FLAG_ECAM) {
 | |
|       REQUIRE(segment.frames[WideRoadCam] != nullptr);
 | |
|     }
 | |
| 
 | |
|     // test LogReader & FrameReader
 | |
|     REQUIRE(segment.log->events.size() > 0);
 | |
|     REQUIRE(std::is_sorted(segment.log->events.begin(), segment.log->events.end(), Event::lessThan()));
 | |
| 
 | |
|     for (auto cam : ALL_CAMERAS) {
 | |
|       auto &fr = segment.frames[cam];
 | |
|       if (!fr) continue;
 | |
| 
 | |
|       if (cam == RoadCam || cam == WideRoadCam) {
 | |
|         REQUIRE(fr->getFrameCount() == 1200);
 | |
|       }
 | |
|       std::unique_ptr<uint8_t[]> rgb_buf = std::make_unique<uint8_t[]>(fr->getRGBSize());
 | |
|       std::unique_ptr<uint8_t[]> yuv_buf = std::make_unique<uint8_t[]>(fr->getYUVSize());
 | |
|       // sequence get 50 frames
 | |
|       for (int i = 0; i < 50; ++i) {
 | |
|         REQUIRE(fr->get(i, rgb_buf.get(), yuv_buf.get()));
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     loop.quit();
 | |
|   });
 | |
|   loop.exec();
 | |
| }
 | |
| 
 | |
| // helper class for unit tests
 | |
| class TestReplay : public Replay {
 | |
|  public:
 | |
|   TestReplay(const QString &route, uint8_t flags = REPLAY_FLAG_NO_FILE_CACHE) : Replay(route, {}, {}, nullptr, flags) {}
 | |
|   void test_seek();
 | |
|   void testSeekTo(int seek_to);
 | |
| };
 | |
| 
 | |
| void TestReplay::testSeekTo(int seek_to) {
 | |
|   seekTo(seek_to, false);
 | |
| 
 | |
|   while (true) {
 | |
|     std::unique_lock lk(stream_lock_);
 | |
|     stream_cv_.wait(lk, [=]() { return events_updated_ == true; });
 | |
|     events_updated_ = false;
 | |
|     if (cur_mono_time_ != route_start_ts_ + seek_to * 1e9) {
 | |
|       // wake up by the previous merging, skip it.
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     Event cur_event(cereal::Event::Which::INIT_DATA, cur_mono_time_);
 | |
|     auto eit = std::upper_bound(events_->begin(), events_->end(), &cur_event, Event::lessThan());
 | |
|     if (eit == events_->end()) {
 | |
|       qDebug() << "waiting for events...";
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     REQUIRE(std::is_sorted(events_->begin(), events_->end(), Event::lessThan()));
 | |
|     const int seek_to_segment = seek_to / 60;
 | |
|     const int event_seconds = ((*eit)->mono_time - route_start_ts_) / 1e9;
 | |
|     current_segment_ = event_seconds / 60;
 | |
|     INFO("seek to [" << seek_to << "s segment " << seek_to_segment << "], events [" << event_seconds << "s segment" << current_segment_ << "]");
 | |
|     REQUIRE(event_seconds >= seek_to);
 | |
|     if (event_seconds > seek_to) {
 | |
|       auto it = segments_.lower_bound(seek_to_segment);
 | |
|       REQUIRE(it->first == current_segment_);
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void TestReplay::test_seek() {
 | |
|   // create a dummy stream thread
 | |
|   stream_thread_ = new QThread(this);
 | |
|   QEventLoop loop;
 | |
|   std::thread thread = std::thread([&]() {
 | |
|     for (int i = 0; i < 50; ++i) {
 | |
|       testSeekTo(random_int(0, 3 * 60));
 | |
|     }
 | |
|     loop.quit();
 | |
|   });
 | |
|   loop.exec();
 | |
|   thread.join();
 | |
| }
 | |
| 
 | |
| TEST_CASE("Replay") {
 | |
|   auto flag = GENERATE(REPLAY_FLAG_NO_FILE_CACHE, REPLAY_FLAG_NONE);
 | |
|   TestReplay replay(DEMO_ROUTE, flag);
 | |
|   REQUIRE(replay.load());
 | |
|   replay.test_seek();
 | |
| }
 | |
| 
 |