accountmanager.c 9.5 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
 /*
 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 <http://www.gnu.org/licenses/>.
 */

19
#include <belle-sip/belle-sip.h>
20
#include "liblinphone_tester.h"
21
#include "tester_utils.h"
22 23 24 25 26

struct _Account{
	LinphoneAddress *identity;
	LinphoneAddress *modified_identity;
	char *password;
27
	int registered;
28
	int done;
29
	int created;
30
	char *phone_alias;
31
	char *uuid;
32 33 34 35
};

typedef struct _Account Account;

36
static Account *account_new(LinphoneAddress *identity, const char *unique_id){
37 38
	char *modified_username;
	Account *obj=ms_new0(Account,1);
39

40 41 42 43
	/* we need to inhibit leak detector because the two LinphoneAddress will remain behond the scope of the test being run */
	belle_sip_object_inhibit_leak_detector(TRUE);
	obj->identity=linphone_address_clone(identity);
	obj->password=sal_get_random_token(8);
44
	obj->phone_alias = NULL;
45 46 47 48 49 50 51 52 53
	obj->modified_identity=linphone_address_clone(identity);
	modified_username=ms_strdup_printf("%s_%s",linphone_address_get_username(identity), unique_id);
	linphone_address_set_username(obj->modified_identity, modified_username);
	ms_free(modified_username);
	belle_sip_object_inhibit_leak_detector(FALSE);
	return obj;
};

void account_destroy(Account *obj){
54
	if (obj->uuid) bctbx_free(obj->uuid);
55 56 57 58 59 60 61 62
	linphone_address_unref(obj->identity);
	linphone_address_unref(obj->modified_identity);
	ms_free(obj->password);
	ms_free(obj);
}

struct _AccountManager{
	char *unique_id;
63
	bctbx_list_t *accounts;
64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80
};

typedef struct _AccountManager AccountManager;

static AccountManager *the_am=NULL;

AccountManager *account_manager_get(void){
	if (the_am==NULL){
		the_am=ms_new0(AccountManager,1);
		the_am->unique_id=sal_get_random_token(6);
	}
	return the_am;
}

void account_manager_destroy(void){
	if (the_am){
		ms_free(the_am->unique_id);
81
		bctbx_list_free_with_data(the_am->accounts,(void(*)(void*))account_destroy);
82 83 84
		ms_free(the_am);
	}
	the_am=NULL;
85
	ms_message("Test account manager destroyed.");
86 87 88
}

Account *account_manager_get_account(AccountManager *m, const LinphoneAddress *identity){
89
	bctbx_list_t *it;
90

91 92 93 94 95 96 97 98 99
	for(it=m->accounts;it!=NULL;it=it->next){
		Account *a=(Account*)it->data;
		if (linphone_address_weak_equal(a->identity,identity)){
			return a;
		}
	}
	return NULL;
}

Ghislain MARY's avatar
Ghislain MARY committed
100 101 102 103 104 105 106 107 108 109 110 111 112 113
LinphoneAddress *account_manager_get_identity_with_modified_identity(const LinphoneAddress *modified_identity){
	AccountManager *m = account_manager_get();
	bctbx_list_t *it;

	for(it=m->accounts;it!=NULL;it=it->next){
		Account *a=(Account*)it->data;
		if (linphone_address_weak_equal(a->modified_identity,modified_identity)){
			return a->identity;
		}
	}
	return NULL;
}


114 115 116
static void account_created_on_server_cb(LinphoneCore *lc, LinphoneProxyConfig *cfg, LinphoneRegistrationState state, const char *info){
	Account *account=(Account*)linphone_core_get_user_data(lc);
	switch(state){
117
		case LinphoneRegistrationOk: {
118
			char * phrase = sal_op_get_error_info(linphone_proxy_config_get_sal_op(cfg))->full_string;
119 120 121 122 123 124 125
			if (phrase && strcasecmp("Test account created", phrase) == 0) {
				account->created=1;
			} else {
				account->registered=1;
			}
			break;
		}
126 127 128 129 130 131 132 133
		case LinphoneRegistrationCleared:
			account->done=1;
		break;
		default:
		break;
	}
}

