summaryrefslogtreecommitdiff
path: root/qt4/TelepathyQt4/pending-contacts.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'qt4/TelepathyQt4/pending-contacts.cpp')
-rw-r--r--qt4/TelepathyQt4/pending-contacts.cpp468
1 files changed, 468 insertions, 0 deletions
diff --git a/qt4/TelepathyQt4/pending-contacts.cpp b/qt4/TelepathyQt4/pending-contacts.cpp
new file mode 100644
index 000000000..061147b5d
--- /dev/null
+++ b/qt4/TelepathyQt4/pending-contacts.cpp
@@ -0,0 +1,468 @@
+/**
+ * This file is part of TelepathyQt4
+ *
+ * @copyright Copyright (C) 2008 Collabora Ltd. <http://www.collabora.co.uk/>
+ * @copyright Copyright (C) 2008 Nokia Corporation
+ * @license LGPL 2.1
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+
+#include <TelepathyQt4/PendingContacts>
+#include "TelepathyQt4/_gen/pending-contacts.moc.hpp"
+
+#include <TelepathyQt4/Connection>
+#include <TelepathyQt4/ConnectionLowlevel>
+#include <TelepathyQt4/ContactManager>
+#include <TelepathyQt4/ContactFactory>
+#include <TelepathyQt4/PendingContactAttributes>
+#include <TelepathyQt4/PendingHandles>
+#include <TelepathyQt4/ReferencedHandles>
+
+#include "TelepathyQt4/debug-internal.h"
+
+namespace Tp
+{
+
+struct TELEPATHY_QT4_NO_EXPORT PendingContacts::Private
+{
+ enum RequestType
+ {
+ ForHandles,
+ ForIdentifiers,
+ Upgrade
+ };
+
+ Private(PendingContacts *parent, const ContactManagerPtr &manager, const UIntList &handles,
+ const Features &features, const Features &missingFeatures,
+ const QMap<uint, ContactPtr> &satisfyingContacts)
+ : parent(parent),
+ manager(manager),
+ features(features),
+ missingFeatures(missingFeatures),
+ satisfyingContacts(satisfyingContacts),
+ requestType(ForHandles),
+ handles(handles),
+ nested(0)
+ {
+ }
+
+ Private(PendingContacts *parent, const ContactManagerPtr &manager, const QStringList &identifiers,
+ const Features &features)
+ : parent(parent),
+ manager(manager),
+ features(features),
+ requestType(ForIdentifiers),
+ identifiers(identifiers),
+ nested(0)
+ {
+ }
+
+ Private(PendingContacts *parent,
+ const ContactManagerPtr &manager, const QList<ContactPtr> &contactsToUpgrade,
+ const Features &features)
+ : parent(parent),
+ manager(manager),
+ features(features),
+ requestType(Upgrade),
+ contactsToUpgrade(contactsToUpgrade),
+ nested(0)
+ {
+ }
+
+ void setFinished();
+
+ // Public object
+ PendingContacts *parent;
+
+ // Generic parameters
+ ContactManagerPtr manager;
+ Features features;
+ Features missingFeatures;
+ QMap<uint, ContactPtr> satisfyingContacts;
+
+ // Request type specific parameters
+ RequestType requestType;
+ UIntList handles;
+ QStringList identifiers;
+ QList<ContactPtr> contactsToUpgrade;
+ PendingContacts *nested;
+
+ // Results
+ QList<ContactPtr> contacts;
+ UIntList invalidHandles;
+ QStringList validIds;
+ QHash<QString, QPair<QString, QString> > invalidIds;
+
+ ReferencedHandles handlesToInspect;
+};
+
+void PendingContacts::Private::setFinished()
+{
+ ConnectionLowlevelPtr connLowlevel = manager->connection()->lowlevel();
+ UIntList handles = invalidHandles;
+ foreach (uint handle, handles) {
+ if (connLowlevel->hasContactId(handle)) {
+ satisfyingContacts.insert(handle, manager->ensureContact(handle,
+ connLowlevel->contactId(handle), missingFeatures));
+ invalidHandles.removeOne(handle);
+ }
+ }
+
+ parent->setFinished();
+}
+
+/**
+ * \class PendingContacts
+ * \ingroup clientconn
+ * \headerfile TelepathyQt4/pending-contacts.h <TelepathyQt4/PendingContacts>
+ *
+ * \brief The PendingContacts class is used by ContactManager when
+ * creating/updating Contact objects.
+ *
+ * See \ref async_model
+ */
+
+PendingContacts::PendingContacts(const ContactManagerPtr &manager,
+ const UIntList &handles,
+ const Features &features,
+ const Features &missingFeatures,
+ const QStringList &interfaces,
+ const QMap<uint, ContactPtr> &satisfyingContacts,
+ const QSet<uint> &otherContacts,
+ const QString &errorName,
+ const QString &errorMessage)
+ : PendingOperation(manager->connection()),
+ mPriv(new Private(this, manager, handles, features, missingFeatures, satisfyingContacts))
+{
+ if (!errorName.isEmpty()) {
+ setFinishedWithError(errorName, errorMessage);
+ return;
+ }
+
+ if (!otherContacts.isEmpty()) {
+ ConnectionPtr conn = manager->connection();
+ if (conn->interfaces().contains(QLatin1String(TELEPATHY_INTERFACE_CONNECTION_INTERFACE_CONTACTS))) {
+ PendingContactAttributes *attributes =
+ conn->lowlevel()->contactAttributes(otherContacts.toList(),
+ interfaces, true);
+
+ connect(attributes,
+ SIGNAL(finished(Tp::PendingOperation*)),
+ SLOT(onAttributesFinished(Tp::PendingOperation*)));
+ } else {
+ // fallback to just create the contacts
+ PendingHandles *handles = conn->lowlevel()->referenceHandles(HandleTypeContact,
+ otherContacts.toList());
+ connect(handles,
+ SIGNAL(finished(Tp::PendingOperation*)),
+ SLOT(onReferenceHandlesFinished(Tp::PendingOperation*)));
+ }
+ } else {
+ allAttributesFetched();
+ }
+}
+
+PendingContacts::PendingContacts(const ContactManagerPtr &manager,
+ const QStringList &identifiers, const Features &features,
+ const QString &errorName, const QString &errorMessage)
+ : PendingOperation(manager->connection()),
+ mPriv(new Private(this, manager, identifiers, features))
+{
+ if (!errorName.isEmpty()) {
+ setFinishedWithError(errorName, errorMessage);
+ return;
+ }
+
+ ConnectionPtr conn = manager->connection();
+ PendingHandles *handles = conn->lowlevel()->requestHandles(HandleTypeContact, identifiers);
+
+ connect(handles,
+ SIGNAL(finished(Tp::PendingOperation*)),
+ SLOT(onRequestHandlesFinished(Tp::PendingOperation*)));
+}
+
+PendingContacts::PendingContacts(const ContactManagerPtr &manager,
+ const QList<ContactPtr> &contacts, const Features &features,
+ const QString &errorName, const QString &errorMessage)
+ : PendingOperation(manager->connection()),
+ mPriv(new Private(this, manager, contacts, features))
+{
+ if (!errorName.isEmpty()) {
+ setFinishedWithError(errorName, errorMessage);
+ return;
+ }
+
+ UIntList handles;
+ foreach (const ContactPtr &contact, contacts) {
+ handles.push_back(contact->handle()[0]);
+ }
+
+ mPriv->nested = manager->contactsForHandles(handles, features);
+ connect(mPriv->nested,
+ SIGNAL(finished(Tp::PendingOperation*)),
+ SLOT(onNestedFinished(Tp::PendingOperation*)));
+}
+
+/**
+ * Class destructor.
+ */
+PendingContacts::~PendingContacts()
+{
+ delete mPriv;
+}
+
+ContactManagerPtr PendingContacts::manager() const
+{
+ return mPriv->manager;
+}
+
+Features PendingContacts::features() const
+{
+ return mPriv->features;
+}
+
+bool PendingContacts::isForHandles() const
+{
+ return mPriv->requestType == Private::ForHandles;
+}
+
+UIntList PendingContacts::handles() const
+{
+ if (!isForHandles()) {
+ warning() << "Tried to get handles from" << this << "which is not for handles!";
+ }
+
+ return mPriv->handles;
+}
+
+bool PendingContacts::isForIdentifiers() const
+{
+ return mPriv->requestType == Private::ForIdentifiers;
+}
+
+QStringList PendingContacts::identifiers() const
+{
+ if (!isForIdentifiers()) {
+ warning() << "Tried to get identifiers from" << this << "which is not for identifiers!";
+ }
+
+ return mPriv->identifiers;
+}
+
+bool PendingContacts::isUpgrade() const
+{
+ return mPriv->requestType == Private::Upgrade;
+}
+
+QList<ContactPtr> PendingContacts::contactsToUpgrade() const
+{
+ if (!isUpgrade()) {
+ warning() << "Tried to get contacts to upgrade from" << this << "which is not an upgrade!";
+ }
+
+ return mPriv->contactsToUpgrade;
+}
+
+QList<ContactPtr> PendingContacts::contacts() const
+{
+ if (!isFinished()) {
+ warning() << "PendingContacts::contacts() called before finished";
+ } else if (isError()) {
+ warning() << "PendingContacts::contacts() called when errored";
+ }
+
+ return mPriv->contacts;
+}
+
+UIntList PendingContacts::invalidHandles() const
+{
+ if (!isFinished()) {
+ warning() << "PendingContacts::invalidHandles() called before finished";
+ } else if (isError()) {
+ warning() << "PendingContacts::invalidHandles() called when errored";
+ } else if (!isForHandles()) {
+ warning() << "PendingContacts::invalidHandles() called for" << this << "which is for IDs!";
+ }
+
+ return mPriv->invalidHandles;
+}
+
+QStringList PendingContacts::validIdentifiers() const
+{
+ if (!isFinished()) {
+ warning() << "PendingContacts::validIdentifiers called before finished";
+ } else if (!isValid()) {
+ warning() << "PendingContacts::validIdentifiers called when not valid";
+ }
+
+ return mPriv->validIds;
+}
+
+QHash<QString, QPair<QString, QString> > PendingContacts::invalidIdentifiers() const
+{
+ if (!isFinished()) {
+ warning() << "PendingContacts::invalidIdentifiers called before finished";
+ }
+
+ return mPriv->invalidIds;
+}
+
+void PendingContacts::onAttributesFinished(PendingOperation *operation)
+{
+ PendingContactAttributes *pendingAttributes =
+ qobject_cast<PendingContactAttributes *>(operation);
+
+ if (pendingAttributes->isError()) {
+ debug() << "PendingAttrs error" << pendingAttributes->errorName()
+ << "message" << pendingAttributes->errorMessage();
+ setFinishedWithError(pendingAttributes->errorName(), pendingAttributes->errorMessage());
+ return;
+ }
+
+ ReferencedHandles validHandles = pendingAttributes->validHandles();
+ ContactAttributesMap attributes = pendingAttributes->attributes();
+
+ foreach (uint handle, mPriv->handles) {
+ if (!mPriv->satisfyingContacts.contains(handle)) {
+ int indexInValid = validHandles.indexOf(handle);
+ if (indexInValid >= 0) {
+ ReferencedHandles referencedHandle = validHandles.mid(indexInValid, 1);
+ QVariantMap handleAttributes = attributes[handle];
+ mPriv->satisfyingContacts.insert(handle, manager()->ensureContact(referencedHandle,
+ mPriv->missingFeatures, handleAttributes));
+ } else {
+ mPriv->invalidHandles.push_back(handle);
+ }
+ }
+ }
+
+ allAttributesFetched();
+}
+
+void PendingContacts::onRequestHandlesFinished(PendingOperation *operation)
+{
+ PendingHandles *pendingHandles = qobject_cast<PendingHandles *>(operation);
+
+ mPriv->validIds = pendingHandles->validNames();
+ mPriv->invalidIds = pendingHandles->invalidNames();
+
+ if (pendingHandles->isError()) {
+ debug() << "RequestHandles error" << operation->errorName()
+ << "message" << operation->errorMessage();
+ setFinishedWithError(operation->errorName(), operation->errorMessage());
+ return;
+ }
+
+ mPriv->nested = manager()->contactsForHandles(pendingHandles->handles(), features());
+ connect(mPriv->nested,
+ SIGNAL(finished(Tp::PendingOperation*)),
+ SLOT(onNestedFinished(Tp::PendingOperation*)));
+}
+
+void PendingContacts::onReferenceHandlesFinished(PendingOperation *operation)
+{
+ PendingHandles *pendingHandles = qobject_cast<PendingHandles *>(operation);
+
+ if (pendingHandles->isError()) {
+ debug() << "ReferenceHandles error" << operation->errorName()
+ << "message" << operation->errorMessage();
+ setFinishedWithError(operation->errorName(), operation->errorMessage());
+ return;
+ }
+
+ ReferencedHandles validHandles = pendingHandles->handles();
+ UIntList invalidHandles = pendingHandles->invalidHandles();
+ ConnectionPtr conn = mPriv->manager->connection();
+ mPriv->handlesToInspect = ReferencedHandles(conn, HandleTypeContact, UIntList());
+ foreach (uint handle, mPriv->handles) {
+ if (!mPriv->satisfyingContacts.contains(handle)) {
+ int indexInValid = validHandles.indexOf(handle);
+ if (indexInValid >= 0) {
+ ReferencedHandles referencedHandle = validHandles.mid(indexInValid, 1);
+ mPriv->handlesToInspect.append(referencedHandle);
+ } else {
+ mPriv->invalidHandles.push_back(handle);
+ }
+ }
+ }
+
+ QDBusPendingCallWatcher *watcher =
+ new QDBusPendingCallWatcher(
+ conn->baseInterface()->InspectHandles(HandleTypeContact,
+ mPriv->handlesToInspect.toList()),
+ this);
+ connect(watcher,
+ SIGNAL(finished(QDBusPendingCallWatcher*)),
+ SLOT(onInspectHandlesFinished(QDBusPendingCallWatcher*)));
+}
+
+void PendingContacts::onNestedFinished(PendingOperation *operation)
+{
+ Q_ASSERT(operation == mPriv->nested);
+
+ if (operation->isError()) {
+ debug() << " error" << operation->errorName()
+ << "message" << operation->errorMessage();
+ setFinishedWithError(operation->errorName(), operation->errorMessage());
+ return;
+ }
+
+ mPriv->contacts = mPriv->nested->contacts();
+ mPriv->nested = 0;
+ mPriv->setFinished();
+}
+
+void PendingContacts::onInspectHandlesFinished(QDBusPendingCallWatcher *watcher)
+{
+ QDBusPendingReply<QStringList> reply = *watcher;
+
+ if (reply.isError()) {
+ debug().nospace() << "InspectHandles: error " << reply.error().name() << ": "
+ << reply.error().message();
+ setFinishedWithError(reply.error());
+ return;
+ }
+
+ QStringList names = reply.value();
+ int i = 0;
+ ConnectionPtr conn = mPriv->manager->connection();
+ foreach (uint handle, mPriv->handlesToInspect) {
+ QVariantMap handleAttributes;
+ handleAttributes.insert(QLatin1String(TELEPATHY_INTERFACE_CONNECTION "/contact-id"),
+ names[i++]);
+ ReferencedHandles referencedHandle(conn, HandleTypeContact,
+ UIntList() << handle);
+ mPriv->satisfyingContacts.insert(handle, manager()->ensureContact(referencedHandle,
+ mPriv->missingFeatures, handleAttributes));
+ }
+
+ allAttributesFetched();
+
+ watcher->deleteLater();
+}
+
+void PendingContacts::allAttributesFetched()
+{
+ foreach (uint handle, mPriv->handles) {
+ if (mPriv->satisfyingContacts.contains(handle)) {
+ mPriv->contacts.push_back(mPriv->satisfyingContacts[handle]);
+ }
+ }
+
+ mPriv->setFinished();
+}
+
+} // Tp