/* liblinphone_tester - liblinphone test suite Copyright (C) 2013 Belledonne Communications SARL This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ #include #include #include "linphone/core.h" #include "linphone/lpconfig.h" #include "private.h" #include "liblinphone_tester.h" #include "mediastreamer2/msutils.h" #include "belle-sip/sipstack.h" #include #ifdef _WIN32 #define unlink _unlink #ifndef F_OK #define F_OK 00 /*visual studio does not define F_OK*/ #endif #endif static void srtp_call(void); // prototype definition for call_recording() #ifdef __ANDROID__ #ifdef HAVE_OPENH264 extern void libmsopenh264_init(MSFactory *factory); #endif #endif void call_state_changed(LinphoneCore *lc, LinphoneCall *call, LinphoneCallState cstate, const char *msg){ char* to=linphone_address_as_string(linphone_call_get_call_log(call)->to); char* from=linphone_address_as_string(linphone_call_get_call_log(call)->from); stats* counters; ms_message(" %s call from [%s] to [%s], new state is [%s]" ,linphone_call_get_call_log(call)->dir==LinphoneCallIncoming?"Incoming":"Outgoing" ,from ,to ,linphone_call_state_to_string(cstate)); ms_free(to); ms_free(from); counters = get_stats(lc); switch (cstate) { case LinphoneCallIncomingReceived:counters->number_of_LinphoneCallIncomingReceived++;break; case LinphoneCallOutgoingInit :counters->number_of_LinphoneCallOutgoingInit++;break; case LinphoneCallOutgoingProgress :counters->number_of_LinphoneCallOutgoingProgress++;break; case LinphoneCallOutgoingRinging :counters->number_of_LinphoneCallOutgoingRinging++;break; case LinphoneCallOutgoingEarlyMedia :counters->number_of_LinphoneCallOutgoingEarlyMedia++;break; case LinphoneCallConnected :counters->number_of_LinphoneCallConnected++;break; case LinphoneCallStreamsRunning :counters->number_of_LinphoneCallStreamsRunning++;break; case LinphoneCallPausing :counters->number_of_LinphoneCallPausing++;break; case LinphoneCallPaused :counters->number_of_LinphoneCallPaused++;break; case LinphoneCallResuming :counters->number_of_LinphoneCallResuming++;break; case LinphoneCallRefered :counters->number_of_LinphoneCallRefered++;break; case LinphoneCallError :counters->number_of_LinphoneCallError++;break; case LinphoneCallEnd :counters->number_of_LinphoneCallEnd++;break; case LinphoneCallPausedByRemote :counters->number_of_LinphoneCallPausedByRemote++;break; case LinphoneCallUpdatedByRemote :counters->number_of_LinphoneCallUpdatedByRemote++;break; case LinphoneCallIncomingEarlyMedia :counters->number_of_LinphoneCallIncomingEarlyMedia++;break; case LinphoneCallUpdating :counters->number_of_LinphoneCallUpdating++;break; case LinphoneCallReleased :counters->number_of_LinphoneCallReleased++;break; case LinphoneCallEarlyUpdating: counters->number_of_LinphoneCallEarlyUpdating++;break; case LinphoneCallEarlyUpdatedByRemote: counters->number_of_LinphoneCallEarlyUpdatedByRemote++;break; default: BC_FAIL("unexpected event");break; } } static bool_t rtcp_is_type(const mblk_t *m, rtcp_type_t type){ const rtcp_common_header_t *ch=rtcp_get_common_header(m); return (ch!=NULL && rtcp_common_header_get_packet_type(ch)==type); } static void rtcp_received(stats* counters, mblk_t *packet) { do{ if (rtcp_is_type(packet, RTCP_RTPFB)){ if (rtcp_RTPFB_get_type(packet) == RTCP_RTPFB_TMMBR) { counters->number_of_tmmbr_received++; counters->last_tmmbr_value_received = (int)rtcp_RTPFB_tmmbr_get_max_bitrate(packet); } } }while (rtcp_next_packet(packet)); rtcp_rewind(packet); } void call_stats_updated(LinphoneCore *lc, LinphoneCall *call, const LinphoneCallStats *lstats) { stats* counters = get_stats(lc); counters->number_of_LinphoneCallStatsUpdated++; if (lstats->updated & LINPHONE_CALL_STATS_RECEIVED_RTCP_UPDATE) { counters->number_of_rtcp_received++; if (lstats->rtcp_received_via_mux){ counters->number_of_rtcp_received_via_mux++; } rtcp_received(counters, lstats->received_rtcp); } if (lstats->updated & LINPHONE_CALL_STATS_SENT_RTCP_UPDATE ) { counters->number_of_rtcp_sent++; } if (lstats->updated & LINPHONE_CALL_STATS_PERIODICAL_UPDATE ) { int tab_size = sizeof (counters->audio_download_bandwidth)/sizeof(int); int index = (counters->current_bandwidth_index[lstats->type]++) % tab_size; LinphoneCallStats *audio_stats, *video_stats; audio_stats = linphone_call_get_audio_stats(call); video_stats = linphone_call_get_video_stats(call); if (lstats->type == LINPHONE_CALL_STATS_AUDIO) { counters->audio_download_bandwidth[index] = (int)audio_stats->download_bandwidth; counters->audio_upload_bandwidth[index] = (int)audio_stats->upload_bandwidth; } else { counters->video_download_bandwidth[index] = (int)video_stats->download_bandwidth; counters->video_upload_bandwidth[index] = (int)video_stats->upload_bandwidth; } linphone_call_stats_unref(audio_stats); linphone_call_stats_unref(video_stats); } } void linphone_call_encryption_changed(LinphoneCore *lc, LinphoneCall *call, bool_t on, const char *authentication_token) { char* to=linphone_address_as_string(linphone_call_get_call_log(call)->to); char* from=linphone_address_as_string(linphone_call_get_call_log(call)->from); stats* counters; ms_message(" %s call from [%s] to [%s], is now [%s]",linphone_call_get_call_log(call)->dir==LinphoneCallIncoming?"Incoming":"Outgoing" ,from ,to ,(on?"encrypted":"unencrypted")); ms_free(to); ms_free(from); counters = get_stats(lc); if (on) counters->number_of_LinphoneCallEncryptedOn++; else counters->number_of_LinphoneCallEncryptedOff++; } void linphone_transfer_state_changed(LinphoneCore *lc, LinphoneCall *transfered, LinphoneCallState new_call_state) { char* to=linphone_address_as_string(linphone_call_get_call_log(transfered)->to); char* from=linphone_address_as_string(linphone_call_get_call_log(transfered)->from); stats* counters; ms_message("Transferred call from [%s] to [%s], new state is [%s]",from,to,linphone_call_state_to_string(new_call_state)); ms_free(to); ms_free(from); counters = get_stats(lc); switch (new_call_state) { case LinphoneCallOutgoingInit :counters->number_of_LinphoneTransferCallOutgoingInit++;break; case LinphoneCallOutgoingProgress :counters->number_of_LinphoneTransferCallOutgoingProgress++;break; case LinphoneCallOutgoingRinging :counters->number_of_LinphoneTransferCallOutgoingRinging++;break; case LinphoneCallOutgoingEarlyMedia :counters->number_of_LinphoneTransferCallOutgoingEarlyMedia++;break; case LinphoneCallConnected :counters->number_of_LinphoneTransferCallConnected++;break; case LinphoneCallStreamsRunning :counters->number_of_LinphoneTransferCallStreamsRunning++;break; case LinphoneCallError :counters->number_of_LinphoneTransferCallError++;break; default: BC_FAIL("unexpected event");break; } } void linphone_call_iframe_decoded_cb(LinphoneCall *call,void * user_data) { char* to=linphone_address_as_string(linphone_call_get_call_log(call)->to); char* from=linphone_address_as_string(linphone_call_get_call_log(call)->from); stats* counters; LinphoneCore* lc=(LinphoneCore*)user_data; ms_message("call from [%s] to [%s] receive iFrame",from,to); ms_free(to); ms_free(from); counters = (stats*)get_stats(lc); counters->number_of_IframeDecoded++; } void liblinphone_tester_check_rtcp(LinphoneCoreManager* caller, LinphoneCoreManager* callee) { LinphoneCall *c1,*c2; MSTimeSpec ts; int max_time_to_wait; LinphoneCallStats *audio_stats1, *video_stats1, *audio_stats2, *video_stats2; c1=linphone_core_get_current_call(caller->lc); c2=linphone_core_get_current_call(callee->lc); BC_ASSERT_PTR_NOT_NULL(c1); BC_ASSERT_PTR_NOT_NULL(c2); if (!c1 || !c2) return; linphone_call_ref(c1); linphone_call_ref(c2); liblinphone_tester_clock_start(&ts); if (linphone_core_rtcp_enabled(caller->lc) && linphone_core_rtcp_enabled(callee->lc)) max_time_to_wait = 15000; else max_time_to_wait = 5000; do { audio_stats1 = linphone_call_get_audio_stats(c1); video_stats1 = linphone_call_get_video_stats(c1); audio_stats2 = linphone_call_get_audio_stats(c2); video_stats2 = linphone_call_get_video_stats(c2); if (audio_stats1->round_trip_delay > 0.0 && audio_stats2->round_trip_delay > 0.0 && (!linphone_call_log_video_enabled(linphone_call_get_call_log(c1)) || video_stats1->round_trip_delay>0.0) && (!linphone_call_log_video_enabled(linphone_call_get_call_log(c2)) || video_stats2->round_trip_delay>0.0)) { break; } linphone_call_stats_unref(audio_stats1); linphone_call_stats_unref(audio_stats2); if (video_stats1) linphone_call_stats_unref(video_stats1); if (video_stats2) linphone_call_stats_unref(video_stats2); wait_for_until(caller->lc,callee->lc,NULL,0,20); /*just to sleep while iterating*/ }while (!liblinphone_tester_clock_elapsed(&ts,max_time_to_wait)); audio_stats1 = linphone_call_get_audio_stats(c1); video_stats1 = linphone_call_get_video_stats(c1); audio_stats2 = linphone_call_get_audio_stats(c2); video_stats2 = linphone_call_get_video_stats(c2); if (linphone_core_rtcp_enabled(caller->lc) && linphone_core_rtcp_enabled(callee->lc)) { BC_ASSERT_GREATER(caller->stat.number_of_rtcp_received, 1, int, "%i"); BC_ASSERT_GREATER(callee->stat.number_of_rtcp_received, 1, int, "%i"); BC_ASSERT_GREATER(audio_stats1->round_trip_delay,0.0,float,"%f"); BC_ASSERT_GREATER(audio_stats2->round_trip_delay,0.0,float,"%f"); if (linphone_call_log_video_enabled(linphone_call_get_call_log(c1))) { BC_ASSERT_GREATER(video_stats1->round_trip_delay,0.0,float,"%f"); } if (linphone_call_log_video_enabled(linphone_call_get_call_log(c2))) { BC_ASSERT_GREATER(video_stats2->round_trip_delay,0.0,float,"%f"); } } else { if (linphone_core_rtcp_enabled(caller->lc)) { BC_ASSERT_EQUAL(audio_stats1->rtp_stats.sent_rtcp_packets, 0, unsigned long long, "%llu"); BC_ASSERT_EQUAL(audio_stats2->rtp_stats.recv_rtcp_packets, 0, unsigned long long, "%llu"); if (linphone_call_log_video_enabled(linphone_call_get_call_log(c1))) { BC_ASSERT_EQUAL(video_stats1->rtp_stats.sent_rtcp_packets, 0, unsigned long long, "%llu"); } if (linphone_call_log_video_enabled(linphone_call_get_call_log(c2))) { BC_ASSERT_EQUAL(video_stats2->rtp_stats.recv_rtcp_packets, 0, unsigned long long, "%llu"); } } if (linphone_core_rtcp_enabled(callee->lc)) { BC_ASSERT_EQUAL(audio_stats2->rtp_stats.sent_rtcp_packets, 0, unsigned long long, "%llu"); BC_ASSERT_EQUAL(audio_stats1->rtp_stats.recv_rtcp_packets, 0, unsigned long long, "%llu"); if (linphone_call_log_video_enabled(linphone_call_get_call_log(c1))) { BC_ASSERT_EQUAL(video_stats1->rtp_stats.recv_rtcp_packets, 0, unsigned long long, "%llu"); } if (linphone_call_log_video_enabled(linphone_call_get_call_log(c2))) { BC_ASSERT_EQUAL(video_stats2->rtp_stats.sent_rtcp_packets, 0, unsigned long long, "%llu"); } } } linphone_call_unref(c1); linphone_call_unref(c2); } static void setup_sdp_handling(const LinphoneCallTestParams* params, LinphoneCoreManager* mgr ){ if( params->sdp_removal ){ sal_default_set_sdp_handling(mgr->lc->sal, SalOpSDPSimulateRemove); } else if( params->sdp_simulate_error ){ sal_default_set_sdp_handling(mgr->lc->sal, SalOpSDPSimulateError); } } /* * CAUTION this function is error prone. you should not use it anymore in new tests. * Creating callee call params before the call is actually received is not the good way * to use the Liblinphone API. Indeed, call params used for receiving calls shall be created by linphone_core_create_call_params() by passing * the call object for which params are to be created. * This function should be used only in test case where the programmer exactly knows the caller params, and then can deduce how * callee params will be set by linphone_core_create_call_params(). * This function was developped at a time where the use of the API about incoming params was not yet clarified. * Tests relying on this function are then not testing the correct way to use the api (through linphone_core_create_call_params()), and so * it is not a so good idea to build new tests based on this function. **/ bool_t call_with_params2(LinphoneCoreManager* caller_mgr ,LinphoneCoreManager* callee_mgr , const LinphoneCallTestParams *caller_test_params , const LinphoneCallTestParams *callee_test_params , bool_t build_callee_params) { int retry=0; stats initial_caller=caller_mgr->stat; stats initial_callee=callee_mgr->stat; bool_t result=FALSE; LinphoneCallParams *caller_params = caller_test_params->base; LinphoneCallParams *callee_params = callee_test_params->base; bool_t did_receive_call; LinphoneCall *callee_call=NULL; LinphoneCall *caller_call=NULL; /* TODO: This should be handled correctly inside the liblinphone library but meanwhile handle this here. */ linphone_core_manager_wait_for_stun_resolution(caller_mgr); linphone_core_manager_wait_for_stun_resolution(callee_mgr); setup_sdp_handling(caller_test_params, caller_mgr); setup_sdp_handling(callee_test_params, callee_mgr); if (!caller_params){ BC_ASSERT_PTR_NOT_NULL((caller_call=linphone_core_invite_address(caller_mgr->lc,callee_mgr->identity))); }else{ BC_ASSERT_PTR_NOT_NULL((caller_call=linphone_core_invite_address_with_params(caller_mgr->lc,callee_mgr->identity,caller_params))); } BC_ASSERT_PTR_NULL(linphone_call_get_remote_params(caller_call)); /*assert that remote params are NULL when no response is received yet*/ did_receive_call = wait_for(callee_mgr->lc ,caller_mgr->lc ,&callee_mgr->stat.number_of_LinphoneCallIncomingReceived ,initial_callee.number_of_LinphoneCallIncomingReceived+1); BC_ASSERT_EQUAL(did_receive_call, !callee_test_params->sdp_simulate_error, int, "%d"); sal_default_set_sdp_handling(caller_mgr->lc->sal, SalOpSDPNormal); sal_default_set_sdp_handling(callee_mgr->lc->sal, SalOpSDPNormal); if (!did_receive_call) return 0; if (linphone_core_get_calls_nb(callee_mgr->lc)<=1) BC_ASSERT_TRUE(linphone_core_inc_invite_pending(callee_mgr->lc)); BC_ASSERT_EQUAL(caller_mgr->stat.number_of_LinphoneCallOutgoingProgress,initial_caller.number_of_LinphoneCallOutgoingProgress+1, int, "%d"); while (caller_mgr->stat.number_of_LinphoneCallOutgoingRinging!=(initial_caller.number_of_LinphoneCallOutgoingRinging + 1) && caller_mgr->stat.number_of_LinphoneCallOutgoingEarlyMedia!=(initial_caller.number_of_LinphoneCallOutgoingEarlyMedia +1) && retry++ < 100) { linphone_core_iterate(caller_mgr->lc); linphone_core_iterate(callee_mgr->lc); ms_usleep(20000); } BC_ASSERT_TRUE((caller_mgr->stat.number_of_LinphoneCallOutgoingRinging==initial_caller.number_of_LinphoneCallOutgoingRinging+1) ||(caller_mgr->stat.number_of_LinphoneCallOutgoingEarlyMedia==initial_caller.number_of_LinphoneCallOutgoingEarlyMedia+1)); if (linphone_core_get_calls_nb(callee_mgr->lc) == 1) BC_ASSERT_PTR_NOT_NULL(linphone_core_get_current_call_remote_address(callee_mgr->lc)); /*only relevant if one call, otherwise, not always set*/ callee_call=linphone_core_get_call_by_remote_address2(callee_mgr->lc,caller_mgr->identity); if(!linphone_core_get_current_call(caller_mgr->lc) || (!callee_call && !linphone_core_get_current_call(callee_mgr->lc)) /*for privacy case*/) { return 0; } else if (caller_mgr->identity){ LinphoneAddress* callee_from=linphone_address_clone(caller_mgr->identity); linphone_address_set_port(callee_from,0); /*remove port because port is never present in from header*/ if (linphone_call_params_get_privacy(linphone_call_get_current_params(linphone_core_get_current_call(caller_mgr->lc))) == LinphonePrivacyNone) { /*don't check in case of p asserted id*/ if (!lp_config_get_int(callee_mgr->lc->config,"sip","call_logs_use_asserted_id_instead_of_from",0)) BC_ASSERT_TRUE(linphone_address_weak_equal(callee_from,linphone_call_get_remote_address(callee_call))); } else { BC_ASSERT_FALSE(linphone_address_weak_equal(callee_from,linphone_call_get_remote_address(linphone_core_get_current_call(callee_mgr->lc)))); } linphone_address_unref(callee_from); } if (callee_params){ linphone_call_accept_with_params(callee_call,callee_params); }else if (build_callee_params){ LinphoneCallParams *default_params=linphone_core_create_call_params(callee_mgr->lc,callee_call); ms_message("Created default call params with video=%i", linphone_call_params_video_enabled(default_params)); linphone_call_accept_with_params(callee_call,default_params); linphone_call_params_unref(default_params); }else if (callee_call) { linphone_call_accept(callee_call); } else { linphone_call_accept(linphone_core_get_current_call(callee_mgr->lc)); } BC_ASSERT_TRUE(wait_for(callee_mgr->lc,caller_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallConnected,initial_callee.number_of_LinphoneCallConnected+1)); BC_ASSERT_TRUE(wait_for(callee_mgr->lc,caller_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallConnected,initial_caller.number_of_LinphoneCallConnected+1)); result = wait_for_until(callee_mgr->lc,caller_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallStreamsRunning,initial_caller.number_of_LinphoneCallStreamsRunning+1, 2000) && wait_for_until(callee_mgr->lc,caller_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallStreamsRunning,initial_callee.number_of_LinphoneCallStreamsRunning+1, 2000); if (linphone_core_get_media_encryption(caller_mgr->lc) != LinphoneMediaEncryptionNone || linphone_core_get_media_encryption(callee_mgr->lc) != LinphoneMediaEncryptionNone) { /*wait for encryption to be on, in case of zrtp or dtls, it can take a few seconds*/ if ( (linphone_core_get_media_encryption(caller_mgr->lc) == LinphoneMediaEncryptionZRTP) || (linphone_core_get_media_encryption(callee_mgr->lc) == LinphoneMediaEncryptionZRTP) /* if callee is ZRTP, wait for it */ || (linphone_core_get_media_encryption(caller_mgr->lc) == LinphoneMediaEncryptionDTLS)) wait_for(callee_mgr->lc,caller_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallEncryptedOn,initial_caller.number_of_LinphoneCallEncryptedOn+1); if ((linphone_core_get_media_encryption(callee_mgr->lc) == LinphoneMediaEncryptionZRTP) || (linphone_core_get_media_encryption(callee_mgr->lc) == LinphoneMediaEncryptionDTLS) || (linphone_core_get_media_encryption(caller_mgr->lc) == LinphoneMediaEncryptionZRTP) || (linphone_core_get_media_encryption(caller_mgr->lc) == LinphoneMediaEncryptionDTLS) /*also take care of caller policy*/ ) wait_for(callee_mgr->lc,caller_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallEncryptedOn,initial_callee.number_of_LinphoneCallEncryptedOn+1); /* when caller is encryptionNone but callee is ZRTP, we expect ZRTP to take place */ if ((linphone_core_get_media_encryption(caller_mgr->lc) == LinphoneMediaEncryptionNone) && (linphone_core_get_media_encryption(callee_mgr->lc) == LinphoneMediaEncryptionZRTP) && linphone_core_media_encryption_supported(caller_mgr->lc, LinphoneMediaEncryptionZRTP)) { const LinphoneCallParams* call_param = linphone_call_get_current_params(callee_call); BC_ASSERT_EQUAL(linphone_call_params_get_media_encryption(call_param), LinphoneMediaEncryptionZRTP, int, "%d"); call_param = linphone_call_get_current_params(linphone_core_get_current_call(caller_mgr->lc)); BC_ASSERT_EQUAL(linphone_call_params_get_media_encryption(call_param), LinphoneMediaEncryptionZRTP, int, "%d"); }else { /* otherwise, final status shall stick to caller core parameter */ const LinphoneCallParams* call_param = linphone_call_get_current_params(callee_call); BC_ASSERT_EQUAL(linphone_call_params_get_media_encryption(call_param),linphone_core_get_media_encryption(caller_mgr->lc), int, "%d"); call_param = linphone_call_get_current_params(linphone_core_get_current_call(caller_mgr->lc)); BC_ASSERT_EQUAL(linphone_call_params_get_media_encryption(call_param),linphone_core_get_media_encryption(caller_mgr->lc), int, "%d"); } } /*wait ice re-invite*/ if (linphone_core_get_firewall_policy(caller_mgr->lc) == LinphonePolicyUseIce && linphone_core_get_firewall_policy(callee_mgr->lc) == LinphonePolicyUseIce && !linphone_core_sdp_200_ack_enabled(caller_mgr->lc) /*ice does not work with sdp less invite*/ && lp_config_get_int(callee_mgr->lc->config, "sip", "update_call_when_ice_completed", TRUE) && lp_config_get_int(caller_mgr->lc->config, "sip", "update_call_when_ice_completed", TRUE) && linphone_core_get_media_encryption(caller_mgr->lc) != LinphoneMediaEncryptionDTLS /*no ice-reinvite with DTLS*/) { BC_ASSERT_TRUE(wait_for(callee_mgr->lc,caller_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallStreamsRunning,initial_caller.number_of_LinphoneCallStreamsRunning+2)); BC_ASSERT_TRUE(wait_for(callee_mgr->lc,caller_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallStreamsRunning,initial_callee.number_of_LinphoneCallStreamsRunning+2)); } else if (linphone_core_get_firewall_policy(caller_mgr->lc) == LinphonePolicyUseIce) { /* check no ice re-invite received*/ BC_ASSERT_FALSE(wait_for_until(callee_mgr->lc,caller_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallStreamsRunning,initial_caller.number_of_LinphoneCallStreamsRunning+2,2000)); BC_ASSERT_FALSE(wait_for_until(callee_mgr->lc,caller_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallStreamsRunning,initial_callee.number_of_LinphoneCallStreamsRunning+2,2000)); } if (linphone_core_get_media_encryption(caller_mgr->lc) == LinphoneMediaEncryptionDTLS ) { if (linphone_core_get_current_call(caller_mgr->lc)->audiostream) BC_ASSERT_TRUE(ms_media_stream_sessions_get_encryption_mandatory(&linphone_core_get_current_call(caller_mgr->lc)->audiostream->ms.sessions)); #ifdef VIDEO_ENABLED if (linphone_core_get_current_call(caller_mgr->lc)->videostream && video_stream_started(linphone_core_get_current_call(caller_mgr->lc)->videostream)) BC_ASSERT_TRUE(ms_media_stream_sessions_get_encryption_mandatory(&linphone_core_get_current_call(caller_mgr->lc)->videostream->ms.sessions)); #endif } return result; } /* * CAUTION this function is error prone. you should not use it anymore in new tests. * Creating callee call params before the call is actually received is not the good way * to use the Liblinphone API. Indeed, call params used for receiving calls shall be created by linphone_core_create_call_params() by passing * the call object for which params are to be created. * This function should be used only in test case where the programmer exactly knows the caller params, and then can deduce how * callee params will be set by linphone_core_create_call_params(). * This function was developped at a time where the use of the API about incoming params was not yet clarified. * Tests relying on this function are then not testing the correct way to use the api (through linphone_core_create_call_params()), and so * it is not a so good idea to build new tests based on this function. **/ bool_t call_with_params(LinphoneCoreManager* caller_mgr ,LinphoneCoreManager* callee_mgr ,const LinphoneCallParams *caller_params ,const LinphoneCallParams *callee_params){ LinphoneCallTestParams caller_test_params = {0}, callee_test_params = {0}; caller_test_params.base = (LinphoneCallParams*)caller_params; callee_test_params.base = (LinphoneCallParams*)callee_params; return call_with_params2(caller_mgr,callee_mgr,&caller_test_params,&callee_test_params,FALSE); } /* * CAUTION this function is error prone. you should not use it anymore in new tests. * Creating callee call params before the call is actually received is not the good way * to use the Liblinphone API. Indeed, call params used for receiving calls shall be created by linphone_core_create_call_params() by passing * the call object for which params are to be created. * This function should be used only in test case where the programmer exactly knows the caller params, and then can deduce how * callee params will be set by linphone_core_create_call_params(). * This function was developped at a time where the use of the API about incoming params was not yet clarified. * Tests relying on this function are then not testing the correct way to use the api (through linphone_core_create_call_params()), and so * it is not a so good idea to build new tests based on this function. **/ bool_t call_with_test_params(LinphoneCoreManager* caller_mgr ,LinphoneCoreManager* callee_mgr ,const LinphoneCallTestParams *caller_test_params ,const LinphoneCallTestParams *callee_test_params){ return call_with_params2(caller_mgr,callee_mgr,caller_test_params,callee_test_params,FALSE); } bool_t call_with_caller_params(LinphoneCoreManager* caller_mgr,LinphoneCoreManager* callee_mgr, const LinphoneCallParams *params) { return call_with_params(caller_mgr,callee_mgr,params,NULL); } bool_t call(LinphoneCoreManager* caller_mgr,LinphoneCoreManager* callee_mgr){ return call_with_params(caller_mgr,callee_mgr,NULL,NULL); } void end_call(LinphoneCoreManager *m1, LinphoneCoreManager *m2){ int previous_count_1 = m1->stat.number_of_LinphoneCallEnd; int previous_count_2 = m2->stat.number_of_LinphoneCallEnd; linphone_core_terminate_all_calls(m1->lc); BC_ASSERT_TRUE(wait_for(m1->lc,m2->lc,&m1->stat.number_of_LinphoneCallEnd,previous_count_1+1)); BC_ASSERT_TRUE(wait_for(m1->lc,m2->lc,&m2->stat.number_of_LinphoneCallEnd,previous_count_2+1)); BC_ASSERT_TRUE(wait_for(m1->lc,m2->lc,&m1->stat.number_of_LinphoneCallReleased,previous_count_1+1)); BC_ASSERT_TRUE(wait_for(m1->lc,m2->lc,&m2->stat.number_of_LinphoneCallReleased,previous_count_2+1)); } void simple_call_base(bool_t enable_multicast_recv_side) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; const LinphoneAddress *from; LinphoneCall *pauline_call; LinphoneProxyConfig* marie_cfg; marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); /* with the account manager, we might lose the identity */ marie_cfg = linphone_core_get_default_proxy_config(marie->lc); { LinphoneAddress* marie_addr = linphone_address_clone(linphone_proxy_config_get_identity_address(marie_cfg)); char* marie_tmp_id = NULL; linphone_address_set_display_name(marie_addr, "Super Marie"); marie_tmp_id = linphone_address_as_string(marie_addr); linphone_proxy_config_edit(marie_cfg); linphone_proxy_config_set_identity(marie_cfg,marie_tmp_id); linphone_proxy_config_done(marie_cfg); ms_free(marie_tmp_id); linphone_address_unref(marie_addr); } linphone_core_enable_audio_multicast(pauline->lc,enable_multicast_recv_side); BC_ASSERT_TRUE(call(marie,pauline)); pauline_call=linphone_core_get_current_call(pauline->lc); BC_ASSERT_PTR_NOT_NULL(pauline_call); /*check that display name is correctly propagated in From */ if (pauline_call){ from=linphone_call_get_remote_address(linphone_core_get_current_call(pauline->lc)); BC_ASSERT_PTR_NOT_NULL(from); if (from){ const char *dname=linphone_address_get_display_name(from); BC_ASSERT_PTR_NOT_NULL(dname); if (dname){ BC_ASSERT_STRING_EQUAL(dname, "Super Marie"); } } } liblinphone_tester_check_rtcp(marie,pauline); end_call(marie,pauline); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(marie); } static void simple_call(void) { simple_call_base(FALSE); } /*This test is added to reproduce a crash when a call is failed synchronously*/ static void simple_call_with_no_sip_transport(void){ LinphoneCoreManager* marie; LinphoneCoreManager* pauline; LinphoneSipTransports tr={0}; LinphoneCall *call; marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); /*disable all transports so that the call will fail synchronously*/ linphone_core_set_sip_transports(marie->lc, &tr); call = linphone_core_invite_address(marie->lc, pauline->identity); BC_ASSERT_PTR_NULL(call); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(marie); } static void simple_call_with_udp(void) { LinphoneCoreManager* michelle; LinphoneCoreManager* laure; const LinphoneAddress *from; LinphoneCall *laure_call; LinphoneProxyConfig* michelle_cfg; michelle = linphone_core_manager_new( "michelle_rc_udp"); laure = linphone_core_manager_new("laure_rc_udp"); /* with the account manager, we might lose the identity */ michelle_cfg = linphone_core_get_default_proxy_config(michelle->lc); { LinphoneAddress* michelle_addr = linphone_address_clone(linphone_proxy_config_get_identity_address(michelle_cfg)); char* michelle_tmp_id = NULL; linphone_address_set_display_name(michelle_addr, "Super michelle"); michelle_tmp_id = linphone_address_as_string(michelle_addr); linphone_proxy_config_edit(michelle_cfg); linphone_proxy_config_set_identity(michelle_cfg,michelle_tmp_id); linphone_proxy_config_done(michelle_cfg); ms_free(michelle_tmp_id); linphone_address_unref(michelle_addr); } BC_ASSERT_TRUE(call(michelle,laure)); laure_call=linphone_core_get_current_call(laure->lc); BC_ASSERT_PTR_NOT_NULL(laure_call); /*check that display name is correctly propagated in From */ if (laure_call){ from=linphone_call_get_remote_address(linphone_core_get_current_call(laure->lc)); BC_ASSERT_PTR_NOT_NULL(from); if (from){ const char *dname=linphone_address_get_display_name(from); BC_ASSERT_PTR_NOT_NULL(dname); if (dname){ BC_ASSERT_STRING_EQUAL(dname, "Super michelle"); } } } liblinphone_tester_check_rtcp(michelle,laure); end_call(michelle,laure); linphone_core_manager_destroy(laure); linphone_core_manager_destroy(michelle); } static void automatic_call_termination(void) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); if (!BC_ASSERT_TRUE(call(marie,pauline))) goto end; liblinphone_tester_check_rtcp(marie,pauline); linphone_core_destroy(pauline->lc); pauline->lc = NULL; /*marie shall receive the BYE*/ BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallEnd, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallReleased, 1)); end: linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(marie); } static void call_with_timed_out_bye(void) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; belle_sip_timer_config_t timer_config; marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); BC_ASSERT_TRUE(call(marie,pauline)); sal_set_send_error(pauline->lc->sal,1500); /*to trash the message without generating error*/ timer_config.T1=50; /*to have timer F = 3s*/ timer_config.T2=4000; timer_config.T3=0; timer_config.T4=5000; belle_sip_stack_set_timer_config(sal_get_stack_impl(pauline->lc->sal),&timer_config); linphone_core_terminate_all_calls(pauline->lc); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for_until(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallReleased,1,timer_config.T1*84)); sal_set_send_error(pauline->lc->sal,0); linphone_core_terminate_all_calls(marie->lc); BC_ASSERT_TRUE(wait_for_until(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallEnd,1,5000)); BC_ASSERT_TRUE(wait_for_until(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallReleased,1,5000)); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void direct_call_over_ipv6(void){ LinphoneCoreManager* marie; LinphoneCoreManager* pauline; if (liblinphone_tester_ipv6_available()){ LinphoneSipTransports pauline_transports; LinphoneAddress* pauline_dest = linphone_address_new("sip:[::1];transport=tcp"); marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); linphone_core_enable_ipv6(marie->lc,TRUE); linphone_core_enable_ipv6(pauline->lc,TRUE); linphone_core_set_default_proxy_config(marie->lc,NULL); linphone_core_set_default_proxy_config(pauline->lc, NULL); linphone_core_get_sip_transports_used(pauline->lc,&pauline_transports); linphone_address_set_port(pauline_dest,pauline_transports.tcp_port); linphone_core_invite_address(marie->lc,pauline_dest); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallOutgoingRinging,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallIncomingReceived,1)); linphone_call_accept(linphone_core_get_current_call(pauline->lc)); 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)); liblinphone_tester_check_rtcp(marie,pauline); end_call(marie,pauline); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_address_unref(pauline_dest); }else ms_warning("Test skipped, no ipv6 available"); } static void call_outbound_with_multiple_proxy(void) { LinphoneCoreManager* marie = linphone_core_manager_new2( "marie_rc", FALSE); LinphoneCoreManager* pauline = linphone_core_manager_new2( "pauline_tcp_rc", FALSE); LinphoneProxyConfig* lpc = NULL; LinphoneProxyConfig* registered_lpc = linphone_core_create_proxy_config(marie->lc); lpc = linphone_core_get_default_proxy_config(marie->lc); linphone_core_set_default_proxy(marie->lc,NULL); if (!BC_ASSERT_PTR_NOT_NULL(lpc) || !BC_ASSERT_PTR_NOT_NULL(registered_lpc)) return; // create new LPC that will successfully register linphone_proxy_config_set_identity(registered_lpc, linphone_proxy_config_get_identity(lpc)); linphone_proxy_config_set_server_addr(registered_lpc, linphone_proxy_config_get_addr(lpc)); linphone_proxy_config_set_route(registered_lpc, linphone_proxy_config_get_route(lpc)); linphone_proxy_config_enable_register(registered_lpc, TRUE); linphone_core_add_proxy_config(marie->lc, registered_lpc); linphone_proxy_config_unref(registered_lpc); // set first LPC to unreacheable proxy addr linphone_proxy_config_edit(lpc); linphone_proxy_config_set_server_addr(lpc,"sip:linphone.org:9016;transport=udp"); linphone_proxy_config_set_route(lpc, "sip:linphone.org:9016;transport=udp;lr"); linphone_proxy_config_done(lpc); BC_ASSERT_TRUE(wait_for_until(pauline->lc, NULL, &pauline->stat.number_of_LinphoneRegistrationOk, 1, 10000)); BC_ASSERT_TRUE(wait_for_until(marie->lc, NULL, &marie->stat.number_of_LinphoneRegistrationProgress, 2, 200)); BC_ASSERT_TRUE(wait_for_until(marie->lc, NULL, &marie->stat.number_of_LinphoneRegistrationOk, 1, 10000)); // calling marie should go through the second proxy config BC_ASSERT_TRUE(call(marie, pauline)); end_call(marie, pauline); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } #if 0 /* TODO: activate test when the implementation is ready */ static void multiple_answers_call(void) { /* Scenario is this: pauline calls marie, which is registered 2 times. Both linphones answer at the same time, and only one should get the call running, the other should be terminated */ LinphoneCoreManager* pauline = linphone_core_manager_new( "pauline_rc" ); LinphoneCoreManager* marie1 = linphone_core_manager_new( "marie_rc" ); LinphoneCoreManager* marie2 = linphone_core_manager_new( "marie_rc" ); LinphoneCall* call1, *call2; bctbx_list_t* lcs = bctbx_list_append(NULL,pauline->lc); lcs = bctbx_list_append(lcs,marie1->lc); lcs = bctbx_list_append(lcs,marie2->lc); BC_ASSERT_TRUE(wait_for_until(pauline->lc, NULL, &pauline->stat.number_of_LinphoneRegistrationOk, 1, 2000)); BC_ASSERT_PTR_NOT_NULL( linphone_core_invite_address(pauline->lc, marie1->identity ) ); BC_ASSERT_TRUE(wait_for_list(lcs,&marie1->stat.number_of_LinphoneCallIncomingReceived, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs,&marie2->stat.number_of_LinphoneCallIncomingReceived, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs,&pauline->stat.number_of_LinphoneCallOutgoingProgress, 1, 2000)); // marie 1 and 2 answer at the same time call1 = linphone_core_get_current_call(marie1->lc); call2 = linphone_core_get_current_call(marie2->lc); BC_ASSERT_PTR_NOT_NULL_FATAL(call1); BC_ASSERT_PTR_NOT_NULL_FATAL(call2); BC_ASSERT_EQUAL( linphone_core_accept_call(marie1->lc, call1), 0, int, "%d"); BC_ASSERT_EQUAL( linphone_core_accept_call(marie2->lc, call2), 0, int, "%d"); BC_ASSERT_TRUE( wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1, 2000) ); BC_ASSERT_TRUE( wait_for_list(lcs, &marie1->stat.number_of_LinphoneCallStreamsRunning, 1, 2000) ); BC_ASSERT_TRUE( wait_for_list(lcs, &marie2->stat.number_of_LinphoneCallEnd, 1, 2000) ); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(marie1); linphone_core_manager_destroy(marie2); } #endif static void multiple_answers_call_with_media_relay(void) { /* Scenario is this: pauline calls marie, which is registered 2 times. * Both linphones answer at the same time, and only one should get the * call running, the other should be terminated */ LinphoneCoreManager* pauline = linphone_core_manager_new( "pauline_tcp_rc" ); LinphoneCoreManager* marie1 = linphone_core_manager_new( "marie_rc" ); LinphoneCoreManager* marie2 = linphone_core_manager_new( "marie_rc" ); LinphoneCall* call1, *call2; bctbx_list_t* lcs = bctbx_list_append(NULL,pauline->lc); lcs = bctbx_list_append(lcs,marie1->lc); lcs = bctbx_list_append(lcs,marie2->lc); linphone_core_set_user_agent(pauline->lc, "Natted Linphone", NULL); linphone_core_set_user_agent(marie1->lc, "Natted Linphone", NULL); linphone_core_set_user_agent(marie2->lc, "Natted Linphone", NULL); BC_ASSERT_TRUE(wait_for_until(pauline->lc, NULL, &pauline->stat.number_of_LinphoneRegistrationOk, 1, 2000)); BC_ASSERT_PTR_NOT_NULL( linphone_core_invite_address(pauline->lc, marie1->identity ) ); BC_ASSERT_TRUE(wait_for_list(lcs,&marie1->stat.number_of_LinphoneCallIncomingReceived, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs,&marie2->stat.number_of_LinphoneCallIncomingReceived, 1, 2000)); BC_ASSERT_TRUE(wait_for_list(lcs,&pauline->stat.number_of_LinphoneCallOutgoingProgress, 1, 2000)); // marie 1 and 2 answer at the same time call1 = linphone_core_get_current_call(marie1->lc); call2 = linphone_core_get_current_call(marie2->lc); if (BC_ASSERT_PTR_NOT_NULL(call1) && BC_ASSERT_PTR_NOT_NULL(call2)) { BC_ASSERT_EQUAL( linphone_call_accept(call1), 0, int, "%d"); ms_sleep(1); /*sleep to make sure that the 200OK of marie1 reaches the server first*/ BC_ASSERT_EQUAL( linphone_call_accept(call2), 0, int, "%d"); BC_ASSERT_TRUE( wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1, 2000) ); BC_ASSERT_TRUE( wait_for_list(lcs, &marie1->stat.number_of_LinphoneCallStreamsRunning, 1, 2000) ); /*the server will send a bye to marie2, as is 200Ok arrived second*/ BC_ASSERT_TRUE( wait_for_list(lcs, &marie2->stat.number_of_LinphoneCallEnd, 1, 4000) ); end_call(marie1, pauline); } linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(marie1); linphone_core_manager_destroy(marie2); bctbx_list_free(lcs); } static void call_with_specified_codec_bitrate(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bool_t call_ok; const char * codec = "opus"; int rate = 48000; int min_bw=24; int max_bw=50; #ifdef __arm__ if (ms_factory_get_cpu_count(marie->lc->factory) <2) { /*2 opus codec channel + resampler is too much for a single core*/ #ifndef __ANDROID__ codec = "speex"; rate = 8000; min_bw=20; max_bw=35; #else BC_PASS("Test requires at least a dual core"); goto end; #endif } #endif /*Force marie to play from file: if soundcard is used and it is silient, then vbr mode will drop down the bitrate Note that a play file is already set by linphone_core_manager_new() (but not used)*/ linphone_core_set_use_files(marie->lc, TRUE); if (linphone_core_find_payload_type(marie->lc,codec,rate,-1)==NULL){ BC_PASS("opus codec not supported, test skipped."); goto end; } disable_all_audio_codecs_except_one(marie->lc,codec,rate); disable_all_audio_codecs_except_one(pauline->lc,codec,rate); linphone_core_set_payload_type_bitrate(marie->lc, linphone_core_find_payload_type(marie->lc,codec,rate,-1), max_bw); linphone_core_set_payload_type_bitrate(pauline->lc, linphone_core_find_payload_type(pauline->lc,codec,rate,-1), min_bw); BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (!call_ok) goto end; liblinphone_tester_check_rtcp(marie,pauline); /*wait a bit that bitstreams are stabilized*/ wait_for_until(marie->lc, pauline->lc, NULL, 0, 2000); BC_ASSERT_LOWER(linphone_core_manager_get_mean_audio_down_bw(marie), (int)(min_bw+5+min_bw*.1), int, "%i"); BC_ASSERT_GREATER(linphone_core_manager_get_mean_audio_down_bw(marie), 10, int, "%i"); /*check that at least something is received */ BC_ASSERT_GREATER(linphone_core_manager_get_mean_audio_down_bw(pauline), (int)(max_bw-5-max_bw*.1), int, "%i"); end_call(pauline, marie); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } void disable_all_codecs(const bctbx_list_t* elem, LinphoneCoreManager* call){ PayloadType *pt; for(;elem!=NULL;elem=elem->next){ pt=(PayloadType*)elem->data; linphone_core_enable_payload_type(call->lc,pt,FALSE); } } /*** Disable all audio codecs , sends an INVITE with RTP port 0 and payload 0. Wait for SIP 488 unacceptable. ***/ static void call_with_no_audio_codec(void){ LinphoneCoreManager* callee = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* caller = linphone_core_manager_new(transport_supported(LinphoneTransportTcp) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* out_call ; const bctbx_list_t* elem =linphone_core_get_audio_codecs(caller->lc); disable_all_codecs(elem, caller); out_call = linphone_core_invite_address(caller->lc,callee->identity); linphone_call_ref(out_call); BC_ASSERT_TRUE(wait_for(caller->lc, callee->lc, &caller->stat.number_of_LinphoneCallOutgoingInit, 1)); BC_ASSERT_TRUE(wait_for_until(caller->lc, callee->lc, &caller->stat.number_of_LinphoneCallError, 1, 6000)); BC_ASSERT_EQUAL(linphone_call_get_reason(out_call), LinphoneReasonNotAcceptable, int, "%d"); BC_ASSERT_EQUAL(callee->stat.number_of_LinphoneCallIncomingReceived, 0, int, "%d"); linphone_call_unref(out_call); linphone_core_manager_destroy(callee); linphone_core_manager_destroy(caller); } static void simple_call_compatibility_mode(void) { char route[256]; LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCore* lc_marie=marie->lc; LinphoneCore* lc_pauline=pauline->lc; stats* stat_marie=&marie->stat; stats* stat_pauline=&pauline->stat; LinphoneProxyConfig* proxy; const LinphoneAddress* identity; LinphoneAddress* proxy_address; char*tmp; LinphoneSipTransports transport; proxy = linphone_core_get_default_proxy_config(lc_marie); BC_ASSERT_PTR_NOT_NULL (proxy); identity = linphone_proxy_config_get_identity_address(proxy); proxy_address=linphone_address_new(linphone_proxy_config_get_addr(proxy)); linphone_address_clean(proxy_address); tmp=linphone_address_as_string_uri_only(proxy_address); linphone_proxy_config_set_server_addr(proxy,tmp); sprintf(route,"sip:%s",test_route); linphone_proxy_config_set_route(proxy,route); ms_free(tmp); linphone_address_unref(proxy_address); linphone_core_get_sip_transports(lc_marie,&transport); transport.udp_port=0; transport.tls_port=0; transport.dtls_port=0; /*only keep tcp*/ linphone_core_set_sip_transports(lc_marie,&transport); stat_marie->number_of_LinphoneRegistrationOk=0; BC_ASSERT_TRUE (wait_for(lc_marie,lc_marie,&stat_marie->number_of_LinphoneRegistrationOk,1)); linphone_core_invite_address(lc_marie,pauline->identity); BC_ASSERT_TRUE (wait_for(lc_pauline,lc_marie,&stat_pauline->number_of_LinphoneCallIncomingReceived,1)); BC_ASSERT_TRUE(linphone_core_inc_invite_pending(lc_pauline)); BC_ASSERT_EQUAL(stat_marie->number_of_LinphoneCallOutgoingProgress,1, int, "%d"); BC_ASSERT_TRUE(wait_for(lc_pauline,lc_marie,&stat_marie->number_of_LinphoneCallOutgoingRinging,1)); BC_ASSERT_PTR_NOT_NULL(linphone_core_get_current_call_remote_address(lc_pauline)); if (linphone_core_get_current_call_remote_address(lc_pauline)) { BC_ASSERT_TRUE(linphone_address_weak_equal(identity,linphone_core_get_current_call_remote_address(lc_pauline))); linphone_call_accept(linphone_core_get_current_call(lc_pauline)); BC_ASSERT_TRUE(wait_for(lc_pauline,lc_marie,&stat_pauline->number_of_LinphoneCallConnected,1)); BC_ASSERT_TRUE(wait_for(lc_pauline,lc_marie,&stat_marie->number_of_LinphoneCallConnected,1)); BC_ASSERT_TRUE(wait_for(lc_pauline,lc_marie,&stat_pauline->number_of_LinphoneCallStreamsRunning,1)); BC_ASSERT_TRUE(wait_for(lc_pauline,lc_marie,&stat_marie->number_of_LinphoneCallStreamsRunning,1)); wait_for(lc_pauline,lc_marie,&stat_marie->number_of_LinphoneCallStreamsRunning,3); end_call(pauline, marie); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void terminate_call_with_error(void) { LinphoneCall* call_callee ; LinphoneErrorInfo *ei ; const LinphoneErrorInfo *rei = NULL; LinphoneCoreManager *callee_mgr = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *caller_mgr = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* out_call = linphone_core_invite_address(caller_mgr->lc,callee_mgr->identity); linphone_call_ref(out_call); ei = linphone_error_info_new(); linphone_error_info_set(ei, NULL, LinphoneReasonNone, 200, "Call refused for security reason", NULL); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallOutgoingInit,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &callee_mgr->stat.number_of_LinphoneCallIncomingReceived, 1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallOutgoingProgress, 1)); call_callee = linphone_core_get_current_call(callee_mgr->lc); linphone_call_ref(call_callee); BC_ASSERT_PTR_NOT_NULL(call_callee); BC_ASSERT_EQUAL( linphone_core_accept_call(callee_mgr->lc,call_callee), 0 , int, "%d"); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallConnected,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallStreamsRunning, 1)); BC_ASSERT_PTR_NOT_NULL(ei); if (ei){ BC_ASSERT_EQUAL(linphone_error_info_get_protocol_code(ei),200, int, "%d"); BC_ASSERT_PTR_NOT_NULL(linphone_error_info_get_phrase(ei)); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_phrase(ei), "Call refused for security reason"); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_protocol(ei), "SIP"); } linphone_call_terminate_with_error_info(out_call,ei); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallReleased,1)); rei = linphone_call_get_error_info(call_callee); BC_ASSERT_PTR_NOT_NULL(rei); if (rei){ BC_ASSERT_EQUAL(linphone_error_info_get_protocol_code(rei),200, int, "%d"); BC_ASSERT_PTR_NOT_NULL(linphone_error_info_get_phrase(rei)); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_phrase(rei), "Call refused for security reason"); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_protocol(ei), "SIP"); } BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallReleased,1)); linphone_error_info_unref(ei); linphone_call_unref(out_call); linphone_call_unref(call_callee); linphone_core_manager_destroy(callee_mgr); linphone_core_manager_destroy(caller_mgr); } static void cancel_call_with_error(void) { LinphoneCall* call_callee ; LinphoneErrorInfo *ei ; const LinphoneErrorInfo *rei = NULL; LinphoneCoreManager *callee_mgr = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *caller_mgr = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* out_call = linphone_core_invite_address(caller_mgr->lc,callee_mgr->identity); linphone_call_ref(out_call); ei = linphone_error_info_new(); linphone_error_info_set(ei, NULL, LinphoneReasonNone, 600, "Call has been cancelled", NULL); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallOutgoingInit,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &callee_mgr->stat.number_of_LinphoneCallIncomingReceived, 1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallOutgoingProgress, 1)); call_callee = linphone_core_get_current_call(callee_mgr->lc); linphone_call_ref(call_callee); BC_ASSERT_PTR_NOT_NULL(call_callee); BC_ASSERT_PTR_NOT_NULL(ei); if (ei){ BC_ASSERT_EQUAL(linphone_error_info_get_protocol_code(ei),600, int, "%d"); BC_ASSERT_PTR_NOT_NULL(linphone_error_info_get_phrase(ei)); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_phrase(ei), "Call has been cancelled"); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_protocol(ei), "SIP"); } linphone_call_terminate_with_error_info(out_call,ei); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallReleased,1)); rei = linphone_call_get_error_info(call_callee); BC_ASSERT_PTR_NOT_NULL(rei); if (rei){ BC_ASSERT_EQUAL(linphone_error_info_get_protocol_code(rei),600, int, "%d"); BC_ASSERT_PTR_NOT_NULL(linphone_error_info_get_phrase(rei)); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_phrase(rei), "Call has been cancelled"); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_protocol(rei), "SIP"); } BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallReleased,1)); linphone_error_info_unref(ei); linphone_call_unref(out_call); linphone_call_unref(call_callee); linphone_core_manager_destroy(callee_mgr); linphone_core_manager_destroy(caller_mgr); } static void cancel_other_device_after_accept(void) { LinphoneCall* call_callee = NULL; LinphoneCall* call_callee_2 = NULL ; const LinphoneErrorInfo *rei = NULL; LinphoneCoreManager *callee_mgr = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *callee_mgr_2 = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *caller_mgr = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* out_call = linphone_core_invite_address(caller_mgr->lc,callee_mgr->identity); linphone_call_ref(out_call); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallOutgoingInit,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &callee_mgr->stat.number_of_LinphoneCallIncomingReceived, 1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallOutgoingProgress, 1)); call_callee = linphone_core_get_current_call(callee_mgr->lc); if (BC_ASSERT_PTR_NOT_NULL(call_callee)) { linphone_call_ref(call_callee); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr_2->lc, &callee_mgr_2->stat.number_of_LinphoneCallIncomingReceived, 1)); call_callee_2 = linphone_core_get_current_call(callee_mgr_2->lc); linphone_call_ref(call_callee_2); BC_ASSERT_PTR_NOT_NULL(call_callee_2); BC_ASSERT_EQUAL( linphone_call_accept(call_callee), 0 , int, "%d"); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallConnected,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallStreamsRunning, 1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr_2->lc,&callee_mgr_2->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr_2->lc,&callee_mgr_2->stat.number_of_LinphoneCallReleased,1)); rei = linphone_call_get_error_info(call_callee_2); BC_ASSERT_PTR_NOT_NULL(rei); if (rei){ BC_ASSERT_EQUAL(linphone_error_info_get_protocol_code(rei),200, int, "%d"); BC_ASSERT_PTR_NOT_NULL(linphone_error_info_get_phrase(rei)); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_phrase(rei), "Call completed elsewhere"); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_protocol(rei), "SIP"); } } linphone_call_terminate(out_call); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallReleased,1)); if (out_call) linphone_call_unref(out_call); if (call_callee) linphone_call_unref(call_callee); if (call_callee_2) linphone_call_unref(call_callee_2); linphone_core_manager_destroy(callee_mgr); linphone_core_manager_destroy(callee_mgr_2); linphone_core_manager_destroy(caller_mgr); } static void cancel_other_device_after_decline(void) { LinphoneCall* call_callee = NULL; LinphoneCall* call_callee_2 = NULL; const LinphoneErrorInfo *rei = NULL; LinphoneCoreManager *callee_mgr = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *callee_mgr_2 = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *caller_mgr = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* out_call = linphone_core_invite_address(caller_mgr->lc,callee_mgr->identity); linphone_call_ref(out_call); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallOutgoingInit,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &callee_mgr->stat.number_of_LinphoneCallIncomingReceived, 1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallOutgoingProgress, 1)); call_callee = linphone_core_get_current_call(callee_mgr->lc); if (BC_ASSERT_PTR_NOT_NULL(call_callee)) { linphone_call_ref(call_callee); BC_ASSERT_TRUE(wait_for(caller_mgr->lc, callee_mgr_2->lc, &callee_mgr_2->stat.number_of_LinphoneCallIncomingReceived, 1)); call_callee_2 = linphone_core_get_current_call(callee_mgr_2->lc); linphone_call_ref(call_callee_2); BC_ASSERT_PTR_NOT_NULL(call_callee_2); BC_ASSERT_EQUAL(linphone_call_decline(call_callee, LinphoneReasonDeclined), 0 , int, "%d"); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc, &caller_mgr->stat.number_of_LinphoneCallReleased, 1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr_2->lc,&callee_mgr_2->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr_2->lc,&callee_mgr_2->stat.number_of_LinphoneCallReleased,1)); rei = linphone_call_get_error_info(call_callee_2); BC_ASSERT_PTR_NOT_NULL(rei); if (rei){ BC_ASSERT_EQUAL(linphone_error_info_get_protocol_code(rei),600, int, "%d"); BC_ASSERT_PTR_NOT_NULL(linphone_error_info_get_phrase(rei)); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_phrase(rei), "Busy Everywhere"); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_protocol(rei), "SIP"); } } if (out_call) linphone_call_unref(out_call); if (call_callee) linphone_call_unref(call_callee); if (call_callee_2) linphone_call_unref(call_callee_2); linphone_core_manager_destroy(callee_mgr); linphone_core_manager_destroy(callee_mgr_2); linphone_core_manager_destroy(caller_mgr); } static void cancelled_call(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* out_call = linphone_core_invite_address(pauline->lc,marie->identity); linphone_call_ref(out_call); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallOutgoingInit,1)); linphone_call_terminate(out_call); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallEnd,1)); //BC_ASSERT_EQUAL(linphone_call_get_reason(out_call),LinphoneReasonCanceled, int, "%d"); BC_ASSERT_EQUAL(pauline->stat.number_of_LinphoneCallEnd,1, int, "%d"); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallIncomingReceived,0, int, "%d"); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallReleased,1)); linphone_call_unref(out_call); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } void disable_all_audio_codecs_except_one(LinphoneCore *lc, const char *mime, int rate){ const bctbx_list_t *elem=linphone_core_get_audio_codecs(lc); PayloadType *pt; for(;elem!=NULL;elem=elem->next){ pt=(PayloadType*)elem->data; linphone_core_enable_payload_type(lc,pt,FALSE); } pt=linphone_core_find_payload_type(lc,mime,rate,-1); if (BC_ASSERT_PTR_NOT_NULL(pt)) { linphone_core_enable_payload_type(lc,pt,TRUE); } } void disable_all_video_codecs_except_one(LinphoneCore *lc, const char *mime) { #ifdef VIDEO_ENABLED const bctbx_list_t *codecs = linphone_core_get_video_codecs(lc); const bctbx_list_t *it = NULL; PayloadType *pt = NULL; for(it = codecs; it != NULL; it = it->next) { linphone_core_enable_payload_type(lc, (PayloadType *)it->data, FALSE); } pt = linphone_core_find_payload_type(lc, mime, -1, -1); if (BC_ASSERT_PTR_NOT_NULL(pt)) { linphone_core_enable_payload_type(lc, pt, TRUE); } #endif } static void call_with_dns_time_out(void) { LinphoneCoreManager* marie = linphone_core_manager_new2( "empty_rc", FALSE); LinphoneSipTransports transport = {9773,0,0,0}; int i; linphone_core_set_sip_transports(marie->lc,&transport); linphone_core_iterate(marie->lc); sal_set_dns_timeout(marie->lc->sal,0); linphone_core_invite(marie->lc,"\"t\x8et\x8e\" "); /*just to use non ascii values*/ for(i=0;i<10;i++){ ms_usleep(200000); linphone_core_iterate(marie->lc); } BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallOutgoingInit,1, int, "%d"); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallOutgoingProgress,1, int, "%d"); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallError,1, int, "%d"); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallReleased,1, int, "%d"); linphone_core_manager_destroy(marie); } static void early_cancelled_call(void) { LinphoneCoreManager* marie = linphone_core_manager_new( "marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new2( "empty_rc",FALSE); LinphoneCall* out_call = linphone_core_invite_address(pauline->lc,marie->identity); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallOutgoingInit,1)); const char *callID = linphone_call_log_get_call_id(linphone_call_get_call_log(out_call)); BC_ASSERT_PTR_NOT_NULL(callID); linphone_call_terminate(out_call); /*since everything is executed in a row, no response can be received from the server, thus the CANCEL cannot be sent. It will ring at Marie's side.*/ BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_EQUAL(pauline->stat.number_of_LinphoneCallEnd,1, int, "%d"); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallIncomingReceived,1)); /* now the CANCEL should have been sent and the the call at marie's side should terminate*/ BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallReleased,1)); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void cancelled_ringing_call(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); const bctbx_list_t * call_history; LinphoneCall* out_call; char * db_path= bctbx_strdup_printf("%s,%s",bc_tester_get_writable_dir_prefix(),"tmp_call_log.db"); linphone_core_set_call_logs_database_path(marie->lc,db_path); out_call = linphone_core_invite_address(pauline->lc,marie->identity); linphone_call_ref(out_call); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallIncomingReceived,1)); linphone_call_terminate(out_call); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallReleased,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallReleased,1)); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallEnd,1, int, "%d"); BC_ASSERT_EQUAL(pauline->stat.number_of_LinphoneCallEnd,1, int, "%d"); call_history = linphone_core_get_call_history(marie->lc); BC_ASSERT_PTR_NOT_NULL(call_history); BC_ASSERT_EQUAL((int)bctbx_list_size(call_history),1, int,"%i"); BC_ASSERT_EQUAL(linphone_call_log_get_status((LinphoneCallLog*)bctbx_list_get_data(call_history)), LinphoneCallMissed, LinphoneCallStatus, "%i"); linphone_call_unref(out_call); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); unlink(db_path); bctbx_free(db_path); } static void early_declined_call(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCallLog* out_call_log; LinphoneCall* out_call; linphone_core_set_max_calls(marie->lc,0); out_call = linphone_core_invite_address(pauline->lc,marie->identity); linphone_call_ref(out_call); /*wait until flexisip transfers the busy...*/ BC_ASSERT_TRUE(wait_for_until(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallError,1,33000)); BC_ASSERT_EQUAL(pauline->stat.number_of_LinphoneCallError,1, int, "%d"); /* FIXME http://git.linphone.org/mantis/view.php?id=757 BC_ASSERT_EQUAL(linphone_call_get_reason(out_call),LinphoneReasonBusy, int, "%d"); */ if (bctbx_list_size(linphone_core_get_call_logs(pauline->lc))>0) { BC_ASSERT_PTR_NOT_NULL(out_call_log=(LinphoneCallLog*)(linphone_core_get_call_logs(pauline->lc)->data)); BC_ASSERT_EQUAL(linphone_call_log_get_status(out_call_log),LinphoneCallAborted, int, "%d"); } linphone_call_unref(out_call); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_busy_when_calling_self(void) { LinphoneCoreManager* marie = linphone_core_manager_new( "marie_rc"); LinphoneCall *out_call=linphone_core_invite_address(marie->lc,marie->identity); linphone_call_ref(out_call); /*wait until flexisip transfers the busy...*/ BC_ASSERT_TRUE(wait_for_until(marie->lc,marie->lc,&marie->stat.number_of_LinphoneCallError,1,33000)); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallError,1, int, "%d"); BC_ASSERT_EQUAL(linphone_call_get_reason(out_call),LinphoneReasonBusy, int, "%d"); linphone_call_unref(out_call); linphone_core_manager_destroy(marie); } static void call_declined_with_error(void) { LinphoneCoreManager* callee_mgr = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* caller_mgr = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* in_call = NULL; LinphoneCall* out_call = linphone_core_invite_address(caller_mgr->lc,callee_mgr->identity); LinphoneFactory* factory = linphone_factory_get(); const LinphoneErrorInfo* rcvd_ei; const LinphoneErrorInfo* sub_rcvd_ei; LinphoneErrorInfo *ei = linphone_factory_create_error_info(factory); LinphoneErrorInfo *reason_ei = linphone_factory_create_error_info(factory); linphone_error_info_set(ei, "SIP", LinphoneReasonUnknown, 603, "Decline", NULL); //ordre des arguments à vérifier linphone_error_info_set(reason_ei, "hardware", LinphoneReasonUnknown, 66, "J'ai plus de batterie", NULL); linphone_error_info_set_sub_error_info(ei, reason_ei); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallIncomingReceived,1)); BC_ASSERT_PTR_NOT_NULL(in_call=linphone_core_get_current_call(callee_mgr->lc)); linphone_call_ref(out_call); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallIncomingReceived,1)); BC_ASSERT_PTR_NOT_NULL(in_call=linphone_core_get_current_call(callee_mgr->lc)); if (in_call) { linphone_call_ref(in_call); linphone_call_decline_with_error_info(in_call, ei); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(callee_mgr->lc,caller_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallEnd,1)); rcvd_ei = linphone_call_get_error_info(out_call); sub_rcvd_ei = linphone_error_info_get_sub_error_info(rcvd_ei); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_phrase(rcvd_ei), "Decline"); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_protocol(rcvd_ei), "SIP"); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_phrase(sub_rcvd_ei), "J'ai plus de batterie"); BC_ASSERT_STRING_EQUAL(linphone_error_info_get_protocol(sub_rcvd_ei), "hardware"); BC_ASSERT_EQUAL(linphone_call_get_reason(in_call),LinphoneReasonDeclined, int, "%d"); BC_ASSERT_EQUAL(linphone_call_log_get_status(linphone_call_get_call_log(in_call)),LinphoneCallDeclined, int, "%d"); BC_ASSERT_EQUAL(linphone_call_get_reason(out_call),LinphoneReasonDeclined, int, "%d"); BC_ASSERT_EQUAL(linphone_call_log_get_status(linphone_call_get_call_log(out_call)),LinphoneCallDeclined, int, "%d"); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallReleased,1)); BC_ASSERT_TRUE(wait_for(caller_mgr->lc,callee_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallReleased,1)); linphone_call_unref(in_call); } linphone_call_unref(out_call); linphone_error_info_unref(reason_ei); linphone_error_info_unref(ei); linphone_core_manager_destroy(callee_mgr); linphone_core_manager_destroy(caller_mgr); } static void call_declined(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* in_call; LinphoneCall* out_call = linphone_core_invite_address(pauline->lc,marie->identity); linphone_call_ref(out_call); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallIncomingReceived,1)); BC_ASSERT_PTR_NOT_NULL(in_call=linphone_core_get_current_call(marie->lc)); if (in_call) { linphone_call_ref(in_call); linphone_call_terminate(in_call); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallReleased,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallReleased,1)); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallEnd,1, int, "%d"); BC_ASSERT_EQUAL(pauline->stat.number_of_LinphoneCallEnd,1, int, "%d"); BC_ASSERT_EQUAL(linphone_call_get_reason(in_call),LinphoneReasonDeclined, int, "%d"); BC_ASSERT_EQUAL(linphone_call_log_get_status(linphone_call_get_call_log(in_call)),LinphoneCallDeclined, int, "%d"); BC_ASSERT_EQUAL(linphone_call_get_reason(out_call),LinphoneReasonDeclined, int, "%d"); BC_ASSERT_EQUAL(linphone_call_log_get_status(linphone_call_get_call_log(out_call)),LinphoneCallDeclined, int, "%d"); linphone_call_unref(in_call); } linphone_call_unref(out_call); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_terminated_by_caller(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); BC_ASSERT_TRUE(call(pauline,marie)); end_call(pauline, marie); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_no_sdp(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); linphone_core_enable_sdp_200_ack(marie->lc,TRUE); BC_ASSERT_TRUE(call(marie,pauline)); end_call(pauline, marie); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_no_sdp_ack_without_sdp(void){ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall *call; linphone_core_enable_sdp_200_ack(marie->lc,TRUE); linphone_core_invite_address(marie->lc,pauline->identity); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallIncomingReceived,1)); call=linphone_core_get_current_call(pauline->lc); if (call){ sal_call_set_sdp_handling(call->op, SalOpSDPSimulateError); /*this will have the effect that the SDP received in the ACK will be ignored*/ linphone_call_accept(call); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallError,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallEnd,1)); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } int check_nb_media_starts(LinphoneCoreManager *caller, LinphoneCoreManager *callee, unsigned int caller_nb_media_starts, unsigned int callee_nb_media_starts) { int c1_ret = FALSE, c2_ret = FALSE; LinphoneCall *c1 = linphone_core_get_current_call(caller->lc); LinphoneCall *c2 = linphone_core_get_current_call(callee->lc); BC_ASSERT_PTR_NOT_NULL(c1); BC_ASSERT_PTR_NOT_NULL(c2); if (!c1 || !c2) return FALSE; if (c1) { c1_ret = c1->nb_media_starts == caller_nb_media_starts; BC_ASSERT_EQUAL(c1->nb_media_starts, caller_nb_media_starts, unsigned int, "%u"); } if (c2) { c2_ret = c2->nb_media_starts == callee_nb_media_starts; BC_ASSERT_EQUAL(c2->nb_media_starts, callee_nb_media_starts, unsigned int, "%u"); } return c1_ret && c2_ret; } void _call_with_ice_base(LinphoneCoreManager* pauline,LinphoneCoreManager* marie, bool_t caller_with_ice, bool_t callee_with_ice, bool_t random_ports, bool_t forced_relay) { linphone_core_set_user_agent(pauline->lc, "Natted Linphone", NULL); linphone_core_set_user_agent(marie->lc, "Natted Linphone", NULL); if (callee_with_ice){ linphone_core_set_firewall_policy(marie->lc,LinphonePolicyUseIce); } if (caller_with_ice){ linphone_core_set_firewall_policy(pauline->lc,LinphonePolicyUseIce); } if (random_ports){ linphone_core_set_audio_port(marie->lc,-1); linphone_core_set_video_port(marie->lc,-1); linphone_core_set_text_port(marie->lc, -1); linphone_core_set_audio_port(pauline->lc,-1); linphone_core_set_video_port(pauline->lc,-1); linphone_core_set_text_port(pauline->lc, -1); } if (forced_relay == TRUE) { linphone_core_enable_forced_ice_relay(marie->lc, TRUE); linphone_core_enable_forced_ice_relay(pauline->lc, TRUE); } if (!BC_ASSERT_TRUE(call(pauline,marie))) return; if (callee_with_ice && caller_with_ice) { /*wait for the ICE reINVITE to complete*/ BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); if (forced_relay == TRUE) { BC_ASSERT_TRUE(check_ice(pauline, marie, LinphoneIceStateRelayConnection)); } else { BC_ASSERT_TRUE(check_ice(pauline,marie,LinphoneIceStateHostConnection)); } check_nb_media_starts(pauline, marie, 1, 1); } liblinphone_tester_check_rtcp(marie,pauline); /*then close the call*/ end_call(pauline, marie); } static void _call_with_ice(bool_t caller_with_ice, bool_t callee_with_ice, bool_t random_ports, bool_t forced_relay, bool_t ipv6) { LinphoneCoreManager* marie = linphone_core_manager_new2("marie_rc", FALSE); LinphoneCoreManager* pauline = linphone_core_manager_new2(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc", FALSE); if (ipv6) { linphone_core_enable_ipv6(marie->lc, TRUE); linphone_core_enable_ipv6(pauline->lc, TRUE); } linphone_core_manager_start(marie, TRUE); linphone_core_manager_start(pauline, TRUE); _call_with_ice_base(pauline,marie,caller_with_ice,callee_with_ice,random_ports,forced_relay); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_ice(void){ _call_with_ice(TRUE,TRUE,FALSE,FALSE,FALSE); } static void call_with_ice_ipv6(void) { if (liblinphone_tester_ipv6_available()) { _call_with_ice(TRUE, TRUE, FALSE, FALSE, TRUE); } else { ms_warning("Test skipped, no ipv6 available"); } } /*ICE is not expected to work in this case, however this should not crash*/ static void call_with_ice_no_sdp(void){ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); linphone_core_enable_sdp_200_ack(pauline->lc,TRUE); linphone_core_set_firewall_policy(marie->lc,LinphonePolicyUseIce); linphone_core_set_firewall_policy(pauline->lc,LinphonePolicyUseIce); BC_ASSERT_TRUE(call(pauline,marie)); liblinphone_tester_check_rtcp(marie,pauline); end_call(pauline, marie); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_ice_random_ports(void){ _call_with_ice(TRUE,TRUE,TRUE,FALSE,FALSE); } static void call_with_ice_forced_relay(void) { _call_with_ice(TRUE, TRUE, TRUE, TRUE, FALSE); } static void ice_to_not_ice(void){ _call_with_ice(TRUE,FALSE,FALSE,FALSE,FALSE); } static void not_ice_to_ice(void){ _call_with_ice(FALSE,TRUE,FALSE,FALSE,FALSE); } static void ice_added_by_reinvite(void){ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneNatPolicy *pol; LinphoneCallParams *params; LinphoneCall *c; bool_t call_ok; lp_config_set_int(linphone_core_get_config(marie->lc), "net", "allow_late_ice", 1); lp_config_set_int(linphone_core_get_config(pauline->lc), "net", "allow_late_ice", 1); BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (!call_ok) goto end; liblinphone_tester_check_rtcp(marie,pauline); /*enable ICE on both ends*/ pol = linphone_core_get_nat_policy(marie->lc); linphone_nat_policy_enable_ice(pol, TRUE); linphone_nat_policy_enable_stun(pol, TRUE); linphone_core_set_nat_policy(marie->lc, pol); pol = linphone_core_get_nat_policy(pauline->lc); linphone_nat_policy_enable_ice(pol, TRUE); linphone_nat_policy_enable_stun(pol, TRUE); linphone_core_set_nat_policy(pauline->lc, pol); linphone_core_manager_wait_for_stun_resolution(marie); linphone_core_manager_wait_for_stun_resolution(pauline); c = linphone_core_get_current_call(marie->lc); params = linphone_core_create_call_params(marie->lc, c); linphone_call_update(c, params); linphone_call_params_unref(params); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallUpdatedByRemote,1)); /*wait for the ICE reINVITE*/ BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,3)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,3)); BC_ASSERT_TRUE(check_ice(marie, pauline, LinphoneIceStateHostConnection)); end_call(pauline, marie); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void on_ack_processing(LinphoneCall *call, LinphoneHeaders *ack, bool_t is_received){ if (!is_received){ linphone_headers_add(ack, "Coucou", "me voila"); /*We put something in user_data to indicate that we went through this callback*/ linphone_call_set_user_data(call, (void*)1); }else{ const char *ack_header; ack_header = linphone_headers_get_value(ack, "Coucou"); BC_ASSERT_PTR_NOT_NULL(ack_header); if (ack_header){ BC_ASSERT_STRING_EQUAL(ack_header, "me voila"); } linphone_call_set_user_data(call, (void*)1); } } static void call_created(LinphoneCore *lc, LinphoneCall *call){ LinphoneCallCbs *cbs = linphone_factory_create_call_cbs(linphone_factory_get()); linphone_call_cbs_set_ack_processing(cbs, on_ack_processing); linphone_call_add_callbacks(call, cbs); linphone_call_cbs_unref(cbs); } static void call_with_custom_headers(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall *call_marie,*call_pauline; LinphoneCallParams *params; const LinphoneCallParams *marie_remote_params; const char *hvalue; char *pauline_remote_contact_header, *pauline_remote_contact, *marie_remote_contact, *marie_remote_contact_header; LinphoneAddress* marie_identity; char* tmp=linphone_address_as_string_uri_only(marie->identity); char tmp2[256]; LinphoneCoreCbs *core_cbs = linphone_factory_create_core_cbs(linphone_factory_get()); snprintf(tmp2,sizeof(tmp2),"%s?uriHeader=myUriHeader",tmp); marie_identity=linphone_address_new(tmp2); ms_free(tmp); linphone_address_unref(marie->identity); marie->identity=marie_identity; params=linphone_core_create_call_params(marie->lc, NULL); linphone_call_params_add_custom_header(params,"Weather","bad"); linphone_call_params_add_custom_header(params,"Working","yes"); linphone_core_cbs_set_call_created(core_cbs, call_created); linphone_core_add_callbacks(marie->lc, core_cbs); linphone_core_add_callbacks(pauline->lc, core_cbs); if (!BC_ASSERT_TRUE(call_with_caller_params(pauline,marie,params))) goto end; call_marie=linphone_core_get_current_call(marie->lc); call_pauline=linphone_core_get_current_call(pauline->lc); BC_ASSERT_PTR_NOT_NULL(call_marie); BC_ASSERT_PTR_NOT_NULL(call_pauline); marie_remote_params=linphone_call_get_remote_params(call_marie); hvalue=linphone_call_params_get_custom_header(marie_remote_params,"Weather"); BC_ASSERT_PTR_NOT_NULL(hvalue); BC_ASSERT_STRING_EQUAL(hvalue,"bad"); hvalue=linphone_call_params_get_custom_header(marie_remote_params,"uriHeader"); BC_ASSERT_PTR_NOT_NULL(hvalue); BC_ASSERT_STRING_EQUAL(hvalue,"myUriHeader"); // FIXME: we have to strdup because successive calls to get_remote_params erase the returned const char*!! pauline_remote_contact = ms_strdup(linphone_call_get_remote_contact(call_pauline)); pauline_remote_contact_header = ms_strdup(linphone_call_params_get_custom_header(linphone_call_get_remote_params(call_pauline), "Contact")); marie_remote_contact = ms_strdup(linphone_call_get_remote_contact(call_marie)); marie_remote_contact_header = ms_strdup(linphone_call_params_get_custom_header(linphone_call_get_remote_params(call_marie), "Contact")); BC_ASSERT_PTR_NOT_NULL(pauline_remote_contact); BC_ASSERT_PTR_NOT_NULL(pauline_remote_contact_header); BC_ASSERT_PTR_NOT_NULL(marie_remote_contact); BC_ASSERT_PTR_NOT_NULL(marie_remote_contact_header); BC_ASSERT_STRING_EQUAL(pauline_remote_contact,pauline_remote_contact_header); BC_ASSERT_STRING_EQUAL(marie_remote_contact,marie_remote_contact_header); /*we need to wait for the ack to arrive*/ wait_for_until(marie->lc, pauline->lc, NULL, 0, 3000); BC_ASSERT_TRUE(linphone_call_get_user_data(call_marie) == (void*)1); BC_ASSERT_TRUE(linphone_call_get_user_data(call_pauline) == (void*)1); ms_free(pauline_remote_contact); ms_free(pauline_remote_contact_header); ms_free(marie_remote_contact); ms_free(marie_remote_contact_header); end_call(pauline, marie); end: linphone_core_cbs_unref(core_cbs); linphone_call_params_unref(params); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_custom_sdp_attributes_cb(LinphoneCore *lc, LinphoneCall *call, LinphoneCallState cstate, const char *message) { if (cstate == LinphoneCallUpdatedByRemote) { LinphoneCallParams *params; const LinphoneCallParams *remote_params = linphone_call_get_remote_params(call); const char *value = linphone_call_params_get_custom_sdp_attribute(remote_params, "weather"); BC_ASSERT_PTR_NOT_NULL(value); if (value) BC_ASSERT_STRING_EQUAL(value, "sunny"); params = linphone_core_create_call_params(lc, call); linphone_call_params_clear_custom_sdp_attributes(params); linphone_call_params_clear_custom_sdp_media_attributes(params, LinphoneStreamTypeAudio); linphone_call_params_add_custom_sdp_attribute(params, "working", "no"); BC_ASSERT_EQUAL(linphone_call_accept_update(call, params), 0, int, "%i"); linphone_call_params_unref(params); } } static void call_with_custom_sdp_attributes(void) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall *call_marie, *call_pauline; LinphoneCallParams *pauline_params; const LinphoneCallParams *marie_remote_params; const LinphoneCallParams *pauline_remote_params; const char *value; LinphoneCoreVTable *vtable; pauline_params = linphone_core_create_call_params(pauline->lc, NULL); linphone_call_params_add_custom_sdp_attribute(pauline_params, "weather", "bad"); linphone_call_params_add_custom_sdp_attribute(pauline_params, "working", "yes"); linphone_call_params_add_custom_sdp_attribute(pauline_params, "attribute_without_value", NULL); /*cannot be tested yet*/ linphone_call_params_add_custom_sdp_media_attribute(pauline_params, LinphoneStreamTypeAudio, "sleeping", "almost"); BC_ASSERT_TRUE(call_with_caller_params(pauline, marie, pauline_params)); linphone_call_params_unref(pauline_params); call_marie = linphone_core_get_current_call(marie->lc); call_pauline = linphone_core_get_current_call(pauline->lc); BC_ASSERT_PTR_NOT_NULL(call_marie); BC_ASSERT_PTR_NOT_NULL(call_pauline); marie_remote_params = linphone_call_get_remote_params(call_marie); value = linphone_call_params_get_custom_sdp_attribute(marie_remote_params, "weather"); BC_ASSERT_PTR_NOT_NULL(value); if (value) BC_ASSERT_STRING_EQUAL(value, "bad"); value = linphone_call_params_get_custom_sdp_media_attribute(marie_remote_params, LinphoneStreamTypeAudio, "sleeping"); BC_ASSERT_PTR_NOT_NULL(value); if (value) BC_ASSERT_STRING_EQUAL(value, "almost"); vtable = linphone_core_v_table_new(); vtable->call_state_changed = call_with_custom_sdp_attributes_cb; linphone_core_add_listener(marie->lc, vtable); pauline_params = linphone_core_create_call_params(pauline->lc, call_pauline); linphone_call_params_clear_custom_sdp_attributes(pauline_params); linphone_call_params_clear_custom_sdp_media_attributes(pauline_params, LinphoneStreamTypeAudio); linphone_call_params_add_custom_sdp_attribute(pauline_params, "weather", "sunny"); linphone_call_update(call_pauline, pauline_params); BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &marie->stat.number_of_LinphoneCallUpdatedByRemote, 1)); BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &pauline->stat.number_of_LinphoneCallUpdating, 1)); linphone_call_params_unref(pauline_params); BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); pauline_remote_params = linphone_call_get_remote_params(call_pauline); value = linphone_call_params_get_custom_sdp_attribute(pauline_remote_params, "working"); BC_ASSERT_PTR_NOT_NULL(value); if (value) BC_ASSERT_STRING_EQUAL(value, "no"); end_call(pauline, marie); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_custom_header_or_sdp_cb(LinphoneCore *lc, LinphoneCall *call, LinphoneCallState cstate, const char *message) { const char *value; if (cstate == LinphoneCallOutgoingInit){ LinphoneCallParams *params = linphone_call_params_copy(linphone_call_get_params(call)); linphone_call_params_add_custom_sdp_attribute(params, "working", "maybe"); linphone_call_set_params(call, params); linphone_call_params_unref(params); } else if (cstate == LinphoneCallIncomingReceived){ const LinphoneCallParams *tparams = linphone_call_get_remote_params(call); LinphoneCallParams *params = linphone_call_params_copy(tparams); //Check received params //SDP value = linphone_call_params_get_custom_sdp_attribute(params, "working"); BC_ASSERT_PTR_NOT_NULL(value); if (value) BC_ASSERT_STRING_EQUAL(value, "maybe"); //header value = linphone_call_params_get_custom_header(params, "weather"); BC_ASSERT_PTR_NOT_NULL(value); if (value) BC_ASSERT_STRING_EQUAL(value, "thunderstorm"); //modify SDP linphone_call_params_add_custom_sdp_attribute(params, "working", "yes"); linphone_call_set_params(call, params); linphone_call_params_unref(params); } } static void call_caller_with_custom_header_or_sdp_attributes(void) { LinphoneCoreManager *callee_mgr = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *caller_mgr = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall *call_caller = NULL, *call_callee = NULL; LinphoneCallParams *caller_params; // *callee_params ; LinphoneCoreVTable *vtable; LinphoneCallTestParams caller_test_params = {0}; LinphoneCallTestParams callee_test_params = {0}; stats initial_caller=caller_mgr->stat; stats initial_callee=callee_mgr->stat; bool_t result=FALSE; bool_t did_receive_call; //Create caller params with custom header and custom SDP caller_params = linphone_core_create_call_params(caller_mgr->lc, NULL); linphone_call_params_add_custom_header(caller_params, "weather", "thunderstorm"); linphone_call_params_add_custom_sdp_media_attribute(caller_params, LinphoneStreamTypeAudio, "sleeping", "almost"); caller_test_params.base = (LinphoneCallParams*)caller_params; callee_test_params.base = NULL; /* TODO: This should be handled correctly inside the liblinphone library but meanwhile handle this here. */ linphone_core_manager_wait_for_stun_resolution(caller_mgr); linphone_core_manager_wait_for_stun_resolution(callee_mgr); setup_sdp_handling(&caller_test_params, caller_mgr); setup_sdp_handling(&callee_test_params, callee_mgr); // Assign dedicated callback to vtable for caller and callee vtable = linphone_core_v_table_new(); vtable->call_state_changed = call_with_custom_header_or_sdp_cb; linphone_core_add_listener(callee_mgr->lc, vtable); linphone_core_add_listener(caller_mgr->lc, vtable); //Caller initates the call with INVITE // caller params not null BC_ASSERT_PTR_NOT_NULL((call_caller=linphone_core_invite_address_with_params(caller_mgr->lc,callee_mgr->identity,caller_params))); BC_ASSERT_PTR_NULL(linphone_call_get_remote_params(call_caller)); /*assert that remote params are NULL when no response is received yet*/ // Wait for Incoming received did_receive_call = wait_for(callee_mgr->lc ,caller_mgr->lc ,&callee_mgr->stat.number_of_LinphoneCallIncomingReceived ,initial_callee.number_of_LinphoneCallIncomingReceived+1); BC_ASSERT_EQUAL(did_receive_call, !callee_test_params.sdp_simulate_error, int, "%d"); linphone_call_params_unref(caller_params); sal_default_set_sdp_handling(caller_mgr->lc->sal, SalOpSDPNormal); sal_default_set_sdp_handling(callee_mgr->lc->sal, SalOpSDPNormal); // Wait for Outgoing Progress if (linphone_core_get_calls_nb(callee_mgr->lc)<=1) BC_ASSERT_TRUE(linphone_core_inc_invite_pending(callee_mgr->lc)); BC_ASSERT_EQUAL(caller_mgr->stat.number_of_LinphoneCallOutgoingProgress,initial_caller.number_of_LinphoneCallOutgoingProgress+1, int, "%d"); LinphoneCallParams *default_params=linphone_core_create_call_params(callee_mgr->lc,call_callee); ms_message("Created default call params with video=%i", linphone_call_params_video_enabled(default_params)); linphone_core_accept_call_with_params(callee_mgr->lc,call_callee,default_params); linphone_call_params_unref(default_params); BC_ASSERT_TRUE(wait_for(callee_mgr->lc,caller_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallConnected,initial_callee.number_of_LinphoneCallConnected+1)); BC_ASSERT_TRUE(wait_for(callee_mgr->lc,caller_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallConnected,initial_caller.number_of_LinphoneCallConnected+1)); result = wait_for_until(callee_mgr->lc,caller_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallStreamsRunning,initial_caller.number_of_LinphoneCallStreamsRunning+1, 2000) && wait_for_until(callee_mgr->lc,caller_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallStreamsRunning,initial_callee.number_of_LinphoneCallStreamsRunning+1, 2000); BC_ASSERT_TRUE(result); caller_params = linphone_core_create_call_params(caller_mgr->lc, call_caller); linphone_call_params_clear_custom_sdp_attributes(caller_params); linphone_call_params_clear_custom_sdp_media_attributes(caller_params, LinphoneStreamTypeAudio); linphone_call_params_add_custom_sdp_attribute(caller_params, "weather", "sunny"); linphone_core_update_call(caller_mgr->lc, call_caller, caller_params); linphone_call_params_unref(caller_params); end_call(caller_mgr, callee_mgr); linphone_core_manager_destroy(callee_mgr); linphone_core_manager_destroy(caller_mgr); } static void call_callee_with_custom_header_or_sdp_cb(LinphoneCore *lc, LinphoneCall *call, LinphoneCallState cstate, const char *message) { const char *value; if (cstate == LinphoneCallOutgoingInit){ LinphoneCallParams *params = linphone_call_params_copy(linphone_call_get_params(call)); linphone_call_params_add_custom_sdp_attribute(params, "working", "maybe"); linphone_call_set_params(call, params); linphone_call_params_unref(params); } else if (cstate == LinphoneCallIncomingReceived){ const LinphoneCallParams *tparams = linphone_call_get_remote_params(call); LinphoneCallParams *params = linphone_call_params_copy(tparams); value = linphone_call_params_get_custom_sdp_attribute(params, "working"); BC_ASSERT_PTR_NOT_NULL(value); if (value) BC_ASSERT_STRING_EQUAL(value, "maybe"); linphone_call_set_params(call, params); linphone_call_params_unref(params); } } static void call_callee_with_custom_header_or_sdp_attributes(void) { int result; LinphoneCoreManager *callee_mgr = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *caller_mgr = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall *call_caller = NULL, *call_callee = NULL; LinphoneCallParams *callee_params, *caller_params ; LinphoneCoreVTable *vtable; const char *value; LinphoneCallTestParams caller_test_params = {0}; LinphoneCallTestParams callee_test_params = {0}; stats initial_caller=caller_mgr->stat; stats initial_callee=callee_mgr->stat; bool_t did_receive_call; const LinphoneCallParams *caller_remote_params; caller_params = linphone_core_create_call_params(caller_mgr->lc, NULL); callee_test_params.base = NULL; caller_test_params.base = NULL; /* TODO: This should be handled correctly inside the liblinphone library but meanwhile handle this here. */ linphone_core_manager_wait_for_stun_resolution(caller_mgr); linphone_core_manager_wait_for_stun_resolution(callee_mgr); setup_sdp_handling(&caller_test_params, caller_mgr); setup_sdp_handling(&callee_test_params, callee_mgr); // Assign dedicated callback to vtable for caller and callee vtable = linphone_core_v_table_new(); vtable->call_state_changed = call_callee_with_custom_header_or_sdp_cb; linphone_core_add_listener(callee_mgr->lc, vtable); linphone_core_add_listener(caller_mgr->lc, vtable); //Caller initates the call with INVITE // caller params not null BC_ASSERT_PTR_NOT_NULL((call_caller=linphone_core_invite_address_with_params(caller_mgr->lc,callee_mgr->identity,caller_params))); BC_ASSERT_PTR_NULL(linphone_call_get_remote_params(call_caller)); /*assert that remote params are NULL when no response is received yet*/ // Wait for Incoming received did_receive_call = wait_for(callee_mgr->lc ,caller_mgr->lc ,&callee_mgr->stat.number_of_LinphoneCallIncomingReceived ,initial_callee.number_of_LinphoneCallIncomingReceived+1); BC_ASSERT_EQUAL(did_receive_call, !callee_test_params.sdp_simulate_error, int, "%d"); sal_default_set_sdp_handling(caller_mgr->lc->sal, SalOpSDPNormal); sal_default_set_sdp_handling(callee_mgr->lc->sal, SalOpSDPNormal); // Wait for Outgoing Progress if (linphone_core_get_calls_nb(callee_mgr->lc)<=1) BC_ASSERT_TRUE(linphone_core_inc_invite_pending(callee_mgr->lc)); BC_ASSERT_EQUAL(caller_mgr->stat.number_of_LinphoneCallOutgoingProgress,initial_caller.number_of_LinphoneCallOutgoingProgress+1, int, "%d"); //Create callee params with custom header and custom SDP callee_params = linphone_core_create_call_params(callee_mgr->lc,call_callee); linphone_call_params_add_custom_header(callee_params, "weather", "thunderstorm"); linphone_call_params_add_custom_sdp_media_attribute(callee_params, LinphoneStreamTypeAudio, "sleeping", "almost"); linphone_call_params_add_custom_sdp_attribute(callee_params, "working", "yes"); ms_message("Created default call params with video=%i", linphone_call_params_video_enabled(callee_params)); linphone_core_accept_call_with_params(callee_mgr->lc,call_callee,callee_params); linphone_call_params_unref(callee_params); BC_ASSERT_TRUE(wait_for(callee_mgr->lc,caller_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallConnected,initial_callee.number_of_LinphoneCallConnected+1)); BC_ASSERT_TRUE(wait_for(callee_mgr->lc,caller_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallConnected,initial_caller.number_of_LinphoneCallConnected+1)); result = wait_for_until(callee_mgr->lc,caller_mgr->lc,&caller_mgr->stat.number_of_LinphoneCallStreamsRunning,initial_caller.number_of_LinphoneCallStreamsRunning+1, 2000) && wait_for_until(callee_mgr->lc,caller_mgr->lc,&callee_mgr->stat.number_of_LinphoneCallStreamsRunning,initial_callee.number_of_LinphoneCallStreamsRunning+1, 2000); BC_ASSERT_TRUE(result); caller_remote_params = linphone_call_get_remote_params(call_caller); value = linphone_call_params_get_custom_sdp_attribute(caller_remote_params, "working"); BC_ASSERT_PTR_NOT_NULL(value); if (value) BC_ASSERT_STRING_EQUAL(value, "yes"); //header value = linphone_call_params_get_custom_header(caller_remote_params, "weather"); BC_ASSERT_PTR_NOT_NULL(value); if (value) BC_ASSERT_STRING_EQUAL(value, "thunderstorm"); linphone_call_params_unref(caller_params); end_call(caller_mgr, callee_mgr); linphone_core_manager_destroy(callee_mgr); linphone_core_manager_destroy(caller_mgr); } void call_paused_resumed_base(bool_t multicast, bool_t with_losses) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* call_pauline; const rtp_stats_t * stats; bool_t call_ok; linphone_core_enable_audio_multicast(pauline->lc,multicast); BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (!call_ok) goto end; call_pauline = linphone_core_get_current_call(pauline->lc); wait_for_until(pauline->lc, marie->lc, NULL, 5, 3000); if (with_losses) { sal_set_send_error(marie->lc->sal,1500); /*to trash 200ok without generating error*/ } linphone_call_pause(call_pauline); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPausing,1)); if (with_losses) { BC_ASSERT_FALSE(wait_for_until(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPaused,1,1000)); sal_set_send_error(marie->lc->sal,0); /*to trash 200ok without generating error*/ } BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPausedByRemote,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPaused,1)); /*stay in pause a little while in order to generate traffic*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 2000); linphone_call_resume(call_pauline); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); /*same here: wait a while for a bit of a traffic, we need to receive a RTCP packet*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 5000); /*since RTCP streams are reset when call is paused/resumed, there should be no loss at all*/ if (BC_ASSERT_PTR_NOT_NULL(call_pauline->sessions->rtp_session)) { stats = rtp_session_get_stats(call_pauline->sessions->rtp_session); BC_ASSERT_EQUAL((int)stats->cum_packet_loss, 0, int, "%d"); } if (with_losses) { /* now we want to loose the ack*/ linphone_call_pause(call_pauline); sal_set_send_error(pauline->lc->sal,1500); /*to trash ACK without generating error*/ BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPausing,2)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPausedByRemote,2)); BC_ASSERT_TRUE(wait_for_until(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPaused,2,1000)); /*now try to resume, it should be OK*/ sal_set_send_error(pauline->lc->sal,0); linphone_call_resume(call_pauline); BC_ASSERT_TRUE(wait_for_until(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,3,2000)); BC_ASSERT_TRUE(wait_for_until(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,3,2000)); } end_call(pauline, marie); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_paused_resumed(void) { call_paused_resumed_base(FALSE,FALSE); } static void call_paused_resumed_with_sip_packets_losses(void) { call_paused_resumed_base(FALSE,TRUE); } static void call_paused_by_both(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* call_pauline, *call_marie; const rtp_stats_t * stats; bctbx_list_t *lcs = NULL; bool_t call_ok; lcs = bctbx_list_append(lcs, pauline->lc); lcs = bctbx_list_append(lcs, marie->lc); BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (!call_ok) goto end; call_pauline = linphone_core_get_current_call(pauline->lc); call_marie = linphone_core_get_current_call(marie->lc); wait_for_until(pauline->lc, marie->lc, NULL, 5, 2000); linphone_call_pause(call_pauline); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPausing,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPausedByRemote,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPaused,1)); /*stay in pause a little while in order to generate traffic*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 2000); /*marie pauses the call also*/ linphone_call_pause(call_marie); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPausing,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPaused,1)); wait_for_until(pauline->lc, marie->lc, NULL, 5, 2000); /*pauline must stay in paused state*/ BC_ASSERT_EQUAL(pauline->stat.number_of_LinphoneCallPaused, 1, int, "%i"); check_media_direction(pauline, call_pauline, lcs, LinphoneMediaDirectionInactive, LinphoneMediaDirectionInvalid); check_media_direction(marie, call_marie, lcs, LinphoneMediaDirectionInactive, LinphoneMediaDirectionInvalid); /*now pauline wants to resume*/ linphone_call_resume(call_pauline); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallResuming,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPausedByRemote,1)); /*Marie must stay in paused state*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 2000); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallPaused, 1, int, "%i"); /*now marie wants to resume also*/ linphone_call_resume(call_marie); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallResuming,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); /*same here: wait a while for a bit of a traffic, we need to receive a RTCP packet*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 5000); /*since RTCP streams are reset when call is paused/resumed, there should be no loss at all*/ stats = rtp_session_get_stats(call_pauline->sessions->rtp_session); BC_ASSERT_EQUAL((int)stats->cum_packet_loss, 0, int, "%d"); end_call(marie, pauline); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); bctbx_list_free(lcs); } #define CHECK_CURRENT_LOSS_RATE() \ rtcp_count_current = pauline->stat.number_of_rtcp_sent; \ /*wait for an RTCP packet to have an accurate cumulative lost value*/ \ BC_ASSERT_TRUE(wait_for_until(pauline->lc, marie->lc, &pauline->stat.number_of_rtcp_sent, rtcp_count_current+1, 10000)); \ stats = rtp_session_get_stats(call_pauline->audiostream->ms.sessions.rtp_session); \ loss_percentage = stats->cum_packet_loss * 100.f / (stats->packet_recv + stats->cum_packet_loss); \ BC_ASSERT_GREATER(loss_percentage, .75f * params.loss_rate, float, "%f"); \ BC_ASSERT_LOWER(loss_percentage , 1.25f * params.loss_rate, float, "%f") static void call_paused_resumed_with_loss(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* call_pauline; const rtp_stats_t * stats; float loss_percentage; int rtcp_count_current; OrtpNetworkSimulatorParams params={0}; params.enabled=TRUE; params.loss_rate=20; BC_ASSERT_TRUE(call(pauline,marie)); call_pauline = linphone_core_get_current_call(pauline->lc); if (call_pauline){ rtp_session_enable_network_simulation(call_pauline->audiostream->ms.sessions.rtp_session,¶ms); /*generate some traffic*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 10000); CHECK_CURRENT_LOSS_RATE(); /*pause call*/ linphone_call_pause(call_pauline); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPausing,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPausedByRemote,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPaused,1)); /*stay in pause a little while in order to generate traffic*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 10000); CHECK_CURRENT_LOSS_RATE(); /*resume*/ linphone_call_resume(call_pauline); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); wait_for_until(pauline->lc, marie->lc, NULL, 5, 10000); /*since stats are NOT totally reset during pause, the stats->packet_recv is computed from the start of call. This test ensures that the loss rate is consistent during the entire call.*/ CHECK_CURRENT_LOSS_RATE(); end_call(marie, pauline); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } bool_t pause_call_1(LinphoneCoreManager* mgr_1,LinphoneCall* call_1,LinphoneCoreManager* mgr_2,LinphoneCall* call_2) { stats initial_call_stat_1=mgr_1->stat; stats initial_call_stat_2=mgr_2->stat; linphone_call_pause(call_1); BC_ASSERT_TRUE(wait_for(mgr_1->lc,mgr_2->lc,&mgr_1->stat.number_of_LinphoneCallPausing,initial_call_stat_1.number_of_LinphoneCallPausing+1)); BC_ASSERT_TRUE(wait_for(mgr_1->lc,mgr_2->lc,&mgr_1->stat.number_of_LinphoneCallPaused,initial_call_stat_1.number_of_LinphoneCallPaused+1)); BC_ASSERT_TRUE(wait_for(mgr_1->lc,mgr_2->lc,&mgr_2->stat.number_of_LinphoneCallPausedByRemote,initial_call_stat_2.number_of_LinphoneCallPausedByRemote+1)); BC_ASSERT_EQUAL(linphone_call_get_state(call_1),LinphoneCallPaused, int, "%d"); BC_ASSERT_EQUAL(linphone_call_get_state(call_2),LinphoneCallPausedByRemote, int, "%d"); return linphone_call_get_state(call_1) == LinphoneCallPaused && linphone_call_get_state(call_2)==LinphoneCallPausedByRemote; } #if 0 void concurrent_paused_resumed_base(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* call_pauline,call_marie; const rtp_stats_t * stats; BC_ASSERT_TRUE(call(pauline,marie)); call_pauline = linphone_core_get_current_call(pauline->lc); call_marie = linphone_core_get_current_call(marie->lc); linphone_core_pause_call(pauline->lc,call_pauline); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPausing,1)); linphone_core_pause_call(marie->lc,call_marie); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPausedByRemote,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPaused,1)); /*stay in pause a little while in order to generate traffic*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 2000); linphone_core_resume_call(pauline->lc,call_pauline); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); /*same here: wait a while for a bit of a traffic, we need to receive a RTCP packet*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 5000); /*since RTCP streams are reset when call is paused/resumed, there should be no loss at all*/ stats = rtp_session_get_stats(call_pauline->sessions->rtp_session); BC_ASSERT_EQUAL(stats->cum_packet_loss, 0, int, "%d"); linphone_core_terminate_all_calls(pauline->lc); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallEnd,1)); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } #endif static void call_paused_resumed_from_callee(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* call_marie; const rtp_stats_t * stats; bool_t call_ok; BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (!call_ok) goto end; call_marie = linphone_core_get_current_call(marie->lc); linphone_call_pause(call_marie); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPausing,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPausedByRemote,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPaused,1)); /*stay in pause a little while in order to generate traffic*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 2000); linphone_call_resume(call_marie); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); /*same here: wait a while for a bit of a traffic, we need to receive a RTCP packet*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 5000); /*since RTCP streams are reset when call is paused/resumed, there should be no loss at all*/ stats = rtp_session_get_stats(call_marie->sessions->rtp_session); BC_ASSERT_EQUAL((int)stats->cum_packet_loss, 0, int, "%d"); end_call(pauline, marie); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void audio_call_with_ice_no_matching_audio_codecs(void) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall *out_call; const bctbx_list_t *logs; LinphoneCallLog *cl; linphone_core_enable_payload_type(marie->lc, linphone_core_find_payload_type(marie->lc, "PCMU", 8000, 1), FALSE); /* Disable PCMU */ linphone_core_enable_payload_type(marie->lc, linphone_core_find_payload_type(marie->lc, "PCMA", 8000, 1), TRUE); /* Enable PCMA */ linphone_core_set_firewall_policy(marie->lc, LinphonePolicyUseIce); linphone_core_set_firewall_policy(pauline->lc, LinphonePolicyUseIce); linphone_core_manager_wait_for_stun_resolution(marie); linphone_core_manager_wait_for_stun_resolution(pauline); out_call = linphone_core_invite_address(marie->lc, pauline->identity); linphone_call_ref(out_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallOutgoingInit, 1)); /* flexisip will retain the 488 until the "urgent reply" timeout arrives. */ BC_ASSERT_TRUE(wait_for_until(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallError, 1, 6000)); BC_ASSERT_EQUAL(linphone_call_get_reason(out_call), LinphoneReasonNotAcceptable, int, "%d"); BC_ASSERT_EQUAL(pauline->stat.number_of_LinphoneCallIncomingReceived, 0, int, "%d"); logs = linphone_core_get_call_logs(pauline->lc); BC_ASSERT_EQUAL((int)bctbx_list_size(logs), 1, int, "%d"); if (logs){ const LinphoneErrorInfo *ei; cl = (LinphoneCallLog*)logs->data; BC_ASSERT_EQUAL(linphone_call_log_get_status(cl), LinphoneCallEarlyAborted, int, "%d"); BC_ASSERT_TRUE(linphone_call_log_get_start_date(cl) != 0); ei = linphone_call_log_get_error_info(cl); BC_ASSERT_PTR_NOT_NULL(ei); if (ei){ BC_ASSERT_EQUAL(linphone_error_info_get_reason(ei), LinphoneReasonNotAcceptable, int, "%d"); } } linphone_call_unref(out_call); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void _call_with_media_relay(bool_t random_ports) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bool_t call_ok; linphone_core_set_user_agent(marie->lc,"Natted Linphone",NULL); linphone_core_set_user_agent(pauline->lc,"Natted Linphone",NULL); linphone_core_set_video_device(pauline->lc,liblinphone_tester_mire_id); linphone_core_set_video_device(marie->lc,liblinphone_tester_mire_id); if (random_ports){ linphone_core_set_audio_port(marie->lc,-1); linphone_core_set_video_port(marie->lc,-1); linphone_core_set_audio_port(pauline->lc,-1); linphone_core_set_video_port(pauline->lc,-1); } BC_ASSERT_TRUE(call_ok=call(pauline,marie)); if (!call_ok) goto end; liblinphone_tester_check_rtcp(pauline,marie); #ifdef VIDEO_ENABLED BC_ASSERT_TRUE(request_video(pauline,marie, TRUE)); liblinphone_tester_check_rtcp(pauline,marie); #endif end_call(pauline, marie); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_media_relay(void) { _call_with_media_relay(FALSE); } static void call_with_media_relay_random_ports(void) { _call_with_media_relay(TRUE); } static void call_with_privacy(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall *c1,*c2; LinphoneCallParams *params; LinphoneProxyConfig* pauline_proxy; params=linphone_core_create_call_params(pauline->lc, NULL); linphone_call_params_set_privacy(params,LinphonePrivacyId); BC_ASSERT_TRUE(call_with_caller_params(pauline,marie,params)); linphone_call_params_unref(params); c1=linphone_core_get_current_call(pauline->lc); c2=linphone_core_get_current_call(marie->lc); BC_ASSERT_PTR_NOT_NULL(c1); BC_ASSERT_PTR_NOT_NULL(c2); if (c1 && c2){ /*make sure local identity is unchanged*/ BC_ASSERT_TRUE(linphone_address_weak_equal(linphone_call_log_get_from(linphone_call_get_call_log(c1)),pauline->identity)); /*make sure remote identity is hidden*/ BC_ASSERT_FALSE(linphone_address_weak_equal(linphone_call_get_remote_address(c2),pauline->identity)); BC_ASSERT_EQUAL(linphone_call_params_get_privacy(linphone_call_get_current_params(c2)),LinphonePrivacyId, int, "%d"); } end_call(pauline, marie); /*test proxy config privacy*/ pauline_proxy = linphone_core_get_default_proxy_config(pauline->lc); linphone_proxy_config_set_privacy(pauline_proxy,LinphonePrivacyId); BC_ASSERT_TRUE(call(pauline,marie)); c1=linphone_core_get_current_call(pauline->lc); c2=linphone_core_get_current_call(marie->lc); BC_ASSERT_PTR_NOT_NULL(c1); BC_ASSERT_PTR_NOT_NULL(c2); if (c1 && c2){ /*make sure remote identity is hidden*/ BC_ASSERT_FALSE(linphone_address_weak_equal(linphone_call_get_remote_address(c2),pauline->identity)); BC_ASSERT_EQUAL(linphone_call_params_get_privacy(linphone_call_get_current_params(c2)),LinphonePrivacyId, int, "%d"); } end_call(pauline, marie); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } /*this ones makes call with privacy without previous registration*/ static void call_with_privacy2(void) { LinphoneCoreManager* marie = linphone_core_manager_new( "marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new2(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc", FALSE); LinphoneCall *c1,*c2; LinphoneCallParams *params; LinphoneProxyConfig* pauline_proxy; params=linphone_core_create_call_params(pauline->lc, NULL); linphone_call_params_set_privacy(params,LinphonePrivacyId); pauline_proxy = linphone_core_get_default_proxy_config(pauline->lc); linphone_proxy_config_edit(pauline_proxy); linphone_proxy_config_enable_register(pauline_proxy,FALSE); linphone_proxy_config_done(pauline_proxy); BC_ASSERT_TRUE(call_with_caller_params(pauline,marie,params)); linphone_call_params_unref(params); c1=linphone_core_get_current_call(pauline->lc); c2=linphone_core_get_current_call(marie->lc); BC_ASSERT_PTR_NOT_NULL(c1); BC_ASSERT_PTR_NOT_NULL(c2); if (c1 && c2){ /*make sure local identity is unchanged*/ BC_ASSERT_TRUE(linphone_address_weak_equal(linphone_call_log_get_from(linphone_call_get_call_log(c1)),pauline->identity)); /*make sure remote identity is hidden*/ BC_ASSERT_FALSE(linphone_address_weak_equal(linphone_call_get_remote_address(c2),pauline->identity)); BC_ASSERT_EQUAL(linphone_call_params_get_privacy(linphone_call_get_current_params(c2)),LinphonePrivacyId, int, "%d"); } end_call(pauline, marie); /*test proxy config privacy*/ linphone_proxy_config_set_privacy(pauline_proxy,LinphonePrivacyId); BC_ASSERT_TRUE(call(pauline,marie)); c1=linphone_core_get_current_call(pauline->lc); c2=linphone_core_get_current_call(marie->lc); BC_ASSERT_PTR_NOT_NULL(c1); BC_ASSERT_PTR_NOT_NULL(c2); if (c1 && c2){ /*make sure remote identity is hidden*/ BC_ASSERT_FALSE(linphone_address_weak_equal(linphone_call_get_remote_address(c2),pauline->identity)); BC_ASSERT_EQUAL(linphone_call_params_get_privacy(linphone_call_get_current_params(c2)),LinphonePrivacyId, int, "%d"); } end_call(marie, pauline); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void srtp_call(void) { call_base(LinphoneMediaEncryptionSRTP,FALSE,FALSE,LinphonePolicyNoFirewall,FALSE); } static void zrtp_call(void) { call_base(LinphoneMediaEncryptionZRTP,FALSE,FALSE,LinphonePolicyNoFirewall,FALSE); } static void zrtp_sas_call(void) { call_base_with_configfile(LinphoneMediaEncryptionZRTP,FALSE,FALSE,LinphonePolicyNoFirewall,FALSE, "marie_zrtp_b256_rc", "pauline_zrtp_b256_rc"); call_base_with_configfile(LinphoneMediaEncryptionZRTP,FALSE,FALSE,LinphonePolicyNoFirewall,FALSE, "marie_zrtp_b256_rc", "pauline_tcp_rc"); } static void zrtp_cipher_call(void) { call_base_with_configfile(LinphoneMediaEncryptionZRTP,FALSE,FALSE,LinphonePolicyNoFirewall,FALSE, "marie_zrtp_srtpsuite_aes256_rc", "pauline_zrtp_srtpsuite_aes256_rc"); call_base_with_configfile(LinphoneMediaEncryptionZRTP,FALSE,FALSE,LinphonePolicyNoFirewall,FALSE, "marie_zrtp_aes256_rc", "pauline_zrtp_aes256_rc"); call_base_with_configfile(LinphoneMediaEncryptionZRTP,FALSE,FALSE,LinphonePolicyNoFirewall,FALSE, "marie_zrtp_aes256_rc", "pauline_tcp_rc"); } static void dtls_srtp_call(void) { call_base(LinphoneMediaEncryptionDTLS,FALSE,FALSE,LinphonePolicyNoFirewall,FALSE); } static void dtls_srtp_call_with_media_realy(void) { call_base(LinphoneMediaEncryptionDTLS,FALSE,TRUE,LinphonePolicyNoFirewall,FALSE); } static void dtls_srtp_ice_call(void) { call_base(LinphoneMediaEncryptionDTLS,FALSE,FALSE,LinphonePolicyUseIce,FALSE); } static void call_with_declined_srtp(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); if (linphone_core_media_encryption_supported(marie->lc,LinphoneMediaEncryptionSRTP)) { linphone_core_set_media_encryption(pauline->lc,LinphoneMediaEncryptionSRTP); BC_ASSERT_TRUE(call(pauline,marie)); end_call(marie, pauline); } else { ms_warning ("not tested because srtp not available"); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_srtp_paused_and_resumed(void) { /* * This test was made to evidence a bug due to internal usage of current_params while not yet filled by linphone_call_get_current_params(). * As a result it must not use the call() function because it calls linphone_call_get_current_params(). */ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); const LinphoneCallParams *params; LinphoneCall *pauline_call; if (!linphone_core_media_encryption_supported(marie->lc,LinphoneMediaEncryptionSRTP)) goto end; linphone_core_set_media_encryption(pauline->lc,LinphoneMediaEncryptionSRTP); linphone_core_invite_address(pauline->lc, marie->identity); if (!BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallIncomingReceived,1))) goto end; pauline_call = linphone_core_get_current_call(pauline->lc); linphone_call_accept(linphone_core_get_current_call(marie->lc)); if (!BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,1))) goto end; if (!BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,1))) goto end; linphone_call_pause(pauline_call); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPaused,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPausedByRemote,1)); linphone_call_resume(pauline_call); if (!BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2))) goto end; if (!BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2))) goto end; /*assert that after pause and resume, SRTP is still being used*/ params = linphone_call_get_current_params(linphone_core_get_current_call(pauline->lc)); BC_ASSERT_EQUAL(linphone_call_params_get_media_encryption(params) , LinphoneMediaEncryptionSRTP, int, "%d"); params = linphone_call_get_current_params(linphone_core_get_current_call(marie->lc)); BC_ASSERT_EQUAL(linphone_call_params_get_media_encryption(params) , LinphoneMediaEncryptionSRTP, int, "%d"); end_call(pauline, marie); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void on_eof(LinphonePlayer *player){ LinphonePlayerCbs *cbs = linphone_player_get_callbacks(player); LinphoneCoreManager *marie=(LinphoneCoreManager*)linphone_player_cbs_get_user_data(cbs); marie->stat.number_of_player_eof++; } static void call_with_file_player(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphonePlayer *player; LinphonePlayerCbs *cbs; char *hellopath = bc_tester_res("sounds/ahbahouaismaisbon.wav"); char *recordpath = bc_tester_file("record-call_with_file_player.wav"); bool_t call_ok; int attempts; double similar=1; const double threshold = 0.9; /*this test is actually attempted three times in case of failure, because the audio comparison at the end is very sensitive to * jitter buffer drifts, which sometimes happen if the machine is unable to run the test in good realtime conditions */ for (attempts=0; attempts<3; attempts++){ reset_counters(&marie->stat); reset_counters(&pauline->stat); /*make sure the record file doesn't already exists, otherwise this test will append new samples to it*/ unlink(recordpath); /*caller uses files instead of soundcard in order to avoid mixing soundcard input with file played using call's player*/ linphone_core_use_files(marie->lc,TRUE); linphone_core_set_play_file(marie->lc,NULL); /*callee is recording and plays file*/ linphone_core_use_files(pauline->lc,TRUE); linphone_core_set_play_file(pauline->lc,NULL); linphone_core_set_record_file(pauline->lc,recordpath); BC_ASSERT_TRUE((call_ok=call(marie,pauline))); if (!call_ok) goto end; player=linphone_call_get_player(linphone_core_get_current_call(marie->lc)); cbs = linphone_player_get_callbacks(player); linphone_player_cbs_set_eof_reached(cbs, on_eof); linphone_player_cbs_set_user_data(cbs, marie); BC_ASSERT_PTR_NOT_NULL(player); if (player){ BC_ASSERT_EQUAL(linphone_player_open(player,hellopath),0, int, "%d"); BC_ASSERT_EQUAL(linphone_player_start(player),0, int, "%d"); } /* This assert should be modified to be at least as long as the WAV file */ BC_ASSERT_TRUE(wait_for_until(pauline->lc,marie->lc,&marie->stat.number_of_player_eof,1,10000)); /*wait one second more for transmission to be fully ended (transmission time + jitter buffer)*/ wait_for_until(pauline->lc,marie->lc,NULL,0,1000); end_call(marie, pauline); /*cannot run on iphone simulator because locks main loop beyond permitted time (should run on another thread) */ BC_ASSERT_EQUAL(ms_audio_diff(hellopath,recordpath,&similar,&audio_cmp_params,NULL,NULL), 0, int, "%d"); if (similar>=threshold) break; } BC_ASSERT_GREATER(similar, threshold, double, "%g"); BC_ASSERT_LOWER(similar, 1.0, double, "%g"); if (similar >= threshold && similar <= 1.0) { remove(recordpath); } end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); ms_free(recordpath); ms_free(hellopath); } static void call_with_mkv_file_player(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphonePlayer *player; char *hellomkv; char *hellowav; char *recordpath; bool_t call_ok; #if !defined(__arm__) && !defined(__arm64__) && !TARGET_IPHONE_SIMULATOR && !defined(__ANDROID__) double similar=0.0; const double threshold = 0.9; #define DO_AUDIO_CMP #endif hellowav = bc_tester_res("sounds/hello8000_mkv_ref.wav"); hellomkv = bc_tester_res("sounds/hello8000.mkv"); if (!linphone_core_file_format_supported(marie->lc,"mkv")){ ms_warning("Test skipped, no mkv support."); goto end; } recordpath = bc_tester_file("record-call_with_mkv_file_player.wav"); /*make sure the record file doesn't already exists, otherwise this test will append new samples to it*/ unlink(recordpath); /*caller uses files instead of soundcard in order to avoid mixing soundcard input with file played using call's player*/ linphone_core_use_files(marie->lc,TRUE); linphone_core_set_play_file(marie->lc,NULL); /*callee is recording and plays file*/ linphone_core_use_files(pauline->lc,TRUE); linphone_core_set_play_file(pauline->lc,hellowav); /*just to send something but we are not testing what is sent by pauline*/ linphone_core_set_record_file(pauline->lc,recordpath); BC_ASSERT_TRUE((call_ok=call(marie,pauline))); if (!call_ok) goto end; player=linphone_call_get_player(linphone_core_get_current_call(marie->lc)); BC_ASSERT_PTR_NOT_NULL(player); if (player){ LinphonePlayerCbs *cbs = linphone_player_get_callbacks(player); linphone_player_cbs_set_eof_reached(cbs, on_eof); linphone_player_cbs_set_user_data(cbs, marie); int res = linphone_player_open(player,hellomkv); //if(!ms_filter_codec_supported("opus")) { if(!ms_factory_codec_supported(marie->lc->factory, "opus") && !ms_factory_codec_supported(pauline->lc->factory, "opus")){ BC_ASSERT_EQUAL(res, -1, int, "%d"); end_call(marie, pauline); goto end; } BC_ASSERT_EQUAL(res, 0, int, "%d"); BC_ASSERT_EQUAL(linphone_player_start(player),0,int,"%d"); BC_ASSERT_TRUE(wait_for_until(pauline->lc,marie->lc,&marie->stat.number_of_player_eof,1,12000)); linphone_player_close(player); /*wait for one second more so that last RTP packets can arrive*/ wait_for_until(pauline->lc,marie->lc,NULL,0,1000); } end_call(marie, pauline); #ifdef DO_AUDIO_CMP BC_ASSERT_EQUAL(ms_audio_diff(hellowav,recordpath,&similar,&audio_cmp_params,NULL,NULL),0,int,"%d"); BC_ASSERT_GREATER(similar,threshold,double,"%f"); BC_ASSERT_LOWER(similar,1.0,double,"%f"); if(similar>threshold && similar<=1.0) { remove(recordpath); } #else /*inter-correlation process is too much CPU consuming ending in a 20 minutes test on arm...*/ remove(recordpath); #endif ms_free(recordpath); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); ms_free(hellomkv); ms_free(hellowav); } static void _call_base_with_configfile(LinphoneMediaEncryption mode, bool_t enable_video,bool_t enable_relay,LinphoneFirewallPolicy policy,bool_t enable_tunnel, const char *marie_rc, const char *pauline_rc, bool_t plays_nothing) { LinphoneCoreManager* marie = linphone_core_manager_new(marie_rc); LinphoneCoreManager* pauline = linphone_core_manager_new(pauline_rc); bool_t call_ok; // important: VP8 has really poor performances with the mire camera, at least // on iOS - so when ever h264 is available, let's use it instead if (linphone_core_find_payload_type(pauline->lc,"h264", -1, -1)!=NULL) { disable_all_video_codecs_except_one(pauline->lc,"h264"); disable_all_video_codecs_except_one(marie->lc,"h264"); } linphone_core_set_video_device(pauline->lc,liblinphone_tester_mire_id); linphone_core_set_video_device(marie->lc,liblinphone_tester_mire_id); if (plays_nothing){ /*This case was for trying to replicate an issue because * zrtp_iterate() was only called when packets are received, which * creates a big problem because no retransmission of HELLO packet will occur * if the remote sends nothing. * However it is not possible to forcibly loose the hello packet, even with network simulator. * If retransmissions fail, this test will fail from time to time*/ linphone_core_use_files(marie->lc, TRUE); linphone_core_set_play_file(marie->lc, NULL); linphone_core_set_play_file(pauline->lc, NULL); linphone_core_set_media_encryption_mandatory(pauline->lc, TRUE); linphone_core_set_media_encryption_mandatory(marie->lc, TRUE); } if (enable_relay) { linphone_core_set_user_agent(marie->lc,"Natted Linphone",NULL); linphone_core_set_user_agent(pauline->lc,"Natted Linphone",NULL); } if (enable_tunnel) { int i; LinphoneTunnelConfig * tunnel_config = linphone_tunnel_config_new(); linphone_tunnel_config_set_host(tunnel_config, "tunnel.linphone.org"); linphone_tunnel_config_set_port(tunnel_config, 443); linphone_tunnel_add_server(linphone_core_get_tunnel(marie->lc),tunnel_config); linphone_tunnel_enable_sip(linphone_core_get_tunnel(marie->lc),FALSE); linphone_tunnel_set_mode(linphone_core_get_tunnel(marie->lc),LinphoneTunnelModeEnable); for (i=0;i<100;i++) { if (linphone_tunnel_connected(linphone_core_get_tunnel(marie->lc))) { linphone_core_iterate(marie->lc); break; } linphone_core_iterate(marie->lc); ms_usleep(20000); } BC_ASSERT_TRUE(linphone_tunnel_connected(linphone_core_get_tunnel(marie->lc))); linphone_tunnel_config_unref(tunnel_config); } if (linphone_core_media_encryption_supported(marie->lc,mode)) { linphone_core_set_media_encryption(marie->lc,mode); linphone_core_set_media_encryption(pauline->lc,mode); if (mode==LinphoneMediaEncryptionDTLS) { /* for DTLS we must access certificates or at least have a directory to store them */ char *path = bc_tester_file("certificates-marie"); marie->lc->user_certificates_path = ms_strdup(path); bc_free(path); path = bc_tester_file("certificates-pauline"); pauline->lc->user_certificates_path = ms_strdup(path); bc_free(path); belle_sip_mkdir(marie->lc->user_certificates_path); belle_sip_mkdir(pauline->lc->user_certificates_path); } linphone_core_set_firewall_policy(marie->lc,policy); linphone_core_set_firewall_policy(pauline->lc,policy); BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (!call_ok) goto end; /* if caller set ZRTP or (callee set ZRTP and caller has no encryption requested), ZRTP shall take place, wait for the SAS */ if ((linphone_core_get_media_encryption(pauline->lc) == LinphoneMediaEncryptionZRTP) || ((linphone_core_get_media_encryption(marie->lc) == LinphoneMediaEncryptionZRTP) && (linphone_core_get_media_encryption(pauline->lc) == LinphoneMediaEncryptionNone))) { /*wait for SAS*/ int i; LinphoneCall *pauline_call; LinphoneCall *marie_call; const char *pauline_token = NULL; const char *marie_token = NULL; for (i=0;i<100;i++) { pauline_call = linphone_core_get_current_call(pauline->lc); marie_call = linphone_core_get_current_call(marie->lc); if (!pauline_call || !marie_call){ /*if one of the two calls was disapeering, don't crash, but report it*/ BC_ASSERT_PTR_NOT_NULL(pauline_call); BC_ASSERT_PTR_NOT_NULL(marie_call); break; } pauline_token = linphone_call_get_authentication_token(pauline_call); marie_token = linphone_call_get_authentication_token(marie_call); if (pauline_token && marie_token) break; linphone_core_iterate(marie->lc); linphone_core_iterate(pauline->lc); ms_usleep(20000); } BC_ASSERT_PTR_NOT_NULL(pauline_token); BC_ASSERT_PTR_NOT_NULL(marie_token); if (marie_token && pauline_token){ BC_ASSERT_STRING_EQUAL(pauline_token, marie_token); BC_ASSERT_TRUE(strlen(pauline_token)>0); BC_ASSERT_TRUE(strlen(marie_token)>0); } if (!plays_nothing) liblinphone_tester_check_rtcp(pauline,marie); } if (policy == LinphonePolicyUseIce){ BC_ASSERT_TRUE(check_ice(pauline,marie,enable_tunnel?LinphoneIceStateReflexiveConnection:LinphoneIceStateHostConnection)); wait_for_until(marie->lc, pauline->lc, NULL, 0, 2000);/*fixme to workaround a crash*/ } #ifdef VIDEO_ENABLED if (enable_video) { if (linphone_core_video_supported(marie->lc)) { BC_ASSERT_TRUE(request_video(pauline,marie, TRUE)); if (policy == LinphonePolicyUseIce){ BC_ASSERT_TRUE(check_ice(pauline, marie, enable_tunnel ? LinphoneIceStateReflexiveConnection : LinphoneIceStateHostConnection)); } liblinphone_tester_check_rtcp(marie,pauline); } else { ms_warning ("not tested because video not available"); } } #endif end_call(marie, pauline); } else { ms_warning ("not tested because %s not available", linphone_media_encryption_to_string(mode)); } end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } void call_base_with_configfile(LinphoneMediaEncryption mode, bool_t enable_video,bool_t enable_relay,LinphoneFirewallPolicy policy,bool_t enable_tunnel, const char *marie_rc, const char *pauline_rc){ _call_base_with_configfile(mode, enable_video, enable_relay, policy, enable_tunnel, marie_rc, pauline_rc, FALSE); } void call_base(LinphoneMediaEncryption mode, bool_t enable_video,bool_t enable_relay,LinphoneFirewallPolicy policy,bool_t enable_tunnel) { call_base_with_configfile(mode, enable_video, enable_relay, policy, enable_tunnel, "marie_rc", "pauline_tcp_rc"); } static void srtp_ice_call(void) { call_base(LinphoneMediaEncryptionSRTP,FALSE,FALSE,LinphonePolicyUseIce,FALSE); } static void zrtp_ice_call(void) { call_base(LinphoneMediaEncryptionZRTP,FALSE,FALSE,LinphonePolicyUseIce,FALSE); } static void zrtp_silent_call(void) { _call_base_with_configfile(LinphoneMediaEncryptionZRTP,FALSE,TRUE,LinphonePolicyNoFirewall,FALSE, "marie_rc", "pauline_tcp_rc", TRUE); } static void zrtp_ice_call_with_relay(void) { call_base(LinphoneMediaEncryptionZRTP,FALSE,TRUE,LinphonePolicyUseIce,FALSE); } static void dtls_ice_call_with_relay(void) { call_base(LinphoneMediaEncryptionDTLS,FALSE,TRUE,LinphonePolicyUseIce,FALSE); } static void early_media_call(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_early_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bool_t call_ok; BC_ASSERT_TRUE(call_ok=call(pauline,marie)); if (!call_ok) goto end; BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallIncomingEarlyMedia,1, int, "%d"); BC_ASSERT_EQUAL(pauline->stat.number_of_LinphoneCallOutgoingEarlyMedia,1, int, "%d"); wait_for_until(pauline->lc,marie->lc,NULL,0,1000); /*added because a bug related to early-media caused the Connected state to be reached two times*/ BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallConnected,1, int, "%d"); end_call(pauline, marie); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void early_media_call_with_ice(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_early_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall *marie_call, *pauline_call; bctbx_list_t *lcs = NULL; lcs = bctbx_list_append(lcs, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); /*in this test, pauline has ICE activated, marie not, but marie proposes early media. * We want to check that ICE processing is not disturbing early media*/ linphone_core_set_firewall_policy(pauline->lc, LinphonePolicyUseIce); pauline_call = linphone_core_invite_address(pauline->lc, marie->identity); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallIncomingReceived,1,3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallIncomingEarlyMedia,1,3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallOutgoingEarlyMedia,1,1000)); BC_ASSERT_TRUE(pauline_call->all_muted); wait_for_until(pauline->lc,marie->lc,NULL,0,1000); marie_call = linphone_core_get_current_call(marie->lc); if (!marie_call) goto end; linphone_call_accept(marie_call); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallConnected,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_LinphoneCallStreamsRunning,1,3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning,1,3000)); BC_ASSERT_FALSE(pauline_call->all_muted); end_call(marie, pauline); end: bctbx_list_free(lcs); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void early_media_call_with_ringing_base(bool_t network_change){ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new("pauline_tcp_rc"); bctbx_list_t* lcs = NULL; LinphoneCall* marie_call; LinphoneCallLog *marie_call_log; uint64_t connected_time=0; uint64_t ended_time=0; int dummy=0; lcs = bctbx_list_append(lcs,marie->lc); lcs = bctbx_list_append(lcs,pauline->lc); /* Marie calls Pauline, and after the call has rung, transitions to an early_media session */ marie_call = linphone_core_invite_address(marie->lc, pauline->identity); marie_call_log = linphone_call_get_call_log(marie_call); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallIncomingReceived,1,3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallOutgoingRinging,1,1000)); if (linphone_core_inc_invite_pending(pauline->lc)) { /* send a 183 to initiate the early media */ linphone_call_accept_early_media(linphone_core_get_current_call(pauline->lc)); BC_ASSERT_TRUE( wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallIncomingEarlyMedia,1,2000) ); BC_ASSERT_TRUE( wait_for_list(lcs, &marie->stat.number_of_LinphoneCallOutgoingEarlyMedia,1,2000) ); BC_ASSERT_TRUE(marie_call->all_muted); liblinphone_tester_check_rtcp(marie, pauline); /* this is a hack to simulate an incoming OK with a different IP address * in the 'c' SDP field. */ if (network_change) { marie_call->localdesc_changed |= SAL_MEDIA_DESCRIPTION_NETWORK_CHANGED; } linphone_call_accept(linphone_core_get_current_call(pauline->lc)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallConnected, 1,1000)); connected_time=ms_get_cur_time_ms(); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, 1,1000)); BC_ASSERT_PTR_EQUAL(marie_call, linphone_core_get_current_call(marie->lc)); BC_ASSERT_FALSE(marie_call->all_muted); liblinphone_tester_check_rtcp(marie, pauline); /*just to have a call duration !=0*/ wait_for_list(lcs,&dummy,1,2000); end_call(pauline, marie); ended_time=ms_get_cur_time_ms(); BC_ASSERT_LOWER( labs((long)((linphone_call_log_get_duration(marie_call_log)*1000) - (int64_t)(ended_time - connected_time))), 1000, long, "%ld"); bctbx_list_free(lcs); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void early_media_call_with_ringing(void) { early_media_call_with_ringing_base(FALSE); } static void early_media_call_with_ringing_and_network_changing(void) { early_media_call_with_ringing_base(TRUE); } static void early_media_call_with_update_base(bool_t media_change){ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bctbx_list_t* lcs = NULL; LinphoneCall *marie_call, *pauline_call; LinphoneCallParams *pauline_params; lcs = bctbx_list_append(lcs,marie->lc); lcs = bctbx_list_append(lcs,pauline->lc); if (media_change) { disable_all_audio_codecs_except_one(marie->lc,"pcmu",-1); disable_all_audio_codecs_except_one(pauline->lc,"pcmu",-1); } /* Marie calls Pauline, and after the call has rung, transitions to an early_media session */ marie_call = linphone_core_invite_address(marie->lc, pauline->identity); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallIncomingReceived,1,5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallOutgoingRinging,1,5000)); pauline_call = linphone_core_get_current_call(pauline->lc); if (!pauline_call) goto end; /* send a 183 to initiate the early media */ linphone_call_accept_early_media(pauline_call); BC_ASSERT_TRUE( wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallIncomingEarlyMedia,1,1000) ); BC_ASSERT_TRUE( wait_for_list(lcs, &marie->stat.number_of_LinphoneCallOutgoingEarlyMedia,1,5000) ); BC_ASSERT_TRUE(marie_call->all_muted); pauline_params = linphone_call_params_copy(linphone_call_get_current_params(pauline_call)); if (media_change) { disable_all_audio_codecs_except_one(marie->lc,"pcma",-1); disable_all_audio_codecs_except_one(pauline->lc,"pcma",-1); } #define UPDATED_SESSION_NAME "nouveau nom de session" linphone_call_params_set_session_name(pauline_params,UPDATED_SESSION_NAME); linphone_call_update(pauline_call, pauline_params); linphone_call_params_unref(pauline_params); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallEarlyUpdating,1,2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallEarlyUpdatedByRemote,1,2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallOutgoingEarlyMedia,1,2000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallIncomingEarlyMedia,1,2000)); BC_ASSERT_TRUE(marie_call->all_muted); /*just to wait 2s*/ liblinphone_tester_check_rtcp(marie, pauline); BC_ASSERT_STRING_EQUAL( linphone_call_params_get_session_name(linphone_call_get_remote_params(marie_call)) , UPDATED_SESSION_NAME); linphone_call_accept(linphone_core_get_current_call(pauline->lc)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallConnected, 1,1000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, 1,1000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallConnected, 1,1000)); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1,1000)); BC_ASSERT_FALSE(marie_call->all_muted); liblinphone_tester_check_rtcp(marie, pauline); end_call(pauline, marie); end: bctbx_list_free(lcs); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void early_media_call_with_session_update(void){ early_media_call_with_update_base(FALSE); } static void early_media_call_with_codec_update(void){ early_media_call_with_update_base(TRUE); } static void check_call_state(LinphoneCoreManager* mgr,LinphoneCallState state) { BC_ASSERT_PTR_NOT_NULL(linphone_core_get_current_call(mgr->lc)); if (linphone_core_get_current_call(mgr->lc)) BC_ASSERT_EQUAL(linphone_call_get_state(linphone_core_get_current_call(mgr->lc)),state, int, "%d"); } static void call_established_with_rejected_info(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneInfoMessage *im1; LinphoneInfoMessage *im2; int dummy=0; bool_t call_ok=FALSE; BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (call_ok){ sal_enable_unconditional_answer(marie->lc->sal,TRUE); im1 = linphone_core_create_info_message(pauline->lc); linphone_call_send_info_message(linphone_core_get_current_call(pauline->lc),im1); wait_for_until(marie->lc,pauline->lc,&dummy,1,1000); /*just to sleep while iterating 1s*/ linphone_info_message_unref(im1); sal_enable_unconditional_answer(marie->lc->sal,FALSE); im2 = linphone_core_create_info_message(pauline->lc); linphone_call_send_info_message(linphone_core_get_current_call(pauline->lc),im2); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_inforeceived,1)); BC_ASSERT_EQUAL(marie->stat.number_of_inforeceived,1, int, "%d"); linphone_info_message_unref(im2); check_call_state(pauline,LinphoneCallStreamsRunning); check_call_state(marie,LinphoneCallStreamsRunning); end_call(pauline, marie); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_established_with_complex_rejected_operation(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bool_t call_ok=FALSE; LinphoneCallParams *params; LinphoneInfoMessage *info; BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (call_ok){ BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,1)); linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMU",8000,1),FALSE); /*disable PCMU*/ linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMA",8000,1),TRUE); /*enable PCMA*/ linphone_core_enable_payload_type(marie->lc,linphone_core_find_payload_type(marie->lc,"PCMU",8000,1),FALSE); /*disable PCMU*/ linphone_core_enable_payload_type(marie->lc,linphone_core_find_payload_type(marie->lc,"PCMA",8000,1),TRUE); /*enable PCMA*/ /*just to authenticate marie*/ info = linphone_core_create_info_message(marie->lc); linphone_call_send_info_message(linphone_core_get_current_call(marie->lc), info); linphone_info_message_unref(info); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_inforeceived,1)); BC_ASSERT_EQUAL(pauline->stat.number_of_inforeceived,1, int, "%d"); /*to give time for 200ok to arrive*/ wait_for_until(marie->lc,pauline->lc,NULL,0,1000); linphone_call_update(linphone_core_get_current_call(pauline->lc),linphone_call_get_current_params(linphone_core_get_current_call(pauline->lc))); linphone_call_update(linphone_core_get_current_call(marie->lc),linphone_call_get_current_params(linphone_core_get_current_call(marie->lc))); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_EQUAL(linphone_call_get_reason(linphone_core_get_current_call(pauline->lc)),LinphoneReasonTemporarilyUnavailable, int, "%d"); BC_ASSERT_EQUAL(linphone_call_get_reason(linphone_core_get_current_call(pauline->lc)),LinphoneReasonTemporarilyUnavailable, int, "%d"); check_call_state(pauline,LinphoneCallStreamsRunning); check_call_state(marie,LinphoneCallStreamsRunning); linphone_call_update(linphone_core_get_current_call(pauline->lc),linphone_call_get_current_params(linphone_core_get_current_call(pauline->lc))); info = linphone_core_create_info_message(marie->lc); linphone_call_send_info_message(linphone_core_get_current_call(marie->lc), info); linphone_info_message_unref(info); params=linphone_core_create_call_params(marie->lc,linphone_core_get_current_call(marie->lc)); sal_enable_pending_trans_checking(marie->lc->sal,FALSE); /*to allow // transactions*/ linphone_core_enable_payload_type(marie->lc,linphone_core_find_payload_type(marie->lc,"PCMU",8000,1),TRUE); linphone_core_enable_payload_type(marie->lc,linphone_core_find_payload_type(marie->lc,"PCMA",8000,1),FALSE); linphone_call_update(linphone_core_get_current_call(marie->lc),params); linphone_call_params_unref(params); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,3)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,3)); BC_ASSERT_EQUAL(linphone_call_get_reason(linphone_core_get_current_call(pauline->lc)),LinphoneReasonTemporarilyUnavailable, int, "%d"); BC_ASSERT_EQUAL(linphone_call_get_reason(linphone_core_get_current_call(pauline->lc)),LinphoneReasonTemporarilyUnavailable, int, "%d"); end_call(pauline, marie); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_established_with_rejected_info_during_reinvite(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bool_t call_ok=FALSE; BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (call_ok){ LinphoneInfoMessage *info; BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,1)); linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMU",8000,1),FALSE); /*disable PCMU*/ linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMA",8000,1),TRUE); /*enable PCMA*/ linphone_core_enable_payload_type(marie->lc,linphone_core_find_payload_type(marie->lc,"PCMU",8000,1),FALSE); /*disable PCMU*/ linphone_core_enable_payload_type(marie->lc,linphone_core_find_payload_type(marie->lc,"PCMA",8000,1),TRUE); /*enable PCMA*/ /*just to authenticate marie*/ info = linphone_core_create_info_message(marie->lc); linphone_call_send_info_message(linphone_core_get_current_call(marie->lc), info); linphone_info_message_unref(info); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_inforeceived,1)); BC_ASSERT_EQUAL(pauline->stat.number_of_inforeceived,1, int, "%d"); /*to give time for 200ok to arrive*/ wait_for_until(marie->lc,pauline->lc,NULL,0,1000); //sal_enable_pending_trans_checking(marie->lc->sal,FALSE); /*to allow // transactions*/ info = linphone_core_create_info_message(marie->lc); linphone_call_send_info_message(linphone_core_get_current_call(marie->lc),info); linphone_info_message_unref(info); //sal_set_send_error(marie->lc->sal, -1); /*to avoid 491 pending to be sent*/ linphone_call_update(linphone_core_get_current_call(pauline->lc),linphone_call_get_current_params(linphone_core_get_current_call(pauline->lc))); wait_for_until(pauline->lc,pauline->lc,NULL,0,2000); /*to avoid 491 pending to be sent to early*/ BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); end_call(pauline, marie); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_established_with_rejected_reinvite(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bool_t call_ok=FALSE; BC_ASSERT_TRUE(call_ok=call(pauline,marie)); if (call_ok){ linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMU",8000,1),FALSE); /*disable PCMU*/ linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMA",8000,1),TRUE); /*enable PCMA*/ linphone_call_update(linphone_core_get_current_call(pauline->lc),linphone_call_get_current_params(linphone_core_get_current_call(pauline->lc))); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_EQUAL(linphone_call_get_reason(linphone_core_get_current_call(pauline->lc)),LinphoneReasonNotAcceptable, int, "%d"); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallStreamsRunning,1, int, "%d"); check_call_state(pauline,LinphoneCallStreamsRunning); check_call_state(marie,LinphoneCallStreamsRunning); end_call(pauline, marie); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_established_with_rejected_incoming_reinvite(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bool_t call_ok=FALSE; BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (call_ok){ /*wait for ACK to be transmitted before going to reINVITE*/ wait_for_until(marie->lc,pauline->lc,NULL,0,1000); linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMU",8000,1),FALSE); /*disable PCMU*/ linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMA",8000,1),TRUE); /*enable PCMA*/ linphone_call_update(linphone_core_get_current_call(marie->lc),linphone_call_get_current_params(linphone_core_get_current_call(marie->lc))); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallUpdating,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_EQUAL(linphone_call_get_reason(linphone_core_get_current_call(marie->lc)),LinphoneReasonNotAcceptable, int, "%d"); BC_ASSERT_EQUAL(pauline->stat.number_of_LinphoneCallStreamsRunning,1, int, "%d"); check_call_state(pauline,LinphoneCallStreamsRunning); check_call_state(marie,LinphoneCallStreamsRunning); end_call(pauline, marie); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_redirect(void){ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCoreManager* laure = linphone_core_manager_new("laure_rc_udp"); bctbx_list_t* lcs = NULL; char *laure_url = NULL; LinphoneCall* marie_call; lcs = bctbx_list_append(lcs,marie->lc); lcs = bctbx_list_append(lcs,pauline->lc); lcs = bctbx_list_append(lcs,laure->lc); /* Marie calls Pauline, which will redirect the call to Laure via a 302 */ marie_call = linphone_core_invite_address(marie->lc, pauline->identity); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallIncomingReceived,1,6000)); if (linphone_core_get_current_call(pauline->lc)){ laure_url = linphone_address_as_string(laure->identity); linphone_call_redirect(linphone_core_get_current_call(pauline->lc), laure_url); ms_free(laure_url); /* laure should be ringing now */ BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallIncomingReceived,1,6000)); /* pauline should have ended the call */ BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallEnd,1,1000)); /* the call should still be ringing on marie's side */ BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallOutgoingRinging, 1, int, "%i"); linphone_call_accept(linphone_core_get_current_call(laure->lc)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, 1,5000)); BC_ASSERT_TRUE(wait_for_list(lcs, &laure->stat.number_of_LinphoneCallStreamsRunning, 1,5000)); BC_ASSERT_PTR_EQUAL(marie_call, linphone_core_get_current_call(marie->lc)); liblinphone_tester_check_rtcp(marie, laure); end_call(laure, marie); } bctbx_list_free(lcs); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(laure); } static void call_established_with_rejected_reinvite_with_error_base(bool_t trans_pending) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bool_t call_ok=TRUE; int result; BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (call_ok){ linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMA",8000,1),TRUE); /*add PCMA*/ if (trans_pending) { LinphoneInfoMessage * info = linphone_core_create_info_message(pauline->lc); linphone_call_send_info_message(linphone_core_get_current_call(pauline->lc),info); linphone_info_message_unref(info); } else sal_enable_unconditional_answer(marie->lc->sal,TRUE); result = linphone_call_update(linphone_core_get_current_call(pauline->lc),linphone_call_get_current_params(linphone_core_get_current_call(pauline->lc))); if (trans_pending) BC_ASSERT_NOT_EQUAL(result,0, int, "%d"); else BC_ASSERT_EQUAL(result,0,int, "%d"); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_EQUAL(linphone_call_get_reason(linphone_core_get_current_call(pauline->lc)),LinphoneReasonTemporarilyUnavailable, int, "%d"); /*might be change later*/ BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallStreamsRunning,1, int, "%d"); check_call_state(pauline,LinphoneCallStreamsRunning); check_call_state(marie,LinphoneCallStreamsRunning); if (!trans_pending) sal_enable_unconditional_answer(marie->lc->sal,FALSE); end_call(pauline, marie); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_established_with_rejected_reinvite_with_error(void) { call_established_with_rejected_reinvite_with_error_base(FALSE); } static void call_established_with_rejected_reinvite_with_trans_pending_error(void) { call_established_with_rejected_reinvite_with_error_base(TRUE); } static void call_rejected_because_wrong_credentials_with_params(const char* user_agent,bool_t enable_auth_req_cb) { LinphoneCoreManager* marie = linphone_core_manager_new( "marie_rc"); LinphoneAuthInfo* good_auth_info=linphone_auth_info_clone(linphone_core_find_auth_info(marie->lc,NULL,linphone_address_get_username(marie->identity),NULL)); LinphoneAuthInfo* wrong_auth_info=linphone_auth_info_clone(good_auth_info); bool_t result=FALSE; linphone_auth_info_set_passwd(wrong_auth_info,"passecretdutout"); linphone_auth_info_set_ha1(wrong_auth_info, NULL); linphone_core_clear_all_auth_info(marie->lc); if (user_agent) { linphone_core_set_user_agent(marie->lc,user_agent,NULL); } if (!enable_auth_req_cb) { ((VTableReference*)(marie->lc->vtable_refs->data))->cbs->vtable->auth_info_requested=NULL; linphone_core_add_auth_info(marie->lc,wrong_auth_info); } BC_ASSERT_PTR_NOT_NULL(linphone_core_invite_address(marie->lc,marie->identity)); result=wait_for(marie->lc,marie->lc,&marie->stat.number_of_auth_info_requested,1); if (enable_auth_req_cb) { BC_ASSERT_TRUE(result); /*automatically re-inititae the call*/ linphone_core_add_auth_info(marie->lc,wrong_auth_info); } BC_ASSERT_TRUE(wait_for(marie->lc,marie->lc,&marie->stat.number_of_LinphoneCallError,1)); if (enable_auth_req_cb) { BC_ASSERT_EQUAL(marie->stat.number_of_auth_info_requested,2, int, "%d"); } /*to make sure unregister will work*/ linphone_core_clear_all_auth_info(marie->lc); linphone_core_add_auth_info(marie->lc,good_auth_info); linphone_auth_info_unref(good_auth_info); linphone_auth_info_unref(wrong_auth_info); linphone_core_manager_destroy(marie); } static void call_rejected_because_wrong_credentials(void) { call_rejected_because_wrong_credentials_with_params(NULL,TRUE); } static void call_rejected_without_403_because_wrong_credentials(void) { call_rejected_because_wrong_credentials_with_params("tester-no-403",TRUE); } static void call_rejected_without_403_because_wrong_credentials_no_auth_req_cb(void) { call_rejected_because_wrong_credentials_with_params("tester-no-403",FALSE); } void check_media_direction(LinphoneCoreManager* mgr, LinphoneCall *call, bctbx_list_t* lcs,LinphoneMediaDirection audio_dir, LinphoneMediaDirection video_dir) { BC_ASSERT_PTR_NOT_NULL(call); if (call) { const LinphoneCallParams *params; wait_for_list(lcs,NULL,0,5000); /*on some device, it may take 3 to 4s to get audio from mic*/ params = linphone_call_get_current_params(call); #ifdef VIDEO_ENABLED if (video_dir != LinphoneMediaDirectionInvalid){ int current_recv_iframe = mgr->stat.number_of_IframeDecoded; int expected_recv_iframe=0; LinphoneCallStats *stats = linphone_call_get_video_stats(call); if (video_dir != LinphoneMediaDirectionInactive){ BC_ASSERT_TRUE(linphone_call_params_video_enabled(params)); BC_ASSERT_EQUAL(linphone_call_params_get_video_direction(params), video_dir, int, "%d"); linphone_call_set_next_video_frame_decoded_callback(call,linphone_call_iframe_decoded_cb,mgr->lc); linphone_call_send_vfu_request(call); } switch (video_dir) { case LinphoneMediaDirectionInactive: BC_ASSERT_LOWER((int)stats->upload_bandwidth, 5, int, "%i"); break; case LinphoneMediaDirectionSendOnly: expected_recv_iframe = 0; BC_ASSERT_LOWER((int)stats->download_bandwidth, 5, int, "%i"); break; case LinphoneMediaDirectionRecvOnly: BC_ASSERT_LOWER((int)stats->upload_bandwidth, 5, int, "%i"); BCTBX_NO_BREAK; /*intentionally no break*/ case LinphoneMediaDirectionSendRecv: expected_recv_iframe = 1; break; default: break; } linphone_call_stats_unref(stats); BC_ASSERT_TRUE(wait_for_list(lcs, &mgr->stat.number_of_IframeDecoded,current_recv_iframe + expected_recv_iframe,10000)); } #endif if (audio_dir != LinphoneMediaDirectionInvalid){ BC_ASSERT_EQUAL(linphone_call_params_get_audio_direction(params), audio_dir, int, "%d"); switch (audio_dir) { case LinphoneMediaDirectionInactive: BC_ASSERT_LOWER(linphone_core_manager_get_mean_audio_up_bw(mgr), 5, int, "%i"); BC_ASSERT_LOWER(linphone_core_manager_get_mean_audio_down_bw(mgr), 5, int, "%i"); break; case LinphoneMediaDirectionSendOnly: BC_ASSERT_GREATER(linphone_core_manager_get_mean_audio_up_bw(mgr), 70, int, "%i"); break; case LinphoneMediaDirectionRecvOnly: BC_ASSERT_LOWER(linphone_core_manager_get_mean_audio_up_bw(mgr), 5, int, "%i"); break; case LinphoneMediaDirectionSendRecv: BC_ASSERT_GREATER(linphone_core_manager_get_mean_audio_down_bw(mgr), 70, int, "%i"); BC_ASSERT_GREATER(linphone_core_manager_get_mean_audio_up_bw(mgr), 70, int, "%i"); break; default: break; } } } } void record_call(const char *filename, bool_t enableVideo, const char *video_codec) { LinphoneCoreManager *marie = NULL; LinphoneCoreManager *pauline = NULL; LinphoneCallParams *marieParams = NULL; LinphoneCallParams *paulineParams = NULL; LinphoneCall *callInst = NULL; const char **formats, *format; char *filepath; int dummy=0, i; bool_t call_succeeded = FALSE; marie = linphone_core_manager_new("marie_h264_rc"); pauline = linphone_core_manager_new("pauline_h264_rc"); // important: VP8 has really poor performances with the mire camera, at least // on iOS - so when ever h264 is available, let's use it instead if (linphone_core_find_payload_type(pauline->lc,"h264", -1, -1)!=NULL) { disable_all_video_codecs_except_one(pauline->lc,"h264"); disable_all_video_codecs_except_one(marie->lc,"h264"); } #if defined(HAVE_OPENH264) && defined(__ANDROID__) libmsopenh264_init(linphone_core_get_ms_factory(marie->lc)); linphone_core_reload_ms_plugins(marie->lc, NULL); libmsopenh264_init(linphone_core_get_ms_factory(pauline->lc)); linphone_core_reload_ms_plugins(pauline->lc, NULL); #endif marieParams = linphone_core_create_call_params(marie->lc, NULL); paulineParams = linphone_core_create_call_params(pauline->lc, NULL); #ifdef VIDEO_ENABLED linphone_core_set_video_device(pauline->lc, liblinphone_tester_mire_id); if(enableVideo) { if(linphone_core_find_payload_type(marie->lc, video_codec, -1, -1) && linphone_core_find_payload_type(pauline->lc, video_codec, -1, -1)) { linphone_call_params_enable_video(marieParams, TRUE); linphone_call_params_enable_video(paulineParams, TRUE); disable_all_video_codecs_except_one(marie->lc, video_codec); disable_all_video_codecs_except_one(pauline->lc, video_codec); } else { ms_warning("call_recording(): the H264 payload has not been found. Only sound will be recorded"); } } #endif formats = linphone_core_get_supported_file_formats(marie->lc); for(i=0, format = formats[0]; format != NULL; i++, format = formats[i]) { char* totalname = ms_strdup_printf("%s.%s", filename, format); filepath = bc_tester_file(totalname); ms_free(totalname); remove(filepath); linphone_call_params_set_record_file(marieParams, filepath); BC_ASSERT_TRUE(call_succeeded = call_with_params(marie, pauline, marieParams, paulineParams)); BC_ASSERT_PTR_NOT_NULL(callInst = linphone_core_get_current_call(marie->lc)); if ((call_succeeded == TRUE) && (callInst != NULL)) { ms_message("call_recording(): start recording into %s", filepath); linphone_call_start_recording(callInst); wait_for_until(marie->lc,pauline->lc,&dummy,1,5000); linphone_call_stop_recording(callInst); end_call(marie, pauline); BC_ASSERT_EQUAL(ortp_file_exist(filepath), 0, int, "%d"); } remove(filepath); ms_free(filepath); } linphone_call_params_unref(paulineParams); linphone_call_params_unref(marieParams); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void audio_call_recording_test(void) { record_call("recording", FALSE, NULL); } static void call_with_in_dialog_update(void) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; LinphoneCallParams *params; bool_t call_ok; marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); BC_ASSERT_TRUE(call_ok=call(pauline,marie)); if (!call_ok) goto end; liblinphone_tester_check_rtcp(marie,pauline); params=linphone_core_create_call_params(marie->lc,linphone_core_get_current_call(marie->lc)); params->no_user_consent=TRUE; linphone_call_update(linphone_core_get_current_call(marie->lc),params); linphone_call_params_unref(params); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallUpdating,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallUpdatedByRemote,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); end_call(marie,pauline); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_very_early_call_update(void) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; LinphoneCallParams *params; marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); linphone_core_invite_address(marie->lc,pauline->identity); BC_ASSERT_TRUE (wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallIncomingReceived,1)); BC_ASSERT_TRUE(linphone_core_inc_invite_pending(pauline->lc)); BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallOutgoingProgress,1, int, "%d"); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallOutgoingRinging,1)); BC_ASSERT_PTR_NOT_NULL(linphone_core_get_current_call_remote_address(pauline->lc)); if (linphone_core_get_current_call_remote_address(pauline->lc)) { linphone_call_accept(linphone_core_get_current_call(pauline->lc)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,1)); } if(linphone_core_get_current_call(pauline->lc)) { params=linphone_core_create_call_params(pauline->lc,linphone_core_get_current_call(pauline->lc)); linphone_call_update(linphone_core_get_current_call(pauline->lc),params); linphone_call_params_unref(params); } BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallUpdating,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallUpdatedByRemote,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); end_call(marie,pauline); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_in_dialog_codec_change_base(bool_t no_sdp) { int dummy=0; LinphoneCoreManager* marie; LinphoneCoreManager* pauline; LinphoneCallParams *params; bool_t call_ok; marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); BC_ASSERT_TRUE(call_ok=call(pauline,marie)); if (!call_ok) goto end; liblinphone_tester_check_rtcp(marie,pauline); params=linphone_core_create_call_params(marie->lc,linphone_core_get_current_call(marie->lc)); linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMU",8000,1),FALSE); /*disable PCMU*/ linphone_core_enable_payload_type(marie->lc,linphone_core_find_payload_type(marie->lc,"PCMU",8000,1),FALSE); /*disable PCMU*/ linphone_core_enable_payload_type(pauline->lc,linphone_core_find_payload_type(pauline->lc,"PCMA",8000,1),TRUE); /*enable PCMA*/ linphone_core_enable_payload_type(marie->lc,linphone_core_find_payload_type(marie->lc,"PCMA",8000,1),TRUE); /*enable PCMA*/ if (no_sdp) { linphone_core_enable_sdp_200_ack(marie->lc,TRUE); } linphone_call_update(linphone_core_get_current_call(marie->lc),params); linphone_call_params_unref(params); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallUpdating,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallUpdatedByRemote,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_STRING_EQUAL("PCMA",payload_type_get_mime(linphone_call_params_get_used_audio_codec(linphone_call_get_current_params(linphone_core_get_current_call(marie->lc))))); wait_for_until(marie->lc, pauline->lc, &dummy, 1, 5000); BC_ASSERT_GREATER(linphone_core_manager_get_max_audio_down_bw(marie),70,int,"%i"); BC_ASSERT_GREATER(linphone_core_manager_get_max_audio_down_bw(pauline),70,int,"%i"); end_call(marie,pauline); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_in_dialog_codec_change(void) { call_with_in_dialog_codec_change_base(FALSE); } static void call_with_in_dialog_codec_change_no_sdp(void) { call_with_in_dialog_codec_change_base(TRUE); } static void call_with_custom_supported_tags(void) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; const LinphoneCallParams *remote_params; const char *recv_supported; LinphoneCall *pauline_call; marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); linphone_core_add_supported_tag(marie->lc,"pouet-tag"); linphone_core_add_supported_tag(marie->lc,"truc-tag"); linphone_core_add_supported_tag(marie->lc,"machin-tag"); linphone_core_invite_address(marie->lc, pauline->identity); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived,1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallOutgoingRinging,1)); pauline_call = linphone_core_get_current_call(pauline->lc); if (!pauline_call) goto end; remote_params=linphone_call_get_remote_params(pauline_call); recv_supported=linphone_call_params_get_custom_header(remote_params,"supported"); BC_ASSERT_PTR_NOT_NULL(recv_supported); if (recv_supported){ BC_ASSERT_PTR_NOT_NULL(strstr(recv_supported,"pouet-tag, truc-tag, machin-tag")); } end_call(marie,pauline); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_log_from_taken_from_p_asserted_id(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall *c1,*c2; LinphoneCallParams *params; const char* pauline_asserted_id ="\"Paupauche\" "; LinphoneAddress *pauline_asserted_id_addr = linphone_address_new(pauline_asserted_id); LpConfig *marie_lp; bool_t call_ok; params=linphone_core_create_call_params(pauline->lc, NULL); linphone_call_params_add_custom_header(params,"P-Asserted-Identity",pauline_asserted_id); /*fixme, should be able to add several time the same header linphone_call_params_add_custom_header(params,"P-Asserted-Identity","\"Paupauche\" ");*/ marie_lp = linphone_core_get_config(marie->lc); lp_config_set_int(marie_lp,"sip","call_logs_use_asserted_id_instead_of_from",1); BC_ASSERT_TRUE(call_ok=call_with_caller_params(pauline,marie,params)); if (!call_ok) goto end; c1=linphone_core_get_current_call(pauline->lc); c2=linphone_core_get_current_call(marie->lc); BC_ASSERT_PTR_NOT_NULL(c1); BC_ASSERT_PTR_NOT_NULL(c2); /*make sure remote identity is hidden*/ BC_ASSERT_TRUE(linphone_address_weak_equal(linphone_call_get_remote_address(c2),pauline_asserted_id_addr)); linphone_address_unref(pauline_asserted_id_addr); end_call(pauline, marie); end: linphone_call_params_unref(params); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void incoming_invite_with_invalid_sdp(void) { LinphoneCoreManager* caller = linphone_core_manager_new( "pauline_tcp_rc"); LinphoneCoreManager* callee = linphone_core_manager_new( "marie_rc"); LinphoneCallTestParams caller_test_params = {0}, callee_test_params = {0}; LinphoneCallLog *cl; const bctbx_list_t *logs; callee_test_params.sdp_simulate_error = TRUE; BC_ASSERT_FALSE(call_with_params2(caller,callee,&caller_test_params, &callee_test_params, FALSE)); BC_ASSERT_PTR_NULL(linphone_core_get_current_call(callee->lc)); BC_ASSERT_EQUAL(caller->stat.number_of_LinphoneCallError,1, int, "%d"); /*call will be drop before presented to the application, because it is invalid*/ BC_ASSERT_EQUAL(callee->stat.number_of_LinphoneCallIncomingReceived,0, int, "%d"); logs = linphone_core_get_call_logs(callee->lc); BC_ASSERT_EQUAL((int)bctbx_list_size(logs), 1, int, "%i"); if (logs){ const LinphoneErrorInfo *ei; cl = (LinphoneCallLog*)logs->data; BC_ASSERT_EQUAL(linphone_call_log_get_status(cl), LinphoneCallEarlyAborted, int, "%d"); BC_ASSERT_TRUE(linphone_call_log_get_start_date(cl) != 0); ei = linphone_call_log_get_error_info(cl); BC_ASSERT_PTR_NOT_NULL(ei); if (ei){ BC_ASSERT_EQUAL(linphone_error_info_get_reason(ei), LinphoneReasonNotAcceptable, int, "%d"); } } linphone_core_manager_destroy(callee); linphone_core_manager_destroy(caller); } static void outgoing_invite_with_invalid_sdp(void) { LinphoneCoreManager* caller = linphone_core_manager_new( "pauline_tcp_rc"); LinphoneCoreManager* callee = linphone_core_manager_new( "marie_rc"); LinphoneCallTestParams caller_test_params = {0}, callee_test_params = {0}; caller_test_params.sdp_simulate_error = TRUE; BC_ASSERT_FALSE(call_with_params2(caller,callee,&caller_test_params, &callee_test_params, FALSE)); BC_ASSERT_PTR_NULL(linphone_core_get_current_call(callee->lc)); BC_ASSERT_EQUAL(callee->stat.number_of_LinphoneCallIncomingReceived,1, int, "%d"); BC_ASSERT_EQUAL(caller->stat.number_of_LinphoneCallError,1, int, "%d"); // actually callee does not receive error, because it just get a BYE from the other part BC_ASSERT_EQUAL(callee->stat.number_of_LinphoneCallError,0, int, "%d"); BC_ASSERT_EQUAL(callee->stat.number_of_LinphoneCallEnd,1, int, "%d"); linphone_core_manager_destroy(callee); linphone_core_manager_destroy(caller); } static void call_with_paused_no_sdp_on_resume(void) { int dummy=0; LinphoneCoreManager* marie; LinphoneCoreManager* pauline; LinphoneCall* call_marie = NULL; LinphoneCallStats *stats; bool_t call_ok; marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); BC_ASSERT_TRUE(call_ok=call(pauline,marie)); if (!call_ok) goto end; liblinphone_tester_check_rtcp(marie,pauline); call_marie = linphone_core_get_current_call(marie->lc); BC_ASSERT_PTR_NOT_NULL(call_marie); ms_message("== Call is OK =="); /* the called party pause the call */ wait_for_until(pauline->lc, marie->lc, NULL, 5, 3000); linphone_call_pause(call_marie); ms_message("== Call pausing =="); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPausing,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&pauline->stat.number_of_LinphoneCallPausedByRemote,1)); BC_ASSERT_TRUE(wait_for(pauline->lc,marie->lc,&marie->stat.number_of_LinphoneCallPaused,1)); /*stay in pause a little while in order to generate traffic*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 2000); ms_message("== Call paused, marie call: %p ==", call_marie); linphone_core_enable_sdp_200_ack(marie->lc,TRUE); linphone_call_resume(call_marie); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); wait_for_until(marie->lc, pauline->lc, &dummy, 1, 3000); BC_ASSERT_GREATER(linphone_core_manager_get_max_audio_down_bw(marie),70,int,"%i"); stats = linphone_call_get_audio_stats(linphone_core_get_current_call(pauline->lc)); BC_ASSERT_TRUE(stats->download_bandwidth>70); linphone_call_stats_unref(stats); end_call(marie,pauline); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } void early_media_without_sdp_in_200_base( bool_t use_video, bool_t use_ice ){ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bctbx_list_t* lcs = NULL; LinphoneCall* marie_call; LinphoneCallParams* params = NULL; LinphoneCallLog *marie_call_log; uint64_t connected_time=0; uint64_t ended_time=0; int dummy=0; lcs = bctbx_list_append(lcs,marie->lc); lcs = bctbx_list_append(lcs,pauline->lc); if (use_ice){ linphone_core_set_firewall_policy(marie->lc, LinphonePolicyUseIce); /* We need RTP symmetric because ICE will put the STUN address in the C line, and no relay is made in this * scenario.*/ lp_config_set_int(linphone_core_get_config(pauline->lc), "rtp", "symmetric", 1); } /* Marie calls Pauline, and after the call has rung, transitions to an early_media session */ params = linphone_core_create_call_params(marie->lc, NULL); if( use_video){ linphone_call_params_enable_video(params, TRUE); linphone_core_enable_video_capture(pauline->lc, TRUE); linphone_core_enable_video_display(pauline->lc, TRUE); linphone_core_enable_video_capture(marie->lc, TRUE); linphone_core_enable_video_display(marie->lc, FALSE); } marie_call = linphone_core_invite_address_with_params(marie->lc, pauline->identity, params); linphone_call_params_unref(params); marie_call_log = linphone_call_get_call_log(marie_call); BC_ASSERT_TRUE(wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallIncomingReceived,1,3000)); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallOutgoingRinging,1,1000)); if (linphone_core_inc_invite_pending(pauline->lc)) { LinphoneCall* pauline_call = linphone_core_get_current_call(pauline->lc); /* send a 183 to initiate the early media */ linphone_call_accept_early_media(pauline_call); BC_ASSERT_TRUE( wait_for_list(lcs, &pauline->stat.number_of_LinphoneCallIncomingEarlyMedia,1,2000) ); BC_ASSERT_TRUE( wait_for_list(lcs, &marie->stat.number_of_LinphoneCallOutgoingEarlyMedia,1,2000) ); liblinphone_tester_check_rtcp(marie, pauline); /* will send the 200OK _without_ SDP. We expect the early-media SDP to be used instead */ sal_call_set_sdp_handling(pauline_call->op, SalOpSDPSimulateRemove); linphone_call_accept(pauline_call); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallConnected, 1,1000)); connected_time=ms_get_cur_time_ms(); BC_ASSERT_TRUE(wait_for_list(lcs, &marie->stat.number_of_LinphoneCallStreamsRunning, 1,3000)); BC_ASSERT_PTR_EQUAL(marie_call, linphone_core_get_current_call(marie->lc)); liblinphone_tester_check_rtcp(marie, pauline); /*just to have a call duration !=0*/ wait_for_list(lcs,&dummy,1,2000); end_call(pauline, marie); ended_time=ms_get_cur_time_ms(); BC_ASSERT_LOWER(labs((long)((linphone_call_log_get_duration(marie_call_log)*1000) - (int64_t)(ended_time - connected_time))), 1000, long, "%ld"); } bctbx_list_free(lcs); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_early_media_and_no_sdp_in_200(void){ early_media_without_sdp_in_200_base(FALSE, FALSE); } static void call_with_early_media_ice_and_no_sdp_in_200(void){ early_media_without_sdp_in_200_base(FALSE, TRUE); } static void call_with_generic_cn(void) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; LinphoneCall *pauline_call; char *audio_file_with_silence=bc_tester_res("sounds/ahbahouaismaisbon.wav"); char *recorded_file=bc_tester_file("result.wav"); marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); remove(recorded_file); linphone_core_use_files(marie->lc,TRUE); linphone_core_use_files(pauline->lc,TRUE); linphone_core_set_play_file(marie->lc, audio_file_with_silence); /*linphone_core_set_play_file(pauline->lc, NULL);*/ linphone_core_set_record_file(pauline->lc, recorded_file); linphone_core_enable_generic_comfort_noise(marie->lc, TRUE); linphone_core_enable_generic_comfort_noise(pauline->lc, TRUE); BC_ASSERT_TRUE(call(marie,pauline)); pauline_call=linphone_core_get_current_call(pauline->lc); BC_ASSERT_PTR_NOT_NULL(pauline_call); if (pauline_call){ const rtp_stats_t *rtps; wait_for_until(marie->lc, pauline->lc, NULL, 0, 8000); rtps=rtp_session_get_stats(pauline_call->audiostream->ms.sessions.rtp_session); BC_ASSERT_TRUE(rtps->packet_recv<=300 && rtps->packet_recv>=200); } end_call(marie,pauline); if (pauline_call){ struct stat stbuf; int err; err=stat(recorded_file,&stbuf); BC_ASSERT_EQUAL(err, 0, int, "%d"); if (err==0){ BC_ASSERT_GREATER(stbuf.st_size,120000,int, "%d"); } } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); ms_free(audio_file_with_silence); bc_free(recorded_file); } static void call_state_changed_2(LinphoneCore *lc, LinphoneCall *call, LinphoneCallState cstate, const char *msg){ LinphoneSipTransports sip_tr; if (cstate==LinphoneCallReleased) { /*to make sure transport is changed*/ sip_tr.udp_port = 0; sip_tr.tcp_port = 45876; sip_tr.tls_port = 0; linphone_core_set_sip_transports(lc,&sip_tr); } } static void call_state_changed_3(LinphoneCore *lc, LinphoneCall *call, LinphoneCallState cstate, const char *msg){ /*just to check multi listener in such situation*/ char* to=linphone_address_as_string(linphone_call_get_call_log(call)->to); char* from=linphone_address_as_string(linphone_call_get_call_log(call)->from); ms_message("Third call listener reports: %s call from [%s] to [%s], new state is [%s]" ,linphone_call_get_call_log(call)->dir==LinphoneCallIncoming?"Incoming":"Outgoing" ,from ,to ,linphone_call_state_to_string(cstate)); ms_free(to); ms_free(from); } static void call_with_transport_change_base(bool_t succesfull_call) { LinphoneSipTransports sip_tr; LinphoneCoreManager* marie; LinphoneCoreManager* pauline; LinphoneCoreVTable * v_table; v_table = linphone_core_v_table_new(); v_table->call_state_changed=call_state_changed_2; marie = linphone_core_manager_new("marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); linphone_core_add_listener(marie->lc,v_table); v_table = linphone_core_v_table_new(); v_table->call_state_changed=call_state_changed_3; linphone_core_add_listener(marie->lc,v_table); sip_tr.udp_port = 0; sip_tr.tcp_port = 45875; sip_tr.tls_port = 0; linphone_core_set_sip_transports(marie->lc,&sip_tr); if (succesfull_call) { BC_ASSERT_TRUE(call(marie,pauline)); end_call(marie, pauline); } else linphone_core_invite(marie->lc,"nexiste_pas"); if (succesfull_call) BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallReleased,1)); if (succesfull_call) { BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallEnd,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallReleased,1)); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_transport_change_after_released(void) { call_with_transport_change_base(TRUE); } static void unsucessfull_call_with_transport_change_after_released(void) { call_with_transport_change_base(FALSE); } #if !defined(__arm__) && !defined(__arm64__) && !TARGET_IPHONE_SIMULATOR && !defined(__ANDROID__) static void completion_cb(void *user_data, int percentage){ fprintf(stdout,"%i %% completed\r",percentage); fflush(stdout); } #endif static void simple_stereo_call(const char *codec_name, int clock_rate, int bitrate_override, bool_t stereo) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; PayloadType *pt; char *stereo_file = bc_tester_res("sounds/vrroom.wav"); char *recordpath = bc_tester_file("stereo-record.wav"); bool_t audio_cmp_failed = FALSE; unlink(recordpath); marie = linphone_core_manager_new( "marie_rc"); pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); /*make sure we have opus*/ pt = linphone_core_find_payload_type(marie->lc, codec_name, clock_rate, 2); if (!pt) { ms_warning("%s not available, stereo with %s not tested.",codec_name, codec_name); goto end; } if (stereo) payload_type_set_recv_fmtp(pt, "stereo=1;sprop-stereo=1"); if (bitrate_override) linphone_core_set_payload_type_bitrate(marie->lc, pt, bitrate_override); pt = linphone_core_find_payload_type(pauline->lc, codec_name, clock_rate, 2); if (stereo) payload_type_set_recv_fmtp(pt, "stereo=1;sprop-stereo=1"); if (bitrate_override) linphone_core_set_payload_type_bitrate(pauline->lc, pt, bitrate_override); disable_all_audio_codecs_except_one(marie->lc, codec_name, clock_rate); disable_all_audio_codecs_except_one(pauline->lc, codec_name, clock_rate); linphone_core_set_use_files(marie->lc, TRUE); linphone_core_set_play_file(marie->lc, stereo_file); linphone_core_set_use_files(pauline->lc, TRUE); linphone_core_set_record_file(pauline->lc, recordpath); /*stereo is supported only without volume control, echo canceller...*/ lp_config_set_string(marie->lc->config,"sound","features","REMOTE_PLAYING"); lp_config_set_string(pauline->lc->config,"sound","features","REMOTE_PLAYING"); if (!BC_ASSERT_TRUE(call(pauline,marie))) goto end; wait_for_until(marie->lc, pauline->lc, NULL, 0, 6000); end_call(pauline, marie); if (clock_rate!=48000) { ms_warning("Similarity checking not implemented for files not having the same sampling rate"); }else{ #if !defined(__arm__) && !defined(__arm64__) && !TARGET_IPHONE_SIMULATOR && !defined(__ANDROID__) double similar; double min_threshold = .75f; /*should be above 0.8 in best conditions*/ double max_threshold = 1.f; if (!stereo){ /*when opus doesn't transmit stereo, the cross correlation is around 0.6 : as expected, it is not as good as in full stereo mode*/ min_threshold = .4f; max_threshold = .68f; } BC_ASSERT_EQUAL(ms_audio_diff(stereo_file, recordpath,&similar,&audio_cmp_params,completion_cb,NULL), 0, int, "%d"); BC_ASSERT_GREATER(similar, min_threshold, double, "%g"); BC_ASSERT_LOWER(similar, max_threshold, double, "%g"); if (similarmax_threshold){ audio_cmp_failed = TRUE; } #endif } if (!audio_cmp_failed) unlink(recordpath); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); ms_free(stereo_file); bc_free(recordpath); } static void simple_stereo_call_l16(void){ simple_stereo_call("L16", 44100, 0, TRUE); } static void simple_stereo_call_opus(void){ simple_stereo_call("opus", 48000, 150, TRUE); } static void call_with_complex_late_offering(void){ LinphoneCallParams *params; LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* call_pauline; LinphoneCall* call_marie; LinphoneVideoPolicy vpol; bool_t call_ok; vpol.automatically_initiate = vpol.automatically_accept = TRUE; linphone_core_enable_video_capture(pauline->lc, TRUE); linphone_core_enable_video_display(pauline->lc, TRUE); linphone_core_enable_video_capture(marie->lc, TRUE); linphone_core_enable_video_display(marie->lc, TRUE); linphone_core_set_video_policy(pauline->lc, &vpol); linphone_core_set_video_policy(marie->lc, &vpol); // important: VP8 has really poor performances with the mire camera, at least // on iOS - so when ever h264 is available, let's use it instead if (linphone_core_find_payload_type(pauline->lc,"h264", -1, -1)!=NULL) { disable_all_video_codecs_except_one(pauline->lc,"h264"); disable_all_video_codecs_except_one(marie->lc,"h264"); } linphone_core_set_video_device(pauline->lc,liblinphone_tester_mire_id); linphone_core_set_video_device(marie->lc,liblinphone_tester_mire_id); BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (!call_ok) goto end; call_pauline = linphone_core_get_current_call(pauline->lc); call_marie = linphone_core_get_current_call(marie->lc); //Invite inactive Audio/video (Marie pause Pauline) ms_message("CONTEXT: Marie sends INVITE with SDP with all streams inactive"); params=linphone_core_create_call_params(marie->lc,call_marie); linphone_call_params_set_audio_direction(params,LinphoneMediaDirectionInactive); linphone_call_params_set_video_direction(params,LinphoneMediaDirectionInactive); linphone_call_update(call_marie ,params); linphone_call_params_unref(params); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallUpdating,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallPausedByRemote,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,2)); //Marie sends INVITE without SDP ms_message("CONTEXT: Marie sends INVITE without SDP for setting streams in send-only mode"); linphone_core_enable_sdp_200_ack(marie->lc,TRUE); params=linphone_core_create_call_params(marie->lc,call_marie); linphone_call_params_set_audio_direction(params,LinphoneMediaDirectionSendOnly); linphone_call_params_set_video_direction(params,LinphoneMediaDirectionSendOnly); linphone_call_update(call_marie , params); linphone_call_params_unref(params); //Pauline OK with sendonly BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallUpdating,2)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,3)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallPausedByRemote,2)); linphone_core_enable_sdp_200_ack(marie->lc,FALSE); //Pauline pause Marie ms_message("CONTEXT: Pauline pauses the call"); linphone_call_pause(call_pauline); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallPausing,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallPaused,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallPausedByRemote,1)); //Pauline resume Marie ms_message("CONTEXT: Pauline resumes the call"); wait_for_until(pauline->lc, marie->lc, NULL, 5, 2000); linphone_call_resume(call_pauline); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,4)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallResuming,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallPausedByRemote,3)); wait_for_until(pauline->lc, marie->lc, NULL, 0, 2000); //Marie invite inactive Audio/Video ms_message("CONTEXT: Marie sends INVITE with SDP with all streams inactive"); params=linphone_core_create_call_params(marie->lc,call_marie); linphone_call_params_set_audio_direction(params,LinphoneMediaDirectionInactive); linphone_call_params_set_video_direction(params,LinphoneMediaDirectionInactive); linphone_call_update(call_marie,params); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallUpdating,3)); linphone_call_params_unref(params); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallPausedByRemote,4)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,5)); //Marie sends INVITE without SDP ms_message("CONTEXT: Marie sends INVITE without SDP in the purpose of re-enabling streams in sendrecv mode"); linphone_core_enable_sdp_200_ack(marie->lc,TRUE); params=linphone_core_create_call_params(marie->lc,call_marie); linphone_call_params_set_audio_direction(params,LinphoneMediaDirectionSendRecv); linphone_call_params_set_video_direction(params,LinphoneMediaDirectionSendRecv); linphone_call_update(call_marie,params); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallUpdating,3)); linphone_call_params_unref(params); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallUpdatedByRemote,1)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&pauline->stat.number_of_LinphoneCallStreamsRunning,2)); BC_ASSERT_TRUE(wait_for(marie->lc,pauline->lc,&marie->stat.number_of_LinphoneCallStreamsRunning,5)); linphone_core_enable_sdp_200_ack(marie->lc,FALSE); end_call(marie,pauline); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void simple_mono_call_opus(void){ /*actually a call where input/output is made with stereo but opus transmits everything as mono*/ simple_stereo_call("opus", 48000, 150, FALSE); } /* because SIP ALG (like in android phones) crash when seing a domain name in SDP, we prefer using SIP/TLS for both participants*/ static void call_with_fqdn_in_sdp(void) { bool_t tls_supported = transport_supported(LinphoneTransportTls); LinphoneCoreManager* marie = linphone_core_manager_new(tls_supported ? "marie_sips_rc" : "marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(tls_supported ? "pauline_rc" : "pauline_tcp_rc"); LpConfig *lp; bool_t call_ok; lp = linphone_core_get_config(marie->lc); lp_config_set_string(lp,"rtp","bind_address","localhost"); lp = linphone_core_get_config(pauline->lc); lp_config_set_string(lp,"rtp","bind_address","localhost"); BC_ASSERT_TRUE(call_ok=call(marie,pauline)); if (!call_ok) goto end; liblinphone_tester_check_rtcp(pauline,marie); #ifdef VIDEO_ENABLED BC_ASSERT_TRUE(request_video(pauline,marie, TRUE)); liblinphone_tester_check_rtcp(pauline,marie); #endif end_call(pauline, marie); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_rtp_io_mode(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphonePlayer *player; char *hellopath = bc_tester_res("sounds/ahbahouaismaisbon.wav"); char *recordpath = bc_tester_file("record-call_with_rtp_io_mode.wav"); bool_t call_ok; int attempts; double similar=1; const double threshold = 0.85; /*this test is actually attempted three times in case of failure, because the audio comparison at the end is very sensitive to * jitter buffer drifts, which sometimes happen if the machine is unable to run the test in good realtime conditions */ for (attempts=0; attempts<3; attempts++){ /* Make sure that the record file doesn't already exists, otherwise this test will append new samples to it. */ unlink(recordpath); reset_counters(&marie->stat); reset_counters(&pauline->stat); /* The caller uses files instead of soundcard in order to avoid mixing soundcard input with file played using call's player. */ linphone_core_use_files(marie->lc, TRUE); linphone_core_set_play_file(marie->lc, NULL); linphone_core_set_record_file(marie->lc, recordpath); linphone_core_use_files(pauline->lc, FALSE); /* The callee uses the RTP IO mode with the PCMU codec to send back audio to the caller. */ disable_all_audio_codecs_except_one(pauline->lc, "pcmu", -1); lp_config_set_int(pauline->lc->config, "sound", "rtp_io", 1); lp_config_set_string(pauline->lc->config, "sound", "rtp_local_addr", linphone_core_ipv6_enabled(pauline->lc) ? "::1" : "127.0.0.1"); lp_config_set_string(pauline->lc->config, "sound", "rtp_remote_addr", linphone_core_ipv6_enabled(pauline->lc) ? "::1" : "127.0.0.1"); lp_config_set_int(pauline->lc->config, "sound", "rtp_local_port", 17076); lp_config_set_int(pauline->lc->config, "sound", "rtp_remote_port", 17076); lp_config_set_string(pauline->lc->config, "sound", "rtp_map", "pcmu/8000/1"); BC_ASSERT_TRUE((call_ok = call(marie, pauline))); if (!call_ok) goto end; player = linphone_call_get_player(linphone_core_get_current_call(marie->lc)); BC_ASSERT_PTR_NOT_NULL(player); if (player) { LinphonePlayerCbs *cbs = linphone_player_get_callbacks(player); linphone_player_cbs_set_eof_reached(cbs, on_eof); linphone_player_cbs_set_user_data(cbs, marie); BC_ASSERT_EQUAL(linphone_player_open(player, hellopath) , 0, int, "%d"); BC_ASSERT_EQUAL(linphone_player_start(player) , 0, int, "%d"); } /* This assert should be modified to be at least as long as the WAV file */ BC_ASSERT_TRUE(wait_for_until(pauline->lc, marie->lc, &marie->stat.number_of_player_eof, 1, 10000)); /*wait for one second more so that last RTP packets can arrive*/ wait_for_until(pauline->lc,marie->lc,NULL,0,1000); end_call(pauline,marie); BC_ASSERT_EQUAL(ms_audio_diff(hellopath, recordpath, &similar, &audio_cmp_params, NULL, NULL), 0, int, "%d"); if (similar>=threshold) break; } BC_ASSERT_GREATER(similar, threshold, double, "%g"); BC_ASSERT_LOWER(similar, 1.0, double, "%g"); if (similar >= threshold && similar <= 1.0) { remove(recordpath); } end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); ms_free(recordpath); ms_free(hellopath); } static void generic_nack_received(const OrtpEventData *evd, stats *st) { if (rtcp_is_RTPFB(evd->packet)) { switch (rtcp_RTPFB_get_type(evd->packet)) { case RTCP_RTPFB_NACK: st->number_of_rtcp_generic_nack++; break; default: break; } } } static void call_with_generic_nack_rtcp_feedback(void) { LinphoneCoreManager *marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LpConfig *lp; LinphoneCall *call_marie; bool_t call_ok; OrtpNetworkSimulatorParams params = { 0 }; params.enabled = TRUE; params.loss_rate = 10; params.consecutive_loss_probability = 0.75; params.mode = OrtpNetworkSimulatorOutbound; linphone_core_set_avpf_mode(marie->lc, LinphoneAVPFEnabled); linphone_core_set_avpf_mode(pauline->lc, LinphoneAVPFEnabled); lp = linphone_core_get_config(pauline->lc); lp_config_set_int(lp, "rtp", "rtcp_fb_generic_nack_enabled", 1); BC_ASSERT_TRUE(call_ok = call(pauline, marie)); if (!call_ok) goto end; BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1)); BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 1)); call_marie = linphone_core_get_current_call(marie->lc); if (call_marie) { rtp_session_enable_network_simulation(call_marie->audiostream->ms.sessions.rtp_session, ¶ms); ortp_ev_dispatcher_connect(media_stream_get_event_dispatcher(&call_marie->audiostream->ms), ORTP_EVENT_RTCP_PACKET_RECEIVED, RTCP_RTPFB, (OrtpEvDispatcherCb)generic_nack_received, &marie->stat); } BC_ASSERT_TRUE(wait_for_until(pauline->lc, marie->lc, &marie->stat.number_of_rtcp_generic_nack, 5, 8000)); end_call(pauline, marie); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } // This is a custom structure used for the tests using custom RTP transport modifier. // It is only used to count the number of sent and received packets typedef struct _RtpTransportModifierData { uint64_t packetSentCount; uint64_t packetReceivedCount; MSQueue to_send; MSQueue to_recv; } RtpTransportModifierData; const char *XOR_KEY = "BELLEDONNECOMMUNICATIONS"; // Callback called when a packet is on it's way to be sent // This is where we can do some changes on it, like encrypt it static int rtptm_on_send(RtpTransportModifier *rtptm, mblk_t *msg) { RtpTransportModifierData *data = rtptm->data; rtp_header_t *rtp = (rtp_header_t*)msg->b_rptr; if (rtp->version == 0) { // This is probably a STUN packet, so don't count it (oRTP won't) and don't encrypt it either return (int)msgdsize(msg); } /*ms_message("rtptm_on_send: rtpm=%p seq=%u", rtptm, (int)ntohs(rtp_get_seqnumber(msg)));*/ data->packetSentCount += 1; ms_queue_put(&data->to_send, dupmsg(msg)); return 0; // Return 0 to drop the packet, it will be injected later } // Callback called when a packet is on it's way to be received // This is where we can do some changes on it, like decrypt it static int rtptm_on_receive(RtpTransportModifier *rtptm, mblk_t *msg) { RtpTransportModifierData *data = rtptm->data; rtp_header_t *rtp = (rtp_header_t*)msg->b_rptr; if (rtp->version == 0) { // This is probably a STUN packet, so don't count it (oRTP won't) and don't decrypt it either return (int)msgdsize(msg); } data->packetReceivedCount += 1; ms_queue_put(&data->to_recv, dupmsg(msg)); return 0; // Return 0 to drop the packet, it will be injected later } static void rtptm_on_schedule(RtpTransportModifier *rtptm) { RtpTransportModifierData *data = rtptm->data; mblk_t *msg = NULL; while ((msg = ms_queue_get(&data->to_send)) != NULL) { int size = 0; unsigned char *src; int i = 0; // Mediastream can create a mblk_t with only the RTP header and setting the b_cont pointer to the actual RTP content buffer // In this scenario, the result of rtp_get_payload will be 0, and we won't be able to do our XOR encryption on the payload // The call to msgpullup will trigger a memcpy of the header and the payload in the same buffer in the msg mblk_t msgpullup(msg, -1); // Now that the mblk_t buffer directly contains the header and the payload, we can get the size of the payload and a pointer to it's start (we don't encrypt the RTP header) size = rtp_get_payload(msg, &src); // Just for fun, let's do a XOR encryption for (i = 0; i < size; i++) { src[i] ^= (unsigned char) XOR_KEY[i % strlen(XOR_KEY)]; } meta_rtp_transport_modifier_inject_packet_to_send(rtptm->transport, rtptm, msg, 0); } msg = NULL; while ((msg = ms_queue_get(&data->to_recv)) != NULL) { int size = 0; unsigned char *src; int i = 0; // On the receiving side, there is no need for a msgpullup, the mblk_t contains the header and the payload in the same buffer // We just ask for the size and a pointer to the payload buffer size = rtp_get_payload(msg, &src); // Since we did a XOR encryption on the send side, we have to do it again to decrypt the payload for (i = 0; i < size; i++) { src[i] ^= (unsigned char) XOR_KEY[i % strlen(XOR_KEY)]; } meta_rtp_transport_modifier_inject_packet_to_recv(rtptm->transport, rtptm, msg, 0); } } // This callback is called when the transport modifier is being destroyed // It is a good place to free the resources allocated for the transport modifier static void rtptm_destroy(RtpTransportModifier *rtptm) { // Do nothing, we'll free it later because we need to access the RtpTransportModifierData structure after the call is ended } // This is the callback called when the state of the call change static void call_state_changed_4(LinphoneCore *lc, LinphoneCall *call, LinphoneCallState cstate, const char *msg) { int i = 0; // To add a custom RTP transport modifier, we have to do it before the call is running, but after the RTP session is created. if (cstate == LinphoneCallIncomingReceived || cstate == LinphoneCallOutgoingProgress) { RtpTransport *rtpt = NULL; RtpTransportModifierData *data = ms_new0(RtpTransportModifierData, 1); RtpTransportModifier *rtptm = ms_new0(RtpTransportModifier, 1); ms_queue_init(&data->to_send); ms_queue_init(&data->to_recv); rtptm->data = data; rtptm->t_process_on_send = rtptm_on_send; rtptm->t_process_on_receive = rtptm_on_receive; rtptm->t_process_on_schedule = rtptm_on_schedule; rtptm->t_destroy = rtptm_destroy; // Here we iterate on each meta rtp transport available for (i = 0; i < linphone_call_get_stream_count(call); i++) { MSFormatType type; rtpt = linphone_call_get_meta_rtp_transport(call, i); // If we wanted, we also could get the RTCP meta transports like this: // rtcpt = linphone_call_get_meta_rtcp_transport(call, i); // If you want to know which stream meta RTP transport is the current one, you can use type = linphone_call_get_stream_type(call, i); // Currently there is only MSAudio and MSVideo types, but this could change later if (type == MSAudio) { // And now we append our RTP transport modifier to the current list of modifiers meta_rtp_transport_append_modifier(rtpt, rtptm); } else if (type == MSVideo) { // Because the call of this test is audio only, we don't have to append our modifier to the meta RTP transport from the video stream } } // We save the pointer to our RtpTransportModifier in the call user_data to be able to get to it later call->user_data = rtptm; } } static void custom_rtp_modifier(bool_t pauseResumeTest, bool_t recordTest) { // This will initialize two linphone core using information contained in the marie_rc and pauline_rc files and wait for them to be correctly registered LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCall* call_pauline = NULL; LinphoneCall* call_marie = NULL; const rtp_stats_t * stats; bool_t call_ok; LinphoneCoreVTable * v_table; RtpTransportModifier *rtptm_marie = NULL; RtpTransportModifier *rtptm_pauline = NULL; RtpTransportModifierData *data_marie = NULL; RtpTransportModifierData *data_pauline = NULL; // The following are only used for the record test LinphonePlayer *player; char *hellopath = bc_tester_res("sounds/ahbahouaismaisbon.wav"); // File to be played char *recordpath = bc_tester_file("record-call_with_file_player.wav"); // File to record the received sound double similar = 1; // The factor of similarity between the played file and the one recorded const double threshold = 0.85; // Minimum similarity value to consider the record file equal to the one sent // We create a new vtable to listen only to the call state changes, in order to plug our RTP Transport Modifier when the call will be established v_table = linphone_core_v_table_new(); v_table->call_state_changed = call_state_changed_4; linphone_core_add_listener(pauline->lc,v_table); v_table = linphone_core_v_table_new(); v_table->call_state_changed = call_state_changed_4; linphone_core_add_listener(marie->lc,v_table); if (recordTest) { // When we do the record test, we need a file player to play the content of a sound file /*make sure the record file doesn't already exists, otherwise this test will append new samples to it*/ unlink(recordpath); linphone_core_use_files(pauline->lc,TRUE); linphone_core_set_play_file(pauline->lc,NULL); linphone_core_set_record_file(pauline->lc,NULL); /*callee is recording and plays file*/ linphone_core_use_files(marie->lc,TRUE); linphone_core_set_play_file(marie->lc,NULL); linphone_core_set_record_file(marie->lc,recordpath); } // Now the the call should be running (call state StreamsRunning) BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (!call_ok) goto end; // Ref the call to keep the pointer valid even after the call is release call_pauline = linphone_call_ref(linphone_core_get_current_call(pauline->lc)); call_marie = linphone_call_ref(linphone_core_get_current_call(marie->lc)); // This is for the pause/resume test, we don't do it in the call record test to be able to check the recorded call matches the file played if (pauseResumeTest) { // This only wait for 3 seconds in order to generate traffic for the test wait_for_until(pauline->lc, marie->lc, NULL, 5, 3000); linphone_call_pause(call_pauline); BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &pauline->stat.number_of_LinphoneCallPausing, 1)); BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &marie->stat.number_of_LinphoneCallPausedByRemote, 1)); BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &pauline->stat.number_of_LinphoneCallPaused, 1)); /*stay in pause a little while in order to generate traffic*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 2000); linphone_call_resume(call_pauline); BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(wait_for(pauline->lc, marie->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 2)); /*same here: wait a while for a bit of a traffic, we need to receive a RTCP packet*/ wait_for_until(pauline->lc, marie->lc, NULL, 5, 5000); /*since RTCP streams are reset when call is paused/resumed, there should be no loss at all*/ stats = rtp_session_get_stats(call_pauline->sessions->rtp_session); BC_ASSERT_EQUAL((int)stats->cum_packet_loss, 0, int, "%d"); end_call(pauline, marie); } else if (recordTest) { player = linphone_call_get_player(call_pauline); BC_ASSERT_PTR_NOT_NULL(player); if (player) { // This will ask pauline to play the file LinphonePlayerCbs *cbs = linphone_player_get_callbacks(player); linphone_player_cbs_set_eof_reached(cbs, on_eof); linphone_player_cbs_set_user_data(cbs, pauline); BC_ASSERT_EQUAL(linphone_player_open(player, hellopath),0, int, "%d"); BC_ASSERT_EQUAL(linphone_player_start(player), 0, int, "%d"); } /* This assert should be modified to be at least as long as the WAV file */ BC_ASSERT_TRUE(wait_for_until(pauline->lc, marie->lc, &pauline->stat.number_of_player_eof, 1, 10000)); /*wait one second more for transmission to be fully ended (transmission time + jitter buffer)*/ wait_for_until(pauline->lc, marie->lc, NULL, 0, 1000); end_call(pauline, marie); // Now we compute a similarity factor between the original file and the one we recorded on the callee side BC_ASSERT_EQUAL(ms_audio_diff(hellopath, recordpath, &similar, &audio_cmp_params, NULL, NULL), 0, int, "%d"); BC_ASSERT_GREATER(similar, threshold, double, "%g"); BC_ASSERT_LOWER(similar, 1.0, double, "%g"); if (similar >= threshold && similar <= 1.0) { // If the similarity value is between perfect (1) and our threshold (0.9), then we delete the file used for the record remove(recordpath); } } else { // This only wait for 3 seconds in order to generate traffic for the test wait_for_until(pauline->lc, marie->lc, NULL, 5, 3000); // We termine the call and check the stats to see if the call is correctly ended on both sides end_call(pauline, marie); } // Now we can go fetch our custom structure and check the number of packets sent/received is the same on both sides rtptm_marie = (RtpTransportModifier *)call_marie->user_data; rtptm_pauline = (RtpTransportModifier *)call_pauline->user_data; data_marie = (RtpTransportModifierData *)rtptm_marie->data; data_pauline = (RtpTransportModifierData *)rtptm_pauline->data; BC_ASSERT_PTR_NOT_NULL(data_marie); BC_ASSERT_PTR_NOT_NULL(data_pauline); ms_message("Marie sent %i RTP packets and received %i (through our modifier)", (int)data_marie->packetSentCount, (int)data_marie->packetReceivedCount); ms_message("Pauline sent %i RTP packets and received %i (through our modifier)", (int)data_pauline->packetSentCount, (int)data_pauline->packetReceivedCount); // There will be a few RTP packets sent on marie's side before the call is ended at pauline's request, so we need the threshold BC_ASSERT_TRUE(data_marie->packetSentCount - data_pauline->packetReceivedCount < 50); BC_ASSERT_TRUE(data_pauline->packetSentCount - data_marie->packetReceivedCount < 50); // At this point, we know each packet that has been processed in the send callback of our RTP modifier also go through the recv callback of the remote. // Now we want to ensure that all sent RTP packets actually go through our RTP transport modifier and thus no packet leave without being processed (by any operation we might want to do on it) { LinphoneCallStats *marie_stats = linphone_call_get_audio_stats(call_marie); LinphoneCallStats *pauline_stats = linphone_call_get_audio_stats(call_pauline); rtp_stats_t marie_rtp_stats = *linphone_call_stats_get_rtp_stats(marie_stats); rtp_stats_t pauline_rtp_stats = *linphone_call_stats_get_rtp_stats(pauline_stats); ms_message("Marie sent %i RTP packets and received %i (for real)", (int)marie_rtp_stats.packet_sent, (int)marie_rtp_stats.packet_recv); ms_message("Pauline sent %i RTP packets and received %i (for real)", (int)pauline_rtp_stats.packet_sent, (int)pauline_rtp_stats.packet_recv); BC_ASSERT_EQUAL(data_marie->packetReceivedCount, marie_rtp_stats.packet_recv, unsigned long long, "%llu"); BC_ASSERT_EQUAL(data_marie->packetSentCount, marie_rtp_stats.packet_sent, unsigned long long, "%llu"); // There can be a small difference between the number of packets received in the modifier and the number processed in reception because the processing is asynchronous BC_ASSERT_TRUE(data_pauline->packetReceivedCount - pauline_rtp_stats.packet_recv < 20); BC_ASSERT_TRUE(data_pauline->packetSentCount == pauline_rtp_stats.packet_sent); linphone_call_stats_unref(marie_stats); linphone_call_stats_unref(pauline_stats); } end: // Since we didn't free the resources of our RTP transport modifier in the rtptm_destroy callback, we'll do it here if (data_pauline) { ms_free(data_pauline); } ms_free(rtptm_pauline); if (data_marie) { ms_free(data_marie); } ms_free(rtptm_marie); // Unref the previously ref calls if (call_marie) { linphone_call_unref(call_marie); } if (call_pauline) { linphone_call_unref(call_pauline); } // The test is finished, the linphone core are no longer needed, we can safely free them linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); bc_free(recordpath); bc_free(hellopath); } static void call_with_custom_rtp_modifier(void) { custom_rtp_modifier(FALSE, FALSE); } static void call_paused_resumed_with_custom_rtp_modifier(void) { custom_rtp_modifier(TRUE, FALSE); } static void call_record_with_custom_rtp_modifier(void) { custom_rtp_modifier(FALSE, TRUE); } static void recovered_call_on_network_switch_in_early_state(LinphoneCoreManager* callerMgr) { const LinphoneCallParams *remote_params; LinphoneCall *incoming_call; LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); linphone_core_invite_address(callerMgr->lc, pauline->identity); if (!BC_ASSERT_TRUE(wait_for(callerMgr->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1))) goto end; if (!BC_ASSERT_TRUE(wait_for(callerMgr->lc, pauline->lc, &callerMgr->stat.number_of_LinphoneCallOutgoingRinging, 1))) goto end; linphone_core_set_network_reachable(callerMgr->lc, FALSE); wait_for(callerMgr->lc, pauline->lc, &callerMgr->stat.number_of_NetworkReachableFalse, 1); linphone_core_set_network_reachable(callerMgr->lc, TRUE); wait_for(callerMgr->lc, pauline->lc, &callerMgr->stat.number_of_NetworkReachableTrue, 2); BC_ASSERT_TRUE(wait_for(callerMgr->lc, pauline->lc, &callerMgr->stat.number_of_LinphoneCallOutgoingRinging, 2)); incoming_call = linphone_core_get_current_call(pauline->lc); remote_params = linphone_call_get_remote_params(incoming_call); BC_ASSERT_PTR_NOT_NULL(remote_params); if (remote_params != NULL) { const char *replaces_header = linphone_call_params_get_custom_header(remote_params, "Replaces"); BC_ASSERT_PTR_NOT_NULL(replaces_header); } linphone_call_accept(incoming_call); BC_ASSERT_TRUE(wait_for(callerMgr->lc, pauline->lc, &callerMgr->stat.number_of_LinphoneCallStreamsRunning, 1)); BC_ASSERT_TRUE(wait_for(callerMgr->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 1)); linphone_call_terminate(incoming_call); BC_ASSERT_TRUE(wait_for(callerMgr->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallEnd, 1)); BC_ASSERT_TRUE(wait_for(callerMgr->lc, pauline->lc, &callerMgr->stat.number_of_LinphoneCallReleased, 1)); BC_ASSERT_TRUE(wait_for(callerMgr->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallReleased, 1)); end: linphone_core_manager_destroy(pauline); } static void recovered_call_on_network_switch_in_early_state_1(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); recovered_call_on_network_switch_in_early_state(marie); linphone_core_manager_destroy(marie); } static void recovered_call_on_network_switch_in_early_state_1_udp(void) { LinphoneCoreManager* laure = linphone_core_manager_new("laure_rc_udp"); recovered_call_on_network_switch_in_early_state(laure); linphone_core_manager_destroy(laure); } static void recovered_call_on_network_switch_in_early_state_2(void) { LinphoneCall *incoming_call; LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); linphone_core_invite_address(marie->lc, pauline->identity); if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1))) goto end; if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallOutgoingRinging, 1))) goto end; incoming_call = linphone_core_get_current_call(pauline->lc); linphone_call_accept(incoming_call); linphone_core_set_network_reachable(marie->lc, FALSE); wait_for(marie->lc, pauline->lc, &marie->stat.number_of_NetworkReachableFalse, 1); linphone_core_set_network_reachable(marie->lc, TRUE); wait_for(marie->lc, pauline->lc, &marie->stat.number_of_NetworkReachableTrue, 2); 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)); linphone_call_terminate(incoming_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallEnd, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallReleased, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallReleased, 1)); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void recovered_call_on_network_switch_in_early_state_3(void) { LinphoneCall *incoming_call; LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); linphone_core_invite_address(marie->lc, pauline->identity); if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1))) goto end; if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallOutgoingRinging, 1))) goto end; linphone_core_set_network_reachable(pauline->lc, FALSE); wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_NetworkReachableFalse, 1); linphone_core_set_network_reachable(pauline->lc, TRUE); wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_NetworkReachableTrue, 2); wait_for_until(marie->lc, pauline->lc, NULL, 1, 2000); incoming_call = linphone_core_get_current_call(pauline->lc); linphone_call_accept(incoming_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)); linphone_call_terminate(incoming_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallEnd, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallReleased, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallReleased, 1)); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void recovered_call_on_network_switch_in_early_state_4(void) { LinphoneCall *incoming_call; LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new("pauline_tcp_rc"); linphone_core_invite_address(marie->lc, pauline->identity); if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1))) goto end; if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallOutgoingRinging, 1))) goto end; incoming_call = linphone_core_get_current_call(pauline->lc); linphone_call_accept(incoming_call); linphone_core_set_network_reachable(pauline->lc, FALSE); wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_NetworkReachableFalse, 1); linphone_core_set_network_reachable(pauline->lc, TRUE); wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_NetworkReachableTrue, 2); 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)); BC_ASSERT_TRUE(sal_call_dialog_request_pending(incoming_call->op)); wait_for_until(marie->lc, pauline->lc, NULL, 1, 2000); BC_ASSERT_FALSE(sal_call_dialog_request_pending(incoming_call->op)); linphone_call_terminate(incoming_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallEnd, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallReleased, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallReleased, 1)); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void recovered_call_on_network_switch_during_reinvite_1(void) { LinphoneCall *incoming_call; LinphoneCall *outgoing_call; LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new("pauline_tcp_rc"); linphone_core_invite_address(marie->lc, pauline->identity); if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1))) goto end; if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallOutgoingRinging, 1))) goto end; incoming_call = linphone_core_get_current_call(pauline->lc); linphone_call_accept(incoming_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)); outgoing_call = linphone_core_get_current_call(marie->lc); linphone_call_pause(outgoing_call); linphone_core_set_network_reachable(marie->lc, FALSE); wait_for(marie->lc, pauline->lc, &marie->stat.number_of_NetworkReachableFalse, 1); linphone_core_set_network_reachable(marie->lc, TRUE); wait_for(marie->lc, pauline->lc, &marie->stat.number_of_NetworkReachableTrue, 2); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallPaused, 1)); linphone_call_terminate(incoming_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallEnd, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallReleased, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallReleased, 1)); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void configure_video_policies_for_network_switch(LinphoneCore *marie, LinphoneCore *pauline) { LinphoneVideoPolicy policy; policy.automatically_accept = FALSE; policy.automatically_initiate = FALSE; linphone_core_enable_video_capture(marie, TRUE); linphone_core_enable_video_display(marie, TRUE); linphone_core_enable_video_capture(pauline, TRUE); linphone_core_enable_video_display(pauline, TRUE); linphone_core_set_video_policy(marie, &policy); linphone_core_set_video_policy(pauline, &policy); lp_config_set_int(pauline->config, "sip", "defer_update_default", TRUE); } static void recovered_call_on_network_switch_during_reinvite_2(void) { LinphoneCall *incoming_call; LinphoneCall *outgoing_call; LinphoneCallParams *params; LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new("pauline_tcp_rc"); configure_video_policies_for_network_switch(marie->lc, pauline->lc); linphone_core_invite_address(marie->lc, pauline->identity); if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1))) goto end; if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallOutgoingRinging, 1))) goto end; incoming_call = linphone_core_get_current_call(pauline->lc); linphone_call_accept(incoming_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)); outgoing_call = linphone_core_get_current_call(marie->lc); params = linphone_core_create_call_params(marie->lc, outgoing_call); linphone_call_params_enable_video(params, TRUE); linphone_call_update(outgoing_call, params); linphone_call_params_unref(params); linphone_core_set_network_reachable(marie->lc, FALSE); wait_for(marie->lc, pauline->lc, &marie->stat.number_of_NetworkReachableFalse, 1); linphone_core_set_network_reachable(marie->lc, TRUE); wait_for(marie->lc, pauline->lc, &marie->stat.number_of_NetworkReachableTrue, 2); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallUpdatedByRemote, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneRegistrationOk, 2)); wait_for_until(marie->lc, pauline->lc, NULL, 1, 2000); params = linphone_core_create_call_params(pauline->lc, incoming_call); linphone_call_params_enable_video(params, TRUE); linphone_call_accept_update(incoming_call, params); linphone_call_params_unref(params); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); wait_for_until(marie->lc, pauline->lc, NULL, 1, 2000); linphone_call_terminate(incoming_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallEnd, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallReleased, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallReleased, 1)); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void recovered_call_on_network_switch_during_reinvite_3(void) { LinphoneCall *incoming_call; LinphoneCall *outgoing_call; LinphoneCallParams *params; LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new("pauline_tcp_rc"); configure_video_policies_for_network_switch(marie->lc, pauline->lc); linphone_core_invite_address(marie->lc, pauline->identity); if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1))) goto end; if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallOutgoingRinging, 1))) goto end; incoming_call = linphone_core_get_current_call(pauline->lc); linphone_call_accept(incoming_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)); outgoing_call = linphone_core_get_current_call(marie->lc); params = linphone_core_create_call_params(marie->lc, outgoing_call); linphone_call_params_enable_video(params, TRUE); linphone_call_update(outgoing_call, params); linphone_call_params_unref(params); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallUpdatedByRemote, 1)); linphone_core_set_network_reachable(pauline->lc, FALSE); wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_NetworkReachableFalse, 1); linphone_core_set_network_reachable(pauline->lc, TRUE); wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_NetworkReachableTrue, 2); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneRegistrationOk, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallUpdatedByRemote, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); params = linphone_core_create_call_params(marie->lc, outgoing_call); linphone_call_params_enable_video(params, TRUE); linphone_call_update(outgoing_call, params); linphone_call_params_unref(params); wait_for_until(marie->lc, pauline->lc, NULL, 1, 2000); params = linphone_core_create_call_params(pauline->lc, incoming_call); linphone_call_params_enable_video(params, TRUE); linphone_call_accept_update(incoming_call, params); linphone_call_params_unref(params); wait_for_until(marie->lc, pauline->lc, NULL, 1, 2000); linphone_call_terminate(incoming_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallEnd, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallReleased, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallReleased, 1)); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void recovered_call_on_network_switch_during_reinvite_4(void) { LinphoneCall *incoming_call; LinphoneCall *outgoing_call; LinphoneCallParams *params; LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new("pauline_tcp_rc"); configure_video_policies_for_network_switch(marie->lc, pauline->lc); linphone_core_invite_address(marie->lc, pauline->identity); if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallIncomingReceived, 1))) goto end; if (!BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallOutgoingRinging, 1))) goto end; incoming_call = linphone_core_get_current_call(pauline->lc); linphone_call_accept(incoming_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)); outgoing_call = linphone_core_get_current_call(marie->lc); params = linphone_core_create_call_params(marie->lc, outgoing_call); linphone_call_params_enable_video(params, TRUE); linphone_call_update(outgoing_call, params); linphone_call_params_unref(params); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallUpdatedByRemote, 1)); params = linphone_core_create_call_params(pauline->lc, incoming_call); linphone_call_params_enable_video(params, TRUE); linphone_call_accept_update(incoming_call, params); linphone_call_params_unref(params); linphone_core_set_network_reachable(pauline->lc, FALSE); wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_NetworkReachableFalse, 1); linphone_core_set_network_reachable(pauline->lc, TRUE); wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_NetworkReachableTrue, 2); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneRegistrationOk, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); wait_for_until(marie->lc, pauline->lc, NULL, 1, 2000); linphone_call_terminate(incoming_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallEnd, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallReleased, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallReleased, 1)); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void _call_with_network_switch(bool_t use_ice, bool_t with_socket_refresh, bool_t enable_rtt, bool_t caller_pause, bool_t callee_pause) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCallParams *pauline_params = NULL; LinphoneCall *marie_call = NULL; LinphoneCall *pauline_call = NULL; bctbx_list_t *lcs = NULL; bool_t call_ok; lcs = bctbx_list_append(lcs, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); if (use_ice){ linphone_core_set_firewall_policy(marie->lc,LinphonePolicyUseIce); linphone_core_set_firewall_policy(pauline->lc,LinphonePolicyUseIce); linphone_core_manager_wait_for_stun_resolution(marie); linphone_core_manager_wait_for_stun_resolution(pauline); } if (with_socket_refresh){ lp_config_set_int(linphone_core_get_config(marie->lc), "net", "recreate_sockets_when_network_is_up", 1); lp_config_set_int(linphone_core_get_config(pauline->lc), "net", "recreate_sockets_when_network_is_up", 1); } if (enable_rtt) { pauline_params = linphone_core_create_call_params(pauline->lc, NULL); linphone_call_params_enable_realtime_text(pauline_params, TRUE); } BC_ASSERT_TRUE((call_ok=call_with_params(pauline, marie, pauline_params, NULL))); if (!call_ok) goto end; wait_for_until(marie->lc, pauline->lc, NULL, 0, 2000); if (use_ice) { /*wait for ICE reINVITE to complete*/ BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(check_ice(pauline,marie,LinphoneIceStateHostConnection)); } if (caller_pause) { pauline_call = linphone_core_get_current_call(pauline->lc); linphone_call_pause(pauline_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallPausedByRemote, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallPaused, 1)); } else if (callee_pause) { marie_call = linphone_core_get_current_call(marie->lc); linphone_call_pause(marie_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallPausedByRemote, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallPaused, 1)); } /*marie looses the network and reconnects*/ linphone_core_set_network_reachable(marie->lc, FALSE); wait_for_until(marie->lc, pauline->lc, NULL, 0, 1000); /*marie will reconnect and register*/ linphone_core_set_network_reachable(marie->lc, TRUE); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneRegistrationOk, 2)); if (use_ice){ BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallUpdating, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallUpdatedByRemote, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 3)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 3)); /*now comes the ICE reINVITE*/ BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallUpdating, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallUpdatedByRemote, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 4)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 4)); }else{ if (caller_pause) { BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallUpdating, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallPausedByRemote, 2)); linphone_call_resume(pauline_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); } else if (callee_pause) { BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallUpdatedByRemote, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallPausedByRemote, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallPaused, 2)); linphone_call_resume(marie_call); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 2)); } else { BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallUpdating, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallUpdatedByRemote, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 2)); } } /*check that media is back*/ check_media_direction(marie, linphone_core_get_current_call(marie->lc), lcs, LinphoneMediaDirectionSendRecv, LinphoneMediaDirectionInvalid); liblinphone_tester_check_rtcp(pauline, marie); if (use_ice) BC_ASSERT_TRUE(check_ice(pauline,marie,LinphoneIceStateHostConnection)); /*pauline shall be able to end the call without problem now*/ end_call(pauline, marie); end: if (pauline_params) { linphone_call_params_unref(pauline_params); } bctbx_list_free(lcs); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_network_switch(void){ _call_with_network_switch(FALSE, FALSE, FALSE, FALSE, FALSE); } static void call_with_network_switch_in_paused_state(void) { _call_with_network_switch(FALSE, FALSE, FALSE, FALSE, TRUE); } static void call_with_network_switch_in_paused_by_remote_state(void) { _call_with_network_switch(FALSE, FALSE, FALSE, TRUE, FALSE); } static void call_with_network_switch_and_ice(void){ _call_with_network_switch(TRUE, FALSE, FALSE, FALSE, FALSE); } static void call_with_network_switch_ice_and_rtt(void) { _call_with_network_switch(TRUE, FALSE, TRUE, FALSE, FALSE); } static void call_with_network_switch_and_socket_refresh(void){ _call_with_network_switch(TRUE, TRUE, FALSE, FALSE, FALSE); } static void call_with_network_switch_no_recovery(void){ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCallParams *pauline_params = NULL; bctbx_list_t *lcs = NULL; bool_t call_ok; lcs = bctbx_list_append(lcs, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); linphone_core_set_nortp_timeout(marie->lc, 50000); BC_ASSERT_TRUE((call_ok=call_with_params(pauline, marie, pauline_params, NULL))); if (!call_ok) goto end; wait_for_until(marie->lc, pauline->lc, NULL, 0, 2000); /*marie looses the network and reconnects*/ linphone_core_set_network_reachable(marie->lc, FALSE); /*but meanwhile pauline terminates the call.*/ linphone_call_terminate(linphone_core_get_current_call(pauline->lc)); /* * We have to wait 32 seconds so that the BYE transaction is terminated, and dialog removed. * This is the condition to receive a 481 when marie sends the reINVITE.*/ wait_for_list(lcs, NULL, 0, 32500); /*marie will reconnect, register, and send an automatic reINVITE to try to repair the call*/ linphone_core_set_network_reachable(marie->lc, TRUE); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneRegistrationOk, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallUpdating, 1)); /*This reINVITE should of course fail, so marie's call should be terminated.*/ BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallEnd, 1)); end: if (pauline_params) { linphone_call_params_unref(pauline_params); } bctbx_list_free(lcs); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_sip_and_rtp_independant_switches(void){ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); bctbx_list_t *lcs = NULL; bool_t call_ok; bool_t use_ice = TRUE; bool_t with_socket_refresh = TRUE; lcs = bctbx_list_append(lcs, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); if (use_ice){ linphone_core_set_firewall_policy(marie->lc,LinphonePolicyUseIce); linphone_core_set_firewall_policy(pauline->lc,LinphonePolicyUseIce); } if (with_socket_refresh){ lp_config_set_int(linphone_core_get_config(marie->lc), "net", "recreate_sockets_when_network_is_up", 1); lp_config_set_int(linphone_core_get_config(pauline->lc), "net", "recreate_sockets_when_network_is_up", 1); } linphone_core_set_media_network_reachable(marie->lc, TRUE); BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (!call_ok) goto end; wait_for_until(marie->lc, pauline->lc, NULL, 0, 2000); if (use_ice) { /*wait for ICE reINVITE to complete*/ BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(check_ice(pauline,marie,LinphoneIceStateHostConnection)); } /*marie looses the SIP network and reconnects*/ linphone_core_set_sip_network_reachable(marie->lc, FALSE); linphone_core_set_media_network_reachable(marie->lc, FALSE); wait_for_until(marie->lc, pauline->lc, NULL, 0, 1000); /*marie will reconnect and register*/ linphone_core_set_sip_network_reachable(marie->lc, TRUE); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneRegistrationOk, 2)); wait_for_until(marie->lc, pauline->lc, NULL, 0, 5000); /*at this stage, no reINVITE is expected to be send*/ BC_ASSERT_EQUAL(marie->stat.number_of_LinphoneCallUpdating, 0, int, "%i"); /*now we notify the a reconnection of media network*/ linphone_core_set_media_network_reachable(marie->lc, TRUE); if (use_ice){ BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallUpdating, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallUpdatedByRemote, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 3)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 3)); /*now comes the ICE reINVITE*/ BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallUpdating, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallUpdatedByRemote, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 4)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 4)); }else{ BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallUpdating, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallUpdatedByRemote, 1)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &pauline->stat.number_of_LinphoneCallStreamsRunning, 2)); BC_ASSERT_TRUE(wait_for(marie->lc, pauline->lc, &marie->stat.number_of_LinphoneCallStreamsRunning, 2)); } /*check that media is back*/ check_media_direction(marie, linphone_core_get_current_call(marie->lc), lcs, LinphoneMediaDirectionSendRecv, LinphoneMediaDirectionInvalid); liblinphone_tester_check_rtcp(pauline, marie); if (use_ice) BC_ASSERT_TRUE(check_ice(pauline,marie,LinphoneIceStateHostConnection)); /*pauline shall be able to end the call without problem now*/ end_call(pauline, marie); end: bctbx_list_free(lcs); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } #ifdef SQLITE_STORAGE_ENABLED static void call_logs_if_no_db_set(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* laure = linphone_core_manager_new("laure_call_logs_rc"); BC_ASSERT_TRUE(bctbx_list_size(laure->lc->call_logs) == 10); BC_ASSERT_TRUE(call(marie, laure)); wait_for_until(marie->lc, laure->lc, NULL, 5, 1000); end_call(marie, laure); BC_ASSERT_TRUE(bctbx_list_size(laure->lc->call_logs) == 11); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(laure); } static void call_logs_migrate(void) { LinphoneCoreManager* laure = linphone_core_manager_new("laure_call_logs_rc"); char *logs_db = bc_tester_file("call_logs.db"); size_t i = 0; int incoming_count = 0, outgoing_count = 0, missed_count = 0, aborted_count = 0, decline_count = 0, video_enabled_count = 0; unlink(logs_db); BC_ASSERT_TRUE(bctbx_list_size(laure->lc->call_logs) == 10); linphone_core_set_call_logs_database_path(laure->lc, logs_db); BC_ASSERT_TRUE(linphone_core_get_call_history_size(laure->lc) == 10); for (; i < bctbx_list_size(laure->lc->call_logs); i++) { LinphoneCallLog *log = bctbx_list_nth_data(laure->lc->call_logs, (int)i); LinphoneCallStatus state = linphone_call_log_get_status(log); LinphoneCallDir direction = linphone_call_log_get_dir(log); if (state == LinphoneCallAborted) { aborted_count += 1; } else if (state == LinphoneCallMissed) { missed_count += 1; } else if (state == LinphoneCallDeclined) { decline_count += 1; } if (direction == LinphoneCallOutgoing) { outgoing_count += 1; } else { incoming_count += 1; } if (linphone_call_log_video_enabled(log)) { video_enabled_count += 1; } } BC_ASSERT_TRUE(incoming_count == 5); BC_ASSERT_TRUE(outgoing_count == 5); BC_ASSERT_TRUE(missed_count == 1); BC_ASSERT_TRUE(aborted_count == 3); BC_ASSERT_TRUE(decline_count == 2); BC_ASSERT_TRUE(video_enabled_count == 3); { LinphoneCallLog *log = linphone_core_get_last_outgoing_call_log(laure->lc); BC_ASSERT_PTR_NOT_NULL(log); if (log) { BC_ASSERT_EQUAL((int)log->start_date_time, 1441738272, int, "%d"); linphone_call_log_unref(log); log = NULL; } } laure->lc->call_logs = bctbx_list_free_with_data(laure->lc->call_logs, (void (*)(void*))linphone_call_log_unref); laure->lc->call_logs = call_logs_read_from_config_file(laure->lc); BC_ASSERT_TRUE(bctbx_list_size(laure->lc->call_logs) == 0); unlink(logs_db); ms_free(logs_db); linphone_core_manager_destroy(laure); } static void call_logs_sqlite_storage(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); char *logs_db = bc_tester_file("call_logs.db"); bctbx_list_t *logs = NULL; LinphoneCallLog *call_log = NULL; LinphoneAddress *laure = NULL; time_t user_data_time = time(NULL); time_t start_time = 0; unlink(logs_db); linphone_core_set_call_logs_database_path(marie->lc, logs_db); BC_ASSERT_TRUE(linphone_core_get_call_history_size(marie->lc) == 0); BC_ASSERT_TRUE(call(marie, pauline)); wait_for_until(marie->lc, pauline->lc, NULL, 5, 500); call_log = linphone_call_get_call_log(linphone_core_get_current_call(marie->lc)); start_time = linphone_call_log_get_start_date(call_log); linphone_call_log_set_user_data(call_log, &user_data_time); linphone_call_log_set_ref_key(call_log, "ref_key"); end_call(marie, pauline); BC_ASSERT_TRUE(linphone_core_get_call_history_size(marie->lc) == 1); logs = linphone_core_get_call_history_for_address(marie->lc, linphone_proxy_config_get_identity_address(linphone_core_get_default_proxy_config(pauline->lc))); BC_ASSERT_TRUE(bctbx_list_size(logs) == 1); bctbx_list_free_with_data(logs, (void (*)(void*))linphone_call_log_unref); laure = linphone_address_new("\"Laure\" "); logs = linphone_core_get_call_history_for_address(marie->lc, laure); BC_ASSERT_TRUE(bctbx_list_size(logs) == 0); linphone_address_unref(laure); logs = linphone_core_get_call_history_for_address(marie->lc, linphone_proxy_config_get_identity_address(linphone_core_get_default_proxy_config(pauline->lc))); if (BC_ASSERT_TRUE(bctbx_list_size(logs) == 1)) { const char *call_id; const char *ref_key; call_log = (LinphoneCallLog *)bctbx_list_get_data(logs); ref_key = linphone_call_log_get_ref_key(call_log); BC_ASSERT_EQUAL(linphone_call_log_get_dir(call_log), LinphoneCallOutgoing, int, "%d"); BC_ASSERT_LOWER(linphone_call_log_get_duration(call_log), 2, int, "%d"); BC_ASSERT_TRUE(linphone_address_equal( linphone_call_log_get_from_address(call_log), linphone_proxy_config_get_identity_address(linphone_core_get_default_proxy_config(marie->lc)))); BC_ASSERT_TRUE(linphone_address_equal( linphone_call_log_get_to_address(call_log), linphone_proxy_config_get_identity_address(linphone_core_get_default_proxy_config(pauline->lc)))); BC_ASSERT_PTR_NOT_NULL(linphone_call_log_get_local_stats(call_log)); BC_ASSERT_GREATER(linphone_call_log_get_quality(call_log), -1, float, "%.1f"); BC_ASSERT_PTR_NOT_NULL(ref_key); if (ref_key) { BC_ASSERT_STRING_EQUAL(ref_key, "ref_key"); } BC_ASSERT_PTR_EQUAL(linphone_call_log_get_user_data(call_log), &user_data_time); call_id = linphone_call_log_get_call_id(call_log); BC_ASSERT_PTR_NOT_NULL(call_id); { LinphoneCallLog* find_call_log = linphone_core_find_call_log_from_call_id(marie->lc, call_id); BC_ASSERT_PTR_NOT_NULL(find_call_log); if (find_call_log) linphone_call_log_unref(find_call_log); } BC_ASSERT_TRUE(linphone_address_equal( linphone_call_log_get_remote_address(call_log), linphone_proxy_config_get_identity_address(linphone_core_get_default_proxy_config(pauline->lc)))); BC_ASSERT_PTR_NOT_NULL(linphone_call_log_get_remote_stats(call_log)); BC_ASSERT_EQUAL(linphone_call_log_get_start_date(call_log), start_time, unsigned long long, "%llu"); BC_ASSERT_EQUAL(linphone_call_log_get_status(call_log), LinphoneCallSuccess, int, "%d"); } linphone_core_delete_call_log(marie->lc, (LinphoneCallLog *)bctbx_list_nth_data(logs, 0)); bctbx_list_free_with_data(logs, (void (*)(void*))linphone_call_log_unref); BC_ASSERT_TRUE(linphone_core_get_call_history_size(marie->lc) == 0); reset_counters(&marie->stat); reset_counters(&pauline->stat); BC_ASSERT_TRUE(call(marie, pauline)); end_call(marie, pauline); reset_counters(&marie->stat); reset_counters(&pauline->stat); BC_ASSERT_TRUE(call(marie, pauline)); end_call(marie, pauline); BC_ASSERT_TRUE(linphone_core_get_call_history_size(marie->lc) == 2); linphone_core_delete_call_history(marie->lc); BC_ASSERT_TRUE(linphone_core_get_call_history_size(marie->lc) == 0); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); unlink(logs_db); ms_free(logs_db); } #endif static void call_with_http_proxy(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new("pauline_rc"); bool_t call_ok; LinphoneCall *marie_call; LinphoneAddress *contact_addr; struct addrinfo *res=NULL; struct addrinfo hints = {0}; char ip[NI_MAXHOST]; int err; if (!transport_supported(LinphoneTransportTls)) { ms_message("Test skipped because no tls support"); goto end; } hints.ai_family = AF_INET; hints.ai_socktype = SOCK_STREAM; err = getaddrinfo("sip.linphone.org","8888", &hints, &res); if (err !=0){ ms_error("call_with_http_proxy(): getaddrinfo() error: %s", gai_strerror(err)); } BC_ASSERT_PTR_NOT_NULL(res); if (!res) goto end; BC_ASSERT_EQUAL(err=bctbx_getnameinfo(res->ai_addr, (socklen_t)res->ai_addrlen, ip, sizeof(ip)-1, NULL, 0, NI_NUMERICHOST), 0, int, "%i"); if (err != 0){ ms_error("call_with_http_proxy(): getnameinfo() error: %s", gai_strerror(err)); goto end; } freeaddrinfo(res); linphone_core_set_http_proxy_host(pauline->lc,"sip.linphone.org"); linphone_core_set_network_reachable(pauline->lc, FALSE); /*to make sure channel is restarted*/ linphone_core_set_network_reachable(pauline->lc, TRUE); BC_ASSERT_TRUE((call_ok=call(pauline,marie))); if (!call_ok) goto end; marie_call = linphone_core_get_current_call(marie->lc); contact_addr = linphone_address_new(linphone_call_get_remote_contact(marie_call)); BC_ASSERT_STRING_EQUAL(linphone_address_get_domain(contact_addr),ip); linphone_address_unref(contact_addr); end_call(marie, pauline); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void _call_with_rtcp_mux(bool_t caller_rtcp_mux, bool_t callee_rtcp_mux, bool_t with_ice,bool_t with_ice_reinvite){ LinphoneCoreManager * marie = linphone_core_manager_new( "marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); const LinphoneCallParams *params; bctbx_list_t *lcs = NULL; lcs = bctbx_list_append(lcs, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); if (caller_rtcp_mux){ lp_config_set_int(linphone_core_get_config(marie->lc), "rtp", "rtcp_mux", 1); } if (callee_rtcp_mux){ lp_config_set_int(linphone_core_get_config(pauline->lc), "rtp", "rtcp_mux", 1); } if (with_ice){ linphone_core_set_user_agent(pauline->lc, "Natted Linphone", NULL); linphone_core_set_user_agent(marie->lc, "Natted Linphone", NULL); linphone_core_set_firewall_policy(marie->lc, LinphonePolicyUseIce); linphone_core_set_firewall_policy(pauline->lc, LinphonePolicyUseIce); } if (!with_ice_reinvite) { lp_config_set_int(linphone_core_get_config(pauline->lc), "sip", "update_call_when_ice_completed", 0); lp_config_set_int(linphone_core_get_config(marie->lc), "sip", "update_call_when_ice_completed", 0); } if (!BC_ASSERT_TRUE(call(marie,pauline))) goto end; params = linphone_call_get_remote_params(linphone_core_get_current_call(pauline->lc)); BC_ASSERT_TRUE(caller_rtcp_mux == (linphone_call_params_get_custom_sdp_media_attribute(params, LinphoneStreamTypeAudio, "rtcp-mux") != NULL)); if (caller_rtcp_mux){ params = linphone_call_get_remote_params(linphone_core_get_current_call(marie->lc)); BC_ASSERT_TRUE(callee_rtcp_mux == (linphone_call_params_get_custom_sdp_media_attribute(params, LinphoneStreamTypeAudio, "rtcp-mux") != NULL)); } if (with_ice){ check_ice(marie, pauline, LinphoneIceStateHostConnection); } liblinphone_tester_check_rtcp(marie,pauline); if (caller_rtcp_mux && callee_rtcp_mux){ BC_ASSERT_EQUAL(marie->stat.number_of_rtcp_received_via_mux, marie->stat.number_of_rtcp_received, int, "%i"); BC_ASSERT_EQUAL(pauline->stat.number_of_rtcp_received_via_mux, pauline->stat.number_of_rtcp_received, int, "%i"); }else{ BC_ASSERT_TRUE(marie->stat.number_of_rtcp_received_via_mux == 0); BC_ASSERT_TRUE(pauline->stat.number_of_rtcp_received_via_mux == 0); } check_media_direction(pauline, linphone_core_get_current_call(pauline->lc), lcs, LinphoneMediaDirectionSendRecv, LinphoneMediaDirectionInvalid); end_call(marie,pauline); end: bctbx_list_free(lcs); linphone_core_manager_destroy(pauline); linphone_core_manager_destroy(marie); } static void call_with_rtcp_mux(void){ _call_with_rtcp_mux(TRUE, TRUE, FALSE,TRUE); } static void call_with_rtcp_mux_not_accepted(void){ _call_with_rtcp_mux(TRUE, FALSE, FALSE,TRUE); } static void call_with_ice_and_rtcp_mux(void){ _call_with_rtcp_mux(TRUE, TRUE, TRUE,TRUE); } static void call_with_ice_and_rtcp_mux_without_reinvite(void){ _call_with_rtcp_mux(TRUE, TRUE, TRUE,FALSE); } static void call_with_zrtp_configured_calling_base(LinphoneCoreManager *marie, LinphoneCoreManager *pauline) { if (ms_zrtp_available()) { bool_t call_ok; linphone_core_set_media_encryption(pauline->lc, LinphoneMediaEncryptionZRTP); BC_ASSERT_TRUE((call_ok=call(pauline,marie))); liblinphone_tester_check_rtcp(marie,pauline); BC_ASSERT_EQUAL(linphone_call_params_get_media_encryption(linphone_call_get_current_params(linphone_core_get_current_call(marie->lc))) , LinphoneMediaEncryptionZRTP, int, "%i"); BC_ASSERT_EQUAL(linphone_call_params_get_media_encryption(linphone_call_get_current_params(linphone_core_get_current_call(pauline->lc))) , LinphoneMediaEncryptionZRTP, int, "%i"); end_call(pauline, marie); } else { ms_warning("Test skipped, ZRTP not available"); } } static void call_with_zrtp_configured_callee_base(LinphoneCoreManager *marie, LinphoneCoreManager *pauline) { if (ms_zrtp_available()) { bool_t call_ok; linphone_core_set_media_encryption(marie->lc, LinphoneMediaEncryptionZRTP); BC_ASSERT_TRUE((call_ok=call(pauline,marie))); liblinphone_tester_check_rtcp(marie,pauline); BC_ASSERT_EQUAL(linphone_call_params_get_media_encryption(linphone_call_get_current_params(linphone_core_get_current_call(marie->lc))) , LinphoneMediaEncryptionZRTP, int, "%i"); BC_ASSERT_EQUAL(linphone_call_params_get_media_encryption(linphone_call_get_current_params(linphone_core_get_current_call(pauline->lc))) , LinphoneMediaEncryptionZRTP, int, "%i"); end_call(pauline, marie); } else { ms_warning("Test skipped, ZRTP not available"); } } /* * this test checks the 'dont_default_to_stun_candidates' mode, where the c= line is left to host * ip instead of stun candidate when ice is enabled*/ static void call_with_ice_with_default_candidate_not_stun(void){ LinphoneCoreManager * marie = linphone_core_manager_new( "marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); char localip[LINPHONE_IPADDR_SIZE]; bool_t call_ok; lp_config_set_int(marie->lc->config, "net", "dont_default_to_stun_candidates", 1); linphone_core_set_firewall_policy(marie->lc, LinphonePolicyUseIce); linphone_core_set_firewall_policy(pauline->lc, LinphonePolicyUseIce); linphone_core_get_local_ip(marie->lc, AF_INET, NULL, localip); call_ok = call(marie, pauline); if (call_ok){ check_ice(marie, pauline, LinphoneIceStateHostConnection); BC_ASSERT_STRING_EQUAL(marie->lc->current_call->localdesc->addr, localip); BC_ASSERT_STRING_EQUAL(pauline->lc->current_call->resultdesc->addr, localip); BC_ASSERT_STRING_EQUAL(marie->lc->current_call->localdesc->streams[0].rtp_addr, localip); BC_ASSERT_STRING_EQUAL(pauline->lc->current_call->resultdesc->streams[0].rtp_addr, ""); } end_call(marie, pauline); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_ice_without_stun(void){ LinphoneCoreManager * marie = linphone_core_manager_new( "marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); linphone_core_set_stun_server(marie->lc, NULL); linphone_core_set_stun_server(pauline->lc, NULL); _call_with_ice_base(marie, pauline, TRUE, TRUE, TRUE, FALSE); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_ice_without_stun2(void){ LinphoneCoreManager * marie = linphone_core_manager_new( "marie_rc"); LinphoneCoreManager *pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); //linphone_core_set_stun_server(marie->lc, NULL); linphone_core_set_stun_server(pauline->lc, NULL); _call_with_ice_base(marie, pauline, TRUE, TRUE, TRUE, FALSE); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_zrtp_configured_calling_side(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); call_with_zrtp_configured_calling_base(marie,pauline); /* now set other encryptions mode for receiver(marie), we shall always fall back to caller preference: ZRTP */ linphone_core_set_media_encryption(marie->lc, LinphoneMediaEncryptionDTLS); call_with_zrtp_configured_calling_base(marie,pauline); linphone_core_set_media_encryption(marie->lc, LinphoneMediaEncryptionSRTP); call_with_zrtp_configured_calling_base(marie,pauline); linphone_core_set_media_encryption(marie->lc, LinphoneMediaEncryptionNone); linphone_core_set_user_agent(pauline->lc, "Natted Linphone", NULL); linphone_core_set_user_agent(marie->lc, "Natted Linphone", NULL); call_with_zrtp_configured_calling_base(marie,pauline); linphone_core_set_firewall_policy(marie->lc,LinphonePolicyUseIce); linphone_core_set_firewall_policy(pauline->lc,LinphonePolicyUseIce); call_with_zrtp_configured_calling_base(marie,pauline); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_with_zrtp_configured_callee_side(void) { LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); call_with_zrtp_configured_callee_base(marie,pauline); linphone_core_set_user_agent(pauline->lc, "Natted Linphone", NULL); linphone_core_set_user_agent(marie->lc, "Natted Linphone", NULL); call_with_zrtp_configured_callee_base(marie,pauline); linphone_core_set_firewall_policy(marie->lc,LinphonePolicyUseIce); linphone_core_set_firewall_policy(pauline->lc,LinphonePolicyUseIce); call_with_zrtp_configured_callee_base(marie,pauline); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static bool_t quick_call(LinphoneCoreManager *m1, LinphoneCoreManager *m2){ linphone_core_invite_address(m1->lc, m2->identity); if (!BC_ASSERT_TRUE(wait_for(m1->lc, m2->lc, &m2->stat.number_of_LinphoneCallIncomingReceived, 1))) return FALSE; linphone_call_accept(linphone_core_get_current_call(m2->lc)); if (!BC_ASSERT_TRUE(wait_for(m1->lc, m2->lc, &m2->stat.number_of_LinphoneCallStreamsRunning, 1))) return FALSE; if (!BC_ASSERT_TRUE(wait_for(m1->lc, m2->lc, &m1->stat.number_of_LinphoneCallStreamsRunning, 1))) return FALSE; return TRUE; } static void call_with_encryption_mandatory(bool_t caller_has_encryption_mandatory){ LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new(transport_supported(LinphoneTransportTls) ? "pauline_rc" : "pauline_tcp_rc"); LinphoneCallStats *marie_stats, *pauline_stats; /*marie doesn't support ZRTP at all*/ marie->lc->zrtp_not_available_simulation=1; /*pauline requests encryption to be mandatory*/ linphone_core_set_media_encryption(pauline->lc, LinphoneMediaEncryptionZRTP); linphone_core_set_media_encryption_mandatory(pauline->lc, TRUE); if (!caller_has_encryption_mandatory){ if (!BC_ASSERT_TRUE(quick_call(marie, pauline))) goto end; }else{ if (!BC_ASSERT_TRUE(quick_call(pauline, marie))) goto end; } wait_for_until(pauline->lc, marie->lc, NULL, 0, 2000); /*assert that no RTP packets have been sent or received by Pauline*/ /*testing packet_sent doesn't work, because packets dropped by the transport layer are counted as if they were sent.*/ #if 0 BC_ASSERT_EQUAL(linphone_call_get_audio_stats(linphone_core_get_current_call(pauline->lc))->rtp_stats.packet_sent, 0, int, "%i"); #endif /*however we can trust packet_recv from the other party instead */ marie_stats = linphone_call_get_audio_stats(linphone_core_get_current_call(marie->lc)); pauline_stats = linphone_call_get_audio_stats(linphone_core_get_current_call(pauline->lc)); BC_ASSERT_EQUAL((int)marie_stats->rtp_stats.packet_recv, 0, int, "%i"); BC_ASSERT_EQUAL((int)pauline_stats->rtp_stats.packet_recv, 0, int, "%i"); linphone_call_stats_unref(marie_stats); linphone_call_stats_unref(pauline_stats); end_call(marie, pauline); end: linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } static void call_from_plain_rtp_to_zrtp(void){ call_with_encryption_mandatory(FALSE); } static void call_from_zrtp_to_plain_rtp(void){ call_with_encryption_mandatory(TRUE); } static void v6_to_v4_call_without_relay(void){ LinphoneCoreManager* marie; LinphoneCoreManager* pauline; bctbx_list_t *lcs = NULL; if (liblinphone_tester_ipv4_available() && liblinphone_tester_ipv6_available()){ marie = linphone_core_manager_new("marie_rc"); pauline = linphone_core_manager_new2("pauline_tcp_rc", FALSE); lcs = bctbx_list_append(lcs, marie->lc); lcs = bctbx_list_append(lcs, pauline->lc); linphone_core_enable_ipv6(pauline->lc, FALSE); /*RTP symmetric must be enabled for this test to pass, because the IPv4-only client cannot send by itself to * the IPv6 address in the SDP*/ lp_config_set_int(linphone_core_get_config(pauline->lc), "rtp", "symmetric", 1); linphone_core_manager_start(pauline, TRUE); if (BC_ASSERT_TRUE(call(marie,pauline))){ check_media_direction(marie, linphone_core_get_current_call(marie->lc), lcs, LinphoneMediaDirectionSendRecv, LinphoneMediaDirectionInvalid); liblinphone_tester_check_rtcp(marie,pauline); end_call(marie,pauline); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); bctbx_list_free(lcs); }else ms_warning("Test skipped, dual stack not available"); } static void v6_call_over_nat_64(void){ LinphoneCoreManager* marie; LinphoneCoreManager* pauline; if (!liblinphone_tester_ipv4_available() && liblinphone_tester_ipv6_available()){ marie = linphone_core_manager_new("marie_nat64_rc"); pauline = linphone_core_manager_new("pauline_nat64_rc"); linphone_core_set_user_agent(pauline->lc, "Natted Linphone", NULL); linphone_core_set_user_agent(marie->lc, "Natted Linphone", NULL); BC_ASSERT_TRUE(wait_for_until(pauline->lc, NULL, &pauline->stat.number_of_LinphoneRegistrationOk, 1, 2000)); BC_ASSERT_TRUE(wait_for_until(pauline->lc, NULL, &marie->stat.number_of_LinphoneRegistrationOk, 1, 2000)); BC_ASSERT_TRUE(call(marie,pauline)); liblinphone_tester_check_rtcp(marie,pauline); end_call(marie,pauline); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); }else ms_warning("Test skipped, no ipv6 nat64 available"); } static void call_with_ice_in_ipv4_with_v6_enabled(void) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; if (liblinphone_tester_ipv4_available() && liblinphone_tester_ipv6_available()){ marie = linphone_core_manager_new("marie_v4proxy_rc"); pauline = linphone_core_manager_new("pauline_v4proxy_rc"); _call_with_ice_base(marie,pauline,TRUE,TRUE,TRUE,TRUE); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } else ms_warning("Test skipped, need both ipv6 and v4 available"); } static void call_with_ice_ipv4_to_ipv6(void) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; if (liblinphone_tester_ipv4_available() && liblinphone_tester_ipv6_available()){ marie = linphone_core_manager_new("marie_v4proxy_rc"); pauline = linphone_core_manager_new("pauline_tcp_rc"); _call_with_ice_base(marie,pauline,TRUE,TRUE,TRUE,TRUE); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } else ms_warning("Test skipped, need both ipv6 and v4 available"); } static void call_with_ice_ipv6_to_ipv4(void) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; if (liblinphone_tester_ipv4_available() && liblinphone_tester_ipv6_available()){ marie = linphone_core_manager_new("marie_rc"); pauline = linphone_core_manager_new("pauline_v4proxy_rc"); _call_with_ice_base(marie, pauline,TRUE,TRUE,TRUE,TRUE); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } else ms_warning("Test skipped, need both ipv6 and v4 available"); } static void call_with_ice_ipv6_to_ipv6(void) { LinphoneCoreManager* marie; LinphoneCoreManager* pauline; if (liblinphone_tester_ipv4_available() && liblinphone_tester_ipv6_available()){ marie = linphone_core_manager_new("marie_rc"); pauline = linphone_core_manager_new("pauline_tcp_rc"); _call_with_ice_base(marie, pauline,TRUE,TRUE,TRUE,TRUE); linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); } else ms_warning("Test skipped, need both ipv6 and v4 available"); } static void my_call_state_changed_cb(LinphoneCore *lc, LinphoneCall*call, LinphoneCallState state, const char *text){ if (state == LinphoneCallError){ linphone_core_set_network_reachable(lc, FALSE); } } /*This test simulates a case where a 404 not found is received from server when attempting to make a call, * and the app decides to immediately shutdown the network reachability*/ static void call_with_network_reachable_down_in_callback(void){ LinphoneCoreManager* marie; LinphoneCoreCbs *cbs = linphone_factory_create_core_cbs(linphone_factory_get()); LinphoneCall *call; linphone_core_cbs_set_call_state_changed(cbs, my_call_state_changed_cb); marie = linphone_core_manager_new("laure_rc_udp"); linphone_core_add_callbacks(marie->lc, cbs); call = linphone_core_invite(marie->lc, "inexistant_username_xbfuuuf"); BC_ASSERT_PTR_NOT_NULL(call); BC_ASSERT_TRUE(wait_for(marie->lc, NULL, &marie->stat.number_of_LinphoneCallError, 1)); linphone_core_cbs_unref(cbs); linphone_core_manager_destroy(marie); } static void recreate_zrtpdb_when_corrupted(void) { #ifdef SQLITE_STORAGE_ENABLED LinphoneCoreManager* marie = linphone_core_manager_new("marie_rc"); LinphoneCoreManager* pauline = linphone_core_manager_new("pauline_tcp_rc"); if (BC_ASSERT_TRUE(linphone_core_media_encryption_supported(marie->lc,LinphoneMediaEncryptionZRTP))) { void *db; const char* db_file; const char *filepath; const char *filepath2; const char *corrupt = "corrupt mwahahahaha"; FILE *f; remove(bc_tester_file("tmpZIDCacheMarie.sqlite")); filepath = bc_tester_file("tmpZIDCacheMarie.sqlite"); remove(bc_tester_file("tmpZIDCachePauline.sqlite")); filepath2 = bc_tester_file("tmpZIDCachePauline.sqlite"); linphone_core_set_media_encryption(marie->lc,LinphoneMediaEncryptionZRTP); linphone_core_set_media_encryption(pauline->lc,LinphoneMediaEncryptionZRTP); linphone_core_set_zrtp_secrets_file(marie->lc, filepath); linphone_core_set_zrtp_secrets_file(pauline->lc, filepath2); BC_ASSERT_TRUE(call(pauline,marie)); linphone_call_set_authentication_token_verified(linphone_core_get_current_call(marie->lc), TRUE); linphone_call_set_authentication_token_verified(linphone_core_get_current_call(pauline->lc), TRUE); BC_ASSERT_TRUE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(marie->lc))); BC_ASSERT_TRUE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(pauline->lc))); end_call(marie, pauline); db = linphone_core_get_zrtp_cache_db(marie->lc); BC_ASSERT_PTR_NOT_NULL(db); BC_ASSERT_TRUE(call(pauline,marie)); BC_ASSERT_TRUE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(marie->lc))); BC_ASSERT_TRUE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(pauline->lc))); end_call(marie, pauline); //Corrupt db file db_file = linphone_core_get_zrtp_secrets_file(marie->lc); BC_ASSERT_PTR_NOT_NULL(db_file); f = fopen(db_file, "wb"); fwrite(corrupt, 1, sizeof(corrupt), f); fclose(f); //Simulate relaunch of linphone core marie linphone_core_set_zrtp_secrets_file(marie->lc, filepath); db = linphone_core_get_zrtp_cache_db(marie->lc); BC_ASSERT_PTR_NULL(db); BC_ASSERT_TRUE(call(pauline,marie)); linphone_call_set_authentication_token_verified(linphone_core_get_current_call(marie->lc), TRUE); linphone_call_set_authentication_token_verified(linphone_core_get_current_call(pauline->lc), TRUE); BC_ASSERT_TRUE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(marie->lc))); BC_ASSERT_TRUE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(pauline->lc))); end_call(marie, pauline); BC_ASSERT_TRUE(call(pauline,marie)); BC_ASSERT_FALSE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(marie->lc))); BC_ASSERT_FALSE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(pauline->lc))); end_call(marie, pauline); //Db file should be recreated after corruption //Simulate relaunch of linphone core marie linphone_core_set_zrtp_secrets_file(marie->lc, filepath); BC_ASSERT_TRUE(call(pauline,marie)); linphone_call_set_authentication_token_verified(linphone_core_get_current_call(marie->lc), TRUE); linphone_call_set_authentication_token_verified(linphone_core_get_current_call(pauline->lc), TRUE); BC_ASSERT_TRUE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(marie->lc))); BC_ASSERT_TRUE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(pauline->lc))); end_call(marie, pauline); db = linphone_core_get_zrtp_cache_db(marie->lc); BC_ASSERT_PTR_NOT_NULL(db); db_file = linphone_core_get_zrtp_secrets_file(marie->lc); BC_ASSERT_PTR_NOT_NULL(db_file); BC_ASSERT_TRUE(call(pauline,marie)); BC_ASSERT_TRUE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(marie->lc))); BC_ASSERT_TRUE(linphone_call_get_authentication_token_verified(linphone_core_get_current_call(pauline->lc))); end_call(marie, pauline); } linphone_core_manager_destroy(marie); linphone_core_manager_destroy(pauline); #endif /* SQLITE_STORAGE_ENABLED */ } test_t call_tests[] = { TEST_NO_TAG("Early declined call", early_declined_call), TEST_NO_TAG("Call declined", call_declined), TEST_NO_TAG("Call declined with error", call_declined_with_error), TEST_NO_TAG("Cancelled call", cancelled_call), TEST_NO_TAG("Early cancelled call", early_cancelled_call), TEST_NO_TAG("Call with DNS timeout", call_with_dns_time_out), TEST_NO_TAG("Cancelled ringing call", cancelled_ringing_call), TEST_NO_TAG("Call busy when calling self", call_busy_when_calling_self), TEST_NO_TAG("Simple call", simple_call), TEST_NO_TAG("Simple call with no SIP transport", simple_call_with_no_sip_transport), TEST_NO_TAG("Simple call with UDP", simple_call_with_udp), TEST_ONE_TAG("Call terminated automatically by linphone_core_destroy", automatic_call_termination, "LeaksMemory"), TEST_NO_TAG("Call with http proxy", call_with_http_proxy), TEST_NO_TAG("Call with timed-out bye", call_with_timed_out_bye), TEST_NO_TAG("Direct call over IPv6", direct_call_over_ipv6), TEST_NO_TAG("Call IPv6 to IPv4 without relay", v6_to_v4_call_without_relay), TEST_NO_TAG("IPv6 call over NAT64", v6_call_over_nat_64), TEST_ONE_TAG("Call with ICE in IPv4 with IPv6 enabled", call_with_ice_in_ipv4_with_v6_enabled, "ICE"), TEST_ONE_TAG("Call with ICE IPv4 to IPv6", call_with_ice_ipv4_to_ipv6, "ICE"), TEST_ONE_TAG("Call with ICE IPv6 to IPv4", call_with_ice_ipv6_to_ipv4, "ICE"), TEST_ONE_TAG("Call with ICE IPv6 to IPv6", call_with_ice_ipv6_to_ipv6, "ICE"), TEST_NO_TAG("Outbound call with multiple proxy possible", call_outbound_with_multiple_proxy), TEST_NO_TAG("Audio call recording", audio_call_recording_test), #if 0 /* not yet activated because not implemented */ TEST_NO_TAG("Multiple answers to a call", multiple_answers_call), #endif TEST_NO_TAG("Multiple answers to a call with media relay", multiple_answers_call_with_media_relay), TEST_NO_TAG("Call with media relay", call_with_media_relay), TEST_NO_TAG("Call with media relay (random ports)", call_with_media_relay_random_ports), TEST_NO_TAG("Simple call compatibility mode", simple_call_compatibility_mode), TEST_NO_TAG("Early-media call", early_media_call), TEST_ONE_TAG("Early-media call with ICE", early_media_call_with_ice, "ICE"), TEST_NO_TAG("Early-media call with ringing", early_media_call_with_ringing), TEST_NO_TAG("Early-media call with ringing and network changing", early_media_call_with_ringing_and_network_changing), TEST_NO_TAG("Early-media call with updated media session", early_media_call_with_session_update), TEST_NO_TAG("Early-media call with updated codec", early_media_call_with_codec_update), TEST_NO_TAG("Call terminated by caller", call_terminated_by_caller), TEST_NO_TAG("Call without SDP", call_with_no_sdp), TEST_NO_TAG("Call without SDP and ACK without SDP", call_with_no_sdp_ack_without_sdp), TEST_NO_TAG("Call paused resumed", call_paused_resumed), TEST_NO_TAG("Call paused resumed with sip packets looses", call_paused_resumed_with_sip_packets_losses), TEST_NO_TAG("Call paused by both parties", call_paused_by_both), TEST_NO_TAG("Call paused resumed with loss", call_paused_resumed_with_loss), TEST_NO_TAG("Call paused resumed from callee", call_paused_resumed_from_callee), TEST_NO_TAG("SRTP call", srtp_call), TEST_NO_TAG("ZRTP call", zrtp_call), TEST_NO_TAG("ZRTP silent call", zrtp_silent_call), TEST_NO_TAG("ZRTP SAS call", zrtp_sas_call), TEST_NO_TAG("ZRTP Cipher call", zrtp_cipher_call), TEST_NO_TAG("DTLS SRTP call", dtls_srtp_call), TEST_NO_TAG("DTLS SRTP call with media relay", dtls_srtp_call_with_media_realy), TEST_NO_TAG("SRTP call with declined srtp", call_with_declined_srtp), TEST_NO_TAG("SRTP call paused and resumed", call_srtp_paused_and_resumed), TEST_NO_TAG("Call with file player", call_with_file_player), TEST_NO_TAG("Call with mkv file player", call_with_mkv_file_player), TEST_ONE_TAG("Audio call with ICE no matching audio codecs", audio_call_with_ice_no_matching_audio_codecs, "ICE"), TEST_ONE_TAG("SRTP ice call", srtp_ice_call, "ICE"), TEST_ONE_TAG("ZRTP ice call", zrtp_ice_call, "ICE"), TEST_ONE_TAG("ZRTP ice call with relay", zrtp_ice_call_with_relay, "ICE"), TEST_ONE_TAG("DTLS SRTP ice call", dtls_srtp_ice_call, "ICE"), TEST_ONE_TAG("DTLS ice call with relay", dtls_ice_call_with_relay, "ICE"), TEST_NO_TAG("Call with privacy", call_with_privacy), TEST_NO_TAG("Call with privacy 2", call_with_privacy2), TEST_NO_TAG("Call rejected because of wrong credential", call_rejected_because_wrong_credentials), TEST_NO_TAG("Call rejected without 403 because of wrong credential", call_rejected_without_403_because_wrong_credentials), TEST_NO_TAG("Call rejected without 403 because of wrong credential and no auth req cb", call_rejected_without_403_because_wrong_credentials_no_auth_req_cb), TEST_ONE_TAG("Call with ICE", call_with_ice, "ICE"), TEST_ONE_TAG("Call with ICE IPv6", call_with_ice_ipv6, "ICE"), TEST_ONE_TAG("Call with ICE without SDP", call_with_ice_no_sdp, "ICE"), TEST_ONE_TAG("Call with ICE (random ports)", call_with_ice_random_ports, "ICE"), TEST_ONE_TAG("Call with ICE (forced relay)", call_with_ice_forced_relay, "ICE"), TEST_ONE_TAG("Call from ICE to not ICE", ice_to_not_ice, "ICE"), TEST_ONE_TAG("Call from not ICE to ICE", not_ice_to_ice, "ICE"), TEST_ONE_TAG("Call with ICE added by reINVITE", ice_added_by_reinvite, "ICE"), TEST_NO_TAG("Call with custom headers", call_with_custom_headers), TEST_NO_TAG("Call with custom SDP attributes", call_with_custom_sdp_attributes), TEST_NO_TAG("Call caller with custom header or sdp", call_caller_with_custom_header_or_sdp_attributes), TEST_NO_TAG("Call callee with custom header or sdp", call_callee_with_custom_header_or_sdp_attributes), TEST_NO_TAG("Call established with rejected INFO", call_established_with_rejected_info), TEST_NO_TAG("Call established with rejected RE-INVITE", call_established_with_rejected_reinvite), TEST_NO_TAG("Call established with rejected incoming RE-INVITE", call_established_with_rejected_incoming_reinvite), TEST_NO_TAG("Call established with rejected RE-INVITE in error", call_established_with_rejected_reinvite_with_error), TEST_NO_TAG("Call established with rejected RE-INVITE with trans pending error", call_established_with_rejected_reinvite_with_trans_pending_error), TEST_NO_TAG("Call established with complex rejected operation", call_established_with_complex_rejected_operation), TEST_NO_TAG("Call established with rejected info during re-invite", call_established_with_rejected_info_during_reinvite), TEST_NO_TAG("Call redirected by callee", call_redirect), TEST_NO_TAG("Call with specified codec bitrate", call_with_specified_codec_bitrate), TEST_NO_TAG("Call with no audio codec", call_with_no_audio_codec), TEST_NO_TAG("Call with in-dialog UPDATE request", call_with_in_dialog_update), TEST_NO_TAG("Call with in-dialog very early call request", call_with_very_early_call_update), TEST_NO_TAG("Call with in-dialog codec change", call_with_in_dialog_codec_change), TEST_NO_TAG("Call with in-dialog codec change no sdp", call_with_in_dialog_codec_change_no_sdp), TEST_NO_TAG("Call with pause no SDP on resume", call_with_paused_no_sdp_on_resume), TEST_NO_TAG("Call with early media and no SDP in 200 Ok", call_with_early_media_and_no_sdp_in_200), TEST_ONE_TAG("Call with ICE and no SDP in 200 OK", call_with_early_media_ice_and_no_sdp_in_200, "ICE"), TEST_NO_TAG("Call with custom supported tags", call_with_custom_supported_tags), TEST_NO_TAG("Call log from taken from asserted id", call_log_from_taken_from_p_asserted_id), TEST_NO_TAG("Incoming INVITE with invalid SDP", incoming_invite_with_invalid_sdp), TEST_NO_TAG("Outgoing INVITE with invalid ACK SDP", outgoing_invite_with_invalid_sdp), TEST_NO_TAG("Call with generic CN", call_with_generic_cn), TEST_NO_TAG("Call with transport change after released", call_with_transport_change_after_released), TEST_NO_TAG("Unsuccessful call with transport change after released", unsucessfull_call_with_transport_change_after_released), TEST_NO_TAG("Simple stereo call with L16", simple_stereo_call_l16), TEST_NO_TAG("Simple stereo call with opus", simple_stereo_call_opus), TEST_NO_TAG("Simple mono call with opus", simple_mono_call_opus), TEST_NO_TAG("Call with FQDN in SDP", call_with_fqdn_in_sdp), TEST_NO_TAG("Call with RTP IO mode", call_with_rtp_io_mode), TEST_NO_TAG("Call with generic NACK RTCP feedback", call_with_generic_nack_rtcp_feedback), TEST_NO_TAG("Call with complex late offering", call_with_complex_late_offering), #ifdef SQLITE_STORAGE_ENABLED TEST_NO_TAG("Call log working if no db set", call_logs_if_no_db_set), TEST_NO_TAG("Call log storage migration from rc to db", call_logs_migrate), TEST_NO_TAG("Call log storage in sqlite database", call_logs_sqlite_storage), #endif TEST_NO_TAG("Call with custom RTP Modifier", call_with_custom_rtp_modifier), TEST_NO_TAG("Call paused resumed with custom RTP Modifier", call_paused_resumed_with_custom_rtp_modifier), TEST_NO_TAG("Call record with custom RTP Modifier", call_record_with_custom_rtp_modifier), TEST_NO_TAG("Call with network switch", call_with_network_switch), TEST_NO_TAG("Call with network switch and no recovery possible", call_with_network_switch_no_recovery), TEST_ONE_TAG("Recovered call on network switch in early state 1", recovered_call_on_network_switch_in_early_state_1, "CallRecovery"), TEST_ONE_TAG("Recovered call on network switch in early state 1 (udp caller)", recovered_call_on_network_switch_in_early_state_1_udp, "CallRecovery"), TEST_ONE_TAG("Recovered call on network switch in early state 2", recovered_call_on_network_switch_in_early_state_2, "CallRecovery"), TEST_ONE_TAG("Recovered call on network switch in early state 3", recovered_call_on_network_switch_in_early_state_3, "CallRecovery"), TEST_ONE_TAG("Recovered call on network switch in early state 4", recovered_call_on_network_switch_in_early_state_4, "CallRecovery"), TEST_ONE_TAG("Recovered call on network switch during re-invite 1", recovered_call_on_network_switch_during_reinvite_1, "CallRecovery"), TEST_ONE_TAG("Recovered call on network switch during re-invite 2", recovered_call_on_network_switch_during_reinvite_2, "CallRecovery"), TEST_ONE_TAG("Recovered call on network switch during re-invite 3", recovered_call_on_network_switch_during_reinvite_3, "CallRecovery"), TEST_ONE_TAG("Recovered call on network switch during re-invite 4", recovered_call_on_network_switch_during_reinvite_4, "CallRecovery"), TEST_ONE_TAG("Call with network switch in paused state", call_with_network_switch_in_paused_state, "CallRecovery"), TEST_ONE_TAG("Call with network switch in paused by remote state", call_with_network_switch_in_paused_by_remote_state, "CallRecovery"), TEST_ONE_TAG("Call with network switch and ICE", call_with_network_switch_and_ice, "ICE"), TEST_ONE_TAG("Call with network switch, ICE and RTT", call_with_network_switch_ice_and_rtt, "ICE"), TEST_NO_TAG("Call with network switch with socket refresh", call_with_network_switch_and_socket_refresh), TEST_NO_TAG("Call with SIP and RTP independant switches", call_with_sip_and_rtp_independant_switches), TEST_NO_TAG("Call with rtcp-mux", call_with_rtcp_mux), TEST_NO_TAG("Call with rtcp-mux not accepted", call_with_rtcp_mux_not_accepted), TEST_ONE_TAG("Call with ICE and rtcp-mux", call_with_ice_and_rtcp_mux, "ICE"), TEST_ONE_TAG("Call with ICE and rtcp-mux without ICE re-invite", call_with_ice_and_rtcp_mux_without_reinvite, "ICE"), TEST_ONE_TAG("Call with ICE with default candidate not stun", call_with_ice_with_default_candidate_not_stun, "ICE"), TEST_ONE_TAG("Call with ICE without stun server", call_with_ice_without_stun, "ICE"), TEST_ONE_TAG("Call with ICE without stun server one side", call_with_ice_without_stun2, "ICE"), TEST_NO_TAG("Call with ZRTP configured calling side only", call_with_zrtp_configured_calling_side), TEST_NO_TAG("Call with ZRTP configured receiver side only", call_with_zrtp_configured_callee_side), TEST_NO_TAG("Call from plain RTP to ZRTP mandatory should be silent", call_from_plain_rtp_to_zrtp), TEST_NO_TAG("Call ZRTP mandatory to plain RTP should be silent", call_from_zrtp_to_plain_rtp), TEST_NO_TAG("Call with network reachable down in callback", call_with_network_reachable_down_in_callback), TEST_NO_TAG("Call terminated with reason", terminate_call_with_error), TEST_NO_TAG("Call cancelled with reason", cancel_call_with_error), TEST_NO_TAG("Call accepted, other ringing device receive CANCEL with reason", cancel_other_device_after_accept), TEST_NO_TAG("Call declined, other ringing device receive CANCEL with reason", cancel_other_device_after_decline), TEST_NO_TAG("Recreate ZRTP db file when corrupted", recreate_zrtpdb_when_corrupted) }; test_suite_t call_test_suite = {"Single Call", NULL, NULL, liblinphone_tester_before_each, liblinphone_tester_after_each, sizeof(call_tests) / sizeof(call_tests[0]), call_tests};