X-Git-Url: https://git.quassel-irc.org/?p=quassel.git;a=blobdiff_plain;f=src%2Fcommon%2Fsignalproxy.cpp;h=f4ca5c1501aa3321228b38d5d33297edd4261454;hp=f05a509f1abe2e405604f77ab1b6d0cbaaf0e8c2;hb=2a89d1b0c473f92c5cf9fb803d99e8b51b2c68e7;hpb=76db8cdfbeffaaba359c8e80cf2146da9e9e7f8a diff --git a/src/common/signalproxy.cpp b/src/common/signalproxy.cpp index f05a509f..f4ca5c15 100644 --- a/src/common/signalproxy.cpp +++ b/src/common/signalproxy.cpp @@ -18,39 +18,31 @@ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * ***************************************************************************/ -#include "signalproxy.h" - -#include -#include -#include +#include #include -#include -#include -#include -#include -#include #include #include -#include +#include + #ifdef HAVE_SSL -#include + #include #endif -#include -#include -#include -#include +#include "signalproxy.h" + +#include "peer.h" #include "protocol.h" #include "syncableobject.h" #include "util.h" +#include "types.h" using namespace Protocol; class RemovePeerEvent : public QEvent { public: - RemovePeerEvent(SignalProxy::AbstractPeer *peer) : QEvent(QEvent::Type(SignalProxy::RemovePeerEvent)), peer(peer) {} - SignalProxy::AbstractPeer *peer; + RemovePeerEvent(Peer *peer) : QEvent(QEvent::Type(SignalProxy::RemovePeerEvent)), peer(peer) {} + Peer *peer; }; @@ -156,7 +148,11 @@ int SignalProxy::SignalRelay::qt_metacall(QMetaObject::Call _c, int _id, void ** params << QVariant(argTypes[i], _a[i+1]); } - proxy()->dispatch(RpcCall(signal.signature, params)); + if (argTypes.size() >= 1 && argTypes[0] == qMetaTypeId() && proxy()->proxyMode() == SignalProxy::Server) { + Peer *peer = params[0].value(); + proxy()->dispatch(peer, RpcCall(signal.signature, params)); + } else + proxy()->dispatch(RpcCall(signal.signature, params)); } _id -= _slots.count(); } @@ -257,7 +253,7 @@ void SignalProxy::setMaxHeartBeatCount(int max) } -bool SignalProxy::addPeer(AbstractPeer *peer) +bool SignalProxy::addPeer(Peer *peer) { if (!peer) return false; @@ -300,14 +296,14 @@ void SignalProxy::removeAllPeers() { Q_ASSERT(proxyMode() == Server || _peers.count() <= 1); // wee need to copy that list since we modify it in the loop - QSet peers = _peers; - foreach(AbstractPeer *peer, peers) { + QSet peers = _peers; + foreach(Peer *peer, peers) { removePeer(peer); } } -void SignalProxy::removePeer(AbstractPeer *peer) +void SignalProxy::removePeer(Peer *peer) { if (!peer) { qWarning() << Q_FUNC_INFO << "Trying to remove a null peer!"; @@ -342,7 +338,7 @@ void SignalProxy::removePeer(AbstractPeer *peer) void SignalProxy::removePeerBySender() { - removePeer(qobject_cast(sender())); + removePeer(qobject_cast(sender())); } @@ -502,7 +498,7 @@ void SignalProxy::stopSynchronize(SyncableObject *obj) template void SignalProxy::dispatch(const T &protoMessage) { - foreach (AbstractPeer *peer, _peers) { + foreach (Peer *peer, _peers) { if (peer->isOpen()) peer->dispatch(protoMessage); else @@ -511,30 +507,40 @@ void SignalProxy::dispatch(const T &protoMessage) } -void SignalProxy::handle(SignalProxy::AbstractPeer *peer, const SyncMessage &syncMessage) +template +void SignalProxy::dispatch(Peer *peer, const T &protoMessage) +{ + if (peer && peer->isOpen()) + peer->dispatch(protoMessage); + else + QCoreApplication::postEvent(this, new ::RemovePeerEvent(peer)); +} + + +void SignalProxy::handle(Peer *peer, const SyncMessage &syncMessage) { - if (!_syncSlave.contains(syncMessage.className()) || !_syncSlave[syncMessage.className()].contains(syncMessage.objectName())) { - qWarning() << QString("no registered receiver for sync call: %1::%2 (objectName=\"%3\"). Params are:").arg(syncMessage.className(), syncMessage.slotName(), syncMessage.objectName()) - << syncMessage.params(); + if (!_syncSlave.contains(syncMessage.className) || !_syncSlave[syncMessage.className].contains(syncMessage.objectName)) { + qWarning() << QString("no registered receiver for sync call: %1::%2 (objectName=\"%3\"). Params are:").arg(syncMessage.className, syncMessage.slotName, syncMessage.objectName) + << syncMessage.params; return; } - SyncableObject *receiver = _syncSlave[syncMessage.className()][syncMessage.objectName()]; + SyncableObject *receiver = _syncSlave[syncMessage.className][syncMessage.objectName]; ExtendedMetaObject *eMeta = extendedMetaObject(receiver); - if (!eMeta->slotMap().contains(syncMessage.slotName())) { - qWarning() << QString("no matching slot for sync call: %1::%2 (objectName=\"%3\"). Params are:").arg(syncMessage.className(), syncMessage.slotName(), syncMessage.objectName()) - << syncMessage.params(); + if (!eMeta->slotMap().contains(syncMessage.slotName)) { + qWarning() << QString("no matching slot for sync call: %1::%2 (objectName=\"%3\"). Params are:").arg(syncMessage.className, syncMessage.slotName, syncMessage.objectName) + << syncMessage.params; return; } - int slotId = eMeta->slotMap()[syncMessage.slotName()]; + int slotId = eMeta->slotMap()[syncMessage.slotName]; if (proxyMode() != eMeta->receiverMode(slotId)) { qWarning("SignalProxy::handleSync(): invokeMethod for \"%s\" failed. Wrong ProxyMode!", eMeta->methodName(slotId).constData()); return; } QVariant returnValue((QVariant::Type)eMeta->returnType(slotId)); - if (!invokeSlot(receiver, slotId, syncMessage.params(), returnValue)) { + if (!invokeSlot(receiver, slotId, syncMessage.params, returnValue, peer)) { qWarning("SignalProxy::handleSync(): invokeMethod for \"%s\" failed ", eMeta->methodName(slotId).constData()); return; } @@ -543,9 +549,9 @@ void SignalProxy::handle(SignalProxy::AbstractPeer *peer, const SyncMessage &syn int receiverId = eMeta->receiveMap()[slotId]; QVariantList returnParams; if (eMeta->argTypes(receiverId).count() > 1) - returnParams << syncMessage.params(); + returnParams << syncMessage.params; returnParams << returnValue; - peer->dispatch(SyncMessage(syncMessage.className(), syncMessage.objectName(), eMeta->methodName(receiverId), returnParams)); + peer->dispatch(SyncMessage(syncMessage.className, syncMessage.objectName, eMeta->methodName(receiverId), returnParams)); } // send emit update signal @@ -553,57 +559,55 @@ void SignalProxy::handle(SignalProxy::AbstractPeer *peer, const SyncMessage &syn } -void SignalProxy::handle(SignalProxy::AbstractPeer *peer, const InitRequest &initRequest) +void SignalProxy::handle(Peer *peer, const InitRequest &initRequest) { - if (!_syncSlave.contains(initRequest.className())) { + if (!_syncSlave.contains(initRequest.className)) { qWarning() << "SignalProxy::handleInitRequest() received initRequest for unregistered Class:" - << initRequest.className(); + << initRequest.className; return; } - if (!_syncSlave[initRequest.className()].contains(initRequest.objectName())) { + if (!_syncSlave[initRequest.className].contains(initRequest.objectName)) { qWarning() << "SignalProxy::handleInitRequest() received initRequest for unregistered Object:" - << initRequest.className() << initRequest.objectName(); + << initRequest.className << initRequest.objectName; return; } - SyncableObject *obj = _syncSlave[initRequest.className()][initRequest.objectName()]; - peer->dispatch(InitData(initRequest.className(), initRequest.objectName(), initData(obj))); + SyncableObject *obj = _syncSlave[initRequest.className][initRequest.objectName]; + peer->dispatch(InitData(initRequest.className, initRequest.objectName, initData(obj))); } -void SignalProxy::handle(SignalProxy::AbstractPeer *peer, const InitData &initData) +void SignalProxy::handle(Peer *peer, const InitData &initData) { Q_UNUSED(peer) - if (!_syncSlave.contains(initData.className())) { + if (!_syncSlave.contains(initData.className)) { qWarning() << "SignalProxy::handleInitData() received initData for unregistered Class:" - << initData.className(); + << initData.className; return; } - if (!_syncSlave[initData.className()].contains(initData.objectName())) { + if (!_syncSlave[initData.className].contains(initData.objectName)) { qWarning() << "SignalProxy::handleInitData() received initData for unregistered Object:" - << initData.className() << initData.objectName(); + << initData.className << initData.objectName; return; } - SyncableObject *obj = _syncSlave[initData.className()][initData.objectName()]; - setInitData(obj, initData.initData()); + SyncableObject *obj = _syncSlave[initData.className][initData.objectName]; + setInitData(obj, initData.initData); } -void SignalProxy::handle(SignalProxy::AbstractPeer *peer, const RpcCall &rpcCall) +void SignalProxy::handle(Peer *peer, const RpcCall &rpcCall) { - Q_UNUSED(peer) - QObject *receiver; int methodId; - SlotHash::const_iterator slot = _attachedSlots.constFind(rpcCall.slotName()); - while (slot != _attachedSlots.constEnd() && slot.key() == rpcCall.slotName()) { + SlotHash::const_iterator slot = _attachedSlots.constFind(rpcCall.slotName); + while (slot != _attachedSlots.constEnd() && slot.key() == rpcCall.slotName) { receiver = (*slot).first; methodId = (*slot).second; - if (!invokeSlot(receiver, methodId, rpcCall.params())) { + if (!invokeSlot(receiver, methodId, rpcCall.params, peer)) { ExtendedMetaObject *eMeta = extendedMetaObject(receiver); qWarning("SignalProxy::handleSignal(): invokeMethod for \"%s\" failed ", eMeta->methodName(methodId).constData()); } @@ -612,7 +616,7 @@ void SignalProxy::handle(SignalProxy::AbstractPeer *peer, const RpcCall &rpcCall } -bool SignalProxy::invokeSlot(QObject *receiver, int methodId, const QVariantList ¶ms, QVariant &returnValue) +bool SignalProxy::invokeSlot(QObject *receiver, int methodId, const QVariantList ¶ms, QVariant &returnValue, Peer *peer) { ExtendedMetaObject *eMeta = extendedMetaObject(receiver); const QList args = eMeta->argTypes(methodId); @@ -640,7 +644,12 @@ bool SignalProxy::invokeSlot(QObject *receiver, int methodId, const QVariantList qWarning() << "SignalProxy::invokeSlot(): incompatible param types to invoke" << eMeta->methodName(methodId); return false; } - _a[i+1] = const_cast(params[i].constData()); + // if first arg is a PeerPtr, replace it by the address of the peer originally receiving the RpcCall + if (peer && i == 0 && args[0] == qMetaTypeId()) { + QVariant v = QVariant::fromValue(peer); + _a[1] = const_cast(v.constData()); + } else + _a[i+1] = const_cast(params[i].constData()); } if (returnValue.type() != QVariant::Invalid) @@ -661,10 +670,10 @@ bool SignalProxy::invokeSlot(QObject *receiver, int methodId, const QVariantList } -bool SignalProxy::invokeSlot(QObject *receiver, int methodId, const QVariantList ¶ms) +bool SignalProxy::invokeSlot(QObject *receiver, int methodId, const QVariantList ¶ms, Peer *peer) { QVariant ret; - return invokeSlot(receiver, methodId, params, ret); + return invokeSlot(receiver, methodId, params, ret, peer); } @@ -732,7 +741,11 @@ void SignalProxy::sync_call__(const SyncableObject *obj, SignalProxy::ProxyMode params << QVariant(argTypes[i], va_arg(ap, void *)); } - dispatch(SyncMessage(eMeta->metaObject()->className(), obj->objectName(), QByteArray(funcname), params)); + if (argTypes.size() >= 1 && argTypes[0] == qMetaTypeId() && proxyMode() == SignalProxy::Server) { + Peer *peer = params[0].value(); + dispatch(peer, SyncMessage(eMeta->metaObject()->className(), obj->objectName(), QByteArray(funcname), params)); + } else + dispatch(SyncMessage(eMeta->metaObject()->className(), obj->objectName(), QByteArray(funcname), params)); } @@ -772,7 +785,7 @@ void SignalProxy::updateSecureState() bool wasSecure = _secure; _secure = !_peers.isEmpty(); - foreach (const AbstractPeer *peer, _peers) { + foreach (const Peer *peer, _peers) { _secure &= peer->isSecure(); }