134
void account_create_on_server(Account *account, const LinphoneProxyConfig *refcfg, const char* phone_alias){
135 136 137 138 139 140
	LinphoneCore *lc;
	LinphoneAddress *tmp_identity=linphone_address_clone(account->modified_identity);
	LinphoneProxyConfig *cfg;
	LinphoneAuthInfo *ai;
	char *tmp;
	LinphoneAddress *server_addr;
141
	LinphoneSipTransports tr;
142
	LinphoneCoreCbs *cbs = linphone_factory_create_core_cbs(linphone_factory_get());
143

144 145 146
	linphone_core_cbs_set_registration_state_changed(cbs, account_created_on_server_cb);
	lc = configure_lc_from(cbs, bc_tester_get_resource_dir_prefix(), NULL, account);
	linphone_core_cbs_unref(cbs);
147 148 149 150
	tr.udp_port=LC_SIP_TRANSPORT_RANDOM;
	tr.tcp_port=LC_SIP_TRANSPORT_RANDOM;
	tr.tls_port=LC_SIP_TRANSPORT_RANDOM;
	linphone_core_set_sip_transports(lc,&tr);
151

152
	cfg=linphone_core_create_proxy_config(lc);
153
	linphone_address_set_secure(tmp_identity, FALSE);
154 155
	linphone_address_set_password(tmp_identity,account->password);
	linphone_address_set_header(tmp_identity,"X-Create-Account","yes");
156
	if (phone_alias) linphone_address_set_header(tmp_identity, "X-Phone-Alias", phone_alias);
157 158 159 160
	tmp=linphone_address_as_string(tmp_identity);
	linphone_proxy_config_set_identity(cfg,tmp);
	ms_free(tmp);
	linphone_address_unref(tmp_identity);
161

162
	server_addr=linphone_address_new(linphone_proxy_config_get_server_addr(refcfg));
163 164
	linphone_address_set_secure(server_addr, FALSE);
	linphone_address_set_transport(server_addr,LinphoneTransportTcp); /*use tcp for account creation, we may not have certificates configured at this stage*/
165
	linphone_address_set_port(server_addr,0);
166 167 168 169
	tmp=linphone_address_as_string(server_addr);
	linphone_proxy_config_set_server_addr(cfg,tmp);
	ms_free(tmp);
	linphone_address_unref(server_addr);
170
	linphone_proxy_config_set_expires(cfg,3*3600); //accounts are valid 3 hours
171

172
	linphone_core_add_proxy_config(lc,cfg);
173 174 175
	/*wait 25 seconds, since the DNS SRV resolution may take a while - and
	especially if router does NOT support DNS SRV and we have to wait its timeout*/
	if (wait_for_until(lc,NULL,&account->created,1,25000)==FALSE){
176 177 178
		ms_fatal("Account for %s could not be created on server.", linphone_proxy_config_get_identity(refcfg));
	}
	linphone_proxy_config_edit(cfg);
179 180 181
	tmp_identity=linphone_address_clone(account->modified_identity);
	linphone_address_set_secure(tmp_identity, FALSE);
	tmp=linphone_address_as_string(tmp_identity);
182
	linphone_proxy_config_set_identity(cfg,tmp); /*remove the X-Create-Account header*/
183
	linphone_address_unref(tmp_identity);
184 185
	ms_free(tmp);
	linphone_proxy_config_done(cfg);
186

187 188 189 190
	ai=linphone_auth_info_new(linphone_address_get_username(account->modified_identity),
				NULL,
				account->password,NULL,NULL,linphone_address_get_domain(account->modified_identity));
	linphone_core_add_auth_info(lc,ai);
191
	linphone_auth_info_unref(ai);
192

193
	if (wait_for_until(lc,NULL,&account->registered,1,3000)==FALSE){
194 195 196 197 198 199 200
		ms_fatal("Account for %s is not working on server.", linphone_proxy_config_get_identity(refcfg));
	}
	linphone_core_remove_proxy_config(lc,cfg);
	linphone_proxy_config_unref(cfg);
	if (wait_for_until(lc,NULL,&account->done,1,3000)==FALSE){
		ms_error("Account creation could not clean the registration context.");
	}
201
	linphone_core_unref(lc);
202 203
}

