2021-03-19 22:07:40 +00:00
|
|
|
using Gee;
|
|
|
|
|
|
|
|
using Xmpp;
|
|
|
|
using Dino.Entities;
|
|
|
|
|
|
|
|
namespace Dino {
|
|
|
|
|
|
|
|
public class Calls : StreamInteractionModule, Object {
|
|
|
|
|
2022-02-08 16:45:57 +00:00
|
|
|
public signal void call_incoming(Call call, CallState state, Conversation conversation, bool video, bool multiparty);
|
2021-11-04 16:33:08 +00:00
|
|
|
public signal void call_outgoing(Call call, CallState state, Conversation conversation);
|
2021-03-19 22:07:40 +00:00
|
|
|
|
|
|
|
public signal void call_terminated(Call call, string? reason_name, string? reason_text);
|
2021-11-04 16:33:08 +00:00
|
|
|
public signal void conference_info_received(Call call, Xep.Coin.ConferenceInfo conference_info);
|
2021-03-19 22:07:40 +00:00
|
|
|
|
|
|
|
public static ModuleIdentity<Calls> IDENTITY = new ModuleIdentity<Calls>("calls");
|
|
|
|
public string id { get { return IDENTITY.id; } }
|
|
|
|
|
|
|
|
private StreamInteractor stream_interactor;
|
2021-04-01 10:03:04 +00:00
|
|
|
private Database db;
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2022-02-06 22:48:58 +00:00
|
|
|
// public HashMap<Account, CallState> current_jmi_request_call = new HashMap<Account, CallState>(Account.hash_func, Account.equals_func);
|
|
|
|
public HashMap<Call, PeerState> jmi_request_peer = new HashMap<Call, PeerState>(Call.hash_func, Call.equals_func);
|
2021-11-04 16:33:08 +00:00
|
|
|
public HashMap<Call, CallState> call_states = new HashMap<Call, CallState>(Call.hash_func, Call.equals_func);
|
2021-03-19 22:07:40 +00:00
|
|
|
|
|
|
|
public static void start(StreamInteractor stream_interactor, Database db) {
|
|
|
|
Calls m = new Calls(stream_interactor, db);
|
|
|
|
stream_interactor.add_module(m);
|
|
|
|
}
|
|
|
|
|
|
|
|
private Calls(StreamInteractor stream_interactor, Database db) {
|
|
|
|
this.stream_interactor = stream_interactor;
|
2021-04-01 10:03:04 +00:00
|
|
|
this.db = db;
|
2021-03-19 22:07:40 +00:00
|
|
|
|
|
|
|
stream_interactor.account_added.connect(on_account_added);
|
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
public async CallState? initiate_call(Conversation conversation, bool video) {
|
2021-03-19 22:07:40 +00:00
|
|
|
Call call = new Call();
|
|
|
|
call.direction = Call.DIRECTION_OUTGOING;
|
|
|
|
call.account = conversation.account;
|
|
|
|
call.counterpart = conversation.counterpart;
|
2021-12-19 23:15:05 +00:00
|
|
|
call.ourpart = stream_interactor.get_module(MucManager.IDENTITY).get_own_jid(conversation.counterpart, conversation.account) ?? conversation.account.full_jid;
|
2021-05-03 11:17:17 +00:00
|
|
|
call.time = call.local_time = call.end_time = new DateTime.now_utc();
|
2022-01-04 11:34:16 +00:00
|
|
|
call.encryption = Encryption.UNKNOWN;
|
2021-03-19 22:07:40 +00:00
|
|
|
call.state = Call.State.RINGING;
|
|
|
|
|
|
|
|
stream_interactor.get_module(CallStore.IDENTITY).add_call(call, conversation);
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
var call_state = new CallState(call, stream_interactor);
|
2022-02-10 14:41:24 +00:00
|
|
|
connect_call_state_signals(call_state);
|
2021-11-04 16:33:08 +00:00
|
|
|
call_state.we_should_send_video = video;
|
|
|
|
call_state.we_should_send_audio = true;
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2021-12-18 20:34:39 +00:00
|
|
|
if (conversation.type_ == Conversation.Type.CHAT) {
|
|
|
|
call.add_peer(conversation.counterpart);
|
|
|
|
PeerState peer_state = call_state.set_first_peer(conversation.counterpart);
|
2022-02-06 22:48:58 +00:00
|
|
|
jmi_request_peer[call] = peer_state;
|
2021-12-18 20:34:39 +00:00
|
|
|
yield peer_state.initiate_call(conversation.counterpart);
|
|
|
|
} else {
|
2021-12-19 23:15:05 +00:00
|
|
|
call_state.initiate_groupchat_call.begin(conversation.counterpart);
|
2021-04-01 10:03:04 +00:00
|
|
|
}
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
call_outgoing(call, call_state, conversation);
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
return call_state;
|
2021-04-01 10:03:04 +00:00
|
|
|
}
|
|
|
|
|
2022-02-02 20:37:05 +00:00
|
|
|
public bool can_we_do_calls(Account account) {
|
2021-04-17 12:50:31 +00:00
|
|
|
Plugins.VideoCallPlugin? plugin = Application.get_default().plugin_registry.video_call_plugin;
|
|
|
|
if (plugin == null) return false;
|
|
|
|
|
2022-02-02 20:37:05 +00:00
|
|
|
return plugin.supports(null);
|
2021-04-17 12:50:31 +00:00
|
|
|
}
|
|
|
|
|
2022-02-02 20:37:05 +00:00
|
|
|
public async bool can_conversation_do_calls(Conversation conversation) {
|
2022-02-08 16:45:57 +00:00
|
|
|
if (!can_we_do_calls(conversation.account)) return false;
|
|
|
|
|
2021-12-18 20:34:39 +00:00
|
|
|
if (conversation.type_ == Conversation.Type.CHAT) {
|
|
|
|
return (yield get_call_resources(conversation.account, conversation.counterpart)).size > 0 || has_jmi_resources(conversation.counterpart);
|
|
|
|
} else {
|
2022-02-02 20:37:05 +00:00
|
|
|
bool is_private = stream_interactor.get_module(MucManager.IDENTITY).is_private_room(conversation.account, conversation.counterpart);
|
2022-02-08 21:04:36 +00:00
|
|
|
return is_private && can_initiate_groupcall(conversation.account);
|
2021-12-18 20:34:39 +00:00
|
|
|
}
|
2021-04-17 12:50:31 +00:00
|
|
|
}
|
|
|
|
|
2022-02-02 20:37:05 +00:00
|
|
|
public bool can_initiate_groupcall(Account account) {
|
|
|
|
return stream_interactor.get_module(MucManager.IDENTITY).default_muc_server[account] != null;
|
2021-04-01 10:03:04 +00:00
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
public async Gee.List<Jid> get_call_resources(Account account, Jid counterpart) {
|
2021-03-19 22:07:40 +00:00
|
|
|
ArrayList<Jid> ret = new ArrayList<Jid>();
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
XmppStream? stream = stream_interactor.get_stream(account);
|
2021-03-19 22:07:40 +00:00
|
|
|
if (stream == null) return ret;
|
|
|
|
|
2022-02-02 20:37:05 +00:00
|
|
|
Presence.Flag? presence_flag = stream.get_flag(Presence.Flag.IDENTITY);
|
|
|
|
if (presence_flag == null) return ret;
|
|
|
|
|
|
|
|
Gee.List<Jid>? full_jids = presence_flag.get_resources(counterpart);
|
2021-03-19 22:07:40 +00:00
|
|
|
if (full_jids == null) return ret;
|
|
|
|
|
|
|
|
foreach (Jid full_jid in full_jids) {
|
2022-02-08 16:45:57 +00:00
|
|
|
var module = stream.get_module(Xep.JingleRtp.Module.IDENTITY);
|
|
|
|
if (module == null) return ret;
|
|
|
|
bool supports_rtc = yield module.is_available(stream, full_jid);
|
2021-03-19 22:07:40 +00:00
|
|
|
if (!supports_rtc) continue;
|
|
|
|
ret.add(full_jid);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
public async bool contains_jmi_resources(Account account, Gee.List<Jid> full_jids) {
|
2021-04-29 13:29:41 +00:00
|
|
|
XmppStream? stream = stream_interactor.get_stream(account);
|
|
|
|
if (stream == null) return false;
|
|
|
|
|
|
|
|
foreach (Jid full_jid in full_jids) {
|
|
|
|
bool does_jmi = yield stream_interactor.get_module(EntityInfo.IDENTITY).has_feature(account, full_jid, Xep.JingleMessageInitiation.NS_URI);
|
|
|
|
if (does_jmi) return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
public bool has_jmi_resources(Jid counterpart) {
|
2021-04-01 10:03:04 +00:00
|
|
|
int64 jmi_resources = db.entity.select()
|
2021-11-04 16:33:08 +00:00
|
|
|
.with(db.entity.jid_id, "=", db.get_jid_id(counterpart))
|
2021-04-01 10:03:04 +00:00
|
|
|
.join_with(db.entity_feature, db.entity.caps_hash, db.entity_feature.entity)
|
|
|
|
.with(db.entity_feature.feature, "=", Xep.JingleMessageInitiation.NS_URI)
|
|
|
|
.count();
|
|
|
|
return jmi_resources > 0;
|
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
public bool is_call_in_progress() {
|
|
|
|
foreach (Call call in call_states.keys) {
|
2021-03-19 22:07:40 +00:00
|
|
|
if (call.state == Call.State.IN_PROGRESS || call.state == Call.State.RINGING || call.state == Call.State.ESTABLISHING) {
|
2021-11-04 16:33:08 +00:00
|
|
|
return true;
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
|
|
|
}
|
2021-11-04 16:33:08 +00:00
|
|
|
return false;
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private void on_incoming_call(Account account, Xep.Jingle.Session session) {
|
2022-02-12 16:18:03 +00:00
|
|
|
Jid? muji_room = session.muji_room;
|
2021-03-19 22:07:40 +00:00
|
|
|
bool counterpart_wants_video = false;
|
2021-03-24 13:12:42 +00:00
|
|
|
foreach (Xep.Jingle.Content content in session.contents) {
|
2021-03-19 22:07:40 +00:00
|
|
|
Xep.JingleRtp.Parameters? rtp_content_parameter = content.content_params as Xep.JingleRtp.Parameters;
|
|
|
|
if (rtp_content_parameter == null) continue;
|
|
|
|
if (rtp_content_parameter.media == "video" && session.senders_include_us(content.senders)) {
|
|
|
|
counterpart_wants_video = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
// Check if this comes from a MUJI MUC => accept
|
2022-02-12 16:18:03 +00:00
|
|
|
if (muji_room != null) {
|
|
|
|
debug("[%s] Incoming call from %s from MUJI muc %s", account.bare_jid.to_string(), session.peer_full_jid.to_string(), muji_room.to_string());
|
2021-11-04 16:33:08 +00:00
|
|
|
|
|
|
|
foreach (CallState call_state in call_states.values) {
|
2022-02-12 16:18:03 +00:00
|
|
|
if (call_state.call.account.equals(account) && call_state.group_call != null && call_state.group_call.muc_jid.equals(muji_room)) {
|
2021-11-04 16:33:08 +00:00
|
|
|
if (call_state.peers.keys.contains(session.peer_full_jid)) {
|
|
|
|
PeerState peer_state = call_state.peers[session.peer_full_jid];
|
2022-01-23 18:00:05 +00:00
|
|
|
debug("[%s] Incoming call, we know the peer. Expected %s", account.bare_jid.to_string(), peer_state.waiting_for_inbound_muji_connection.to_string());
|
2021-11-04 16:33:08 +00:00
|
|
|
if (!peer_state.waiting_for_inbound_muji_connection) return;
|
|
|
|
|
|
|
|
peer_state.set_session(session);
|
|
|
|
debug(@"[%s] Accepting incoming MUJI call from %s", account.bare_jid.to_string(), session.peer_full_jid.to_string());
|
|
|
|
peer_state.accept();
|
|
|
|
} else {
|
|
|
|
debug(@"[%s] Incoming call, but didn't see peer in MUC yet", account.bare_jid.to_string());
|
2022-02-06 22:48:58 +00:00
|
|
|
PeerState peer_state = new PeerState(session.peer_full_jid, call_state.call, call_state, stream_interactor);
|
2021-11-04 16:33:08 +00:00
|
|
|
peer_state.set_session(session);
|
|
|
|
call_state.add_peer(peer_state);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
debug(@"[%s] Incoming call from %s", account.bare_jid.to_string(), session.peer_full_jid.to_string());
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2022-02-12 13:07:19 +00:00
|
|
|
// Check if we already got this call via Jingle Message Initiation => accept
|
|
|
|
// PeerState.accept() checks if the call was accepted and ensures that we don't accidentally send video
|
|
|
|
PeerState? peer_state = get_peer_by_sid(account, session.sid, session.peer_full_jid);
|
|
|
|
if (peer_state != null) {
|
|
|
|
jmi_request_peer[peer_state.call].set_session(session);
|
|
|
|
jmi_request_peer[peer_state.call].accept();
|
|
|
|
jmi_request_peer.unset(peer_state.call);
|
2021-11-04 16:33:08 +00:00
|
|
|
return;
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
2021-11-04 16:33:08 +00:00
|
|
|
|
|
|
|
// This is a direct call without prior JMI. Ask user.
|
2022-02-12 13:07:19 +00:00
|
|
|
if (stream_interactor.get_module(MucManager.IDENTITY).might_be_groupchat(session.peer_full_jid.bare_jid, account)) return;
|
|
|
|
peer_state = create_received_call(account, session.peer_full_jid, account.full_jid, counterpart_wants_video);
|
2021-11-04 16:33:08 +00:00
|
|
|
peer_state.set_session(session);
|
2022-02-10 14:41:24 +00:00
|
|
|
Conversation conversation = stream_interactor.get_module(ConversationManager.IDENTITY).get_conversation(peer_state.call.counterpart.bare_jid, account, Conversation.Type.CHAT);
|
|
|
|
call_incoming(peer_state.call, peer_state.call_state, conversation, counterpart_wants_video, false);
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
stream_interactor.module_manager.get_module(account, Xep.JingleRtp.Module.IDENTITY).session_info_type.send_ringing(session);
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
private PeerState create_received_call(Account account, Jid from, Jid to, bool video_requested) {
|
2021-03-19 22:07:40 +00:00
|
|
|
Call call = new Call();
|
|
|
|
if (from.equals_bare(account.bare_jid)) {
|
|
|
|
// Call requested by another of our devices
|
|
|
|
call.direction = Call.DIRECTION_OUTGOING;
|
|
|
|
call.ourpart = from;
|
2021-11-15 12:29:13 +00:00
|
|
|
call.state = Call.State.OTHER_DEVICE;
|
2021-03-19 22:07:40 +00:00
|
|
|
call.counterpart = to;
|
|
|
|
} else {
|
|
|
|
call.direction = Call.DIRECTION_INCOMING;
|
|
|
|
call.ourpart = account.full_jid;
|
2021-11-15 12:29:13 +00:00
|
|
|
call.state = Call.State.RINGING;
|
2021-03-19 22:07:40 +00:00
|
|
|
call.counterpart = from;
|
|
|
|
}
|
2021-12-19 23:15:05 +00:00
|
|
|
call.add_peer(call.counterpart);
|
2021-03-19 22:07:40 +00:00
|
|
|
call.account = account;
|
2021-05-03 11:17:17 +00:00
|
|
|
call.time = call.local_time = call.end_time = new DateTime.now_utc();
|
2022-01-04 11:34:16 +00:00
|
|
|
call.encryption = Encryption.UNKNOWN;
|
2021-03-19 22:07:40 +00:00
|
|
|
|
|
|
|
Conversation conversation = stream_interactor.get_module(ConversationManager.IDENTITY).create_conversation(call.counterpart.bare_jid, account, Conversation.Type.CHAT);
|
|
|
|
stream_interactor.get_module(CallStore.IDENTITY).add_call(call, conversation);
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
var call_state = new CallState(call, stream_interactor);
|
|
|
|
connect_call_state_signals(call_state);
|
2021-12-19 23:15:05 +00:00
|
|
|
PeerState peer_state = call_state.set_first_peer(call.counterpart);
|
2021-11-04 16:33:08 +00:00
|
|
|
call_state.we_should_send_video = video_requested;
|
|
|
|
call_state.we_should_send_audio = true;
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
return peer_state;
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
|
|
|
|
2022-02-12 18:16:38 +00:00
|
|
|
private CallState? get_call_state_by_call_id(Account account, string call_id, Jid? counterpart_jid = null) {
|
2021-11-04 16:33:08 +00:00
|
|
|
foreach (CallState call_state in call_states.values) {
|
2021-12-18 20:34:39 +00:00
|
|
|
if (!call_state.call.account.equals(account)) continue;
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2022-02-07 21:09:51 +00:00
|
|
|
if (call_state.cim_call_id == call_id) {
|
2022-02-12 18:16:38 +00:00
|
|
|
if (counterpart_jid == null) return call_state;
|
|
|
|
|
2022-01-19 15:54:56 +00:00
|
|
|
foreach (Jid jid in call_state.peers.keys) {
|
2022-02-12 18:16:38 +00:00
|
|
|
if (jid.equals_bare(counterpart_jid)) {
|
2022-01-19 15:54:56 +00:00
|
|
|
return call_state;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-02-06 22:48:58 +00:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-02-12 13:07:19 +00:00
|
|
|
private PeerState? get_peer_by_sid(Account account, string sid, Jid jid1, Jid? jid2 = null) {
|
|
|
|
Jid relevant_jid = jid1.equals_bare(account.bare_jid) && jid2 != null ? jid2 : jid1;
|
2022-02-06 22:48:58 +00:00
|
|
|
|
|
|
|
foreach (CallState call_state in call_states.values) {
|
|
|
|
if (!call_state.call.account.equals(account)) continue;
|
|
|
|
|
|
|
|
foreach (PeerState peer_state in call_state.peers.values) {
|
|
|
|
if (peer_state.sid != sid) continue;
|
|
|
|
if (peer_state.jid.equals_bare(relevant_jid)) {
|
|
|
|
return peer_state;
|
|
|
|
}
|
|
|
|
}
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
2021-11-04 16:33:08 +00:00
|
|
|
return null;
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
|
|
|
|
2022-02-12 18:16:38 +00:00
|
|
|
private CallState? create_recv_muji_call(Account account, string call_id, Jid inviter_jid, Jid muc_jid, string message_type) {
|
2021-12-18 20:34:39 +00:00
|
|
|
debug("[%s] Muji call received from %s for MUC %s, type %s", account.bare_jid.to_string(), inviter_jid.to_string(), muc_jid.to_string(), message_type);
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
foreach (Call call in call_states.keys) {
|
2022-02-12 16:18:03 +00:00
|
|
|
if (!call.account.equals(account)) continue;
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2022-01-19 15:54:56 +00:00
|
|
|
CallState call_state = call_states[call];
|
|
|
|
|
2022-02-12 13:07:19 +00:00
|
|
|
if (call.counterparts.size == 1 && call.counterparts.contains(inviter_jid) && call_state.accepted) {
|
2021-11-04 16:33:08 +00:00
|
|
|
// A call is converted into a group call.
|
2022-02-12 18:16:38 +00:00
|
|
|
call_state.cim_call_id = call_id;
|
2022-02-06 22:48:58 +00:00
|
|
|
call_state.join_group_call.begin(muc_jid);
|
|
|
|
return null;
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
2021-04-08 10:07:04 +00:00
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
Call call = new Call();
|
|
|
|
call.direction = Call.DIRECTION_INCOMING;
|
|
|
|
call.ourpart = account.full_jid;
|
2021-12-19 23:15:05 +00:00
|
|
|
call.counterpart = inviter_jid;
|
2021-11-04 16:33:08 +00:00
|
|
|
call.account = account;
|
|
|
|
call.time = call.local_time = call.end_time = new DateTime.now_utc();
|
2022-01-04 11:34:16 +00:00
|
|
|
call.encryption = Encryption.UNKNOWN;
|
2021-11-04 16:33:08 +00:00
|
|
|
call.state = Call.State.RINGING;
|
2021-04-10 21:06:13 +00:00
|
|
|
|
2022-02-06 22:48:58 +00:00
|
|
|
// TODO create conv
|
2021-11-04 16:33:08 +00:00
|
|
|
Conversation? conversation = stream_interactor.get_module(ConversationManager.IDENTITY).get_conversation(inviter_jid.bare_jid, account);
|
2022-02-10 14:41:24 +00:00
|
|
|
if (conversation == null) return null;
|
2021-11-04 16:33:08 +00:00
|
|
|
stream_interactor.get_module(CallStore.IDENTITY).add_call(call, conversation);
|
2021-04-25 17:49:10 +00:00
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
CallState call_state = new CallState(call, stream_interactor);
|
|
|
|
connect_call_state_signals(call_state);
|
|
|
|
call_state.invited_to_group_call = muc_jid;
|
2022-02-06 22:48:58 +00:00
|
|
|
call_state.parent_muc = inviter_jid.bare_jid;
|
2021-04-25 17:49:10 +00:00
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
debug("[%s] on_muji_call_received accepting", account.bare_jid.to_string());
|
2022-02-06 22:48:58 +00:00
|
|
|
|
|
|
|
return call_state;
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
2021-04-25 17:49:10 +00:00
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
private void remove_call_from_datastructures(Call call) {
|
2022-02-06 22:48:58 +00:00
|
|
|
jmi_request_peer.unset(call);
|
2021-11-04 16:33:08 +00:00
|
|
|
call_states.unset(call);
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
private void connect_call_state_signals(CallState call_state) {
|
|
|
|
call_states[call_state.call] = call_state;
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
ulong terminated_handler_id = -1;
|
|
|
|
terminated_handler_id = call_state.terminated.connect((who_terminated, reason_name, reason_text) => {
|
|
|
|
remove_call_from_datastructures(call_state.call);
|
|
|
|
call_terminated(call_state.call, reason_name, reason_text);
|
|
|
|
call_state.disconnect(terminated_handler_id);
|
2021-03-19 22:07:40 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private void on_account_added(Account account) {
|
|
|
|
Xep.Jingle.Module jingle_module = stream_interactor.module_manager.get_module(account, Xep.Jingle.Module.IDENTITY);
|
|
|
|
jingle_module.session_initiate_received.connect((stream, session) => {
|
2021-03-24 13:12:42 +00:00
|
|
|
foreach (Xep.Jingle.Content content in session.contents) {
|
2021-03-19 22:07:40 +00:00
|
|
|
Xep.JingleRtp.Parameters? rtp_content_parameter = content.content_params as Xep.JingleRtp.Parameters;
|
|
|
|
if (rtp_content_parameter != null) {
|
|
|
|
on_incoming_call(account, session);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
Xep.JingleMessageInitiation.Module mi_module = stream_interactor.module_manager.get_module(account, Xep.JingleMessageInitiation.Module.IDENTITY);
|
|
|
|
mi_module.session_proposed.connect((from, to, sid, descriptions) => {
|
2022-02-10 14:41:24 +00:00
|
|
|
|
|
|
|
if (stream_interactor.get_module(MucManager.IDENTITY).might_be_groupchat(from.bare_jid, account)) return;
|
|
|
|
|
2021-03-19 22:07:40 +00:00
|
|
|
bool audio_requested = descriptions.any_match((description) => description.ns_uri == Xep.JingleRtp.NS_URI && description.get_attribute("media") == "audio");
|
|
|
|
bool video_requested = descriptions.any_match((description) => description.ns_uri == Xep.JingleRtp.NS_URI && description.get_attribute("media") == "video");
|
|
|
|
if (!audio_requested && !video_requested) return;
|
2021-11-04 16:33:08 +00:00
|
|
|
|
|
|
|
PeerState peer_state = create_received_call(account, from, to, video_requested);
|
|
|
|
peer_state.sid = sid;
|
2022-02-06 22:48:58 +00:00
|
|
|
CallState call_state = call_states[peer_state.call];
|
|
|
|
|
2022-02-10 14:41:24 +00:00
|
|
|
jmi_request_peer[peer_state.call] = peer_state;
|
|
|
|
|
|
|
|
Conversation conversation = stream_interactor.get_module(ConversationManager.IDENTITY).get_conversation(call_state.call.counterpart.bare_jid, account, Conversation.Type.CHAT);
|
|
|
|
if (call_state.call.direction == Call.DIRECTION_INCOMING) {
|
|
|
|
call_incoming(call_state.call, call_state, conversation, video_requested, false);
|
|
|
|
} else {
|
|
|
|
call_outgoing(call_state.call, call_state, conversation);
|
|
|
|
}
|
2021-03-19 22:07:40 +00:00
|
|
|
});
|
2021-11-15 12:29:13 +00:00
|
|
|
mi_module.session_accepted.connect((from, to, sid) => {
|
2022-02-06 22:48:58 +00:00
|
|
|
PeerState? peer_state = get_peer_by_sid(account, sid, from, to);
|
|
|
|
if (peer_state == null) return;
|
|
|
|
Call call = peer_state.call;
|
2021-03-19 22:07:40 +00:00
|
|
|
|
2022-02-08 21:04:36 +00:00
|
|
|
// Carboned message from our account
|
|
|
|
if (from.equals_bare(account.bare_jid)) {
|
2021-04-01 10:03:04 +00:00
|
|
|
// Ignore carbon from ourselves
|
|
|
|
if (from.equals(account.full_jid)) return;
|
|
|
|
|
2021-12-19 23:15:05 +00:00
|
|
|
call.ourpart = from;
|
2021-11-15 12:29:13 +00:00
|
|
|
call.state = Call.State.OTHER_DEVICE;
|
2021-03-19 22:07:40 +00:00
|
|
|
remove_call_from_datastructures(call);
|
2022-02-08 21:04:36 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We proposed the call. This is a message from our peer.
|
|
|
|
if (call.direction == Call.DIRECTION_OUTGOING &&
|
|
|
|
from.equals_bare(peer_state.jid) && to.equals(account.full_jid)) {
|
2021-11-04 16:33:08 +00:00
|
|
|
// We know the full jid of our peer now
|
2022-02-06 22:48:58 +00:00
|
|
|
call_states[call].rename_peer(jmi_request_peer[call].jid, from);
|
|
|
|
jmi_request_peer[call].call_resource.begin(from);
|
2021-03-19 22:07:40 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
mi_module.session_rejected.connect((from, to, sid) => {
|
2022-02-06 22:48:58 +00:00
|
|
|
PeerState? peer_state = get_peer_by_sid(account, sid, from, to);
|
|
|
|
if (peer_state == null) return;
|
|
|
|
Call call = peer_state.call;
|
2021-04-29 13:03:37 +00:00
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
bool outgoing_reject = call.direction == Call.DIRECTION_OUTGOING && from.equals_bare(call.counterparts[0]);
|
2021-04-29 13:03:37 +00:00
|
|
|
bool incoming_reject = call.direction == Call.DIRECTION_INCOMING && from.equals_bare(account.bare_jid);
|
2021-11-04 16:33:08 +00:00
|
|
|
if (!outgoing_reject && !incoming_reject) return;
|
2021-04-29 13:03:37 +00:00
|
|
|
|
2021-12-19 23:15:05 +00:00
|
|
|
// We don't care if a single person in a group call rejected the call
|
|
|
|
if (incoming_reject && call_states[call].group_call != null) return;
|
2021-04-29 13:03:37 +00:00
|
|
|
|
2021-03-19 22:07:40 +00:00
|
|
|
call.state = Call.State.DECLINED;
|
2021-11-04 16:33:08 +00:00
|
|
|
call_states[call].terminated(from, Xep.Jingle.ReasonElement.DECLINE, "JMI reject");
|
2021-03-19 22:07:40 +00:00
|
|
|
remove_call_from_datastructures(call);
|
|
|
|
});
|
|
|
|
mi_module.session_retracted.connect((from, to, sid) => {
|
2022-02-06 22:48:58 +00:00
|
|
|
PeerState? peer_state = get_peer_by_sid(account, sid, from, to);
|
|
|
|
if (peer_state == null) return;
|
|
|
|
Call call = peer_state.call;
|
2021-04-29 13:03:37 +00:00
|
|
|
|
2021-12-12 11:56:58 +00:00
|
|
|
bool outgoing_retract = call.direction == Call.DIRECTION_OUTGOING && from.equals_bare(account.bare_jid);
|
|
|
|
bool incoming_retract = call.direction == Call.DIRECTION_INCOMING && from.equals_bare(call.counterpart);
|
2021-04-29 13:03:37 +00:00
|
|
|
if (!(outgoing_retract || incoming_retract)) return;
|
|
|
|
|
2021-03-19 22:07:40 +00:00
|
|
|
call.state = Call.State.MISSED;
|
2021-11-04 16:33:08 +00:00
|
|
|
call_states[call].terminated(from, Xep.Jingle.ReasonElement.CANCEL, "JMI retract");
|
2021-03-19 22:07:40 +00:00
|
|
|
remove_call_from_datastructures(call);
|
2021-11-04 16:33:08 +00:00
|
|
|
});
|
|
|
|
|
2022-01-19 15:54:56 +00:00
|
|
|
Xep.CallInvites.Module call_invites_module = stream_interactor.module_manager.get_module(account, Xep.CallInvites.Module.IDENTITY);
|
2022-02-07 21:09:51 +00:00
|
|
|
call_invites_module.call_proposed.connect((from_jid, to_jid, call_id, video_requested, join_methods, message_stanza) => {
|
2022-01-19 15:54:56 +00:00
|
|
|
if (from_jid.equals_bare(account.bare_jid)) return;
|
2022-02-07 21:09:51 +00:00
|
|
|
if (stream_interactor.get_module(MucManager.IDENTITY).is_own_muc_jid(from_jid, account)) return;
|
2022-02-06 22:48:58 +00:00
|
|
|
|
2022-02-08 16:45:57 +00:00
|
|
|
bool multiparty = false;
|
2022-02-06 22:48:58 +00:00
|
|
|
CallState? call_state = null;
|
|
|
|
|
2022-01-19 15:54:56 +00:00
|
|
|
foreach (StanzaNode join_method_node in join_methods) {
|
2022-02-06 22:48:58 +00:00
|
|
|
if (join_method_node.name == "muji" && join_method_node.ns_uri == Xep.Muji.NS_URI) {
|
|
|
|
// This is a MUJI invite
|
2022-02-07 21:09:51 +00:00
|
|
|
|
|
|
|
// Disregard calls from muc history
|
|
|
|
DateTime? delay = Xep.DelayedDelivery.get_time_for_message(message_stanza, from_jid.bare_jid);
|
|
|
|
if (delay != null) return;
|
|
|
|
|
2022-01-19 15:54:56 +00:00
|
|
|
string? room_jid_str = join_method_node.get_attribute("room");
|
|
|
|
if (room_jid_str == null) return;
|
|
|
|
Jid room_jid = new Jid(room_jid_str);
|
2022-02-12 18:16:38 +00:00
|
|
|
call_state = create_recv_muji_call(account, call_id, from_jid, room_jid, message_stanza.type_);
|
2022-02-08 16:45:57 +00:00
|
|
|
|
|
|
|
multiparty = true;
|
2022-02-06 22:48:58 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
} else if (join_method_node.name == "jingle" && join_method_node.ns_uri == Xep.CallInvites.NS_URI) {
|
|
|
|
// This is an invite for a direct Jingle session
|
2022-02-07 21:09:51 +00:00
|
|
|
|
2022-02-10 14:41:24 +00:00
|
|
|
if (stream_interactor.get_module(MucManager.IDENTITY).might_be_groupchat(from_jid.bare_jid, account)) return;
|
2022-02-06 22:48:58 +00:00
|
|
|
|
|
|
|
string? sid = join_method_node.get_attribute("sid");
|
|
|
|
if (sid == null) return;
|
|
|
|
|
|
|
|
PeerState peer_state = create_received_call(account, from_jid, to_jid, video_requested);
|
|
|
|
peer_state.sid = sid;
|
|
|
|
|
|
|
|
call_state = call_states[peer_state.call];
|
|
|
|
|
|
|
|
jmi_request_peer[call_state.call] = peer_state;
|
|
|
|
break;
|
2022-01-19 15:54:56 +00:00
|
|
|
}
|
|
|
|
}
|
2021-11-04 16:33:08 +00:00
|
|
|
|
2022-02-06 22:48:58 +00:00
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
if (call_state == null) return;
|
|
|
|
|
2022-02-06 22:48:58 +00:00
|
|
|
call_state.we_should_send_audio = true;
|
|
|
|
call_state.we_should_send_video = video_requested;
|
|
|
|
|
|
|
|
call_state.use_cim = true;
|
2022-02-07 21:09:51 +00:00
|
|
|
call_state.cim_call_id = call_id;
|
2022-02-06 22:48:58 +00:00
|
|
|
call_state.cim_counterpart = message_stanza.type_ == MessageStanza.TYPE_GROUPCHAT ? from_jid.bare_jid : from_jid;
|
|
|
|
call_state.cim_message_type = message_stanza.type_;
|
|
|
|
|
|
|
|
Conversation? conversation = stream_interactor.get_module(ConversationManager.IDENTITY).approx_conversation_for_stanza(from_jid, to_jid, account, message_stanza.type_);
|
|
|
|
if (conversation == null) return;
|
|
|
|
|
2022-02-10 14:41:24 +00:00
|
|
|
if (call_state.call.direction == Call.DIRECTION_INCOMING) {
|
|
|
|
call_incoming(call_state.call, call_state, conversation, video_requested, multiparty);
|
|
|
|
} else {
|
|
|
|
call_outgoing(call_state.call, call_state, conversation);
|
|
|
|
}
|
2022-02-06 22:48:58 +00:00
|
|
|
});
|
2022-02-07 21:09:51 +00:00
|
|
|
call_invites_module.call_accepted.connect((from_jid, to_jid, call_id, message_type) => {
|
2022-02-08 21:04:36 +00:00
|
|
|
// Carboned message from our account
|
|
|
|
if (from_jid.equals_bare(account.bare_jid)) {
|
2022-02-12 18:16:38 +00:00
|
|
|
|
|
|
|
CallState? call_state = get_call_state_by_call_id(account, call_id);
|
|
|
|
if (call_state == null) return;
|
|
|
|
Call call = call_state.call;
|
|
|
|
|
2022-02-06 22:48:58 +00:00
|
|
|
// Ignore carbon from ourselves
|
|
|
|
if (from_jid.equals(account.full_jid)) return;
|
|
|
|
|
|
|
|
// We accepted the call from another device
|
|
|
|
call.ourpart = from_jid;
|
|
|
|
call.state = Call.State.OTHER_DEVICE;
|
|
|
|
remove_call_from_datastructures(call);
|
2022-02-08 21:04:36 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-02-12 18:16:38 +00:00
|
|
|
CallState? call_state = get_call_state_by_call_id(account, call_id, from_jid);
|
|
|
|
if (call_state == null) return;
|
|
|
|
Call call = call_state.call;
|
|
|
|
|
2022-02-08 21:04:36 +00:00
|
|
|
// We proposed the call. This is a message from our peer.
|
|
|
|
if (call.direction == Call.DIRECTION_OUTGOING &&
|
|
|
|
to_jid.equals(account.full_jid)) {
|
2022-02-06 22:48:58 +00:00
|
|
|
// We know the full jid of our peer now
|
2022-02-08 21:04:36 +00:00
|
|
|
call_state.rename_peer(jmi_request_peer[call].jid, from_jid);
|
2022-02-06 22:48:58 +00:00
|
|
|
jmi_request_peer[call].call_resource.begin(from_jid);
|
|
|
|
}
|
2021-11-04 16:33:08 +00:00
|
|
|
});
|
2022-02-07 21:09:51 +00:00
|
|
|
call_invites_module.call_retracted.connect((from_jid, to_jid, call_id, message_type) => {
|
2021-11-04 16:33:08 +00:00
|
|
|
if (from_jid.equals_bare(account.bare_jid)) return;
|
|
|
|
|
|
|
|
// The call was retracted by the counterpart
|
2022-02-12 18:16:38 +00:00
|
|
|
CallState? call_state = get_call_state_by_call_id(account, call_id, from_jid);
|
2021-11-04 16:33:08 +00:00
|
|
|
if (call_state == null) return;
|
|
|
|
|
2021-12-18 20:34:39 +00:00
|
|
|
if (call_state.call.state != Call.State.RINGING) {
|
|
|
|
debug("%s tried to retract a call that's in state %s. Ignoring.", from_jid.to_string(), call_state.call.state.to_string());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO prevent other MUC occupants from retracting a call
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
call_state.call.state = Call.State.MISSED;
|
|
|
|
remove_call_from_datastructures(call_state.call);
|
|
|
|
});
|
2022-02-08 21:04:36 +00:00
|
|
|
call_invites_module.call_rejected.connect((from_jid, to_jid, call_id, message_type) => {
|
2022-02-12 18:16:38 +00:00
|
|
|
// We rejected an invite from another device
|
|
|
|
if (from_jid.equals_bare(account.bare_jid)) {
|
|
|
|
CallState? call_state = get_call_state_by_call_id(account, call_id);
|
|
|
|
if (call_state == null) return;
|
|
|
|
Call call = call_state.call;
|
|
|
|
call.state = Call.State.DECLINED;
|
|
|
|
}
|
|
|
|
|
2021-11-04 16:33:08 +00:00
|
|
|
if (from_jid.equals_bare(account.bare_jid)) return;
|
2022-02-08 21:04:36 +00:00
|
|
|
debug(@"[%s] %s rejected our MUJI invite", account.bare_jid.to_string(), from_jid.to_string());
|
2021-11-04 16:33:08 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
stream_interactor.module_manager.get_module(account, Xep.Coin.Module.IDENTITY).coin_info_received.connect((jid, info) => {
|
|
|
|
foreach (Call call in call_states.keys) {
|
|
|
|
if (call.counterparts[0].equals_bare(jid)) {
|
|
|
|
conference_info_received(call, info);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2021-03-19 22:07:40 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|