mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 18:50:44 +00:00
Update PRS1 tests to allow for session chunks to be split between files.
Also warn when waveform files are being skipped on import due to this kind of splitting. The chunks YAML now emits all unique chunks found in files with the same session ID. Note that a single file can contain multiple chunks covering multiple sessions. These will all be saved in the YAML file corresponding to the original file's name, rather than the session ID encoded in any chunk. This slight discrepancy is intentional, since the chunk YAML is meant to test the parsers, to verify that they correctly decode a specific input file. When importing data into a session, we use the actual session ID specified by each chunk. Thus the session YAML files will be derived from the proper chunks, regardless of their original containing file. (Well, except for waveforms, but they don't appear to have more than one session ID per file.)
This commit is contained in:
parent
3267078608
commit
3e42703399
@ -829,7 +829,7 @@ static QString relativePath(const QString & inpath)
|
||||
|
||||
static bool chunksIdentical(const PRS1DataChunk* a, const PRS1DataChunk* b)
|
||||
{
|
||||
return (a->timestamp == b->timestamp && a->storedCrc == b->storedCrc);
|
||||
return (a->hash() == b->hash());
|
||||
}
|
||||
|
||||
static QString chunkComparison(const PRS1DataChunk* a, const PRS1DataChunk* b)
|
||||
@ -942,11 +942,19 @@ void PRS1Loader::ScanFiles(const QStringList & paths, int sessionid_base, Machin
|
||||
}
|
||||
|
||||
if (ext == 5) {
|
||||
if (!task->wavefile.isEmpty()) continue;
|
||||
if (!task->wavefile.isEmpty()) {
|
||||
qDebug() << sid << "already has waveform file" << relativePath(task->wavefile)
|
||||
<< "skipping" << relativePath(fi.canonicalFilePath());
|
||||
continue;
|
||||
}
|
||||
task->wavefile = fi.canonicalFilePath();
|
||||
} else if (ext == 6) {
|
||||
qWarning() << fi.canonicalFilePath() << "oximetry is untested"; // TODO: mark as untested/unexpected
|
||||
if (!task->oxifile.isEmpty()) continue;
|
||||
if (!task->oxifile.isEmpty()) {
|
||||
qDebug() << sid << "already has oximetry file" << relativePath(task->oxifile)
|
||||
<< "skipping" << relativePath(fi.canonicalFilePath());
|
||||
continue;
|
||||
}
|
||||
task->oxifile = fi.canonicalFilePath();
|
||||
}
|
||||
|
||||
|
@ -119,6 +119,9 @@ public:
|
||||
quint32 storedCrc; // header + data CRC stored in file, last 2-4 bytes of chunk
|
||||
quint32 calcCrc; // header + data CRC as calculated when parsing
|
||||
|
||||
//! \brief Calculate a simplistic hash to check whether two chunks are identical.
|
||||
inline quint64 hash(void) const { return ((((quint64) this->calcCrc) << 32) | this->timestamp); }
|
||||
|
||||
//! \brief Parse and return the next chunk from a PRS1 file
|
||||
static PRS1DataChunk* ParseNext(class QFile & f);
|
||||
|
||||
|
@ -238,6 +238,7 @@ void parseAndEmitChunkYaml(const QString & path)
|
||||
{
|
||||
qDebug() << path;
|
||||
|
||||
QHash<QString,QSet<quint64>> written;
|
||||
QStringList paths;
|
||||
QString propertyfile;
|
||||
int sessionid_base;
|
||||
@ -293,7 +294,9 @@ void parseAndEmitChunkYaml(const QString & path)
|
||||
QString suffix = QString(".%1-chunks.yml").arg(ext, 3, 10, QChar('0'));
|
||||
QString outpath = prs1OutputPath(path, m->serial(), sessionid, suffix);
|
||||
QFile file(outpath);
|
||||
if (!file.open(QFile::WriteOnly | QFile::Truncate)) {
|
||||
// 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))) {
|
||||
qDebug() << outpath;
|
||||
Q_ASSERT(false);
|
||||
}
|
||||
@ -302,30 +305,38 @@ void parseAndEmitChunkYaml(const QString & path)
|
||||
// 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());
|
||||
out << "file: " << relative << endl;
|
||||
bool first_chunk_from_file = true;
|
||||
|
||||
// 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);
|
||||
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
|
||||
case 6: // skip oximetry data (but log it)
|
||||
qWarning() << relative << "oximetry is untested"; // never encountered
|
||||
break;
|
||||
default:
|
||||
qWarning() << relative << "unexpected file type";
|
||||
break;
|
||||
// Only write unique chunks to the file.
|
||||
if (written[outpath].contains(chunk->hash()) == false) {
|
||||
if (first_chunk_from_file) {
|
||||
out << "file: " << relative << endl;
|
||||
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
|
||||
case 6: // skip oximetry data (but log it)
|
||||
qWarning() << relative << "oximetry is untested"; // never encountered
|
||||
break;
|
||||
default:
|
||||
qWarning() << relative << "unexpected file type";
|
||||
break;
|
||||
}
|
||||
|
||||
// Emit the YAML.
|
||||
ChunkToYaml(out, chunk, ok);
|
||||
written[outpath] += chunk->hash();
|
||||
}
|
||||
|
||||
// Emit the YAML.
|
||||
ChunkToYaml(out, chunk, ok);
|
||||
delete chunk;
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user