/* -*- mode: C++; tab-width: 4; c-basic-offset: 4; -*- */ /* AbiCollab- Code to enable the modification of remote documents. * Copyright (C) 2005 by Martin Sevior * Copyright (C) 2006,2007 by Marc Maurer * Copyright (C) 2008 by AbiSource Corporation B.V. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA * 02111-1307, USA. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "ut_assert.h" #include "ut_debugmsg.h" #include "xap_App.h" #include "xap_Frame.h" #include "fv_View.h" #include "xav_View.h" #include "pd_Document.h" #include "ut_assert.h" #include "ie_types.h" #include "ut_misc.h" #include "ut_units.h" #include "ap_Dialog_Id.h" #include "xap_DialogFactory.h" #include "xap_Dlg_MessageBox.h" #include "ap_Strings.h" #include "xap_Prefs.h" #include "ap_Frame.h" #include "pp_Author.h" #include "pp_AttrProp.h" #include "fv_View.h" #include "fl_DocLayout.h" #ifdef WIN32 #define WIN32_LEAN_AND_MEAN #include #else #include #include "ut_files.h" #endif // system includes #include // dialog includes #include "xap_DialogFactory.h" #include #include #include #include #include // packet includes #include // session includes #include // account handler includes #ifdef ABICOLLAB_HANDLER_TELEPATHY #include #endif #ifdef ABICOLLAB_HANDLER_XMPP #include #endif #ifdef ABICOLLAB_HANDLER_TCP #include #endif #ifdef ABICOLLAB_HANDLER_SUGAR #include #endif #ifdef ABICOLLAB_HANDLER_SERVICE #include #include #include #endif // event includes #include #include #include #include // importer/exporter includes #include "ie_exp.h" #include "ie_imp_AbiWord_1.h" #include #include #include #include #include // packet includes #include // ... and include ourselves :) #include "AbiCollabSessionManager.h" // We, the mighty SessionManager, manage it all! AbiCollabSessionManager s_AbiCollabSessionManager; AbiCollabSessionManager * AbiCollabSessionManager::m_pManager = NULL; AbiCollabSessionManager * AbiCollabSessionManager::getManager() { return m_pManager; } UT_Error AbiCollabSessionManager::serializeDocument(const PD_Document* pDoc, std::string& document, bool encodeBase64) { UT_return_val_if_fail(pDoc, false); // Don't put this auto-save in the most recent list. XAP_App::getApp()->getPrefs()->setIgnoreNextRecent(); // maskExport(); GsfOutputMemory* sink = GSF_OUTPUT_MEMORY(gsf_output_memory_new()); GsfOutput* gzSink = gsf_output_gzip_new(GSF_OUTPUT(sink), NULL); bool bAuthor = pDoc->isExportAuthorAtts(); const_cast(pDoc)->setExportAuthorAtts(true); UT_Error result = const_cast(pDoc)->saveAs(GSF_OUTPUT(gzSink), IE_Exp::fileTypeForSuffix(".abw"), true); const_cast(pDoc)->setExportAuthorAtts(bAuthor); gsf_output_close(GSF_OUTPUT(gzSink)); // unmaskExport(); if (result == UT_OK) { guint32 size = gsf_output_size (GSF_OUTPUT(sink)); const guint8* zabwBuf = gsf_output_memory_get_bytes (sink); if (encodeBase64) { // this would be more efficient if we had a GsfOutputBase64.. ah well, this will do for now guint8* base64zabwBuf = gsf_base64_encode_simple(zabwBuf, size); document += (char*)base64zabwBuf; g_free(base64zabwBuf); } else { // just copy raw zipped data into string document.resize( size ); memcpy( &document[0], zabwBuf, size ); } } else { UT_DEBUGMSG(("Failed to export! Handle this gracefully!\n")); } g_object_unref(G_OBJECT(gzSink)); g_object_unref(G_OBJECT(sink)); return result; } UT_Error AbiCollabSessionManager::deserializeDocument(PD_Document** pDoc, const std::string& document, bool isEncodedBase64) { UT_return_val_if_fail(pDoc, UT_ERROR); UT_Error res = UT_ERROR; // really bad abuse of std::string's below, but more efficient than copying // the whole buffer (which could be massive) into a temporary buffer GsfInput *source; if (isEncodedBase64) { char* base64gzBuf = const_cast(document.c_str()); size_t gzbufLen = gsf_base64_decode_simple((guint8*)base64gzBuf, strlen(base64gzBuf)); char* gzBuf = base64gzBuf; source = gsf_input_memory_new((guint8*)gzBuf, gzbufLen, false); } else { // string contains raw zipped data source = gsf_input_memory_new((guint8*)document.c_str(), document.size(), false); } if (source) { GsfInput *gzabwBuf = gsf_input_gzip_new (source, NULL); // todo: don't pass null here, but check for errors if (gzabwBuf) { bool create = (*pDoc == NULL); if (create) { UT_DEBUGMSG(("Creating a new document in AbiCollabSessionManager::deserializeDocument()\n")); (*pDoc) = new PD_Document(); (*pDoc)->createRawDocument(); } IE_Imp* imp = (IE_Imp*)new IE_Imp_AbiWord_1(*pDoc); imp->importFile(gzabwBuf); // todo: check for errors (*pDoc)->repairDoc(); if (create) (*pDoc)->finishRawCreation(); DELETEP(imp); g_object_unref(G_OBJECT(gzabwBuf)); res = UT_OK; } else UT_ASSERT(UT_SHOULD_NOT_HAPPEN); g_object_unref(G_OBJECT(source)); } else UT_ASSERT(UT_SHOULD_NOT_HAPPEN); return res; } AbiCollabSessionManager::AbiCollabSessionManager(void) : m_iDialogShare(0), m_iDialogJoin(0), m_iDialogAccounts(0), m_iDialogAddAccount(0), m_iDialogAddBuddy(0) { m_pManager = this; } AbiCollabSessionManager::~AbiCollabSessionManager(void) { disconnectSessions(); destroyAccounts(); m_pManager = NULL; } bool AbiCollabSessionManager::registerDialogs(void) { XAP_DialogFactory * pFactory = static_cast(XAP_App::getApp()->getDialogFactory()); m_iDialogShare = pFactory->registerDialog(ap_Dialog_CollaborationShare_Constructor, XAP_DLGT_NON_PERSISTENT); m_iDialogJoin = pFactory->registerDialog(ap_Dialog_CollaborationJoin_Constructor, XAP_DLGT_NON_PERSISTENT); m_iDialogAccounts = pFactory->registerDialog(ap_Dialog_CollaborationAccounts_Constructor, XAP_DLGT_NON_PERSISTENT); m_iDialogAddAccount = pFactory->registerDialog(ap_Dialog_CollaborationAddAccount_Constructor, XAP_DLGT_NON_PERSISTENT); m_iDialogAddBuddy = pFactory->registerDialog(ap_Dialog_CollaborationAddBuddy_Constructor, XAP_DLGT_NON_PERSISTENT); return true; } bool AbiCollabSessionManager::registerAccountHandlers() { UT_DEBUGMSG(("AbiCollabSessionManager::registerAccountHandlers()\n")); #ifdef ABICOLLAB_HANDLER_TELEPATHY // we don't want to register a d-bus tube account handler here so // swe can construct multiple sugar account handlers later: the // d-bus tube account handler is a singleton, that should always // be active if it is compiled in UT_DEBUGMSG(("Registering the telepathy account handler!\n")); AccountHandler* pDTubeHandler = new DTubeAccountHandler(); addAccount(pDTubeHandler); #endif #ifdef ABICOLLAB_HANDLER_XMPP m_regAccountHandlers[XMPPAccountHandler::getStaticStorageType()] = XMPPAccountHandlerConstructor; #endif #ifdef ABICOLLAB_HANDLER_TCP m_regAccountHandlers[TCPAccountHandler::getStaticStorageType()] = TCPAccountHandlerConstructor; #endif #ifdef ABICOLLAB_HANDLER_SUGAR // we don't want to register a sugar account handler here so // we can construct multiple sugar account handlers later: the // sugar account handler is a singleton, that should always // be active if it is compiled in UT_DEBUGMSG(("Registering the sugar account handler!\n")); AccountHandler* pSugarHandler = new SugarAccountHandler(); addAccount(pSugarHandler); #endif #ifdef ABICOLLAB_HANDLER_SERVICE if (tls_tunnel::Proxy::tls_tunnel_init()) m_regAccountHandlers[ServiceAccountHandler::getStaticStorageType()] = ServiceAccountHandlerConstructor; IE_Imp_AbiCollabSniffer* pAbiCollabSniffer = new IE_Imp_AbiCollabSniffer(); IE_Imp::registerImporter(pAbiCollabSniffer); m_vImpSniffers.push_back(pAbiCollabSniffer); #endif return true; } bool AbiCollabSessionManager::unregisterAccountHandlers(void) { UT_DEBUGMSG(("AbiCollabSessionManager::unregisterAccountHandlers()\n")); // no need to "free/delete" items, as they are just function pointers (ie. basic types) m_regAccountHandlers.clear(); #ifdef ABICOLLAB_HANDLER_SERVICE tls_tunnel::Proxy::tls_tunnel_deinit(); #endif return true; } void AbiCollabSessionManager::unregisterSniffers(void) { for (UT_uint32 i = 0; i < m_vImpSniffers.size(); i++) { IE_ImpSniffer* pSniffer = m_vImpSniffers[i]; UT_continue_if_fail(pSniffer); IE_Imp::unregisterImporter(pSniffer); } m_vImpSniffers.clear(); } bool AbiCollabSessionManager::unregisterDialogs(void) { XAP_DialogFactory * pFactory = static_cast(XAP_App::getApp()->getDialogFactory()); pFactory->unregisterDialog(m_iDialogShare); pFactory->unregisterDialog(m_iDialogJoin); pFactory->unregisterDialog(m_iDialogAccounts); pFactory->unregisterDialog(m_iDialogAddAccount); pFactory->unregisterDialog(m_iDialogAddBuddy); return true; } void AbiCollabSessionManager::loadProfile() { UT_DEBUGMSG(("AbiCollabSessionManager::loadProfile()\n")); gchar *s = g_build_filename(XAP_App::getApp()->getUserPrivateDirectory(), "AbiCollab.Profile", (void*)0); UT_UTF8String profile(s); FREEP(s); GsfInput* in = NULL; char *uri = UT_go_filename_to_uri(profile.utf8_str()); UT_return_if_fail(uri); in = UT_go_file_open(uri, NULL); // TODO: shouldn't use NULL here, but check for errors FREEP(uri); if (!in) return; guint8 const* contents = gsf_input_read(in, gsf_input_size(in), NULL); if (contents) { xmlDocPtr reader = xmlReadMemory(reinterpret_cast(contents), strlen(reinterpret_cast(contents)), 0, "UTF-8", 0); if (reader) { xmlNode* node = xmlDocGetRootElement(reader); if (node) { if (strcmp(reinterpret_cast(node->name), "AbiCollabProfile") == 0) { for (xmlNode* accountNode = node->children; accountNode; accountNode = accountNode->next) { // TODO: check if this node is really an AccountHandler node // find the account handler belonging to this type UT_UTF8String handlerType = reinterpret_cast(xmlGetProp(accountNode, reinterpret_cast("type"))); std::map::iterator handler_iter = m_regAccountHandlers.find(handlerType); if (handler_iter == m_regAccountHandlers.end()) continue; // could happen for example when the sugar backend is found in the profile, which does not have a registered account handler belowing to it for now AccountHandlerConstructor constructor = handler_iter->second; AccountHandler* pHandler = constructor(); UT_continue_if_fail(pHandler); for (xmlNode* accountProp = accountNode->children; accountProp; accountProp = accountProp->next) { if (accountProp->type == XML_ELEMENT_NODE) { // some node names are pre-defined... if (strcmp(reinterpret_cast(accountProp->name), "buddies") == 0) { for (xmlNode* buddyNode = accountProp->children; buddyNode; buddyNode = buddyNode->next) { if (buddyNode->type != XML_ELEMENT_NODE) continue; UT_continue_if_fail(strcmp(reinterpret_cast(buddyNode->name), "buddy") == 0); UT_continue_if_fail(buddyNode->children); // read all buddy properties PropertyMap vBuddyProps; for (xmlNode* buddyPropertyNode = buddyNode->children; buddyPropertyNode; buddyPropertyNode = buddyPropertyNode->next) { UT_continue_if_fail(buddyPropertyNode->type == XML_ELEMENT_NODE); UT_UTF8String buddyPropValue = reinterpret_cast(xmlNodeGetContent(buddyPropertyNode)); UT_continue_if_fail(buddyPropertyNode->name && *buddyPropertyNode->name && buddyPropValue.size() > 0); vBuddyProps.insert(PropertyMap::value_type( reinterpret_cast(buddyPropertyNode->name), buddyPropValue.utf8_str()) ); } // construct the buddy BuddyPtr pBuddy = pHandler->constructBuddy(vBuddyProps); if (pBuddy) { // add the buddy to the account handler pHandler->addBuddy(pBuddy); } } } else { // ... the rest are generic properties UT_UTF8String propValue = reinterpret_cast(xmlNodeGetContent(accountProp)); pHandler->addProperty(reinterpret_cast(accountProp->name), propValue.utf8_str()); } } } // add the account to the account list if it is not a duplicate if (addAccount(pHandler)) { if (pHandler->autoConnect()) pHandler->connect(); } else { _deleteAccount(pHandler); } } } } xmlFreeDoc(reader); } } g_object_unref(G_OBJECT(in)); } void AbiCollabSessionManager::storeProfile() { UT_DEBUGMSG(("AbiCollabSessionManager::storeProfile()\n")); xmlBufferPtr doc = xmlBufferCreate(); if (doc) { xmlTextWriterPtr writer = xmlNewTextWriterMemory(doc, 0); if (writer) { int rc = xmlTextWriterStartDocument(writer, NULL, "UTF-8", NULL); if (rc >= 0) { // TODO: one could check every return value here, but I'm lazy right now xmlTextWriterStartElement(writer, reinterpret_cast("AbiCollabProfile")); for (UT_uint32 i = 0; i < m_vecAccounts.size(); i++) { AccountHandler* pHandler = m_vecAccounts[i]; UT_continue_if_fail(pHandler); xmlTextWriterStartElement(writer, reinterpret_cast("AccountHandler")); // write out the account handler type xmlTextWriterWriteAttribute(writer, reinterpret_cast("type"), BAD_CAST pHandler->getStorageType().utf8_str()); // write out the account handler properties for (PropertyMap::const_iterator cit = pHandler->getProperties().begin(); cit != pHandler->getProperties().end(); cit++) { xmlTextWriterWriteElement(writer, BAD_CAST cit->first.c_str(), BAD_CAST BAD_CAST cit->second.c_str()); } // write out the account handler buddies xmlTextWriterStartElement(writer, reinterpret_cast("buddies")); for (UT_uint32 j = 0; j < pHandler->getBuddies().size(); j++) { BuddyPtr pBuddy = pHandler->getBuddies()[j]; UT_continue_if_fail(pBuddy); if (!pBuddy->isVolatile()) { // we need to be able to store buddy properties UT_ASSERT_HARMLESS(UT_NOT_IMPLEMENTED); /*xmlTextWriterStartElement(writer, reinterpret_cast("buddy")); // write out the buddy properties // TODO: for now, the only useful property a buddy has is its "name"; // However in the future we really should write out a generic property list xmlTextWriterWriteElement( writer, reinterpret_cast("name"), reinterpret_cast(pBuddy->getName().utf8_str()) ); xmlTextWriterEndElement(writer);*/ /* end buddy */ } } xmlTextWriterEndElement(writer); /* end buddies */ xmlTextWriterEndElement(writer); /* end AccountHandler */ } xmlTextWriterEndElement(writer); /* end AbiCollabProfile */ } xmlTextWriterEndDocument(writer); xmlFreeTextWriter(writer); gchar * s = g_build_filename(XAP_App::getApp()->getUserPrivateDirectory(), "AbiCollab.Profile",NULL); UT_UTF8String profile(s); FREEP(s); char *uri = UT_go_filename_to_uri(profile.utf8_str()); GError* error = 0; GsfOutput* out = UT_go_file_create (uri, &error); if (out) { gsf_output_write(out, strlen(reinterpret_cast(const_cast(doc->content))), reinterpret_cast(const_cast(doc->content)) ); gsf_output_close(out); g_object_unref(G_OBJECT(out)); } else { UT_DEBUGMSG(("Error creating AbiCollab Profile %s: %s!\n", profile.utf8_str(), error ? error->message : "unknown error")); } FREEP(uri); } else { UT_DEBUGMSG(("Error creating XML output writer\n")); } xmlBufferFree(doc); } else { UT_DEBUGMSG(("Error creating XML output buffer\n")); } } bool AbiCollabSessionManager::destroySession(PD_Document * pDoc) { UT_DEBUGMSG(("AbiCollabSessionManager::destroySession(PD_Document * pDoc)\n")); UT_sint32 i = 0; for (i = 0; i < m_vecSessions.getItemCount(); i++) { AbiCollab* pSession = m_vecSessions.getNthItem(i); UT_continue_if_fail(pSession); PD_Document* pSessionDoc = pSession->getDocument(); if (pSessionDoc == pDoc) { _deleteSession(pSession); m_vecSessions.deleteNthItem(i); return true; } } return false; } bool AbiCollabSessionManager::destroySession(AbiCollab* pSession) { UT_DEBUGMSG(("AbiCollabSessionManager::destroySession(AbiCollab* pSession)\n")); UT_sint32 i = 0; for (i = 0; i < m_vecSessions.getItemCount(); i++) { AbiCollab* pActiveSession = m_vecSessions.getNthItem(i); UT_continue_if_fail(pActiveSession); if (pActiveSession == pSession) { _deleteSession(pSession); m_vecSessions.deleteNthItem(i); return true; } } return false; } void AbiCollabSessionManager::disconnectSession(AbiCollab* pSession) { UT_return_if_fail(pSession); UT_DEBUGMSG(("AbiCollabSessionManager::disconnectSession() - pSession: %s\n", pSession->getSessionId().utf8_str())); if (isLocallyControlled(pSession->getDocument())) { /* Before we close this session, try to see of we can hand over session ownership to someone else. NOTE: we don't do any fancy determination yet who to hand over the session to; we just hand it over to the first buddy in the list. */ if (_canInitiateSessionTakeover(pSession)) { UT_DEBUGMSG(("Can initiate session takeover\n")); if (pSession->getCollaborators().size() > 0) { std::map::const_iterator cit = pSession->getCollaborators().begin(); pSession->initiateSessionTakeover((*cit).first); } } closeSession(pSession, false); } else { disjoinSession(pSession->getSessionId()); } // TODO: mark the session as disconnected/closed, for additional safety } void AbiCollabSessionManager::disconnectSessions() { for(UT_sint32 i = 0; i < m_vecSessions.getItemCount(); i++) { AbiCollab* pSession = m_vecSessions.getNthItem(i); UT_continue_if_fail(pSession); disconnectSession(pSession); } } AbiCollab* AbiCollabSessionManager::getSession(PD_Document* pDoc) { UT_return_val_if_fail(pDoc, NULL); AbiCollab * pCollab = NULL; UT_sint32 i = 0; for(i = 0; i < m_vecSessions.getItemCount(); i++) { pCollab = m_vecSessions.getNthItem(i); if (pCollab) { if (pDoc == pCollab->getDocument()) { return pCollab; } } } return NULL; } AbiCollab* AbiCollabSessionManager::getSessionFromDocumentId(const UT_UTF8String& sDocumentId) { AbiCollab * pCollab = NULL; UT_sint32 i = 0; for(i = 0; i < m_vecSessions.getItemCount(); i++) { pCollab = m_vecSessions.getNthItem(i); if (pCollab) { PD_Document* pDoc = pCollab->getDocument(); if (strcmp(pDoc->getDocUUIDString(), sDocumentId.utf8_str()) == 0) { return pCollab; } } } return NULL; } AbiCollab* AbiCollabSessionManager::getSessionFromSessionId(const UT_UTF8String& sSessionId) { AbiCollab * pCollab = NULL; UT_sint32 i = 0; for(i = 0; i < m_vecSessions.getItemCount(); i++) { pCollab = m_vecSessions.getNthItem(i); if (pCollab) { if (pCollab->getSessionId() == sSessionId) { return pCollab; } } } return NULL; } AbiCollab* AbiCollabSessionManager::startSession(PD_Document* pDoc, UT_UTF8String& sSessionId, AccountHandler* pAclAccount, bool bLocallyOwned, XAP_Frame* pFrame, const UT_UTF8String& masterDescriptor) { UT_DEBUGMSG(("Starting collaboration session for document with id %s, master descriptor: %s\n", pDoc->getDocUUIDString(), masterDescriptor.utf8_str())); UT_return_val_if_fail(pDoc, NULL); UT_return_val_if_fail(pAclAccount, NULL); if (sSessionId == "") { XAP_App* pApp = XAP_App::getApp(); UT_UUID* pUUID = pApp->getUUIDGenerator()->createUUID(); pUUID->toString(sSessionId); } if (masterDescriptor != "") { // search for a buddy descriptor in the authors list that matches this // descriptor, and make that the active author; if such an author does // not exist, then search for an author with no abicollab property set // at all, and give it this buddy descriptor (ie, we assume that it is // "us", even though it might not always be a valid assumption). int iAuthorId = -1; UT_GenericVector authors = pDoc->getAuthors(); pp_Author* pEmptyAuthor = NULL; UT_DEBUGMSG(("Scanning %d authors to see if we recognize this master buddy\n", authors.getItemCount())); for (UT_sint32 i = 0; i < authors.getItemCount(); i++) { pp_Author* pAuthor = authors.getNthItem(i); UT_continue_if_fail(pAuthor); const gchar* szDescriptor = NULL; pAuthor->getProperty("abicollab-descriptor", szDescriptor); if (!szDescriptor) { if (!pEmptyAuthor && !pAuthor->getAttrProp()->hasProperties()) pEmptyAuthor = pAuthor; continue; } if (masterDescriptor != szDescriptor) continue; // yay, we already editted this document ourselves! iAuthorId = pAuthor->getAuthorInt(); pDoc->setMyAuthorInt(iAuthorId); UT_DEBUGMSG(("Found our own author object with descriptior %s, id %d!\n", masterDescriptor.utf8_str(), iAuthorId)); break; } if (iAuthorId == -1) { if (pEmptyAuthor) { // reuse this author object and make it our own iAuthorId = pEmptyAuthor->getAuthorInt(); PP_AttrProp * pPA = pEmptyAuthor->getAttrProp(); pPA->setProperty("abicollab-descriptor", masterDescriptor.utf8_str()); pDoc->setMyAuthorInt(iAuthorId); pDoc->sendChangeAuthorCR(pEmptyAuthor); UT_DEBUGMSG(("Reusing empty author object with id %d, setting descriptor to %s!\n", iAuthorId, masterDescriptor.utf8_str())); } else { UT_DEBUGMSG(("No suitable author found in the document for descriptor %s\n", masterDescriptor.utf8_str())); iAuthorId = pDoc->findFirstFreeAuthorInt(); pp_Author * pA = pDoc->addAuthor(iAuthorId); pDoc->setMyAuthorInt(iAuthorId); PP_AttrProp * pPA = pA->getAttrProp(); pPA->setProperty("abicollab-descriptor", masterDescriptor.utf8_str()); pDoc->sendAddAuthorCR(pA); UT_DEBUGMSG(("Added a new author to the documument with descriptor %s, id %d\n", masterDescriptor.utf8_str(), iAuthorId)); } } } UT_DEBUGMSG(("Creating a new collaboration session with UUID: %s\n", sSessionId.utf8_str())); UT_return_val_if_fail(_setupFrame(&pFrame, pDoc), NULL); AbiCollab* pAbiCollab = new AbiCollab(pDoc, sSessionId, pAclAccount, bLocallyOwned); m_vecSessions.push_back(pAbiCollab); // notify all people we are sharing a new document // FIXME: since we only allow a session to be shared on 1 account, we should // only notify the buddies on that account, instead of notifying the buddies // on all accounts. StartSessionEvent event; event.setBroadcast(true); signal(event); return pAbiCollab; } void AbiCollabSessionManager::closeSession(AbiCollab* pSession, bool canConfirm) { UT_DEBUGMSG(("Stopping collaboration session %s\n", pSession->getSessionId().utf8_str())); UT_return_if_fail(pSession); // TODO: in the future, we should hand over control to someone // else within this session first, if possible // ask for confirmation if we are in control of this session, and people are connected to us, if (pSession->isLocallyControlled() && pSession->getCollaborators().size() > 0 && canConfirm) { XAP_Frame *pFrame = XAP_App::getApp()->getLastFocussedFrame(); UT_return_if_fail(pFrame); UT_UTF8String msg; // TODO: make this localizable UT_UTF8String_sprintf(msg, "This document is currently being shared with %u people. Are you sure you want to stop sharing this document?", pSession->getCollaborators().size()); if (pFrame->showMessageBox(msg.utf8_str(), XAP_Dialog_MessageBox::b_YN, XAP_Dialog_MessageBox::a_NO) != XAP_Dialog_MessageBox::a_YES) return; } // check who is controlling this session if (pSession->isLocallyControlled()) { UT_UTF8String pDestroyedSession = pSession->getSessionId(); // kill the session destroySession(pSession); // notify all that this session is closed CloseSessionEvent event(pDestroyedSession); event.setBroadcast(true); signal(event); } else { // we are joined to this session, so we aren't allowed to close it; // we should disjoin it instead UT_ASSERT(UT_NOT_REACHED); } } void AbiCollabSessionManager::joinSessionInitiate(BuddyPtr pBuddy, DocHandle* pDocHandle) { UT_return_if_fail(pBuddy); UT_return_if_fail(pDocHandle); UT_DEBUGMSG(("Initiating join on buddy |%s|, document |%s|\n", pBuddy->getDescription().utf8_str(), pDocHandle->getSessionId().utf8_str())); AccountHandler* pHandler = pBuddy->getHandler(); UT_return_if_fail(pHandler); pHandler->joinSessionAsync(pBuddy, *pDocHandle); // TODO: do some accounting here, so we know we send an auth request in ::joinSession() } void AbiCollabSessionManager::joinSession(const UT_UTF8String& sSessionId, PD_Document* pDoc, const UT_UTF8String& docUUID, UT_sint32 iRev, UT_sint32 iAuthorId, BuddyPtr pCollaborator, AccountHandler* pAclAccount, bool bLocallyOwned, XAP_Frame *pFrame) { UT_DEBUGMSG(("AbiCollabSessionManager::joinSession()\n")); UT_return_if_fail(pCollaborator); UT_return_if_fail(pDoc); UT_return_if_fail(pAclAccount); if (pAclAccount->getStorageType() == "com.abisource.abiword.abicollab.backend.sugar") { // HACK, remove this some day: the sugar backend should just pass us a frame to use pFrame = XAP_App::getApp()->getLastFocussedFrame(); pFrame->loadDocument(pDoc); } else { UT_return_if_fail(_setupFrame(&pFrame, pDoc)); } AbiCollab* pSession = new AbiCollab(sSessionId, pDoc, docUUID, iRev, pCollaborator, pAclAccount, bLocallyOwned); m_vecSessions.push_back(pSession); // signal everyone that we have joined this session JoinSessionEvent event(sSessionId); event.addRecipient(pCollaborator); signal(event); // the author object representing us should already be present in the document // that was sent to us; fetch it using the author id the master gave us pp_Author * pA = pDoc->getAuthorByInt(iAuthorId); UT_return_if_fail(pA); pDoc->setMyAuthorInt(iAuthorId); } void AbiCollabSessionManager::joinSession(AbiCollab* pSession, BuddyPtr pCollaborator) { UT_DEBUGMSG(("AbiCollabSessionManager::joinSession()\n")); UT_return_if_fail(pSession); UT_return_if_fail(pCollaborator); m_vecSessions.push_back(pSession); // signal everyone that we have joined this session JoinSessionEvent event(pSession->getSessionId()); event.addRecipient(pCollaborator); signal(event); } void AbiCollabSessionManager::disjoinSession(const UT_UTF8String& sSessionId) { UT_DEBUGMSG(("AbiCollabSessionManager::disjoinSession(%s)\n", sSessionId.utf8_str())); AbiCollab* pSession = getSessionFromSessionId(sSessionId); UT_return_if_fail(pSession); const std::map vCollaborators = pSession->getCollaborators(); if (!isLocallyControlled(pSession->getDocument())) { // we are joined to a session, so there should only be one collaborator: // the person sharing the document UT_return_if_fail(vCollaborators.size() == 1); std::map::const_iterator cit = vCollaborators.begin(); BuddyPtr pCollaborator = (*cit).first; destroySession(pSession); DisjoinSessionEvent event(sSessionId); event.addRecipient(pCollaborator); signal(event); } else { UT_ASSERT(UT_NOT_REACHED); return; } } bool AbiCollabSessionManager::isLocallyControlled(PD_Document* pDoc) { UT_return_val_if_fail(pDoc, false); // find the session belonging to this document, if any for (UT_sint32 i = 0; i < m_vecSessions.getItemCount(); i++) { AbiCollab* pSession = m_vecSessions.getNthItem(i); if (pSession) { PD_Document* pSessionDoc = pSession->getDocument(); if (pSessionDoc && pSessionDoc == pDoc) { return pSession->isLocallyControlled(); } } } return false; } bool AbiCollabSessionManager::isInSession(PD_Document* pDoc) { UT_return_val_if_fail(pDoc, false); // find the session belonging to this document, if any for (UT_sint32 i = 0; i < m_vecSessions.getItemCount(); i++) { AbiCollab* pSession = m_vecSessions.getNthItem(i); if (pSession) { PD_Document* pSessionDoc = pSession->getDocument(); if (pSessionDoc && pSessionDoc == pDoc) { return true; } } } return false; } bool AbiCollabSessionManager::isActive(const UT_UTF8String& sSessionId) { // find a session with this session id for (UT_sint32 i = 0; i < m_vecSessions.getItemCount(); i++) { AbiCollab* pSession = m_vecSessions.getNthItem(i); if (pSession) { if (pSession->getSessionId() == sSessionId) { return true; } } } return false; } void AbiCollabSessionManager::removeBuddy(BuddyPtr pBuddy, bool graceful) { UT_return_if_fail(pBuddy); UT_DEBUGMSG(("Dropping buddy '%s' from all sessions\n", pBuddy->getDescription().utf8_str())); // TODO: should we send out events for every buddy we drop, or session // we delete? for (UT_sint32 i = m_vecSessions.getItemCount() - 1; i >= 0; i--) { AbiCollab* pSession = m_vecSessions.getNthItem(i); UT_continue_if_fail(pSession); if (pSession->isLocallyControlled()) { pSession->removeCollaborator(pBuddy); } else { // we don't control this session, meaning we can drop it completely // if this buddy controlled it // TODO: when we allow more than 1 buddy in a non-locally controlled, // then remove it from that list here if (pSession->isController(pBuddy)) { UT_DEBUGMSG(("This buddy controlled a session, destroying the session...\n")); UT_UTF8String docName = pSession->getDocument()->getFilename(); if (docName == "") docName = "Untitled"; // TODO: fetch the title from the frame somehow (which frame?) - MARCM destroySession(pSession); if (!graceful) { XAP_Frame *pFrame = XAP_App::getApp()->getLastFocussedFrame(); UT_continue_if_fail(pFrame); // TODO: make this localizable UT_UTF8String msg; UT_UTF8String_sprintf(msg, "You've been disconnected from buddy %s. The collaboration session for document %s has been stopped.", pBuddy->getDescription().utf8_str(), docName.utf8_str()); pFrame->showMessageBox(msg.utf8_str(), XAP_Dialog_MessageBox::b_O, XAP_Dialog_MessageBox::a_OK); } } } } } // should we move this to AbiCollab.cpp ? void AbiCollabSessionManager::updateAcl(AbiCollab* pSession, AccountHandler* pAccount, const std::vector vAcl) { UT_return_if_fail(pSession); UT_return_if_fail(pAccount); // check if all current collaborators are still allowed to collaborate; if not, // then remove them from the session const std::map collaborators = pSession->getCollaborators(); for (std::map::const_iterator cit = collaborators.begin(); cit != collaborators.end(); cit++) { BuddyPtr pBuddy = (*cit).first; UT_continue_if_fail(pBuddy); AccountHandler* pBuddyAccount = pBuddy->getHandler(); UT_continue_if_fail(pBuddyAccount); UT_continue_if_fail(pBuddyAccount == pAccount); if (!pBuddyAccount->hasAccess(vAcl, pBuddy)) { // this current collaborator has been banned from the session, so // disconnect him UT_ASSERT_HARMLESS(UT_NOT_IMPLEMENTED); } } // set the new ACL on the account handler pAccount->setAcl(pSession, vAcl); // set the new access control list on the session pSession->setAcl(vAcl); } XAP_Frame* AbiCollabSessionManager::findFrameForSession(AbiCollab* pSession) { UT_return_val_if_fail(pSession, NULL); UT_return_val_if_fail(pSession->getDocument(), NULL); for (UT_sint32 i = 0; i < XAP_App::getApp()->getFrameCount(); i++) { XAP_Frame* pFrame = XAP_App::getApp()->getFrame(i); UT_continue_if_fail(pFrame); if (pSession->getDocument() == pFrame->getCurrentDoc()) return pFrame; } return NULL; } bool AbiCollabSessionManager::addAccount(AccountHandler* pHandler) { UT_return_val_if_fail(pHandler, false); bool bUnique = true; for (UT_uint32 i = 0; i < m_vecAccounts.size() && bUnique; i++) { UT_continue_if_fail(m_vecAccounts[i]); if (pHandler->getStorageType() == m_vecAccounts[i]->getStorageType()) { // Accounts are of same backend - compare if equal bUnique = !((*pHandler) == (*m_vecAccounts[i])); } } if (bUnique) { m_vecAccounts.push_back(pHandler); } else { _deleteAccount(pHandler); UT_DEBUGMSG(("User attempted to add duplicate account - request ignored.\n")); } return bUnique; } void AbiCollabSessionManager::destroyAccounts(void) { for (UT_uint32 i = 0; i < m_vecAccounts.size(); i++) _deleteAccount(m_vecAccounts[i]); m_vecAccounts.clear(); } bool AbiCollabSessionManager::destroyAccount(AccountHandler* pHandler) { UT_return_val_if_fail(pHandler, false); for (UT_uint32 i = 0; i < m_vecAccounts.size(); i++) { UT_continue_if_fail(m_vecAccounts[i]); if (pHandler == m_vecAccounts[i]) { UT_DEBUGMSG(("Destroying account handler %s\n", pHandler->getDescription().utf8_str())); // TODO: if a number of buddies are connected, then we should ask for confirmation // drop all buddies that belong to the account that is being deleted // from all active sessions for (UT_sint32 j = 0; j < m_vecSessions.getItemCount(); j++) { AbiCollab* pSession = m_vecSessions.getNthItem(j); UT_continue_if_fail(pSession); // There can only be buddies from 1 account in an active session these days/ // That means that if this session's account is the account we are destroying, // then we can kill off the entire session. Do nothing otherwise. if (pSession->getAclAccount() == pHandler) { UT_DEBUGMSG(("Session %s is running on this account, destroying it!\n", pSession->getSessionId().utf8_str())); destroySession(pSession); } } m_vecAccounts.erase(m_vecAccounts.begin() + i, m_vecAccounts.begin() + i + 1); _deleteAccount(pHandler); return true; } } return false; } void AbiCollabSessionManager::setDocumentHandles(BuddyPtr pBuddy, const UT_GenericVector& vDocHandles) { UT_DEBUGMSG(("Setting document handles for buddy %s\n", pBuddy->getDescriptor().utf8_str())); UT_return_if_fail(pBuddy); // create a copy of the current document handles, which // we'll use to determine which document handles do not exist anymore std::vector oldDocHandles(pBuddy->getDocHandles()); for (UT_sint32 i = 0; i < vDocHandles.size(); i++) { DocHandle* pDocHandle = vDocHandles.getNthItem(i); UT_continue_if_fail(pDocHandle); // sanity checking UT_UTF8String sId = pDocHandle->getSessionId(); UT_continue_if_fail(sId.size() > 0); // construct a nice document name UT_UTF8String sDocumentName = pDocHandle->getName(); if (sDocumentName.size() == 0) { // this document has no name yet; give it an untitled name const XAP_StringSet * pSS = XAP_App::getApp()->getStringSet(); UT_UTF8String sUntitled; pSS->getValueUTF8(XAP_STRING_ID_UntitledDocument, sUntitled); UT_UTF8String_sprintf(sDocumentName, sUntitled.utf8_str(), 0); // TODO: as should append a number here, but at the moment // XAP_Frame::m_iUntitled is not accessible from here } // check to see if we already have a document handle with this ID DocHandle* pCurDocHandle = pBuddy->getDocHandle(sId); if (!pCurDocHandle) { // Ok, all set. Get the buddy from the AccountHandler, and assign // the document handle to the buddy DocHandle * pNewDocHandle = new DocHandle(sId, sDocumentName); pBuddy->addDocHandle(pNewDocHandle); UT_DEBUGMSG(("Added DocHandle (%s) to buddy (%s)\n", sId.utf8_str(), pBuddy->getDescription().utf8_str())); // signal that a buddy has a new session AccountBuddyAddDocumentEvent event(pNewDocHandle); signal(event, pBuddy); } else { UT_DEBUGMSG(("Found an existing DocHandle (%s) for buddy (%s)\n", sId.utf8_str(), pBuddy->getDescription().utf8_str())); // we already have a handle for this document, remove it from the old document handles copy for (std::vector::iterator it = oldDocHandles.begin(); it != oldDocHandles.end(); it++) { DocHandle* pOldDocHandle = *it; if (pCurDocHandle == pOldDocHandle) { oldDocHandles.erase(it); break; } } } } // every document that is still in the old document handles list does not // exist anymore, so let's delete it std::vector::iterator it = oldDocHandles.begin(); while (it != oldDocHandles.end()) { DocHandle* pDocHandle = *it; UT_continue_if_fail(pDocHandle); // TODO: when we are a part of this session, then handle that properly UT_DEBUGMSG(("Purging existing DocHandle (%s) for buddy (%s)\n", pDocHandle->getSessionId().utf8_str(), pBuddy->getDescription().utf8_str())); UT_UTF8String pDestroyedSessionId = pDocHandle->getSessionId(); pBuddy->destroyDocHandle(pDestroyedSessionId); CloseSessionEvent event(pDestroyedSessionId); signal(event, pBuddy); it = oldDocHandles.erase(it); } } BuddyPtr AbiCollabSessionManager::constructBuddy(const std::string& identifier, BuddyPtr pBuddy) { UT_DEBUGMSG(("AbiCollabSessionManager::constructBuddy() - identifier: %s\n", identifier.c_str())); // find an account hander to handle this identifier for (UT_uint32 i = 0; i < m_vecAccounts.size(); i++) { AccountHandler* pHandler = m_vecAccounts[i]; UT_continue_if_fail(pHandler); if (pHandler->recognizeBuddyIdentifier(identifier)) return pHandler->constructBuddy(identifier, pBuddy); } UT_ASSERT_HARMLESS(UT_SHOULD_NOT_HAPPEN); return BuddyPtr(); } bool AbiCollabSessionManager::processPacket(AccountHandler& /*handler*/, Packet* packet, BuddyPtr buddy) { UT_DEBUGMSG(("AbiCollabSessionManager::processPacket()\n")); UT_return_val_if_fail(packet, false); UT_return_val_if_fail(buddy, false); // check if this is a simple import-meh-now-packet PClassType pct = packet->getClassType(); if (pct >= _PCT_FirstSessionPacket && pct <= _PCT_LastSessionPacket) { // lookup session SessionPacket* dsp = static_cast( packet ); const UT_UTF8String& sessionId = dsp->getSessionId(); AbiCollab* pAbiCollab = getSessionFromSessionId(sessionId); if (!pAbiCollab) { UT_DEBUGMSG(("Unknown session id: '%s'", sessionId.utf8_str())); UT_return_val_if_fail(pAbiCollab, true); } // handle packet! pAbiCollab->import(dsp, buddy); return true; } // handle packet switch (pct) { case PCT_StartSessionEvent: { StartSessionEvent event; event.setBroadcast(true); signal(event, buddy); return true; } case PCT_JoinSessionEvent: { JoinSessionEvent* jse = static_cast(packet); const UT_UTF8String& joinedSessionId = jse->getSessionId(); // someone who joined this session disconnected, remove him from the collaboration session AbiCollab* pSession = getSessionFromSessionId(joinedSessionId); if (pSession) { if (isLocallyControlled( pSession->getDocument() )) { // we should already know this buddy, as we sent should have already added this // buddy when responding to his JoinSessionRequest // TODO: check this } // signal all JoinSessionEvent event(joinedSessionId); signal( event, buddy ); } else { // we don't know this session, don't forward the packet UT_ASSERT_HARMLESS(UT_NOT_REACHED); } return true; } case PCT_DisjoinSessionEvent: { DisjoinSessionEvent* dse = static_cast(packet); const UT_UTF8String& disjoinedSessionId = dse->getSessionId(); // someone who joined this session disconnected, remove him from the collaboration session AbiCollab* pSession = getSessionFromSessionId(disjoinedSessionId); if (pSession) { pSession->removeCollaborator(buddy); // signal all DisjoinSessionEvent event(disjoinedSessionId); signal(event, buddy); } else { // we don't know this session, don't forward the packet UT_ASSERT_HARMLESS(UT_NOT_REACHED); } return true; } case PCT_CloseSessionEvent: { CloseSessionEvent* cse = static_cast(packet); const UT_UTF8String& destroyedSessionId = cse->getSessionId(); buddy->destroyDocHandle( destroyedSessionId ); // handle the event outselves AbiCollab* pSession = getSessionFromSessionId(destroyedSessionId); if (pSession) { if (!isLocallyControlled(pSession->getDocument())) { UT_UTF8String docName = pSession->getDocument()->getFilename(); if (docName == "") docName = "Untitled"; // TODO: fetch the title from the frame somehow (which frame?) - MARCM // the server hosting this session is gone, so let's disconnect as well if (!destroySession(pSession)) { UT_ASSERT_HARMLESS(UT_SHOULD_NOT_HAPPEN); } // signal all CloseSessionEvent event( destroyedSessionId ); signal( event, buddy ); // inform the user of the disconnect XAP_Frame *pFrame = XAP_App::getApp()->getLastFocussedFrame(); UT_return_val_if_fail(pFrame, true); UT_UTF8String msg; // TODO: make this localizable UT_UTF8String_sprintf(msg, "Document %s is not being shared anymore by buddy %s. You are disconnected from the collaboration session.", docName.utf8_str(), buddy->getDescription().utf8_str()); pFrame->showMessageBox(msg.utf8_str(), XAP_Dialog_MessageBox::b_O, XAP_Dialog_MessageBox::a_OK); } else { // someone who is not controlling this session sends out messages he closed it! // we will not forward this packet UT_ASSERT_HARMLESS(UT_NOT_REACHED); } } else { UT_DEBUGMSG(("Ignoring a CloseSession event for unknown session (%s)\n", destroyedSessionId.utf8_str())); } return true; } case PCT_AccountAddBuddyRequestEvent: { // look at this packet; I have a feeling we need to deprecate it - MARCM UT_ASSERT_HARMLESS(UT_NOT_IMPLEMENTED); return true; } default: break; } return false; } void AbiCollabSessionManager::registerEventListener(EventListener* pListener) { UT_return_if_fail(pListener); m_vecEventListeners.push_back(pListener); } void AbiCollabSessionManager::unregisterEventListener(EventListener* pListener) { UT_return_if_fail(pListener); for (UT_sint32 i = 0; i < m_vecEventListeners.getItemCount(); i++) { EventListener* pRegListener = m_vecEventListeners.getNthItem(i); if (pRegListener == pListener) { m_vecEventListeners.deleteNthItem(i); break; } } } void AbiCollabSessionManager::signal(const Event& event, BuddyPtr pSource) { UT_DEBUGMSG(("AbiCollabSessionManager::signal()\n")); // forward the event to all listeners for (UT_sint32 i = 0; i < m_vecEventListeners.getItemCount(); i++) { EventListener* pListener = m_vecEventListeners.getNthItem(i); if (pListener) pListener->signal(event, pSource); } } void AbiCollabSessionManager::beginAsyncOperation(AbiCollab* pSession) { UT_DEBUGMSG(("AbiCollabSessionManager::beginAsyncOperation(AbiCollab*)\n")); UT_return_if_fail(pSession); m_asyncSessionOps[pSession]++; } void AbiCollabSessionManager::endAsyncOperation(AbiCollab* pSession) { UT_DEBUGMSG(("AbiCollabSessionManager::endAsyncOperation(AbiCollab*)\n")); UT_return_if_fail(pSession); UT_return_if_fail(m_asyncSessionOps[pSession] > 0); m_asyncSessionOps[pSession]--; } void AbiCollabSessionManager::beginAsyncOperation(AccountHandler* pAccount) { UT_DEBUGMSG(("AbiCollabSessionManager::beginAsyncOperation(AccountHandler*)\n")); UT_return_if_fail(pAccount); m_asyncAccountOps[pAccount]++; } void AbiCollabSessionManager::endAsyncOperation(AccountHandler* pAccount) { UT_DEBUGMSG(("AbiCollabSessionManager::endAsyncOperation(AccountHandler*)\n")); UT_return_if_fail(pAccount); UT_return_if_fail(m_asyncAccountOps[pAccount] > 0); m_asyncAccountOps[pAccount]--; } bool AbiCollabSessionManager::_setupFrame(XAP_Frame** pFrame, PD_Document* pDoc) { UT_return_val_if_fail(pFrame, false); if (*pFrame) { UT_DEBUGMSG(("Frame is non-NULL, NOT loading document in the frame\n")); return true; } // if the document doesn't belong to a frame already, then create a // new frame for this session (except when the document in the current // frame is not dirty, doesn't have a filename yet (which means it // is a brand new empty document), and isn't being shared at the moment) XAP_Frame* pCurFrame = XAP_App::getApp()->getLastFocussedFrame(); UT_return_val_if_fail(pCurFrame, false); bool isNewFrame = false; PD_Document * pFrameDoc = static_cast(pCurFrame->getCurrentDoc()); if (pFrameDoc != pDoc) { if (!pFrameDoc || (!pFrameDoc->getFilename() && !pFrameDoc->isDirty() && !isInSession(pFrameDoc))) { // we can replace the document in this frame safely, as it is // brand new, and doesn't have any contents yet } else { // the current frame has already a document loaded, let's create // a new frame pCurFrame = XAP_App::getApp()->newFrame(); isNewFrame = true; } } else { UT_DEBUGMSG(("This document is already in the current frame; using this frame\n")); } UT_return_val_if_fail(pCurFrame, false); *pFrame = pCurFrame; // load the document in the frame; this will also delete the old document (or at least, it should) if (static_cast((*pFrame)->getCurrentDoc()) != pDoc) { UT_DEBUGMSG(("Loading the document in the frame\n")); (*pFrame)->loadDocument(pDoc); } else { UT_DEBUGMSG(("Not loading the document in the frame, as the frame already has it\n")); } if (isNewFrame) (*pFrame)->show(); return true; } void AbiCollabSessionManager::_deleteSession(AbiCollab* pSession) { UT_DEBUGMSG(("Waiting for all async actions to complete...\n")); UT_return_if_fail(pSession); // wait for all async actions to complete // TODO: some sort of feedback to the user would be nice bool caughtQuit = false; while (m_asyncSessionOps[pSession] > 0) caughtQuit |= _nullUpdate(); #if defined(TOOLKIT_WIN) if (caughtQuit) PostQuitMessage(0); #endif DELETEP(pSession); } void AbiCollabSessionManager::_deleteAccount(AccountHandler* pHandler) { UT_DEBUGMSG(("Waiting for all async actions to complete...\n")); UT_return_if_fail(pHandler); // wait for all async actions to complete // TODO: some sort of feedback to the user would be nice bool caughtQuit = false; while (m_asyncAccountOps[pHandler] > 0) caughtQuit |= _nullUpdate(); #if defined(TOOLKIT_WIN) if (caughtQuit) PostQuitMessage(0); #endif DELETEP(pHandler); } // Note: the return value only makes sense on Win32, other platforms // should ignore it. On Win32, it denotes wether a WM_QUIT message // has been eaten (and thus should be reposted to the main message // queue later. bool AbiCollabSessionManager::_nullUpdate() { bool caughtQuit = false; #if defined(TOOLKIT_WIN) MSG msg; for (UT_sint32 i = 0 ; i < 10 ; i++ ) { if (PeekMessage(&msg, (HWND) NULL, 0, 0, PM_REMOVE)) { if (msg.message == WM_QUIT) caughtQuit = true; DispatchMessage(&msg); } } Sleep(10); #elif defined(TOOLKIT_GTK) for (UT_sint32 i = 0; (i < 10) && gtk_events_pending(); i++) gtk_main_iteration (); usleep(1000*10); #elif defined(TOOLKIT_COCOA) #warning _nullUpdate needs implementation for Cocoa #else #error unknown platform #endif return caughtQuit; } bool AbiCollabSessionManager::_canInitiateSessionTakeover(AbiCollab* pSession) { UT_return_val_if_fail(pSession, false); UT_return_val_if_fail(pSession->isLocallyControlled(), false); const std::map collaborators = pSession->getCollaborators(); if (collaborators.size() == 0) return false; /* NOTE: for now, we only allow session takeover, if: 1. All buddies use the same account handler type 2. The account handler actually supports session takeover 3. All buddies are actually owned by the same account handler instance Condition 1 and 2 are ok, but 3 needs fixing soon. */ AccountHandler* pHandler = (*collaborators.begin()).first->getHandler(); if (!pHandler->allowsSessionTakeover()) return false; for (std::map::const_iterator cit = ++collaborators.begin(); cit != collaborators.end(); cit++) if (((*cit).first)->getHandler() != pHandler) return false; return true; }