call_multi_tester.c 67.21 KiB
/*
 * Copyright (c) 2010-2022 Belledonne Communications SARL.
 * This file is part of Liblinphone
 * (see https://gitlab.linphone.org/BC/public/liblinphone).
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Affero General Public License as
 * published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
 * You should have received a copy of the GNU Affero General Public License
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
#include "belle-sip/sipstack.h"
#include "liblinphone_tester.h"
#include "linphone/core.h"
#include "linphone/lpconfig.h"
#include "mediastreamer2/msutils.h"
#include "shared_tester_functions.h"
#include "tester_utils.h"
#include <sys/stat.h>
#include <sys/types.h>
#ifdef _WIN32
#define unlink _unlink
#endif
 * With IPV6, Flexisip automatically switches to TCP, so it's no more possible to really have Laure configured with UDP
 * Anyway for IPV4, it's still a good opportunity to test UDP.
static const char *get_laure_rc(void) {
	if (liblinphone_tester_ipv6_available()) {
		return "laure_tcp_rc";
	} else {
		return "laure_rc_udp";
static void call_waiting_indication_with_param(bool_t enable_caller_privacy) {
	bctbx_list_t *iterator;
	bctbx_list_t *lcs;
	LinphoneCall *pauline_called_by_marie;
	LinphoneCall *pauline_called_by_laure = NULL;
	LinphoneCoreManager *marie = ms_new0(LinphoneCoreManager, 1);
	linphone_core_manager_init(marie, "marie_rc", NULL);
	linphone_core_remove_supported_tag(marie->lc, "gruu");
	linphone_core_manager_start(marie, TRUE);
	LinphoneCoreManager *pauline = ms_new0(LinphoneCoreManager, 1);
	linphone_core_manager_init(pauline, "pauline_tcp_rc", NULL);
	linphone_core_remove_supported_tag(pauline->lc, "gruu");
	linphone_core_manager_start(pauline, TRUE);
	LinphoneCoreManager *laure = ms_new0(LinphoneCoreManager, 1);
	linphone_core_manager_init(laure, get_laure_rc(), NULL);
	linphone_core_remove_supported_tag(laure->lc, "gruu");
	linphone_core_manager_start(laure, TRUE);
	LinphoneCallParams *laure_params = linphone_core_create_call_params(laure->lc, NULL);
	LinphoneCallParams *marie_params = linphone_core_create_call_params(marie->lc, NULL);
	if (enable_caller_privacy) linphone_call_params_set_privacy(marie_params, LinphonePrivacyId);
	lcs = bctbx_list_append(NULL, marie->lc);
	lcs = bctbx_list_append(lcs, pauline->lc);
7172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
lcs = bctbx_list_append(lcs, laure->lc); BC_ASSERT_TRUE(call_with_caller_params(marie, pauline, marie_params)); linphone_call_params_unref(marie_params); pauline_called_by_marie = linphone_core_get_current_call(pauline->lc); if (enable_caller_privacy) linphone_call_params_set_privacy(laure_params, LinphonePrivacyId); BC_ASSERT_PTR_NOT_NULL(linphone_core_invite_address_with_params(laure->lc, pauline->identity, laure_params)); linphone_call_params_unref(laure_params); BC_ASSERT_TRUE(wait_for(laure->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived, 2)); BC_ASSERT_EQUAL(laure->stat.number_of_LinphoneCallOutgoingProgress, 1, int, "%d"); BC_ASSERT_TRUE(wait_for(laure->lc, pauline->lc, &laure->stat.number_of_LinphoneCallOutgoingRinging, 1)); bctbx_list_t *calls = bctbx_list_copy(linphone_core_get_calls(pauline->lc)); for (iterator = calls; iterator; iterator = bctbx_list_next(iterator)) { LinphoneCall *call = (LinphoneCall *)bctbx_list_get_data(iterator); if (call != pauline_called_by_marie) { /*fine, this is the call waiting*/ pauline_called_by_laure = call; linphone_call_accept(pauline_called_by_laure); } } bctbx_list_free(calls); BC_ASSERT_TRUE(wait_for(laure->lc, pauline->lc, &laure->stat.number_of_LinphoneCallConnected, 1)); BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &marie->stat.number_of_LinphoneCallPausedByRemote, 1)); if (pauline_called_by_laure && enable_caller_privacy) BC_ASSERT_EQUAL(linphone_call_params_get_privacy(linphone_call_get_current_params(pauline_called_by_laure)), LinphonePrivacyId, int, "%d"); /*wait a bit for ACK to be sent*/ wait_for_list(lcs, NULL, 0, 1000); linphone_core_terminate_all_calls(pauline->lc); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallReleased, 1, 10000)); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(laure); bctbx_list_free(lcs); } static void call_waiting_indication(void) { call_waiting_indication_with_param(FALSE); } static void call_waiting_indication_with_privacy(void) { call_waiting_indication_with_param(TRUE); } static void second_call_rejection(bool_t second_without_audio) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); LinphoneCall *pauline_call; LinphoneCallParams *params; LinphoneCall *marie_call; /*start a call to pauline*/ linphone_core_invite_address(marie->lc, pauline->identity); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallOutgoingRinging, 1));
141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
/*attempt to send a second call while the first one is not answered. * It must be rejected by the core, since the audio resources are already engaged for the first call*/ params = linphone_core_create_call_params(marie->lc, NULL); linphone_call_params_enable_audio(params, !second_without_audio); marie_call = linphone_core_invite_with_params(marie->lc, "sip:laure_non_existent@sip.example.org", params); linphone_call_params_unref(params); if (second_without_audio) { BC_ASSERT_PTR_NOT_NULL(marie_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallError, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallReleased, 1)); } else { BC_ASSERT_PTR_NULL(marie_call); } pauline_call = linphone_core_get_current_call(pauline->lc); BC_ASSERT_PTR_NOT_NULL(pauline_call); if (pauline_call) { linphone_call_accept(pauline_call); } BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1)); end_call(pauline, marie); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void second_call_rejected_if_first_one_in_progress(void) { second_call_rejection(FALSE); } static void second_call_allowed_if_not_using_audio(void) { second_call_rejection(TRUE); } static void incoming_call_accepted_when_outgoing_call_in_state(LinphoneCallState state) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); LinphoneCoreManager *laure = linphone_core_manager_new(get_laure_rc()); bctbx_list_t *lcs; LinphoneCallParams *laure_params = linphone_core_create_call_params(laure->lc, NULL); LinphoneCallParams *marie_params = linphone_core_create_call_params(marie->lc, NULL); lcs = bctbx_list_append(NULL, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); lcs = bctbx_list_append(lcs, laure->lc); if (state == LinphoneCallOutgoingRinging || state == LinphoneCallOutgoingEarlyMedia) { BC_ASSERT_PTR_NOT_NULL(linphone_core_invite_address_with_params(marie->lc, pauline->identity, marie_params)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1)); if (state == LinphoneCallOutgoingEarlyMedia) linphone_call_accept_early_media(linphone_core_get_current_call(pauline->lc)); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallOutgoingProgress, 1, int, "%d"); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, state == LinphoneCallOutgoingEarlyMedia ? &marie->stat.number_of_LinphoneCallOutgoingEarlyMedia : &marie->stat.number_of_LinphoneCallOutgoingRinging, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, (int *)&linphone_core_get_tone_manager_stats(marie->lc)->number_of_startRingbackTone, 1)); } else if (state == LinphoneCallOutgoingProgress) { BC_ASSERT_PTR_NOT_NULL(linphone_core_invite_address(marie->lc, pauline->identity)); } else {
211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
ms_error("Unsupported state"); return; } BC_ASSERT_TRUE(call_with_caller_params(laure, marie, laure_params)); if (state == LinphoneCallOutgoingRinging) { BC_ASSERT_EQUAL(linphone_core_get_tone_manager_stats(marie->lc)->number_of_startRingtone, 0, int, "%d"); } else if (state == LinphoneCallOutgoingProgress || state == LinphoneCallOutgoingEarlyMedia) { BC_ASSERT_EQUAL(linphone_core_get_tone_manager_stats(marie->lc)->number_of_startRingtone, 1, int, "%d"); } BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallEnd, 1, 10000)); linphone_core_terminate_all_calls(marie->lc); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallEnd, 2, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallReleased, 2, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallReleased, 1, 10000)); linphone_call_params_unref(laure_params); linphone_call_params_unref(marie_params); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(laure); bctbx_list_free(lcs); } static void incoming_call_accepted_when_outgoing_call_in_progress(void) { incoming_call_accepted_when_outgoing_call_in_state(LinphoneCallOutgoingProgress); } static void incoming_call_accepted_when_outgoing_call_in_outgoing_ringing(void) { incoming_call_accepted_when_outgoing_call_in_state(LinphoneCallOutgoingRinging); } static void incoming_call_accepted_when_outgoing_call_in_outgoing_ringing_early_media(void) { incoming_call_accepted_when_outgoing_call_in_state(LinphoneCallOutgoingEarlyMedia); } static void _simple_call_transfer(bool_t transferee_is_default_account) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_dual_proxy_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); LinphoneCoreManager *laure = linphone_core_manager_new(get_laure_rc()); LinphoneCall *pauline_called_by_marie; LinphoneCall *marie_calling_pauline; LinphoneCall *marie_calling_laure; char *laure_identity = linphone_address_as_string(laure->identity); bctbx_list_t *lcs = bctbx_list_append(NULL, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); lcs = bctbx_list_append(lcs, laure->lc); char *marie_identity = linphone_address_as_string(marie->identity); // Marie calls Pauline BC_ASSERT_TRUE(call(marie, pauline)); marie_calling_pauline = linphone_core_get_current_call(marie->lc); pauline_called_by_marie = linphone_core_get_current_call(pauline->lc); if (!BC_ASSERT_PTR_NOT_NULL(marie_calling_pauline)) goto end; if (!BC_ASSERT_PTR_NOT_NULL(pauline_called_by_marie)) goto end; reset_counters(&marie->stat); reset_counters(&pauline->stat); reset_counters(&laure->stat); if (!transferee_is_default_account) { // Set Marie's second account (marie2) as default const bctbx_list_t *accounts = linphone_core_get_account_list(marie->lc);
281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
BC_ASSERT_EQUAL(bctbx_list_size(accounts), 2, size_t, "%zu"); bctbx_list_t *it = (bctbx_list_t *)accounts; it = bctbx_list_next(it); LinphoneAccount *secondAccount = (LinphoneAccount *)it->data; linphone_core_set_default_account(marie->lc, secondAccount); } // Pauline transfers call from Marie to Laure linphone_call_transfer(pauline_called_by_marie, laure_identity); bctbx_free(laure_identity); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallRefered, 1, 2000)); // marie pausing pauline BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPausing, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallPausedByRemote, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPaused, 1, 2000)); // marie calling laure BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallOutgoingProgress, 1, 2000)); BC_ASSERT_PTR_NOT_NULL(linphone_call_get_transfer_target_call(marie_calling_pauline)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneTransferCallOutgoingInit, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallIncomingReceived, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallOutgoingRinging, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneTransferCallOutgoingProgress, 1, 2000)); LinphoneCall *laure_call = linphone_core_get_current_call(laure->lc); if (!BC_ASSERT_PTR_NOT_NULL(laure_call)) goto end; linphone_call_accept(laure_call); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallConnected, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallStreamsRunning, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallConnected, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, 1, 2000)); marie_calling_laure = linphone_core_get_current_call(marie->lc); if (!BC_ASSERT_PTR_NOT_NULL(marie_calling_laure)) goto end; BC_ASSERT_PTR_EQUAL(linphone_call_get_transferer_call(marie_calling_laure), marie_calling_pauline); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneTransferCallConnected, 1, 2000)); char *remote_address_str = linphone_call_get_remote_address_as_string(laure_call); BC_ASSERT_STRING_EQUAL(remote_address_str, marie_identity); ms_free(remote_address_str); // terminate marie to pauline call BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallReleased, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallReleased, 1, 2000)); end_call(marie, laure); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallReleased, 1, 2000)); end: bctbx_free(marie_identity); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(laure); bctbx_list_free(lcs); } static void simple_call_transfer(void) { _simple_call_transfer(TRUE); } static void simple_call_transfer_from_non_default_account(void) { _simple_call_transfer(FALSE); } static void unattended_call_transfer(void) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc");
351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420
LinphoneCoreManager *laure = linphone_core_manager_new(get_laure_rc()); LinphoneCall *pauline_called_by_marie; char *laure_identity = linphone_address_as_string(laure->identity); bctbx_list_t *lcs = bctbx_list_append(NULL, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); lcs = bctbx_list_append(lcs, laure->lc); BC_ASSERT_TRUE(call(marie, pauline)); pauline_called_by_marie = linphone_core_get_current_call(marie->lc); if (!BC_ASSERT_PTR_NOT_NULL(pauline_called_by_marie)) goto end; reset_counters(&marie->stat); reset_counters(&pauline->stat); reset_counters(&laure->stat); linphone_call_transfer(pauline_called_by_marie, laure_identity); bctbx_free(laure_identity); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallRefered, 1, 3000)); // marie ends the call linphone_call_terminate(pauline_called_by_marie); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallEnd, 1, 3000)); // Pauline starts the transfer BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallOutgoingInit, 1, 3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallOutgoingProgress, 1, 3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallIncomingReceived, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallOutgoingRinging, 1, 3000)); if (!BC_ASSERT_PTR_NOT_NULL(linphone_core_get_current_call(laure->lc))) goto end; linphone_call_accept(linphone_core_get_current_call(laure->lc)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallConnected, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallStreamsRunning, 1, 3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallConnected, 1, 3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1, 3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallConnected, 1, 3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallReleased, 1, 3000)); end_call(laure, pauline); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(laure); bctbx_list_free(lcs); } static void unattended_call_transfer_with_error(void) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); LinphoneCall *pauline_called_by_marie; bool_t call_ok = TRUE; bctbx_list_t *lcs = bctbx_list_append(NULL, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); BC_ASSERT_TRUE((call_ok = call(marie, pauline))); if (call_ok) { pauline_called_by_marie = linphone_core_get_current_call(marie->lc); if (!BC_ASSERT_PTR_NOT_NULL(pauline_called_by_marie)) goto end; reset_counters(&marie->stat); reset_counters(&pauline->stat); linphone_call_transfer(pauline_called_by_marie, "unknown_user"); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallRefered, 1, 2000)); // Pauline starts the transfer BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallOutgoingInit, 1, 2000));
421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490
// and immediately get an error BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallError, 1, 2000)); // the error must be reported back to marie BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneTransferCallError, 1, 2000)); // and pauline should resume the call automatically BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallResuming, 1, 2000)); // and call should be resumed BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, 1, 2000)); end_call(marie, pauline); } end: linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(marie); bctbx_list_free(lcs); } static void call_transfer_existing_call(bool_t outgoing_call, bool_t auto_answer_replacing_calls) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); LinphoneCoreManager *laure = linphone_core_manager_new(get_laure_rc()); LinphoneCall *marie_call_pauline; LinphoneCall *pauline_called_by_marie; LinphoneCall *marie_call_laure; LinphoneCall *laure_called_by_marie; LinphoneCall *lcall; bool_t call_ok = TRUE; const bctbx_list_t *calls; bctbx_list_t *lcs = bctbx_list_append(NULL, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); lcs = bctbx_list_append(lcs, laure->lc); if (!auto_answer_replacing_calls) linphone_config_set_int(linphone_core_get_config(laure->lc), "sip", "auto_answer_replacing_calls", 0); // marie call pauline BC_ASSERT_TRUE((call_ok = call(marie, pauline))); if (call_ok) { marie_call_pauline = linphone_core_get_current_call(marie->lc); pauline_called_by_marie = linphone_core_get_current_call(pauline->lc); // marie pause pauline if (!BC_ASSERT_TRUE(pause_call_1(marie, marie_call_pauline, pauline, pauline_called_by_marie))) { goto end; } if (outgoing_call) { // marie call laure if (!BC_ASSERT_TRUE(call(marie, laure))) { end_call(marie, pauline); goto end; } } else { // laure call pauline if (!BC_ASSERT_TRUE(call(laure, marie))) { end_call(marie, pauline); goto end; } } marie_call_laure = linphone_core_get_current_call(marie->lc); laure_called_by_marie = linphone_core_get_current_call(laure->lc); if (!BC_ASSERT_PTR_NOT_NULL(marie_call_laure)) goto end; if (!BC_ASSERT_PTR_NOT_NULL(laure_called_by_marie)) goto end; // marie pause laure BC_ASSERT_TRUE(pause_call_1(marie, marie_call_laure, laure, laure_called_by_marie));
491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560
reset_counters(&marie->stat); reset_counters(&pauline->stat); reset_counters(&laure->stat); linphone_call_transfer_to_another(marie_call_pauline, marie_call_laure); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallRefered, 1, 2000)); // pauline pausing marie BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallPausing, 1, 4000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallPaused, 1, 4000)); // pauline calling laure BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallOutgoingProgress, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneTransferCallOutgoingInit, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallIncomingReceived, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallOutgoingRinging, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneTransferCallOutgoingProgress, 1, 2000)); // laure accept call if (!auto_answer_replacing_calls) { for (calls = linphone_core_get_calls(laure->lc); calls != NULL; calls = calls->next) { lcall = (LinphoneCall *)calls->data; if (linphone_call_get_state(lcall) == LinphoneCallIncomingReceived) { BC_ASSERT_PTR_EQUAL(linphone_call_get_replaced_call(lcall), laure_called_by_marie); linphone_call_accept(lcall); break; } } } BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallConnected, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallStreamsRunning, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallConnected, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneTransferCallConnected, 1, 2000)); // terminate marie to pauline/laure call BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallEnd, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallEnd, 2, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallReleased, 2, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallEnd, 1, 2000)); end_call(pauline, laure); } end: linphone_core_manager_destroy(laure); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(marie); bctbx_list_free(lcs); } static void call_transfer_existing_call_outgoing_call(void) { call_transfer_existing_call(TRUE, TRUE); } static void call_transfer_existing_call_outgoing_call_no_auto_answer(void) { call_transfer_existing_call(TRUE, FALSE); } static void call_transfer_existing_call_incoming_call(void) { call_transfer_existing_call(FALSE, TRUE); } static void call_transfer_existing_ringing_call(void) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); LinphoneCoreManager *laure = linphone_core_manager_new(get_laure_rc()); LinphoneCall *marie_call_pauline; LinphoneCall *pauline_called_by_marie; LinphoneCall *marie_call_laure; LinphoneCall *lcall; LinphoneCall *laure_call;
561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630
bool_t call_ok = TRUE; const bctbx_list_t *calls; stats initial_marie_stats; bctbx_list_t *lcs = bctbx_list_append(NULL, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); lcs = bctbx_list_append(lcs, laure->lc); // marie calls pauline BC_ASSERT_TRUE((call_ok = call(marie, pauline))); if (call_ok) { marie_call_pauline = linphone_core_get_current_call(marie->lc); pauline_called_by_marie = linphone_core_get_current_call(pauline->lc); // marie pauses pauline if (!BC_ASSERT_TRUE(pause_call_1(marie, marie_call_pauline, pauline, pauline_called_by_marie))) goto end; initial_marie_stats = marie->stat; BC_ASSERT_PTR_NOT_NULL((marie_call_laure = linphone_core_invite_address(marie->lc, laure->identity))); if (!marie_call_laure) goto end; BC_ASSERT_TRUE(wait_for(marie->lc, laure->lc, &marie->stat.number_of_LinphoneCallOutgoingRinging, initial_marie_stats.number_of_LinphoneCallOutgoingRinging + 1)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallIncomingReceived, 1, 10000)); laure_call = linphone_core_get_current_call(laure->lc); if (!BC_ASSERT_PTR_NOT_NULL(laure_call)) goto end; linphone_call_transfer_to_another(marie_call_pauline, marie_call_laure); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallRefered, 1, 10000)); // pauline pausing marie BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallPausing, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallPaused, 1, 10000)); // pauline calling laure BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallOutgoingProgress, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneTransferCallOutgoingInit, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallOutgoingRinging, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneTransferCallOutgoingProgress, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallIncomingReceived, 2, 10000)); // laure accepts the new (replacing) call for (calls = linphone_core_get_calls(laure->lc); calls != NULL; calls = calls->next) { lcall = (LinphoneCall *)calls->data; if (lcall == laure_call) continue; if (linphone_call_get_state(lcall) == LinphoneCallIncomingReceived) { linphone_call_accept(lcall); break; } } BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallConnected, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallStreamsRunning, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallConnected, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneTransferCallConnected, 1, 10000)); // terminate marie to pauline/laure call BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallEnd, 2, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallEnd, 1, 10000)); end_call(pauline, laure); } end: linphone_core_manager_destroy(laure); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(marie); bctbx_list_free(lcs); } void do_not_stop_ringing_when_declining_one_of_two_incoming_calls(void) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc");
631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700
LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); LinphoneCoreManager *laure = linphone_core_manager_new(get_laure_rc()); LinphoneCall *pauline_called_by_marie; LinphoneCall *pauline_called_by_laure; LinphoneCallParams *laure_params = linphone_core_create_call_params(laure->lc, NULL); LinphoneCallParams *marie_params = linphone_core_create_call_params(marie->lc, NULL); bctbx_list_t *core_list = NULL; // Do not allow Pauline to use files as the goal of the test is to test audio routes on ring stream linphone_core_set_use_files(pauline->lc, FALSE); core_list = bctbx_list_append(core_list, marie->lc); core_list = bctbx_list_append(core_list, pauline->lc); core_list = bctbx_list_append(core_list, laure->lc); // Laure call Pauline BC_ASSERT_PTR_NOT_NULL(linphone_core_invite_address_with_params(laure->lc, pauline->identity, laure_params)); linphone_call_params_unref(laure_params); BC_ASSERT_TRUE(wait_for_list(core_list, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1, 10000)); pauline_called_by_laure = linphone_core_get_current_call(pauline->lc); const LinphoneCoreToneManagerStats *paulineToneManagerStats = linphone_core_get_tone_manager_stats(pauline->lc); BC_ASSERT_EQUAL(paulineToneManagerStats->number_of_startRingtone, 1, int, "%d"); // Marie call Pauline BC_ASSERT_PTR_NOT_NULL(linphone_core_invite_address_with_params(marie->lc, pauline->identity, marie_params)); linphone_call_params_unref(marie_params); BC_ASSERT_TRUE(wait_for_list(core_list, &pauline->stat.number_of_LinphoneCallIncomingReceived, 2, 10000)); pauline_called_by_marie = linphone_core_get_current_call(marie->lc); BC_ASSERT_EQUAL(paulineToneManagerStats->number_of_startRingtone, 1, int, "%d"); BC_ASSERT_TRUE(linphone_ringtoneplayer_is_started(linphone_core_get_ringtoneplayer(pauline->lc))); // Pauline decline Laure linphone_call_decline(pauline_called_by_laure, LinphoneReasonDeclined); BC_ASSERT_TRUE(wait_for_list(core_list, &pauline->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &pauline->stat.number_of_LinphoneCallReleased, 1, 10000)); // check that ringtone player restart BC_ASSERT_TRUE(wait_for_list(core_list, (int *)&paulineToneManagerStats->number_of_stopRingtone, 1, 2000)); BC_ASSERT_EQUAL(paulineToneManagerStats->number_of_stopRingtone, 1, int, "%d"); BC_ASSERT_TRUE(wait_for_list(core_list, (int *)&paulineToneManagerStats->number_of_startRingtone, 2, 2000)); BC_ASSERT_TRUE(linphone_ringtoneplayer_is_started(linphone_core_get_ringtoneplayer(pauline->lc))); linphone_call_terminate(pauline_called_by_marie); BC_ASSERT_TRUE(wait_for_list(core_list, &marie->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &marie->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &laure->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &laure->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &pauline->stat.number_of_LinphoneCallEnd, 2, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &pauline->stat.number_of_LinphoneCallReleased, 2, 10000)); BC_ASSERT_EQUAL(paulineToneManagerStats->number_of_stopRingtone, 2, int, "%d"); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(laure); bctbx_list_free(core_list); } void stop_ringing_when_accepting_call_while_holding_another(bool_t activate_ice) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); LinphoneCoreManager *laure = linphone_core_manager_new(get_laure_rc()); LinphoneCall *pauline_called_by_marie; LinphoneCall *marie_called_by_laure; LinphoneCall *laure_call = NULL; LinphoneCallParams *marie_params = linphone_core_create_call_params(marie->lc, NULL); LinphoneCallParams *laure_params = linphone_core_create_call_params(laure->lc, NULL); bctbx_list_t *core_list = NULL; // Do not allow Pauline to use files as the goal of the test is to test audio routes on ring stream linphone_core_set_use_files(marie->lc, FALSE);
701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770
core_list = bctbx_list_append(core_list, marie->lc); core_list = bctbx_list_append(core_list, pauline->lc); core_list = bctbx_list_append(core_list, laure->lc); // Enable ICE enable_stun_in_mgr(marie, TRUE, activate_ice, TRUE, TRUE); enable_stun_in_mgr(pauline, TRUE, activate_ice, TRUE, TRUE); enable_stun_in_mgr(laure, TRUE, activate_ice, TRUE, TRUE); // Marie calls Pauline BC_ASSERT_PTR_NOT_NULL(linphone_core_invite_address_with_params(marie->lc, pauline->identity, marie_params)); linphone_call_params_unref(marie_params); BC_ASSERT_TRUE(wait_for_list(core_list, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &marie->stat.number_of_LinphoneCallOutgoingRinging, 1, 10000)); pauline_called_by_marie = linphone_core_get_current_call(pauline->lc); BC_ASSERT_EQUAL(linphone_core_get_tone_manager_stats(pauline->lc)->number_of_startRingtone, 1, int, "%d"); BC_ASSERT_EQUAL(linphone_core_get_tone_manager_stats(marie->lc)->number_of_startRingbackTone, 1, int, "%d"); // Laure calls Marie laure_call = linphone_core_invite_address_with_params(laure->lc, marie->identity, laure_params); BC_ASSERT_PTR_NOT_NULL(laure_call); LinphoneCallLog *laure_call_log = linphone_call_get_call_log(laure_call); linphone_call_params_unref(laure_params); BC_ASSERT_TRUE(wait_for_list(core_list, &marie->stat.number_of_LinphoneCallIncomingReceived, 1, 10000)); const char *laure_call_id = linphone_call_log_get_call_id(laure_call_log); marie_called_by_laure = linphone_core_get_call_by_callid(marie->lc, laure_call_id); // marie_called_by_laure=linphone_core_get_current_call(marie->lc); BC_ASSERT_TRUE(wait_for_list(core_list, &marie->stat.number_of_LinphoneCallIncomingReceived, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &laure->stat.number_of_LinphoneCallOutgoingRinging, 1, 10000)); const LinphoneCoreToneManagerStats *marieToneStats = linphone_core_get_tone_manager_stats(marie->lc); wait_for_list(core_list, 0, 1, 2000); // Let some time ringing // Pauline accept Marie linphone_call_accept(pauline_called_by_marie); BC_ASSERT_TRUE(wait_for_list(core_list, (int *)&marieToneStats->number_of_stopRingbackTone, 1, 5000)); BC_ASSERT_TRUE(wait_for_list(core_list, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &marie->stat.number_of_LinphoneCallStreamsRunning, 1, 10000)); /* the call from laure should ring as a waiting tone */ BC_ASSERT_TRUE(wait_for_list(core_list, (int *)&marieToneStats->number_of_startNamedTone, 1, 5000)); // Marie accept Laure linphone_call_accept(marie_called_by_laure); BC_ASSERT_TRUE(wait_for_list(core_list, (int *)&marieToneStats->number_of_stopTone, 1, 5000)); // End of named tone linphone_call_terminate(marie_called_by_laure); linphone_call_terminate(pauline_called_by_marie); BC_ASSERT_TRUE(wait_for_list(core_list, &marie->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &marie->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &laure->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &laure->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &pauline->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(core_list, &pauline->stat.number_of_LinphoneCallReleased, 1, 10000)); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(laure); bctbx_list_free(core_list); } void stop_ringing_when_accepting_call_while_holding_another_with_ice(void) { stop_ringing_when_accepting_call_while_holding_another(TRUE);
771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840
} void stop_ringing_when_accepting_call_while_holding_another_without_ice(void) { stop_ringing_when_accepting_call_while_holding_another(FALSE); } void no_auto_answer_on_fake_call_with_replaces_header(void) { LinphoneCoreManager *marie1 = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *marie2 = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); bctbx_list_t *lcs = bctbx_list_append(NULL, marie1->lc); lcs = bctbx_list_append(lcs, marie2->lc); lcs = bctbx_list_append(lcs, pauline->lc); LinphoneCallParams *params = linphone_core_create_call_params(marie1->lc, NULL); linphone_call_params_set_privacy(params, LinphonePrivacyId); LinphoneCall *call1 = linphone_core_invite_address_with_params(marie1->lc, pauline->identity, params); linphone_call_params_unref(params); BC_ASSERT_PTR_NOT_NULL(call1); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1, 3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie1->stat.number_of_LinphoneCallOutgoingRinging, 1, 3000)); reset_counters(&marie1->stat); reset_counters(&marie2->stat); reset_counters(&pauline->stat); const char *callId = linphone_call_log_get_call_id(linphone_call_get_call_log(call1)); BC_ASSERT_PTR_NOT_NULL(callId); SalOp *op = linphone_call_get_op_as_sal_op(call1); const char *fromTag = sal_call_get_local_tag(op); const char *toTag = sal_call_get_remote_tag(op); BC_ASSERT_PTR_NOT_NULL(fromTag); BC_ASSERT_PTR_NOT_NULL(toTag); params = linphone_core_create_call_params(marie2->lc, NULL); char *headerValue = bctbx_strdup_printf("%s;from-tag=%s;to-tag=%s", callId, fromTag, toTag); linphone_call_params_add_custom_header(params, "Replaces", headerValue); bctbx_free(headerValue); LinphoneCall *call2 = linphone_core_invite_address_with_params(marie2->lc, pauline->identity, params); linphone_call_params_unref(params); BC_ASSERT_PTR_NOT_NULL(call2); BC_ASSERT_EQUAL(marie2->stat.number_of_LinphoneCallOutgoingProgress, 1, int, "%d"); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1, 3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie2->stat.number_of_LinphoneCallOutgoingRinging, 1, 3000)); BC_ASSERT_FALSE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallConnected, 1, 2000)); linphone_call_terminate(call1); linphone_call_terminate(call2); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallReleased, 1, 3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie1->stat.number_of_LinphoneCallReleased, 1, 3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie2->stat.number_of_LinphoneCallReleased, 1, 3000)); linphone_core_manager_destroy(marie1); linphone_core_manager_destroy(marie2); linphone_core_manager_destroy(pauline); bctbx_list_free(lcs); } static void call_accepted_while_another_one_is_updating(bool_t update_from_callee) { int call_ring_timeout = 10000000; // Core that is called by other cores // ICE is enabled LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); linphone_core_set_inc_timeout(marie->lc, call_ring_timeout); bctbx_list_t *participants = NULL; LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc");
841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910
linphone_core_set_inc_timeout(pauline->lc, call_ring_timeout); participants = bctbx_list_append(participants, pauline); LinphoneCoreManager *chloe = linphone_core_manager_new("chloe_rc"); linphone_core_set_inc_timeout(chloe->lc, call_ring_timeout); participants = bctbx_list_append(participants, chloe); LinphoneCoreManager *laure = linphone_core_manager_new(get_laure_rc()); linphone_core_set_inc_timeout(laure->lc, call_ring_timeout); participants = bctbx_list_append(participants, laure); LinphoneVideoActivationPolicy *pol = linphone_factory_create_video_activation_policy(linphone_factory_get()); linphone_video_activation_policy_set_automatically_accept(pol, TRUE); linphone_core_set_video_activation_policy(pauline->lc, pol); linphone_core_set_video_activation_policy(marie->lc, pol); linphone_core_set_video_activation_policy(laure->lc, pol); linphone_core_set_video_activation_policy(chloe->lc, pol); linphone_video_activation_policy_unref(pol); linphone_core_set_video_device(marie->lc, liblinphone_tester_mire_id); linphone_core_enable_video_capture(marie->lc, TRUE); linphone_core_enable_video_display(marie->lc, TRUE); for (bctbx_list_t *it = participants; it; it = bctbx_list_next(it)) { LinphoneCoreManager *m = (LinphoneCoreManager *)bctbx_list_get_data(it); LinphoneCore *c = m->lc; linphone_core_set_video_device(c, liblinphone_tester_mire_id); linphone_core_enable_video_capture(c, TRUE); linphone_core_enable_video_display(c, TRUE); } bctbx_list_t *lcs = NULL; lcs = bctbx_list_append(lcs, marie->lc); initiate_calls(participants, marie); unsigned int no_callers = (unsigned int)bctbx_list_size(participants); unsigned int idx = 0; for (bctbx_list_t *it = participants; it; it = bctbx_list_next(it)) { LinphoneCoreManager *m = (LinphoneCoreManager *)bctbx_list_get_data(it); LinphoneCore *c = m->lc; lcs = bctbx_list_append(lcs, c); LinphoneCall *marie_call = linphone_core_get_call_by_remote_address2(marie->lc, m->identity); BC_ASSERT_PTR_NOT_NULL(marie_call); // Take call - ringing ends linphone_call_accept(marie_call); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallConnected, (idx + 1), 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, (idx + 1), 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &m->stat.number_of_LinphoneCallConnected, 1, 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &m->stat.number_of_LinphoneCallStreamsRunning, 1, 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPausing, idx, 5000)); // Send update only on the first call taken if (it == participants) { LinphoneCall *call_to_update = NULL; if (update_from_callee) { call_to_update = linphone_core_get_call_by_remote_address2(marie->lc, m->identity); } else { call_to_update = linphone_core_get_current_call(c); } BC_ASSERT_PTR_NOT_NULL(call_to_update); if (call_to_update) { LinphoneCallParams *new_params = linphone_core_create_call_params(c, call_to_update);
911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980
linphone_call_params_enable_video(new_params, TRUE); linphone_call_update(call_to_update, new_params); linphone_call_params_unref(new_params); } } idx++; } BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallConnected, no_callers, 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, no_callers, 5000)); LinphoneCoreManager *phead = (LinphoneCoreManager *)bctbx_list_get_data(participants); if (update_from_callee) { BC_ASSERT_TRUE(wait_for_list(lcs, &phead->stat.number_of_LinphoneCallUpdatedByRemote, 1, 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallUpdating, 1, 5000)); } else { BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallUpdatedByRemote, 1, 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &phead->stat.number_of_LinphoneCallUpdating, 1, 5000)); } BC_ASSERT_TRUE(wait_for_list(lcs, &phead->stat.number_of_LinphoneCallStreamsRunning, 2, 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &phead->stat.number_of_LinphoneCallPausedByRemote, 1, 5000)); // Only one call is not paused unsigned int no_call_paused = no_callers - 1; BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPausing, no_call_paused, 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPaused, no_call_paused, 5000)); LinphoneCall *marie_call = linphone_core_get_call_by_remote_address2(marie->lc, phead->identity); BC_ASSERT_PTR_NOT_NULL(marie_call); const LinphoneCallParams *marie_params = linphone_call_get_params(marie_call); BC_ASSERT_TRUE(linphone_call_params_video_enabled(marie_params)); LinphoneCall *phead_call = linphone_core_get_current_call(phead->lc); BC_ASSERT_PTR_NOT_NULL(phead_call); const LinphoneCallParams *phead_params = linphone_call_get_params(phead_call); BC_ASSERT_TRUE(linphone_call_params_video_enabled(phead_params)); LinphoneCall *pcall = NULL; unsigned int no_paused_by_remote = 0; LinphoneCoreManager *pm = NULL; for (bctbx_list_t *it = participants; it; it = bctbx_list_next(it)) { LinphoneCoreManager *m = (LinphoneCoreManager *)bctbx_list_get_data(it); LinphoneCall *call = linphone_core_get_current_call(m->lc); BC_ASSERT_PTR_NOT_NULL(call); if (call) { no_paused_by_remote += (linphone_call_get_state(call) == LinphoneCallPausedByRemote) ? 1 : 0; if (linphone_call_get_state(call) == LinphoneCallStreamsRunning) { pcall = call; pm = m; } } } BC_ASSERT_PTR_NOT_NULL(pcall); if (pcall) { char *p_remote_address = linphone_call_get_remote_address_as_string(pcall); char *marie_identity = linphone_address_as_string(marie->identity); BC_ASSERT_EQUAL(strcmp(p_remote_address, marie_identity), 0, int, "%d"); ms_free(p_remote_address); ms_free(marie_identity); } BC_ASSERT_EQUAL(no_paused_by_remote, no_call_paused, int, "%d"); LinphoneCall *hcall = NULL; int no_active_calls_stream_running = 0; bctbx_list_t *calls = bctbx_list_copy(linphone_core_get_calls(marie->lc)); for (bctbx_list_t *it = calls; it; it = bctbx_list_next(it)) { LinphoneCall *call = (LinphoneCall *)bctbx_list_get_data(it); BC_ASSERT_PTR_NOT_NULL(call);
981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050
if (call) { if (linphone_call_get_state(call) == LinphoneCallStreamsRunning) { no_active_calls_stream_running += 1; hcall = call; } } } bctbx_list_free(calls); BC_ASSERT_PTR_NOT_NULL(hcall); BC_ASSERT_PTR_NOT_NULL(pm); if (hcall && pm) { char *h_remote_address = linphone_call_get_remote_address_as_string(hcall); char *pm_address = linphone_address_as_string(pm->identity); BC_ASSERT_EQUAL(strcmp(h_remote_address, pm_address), 0, int, "%d"); ms_free(h_remote_address); ms_free(pm_address); } BC_ASSERT_EQUAL(no_active_calls_stream_running, 1, int, "%d"); linphone_core_terminate_all_calls(marie->lc); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &chloe->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallEnd, no_callers, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &chloe->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallReleased, no_callers, 10000)); bctbx_list_free(participants); bctbx_list_free(lcs); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(laure); linphone_core_manager_destroy(chloe); } static void call_accepted_while_callee_is_updating_another_one(void) { call_accepted_while_another_one_is_updating(TRUE); } #if 0 static void call_accepted_while_caller_is_updating_to_same_callee(void) { call_accepted_while_another_one_is_updating(FALSE); } static void call_with_ice_negotiations_ending_while_accepting_call_base(bool_t back_to_back_accept) { LinphoneMediaEncryption mode = LinphoneMediaEncryptionNone; // Core that is called by other cores // ICE is enabled LinphoneCoreManager* marie = linphone_core_manager_new( "marie_rc"); linphone_core_set_inc_timeout(marie->lc, 10000); if (linphone_core_media_encryption_supported(marie->lc,mode)) { linphone_core_set_firewall_policy(marie->lc,LinphonePolicyUseIce); linphone_core_set_media_encryption(marie->lc,mode); } bctbx_list_t* ice_participants=NULL; // ICE is enabled LinphoneCoreManager* pauline = linphone_core_manager_new( "pauline_tcp_rc"); linphone_core_set_inc_timeout(pauline->lc, 10000); if (linphone_core_media_encryption_supported(pauline->lc,mode)) { linphone_core_set_firewall_policy(pauline->lc,LinphonePolicyUseIce); linphone_core_set_media_encryption(pauline->lc,mode); } ice_participants=bctbx_list_append(ice_participants,pauline);
1051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120
// ICE is enabled LinphoneCoreManager* chloe = linphone_core_manager_new( "chloe_rc"); linphone_core_set_inc_timeout(chloe->lc, 10000); if (linphone_core_media_encryption_supported(chloe->lc,mode)) { linphone_core_set_firewall_policy(chloe->lc,LinphonePolicyUseIce); linphone_core_set_media_encryption(chloe->lc,mode); } ice_participants=bctbx_list_append(ice_participants,chloe); bctbx_list_t* non_ice_participants=NULL; // ICE is disabled LinphoneCoreManager* laure = linphone_core_manager_new( liblinphone_tester_ipv6_available() ? "laure_tcp_rc" : "laure_rc_udp"); linphone_core_set_inc_timeout(laure->lc, 10000); non_ice_participants=bctbx_list_append(non_ice_participants,laure); // ICE is disabled LinphoneCoreManager* michelle = linphone_core_manager_new( "michelle_rc_udp"); linphone_core_set_inc_timeout(michelle->lc, 10000); non_ice_participants=bctbx_list_append(non_ice_participants,michelle); bctbx_list_t* lcs = NULL; lcs=bctbx_list_append(lcs,marie->lc); initiate_calls(non_ice_participants, marie); initiate_calls(ice_participants, marie); unsigned int no_ice_callers = (unsigned int)bctbx_list_size(ice_participants); unsigned int no_non_ice_callers = (unsigned int)bctbx_list_size(non_ice_participants); unsigned int no_calls = no_ice_callers + no_non_ice_callers; bctbx_list_t* participants = NULL; participants=bctbx_list_append(participants,laure); participants=bctbx_list_append(participants,pauline); participants=bctbx_list_append(participants,michelle); bool_t callee_uses_ice = (linphone_core_get_firewall_policy(marie->lc) == LinphonePolicyUseIce); unsigned int no_calls_paused_by_remote = 0; LinphoneCoreManager * pm = NULL; if (back_to_back_accept) { lcs=bctbx_list_append(lcs,chloe->lc); const LinphoneAddress *chloe_uri = chloe->identity; LinphoneCall * marie_call = linphone_core_get_call_by_remote_address2(marie->lc, chloe_uri); BC_ASSERT_PTR_NOT_NULL(marie_call); // Take call - ringing ends linphone_call_accept(marie_call); BC_ASSERT_TRUE(wait_for_until(chloe->lc,marie->lc, &chloe->stat.number_of_LinphoneCallConnected, 1, 5000)); BC_ASSERT_TRUE(wait_for_until(chloe->lc,marie->lc, &chloe->stat.number_of_LinphoneCallStreamsRunning, 1, 5000)); if (chloe && callee_uses_ice && linphone_core_get_firewall_policy(chloe->lc) == LinphonePolicyUseIce) { no_calls_paused_by_remote++; BC_ASSERT_TRUE(wait_for_until(chloe->lc,marie->lc, &chloe->stat.number_of_LinphoneCallUpdating,1,5000)); } pm = chloe; for (bctbx_list_t *it = participants; it; it = bctbx_list_next(it)) { LinphoneCoreManager * m = (LinphoneCoreManager *)bctbx_list_get_data(it); const LinphoneAddress *caller_uri = m->identity; LinphoneCall * marie_call = linphone_core_get_call_by_remote_address2(marie->lc, caller_uri); BC_ASSERT_PTR_NOT_NULL(marie_call); // Take call - ringing ends linphone_call_accept(marie_call); } } else { participants=bctbx_list_prepend(participants,chloe); }
1121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190
for (bctbx_list_t *it = participants; it; it = bctbx_list_next(it)) { LinphoneCoreManager * m = (LinphoneCoreManager *)bctbx_list_get_data(it); LinphoneCore * c = m->lc; lcs=bctbx_list_append(lcs,c); if (!back_to_back_accept) { const LinphoneAddress *caller_uri = m->identity; LinphoneCall * marie_call = linphone_core_get_call_by_remote_address2(marie->lc, caller_uri); BC_ASSERT_PTR_NOT_NULL(marie_call); // Take call - ringing ends linphone_call_accept(marie_call); } if (pm) { BC_ASSERT_TRUE(wait_for_until(pm->lc,marie->lc, &marie->stat.number_of_LinphoneCallPaused, no_calls_paused_by_remote, 5000)); BC_ASSERT_TRUE(wait_for_until(pm->lc,marie->lc, &pm->stat.number_of_LinphoneCallPausedByRemote,1,5000)); } BC_ASSERT_TRUE(wait_for_until(m->lc,marie->lc, &m->stat.number_of_LinphoneCallConnected, 1, 5000)); BC_ASSERT_TRUE(wait_for_until(m->lc,marie->lc, &m->stat.number_of_LinphoneCallStreamsRunning, 1, 5000)); if (m && callee_uses_ice && linphone_core_get_firewall_policy(m->lc) == LinphonePolicyUseIce) { no_calls_paused_by_remote++; BC_ASSERT_TRUE(wait_for_until(m->lc,marie->lc, &m->stat.number_of_LinphoneCallUpdating,1,5000)); } pm = m; } if (pm && callee_uses_ice && linphone_core_get_firewall_policy(pm->lc) == LinphonePolicyUseIce) { BC_ASSERT_TRUE(wait_for_until(pm->lc, marie->lc, &marie->stat.number_of_LinphoneCallUpdatedByRemote, no_calls_paused_by_remote, 5000)); } BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallConnected, no_calls, 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, no_calls, 5000)); for (bctbx_list_t *it = ice_participants; it; it = bctbx_list_next(it)) { LinphoneCoreManager * m = (LinphoneCoreManager *)bctbx_list_get_data(it); BC_ASSERT_TRUE(wait_for_list(lcs, &m->stat.number_of_LinphoneCallStreamsRunning, 2, 5000)); BC_ASSERT_TRUE(check_ice(m,marie,LinphoneIceStateHostConnection)); BC_ASSERT_TRUE(check_ice(marie,m,LinphoneIceStateHostConnection)); } // Calls whose core enables ICE send a reINVITE hence they go twice to stream running BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, (2*no_ice_callers) + no_non_ice_callers, 5000)); // Only one call is not paused unsigned int no_call_paused = no_ice_callers + no_non_ice_callers - 1; BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPausing, no_call_paused, 5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPaused, no_call_paused, 5000)); unsigned int no_paused_by_remote = 0; for (bctbx_list_t *it = ice_participants; it; it = bctbx_list_next(it)) { LinphoneCoreManager * m = (LinphoneCoreManager *)bctbx_list_get_data(it); no_paused_by_remote += m->stat.number_of_LinphoneCallPausedByRemote; } for (bctbx_list_t *it = non_ice_participants; it; it = bctbx_list_next(it)) { LinphoneCoreManager * m = (LinphoneCoreManager *)bctbx_list_get_data(it); no_paused_by_remote += m->stat.number_of_LinphoneCallPausedByRemote; } BC_ASSERT_EQUAL(no_paused_by_remote,no_call_paused, int, "%d"); int no_active_calls_stream_running = 0; bctbx_list_t *calls = bctbx_list_copy(linphone_core_get_calls(marie->lc)); for (bctbx_list_t *it = calls; it; it = bctbx_list_next(it)) { LinphoneCall *call = (LinphoneCall *)bctbx_list_get_data(it); no_active_calls_stream_running += (linphone_call_get_state(call) == LinphoneCallStreamsRunning) ? 1 : 0; } bctbx_list_free(calls);
1191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260
BC_ASSERT_EQUAL(no_active_calls_stream_running,1, int, "%d"); bctbx_list_free(participants); bctbx_list_free(non_ice_participants); bctbx_list_free(ice_participants); bctbx_list_free(lcs); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(laure); linphone_core_manager_destroy(michelle); linphone_core_manager_destroy(chloe); } static void call_with_ice_negotiations_ending_while_accepting_call(void) { call_with_ice_negotiations_ending_while_accepting_call_base(FALSE); } static void call_with_ice_negotiations_ending_while_accepting_call_back_to_back(void) { call_with_ice_negotiations_ending_while_accepting_call_base(TRUE); } #endif /* Workflow: * Marie-Pauline-Laure : Accounts * - Pauline call Marie * - Pauline pauses Marie with inactive stream * - Laure call Marie * - Laure pauses Marie * - Pauline resumes * - Marie resumes Pauline * - Pauline pauses Marie with inactive stream * - Pauline resumes * */ void resuming_inactive_stream(void) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); LinphoneCoreManager *laure = linphone_core_manager_new(get_laure_rc()); LinphoneCall *pauline_call_marie, *laure_call_marie; LinphoneCall *marie_called_by_pauline; LinphoneCallParams *pauline_call_marie_params; bctbx_list_t *lcs = bctbx_list_append(NULL, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); lcs = bctbx_list_append(lcs, laure->lc); // Pauline call Marie. Prepare early media to have meta-data debug for checking stream state (inactive/sendrecv) pauline_call_marie_params = linphone_core_create_call_params(marie->lc, NULL); linphone_call_params_enable_early_media_sending(pauline_call_marie_params, TRUE); pauline_call_marie = linphone_core_invite_address_with_params(pauline->lc, marie->identity, pauline_call_marie_params); linphone_call_params_unref(pauline_call_marie_params); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallIncomingReceived, 1, 2000)); linphone_call_accept_early_media(linphone_core_get_current_call(marie->lc)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallOutgoingEarlyMedia, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallIncomingEarlyMedia, 1, 2000)); BC_ASSERT_EQUAL(pauline->stat.number_of_LinphoneCallOutgoingProgress, 1, int, "%d"); if (pauline_call_marie != NULL) { marie_called_by_pauline = linphone_core_get_current_call(marie->lc); pauline_call_marie = linphone_core_get_current_call(pauline->lc); // Marie accept call linphone_call_accept(marie_called_by_pauline); if (!BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, 1, 2000))) goto end; // Marie is in call with Pauline if (!BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1, 2000))) goto end; // Pauline set inactive stream pauline_call_marie_params = linphone_core_create_call_params(pauline->lc, NULL); linphone_call_params_set_audio_direction(pauline_call_marie_params, LinphoneMediaDirectionInactive); linphone_call_update(pauline_call_marie, pauline_call_marie_params);
1261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330
linphone_call_params_unref(pauline_call_marie_params); if (!BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2, 2000))) goto end; // By setting the current stream to inactive, Pauline has been entered in Running state if (!BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &marie->stat.number_of_LinphoneCallPausedByRemote, 1))) goto end; // Marie detect the inactive stream as a Paused from remote // Laure call Marie if (BC_ASSERT_TRUE(call(laure, marie))) { if (!BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPausing, 1, 2000))) goto end; // Marie do pause when accepting the new call if (!BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPaused, 1, 2000))) goto end; // Marie is paused on Pauline inactive call if (!BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallPausedByRemote, 1, 2000))) goto end; // Pauline receive the paused state from Marie laure_call_marie = linphone_core_get_current_call(laure->lc); // Laure pauses Marie linphone_call_pause(laure_call_marie); if (!BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallPaused, 1, 2000))) goto end; // Laure do pauses if (!BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPausedByRemote, 2, 2000))) goto end; // Marie get paused // Pauline set sendrecv stream pauline_call_marie_params = linphone_core_create_call_params(pauline->lc, NULL); linphone_call_params_set_audio_direction(pauline_call_marie_params, LinphoneMediaDirectionSendRecv); linphone_call_update(pauline_call_marie, pauline_call_marie_params); linphone_call_params_unref(pauline_call_marie_params); if (!BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallPausedByRemote, 2, 2000))) goto end; // When reactivating the call by stream, Pauline get the paused state from Marie. That allows // Marie to resume // Marie resumes Pauline linphone_call_resume(marie_called_by_pauline); if (!BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPaused, 2, 2000))) goto end; // Marie pauses Laure call if (!BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallResuming, 1, 2000))) goto end; // Marie resumes if (!BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, 3, 2000))) goto end; // Marie open the stream if (!BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 3, 2000))) goto end; // Pauline run the stream as it is unpaused from Marie and in SendRecv state // Pauline set inactive pauline_call_marie_params = linphone_core_create_call_params(pauline->lc, NULL); linphone_call_params_set_audio_direction(pauline_call_marie_params, LinphoneMediaDirectionInactive); linphone_call_update(pauline_call_marie, pauline_call_marie_params); linphone_call_params_unref(pauline_call_marie_params); if (!BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPausedByRemote, 3, 2000))) goto end; // Marie has been paused from Inactive stream if (!BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 4, 2000))) goto end; // By setting the stream to inactive, Pauline has been entered in Running state // Pauline set sendrecv stream pauline_call_marie_params = linphone_core_create_call_params(pauline->lc, NULL); linphone_call_params_set_audio_direction(pauline_call_marie_params, LinphoneMediaDirectionSendRecv); linphone_call_update(pauline_call_marie, pauline_call_marie_params); linphone_call_params_unref(pauline_call_marie_params); if (!BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, 4, 2000))) goto end; // comes from : Pauline call, Laure Call, 2 resumes if (!BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 5, 2000))) goto end; // comes from : first call, 2 inactives, 2 sendrecv wait_for_list(lcs, NULL, 5, 200); end_call(laure, marie); } end_call(pauline, marie); } end: linphone_core_manager_destroy(laure); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(marie); bctbx_list_free(lcs); } static void second_call_with_early_media(void) {
1331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400
LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new("pauline_tcp_rc"); LinphoneCoreManager *laure = linphone_core_manager_new(get_laure_rc()); bctbx_list_t *lcs; LinphoneCallParams *laure_params = linphone_core_create_call_params(laure->lc, NULL); LinphoneCallParams *marie_params = linphone_core_create_call_params(marie->lc, NULL); lcs = bctbx_list_append(NULL, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); lcs = bctbx_list_append(lcs, laure->lc); /* Marie makes a first call to pauline, who accepts it. */ if (!BC_ASSERT_TRUE(call(marie, pauline))) { goto end; } /* Marie pauses this call and makes a second call to laure, who answers with early-media first */ linphone_call_pause(linphone_core_get_current_call(marie->lc)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallPaused, 1, 10000)); BC_ASSERT_PTR_NOT_NULL(linphone_core_invite_address_with_params(marie->lc, laure->identity, marie_params)); if (BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallIncomingReceived, 1, 10000))) { linphone_call_accept_early_media(linphone_core_get_current_call(laure->lc)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallIncomingEarlyMedia, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallOutgoingEarlyMedia, 1, 10000)); /* The OutgoingEarlyMedia state should stop the "waiting" tone that was started for the first call being paused. */ BC_ASSERT_TRUE( wait_for_list(lcs, (int *)&linphone_core_get_tone_manager_stats(marie->lc)->number_of_stopTone, 1, 5000)); } linphone_core_terminate_all_calls(marie->lc); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallEnd, 2, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallEnd, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallReleased, 1, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallReleased, 2, 10000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallReleased, 1, 10000)); end: linphone_call_params_unref(laure_params); linphone_call_params_unref(marie_params); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(laure); bctbx_list_free(lcs); } test_t multi_call_tests[] = { TEST_NO_TAG("Call waiting indication", call_waiting_indication), TEST_NO_TAG("Call waiting indication with privacy", call_waiting_indication_with_privacy), TEST_NO_TAG("Second call rejected if first one in progress", second_call_rejected_if_first_one_in_progress), TEST_NO_TAG("Second call allowed if not using audio", second_call_allowed_if_not_using_audio), TEST_NO_TAG("Incoming call accepted when outgoing call in progress", incoming_call_accepted_when_outgoing_call_in_progress), TEST_NO_TAG("Incoming call accepted when outgoing call in outgoing ringing", incoming_call_accepted_when_outgoing_call_in_outgoing_ringing), TEST_NO_TAG("Incoming call accepted when outgoing call in outgoing ringing early media", incoming_call_accepted_when_outgoing_call_in_outgoing_ringing_early_media), TEST_NO_TAG("Call accepted while callee is updating another one", call_accepted_while_callee_is_updating_another_one), // TEST_NO_TAG("Call accepted while caller is updating to same callee", // call_accepted_while_caller_is_updating_to_same_callee), TEST_ONE_TAG("Call with ICE negotiations ending while // accepting call", call_with_ice_negotiations_ending_while_accepting_call, "ICE"), TEST_ONE_TAG("Call with ICE // negotiations ending while accepting call back to back", // call_with_ice_negotiations_ending_while_accepting_call_back_to_back, "ICE"), TEST_NO_TAG("Simple call transfer", simple_call_transfer),
140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427
TEST_NO_TAG("Simple call transfer from non default account", simple_call_transfer_from_non_default_account), TEST_NO_TAG("Unattended call transfer", unattended_call_transfer), TEST_NO_TAG("Unattended call transfer with error", unattended_call_transfer_with_error), TEST_NO_TAG("Call transfer existing outgoing call", call_transfer_existing_call_outgoing_call), TEST_NO_TAG("Call transfer existing outgoing call without auto answer of replacing call", call_transfer_existing_call_outgoing_call_no_auto_answer), TEST_NO_TAG("Call transfer existing incoming call", call_transfer_existing_call_incoming_call), TEST_NO_TAG("Call transfer existing ringing call", call_transfer_existing_ringing_call), TEST_NO_TAG("Do not stop ringing when declining one of two incoming calls", do_not_stop_ringing_when_declining_one_of_two_incoming_calls), TEST_NO_TAG("No auto answer on fake call with Replaces header", no_auto_answer_on_fake_call_with_replaces_header), TEST_NO_TAG("Resuming on inactive stream", resuming_inactive_stream), TEST_NO_TAG("Stop ringing when accepting call while holding another with ICE", stop_ringing_when_accepting_call_while_holding_another_with_ice), TEST_NO_TAG("Stop ringing when accepting call while holding another without ICE", stop_ringing_when_accepting_call_while_holding_another_without_ice), TEST_NO_TAG("Stop paused tone when second call is accepted with early media", second_call_with_early_media)}; test_suite_t multi_call_test_suite = {"Multi call", NULL, NULL, liblinphone_tester_before_each, liblinphone_tester_after_each, sizeof(multi_call_tests) / sizeof(multi_call_tests[0]), multi_call_tests, 0};