annotate vamp-client/ProcessQtTransport.h @ 113:ac4a9518e1cc

Only if server not running
author Chris Cannam <cannam@all-day-breakfast.com>
date Tue, 25 Oct 2016 15:51:55 +0100
parents d74dfc11927c
children 95c94a86c902
rev   line source
cannam@111 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
cannam@111 2
c@94 3 #ifndef PIPER_PROCESS_QT_TRANSPORT_H
c@94 4 #define PIPER_PROCESS_QT_TRANSPORT_H
c@94 5
c@94 6 #include "SynchronousTransport.h"
c@94 7
c@94 8 #include <QProcess>
c@94 9 #include <QString>
c@100 10 #include <QMutex>
c@94 11
c@94 12 #include <iostream>
c@94 13
c@97 14 namespace piper_vamp {
c@97 15 namespace client {
c@94 16
c@100 17 /**
c@100 18 * A SynchronousTransport implementation that spawns a sub-process
c@100 19 * using Qt's QProcess abstraction and talks to it via stdin/stdout
c@100 20 * channels. Calls are completely serialized; the protocol only
c@100 21 * supports one call in process at a time, and therefore the transport
c@100 22 * only allows one at a time. This class is thread-safe because it
c@100 23 * serializes explicitly using a mutex.
c@100 24 */
c@94 25 class ProcessQtTransport : public SynchronousTransport
c@94 26 {
c@94 27 public:
c@101 28 ProcessQtTransport(std::string processName) :
c@94 29 m_completenessChecker(0) {
c@94 30 m_process = new QProcess();
c@94 31 m_process->setReadChannel(QProcess::StandardOutput);
c@94 32 m_process->setProcessChannelMode(QProcess::ForwardedErrorChannel);
cannam@113 33 m_process->start(processName.c_str());
c@94 34 if (!m_process->waitForStarted()) {
cannam@113 35 if (m_process->state() == QProcess::NotRunning) {
cannam@113 36 QProcess::ProcessError err = m_process->error();
cannam@113 37 if (err == QProcess::FailedToStart) {
cannam@113 38 std::cerr << "Unable to start server process "
cannam@113 39 << processName << std::endl;
cannam@113 40 } else if (err == QProcess::Crashed) {
cannam@113 41 std::cerr << "Server process " << processName
cannam@113 42 << " crashed on startup" << std::endl;
cannam@113 43 } else {
cannam@113 44 std::cerr << "Server process " << processName
cannam@113 45 << " failed on startup with error code "
cannam@113 46 << err << std::endl;
cannam@113 47 }
cannam@113 48 delete m_process;
cannam@113 49 m_process = nullptr;
cannam@111 50 }
c@94 51 }
c@94 52 }
c@94 53
c@94 54 ~ProcessQtTransport() {
c@94 55 if (m_process) {
c@94 56 if (m_process->state() != QProcess::NotRunning) {
c@94 57 m_process->closeWriteChannel();
c@94 58 m_process->waitForFinished(200);
c@94 59 m_process->close();
c@94 60 m_process->waitForFinished();
c@94 61 std::cerr << "server exited" << std::endl;
c@94 62 }
c@94 63 delete m_process;
c@94 64 }
c@94 65 }
c@94 66
c@94 67 void
cannam@111 68 setCompletenessChecker(MessageCompletenessChecker *checker) override {
c@94 69 //!!! ownership?
c@94 70 m_completenessChecker = checker;
c@94 71 }
c@94 72
c@94 73 bool
c@94 74 isOK() const override {
c@94 75 return m_process != nullptr;
c@94 76 }
c@94 77
c@94 78 std::vector<char>
c@94 79 call(const char *ptr, size_t size) override {
c@94 80
c@100 81 QMutexLocker locker(&m_mutex);
c@100 82
c@94 83 if (!m_completenessChecker) {
c@94 84 throw std::logic_error("No completeness checker set on transport");
c@94 85 }
c@94 86
c@94 87 m_process->write(ptr, size);
c@94 88
c@94 89 std::vector<char> buffer;
c@94 90 bool complete = false;
c@94 91
c@94 92 while (!complete) {
c@94 93
c@94 94 qint64 byteCount = m_process->bytesAvailable();
c@94 95
c@101 96 if (!byteCount) {
c@108 97 std::cerr << "waiting for data from server..." << std::endl;
c@101 98 m_process->waitForReadyRead(1000);
c@94 99 if (m_process->state() == QProcess::NotRunning) {
c@94 100 std::cerr << "ERROR: Subprocess exited: Load failed" << std::endl;
c@94 101 throw std::runtime_error("Piper server exited unexpectedly");
c@94 102 }
c@94 103 } else {
c@94 104 size_t formerSize = buffer.size();
c@94 105 buffer.resize(formerSize + byteCount);
c@94 106 m_process->read(buffer.data() + formerSize, byteCount);
c@94 107 complete = m_completenessChecker->isComplete(buffer);
c@94 108 }
c@94 109 }
c@94 110
c@94 111 return buffer;
c@94 112 }
c@94 113
c@94 114 private:
c@94 115 MessageCompletenessChecker *m_completenessChecker; //!!! I don't own this (currently)
c@94 116 QProcess *m_process; // I own this
c@100 117 QMutex m_mutex;
c@94 118 };
c@94 119
c@94 120 }
c@94 121 }
c@94 122
c@94 123 #endif