204
static LinphoneAddress *account_manager_check_account(AccountManager *m, LinphoneProxyConfig *cfg, LinphoneCoreManager *cm){
205 206 207 208 209 210
	LinphoneCore *lc=linphone_proxy_config_get_core(cfg);
	const char *identity=linphone_proxy_config_get_identity(cfg);
	LinphoneAddress *id_addr=linphone_address_new(identity);
	Account *account=account_manager_get_account(m,id_addr);
	LinphoneAuthInfo *ai;
	bool_t create_account=FALSE;
211 212 213 214
	const LinphoneAuthInfo *original_ai = linphone_core_find_auth_info(lc
																		,NULL
																		, linphone_address_get_username(id_addr)
																		, linphone_address_get_domain(id_addr));
215
	const char *phone_alias = cm->phone_alias;
216

217 218 219 220 221
	if (!account||(phone_alias&&(!account->phone_alias||strcmp(phone_alias,account->phone_alias)!=0))){
		if (account) {
			m->accounts=bctbx_list_remove(m->accounts,account);
			account_destroy(account);
		}
222
		account=account_new(id_addr,m->unique_id);
223
		account->phone_alias=ms_strdup(phone_alias);
224 225
		ms_message("No account for %s exists, going to create one.",identity);
		create_account=TRUE;
226
		m->accounts=bctbx_list_append(m->accounts,account);
227
	}
228 229
	/*modify the username of the identity of the proxy config*/
	linphone_address_set_username(id_addr, linphone_address_get_username(account->modified_identity));
230
	linphone_proxy_config_set_identity_address(cfg, id_addr);
231

232
	if (create_account){
233
		account_create_on_server(account,cfg,phone_alias);
234
	}
235

236 237 238 239
	if (liblinphone_tester_keep_uuid) {
		/* create and/or set uuid */
		if (account->uuid == NULL) {
			char tmp[64];
240
			sal_create_uuid(linphone_core_get_sal(cm->lc), tmp, sizeof(tmp));
241 242
			account->uuid = bctbx_strdup(tmp);
		}
243
		sal_set_uuid(linphone_core_get_sal(cm->lc), account->uuid);
244 245
	}

246 247 248 249
	/*remove previous auth info to avoid mismatching*/
	if (original_ai)
		linphone_core_remove_auth_info(lc,original_ai);

250 251 252 253
	ai=linphone_auth_info_new(linphone_address_get_username(account->modified_identity),
				NULL,
				account->password,NULL,NULL,linphone_address_get_domain(account->modified_identity));
	linphone_core_add_auth_info(lc,ai);
254
	linphone_auth_info_unref(ai);
255

256 257 258 259
	linphone_address_unref(id_addr);
	return account->modified_identity;
}

260
void linphone_core_manager_check_accounts(LinphoneCoreManager *m){
261
	const bctbx_list_t *it;
262
	AccountManager *am=account_manager_get();
Ronan's avatar
Ronan committed
263 264
	unsigned int logmask = linphone_core_get_log_level_mask();

265
	if (!liblinphonetester_show_account_manager_logs) linphone_core_set_log_level_mask(ORTP_ERROR|ORTP_FATAL);
266
	for(it=linphone_core_get_proxy_config_list(m->lc);it!=NULL;it=it->next){
267
		LinphoneProxyConfig *cfg=(LinphoneProxyConfig *)it->data;
268
		account_manager_check_account(am,cfg,m);
269
	}
270
	if (!liblinphonetester_show_account_manager_logs) linphone_core_set_log_level_mask(logmask);
271
}