2019-05-05 01:53:02 +00:00
|
|
|
/* PRS1 Unit Tests
|
|
|
|
*
|
2021-11-02 20:34:12 +00:00
|
|
|
* Copyright (c) 2019-2022 The OSCAR Team
|
2019-05-05 01:53:02 +00:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
|
|
|
* License. See the file COPYING in the main directory of the source code
|
|
|
|
* for more details. */
|
|
|
|
|
2019-05-03 01:51:56 +00:00
|
|
|
#include "prs1tests.h"
|
2019-05-05 01:53:02 +00:00
|
|
|
#include "sessiontests.h"
|
|
|
|
|
2021-05-31 18:53:23 +00:00
|
|
|
#include "../SleepLib/loader_plugins/prs1_loader.h"
|
|
|
|
#include "../SleepLib/loader_plugins/prs1_parser.h"
|
2021-09-01 19:31:06 +00:00
|
|
|
#include "../SleepLib/importcontext.h"
|
2021-05-31 18:53:23 +00:00
|
|
|
|
2019-05-05 01:53:02 +00:00
|
|
|
#define TESTDATA_PATH "./testdata/"
|
2021-10-26 18:43:14 +00:00
|
|
|
//#define TEST_OSCAR_CALCS
|
2019-05-05 01:53:02 +00:00
|
|
|
|
|
|
|
static PRS1Loader* s_loader = nullptr;
|
|
|
|
static void iterateTestCards(const QString & root, void (*action)(const QString &));
|
2019-05-18 23:20:36 +00:00
|
|
|
static QString prs1OutputPath(const QString & inpath, const QString & serial, const QString & basename, const QString & suffix);
|
2019-05-05 01:53:02 +00:00
|
|
|
static QString prs1OutputPath(const QString & inpath, const QString & serial, int session, const QString & suffix);
|
2019-05-03 01:51:56 +00:00
|
|
|
|
|
|
|
void PRS1Tests::initTestCase(void)
|
|
|
|
{
|
2020-01-29 14:10:29 +00:00
|
|
|
p_profile = new Profile(TESTDATA_PATH "profile/", false);
|
2019-05-05 01:53:02 +00:00
|
|
|
|
2021-10-26 18:43:14 +00:00
|
|
|
#ifdef TEST_OSCAR_CALCS
|
|
|
|
p_pref = new Preferences("Preferences");
|
|
|
|
p_pref->Open();
|
|
|
|
AppSetting = new AppWideSetting(p_pref);
|
|
|
|
#endif
|
|
|
|
|
2019-05-05 01:53:02 +00:00
|
|
|
schema::init();
|
|
|
|
PRS1Loader::Register();
|
|
|
|
s_loader = dynamic_cast<PRS1Loader*>(lookupLoader(prs1_class_name));
|
2019-05-03 01:51:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void PRS1Tests::cleanupTestCase(void)
|
|
|
|
{
|
2020-01-29 14:10:29 +00:00
|
|
|
delete p_profile;
|
|
|
|
p_profile = nullptr;
|
2021-10-26 18:43:14 +00:00
|
|
|
#ifdef TEST_OSCAR_CALCS
|
|
|
|
delete p_pref;
|
|
|
|
#endif
|
2019-05-03 01:51:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-27 14:05:16 +00:00
|
|
|
// ====================================================================================================
|
|
|
|
|
|
|
|
extern PRS1ModelInfo s_PRS1ModelInfo;
|
|
|
|
void PRS1Tests::testMachineSupport()
|
|
|
|
{
|
|
|
|
QHash<QString,QString> tested = {
|
|
|
|
{ "ModelNumber", "550P" },
|
|
|
|
{ "Family", "0" },
|
|
|
|
{ "FamilyVersion", "3" },
|
|
|
|
};
|
|
|
|
QHash<QString,QString> supported = {
|
|
|
|
{ "ModelNumber", "700X999" },
|
|
|
|
{ "Family", "0" },
|
|
|
|
{ "FamilyVersion", "6" },
|
|
|
|
};
|
|
|
|
QHash<QString,QString> unsupported = {
|
|
|
|
{ "ModelNumber", "550P" },
|
|
|
|
{ "Family", "0" },
|
|
|
|
{ "FamilyVersion", "9" },
|
|
|
|
};
|
|
|
|
|
|
|
|
Q_ASSERT(s_PRS1ModelInfo.IsSupported(5, 3));
|
|
|
|
Q_ASSERT(!s_PRS1ModelInfo.IsSupported(5, 9));
|
|
|
|
Q_ASSERT(!s_PRS1ModelInfo.IsSupported(9, 9));
|
|
|
|
Q_ASSERT(s_PRS1ModelInfo.IsTested("550P", 0, 2));
|
|
|
|
Q_ASSERT(s_PRS1ModelInfo.IsTested("550P", 0, 3));
|
|
|
|
Q_ASSERT(s_PRS1ModelInfo.IsTested("760P", 0, 4));
|
|
|
|
Q_ASSERT(s_PRS1ModelInfo.IsTested("700X110", 0, 6));
|
|
|
|
Q_ASSERT(!s_PRS1ModelInfo.IsTested("700X999", 0, 6));
|
|
|
|
|
|
|
|
Q_ASSERT(s_PRS1ModelInfo.IsTested(tested));
|
|
|
|
Q_ASSERT(!s_PRS1ModelInfo.IsTested(supported));
|
|
|
|
Q_ASSERT(s_PRS1ModelInfo.IsSupported(tested));
|
|
|
|
Q_ASSERT(s_PRS1ModelInfo.IsSupported(supported));
|
|
|
|
Q_ASSERT(!s_PRS1ModelInfo.IsSupported(unsupported));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ====================================================================================================
|
|
|
|
|
|
|
|
|
2019-05-05 01:53:02 +00:00
|
|
|
void parseAndEmitSessionYaml(const QString & path)
|
|
|
|
{
|
|
|
|
qDebug() << path;
|
|
|
|
|
2021-09-01 19:31:06 +00:00
|
|
|
ImportContext* ctx = new ProfileImportContext(p_profile);
|
|
|
|
s_loader->SetContext(ctx);
|
|
|
|
|
|
|
|
ctx->connect(ctx, &ImportContext::importEncounteredUnexpectedData, [=]() {
|
|
|
|
qWarning() << "*** Found unexpected data";
|
|
|
|
});
|
|
|
|
|
2019-05-05 01:53:02 +00:00
|
|
|
// This mirrors the functional bits of PRS1Loader::OpenMachine.
|
2020-01-05 01:47:28 +00:00
|
|
|
// TODO: Refactor PRS1Loader so that the tests can use the same
|
|
|
|
// underlying logic as OpenMachine rather than duplicating it here.
|
2019-05-05 01:53:02 +00:00
|
|
|
|
|
|
|
QStringList paths;
|
|
|
|
QString propertyfile;
|
|
|
|
int sessionid_base;
|
|
|
|
sessionid_base = s_loader->FindSessionDirsAndProperties(path, paths, propertyfile);
|
|
|
|
|
2021-10-26 19:23:08 +00:00
|
|
|
bool supported = s_loader->CreateMachineFromProperties(propertyfile);
|
|
|
|
if (!supported) {
|
2020-04-22 00:49:15 +00:00
|
|
|
qWarning() << "*** Skipping unsupported machine!";
|
|
|
|
return;
|
|
|
|
}
|
2021-10-26 19:23:08 +00:00
|
|
|
Machine* m = ctx->m_machine;
|
2019-05-05 01:53:02 +00:00
|
|
|
|
2021-09-03 16:45:00 +00:00
|
|
|
s_loader->ScanFiles(paths, sessionid_base);
|
2019-05-05 01:53:02 +00:00
|
|
|
|
2019-07-13 13:59:57 +00:00
|
|
|
// Each session now has a PRS1Import object in m_MLtasklist
|
2019-05-05 01:53:02 +00:00
|
|
|
QList<ImportTask*>::iterator i;
|
2019-07-13 13:59:57 +00:00
|
|
|
while (!s_loader->m_MLtasklist.isEmpty()) {
|
|
|
|
ImportTask* task = s_loader->m_MLtasklist.takeFirst();
|
2019-05-05 20:56:49 +00:00
|
|
|
|
2019-05-05 01:53:02 +00:00
|
|
|
// Run the parser
|
2019-05-05 20:56:49 +00:00
|
|
|
PRS1Import* import = dynamic_cast<PRS1Import*>(task);
|
2019-06-06 20:50:40 +00:00
|
|
|
bool ok = import->ParseSession();
|
2019-05-05 01:53:02 +00:00
|
|
|
|
|
|
|
// Emit the parsed session data to compare against our regression benchmarks
|
|
|
|
Session* session = import->session;
|
|
|
|
QString outpath = prs1OutputPath(path, m->serial(), session->session(), "-session.yml");
|
2021-10-26 18:43:14 +00:00
|
|
|
#ifdef TEST_OSCAR_CALCS
|
|
|
|
session->s_events_loaded = true;
|
|
|
|
session->UpdateSummaries();
|
|
|
|
#endif
|
2019-06-06 20:50:40 +00:00
|
|
|
SessionToYaml(outpath, session, ok);
|
2019-05-05 01:53:02 +00:00
|
|
|
|
|
|
|
delete session;
|
2019-05-05 20:56:49 +00:00
|
|
|
delete task;
|
2020-01-05 01:47:28 +00:00
|
|
|
}
|
2021-09-01 19:31:06 +00:00
|
|
|
|
|
|
|
delete ctx;
|
2019-05-05 01:53:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void PRS1Tests::testSessionsToYaml()
|
2019-05-03 01:51:56 +00:00
|
|
|
{
|
2019-10-09 14:24:29 +00:00
|
|
|
iterateTestCards(TESTDATA_PATH "prs1/input/", parseAndEmitSessionYaml);
|
2019-05-03 01:51:56 +00:00
|
|
|
}
|
|
|
|
|
2019-05-05 01:53:02 +00:00
|
|
|
|
2019-05-18 23:20:36 +00:00
|
|
|
// ====================================================================================================
|
|
|
|
|
2019-06-05 21:08:45 +00:00
|
|
|
static QString dur(qint64 msecs)
|
|
|
|
{
|
|
|
|
qint64 s = msecs / 1000L;
|
|
|
|
int h = s / 3600; s -= h * 3600;
|
|
|
|
int m = s / 60; s -= m * 60;
|
|
|
|
return QString("%1:%2:%3")
|
|
|
|
.arg(h, 2, 10, QChar('0'))
|
|
|
|
.arg(m, 2, 10, QChar('0'))
|
|
|
|
.arg(s, 2, 10, QChar('0'));
|
|
|
|
}
|
|
|
|
|
2019-05-29 01:09:17 +00:00
|
|
|
static QString byteList(QByteArray data, int limit=-1)
|
2019-05-18 23:20:36 +00:00
|
|
|
{
|
2019-05-29 01:09:17 +00:00
|
|
|
int count = data.size();
|
|
|
|
if (limit == -1 || limit > count) limit = count;
|
2020-01-05 21:02:04 +00:00
|
|
|
int first = limit / 2;
|
|
|
|
int last = limit - first;
|
2021-05-24 01:26:17 +00:00
|
|
|
#if 1
|
|
|
|
// Optimized after profiling regression tests.
|
|
|
|
QString s;
|
|
|
|
s.reserve(3 * limit + 4); // "NN " for each byte + possible "... " in the middle
|
|
|
|
const unsigned char* b = (const unsigned char*) data.constData();
|
|
|
|
for (int i = 0; i < first; i++) {
|
|
|
|
s.append(QString::asprintf("%02X ", b[i]));
|
|
|
|
}
|
|
|
|
if (limit < count) {
|
|
|
|
s.append(QStringLiteral("... "));
|
|
|
|
}
|
|
|
|
for (int i = count - last; i < count; i++) {
|
|
|
|
s.append(QString::asprintf("%02X ", b[i]));
|
|
|
|
}
|
|
|
|
s.resize(s.size() - 1); // remove trailing space
|
|
|
|
#else
|
|
|
|
// Unoptimized original, slows down regression tests.
|
2019-05-18 23:20:36 +00:00
|
|
|
QStringList l;
|
2020-01-05 21:02:04 +00:00
|
|
|
for (int i = 0; i < first; i++) {
|
2019-05-18 23:20:36 +00:00
|
|
|
l.push_back(QString( "%1" ).arg((int) data[i] & 0xFF, 2, 16, QChar('0') ).toUpper());
|
|
|
|
}
|
2019-05-29 01:09:17 +00:00
|
|
|
if (limit < count) l.push_back("...");
|
2020-01-05 21:02:04 +00:00
|
|
|
for (int i = count - last; i < count; i++) {
|
|
|
|
l.push_back(QString( "%1" ).arg((int) data[i] & 0xFF, 2, 16, QChar('0') ).toUpper());
|
|
|
|
}
|
2019-05-29 16:11:53 +00:00
|
|
|
QString s = l.join(" ");
|
2021-05-24 01:26:17 +00:00
|
|
|
#endif
|
2019-05-18 23:20:36 +00:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2019-06-08 01:12:55 +00:00
|
|
|
void ChunkToYaml(QTextStream & out, PRS1DataChunk* chunk, bool ok)
|
2019-05-18 23:20:36 +00:00
|
|
|
{
|
|
|
|
// chunk header
|
2021-05-24 01:26:17 +00:00
|
|
|
out << "chunk:" << '\n';
|
|
|
|
out << " at: " << hex << chunk->m_filepos << '\n';
|
|
|
|
out << " parsed: " << ok << '\n';
|
|
|
|
out << " version: " << dec << chunk->fileVersion << '\n';
|
|
|
|
out << " size: " << chunk->blockSize << '\n';
|
|
|
|
out << " htype: " << chunk->htype << '\n';
|
|
|
|
out << " family: " << chunk->family << '\n';
|
|
|
|
out << " familyVersion: " << chunk->familyVersion << '\n';
|
|
|
|
out << " ext: " << chunk->ext << '\n';
|
|
|
|
out << " session: " << chunk->sessionid << '\n';
|
|
|
|
out << " start: " << ts(chunk->timestamp * 1000L) << '\n';
|
|
|
|
out << " duration: " << dur(chunk->duration * 1000L) << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
|
|
|
|
// hblock for V3 non-waveform chunks
|
|
|
|
if (chunk->fileVersion == 3 && chunk->htype == 0) {
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " hblock:" << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
QMapIterator<unsigned char, short> i(chunk->hblock);
|
|
|
|
while (i.hasNext()) {
|
|
|
|
i.next();
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " " << (int) i.key() << ": " << i.value() << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// waveform chunks
|
|
|
|
if (chunk->htype == 1) {
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " intervals: " << chunk->interval_count << '\n';
|
|
|
|
out << " intervalSeconds: " << (int) chunk->interval_seconds << '\n';
|
|
|
|
out << " interleave:" << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
for (int i=0; i < chunk->waveformInfo.size(); i++) {
|
|
|
|
const PRS1Waveform & w = chunk->waveformInfo.at(i);
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " " << i << ": " << w.interleave << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
}
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " end: " << ts((chunk->timestamp + chunk->duration) * 1000L) << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// header checksum
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " checksum: " << hex << chunk->storedChecksum << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
if (chunk->storedChecksum != chunk->calcChecksum) {
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " calcChecksum: " << hex << chunk->calcChecksum << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// data
|
2019-05-29 16:11:53 +00:00
|
|
|
bool dump_data = true;
|
|
|
|
if (chunk->m_parsedData.size() > 0) {
|
|
|
|
dump_data = false;
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " events:" << '\n';
|
2019-05-29 16:11:53 +00:00
|
|
|
for (auto & e : chunk->m_parsedData) {
|
2021-05-31 18:53:23 +00:00
|
|
|
QString name = e->typeName();
|
2019-05-29 16:11:53 +00:00
|
|
|
if (name == "raw" || name == "unknown") {
|
|
|
|
dump_data = true;
|
|
|
|
}
|
2021-05-31 18:53:23 +00:00
|
|
|
QMap<QString,QString> contents = e->contents();
|
2019-05-29 16:11:53 +00:00
|
|
|
if (name == "setting" && contents.size() == 1) {
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " - set_" << contents.firstKey() << ": " << contents.first() << '\n';
|
2019-05-29 16:11:53 +00:00
|
|
|
}
|
|
|
|
else {
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " - " << name << ":" << '\n';
|
2019-05-29 16:11:53 +00:00
|
|
|
|
|
|
|
// Always emit start first if present
|
|
|
|
if (contents.contains("start")) {
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " " << "start" << ": " << contents["start"] << '\n';
|
2019-05-29 16:11:53 +00:00
|
|
|
}
|
|
|
|
for (auto & key : contents.keys()) {
|
|
|
|
if (key == "start") continue;
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " " << key << ": " << contents[key] << '\n';
|
2019-05-29 16:11:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-06-08 01:12:55 +00:00
|
|
|
if (dump_data || !ok) {
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " data: " << byteList(chunk->m_data, 100) << '\n';
|
2019-05-29 16:11:53 +00:00
|
|
|
}
|
2019-05-18 23:20:36 +00:00
|
|
|
|
|
|
|
// data CRC
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " crc: " << hex << chunk->storedCrc << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
if (chunk->storedCrc != chunk->calcCrc) {
|
2021-05-24 01:26:17 +00:00
|
|
|
out << " calcCrc: " << hex << chunk->calcCrc << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
}
|
2021-05-24 01:26:17 +00:00
|
|
|
out << '\n';
|
2019-05-18 23:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void parseAndEmitChunkYaml(const QString & path)
|
|
|
|
{
|
2020-01-12 00:14:01 +00:00
|
|
|
bool FV = false; // set this to true to emit family/familyVersion for this path
|
2019-05-18 23:20:36 +00:00
|
|
|
qDebug() << path;
|
|
|
|
|
2021-09-01 19:31:06 +00:00
|
|
|
ImportContext* ctx = new ProfileImportContext(p_profile);
|
|
|
|
s_loader->SetContext(ctx);
|
|
|
|
|
2019-10-25 21:27:35 +00:00
|
|
|
QHash<QString,QSet<quint64>> written;
|
2019-05-18 23:20:36 +00:00
|
|
|
QStringList paths;
|
|
|
|
QString propertyfile;
|
|
|
|
int sessionid_base;
|
|
|
|
sessionid_base = s_loader->FindSessionDirsAndProperties(path, paths, propertyfile);
|
|
|
|
|
2021-10-26 19:23:08 +00:00
|
|
|
bool supported = s_loader->CreateMachineFromProperties(propertyfile);
|
|
|
|
if (!supported) {
|
2020-04-22 00:49:15 +00:00
|
|
|
qWarning() << "*** Skipping unsupported machine!";
|
|
|
|
return;
|
|
|
|
}
|
2021-10-26 19:23:08 +00:00
|
|
|
Machine* m = ctx->m_machine;
|
2019-05-18 23:20:36 +00:00
|
|
|
|
|
|
|
// This mirrors the functional bits of PRS1Loader::ScanFiles.
|
|
|
|
|
|
|
|
QDir dir;
|
|
|
|
dir.setFilter(QDir::NoDotAndDotDot | QDir::Dirs | QDir::Files | QDir::Hidden | QDir::NoSymLinks);
|
|
|
|
dir.setSorting(QDir::Name);
|
|
|
|
|
|
|
|
int size = paths.size();
|
|
|
|
|
|
|
|
// for each p0/p1/p2/etc... folder
|
|
|
|
for (int p=0; p < size; ++p) {
|
|
|
|
dir.setPath(paths.at(p));
|
|
|
|
if (!dir.exists() || !dir.isReadable()) {
|
|
|
|
qWarning() << dir.canonicalPath() << "can't read directory";
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
QFileInfoList flist = dir.entryInfoList();
|
|
|
|
|
|
|
|
// Scan for individual .00X files
|
|
|
|
for (int i = 0; i < flist.size(); i++) {
|
|
|
|
QFileInfo fi = flist.at(i);
|
|
|
|
QString inpath = fi.canonicalFilePath();
|
|
|
|
bool ok;
|
2019-05-31 20:58:58 +00:00
|
|
|
|
|
|
|
if (fi.fileName() == ".DS_Store") {
|
|
|
|
continue;
|
|
|
|
}
|
2019-05-18 23:20:36 +00:00
|
|
|
|
|
|
|
QString ext_s = fi.fileName().section(".", -1);
|
2021-11-03 20:38:20 +00:00
|
|
|
if (ext_s.toUpper().startsWith("B")) { // .B01, .B02, etc.
|
|
|
|
ext_s = ext_s.mid(1);
|
|
|
|
}
|
2019-06-08 01:12:55 +00:00
|
|
|
int ext = ext_s.toInt(&ok);
|
2019-05-18 23:20:36 +00:00
|
|
|
if (!ok) {
|
|
|
|
// not a numerical extension
|
2019-09-20 19:38:14 +00:00
|
|
|
qInfo() << inpath << "unexpected filename";
|
2019-05-18 23:20:36 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString session_s = fi.fileName().section(".", 0, -2);
|
2019-06-08 01:12:55 +00:00
|
|
|
int sessionid = session_s.toInt(&ok, sessionid_base);
|
2019-05-18 23:20:36 +00:00
|
|
|
if (!ok) {
|
|
|
|
// not a numerical session ID
|
2019-09-20 19:38:14 +00:00
|
|
|
qInfo() << inpath << "unexpected filename";
|
2019-05-18 23:20:36 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the YAML file.
|
2019-06-08 01:12:55 +00:00
|
|
|
QString suffix = QString(".%1-chunks.yml").arg(ext, 3, 10, QChar('0'));
|
|
|
|
QString outpath = prs1OutputPath(path, m->serial(), sessionid, suffix);
|
2019-05-18 23:20:36 +00:00
|
|
|
QFile file(outpath);
|
2019-10-25 21:27:35 +00:00
|
|
|
// Truncate the first time we open this file, to clear out any previous test data.
|
|
|
|
// Otherwise append, allowing session chunks to be split among multiple files.
|
|
|
|
if (!file.open(QFile::WriteOnly | (written.contains(outpath) ? QFile::Append : QFile::Truncate))) {
|
2019-05-18 23:20:36 +00:00
|
|
|
qDebug() << outpath;
|
|
|
|
Q_ASSERT(false);
|
|
|
|
}
|
2019-06-08 01:12:55 +00:00
|
|
|
QTextStream out(&file);
|
|
|
|
|
|
|
|
// keep only P1234568/Pn/00000000.001
|
|
|
|
QStringList pathlist = QDir::toNativeSeparators(inpath).split(QDir::separator(), QString::SkipEmptyParts);
|
|
|
|
QString relative = pathlist.mid(pathlist.size()-3).join(QDir::separator());
|
2019-10-25 21:27:35 +00:00
|
|
|
bool first_chunk_from_file = true;
|
2019-05-18 23:20:36 +00:00
|
|
|
|
|
|
|
// Parse the chunks in the file.
|
|
|
|
QList<PRS1DataChunk *> chunks = s_loader->ParseFile(inpath);
|
|
|
|
for (int i=0; i < chunks.size(); i++) {
|
|
|
|
PRS1DataChunk * chunk = chunks.at(i);
|
2020-01-12 00:14:01 +00:00
|
|
|
if (i == 0 && FV) { qWarning() << QString("F%1V%2").arg(chunk->family).arg(chunk->familyVersion); FV=false; }
|
2019-10-25 21:27:35 +00:00
|
|
|
// Only write unique chunks to the file.
|
|
|
|
if (written[outpath].contains(chunk->hash()) == false) {
|
|
|
|
if (first_chunk_from_file) {
|
2021-05-24 01:26:17 +00:00
|
|
|
out << "file: " << relative << '\n';
|
2019-10-25 21:27:35 +00:00
|
|
|
first_chunk_from_file = false;
|
|
|
|
}
|
|
|
|
bool ok = true;
|
|
|
|
|
|
|
|
// Parse the inner data.
|
|
|
|
switch (chunk->ext) {
|
|
|
|
case 0: ok = chunk->ParseCompliance(); break;
|
|
|
|
case 1: ok = chunk->ParseSummary(); break;
|
|
|
|
case 2: ok = chunk->ParseEvents(); break;
|
|
|
|
case 5: break; // skip flow/pressure waveforms
|
2019-12-02 22:30:28 +00:00
|
|
|
case 6: break; // skip oximetry data
|
2019-10-25 21:27:35 +00:00
|
|
|
default:
|
|
|
|
qWarning() << relative << "unexpected file type";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Emit the YAML.
|
|
|
|
ChunkToYaml(out, chunk, ok);
|
|
|
|
written[outpath] += chunk->hash();
|
2019-05-29 16:11:53 +00:00
|
|
|
}
|
2019-05-18 23:20:36 +00:00
|
|
|
delete chunk;
|
|
|
|
}
|
|
|
|
|
|
|
|
file.close();
|
|
|
|
}
|
|
|
|
}
|
2020-01-29 14:10:29 +00:00
|
|
|
|
2021-09-01 19:31:06 +00:00
|
|
|
delete ctx;
|
|
|
|
|
2020-01-29 14:10:29 +00:00
|
|
|
p_profile->removeMachine(m);
|
|
|
|
delete m;
|
2019-05-18 23:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void PRS1Tests::testChunksToYaml()
|
|
|
|
{
|
2019-10-24 15:44:47 +00:00
|
|
|
iterateTestCards(TESTDATA_PATH "prs1/input/", parseAndEmitChunkYaml);
|
2019-05-18 23:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-05 01:53:02 +00:00
|
|
|
// ====================================================================================================
|
|
|
|
|
|
|
|
QString prs1OutputPath(const QString & inpath, const QString & serial, int session, const QString & suffix)
|
2019-05-18 23:20:36 +00:00
|
|
|
{
|
|
|
|
QString basename = QString("%1").arg(session, 8, 10, QChar('0'));
|
|
|
|
return prs1OutputPath(inpath, serial, basename, suffix);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString prs1OutputPath(const QString & inpath, const QString & serial, const QString & basename, const QString & suffix)
|
2019-05-05 01:53:02 +00:00
|
|
|
{
|
|
|
|
// Output to prs1/output/FOLDER/SERIAL-000000(-session.yml, etc.)
|
|
|
|
QDir path(inpath);
|
|
|
|
QStringList pathlist = QDir::toNativeSeparators(inpath).split(QDir::separator(), QString::SkipEmptyParts);
|
|
|
|
pathlist.pop_back(); // drop serial number directory
|
|
|
|
pathlist.pop_back(); // drop P-Series directory
|
|
|
|
QString foldername = pathlist.last();
|
|
|
|
|
|
|
|
QDir outdir(TESTDATA_PATH "prs1/output/" + foldername);
|
|
|
|
outdir.mkpath(".");
|
|
|
|
|
|
|
|
QString filename = QString("%1-%2%3")
|
|
|
|
.arg(serial)
|
2019-05-18 23:20:36 +00:00
|
|
|
.arg(basename)
|
2019-05-05 01:53:02 +00:00
|
|
|
.arg(suffix);
|
|
|
|
return outdir.path() + QDir::separator() + filename;
|
|
|
|
}
|
|
|
|
|
|
|
|
void iterateTestCards(const QString & root, void (*action)(const QString &))
|
|
|
|
{
|
|
|
|
QDir dir(root);
|
2020-03-27 16:33:23 +00:00
|
|
|
dir.setFilter(QDir::NoDotAndDotDot | QDir::Dirs);
|
2019-05-05 01:53:02 +00:00
|
|
|
dir.setSorting(QDir::Name);
|
|
|
|
QFileInfoList flist = dir.entryInfoList();
|
|
|
|
|
|
|
|
// Look through each folder in the given root
|
2020-03-09 00:58:36 +00:00
|
|
|
for (auto & fi : flist) {
|
2020-03-27 16:33:23 +00:00
|
|
|
QStringList machinePaths = s_loader->FindMachinesOnCard(fi.canonicalFilePath());
|
|
|
|
|
|
|
|
// Tests should be run newest to oldest, since older sets tend to have more
|
|
|
|
// complete data. (These are usually previously cleared data in the Clear0/Cn
|
|
|
|
// directories.) The machines themselves will write out the summary data they
|
|
|
|
// remember when they see an empty folder, without event or waveform data.
|
|
|
|
// And since these tests (by design) overwrite existing output, we want the
|
|
|
|
// earlier (more complete) data to be what's written last.
|
|
|
|
//
|
|
|
|
// Since the loader itself keeps only the first set of data it sees for a session,
|
|
|
|
// we want to leave its earliest-to-latest ordering in place, and just reverse it
|
|
|
|
// here.
|
|
|
|
for (auto i = machinePaths.crbegin(); i != machinePaths.crend(); i++) {
|
|
|
|
action(*i);
|
2019-05-05 01:53:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|