| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439 | /*=========================================================================  Library:   CTK  Copyright (c) Kitware Inc.  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.commontk.org/LICENSE  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.=========================================================================*/// Qt includes#include <QSqlQuery>#include <QSqlRecord>#include <QVariant>#include <QDate>#include <QStringList>#include <QSet>#include <QFile>#include <QDirIterator>#include <QFileInfo>#include <QDebug>// ctkDICOM includes#include "ctkDICOMIndexer.h"// DCMTK includes#ifndef WIN32  #define HAVE_CONFIG_H #endif#include <dcmtk/dcmdata/dcfilefo.h>#include <dcmtk/dcmdata/dcfilefo.h>#include <dcmtk/dcmdata/dcdeftag.h>#include <dcmtk/dcmdata/dcdatset.h>#include <dcmtk/ofstd/ofcond.h>#include <dcmtk/ofstd/ofstring.h>#include <dcmtk/ofstd/ofstd.h>        /* for class OFStandard */#include <dcmtk/dcmdata/dcddirif.h>   /* for class DicomDirInterface */#define MITK_ERROR std::cout#define MITK_INFO std::cout//------------------------------------------------------------------------------class ctkDICOMIndexerPrivate{public:  ctkDICOMIndexerPrivate();  ~ctkDICOMIndexerPrivate();};//------------------------------------------------------------------------------// ctkDICOMIndexerPrivate methods//------------------------------------------------------------------------------ctkDICOMIndexerPrivate::ctkDICOMIndexerPrivate(){}//------------------------------------------------------------------------------ctkDICOMIndexerPrivate::~ctkDICOMIndexerPrivate(){}//------------------------------------------------------------------------------// ctkDICOMIndexer methods//------------------------------------------------------------------------------ctkDICOMIndexer::ctkDICOMIndexer():d_ptr(new ctkDICOMIndexerPrivate){}//------------------------------------------------------------------------------ctkDICOMIndexer::~ctkDICOMIndexer(){}//------------------------------------------------------------------------------void ctkDICOMIndexer::addDirectory(QSqlDatabase database, const QString& directoryName,const QString& destinationDirectoryName){  QSqlDatabase db = database;  const std::string src_directory(directoryName.toStdString());  OFList<OFString> originalDcmtkFileNames;  OFList<OFString> dcmtkFileNames;  OFStandard::searchDirectoryRecursively( src_directory.c_str(), originalDcmtkFileNames, "", "");  // hack to reverse list of filenames (not neccessary when image loading works correctly)  for ( OFListIterator(OFString) iter = originalDcmtkFileNames.begin(); iter != originalDcmtkFileNames.end(); ++iter )  {    dcmtkFileNames.push_front( *iter );  }  DcmFileFormat fileformat;  DcmDataset *dataset;  OFListIterator(OFString) iter = dcmtkFileNames.begin();  OFListIterator(OFString) last = dcmtkFileNames.end();  if(iter == last) return;  QSqlQuery query(database);  /// these are for optimizing the import of image sequences  /// since most information are identical for all slices  OFString lastPatientID = "";  OFString lastPatientsName = "";  OFString lastPatientsBirthDate = "";  OFString lastStudyInstanceUID = "";  OFString lastSeriesInstanceUID = "";  int lastPatientUID = -1;  /* iterate over all input filenames */  while (iter != last)  {    std::string filename((*iter).c_str());    QString qfilename(filename.c_str());     /// first we check if the file is already in the database    QSqlQuery fileExists(database);    fileExists.prepare("SELECT InsertTimestamp FROM Images WHERE Filename == ?");     fileExists.bindValue(0,qfilename);    fileExists.exec();    if (      fileExists.next() &&       QFileInfo(qfilename).lastModified() < QDateTime::fromString(fileExists.value(0).toString(),Qt::ISODate)            )      {      MITK_INFO << "File " << filename << " already added.";      continue;      }    MITK_INFO << filename << "\n";    OFCondition status = fileformat.loadFile(filename.c_str());    ++iter;    dataset = fileformat.getDataset();    if (!status.good())    {      MITK_ERROR << "Could not load " << filename << "\nDCMTK says: " << status.text();      continue;    }    OFString patientsName, patientID, patientsBirthDate, patientsBirthTime, patientsSex,      patientComments, patientsAge;    OFString studyInstanceUID, studyID, studyDate, studyTime,      accessionNumber, modalitiesInStudy, institutionName, performingPhysiciansName, referringPhysician, studyDescription;    OFString seriesInstanceUID, seriesDate, seriesTime,      seriesDescription, bodyPartExamined, frameOfReferenceUID,      contrastAgent, scanningSequence;    OFString instanceNumber;    Sint32 seriesNumber = 0, acquisitionNumber = 0, echoNumber = 0, temporalPosition = 0;    //The patient UID is a unique number within the database, generated by the sqlite autoincrement    int patientUID = -1;    //If the following fields can not be evaluated, cancel evaluation of the DICOM file    if (!dataset->findAndGetOFString(DCM_PatientsName, patientsName).good())    {      MITK_ERROR << "Could not read DCM_PatientsName from " << filename;      continue;    }    if (!dataset->findAndGetOFString(DCM_StudyInstanceUID, studyInstanceUID).good())    {      MITK_ERROR << "Could not read DCM_StudyInstanceUID from " << filename;      continue;    }    if (!dataset->findAndGetOFString(DCM_SeriesInstanceUID, seriesInstanceUID).good())    {      MITK_ERROR << "Could not read DCM_SeriesInstanceUID from " << filename;      continue;    }    if (!dataset->findAndGetOFString(DCM_InstanceNumber, instanceNumber).good())    {      MITK_ERROR << "Could not read DCM_InstanceNumber from " << filename;      continue;    }    dataset->findAndGetOFString(DCM_PatientID, patientID);    dataset->findAndGetOFString(DCM_PatientsBirthDate, patientsBirthDate);    dataset->findAndGetOFString(DCM_PatientsBirthTime, patientsBirthTime);    dataset->findAndGetOFString(DCM_PatientsSex, patientsSex);    dataset->findAndGetOFString(DCM_PatientsAge, patientsAge);    dataset->findAndGetOFString(DCM_PatientComments, patientComments);    dataset->findAndGetOFString(DCM_StudyID, studyID);    dataset->findAndGetOFString(DCM_StudyDate, studyDate);    dataset->findAndGetOFString(DCM_StudyTime, studyTime);    dataset->findAndGetOFString(DCM_AccessionNumber, accessionNumber);    dataset->findAndGetOFString(DCM_ModalitiesInStudy, modalitiesInStudy);    dataset->findAndGetOFString(DCM_InstitutionName, institutionName);    dataset->findAndGetOFString(DCM_PerformingPhysiciansName, performingPhysiciansName);    dataset->findAndGetOFString(DCM_ReferringPhysiciansName, referringPhysician);    dataset->findAndGetOFString(DCM_StudyDescription, studyDescription);    dataset->findAndGetOFString(DCM_SeriesDate, seriesDate);    dataset->findAndGetOFString(DCM_SeriesTime, seriesTime);    dataset->findAndGetOFString(DCM_SeriesDescription, seriesDescription);    dataset->findAndGetOFString(DCM_BodyPartExamined, bodyPartExamined);    dataset->findAndGetOFString(DCM_FrameOfReferenceUID, frameOfReferenceUID);    dataset->findAndGetOFString(DCM_ContrastBolusAgent, contrastAgent);    dataset->findAndGetOFString(DCM_ScanningSequence, scanningSequence);    dataset->findAndGetSint32(DCM_SeriesNumber, seriesNumber);    dataset->findAndGetSint32(DCM_AcquisitionNumber, acquisitionNumber);    dataset->findAndGetSint32(DCM_EchoNumbers, echoNumber);    dataset->findAndGetSint32(DCM_TemporalPositionIdentifier, temporalPosition);    MITK_INFO << "Adding new items to database:";    MITK_INFO << "studyID: " << studyID;    MITK_INFO << "seriesInstanceUID: " << seriesInstanceUID;    MITK_INFO << "Patient's Name: " << patientsName;    //-----------------------    //Add Patient to Database    //-----------------------    //Speed up: Check if patient is the same as in last file; very probable, as all images belonging to a study have the same patient    bool patientExists = false;    if(lastPatientID.compare(patientID) || lastPatientsBirthDate.compare(patientsBirthDate) || lastPatientsName.compare(patientsName))    {      //Check if patient is already present in the db      QSqlQuery check_exists_query(database);      std::stringstream check_exists_query_string;      check_exists_query_string << "SELECT * FROM Patients WHERE PatientID = '" << patientID << "'";      check_exists_query.exec(check_exists_query_string.str().c_str());      /// we check only patients with the same PatientID      /// PatientID is not unique in DICOM, so we also compare Name and BirthDate      /// and assume this is sufficient      while (check_exists_query.next())      {        if (            check_exists_query.record().value("PatientsName").toString() == patientsName.c_str() &&            check_exists_query.record().value("PatientsBirthDate").toString() == patientsBirthDate.c_str()           )        {          /// found it          patientUID = check_exists_query.value(check_exists_query.record().indexOf("UID")).toInt();          break;        }      }      if(!patientExists)      {        std::stringstream query_string;        query_string << "INSERT INTO Patients VALUES( NULL,'"         << patientsName << "','"         << patientID << "','"         << patientsBirthDate << "','"        << patientsBirthTime << "','"         << patientsSex << "','"         << patientsAge << "','"         << patientComments << "')";        query.exec(query_string.str().c_str());        patientUID = query.lastInsertId().toInt();        MITK_INFO << "New patient inserted: " << patientUID << "\n";      }    }    else       {      patientUID = lastPatientUID;      }             /// keep this for the next image    lastPatientUID = patientUID;    lastPatientID = patientID;    lastPatientsBirthDate = patientsBirthDate;    lastPatientsName = patientsName;    //---------------------    //Add Study to Database    //---------------------    if(lastStudyInstanceUID.compare(studyInstanceUID))    {      QSqlQuery check_exists_query(database);      std::stringstream check_exists_query_string;      check_exists_query_string << "SELECT * FROM Studies WHERE StudyInstanceUID = '" << studyInstanceUID << "'";      check_exists_query.exec(check_exists_query_string.str().c_str());      if(!check_exists_query.next())      {        std::stringstream query_string;        query_string << "INSERT INTO Studies VALUES('"          << studyInstanceUID << "','"           << patientUID << "','"           << studyID << "','"          << QDate::fromString(studyDate.c_str(), "yyyyMMdd").toString("yyyy-MM-dd").toStdString() << "','"          << studyTime << "','"           << accessionNumber << "','"           << modalitiesInStudy << "','"           << institutionName << "','"           << referringPhysician << "','"           << performingPhysiciansName << "','"           << studyDescription << "')";        query.exec(query_string.str().c_str());      }    }    lastStudyInstanceUID = studyInstanceUID;    //----------------------    //Add Series to Database    //----------------------    if(lastSeriesInstanceUID.compare(seriesInstanceUID))    {      QSqlQuery check_exists_query(database);      std::stringstream check_exists_query_string;      check_exists_query_string << "SELECT * FROM Series WHERE SeriesInstanceUID = '" << seriesInstanceUID << "'";      check_exists_query.exec(check_exists_query_string.str().c_str());      if(!check_exists_query.next())      {        std::stringstream query_string;        query_string << "INSERT INTO Series VALUES('"          << seriesInstanceUID << "','"          << studyInstanceUID << "','"          << static_cast<int>(seriesNumber) << "','"          << QDate::fromString(seriesDate.c_str(), "yyyyMMdd").toString("yyyy-MM-dd").toStdString() << "','"          << seriesTime << "','"          << seriesDescription << "','"          << bodyPartExamined << "','"          << frameOfReferenceUID << "','"          << static_cast<int>(acquisitionNumber) << "','"          << contrastAgent << "','"          << scanningSequence << "','"          << static_cast<int>(echoNumber) << "','"          << static_cast<int>(temporalPosition) << "')";        query.exec(query_string.str().c_str());      }    }    lastSeriesInstanceUID = seriesInstanceUID;    //----------------------------------    //Move file to destination directory    //----------------------------------    if (!destinationDirectoryName.isEmpty())      {      QFile currentFile( qfilename );      QDir destinationDir(destinationDirectoryName);      QString uniqueDirName = QString(studyInstanceUID.c_str()) + "/" + seriesInstanceUID.c_str();      qDebug() << "MKPath: " << uniqueDirName;      destinationDir.mkpath(uniqueDirName);      QString destFileName = destinationDir.absolutePath().append("/").append(instanceNumber.c_str());      qDebug() << "Copy: " << qfilename << " -> " << destFileName;      currentFile.copy(destFileName);      //for testing only: copy file instead of moving      //currentFile.copyTo(destDirectoryPath.str());    }    // */    //------------------------    //Add Filename to Database    //------------------------//    std::stringstream relativeFilePath;//    relativeFilePath << seriesInstanceUID.c_str() << "/" << currentFilePath.getFileName();    QSqlQuery check_exists_query(database);    std::stringstream check_exists_query_string;//    check_exists_query_string << "SELECT * FROM Images WHERE Filename = '" << relativeFilePath.str() << "'";    check_exists_query_string << "SELECT * FROM Images WHERE Filename = '" << filename << "'";    check_exists_query.exec(check_exists_query_string.str().c_str());    if(!check_exists_query.next())    {      std::stringstream query_string;      //To save absolute path: destDirectoryPath.str()      query_string << "INSERT INTO Images VALUES('"        << /*relativeFilePath.str()*/ filename << "','" << seriesInstanceUID << "','" << QDateTime::currentDateTime().toString(Qt::ISODate).toStdString() << "')";      query.exec(query_string.str().c_str());    }  }  // db.commit();  // db.close();}//------------------------------------------------------------------------------void ctkDICOMIndexer::refreshDatabase(QSqlDatabase database, const QString& directoryName){  /// get all filenames from the database  QSqlQuery allFilesQuery(database);  QStringList databaseFileNames;  QStringList filesToRemove;  allFilesQuery.exec("SELECT Filename from Images;");  while (allFilesQuery.next())    {    QString fileName = allFilesQuery.value(0).toString();    databaseFileNames.append(fileName);    if (! QFile::exists(fileName) )       {      filesToRemove.append(fileName);      }    }  QSet<QString> filesytemFiles;  QDirIterator dirIt(directoryName);  while (dirIt.hasNext())     {    filesytemFiles.insert(dirIt.next());    }}
 |