mirror of
https://gitlab.linphone.org/BC/public/linphone-iphone.git
synced 2026-01-28 16:49:20 +00:00
461 lines
16 KiB
C++
461 lines
16 KiB
C++
/*
|
|
* client-group-chat-room.cpp
|
|
* Copyright (C) 2010-2017 Belledonne Communications SARL
|
|
*
|
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "linphone/utils/utils.h"
|
|
|
|
#include "address/address-p.h"
|
|
#include "c-wrapper/c-wrapper.h"
|
|
#include "client-group-chat-room-p.h"
|
|
#include "conference/participant-p.h"
|
|
#include "conference/remote-conference-event-handler.h"
|
|
#include "conference/remote-conference-p.h"
|
|
#include "conference/session/call-session-p.h"
|
|
#include "core/core-p.h"
|
|
#include "event-log/events.h"
|
|
#include "logger/logger.h"
|
|
#include "sal/refer-op.h"
|
|
|
|
// =============================================================================
|
|
|
|
using namespace std;
|
|
|
|
LINPHONE_BEGIN_NAMESPACE
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
shared_ptr<CallSession> ClientGroupChatRoomPrivate::createSession () {
|
|
L_Q();
|
|
L_Q_T(RemoteConference, qConference);
|
|
|
|
CallSessionParams csp;
|
|
csp.addCustomHeader("Require", "recipient-list-invite");
|
|
|
|
shared_ptr<Participant> focus = qConference->getPrivate()->focus;
|
|
shared_ptr<CallSession> session = focus->getPrivate()->createSession(*q, &csp, false, q);
|
|
const Address &myAddress = q->getMe()->getAddress();
|
|
session->configure(LinphoneCallOutgoing, nullptr, nullptr, myAddress, focus->getAddress());
|
|
session->initiateOutgoing();
|
|
|
|
Address addr = myAddress;
|
|
addr.setParam("text");
|
|
session->getPrivate()->getOp()->set_contact_address(addr.getPrivate()->getInternalAddress());
|
|
return session;
|
|
}
|
|
|
|
void ClientGroupChatRoomPrivate::notifyReceived (string body) {
|
|
L_Q_T(RemoteConference, qConference);
|
|
qConference->getPrivate()->eventHandler->notifyReceived(body);
|
|
}
|
|
|
|
// =============================================================================
|
|
|
|
ClientGroupChatRoom::ClientGroupChatRoom (
|
|
const std::shared_ptr<Core> &core,
|
|
const Address &me,
|
|
const std::string &uri,
|
|
const std::string &subject
|
|
) : ChatRoom(*new ClientGroupChatRoomPrivate, core, me), RemoteConference(core->getCCore(), me, nullptr) {
|
|
L_D_T(RemoteConference, dConference);
|
|
dConference->focus = make_shared<Participant>(Address(uri));
|
|
RemoteConference::setSubject(subject);
|
|
}
|
|
|
|
int ClientGroupChatRoom::getCapabilities () const {
|
|
return static_cast<int>(Capabilities::Conference);
|
|
}
|
|
|
|
void ClientGroupChatRoom::addParticipant (const Address &addr, const CallSessionParams *params, bool hasMedia) {
|
|
list<Address> addresses;
|
|
addresses.push_back(addr);
|
|
addParticipants(addresses, params, hasMedia);
|
|
}
|
|
|
|
void ClientGroupChatRoom::addParticipants (
|
|
const list<Address> &addresses,
|
|
const CallSessionParams *params,
|
|
bool hasMedia
|
|
) {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
if (addresses.empty())
|
|
return;
|
|
|
|
if ((d->state != ChatRoom::State::Instantiated) && (d->state != ChatRoom::State::Created)) {
|
|
lError() << "Cannot add participants to the ClientGroupChatRoom in a state other than Instantiated or Created";
|
|
return;
|
|
}
|
|
|
|
list<Address> sortedAddresses(addresses);
|
|
sortedAddresses.sort();
|
|
sortedAddresses.unique();
|
|
|
|
Content content;
|
|
content.setBody(getResourceLists(sortedAddresses));
|
|
content.setContentType("application/resource-lists+xml");
|
|
content.setContentDisposition("recipient-list");
|
|
|
|
shared_ptr<CallSession> session = dConference->focus->getPrivate()->getSession();
|
|
if (session)
|
|
session->update(nullptr, getSubject(), &content);
|
|
else {
|
|
session = d->createSession();
|
|
session->startInvite(nullptr, getSubject(), &content);
|
|
if (d->state == ChatRoom::State::Instantiated)
|
|
d->setState(ChatRoom::State::CreationPending);
|
|
}
|
|
}
|
|
|
|
bool ClientGroupChatRoom::canHandleParticipants () const {
|
|
return RemoteConference::canHandleParticipants();
|
|
}
|
|
|
|
shared_ptr<Participant> ClientGroupChatRoom::findParticipant (const Address &addr) const {
|
|
return RemoteConference::findParticipant(addr);
|
|
}
|
|
|
|
const Address &ClientGroupChatRoom::getConferenceAddress () const {
|
|
return RemoteConference::getConferenceAddress();
|
|
}
|
|
|
|
int ClientGroupChatRoom::getNbParticipants () const {
|
|
return RemoteConference::getNbParticipants();
|
|
}
|
|
|
|
list<shared_ptr<Participant>> ClientGroupChatRoom::getParticipants () const {
|
|
return RemoteConference::getParticipants();
|
|
}
|
|
|
|
const string &ClientGroupChatRoom::getSubject () const {
|
|
return RemoteConference::getSubject();
|
|
}
|
|
|
|
void ClientGroupChatRoom::join () {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
shared_ptr<CallSession> session = dConference->focus->getPrivate()->getSession();
|
|
if (!session && d->state == ChatRoom::State::Instantiated) {
|
|
session = d->createSession();
|
|
session->startInvite(nullptr, "", nullptr);
|
|
d->setState(ChatRoom::State::CreationPending);
|
|
}
|
|
}
|
|
|
|
void ClientGroupChatRoom::leave () {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
dConference->eventHandler->unsubscribe();
|
|
|
|
shared_ptr<CallSession> session = dConference->focus->getPrivate()->getSession();
|
|
if (session)
|
|
session->terminate();
|
|
else {
|
|
session = d->createSession();
|
|
session->startInvite(nullptr, "", nullptr);
|
|
}
|
|
|
|
d->setState(ChatRoom::State::TerminationPending);
|
|
}
|
|
|
|
void ClientGroupChatRoom::removeParticipant (const shared_ptr<const Participant> &participant) {
|
|
LinphoneCore *cCore = CoreAccessor::getCore()->getCCore();
|
|
|
|
SalReferOp *referOp = new SalReferOp(cCore->sal);
|
|
LinphoneAddress *lAddr = linphone_address_new(getConferenceAddress().asString().c_str());
|
|
linphone_configure_op(cCore, referOp, lAddr, nullptr, false);
|
|
linphone_address_unref(lAddr);
|
|
Address referToAddr = participant->getAddress();
|
|
referToAddr.setParam("text");
|
|
referToAddr.setUriParam("method", "BYE");
|
|
referToAddr.setDomain("");
|
|
referToAddr.setPort(-1);
|
|
referOp->send_refer(referToAddr.getPrivate()->getInternalAddress());
|
|
referOp->unref();
|
|
}
|
|
|
|
void ClientGroupChatRoom::removeParticipants (const list<shared_ptr<Participant>> &participants) {
|
|
RemoteConference::removeParticipants(participants);
|
|
}
|
|
|
|
void ClientGroupChatRoom::setParticipantAdminStatus (shared_ptr<Participant> &participant, bool isAdmin) {
|
|
if (isAdmin == participant->isAdmin())
|
|
return;
|
|
|
|
if (!getMe()->isAdmin()) {
|
|
lError() << "Cannot change the participant admin status because I am not admin";
|
|
return;
|
|
}
|
|
|
|
LinphoneCore *cCore = CoreAccessor::getCore()->getCCore();
|
|
|
|
SalReferOp *referOp = new SalReferOp(cCore->sal);
|
|
LinphoneAddress *lAddr = linphone_address_new(getConferenceAddress().asString().c_str());
|
|
linphone_configure_op(cCore, referOp, lAddr, nullptr, false);
|
|
linphone_address_unref(lAddr);
|
|
Address referToAddr = participant->getAddress();
|
|
referToAddr.setParam("text");
|
|
referToAddr.setParam("admin", Utils::toString(isAdmin));
|
|
referToAddr.setDomain("");
|
|
referToAddr.setPort(-1);
|
|
referOp->send_refer(referToAddr.getPrivate()->getInternalAddress());
|
|
referOp->unref();
|
|
}
|
|
|
|
void ClientGroupChatRoom::setSubject (const string &subject) {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
if (d->state != ChatRoom::State::Created) {
|
|
lError() << "Cannot change the ClientGroupChatRoom subject in a state other than Created";
|
|
return;
|
|
}
|
|
|
|
if (!getMe()->isAdmin()) {
|
|
lError() << "Cannot change the ClientGroupChatRoom subject because I am not admin";
|
|
return;
|
|
}
|
|
|
|
shared_ptr<CallSession> session = dConference->focus->getPrivate()->getSession();
|
|
if (session)
|
|
session->update(nullptr, subject);
|
|
else {
|
|
session = d->createSession();
|
|
session->startInvite(nullptr, subject, nullptr);
|
|
}
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
void ClientGroupChatRoom::onChatMessageReceived (const shared_ptr<ChatMessage> &msg) {
|
|
|
|
}
|
|
|
|
void ClientGroupChatRoom::onConferenceCreated (const Address &addr) {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
dConference->conferenceAddress = addr;
|
|
d->peerAddress = addr;
|
|
CoreAccessor::getCore()->getPrivate()->insertChatRoom(getSharedFromThis());
|
|
d->setState(ChatRoom::State::Created);
|
|
CoreAccessor::getCore()->getPrivate()->insertChatRoomWithDb(getSharedFromThis());
|
|
}
|
|
|
|
void ClientGroupChatRoom::onConferenceTerminated (const Address &addr) {
|
|
L_D();
|
|
d->setState(ChatRoom::State::Terminated);
|
|
}
|
|
|
|
void ClientGroupChatRoom::onParticipantAdded (time_t tm, const Address &addr) {
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
if (isMe(addr))
|
|
return;
|
|
|
|
shared_ptr<Participant> participant = findParticipant(addr);
|
|
if (participant) {
|
|
lWarning() << "Participant " << participant << " added but already in the list of participants!";
|
|
return;
|
|
}
|
|
|
|
participant = make_shared<Participant>(addr);
|
|
dConference->participants.push_back(participant);
|
|
LinphoneChatRoom *cr = L_GET_C_BACK_PTR(this);
|
|
LinphoneChatRoomCbs *cbs = linphone_chat_room_get_callbacks(cr);
|
|
LinphoneChatRoomCbsParticipantAddedCb cb = linphone_chat_room_cbs_get_participant_added(cbs);
|
|
shared_ptr<ConferenceParticipantEvent> event = make_shared<ConferenceParticipantEvent>(
|
|
EventLog::Type::ConferenceParticipantAdded,
|
|
tm,
|
|
dConference->conferenceAddress,
|
|
dConference->eventHandler->getLastNotify(),
|
|
addr
|
|
);
|
|
Conference::getCore()->cppCore->getPrivate()->mainDb->addEvent(event);
|
|
|
|
if (cb)
|
|
cb(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
void ClientGroupChatRoom::onParticipantRemoved (time_t tm, const Address &addr) {
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
shared_ptr<Participant> participant = findParticipant(addr);
|
|
if (!participant) {
|
|
lWarning() << "Participant " << addr.asString() << " removed but not in the list of participants!";
|
|
return;
|
|
}
|
|
|
|
LinphoneChatRoom *cr = L_GET_C_BACK_PTR(this);
|
|
LinphoneChatRoomCbs *cbs = linphone_chat_room_get_callbacks(cr);
|
|
LinphoneChatRoomCbsParticipantRemovedCb cb = linphone_chat_room_cbs_get_participant_removed(cbs);
|
|
shared_ptr<ConferenceParticipantEvent> event = make_shared<ConferenceParticipantEvent>(
|
|
EventLog::Type::ConferenceParticipantRemoved,
|
|
tm,
|
|
dConference->conferenceAddress,
|
|
dConference->eventHandler->getLastNotify(),
|
|
addr
|
|
);
|
|
Conference::getCore()->cppCore->getPrivate()->mainDb->addEvent(event);
|
|
|
|
if (cb)
|
|
cb(cr, L_GET_C_BACK_PTR(event));
|
|
|
|
dConference->participants.remove(participant);
|
|
}
|
|
|
|
void ClientGroupChatRoom::onParticipantSetAdmin (time_t tm, const Address &addr, bool isAdmin) {
|
|
L_D_T(RemoteConference, dConference);
|
|
shared_ptr<Participant> participant;
|
|
if (isMe(addr))
|
|
participant = getMe();
|
|
else
|
|
participant = findParticipant(addr);
|
|
if (!participant) {
|
|
lWarning() << "Participant " << participant << " admin status has been changed but is not in the list of participants!";
|
|
return;
|
|
}
|
|
|
|
participant->getPrivate()->setAdmin(isAdmin);
|
|
LinphoneChatRoom *cr = L_GET_C_BACK_PTR(this);
|
|
LinphoneChatRoomCbs *cbs = linphone_chat_room_get_callbacks(cr);
|
|
LinphoneChatRoomCbsParticipantAdminStatusChangedCb cb = linphone_chat_room_cbs_get_participant_admin_status_changed(cbs);
|
|
shared_ptr<ConferenceParticipantEvent> event = make_shared<ConferenceParticipantEvent>(
|
|
isAdmin ? EventLog::Type::ConferenceParticipantSetAdmin : EventLog::Type::ConferenceParticipantUnsetAdmin,
|
|
tm,
|
|
dConference->conferenceAddress,
|
|
dConference->eventHandler->getLastNotify(),
|
|
addr
|
|
);
|
|
Conference::getCore()->cppCore->getPrivate()->mainDb->addEvent(event);
|
|
|
|
if (cb)
|
|
cb(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
void ClientGroupChatRoom::onSubjectChanged (time_t tm, const std::string &subject) {
|
|
L_D_T(RemoteConference, dConference);
|
|
RemoteConference::setSubject(subject);
|
|
LinphoneChatRoom *cr = L_GET_C_BACK_PTR(this);
|
|
LinphoneChatRoomCbs *cbs = linphone_chat_room_get_callbacks(cr);
|
|
LinphoneChatRoomCbsSubjectChangedCb cb = linphone_chat_room_cbs_get_subject_changed(cbs);
|
|
shared_ptr<ConferenceSubjectEvent> event = make_shared<ConferenceSubjectEvent>(
|
|
tm,
|
|
dConference->conferenceAddress,
|
|
dConference->eventHandler->getLastNotify(),
|
|
subject
|
|
);
|
|
Conference::getCore()->cppCore->getPrivate()->mainDb->addEvent(event);
|
|
|
|
if (cb)
|
|
cb(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
void ClientGroupChatRoom::onParticipantDeviceAdded (time_t tm, const Address &addr, const Address &gruu) {
|
|
L_D_T(RemoteConference, dConference);
|
|
shared_ptr<Participant> participant;
|
|
if (isMe(addr))
|
|
participant = getMe();
|
|
else
|
|
participant = findParticipant(addr);
|
|
if (!participant) {
|
|
lWarning() << "Participant " << participant << " added a device but is not in the list of participants!";
|
|
return;
|
|
}
|
|
participant->getPrivate()->addDevice(gruu);
|
|
LinphoneChatRoom *cr = L_GET_C_BACK_PTR(this);
|
|
LinphoneChatRoomCbs *cbs = linphone_chat_room_get_callbacks(cr);
|
|
LinphoneChatRoomCbsParticipantDeviceAddedCb cb = linphone_chat_room_cbs_get_participant_device_added(cbs);
|
|
shared_ptr<ConferenceParticipantDeviceEvent> event = make_shared<ConferenceParticipantDeviceEvent>(
|
|
EventLog::Type::ConferenceParticipantDeviceAdded,
|
|
tm,
|
|
dConference->conferenceAddress,
|
|
dConference->eventHandler->getLastNotify(),
|
|
addr,
|
|
gruu
|
|
);
|
|
Conference::getCore()->cppCore->getPrivate()->mainDb->addEvent(event);
|
|
|
|
if (cb)
|
|
cb(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
void ClientGroupChatRoom::onParticipantDeviceRemoved (time_t tm, const Address &addr, const Address &gruu) {
|
|
L_D_T(RemoteConference, dConference);
|
|
shared_ptr<Participant> participant;
|
|
if (isMe(addr))
|
|
participant = getMe();
|
|
else
|
|
participant = findParticipant(addr);
|
|
if (!participant) {
|
|
lWarning() << "Participant " << participant << " removed a device but is not in the list of participants!";
|
|
return;
|
|
}
|
|
participant->getPrivate()->removeDevice(gruu);
|
|
LinphoneChatRoom *cr = L_GET_C_BACK_PTR(this);
|
|
LinphoneChatRoomCbs *cbs = linphone_chat_room_get_callbacks(cr);
|
|
LinphoneChatRoomCbsParticipantDeviceRemovedCb cb = linphone_chat_room_cbs_get_participant_device_removed(cbs);
|
|
shared_ptr<ConferenceParticipantDeviceEvent> event = make_shared<ConferenceParticipantDeviceEvent>(
|
|
EventLog::Type::ConferenceParticipantDeviceRemoved,
|
|
tm,
|
|
dConference->conferenceAddress,
|
|
dConference->eventHandler->getLastNotify(),
|
|
addr,
|
|
gruu
|
|
);
|
|
Conference::getCore()->cppCore->getPrivate()->mainDb->addEvent(event);
|
|
|
|
if (cb)
|
|
cb(cr, L_GET_C_BACK_PTR(event));
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
void ClientGroupChatRoom::onCallSessionSetReleased (const std::shared_ptr<const CallSession> &session) {
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
ParticipantPrivate *participantPrivate = dConference->focus->getPrivate();
|
|
if (session == participantPrivate->getSession())
|
|
participantPrivate->removeSession();
|
|
}
|
|
|
|
void ClientGroupChatRoom::onCallSessionStateChanged (
|
|
const std::shared_ptr<const CallSession> &session,
|
|
LinphoneCallState state,
|
|
const string &message
|
|
) {
|
|
L_D();
|
|
L_D_T(RemoteConference, dConference);
|
|
|
|
if (state == LinphoneCallConnected) {
|
|
if (d->state == ChatRoom::State::CreationPending) {
|
|
Address addr(session->getRemoteContactAddress()->asStringUriOnly());
|
|
onConferenceCreated(addr);
|
|
if (session->getRemoteContactAddress()->hasParam("isfocus"))
|
|
dConference->eventHandler->subscribe(getConferenceAddress());
|
|
} else if (d->state == ChatRoom::State::TerminationPending)
|
|
dConference->focus->getPrivate()->getSession()->terminate();
|
|
} else if (state == LinphoneCallReleased && d->state == ChatRoom::State::TerminationPending) {
|
|
onConferenceTerminated(getConferenceAddress());
|
|
} else if (state == LinphoneCallError && d->state == ChatRoom::State::CreationPending) {
|
|
d->setState(ChatRoom::State::CreationFailed);
|
|
}
|
|
}
|
|
|
|
LINPHONE_END_NAMESPACE
|