From 36628c4f0dbd34f48031d1cd62a7d560e791f94c Mon Sep 17 00:00:00 2001 From: Sylvain Berfini <sylvain.berfini@belledonne-communications.com> Date: Mon, 10 Jul 2023 11:30:31 +0200 Subject: [PATCH] Fixed find friend by phone number when number has international prefix but no '+' + improved performances a bit --- coreapi/friend.c | 91 +++++++++++++++++++++++--------- coreapi/friendlist.c | 37 +++++++++++-- coreapi/private_functions.h | 6 +++ include/linphone/api/c-account.h | 4 +- src/c-wrapper/api/c-account.cpp | 47 +++++++++++------ tester/setup_tester.c | 38 +++++++++++-- 6 files changed, 171 insertions(+), 52 deletions(-) diff --git a/coreapi/friend.c b/coreapi/friend.c index b736eaabc9..0e9d7f4260 100644 --- a/coreapi/friend.c +++ b/coreapi/friend.c @@ -576,6 +576,31 @@ bctbx_list_t *linphone_friend_get_phone_numbers_with_label(const LinphoneFriend return NULL; } +bool_t _linphone_friend_has_phone_number(const LinphoneFriend *lf, + const LinphoneAccount *account, + const char *normalized_phone_number) { + if (!lf || !normalized_phone_number) return FALSE; + + bool_t found = FALSE; + bctbx_list_t *numbers = linphone_friend_get_phone_numbers(lf); + bctbx_list_t *it = NULL; + for (it = numbers; it != NULL; it = bctbx_list_next(it)) { + const char *value = (const char *)bctbx_list_get_data(it); + char *normalized_value = linphone_account_normalize_phone_number(account, value); + if (normalized_value) { + if (strcmp(normalized_value, normalized_phone_number) == 0) { + found = TRUE; + ms_free(normalized_value); + break; + } + ms_free(normalized_value); + } + } + bctbx_list_free(numbers); + + return found; +} + bool_t linphone_friend_has_phone_number(const LinphoneFriend *lf, const char *phoneNumber) { if (!lf || !phoneNumber) return FALSE; @@ -587,6 +612,11 @@ bool_t linphone_friend_has_phone_number(const LinphoneFriend *lf, const char *ph return FALSE; } + if (!linphone_core_vcard_supported()) { + ms_warning("SDK built without vCard support, can't do a phone number search without it"); + return FALSE; + } + bool_t found = FALSE; const bctbx_list_t *elem; const bctbx_list_t *accounts = linphone_core_get_account_list(lf->lc); @@ -594,21 +624,7 @@ bool_t linphone_friend_has_phone_number(const LinphoneFriend *lf, const char *ph account = (LinphoneAccount *)bctbx_list_get_data(elem); char *normalized_phone_number = linphone_account_normalize_phone_number(account, phoneNumber); - if (linphone_core_vcard_supported()) { - bctbx_list_t *numbers = linphone_friend_get_phone_numbers(lf); - bctbx_list_t *it = NULL; - for (it = numbers; it != NULL; it = bctbx_list_next(it)) { - const char *value = (const char *)bctbx_list_get_data(it); - char *normalized_value = linphone_account_normalize_phone_number(account, value); - if (normalized_value && strcmp(normalized_value, normalized_phone_number) == 0) { - found = TRUE; - ms_free(normalized_value); - break; - } - if (normalized_value) ms_free(normalized_value); - } - bctbx_list_free(numbers); - } + found = _linphone_friend_has_phone_number(lf, account, normalized_phone_number); if (normalized_phone_number) ms_free(normalized_phone_number); if (found) break; @@ -617,6 +633,40 @@ bool_t linphone_friend_has_phone_number(const LinphoneFriend *lf, const char *ph return found; } +LinphoneFriend *linphone_core_find_friend_by_phone_number(const LinphoneCore *lc, const char *phoneNumber) { + LinphoneAccount *account = linphone_core_get_default_account(lc); + // Account can be null, both linphone_account_is_phone_number and linphone_account_normalize_phone_number can handle + // it + if (phoneNumber == NULL || !linphone_account_is_phone_number(account, phoneNumber)) { + ms_warning("Phone number [%s] isn't valid", phoneNumber); + return NULL; + } + + if (!linphone_core_vcard_supported()) { + ms_warning("SDK built without vCard support, can't do a phone number search without it"); + return NULL; + } + + const bctbx_list_t *elem; + const bctbx_list_t *accounts = linphone_core_get_account_list(lc); + for (elem = accounts; elem != NULL; elem = bctbx_list_next(elem)) { + account = (LinphoneAccount *)bctbx_list_get_data(elem); + char *normalized_phone_number = linphone_account_normalize_phone_number(account, phoneNumber); + + bctbx_list_t *lists = lc->friends_lists; + LinphoneFriend *lf = NULL; + while (lists && !lf) { + LinphoneFriendList *list = (LinphoneFriendList *)bctbx_list_get_data(lists); + lf = _linphone_friend_list_find_friend_by_phone_number(list, account, normalized_phone_number); + lists = bctbx_list_next(lists); + } + + ms_free(normalized_phone_number); + if (lf) return lf; + } + return NULL; +} + void linphone_friend_remove_phone_number(LinphoneFriend *lf, const char *phone) { if (!lf || !phone || !lf->vcard) return; @@ -1182,17 +1232,6 @@ LinphoneFriend *linphone_core_find_friend(const LinphoneCore *lc, const Linphone return lf; } -LinphoneFriend *linphone_core_find_friend_by_phone_number(const LinphoneCore *lc, const char *phoneNumber) { - bctbx_list_t *lists = lc->friends_lists; - LinphoneFriend *lf = NULL; - while (lists && !lf) { - LinphoneFriendList *list = (LinphoneFriendList *)bctbx_list_get_data(lists); - lf = linphone_friend_list_find_friend_by_phone_number(list, phoneNumber); - lists = bctbx_list_next(lists); - } - return lf; -} - bctbx_list_t *linphone_core_find_friends(const LinphoneCore *lc, const LinphoneAddress *addr) { bctbx_list_t *result = NULL; bctbx_list_t *lists = lc->friends_lists; diff --git a/coreapi/friendlist.c b/coreapi/friendlist.c index a2f80aff5d..997ea07bcf 100644 --- a/coreapi/friendlist.c +++ b/coreapi/friendlist.c @@ -1098,14 +1098,14 @@ LinphoneFriend *linphone_friend_list_find_friend_by_address(const LinphoneFriend return lf; } -LinphoneFriend *linphone_friend_list_find_friend_by_phone_number(const LinphoneFriendList *list, - const char *phoneNumber) { +LinphoneFriend *_linphone_friend_list_find_friend_by_phone_number(const LinphoneFriendList *list, + const LinphoneAccount *account, + const char *normalized_phone_number) { LinphoneFriend *result = NULL; - const bctbx_list_t *elem; for (elem = list->friends; elem != NULL; elem = bctbx_list_next(elem)) { LinphoneFriend *lf = (LinphoneFriend *)bctbx_list_get_data(elem); - if (linphone_friend_has_phone_number(lf, phoneNumber)) { + if (_linphone_friend_has_phone_number(lf, account, normalized_phone_number)) { result = lf; break; } @@ -1114,6 +1114,35 @@ LinphoneFriend *linphone_friend_list_find_friend_by_phone_number(const LinphoneF return result; } +LinphoneFriend *linphone_friend_list_find_friend_by_phone_number(const LinphoneFriendList *list, + const char *phoneNumber) { + LinphoneFriend *result = NULL; + LinphoneAccount *account = linphone_core_get_default_account(list->lc); + // Account can be null, both linphone_account_is_phone_number and linphone_account_normalize_phone_number can handle + // it + if (phoneNumber == NULL || !linphone_account_is_phone_number(account, phoneNumber)) { + ms_warning("Phone number [%s] isn't valid", phoneNumber); + return result; + } + + if (!linphone_core_vcard_supported()) { + ms_warning("SDK built without vCard support, can't do a phone number search without it"); + return NULL; + } + + const bctbx_list_t *elem; + const bctbx_list_t *accounts = linphone_core_get_account_list(list->lc); + for (elem = accounts; elem != NULL; elem = bctbx_list_next(elem)) { + account = (LinphoneAccount *)bctbx_list_get_data(elem); + char *normalized_phone_number = linphone_account_normalize_phone_number(account, phoneNumber); + result = _linphone_friend_list_find_friend_by_phone_number(list, account, normalized_phone_number); + ms_free(normalized_phone_number); + if (result) return result; + } + + return result; +} + bctbx_list_t *linphone_friend_list_find_friends_by_address(const LinphoneFriendList *list, const LinphoneAddress *address) { LinphoneAddress *clean_addr = linphone_address_clone(address); diff --git a/coreapi/private_functions.h b/coreapi/private_functions.h index 81d0b896c3..5a8ef6f1f7 100644 --- a/coreapi/private_functions.h +++ b/coreapi/private_functions.h @@ -214,6 +214,9 @@ void linphone_friend_list_subscription_state_changed(LinphoneCore *lc, LinphoneEvent *lev, LinphoneSubscriptionState state); void linphone_friend_list_invalidate_friends_maps(LinphoneFriendList *list); +LinphoneFriend *_linphone_friend_list_find_friend_by_phone_number(const LinphoneFriendList *list, + const LinphoneAccount *account, + const char *normalized_phone_number); /** * Removes all bodyless friend lists. @@ -265,6 +268,9 @@ LinphoneFriendListCbs *linphone_friend_list_cbs_new(void); void linphone_friend_list_set_current_callbacks(LinphoneFriendList *friend_list, LinphoneFriendListCbs *cbs); void linphone_friend_add_addresses_and_numbers_into_maps(LinphoneFriend *lf, LinphoneFriendList *list); void linphone_friend_notify_presence_received(LinphoneFriend *lf); +bool_t _linphone_friend_has_phone_number(const LinphoneFriend *lf, + const LinphoneAccount *account, + const char *normalized_phone_number); int linphone_parse_host_port(const char *input, char *host, size_t hostlen, int *port); int parse_hostname_to_addr(const char *server, struct sockaddr_storage *ss, socklen_t *socklen, int default_port); diff --git a/include/linphone/api/c-account.h b/include/linphone/api/c-account.h index fba62277e3..10ddcdc110 100644 --- a/include/linphone/api/c-account.h +++ b/include/linphone/api/c-account.h @@ -263,7 +263,7 @@ LINPHONE_PUBLIC int linphone_account_get_unread_chat_message_count(LinphoneAccou * @param username The string to parse. @notnil * @return TRUE if input is a phone number, FALSE otherwise. **/ -LINPHONE_PUBLIC bool_t linphone_account_is_phone_number(LinphoneAccount *account, const char *username); +LINPHONE_PUBLIC bool_t linphone_account_is_phone_number(const LinphoneAccount *account, const char *username); /** * Normalize a human readable phone number into a basic string. 888-444-222 becomes 888444222 @@ -275,7 +275,7 @@ LINPHONE_PUBLIC bool_t linphone_account_is_phone_number(LinphoneAccount *account * @return NULL if input is an invalid phone number, normalized phone number from username input otherwise. @maybenil * @tobefreed */ -LINPHONE_PUBLIC char *linphone_account_normalize_phone_number(LinphoneAccount *account, const char *username); +LINPHONE_PUBLIC char *linphone_account_normalize_phone_number(const LinphoneAccount *account, const char *username); /** * Normalize a human readable sip uri into a fully qualified LinphoneAddress. diff --git a/src/c-wrapper/api/c-account.cpp b/src/c-wrapper/api/c-account.cpp index 8f5d12e463..ea37052049 100644 --- a/src/c-wrapper/api/c-account.cpp +++ b/src/c-wrapper/api/c-account.cpp @@ -182,7 +182,7 @@ void _linphone_account_notify_registration_state_changed(LinphoneAccount *accoun linphone_account_cbs_get_registration_state_changed, state, message); } -bool_t linphone_account_is_phone_number(BCTBX_UNUSED(LinphoneAccount *account), const char *username) { +bool_t linphone_account_is_phone_number(BCTBX_UNUSED(const LinphoneAccount *account), const char *username) { if (!username) return FALSE; const char *p; @@ -223,16 +223,26 @@ static char *replace_icp_with_plus(char *phone, const char *icp) { return (strstr(phone, icp) == phone) ? ms_strdup_printf("+%s", phone + strlen(icp)) : ms_strdup(phone); } -char *linphone_account_normalize_phone_number(LinphoneAccount *account, const char *username) { - LinphoneAccountParams *tmpparams = account ? NULL : linphone_account_params_new(NULL); - LinphoneAccount *tmpaccount = account ? account : linphone_account_new(NULL, tmpparams); - if (tmpparams) linphone_account_params_unref(tmpparams); +char *linphone_account_normalize_phone_number(const LinphoneAccount *account, const char *username) { char *result = NULL; std::shared_ptr<DialPlan> dialplan; char *nationnal_significant_number = NULL; int ccc = -1; + const char *dial_prefix; + bool_t dial_escape_plus; - if (linphone_account_is_phone_number(tmpaccount, username)) { + if (account) { + const LinphoneAccountParams *accountParams = linphone_account_get_params(account); + dial_prefix = linphone_account_params_get_international_prefix(accountParams); + dial_escape_plus = linphone_account_params_get_dial_escape_plus_enabled(accountParams); + } else { + LinphoneAccountParams *accountParams = linphone_account_params_new(NULL); + dial_prefix = linphone_account_params_get_international_prefix(accountParams); + dial_escape_plus = linphone_account_params_get_dial_escape_plus_enabled(accountParams); + linphone_account_params_unref(accountParams); + } + + if (linphone_account_is_phone_number(account, username)) { char *flatten = linphone_account_flatten_phone_number(username); ms_debug("Flattened number is '%s' for '%s'", flatten, username); @@ -247,24 +257,35 @@ char *linphone_account_normalize_phone_number(LinphoneAccount *account, const ch ms_message("Unknown ccc for e164 like number [%s]", flatten); goto end; } else { - const char *dial_prefix = - linphone_account_params_get_international_prefix(linphone_account_get_params(tmpaccount)); if (dial_prefix) { dialplan = DialPlan::findByCcc(dial_prefix); // copy dial plan; } else { dialplan = DialPlan::MostCommon; } + if (dial_prefix) { - if (strcmp(dial_prefix, dialplan->getCountryCallingCode().c_str()) != 0) { + const char *country_calling_code = dialplan->getCountryCallingCode().c_str(); + if (strcmp(dial_prefix, country_calling_code) != 0) { // probably generic dialplan, preserving proxy dial prefix dialplan->setCountryCallingCode(dial_prefix); + country_calling_code = dial_prefix; + } + + // If phone number starts by international prefix but without +, add it + if (strstr(flatten, country_calling_code) == flatten && + strlen(flatten) > strlen(country_calling_code)) { + ms_warning("Phone number seems to start by international prefix but without '+', adding it"); + char *e164 = ms_strdup_printf("+%s", flatten); + result = linphone_account_normalize_phone_number(account, e164); + ms_free(e164); + goto end; } /*it does not make sens to try replace icp with + if we are not sure from the country we are (I.E * dial_prefix==NULL)*/ if (strstr(flatten, dialplan->getInternationalCallPrefix().c_str()) == flatten) { char *e164 = replace_icp_with_plus(flatten, dialplan->getInternationalCallPrefix().c_str()); - result = linphone_account_normalize_phone_number(tmpaccount, e164); + result = linphone_account_normalize_phone_number(account, e164); ms_free(e164); goto end; } @@ -286,8 +307,6 @@ char *linphone_account_normalize_phone_number(LinphoneAccount *account, const ch /*1. First prepend international calling prefix or +*/ /*2. Second add prefix*/ /*3. Finally add user digits */ - bool_t dial_escape_plus = - linphone_account_params_get_dial_escape_plus_enabled(linphone_account_get_params(tmpaccount)); result = ms_strdup_printf("%s%s%s", dial_escape_plus ? dialplan->getInternationalCallPrefix().c_str() : "+", dialplan->getCountryCallingCode().c_str(), nationnal_significant_number_start); ms_debug("Prepended prefix resulted in %s", result); @@ -300,10 +319,6 @@ char *linphone_account_normalize_phone_number(LinphoneAccount *account, const ch ms_free(flatten); } } - if (account == NULL) { - // linphone_account_params_unref(tmpparams); - linphone_account_unref(tmpaccount); - } return result; } diff --git a/tester/setup_tester.c b/tester/setup_tester.c index e074aeca97..00520fcc40 100644 --- a/tester/setup_tester.c +++ b/tester/setup_tester.c @@ -3155,9 +3155,10 @@ static void ldap_features_more_results(void) { // Check when magic search is limited to 30 items linphone_magic_search_get_contacts_list_async(magicSearch, "Big", "", LinphoneMagicSearchSourceLdapServers, - LinphoneMagicSearchAggregationNone); + LinphoneMagicSearchAggregationNone); // add 10ms to let some times to timeout callbacks. - BC_ASSERT_TRUE(wait_for_until(manager->lc, NULL, &stat->number_of_LinphoneMagicSearchResultReceived, 1, maxTimeout*1000 + 10)); + BC_ASSERT_TRUE(wait_for_until(manager->lc, NULL, &stat->number_of_LinphoneMagicSearchResultReceived, 1, + maxTimeout * 1000 + 10)); resultList = linphone_magic_search_get_last_search(magicSearch); BC_ASSERT_EQUAL((int)bctbx_list_size(resultList), 30, int, "%d"); bctbx_list_free_with_data(resultList, (bctbx_list_free_func)linphone_search_result_unref); @@ -3167,9 +3168,10 @@ static void ldap_features_more_results(void) { linphone_magic_search_set_search_limit(magicSearch, 100); linphone_magic_search_get_contacts_list_async(magicSearch, "Big", "", LinphoneMagicSearchSourceLdapServers, - LinphoneMagicSearchAggregationNone); + LinphoneMagicSearchAggregationNone); // add 10ms to let some times to timeout callbacks. - BC_ASSERT_TRUE(wait_for_until(manager->lc, NULL, &stat->number_of_LinphoneMagicSearchResultReceived, 1, maxTimeout*1000 + 10)); + BC_ASSERT_TRUE(wait_for_until(manager->lc, NULL, &stat->number_of_LinphoneMagicSearchResultReceived, 1, + maxTimeout * 1000 + 10)); resultList = linphone_magic_search_get_last_search(magicSearch); BC_ASSERT_EQUAL((int)bctbx_list_size(resultList), 100, int, "%d"); bctbx_list_free_with_data(resultList, (bctbx_list_free_func)linphone_search_result_unref); @@ -3290,6 +3292,33 @@ static void dial_plan(void) { bctbx_list_free_with_data(dial_plans, (bctbx_list_free_func)linphone_dial_plan_unref); } +static void friend_phone_number_lookup_without_plus(void) { + LinphoneCoreManager *manager = linphone_core_manager_new("marie_rc"); + LinphoneCore *core = manager->lc; + + LinphoneFriend *lf = linphone_core_create_friend(core); + linphone_friend_set_name(lf, "Test Number"); + linphone_friend_add_phone_number(lf, "+4912345678901"); + linphone_core_add_friend(core, lf); + linphone_friend_unref(lf); + + LinphoneFriend *found = linphone_core_find_friend_by_phone_number(core, "4912345678901"); + BC_ASSERT_PTR_NULL(found); + + const bctbx_list_t *accounts = linphone_core_get_account_list(core); + LinphoneAccount *account = (LinphoneAccount *)bctbx_list_get_data(accounts); + const LinphoneAccountParams *params = linphone_account_get_params(account); + LinphoneAccountParams *cloned_params = linphone_account_params_clone(params); + linphone_account_params_set_international_prefix(cloned_params, "49"); + linphone_account_set_params(account, cloned_params); + linphone_account_params_unref(cloned_params); + + found = linphone_core_find_friend_by_phone_number(core, "4912345678901"); + BC_ASSERT_PTR_NOT_NULL(found); + + linphone_core_manager_destroy(manager); +} + static void audio_devices(void) { LinphoneCoreManager *manager = linphone_core_manager_new("marie_rc"); LinphoneCore *core = manager->lc; @@ -3539,6 +3568,7 @@ test_t setup_tests[] = { TEST_NO_TAG("Ldap params edition with check", ldap_params_edition_with_check), TEST_NO_TAG("Delete friend in linphone rc", delete_friend_from_rc), TEST_NO_TAG("Dialplan", dial_plan), + TEST_NO_TAG("Friend phone number lookup without plus", friend_phone_number_lookup_without_plus), TEST_NO_TAG("Audio devices", audio_devices), TEST_NO_TAG("Migrate from call history database", migration_from_call_history_db), }; -- GitLab