| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279 | /*=============================================================================  Library: CTK  Copyright (c) German Cancer Research Center,    Division of Medical and Biological Informatics  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.apache.org/licenses/LICENSE-2.0  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.=============================================================================*/#include "ctkCmdLineModuleXmlProgressWatcher.h"#include <QIODevice>#include <QProcess>#include <QXmlStreamReader>#include <QDebug>namespace {static QString FILTER_START = "filter-start";static QString FILTER_NAME = "filter-name";static QString FILTER_COMMENT = "filter-comment";static QString FILTER_PROGRESS = "filter-progress";static QString FILTER_PROGRESS_TEXT = "filter-progress-text";static QString FILTER_RESULT = "filter-result";static QString FILTER_END = "filter-end";}//----------------------------------------------------------------------------class ctkCmdLineModuleXmlProgressWatcherPrivate{public:  ctkCmdLineModuleXmlProgressWatcherPrivate(QIODevice* input, ctkCmdLineModuleXmlProgressWatcher* qq)    : input(input), process(NULL), readPos(0), q(qq), error(false), currentProgress(0)  {    // wrap the content in an artifical root element    reader.addData("<module-root>");  }  ctkCmdLineModuleXmlProgressWatcherPrivate(QProcess* input, ctkCmdLineModuleXmlProgressWatcher* qq)    : input(input), process(input), readPos(0), q(qq), error(false), currentProgress(0)  {    // wrap the content in an artifical root element    reader.addData("<module-root>");  }  void _q_readyRead()  {    input->seek(readPos);    reader.addData(input->readAll());    readPos = input->pos();    parseProgressXml();  }  void _q_readyReadError()  {    emit q->errorDataAvailable(process->readAllStandardError());  }  void parseProgressXml()  {    QXmlStreamReader::TokenType type = reader.readNext();    QByteArray outputData;    while(type != QXmlStreamReader::Invalid)    {      switch(type)      {      case QXmlStreamReader::NoToken: break;      case QXmlStreamReader::Characters:      {        if (stack.empty())        {          QByteArray output(reader.text().toString().toAscii());          // get rid of a possible newline after the last xml end tag          if (output.startsWith('\n')) output = output.remove(0,1);          outputData.append(output);          break;        }        if (stack.size() == 2 &&            (stack.front() == FILTER_START || stack.front() == FILTER_END))        {          if (stack.back() == FILTER_NAME)          {            currentName = reader.text().toString().trimmed();          }          else if (stack.back() == FILTER_COMMENT)          {            currentComment = reader.text().toString().trimmed();          }        }        else if (stack.size() == 1 && stack.back() == FILTER_PROGRESS)        {          currentProgress = reader.text().toString().toFloat();        }        else if (stack.size() == 1 && stack.back() == FILTER_PROGRESS_TEXT)        {          currentComment = reader.text().toString();        }        else if (stack.size() == 1 && stack.back() == FILTER_RESULT)        {          currentResultValue = reader.text().toString();        }        break;      }      case QXmlStreamReader::StartElement:      {        QStringRef name = reader.name();        QString parent;        if (!stack.empty()) parent = stack.back();        if (name.compare("module-root") != 0)        {          stack.push_back(name.toString());        }        if (name.compare(FILTER_START, Qt::CaseInsensitive) == 0 ||            name.compare(FILTER_PROGRESS, Qt::CaseInsensitive) == 0 ||            name.compare(FILTER_PROGRESS_TEXT, Qt::CaseInsensitive) == 0 ||            name.compare(FILTER_RESULT, Qt::CaseInsensitive) == 0 ||            name.compare(FILTER_END, Qt::CaseInsensitive) == 0)        {          if (!parent.isEmpty())          {            unexpectedNestedElement(name.toString());            break;          }          if (name.compare(FILTER_START, Qt::CaseInsensitive) == 0)          {            currentName = QString();            currentComment = QString();            currentProgress = 0;          }          else if (name.compare(FILTER_PROGRESS_TEXT, Qt::CaseInsensitive) == 0)          {            currentProgress = reader.attributes().value("progress").toString().toFloat();          }          else if (name.compare(FILTER_RESULT, Qt::CaseInsensitive) == 0)          {            currentResultParameter = reader.attributes().value("name").toString();            currentResultValue.clear();          }        }        break;      }      case QXmlStreamReader::EndElement:      {        QStringRef name = reader.name();        QString curr;        QString parent;        if (!stack.empty())        {          curr = stack.back();          stack.pop_back();          if (!stack.empty()) parent = stack.back();        }        if (parent.isEmpty())        {          if (!outputData.isEmpty())          {            emit q->outputDataAvailable(outputData);            outputData.clear();          }          if (name.compare(FILTER_START, Qt::CaseInsensitive) == 0)          {            emit q->filterStarted(currentName, currentComment);            currentComment = QString();          }          else if (name.compare(FILTER_PROGRESS, Qt::CaseInsensitive) == 0)          {            emit q->filterProgress(currentProgress, QString());          }          else if (name.compare(FILTER_PROGRESS_TEXT, Qt::CaseInsensitive) == 0)          {            emit q->filterProgress(currentProgress, currentComment);            currentComment = QString();          }          else if (name.compare(FILTER_RESULT, Qt::CaseInsensitive) == 0)          {            emit q->filterResult(currentResultParameter, currentResultValue);          }          else if (name.compare(FILTER_END, Qt::CaseInsensitive) == 0)          {            emit q->filterFinished(currentName, currentComment);            currentName = QString();            currentComment = QString();          }        }        break;      }      default:        break;      }      type = reader.readNext();    }    if (type == QXmlStreamReader::Invalid && reader.error() != QXmlStreamReader::PrematureEndOfDocumentError)    {      if (!error)      {        error = true;        emit q->filterXmlError(QString("Error parsing XML at line %1, column %2: ")                               .arg(reader.lineNumber()).arg(reader.columnNumber()) + reader.errorString());      }    }  }  void unexpectedNestedElement(const QString& element)  {    if (!error)    {      error = true;      emit q->filterXmlError(QString("\"%1\" must be a top-level element, found at line %2.")                             .arg(element).arg(reader.lineNumber()));    }  }  QIODevice* input;  QProcess* process;  qint64 readPos;  ctkCmdLineModuleXmlProgressWatcher* q;  bool error;  QXmlStreamReader reader;  QList<QString> stack;  QString currentName;  QString currentComment;  float currentProgress;  QString currentResultParameter;  QString currentResultValue;};//----------------------------------------------------------------------------ctkCmdLineModuleXmlProgressWatcher::ctkCmdLineModuleXmlProgressWatcher(QIODevice* input)  : d(new ctkCmdLineModuleXmlProgressWatcherPrivate(input, this)){  if (d->input == NULL) return;  if (!(d->input->openMode() & QIODevice::ReadOnly))  {    input->open(QIODevice::ReadOnly);  }  connect(d->input, SIGNAL(readyRead()), SLOT(_q_readyRead()));}//----------------------------------------------------------------------------ctkCmdLineModuleXmlProgressWatcher::ctkCmdLineModuleXmlProgressWatcher(QProcess* input)  : d(new ctkCmdLineModuleXmlProgressWatcherPrivate(input, this)){  if (d->input == NULL) return;  connect(input, SIGNAL(readyReadStandardOutput()), SLOT(_q_readyRead()));  connect(input, SIGNAL(readyReadStandardError()), SLOT(_q_readyReadError()));}//----------------------------------------------------------------------------ctkCmdLineModuleXmlProgressWatcher::~ctkCmdLineModuleXmlProgressWatcher(){}#include "moc_ctkCmdLineModuleXmlProgressWatcher.h"
 |