* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
***************************************************************************/
+#include <QPointer>
+#include <QTimer>
+
#include "core.h"
#include "coresession.h"
#include "internalpeer.h"
#include "sessionthread.h"
#include "signalproxy.h"
-SessionThread::SessionThread(UserId uid, bool restoreState, bool strictIdentEnabled, QObject *parent)
- : QThread(parent),
- _session(0),
- _user(uid),
- _sessionInitialized(false),
- _restoreState(restoreState),
- _strictIdentEnabled(strictIdentEnabled)
-{
- connect(this, SIGNAL(initialized()), this, SLOT(setSessionInitialized()));
-}
-
+namespace {
-SessionThread::~SessionThread()
+class Worker : public QObject
{
- // shut down thread gracefully
- quit();
- wait();
-}
+ Q_OBJECT
+
+public:
+ Worker(UserId userId, bool restoreState, bool strictIdentEnabled)
+ : _userId{userId}
+ , _restoreState{restoreState}
+ , _strictIdentEnabled{strictIdentEnabled}
+ {
+ }
+public slots:
+ void initialize()
+ {
+ _session = new CoreSession{_userId, _restoreState, _strictIdentEnabled, this};
+ connect(_session, SIGNAL(destroyed()), QThread::currentThread(), SLOT(quit()));
+ connect(_session, SIGNAL(sessionState(Protocol::SessionState)), Core::instance(), SIGNAL(sessionState(Protocol::SessionState)));
+ emit initialized();
+ }
-CoreSession *SessionThread::session()
-{
- return _session;
-}
+ void shutdown()
+ {
+ if (_session) {
+ _session->shutdown();
+ }
+ }
+ void addClient(Peer *peer)
+ {
+ if (!_session) {
+ qWarning() << "Session not initialized!";
+ return;
+ }
+
+ auto remotePeer = qobject_cast<RemotePeer*>(peer);
+ if (remotePeer) {
+ _session->addClient(remotePeer);
+ return;
+ }
+ auto internalPeer = qobject_cast<InternalPeer*>(peer);
+ if (internalPeer) {
+ _session->addClient(internalPeer);
+ return;
+ }
+
+ qWarning() << "SessionThread::addClient() received invalid peer!" << peer;
+ }
-UserId SessionThread::user()
-{
- return _user;
-}
+signals:
+ void initialized();
+private:
+ UserId _userId;
+ bool _restoreState;
+ bool _strictIdentEnabled; ///< Whether or not strict ident mode is enabled, locking users' idents to Quassel username
+ QPointer<CoreSession> _session;
+};
-bool SessionThread::isSessionInitialized()
+} // anon
+
+SessionThread::SessionThread(UserId uid, bool restoreState, bool strictIdentEnabled, QObject *parent)
+ : QObject(parent)
{
- return _sessionInitialized;
+ auto worker = new Worker(uid, restoreState, strictIdentEnabled);
+ worker->moveToThread(&_sessionThread);
+ connect(&_sessionThread, SIGNAL(started()), worker, SLOT(initialize()));
+ connect(&_sessionThread, SIGNAL(finished()), worker, SLOT(deleteLater()));
+ connect(worker, SIGNAL(initialized()), this, SLOT(onSessionInitialized()));
+ connect(worker, SIGNAL(destroyed()), this, SLOT(onSessionDestroyed()));
+
+ connect(this, SIGNAL(addClientToWorker(Peer*)), worker, SLOT(addClient(Peer*)));
+ connect(this, SIGNAL(shutdownSession()), worker, SLOT(shutdown()));
+
+ // Defer thread start through the event loop, so the SessionThread instance is fully constructed before
+ QTimer::singleShot(0, &_sessionThread, SLOT(start()));
}
-void SessionThread::setSessionInitialized()
+SessionThread::~SessionThread()
{
- _sessionInitialized = true;
- foreach(QObject *peer, clientQueue) {
- addClientToSession(peer);
- }
- clientQueue.clear();
+ // shut down thread gracefully
+ _sessionThread.quit();
+ _sessionThread.wait(30000);
}
-// this and the following related methods are executed in the Core thread!
-void SessionThread::addClient(QObject *peer)
+void SessionThread::shutdown()
{
- if (isSessionInitialized()) {
- addClientToSession(peer);
- }
- else {
- clientQueue.append(peer);
- }
+ emit shutdownSession();
}
-void SessionThread::addClientToSession(QObject *peer)
+void SessionThread::onSessionInitialized()
{
- RemotePeer *remote = qobject_cast<RemotePeer *>(peer);
- if (remote) {
- addRemoteClientToSession(remote);
- return;
- }
-
- InternalPeer *internal = qobject_cast<InternalPeer *>(peer);
- if (internal) {
- addInternalClientToSession(internal);
- return;
+ _sessionInitialized = true;
+ for (auto &&peer : _clientQueue) {
+ peer->setParent(nullptr);
+ peer->moveToThread(&_sessionThread);
+ emit addClientToWorker(peer);
}
-
- qWarning() << "SessionThread::addClient() received invalid peer!" << peer;
+ _clientQueue.clear();
}
-void SessionThread::addRemoteClientToSession(RemotePeer *remotePeer)
+void SessionThread::onSessionDestroyed()
{
- remotePeer->setParent(0);
- remotePeer->moveToThread(session()->thread());
- emit addRemoteClient(remotePeer);
+ emit shutdownComplete(this);
}
-
-void SessionThread::addInternalClientToSession(InternalPeer *internalPeer)
+void SessionThread::addClient(Peer *peer)
{
- internalPeer->setParent(0);
- internalPeer->moveToThread(session()->thread());
- emit addInternalClient(internalPeer);
+ if (_sessionInitialized) {
+ peer->setParent(nullptr);
+ peer->moveToThread(&_sessionThread);
+ emit addClientToWorker(peer);
+ }
+ else {
+ _clientQueue.push_back(peer);
+ }
}
-
-void SessionThread::run()
-{
- _session = new CoreSession(user(), _restoreState, _strictIdentEnabled);
- connect(this, SIGNAL(addRemoteClient(RemotePeer*)), _session, SLOT(addClient(RemotePeer*)));
- connect(this, SIGNAL(addInternalClient(InternalPeer*)), _session, SLOT(addClient(InternalPeer*)));
- connect(_session, SIGNAL(sessionState(Protocol::SessionState)), Core::instance(), SIGNAL(sessionState(Protocol::SessionState)));
- emit initialized();
- exec();
- delete _session;
-}
+#include "sessionthread.moc"
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
***************************************************************************/
-#ifndef SESSIONTHREAD_H
-#define SESSIONTHREAD_H
+#pragma once
+
+#include <memory>
-#include <QMutex>
#include <QThread>
#include "types.h"
class CoreSession;
+class Peer;
class InternalPeer;
class RemotePeer;
-class QIODevice;
-class SessionThread : public QThread
+class SessionThread : public QObject
{
Q_OBJECT
SessionThread(UserId user, bool restoreState, bool strictIdentEnabled, QObject *parent = 0);
~SessionThread();
- void run();
-
- CoreSession *session();
- UserId user();
-
public slots:
- void addClient(QObject *peer);
+ void addClient(Peer *peer);
+ void shutdown();
private slots:
- void setSessionInitialized();
+ void onSessionInitialized();
+ void onSessionDestroyed();
signals:
void initialized();
- void shutdown();
+ void shutdownSession();
+ void shutdownComplete(SessionThread *);
- void addRemoteClient(RemotePeer *peer);
- void addInternalClient(InternalPeer *peer);
+ void addClientToWorker(Peer *peer);
private:
- CoreSession *_session;
- UserId _user;
- QList<QObject *> clientQueue;
- bool _sessionInitialized;
- bool _restoreState;
-
- /// Whether or not strict ident mode is enabled, locking users' idents to Quassel username
- bool _strictIdentEnabled;
+ QThread _sessionThread;
+ bool _sessionInitialized{false};
- bool isSessionInitialized();
- void addClientToSession(QObject *peer);
- void addRemoteClientToSession(RemotePeer *remotePeer);
- void addInternalClientToSession(InternalPeer *internalPeer);
+ std::vector<Peer *> _clientQueue;
};
-
-
-#endif