summaryrefslogtreecommitdiffstats
blob: 36f02cc093907691bc18a94e96e3fbe146aae3c3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
/*
 * Copyright (c) 2012 Mateusz Piękos
 * Licensed under the simplified BSD license.
 * See Documentation/Licenses/BSD-simplified.txt for more information.
 */

/*
 * Copyright (c) 2016 Isode Limited.
 * All rights reserved.
 * See the COPYING file for more information.
 */


#include <Swiften/Whiteboard/WhiteboardSessionManager.h>

#include <boost/bind.hpp>
#include <boost/smart_ptr/make_shared.hpp>

#include <Swiften/Base/foreach.h>
#include <Swiften/Disco/EntityCapsProvider.h>
#include <Swiften/Presence/PresenceOracle.h>
#include <Swiften/Queries/IQRouter.h>
#include <Swiften/Whiteboard/WhiteboardResponder.h>

namespace Swift {
    WhiteboardSessionManager::WhiteboardSessionManager(IQRouter* router, StanzaChannel* stanzaChannel, PresenceOracle* presenceOracle, EntityCapsProvider* capsProvider) : router_(router), stanzaChannel_(stanzaChannel), presenceOracle_(presenceOracle), capsProvider_(capsProvider) {
        responder = new WhiteboardResponder(this, router);
        responder->start();
        stanzaChannel_->onPresenceReceived.connect(boost::bind(&WhiteboardSessionManager::handlePresenceReceived, this, _1));
        stanzaChannel_->onAvailableChanged.connect(boost::bind(&WhiteboardSessionManager::handleAvailableChanged, this, _1));
    }

    WhiteboardSessionManager::~WhiteboardSessionManager() {
        responder->stop();
        delete responder;
    }

    WhiteboardSession::ref WhiteboardSessionManager::getSession(const JID& to) {
        if (sessions_.find(to) == sessions_.end()) {
            return boost::shared_ptr<WhiteboardSession>();
        }
        return sessions_[to];
    }

    OutgoingWhiteboardSession::ref WhiteboardSessionManager::createOutgoingSession(const JID& to) {
        JID fullJID = to;
        if (fullJID.isBare()) {
            fullJID = getFullJID(fullJID);
        }
        OutgoingWhiteboardSession::ref session = boost::make_shared<OutgoingWhiteboardSession>(fullJID, router_);
        sessions_[fullJID] = session;
        session->onSessionTerminated.connect(boost::bind(&WhiteboardSessionManager::deleteSessionEntry, this, _1));
        session->onRequestRejected.connect(boost::bind(&WhiteboardSessionManager::deleteSessionEntry, this, _1));
        return session;
    }

    WhiteboardSession::ref WhiteboardSessionManager::requestSession(const JID& to) {
        WhiteboardSession::ref session = getSession(to);
        if (!session) {
            OutgoingWhiteboardSession::ref outgoingSession = createOutgoingSession(to);
            outgoingSession->startSession();
            return outgoingSession;
        } else {
            return session;
        }
    }

    void WhiteboardSessionManager::handleIncomingSession(IncomingWhiteboardSession::ref session) {
        sessions_[session->getTo()] = session;
        session->onSessionTerminated.connect(boost::bind(&WhiteboardSessionManager::deleteSessionEntry, this, _1));
        onSessionRequest(session);
    }

    JID WhiteboardSessionManager::getFullJID(const JID& bareJID) {
        JID fullReceipientJID;
        int priority = INT_MIN;

        //getAllPresence(bareJID) gives you all presences for the bare JID (i.e. all resources) Remko Tronçon @ 11:11
        std::vector<Presence::ref> presences = presenceOracle_->getAllPresence(bareJID);

        //iterate over them
        foreach(Presence::ref pres, presences) {
            if (pres->getPriority() > priority) {
                // look up caps from the jid
                DiscoInfo::ref info = capsProvider_->getCaps(pres->getFrom());
                if (info && info->hasFeature(DiscoInfo::WhiteboardFeature)) {
                    priority = pres->getPriority();
                    fullReceipientJID = pres->getFrom();
                }
            }
        }

        return fullReceipientJID;
    }

    void WhiteboardSessionManager::deleteSessionEntry(const JID& contact) {
        sessions_.erase(contact);
    }

    void WhiteboardSessionManager::handlePresenceReceived(Presence::ref presence) {
        if (!presence->isAvailable()) {
            WhiteboardSession::ref session = getSession(presence->getFrom());
            if (session) {
                session->cancel();
            }
        }
    }

    void WhiteboardSessionManager::handleAvailableChanged(bool available) {
        if (!available) {
            std::map<JID, WhiteboardSession::ref> sessionsCopy = sessions_;
            std::map<JID, WhiteboardSession::ref>::iterator it;
            for (it = sessionsCopy.begin(); it != sessionsCopy.end(); ++it) {
                it->second->cancel();
            }
        }
    }
}