common: Represent core/client features as string list in the protocol
[quassel.git] / src / common / internalpeer.cpp
1 /***************************************************************************
2  *   Copyright (C) 2005-2016 by the Quassel Project                        *
3  *   devel@quassel-irc.org                                                 *
4  *                                                                         *
5  *   This program is free software; you can redistribute it and/or modify  *
6  *   it under the terms of the GNU General Public License as published by  *
7  *   the Free Software Foundation; either version 2 of the License, or     *
8  *   (at your option) version 3.                                           *
9  *                                                                         *
10  *   This program is distributed in the hope that it will be useful,       *
11  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
12  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
13  *   GNU General Public License for more details.                          *
14  *                                                                         *
15  *   You should have received a copy of the GNU General Public License     *
16  *   along with this program; if not, write to the                         *
17  *   Free Software Foundation, Inc.,                                       *
18  *   51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.         *
19  ***************************************************************************/
20
21 #include <QCoreApplication>
22 #include <QThread>
23
24 #include "internalpeer.h"
25
26 using namespace Protocol;
27
28 template<class T>
29 class PeerMessageEvent : public QEvent
30 {
31 public:
32     PeerMessageEvent(InternalPeer *sender, InternalPeer::EventType eventType, const T &message)
33     : QEvent(QEvent::Type(eventType)), sender(sender), message(message) {}
34     InternalPeer *sender;
35     T message;
36 };
37
38
39 InternalPeer::InternalPeer(QObject *parent)
40     : Peer(0, parent),
41     _proxy(0),
42     _peer(0),
43     _isOpen(true)
44 {
45     setFeatures(Quassel::Features{});
46 }
47
48
49 InternalPeer::~InternalPeer()
50 {
51     if (_isOpen)
52         emit disconnected();
53 }
54
55
56 QString InternalPeer::description() const
57 {
58     return tr("internal connection");
59 }
60
61 QString InternalPeer::address() const
62 {
63     return tr("internal connection");
64 }
65
66 quint16 InternalPeer::port() const
67 {
68     return 0;
69 }
70
71 bool InternalPeer::isOpen() const
72 {
73     return true;
74 }
75
76
77 bool InternalPeer::isSecure() const
78 {
79     return true;
80 }
81
82
83 bool InternalPeer::isLocal() const
84 {
85     return true;
86 }
87
88
89 void InternalPeer::close(const QString &reason)
90 {
91     // FIXME
92     Q_UNUSED(reason)
93     qWarning() << "closing not implemented!";
94 }
95
96
97 int InternalPeer::lag() const
98 {
99     return 0;
100 }
101
102
103 ::SignalProxy *InternalPeer::signalProxy() const
104 {
105     return _proxy;
106 }
107
108
109 void InternalPeer::setSignalProxy(::SignalProxy *proxy)
110 {
111     if (!proxy && _proxy) {
112         _proxy = 0;
113         if (_isOpen) {
114             _isOpen = false;
115             emit disconnected();
116         }
117         return;
118     }
119
120     if (proxy && !_proxy) {
121         _proxy = proxy;
122         return;
123     }
124
125     qWarning() << Q_FUNC_INFO << "Changing the SignalProxy is not supported!";
126 }
127
128
129 void InternalPeer::setPeer(InternalPeer *peer)
130 {
131     if (_peer) {
132         qWarning() << Q_FUNC_INFO << "Peer already set, ignoring!";
133         return;
134     }
135     _peer = peer;
136     connect(peer, SIGNAL(disconnected()), SLOT(peerDisconnected()));
137 }
138
139
140 void InternalPeer::peerDisconnected()
141 {
142     disconnect(_peer, 0, this, 0);
143     _peer = 0;
144     if (_isOpen) {
145         _isOpen = false;
146         emit disconnected();
147     }
148 }
149
150
151 void InternalPeer::dispatch(const SyncMessage &msg)
152 {
153     dispatch(SyncMessageEvent, msg);
154 }
155
156
157 void InternalPeer::dispatch(const RpcCall &msg)
158 {
159     dispatch(RpcCallEvent, msg);
160 }
161
162
163 void InternalPeer::dispatch(const InitRequest &msg)
164 {
165     dispatch(InitRequestEvent, msg);
166 }
167
168
169 void InternalPeer::dispatch(const InitData &msg)
170 {
171     dispatch(InitDataEvent, msg);
172 }
173
174
175 template<class T>
176 void InternalPeer::dispatch(EventType eventType, const T &msg)
177 {
178     if (!_peer) {
179         qWarning() << Q_FUNC_INFO << "Cannot dispatch a message without a peer!";
180         return;
181     }
182
183     if(QThread::currentThread() == _peer->thread())
184         _peer->handle(msg);
185     else
186         QCoreApplication::postEvent(_peer, new PeerMessageEvent<T>(this, eventType, msg));
187 }
188
189
190 void InternalPeer::customEvent(QEvent *event)
191 {
192     switch ((int)event->type()) {
193         case SyncMessageEvent: {
194             PeerMessageEvent<SyncMessage> *e = static_cast<PeerMessageEvent<SyncMessage> *>(event);
195             handle(e->message);
196             break;
197         }
198         case RpcCallEvent: {
199             PeerMessageEvent<RpcCall> *e = static_cast<PeerMessageEvent<RpcCall> *>(event);
200             handle(e->message);
201             break;
202         }
203         case InitRequestEvent: {
204             PeerMessageEvent<InitRequest> *e = static_cast<PeerMessageEvent<InitRequest> *>(event);
205             handle(e->message);
206             break;
207         }
208         case InitDataEvent: {
209             PeerMessageEvent<InitData> *e = static_cast<PeerMessageEvent<InitData> *>(event);
210             handle(e->message);
211             break;
212         }
213
214         default:
215             qWarning() << Q_FUNC_INFO << "Received unknown custom event:" << event->type();
216             return;
217     }
218
219     event->accept();
220 }