| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170 | /*=============================================================================  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 "ctkCmdLineModuleProcessWatcher_p.h"#include "ctkCmdLineModuleFuture.h"#include <QVariant>#include <QProcess>#ifdef Q_OS_UNIX#include <signal.h>#endif//----------------------------------------------------------------------------ctkCmdLineModuleProcessWatcher::ctkCmdLineModuleProcessWatcher(QProcess& process, const QString& location,                                                               ctkCmdLineModuleFutureInterface &futureInterface)  : process(process), location(location), futureInterface(futureInterface), processXmlWatcher(&process),    processPaused(false), progressValue(0){  // The reported float value in the range [0.0,1.0] for the progress is scaled to [0,1000].  // Value 1001 is reserved for the last "filter-end" output, which is reported as a progress event.  // Value 1002 is reserved internally to report process termination.  futureInterface.setProgressRange(0, 1002);  connect(&processXmlWatcher, SIGNAL(filterStarted(QString,QString)), SLOT(filterStarted(QString,QString)));  connect(&processXmlWatcher, SIGNAL(filterProgress(float,QString)), SLOT(filterProgress(float,QString)));  connect(&processXmlWatcher, SIGNAL(filterResult(QString,QString)), SLOT(filterResult(QString,QString)));  connect(&processXmlWatcher, SIGNAL(filterFinished(QString,QString)), SLOT(filterFinished(QString,QString)));  connect(&processXmlWatcher, SIGNAL(filterXmlError(QString)), SLOT(filterXmlError(QString)));  connect(&processXmlWatcher, SIGNAL(outputDataAvailable(QByteArray)), SLOT(outputDataAvailable(QByteArray)));  connect(&processXmlWatcher, SIGNAL(errorDataAvailable(QByteArray)), SLOT(errorDataAvailable(QByteArray)));  connect(&futureWatcher, SIGNAL(canceled()), SLOT(cancelProcess()));#ifdef Q_OS_UNIX  connect(&futureWatcher, SIGNAL(resumed()), SLOT(resumeProcess()));  // Due to Qt bug 12152, we cannot listen to the "paused" signal because it is  // not emitted directly when the QFuture is paused. Instead, it is emitted after  // resuming the future, after the "resume" signal has been emitted...  //connect(&futureWatcher, SIGNAL(paused()), SLOT(pauseProcess()));  connect(&pollPauseTimer, SIGNAL(timeout()), this, SLOT(pauseProcess()));  pollPauseTimer.start(500);#endif  futureWatcher.setFuture(futureInterface.future());}//----------------------------------------------------------------------------void ctkCmdLineModuleProcessWatcher::filterStarted(const QString& name, const QString& comment){  futureInterface.setProgressValueAndText(incrementProgress(), comment.isEmpty() ? tr("Starting") + name : comment);}//----------------------------------------------------------------------------void ctkCmdLineModuleProcessWatcher::filterProgress(float progress, const QString& comment){  futureInterface.setProgressValueAndText(updateProgress(progress), comment);}//----------------------------------------------------------------------------void ctkCmdLineModuleProcessWatcher::filterResult(const QString ¶meter, const QString &value){  futureInterface.reportResult(ctkCmdLineModuleResult(parameter, value));}//----------------------------------------------------------------------------void ctkCmdLineModuleProcessWatcher::filterFinished(const QString& name, const QString& comment){  int progressValue = incrementProgress();  if (progressValue == 1000) progressValue = 1001;  futureInterface.setProgressValueAndText(progressValue, comment.isEmpty() ? tr("Finished ") + name : comment);}//----------------------------------------------------------------------------void ctkCmdLineModuleProcessWatcher::filterXmlError(const QString &error){  qDebug().nospace() << "[Module " << location << "]: " << error;}//----------------------------------------------------------------------------void ctkCmdLineModuleProcessWatcher::pauseProcess(){  if (processPaused || !futureInterface.isPaused()) return;#ifdef Q_OS_UNIX  if (::kill(process.pid(), SIGSTOP))  {    // error    futureInterface.setPaused(false);  }  else  {    processPaused = true;  }#endif}//----------------------------------------------------------------------------void ctkCmdLineModuleProcessWatcher::resumeProcess(){  if (!processPaused) return;#ifdef Q_OS_UNIX  if(::kill(process.pid(), SIGCONT))  {    // error    futureInterface.setPaused(true);  }  else  {    processPaused = false;  }#endif}//----------------------------------------------------------------------------void ctkCmdLineModuleProcessWatcher::cancelProcess(){  process.kill();}//----------------------------------------------------------------------------void ctkCmdLineModuleProcessWatcher::outputDataAvailable(const QByteArray &outputData){  futureInterface.reportOutputData(outputData);}//----------------------------------------------------------------------------void ctkCmdLineModuleProcessWatcher::errorDataAvailable(const QByteArray &errorData){  futureInterface.reportErrorData(errorData);}//----------------------------------------------------------------------------int ctkCmdLineModuleProcessWatcher::updateProgress(float progress){  progressValue = static_cast<int>(progress * 1000.0f);  // normalize the value to lie between 0 and 1000.  // 0 is reported when the process starts and 1001 is reserved for  // reporting completion.  if (progressValue < 1) progressValue = 1;  if (progressValue > 1000) progressValue = 1000;  return progressValue;}//----------------------------------------------------------------------------int ctkCmdLineModuleProcessWatcher::incrementProgress(){  if (++progressValue > 1000) progressValue = 1000;  return progressValue;}
 |