remote_provisioning.c 5.13 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
/*
remote_provisioning.c
Copyright (C) 2010  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, write to the Free Software
17
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
18
*/
19
#include "private.h"
20
#include "xml2lpc.h"
21
#include "linphone/lpconfig.h"
22 23 24 25 26

#define XML2LPC_CALLBACK_BUFFER_SIZE  1024


static void linphone_remote_provisioning_apply(LinphoneCore *lc, const char *xml) {
27 28 29 30 31 32 33 34 35
	LinphoneConfig *config = linphone_core_get_config(lc);
	const char *error_msg = _linphone_config_load_from_xml_string(config, xml);

	_linphone_config_apply_factory_config(config);
	linphone_configuring_terminated(
		lc,
		error_msg ? LinphoneConfiguringFailed : LinphoneConfiguringSuccessful,
		error_msg
	);
36 37
}

38
int linphone_remote_provisioning_load_file( LinphoneCore* lc, const char* file_path){
39
	int status = -1;
40
	char* provisioning=ms_load_path_content(file_path, NULL);
41 42 43 44

	if (provisioning){
		linphone_remote_provisioning_apply(lc, provisioning);
		status = 0;
45
		ms_free(provisioning);
46 47 48 49
	}
	return status;
}

50 51
static void belle_request_process_response_event(void *ctx, const belle_http_response_event_t *event) {
	LinphoneCore *lc = (LinphoneCore *)ctx;
52 53
	belle_sip_message_t *message = BELLE_SIP_MESSAGE(event->response);
	const char *body = belle_sip_message_get_body(message);
54

55
	if (belle_http_response_get_status_code(event->response) == 200) {
56
		linphone_remote_provisioning_apply(lc, body);
57
	} else {
58
		linphone_configuring_terminated(lc, LinphoneConfiguringFailed, "http error");
59 60 61 62 63
	}
}

static void belle_request_process_io_error(void *ctx, const belle_sip_io_error_event_t *event) {
	LinphoneCore *lc = (LinphoneCore *)ctx;
64
	linphone_configuring_terminated(lc, LinphoneConfiguringFailed, "http io error");
65 66 67 68
}

static void belle_request_process_timeout(void *ctx, const belle_sip_timeout_event_t *event) {
	LinphoneCore *lc = (LinphoneCore *)ctx;
69
	linphone_configuring_terminated(lc, LinphoneConfiguringFailed, "http timeout");
70 71 72 73
}

static void belle_request_process_auth_requested(void *ctx, belle_sip_auth_event_t *event) {
	LinphoneCore *lc = (LinphoneCore *)ctx;
74
	linphone_configuring_terminated(lc, LinphoneConfiguringFailed, "http auth requested");
75 76
}

77
int linphone_remote_provisioning_download_and_apply(LinphoneCore *lc, const char *remote_provisioning_uri) {
78

79 80
	belle_generic_uri_t *uri=belle_generic_uri_parse(remote_provisioning_uri);
	const char* scheme = uri ? belle_generic_uri_get_scheme(uri) : NULL;
81
	const char *host = uri ? belle_generic_uri_get_host(uri) : NULL;
82 83 84 85

	if( scheme && (strcmp(scheme,"file") == 0) ){
		// We allow for 'local remote-provisioning' in case the file is to be opened from the hard drive.
		const char* file_path = remote_provisioning_uri + strlen("file://"); // skip scheme
86 87 88
		if (uri) {
			belle_sip_object_unref(uri);
		}
89
		return linphone_remote_provisioning_load_file(lc, file_path);
90
	} else if( scheme && strncmp(scheme, "http", 4) == 0 && host && strlen(host) > 0) {
91
		belle_http_request_listener_callbacks_t belle_request_listener={0};
92
		belle_http_request_t *request;
93

94 95 96 97
		belle_request_listener.process_response=belle_request_process_response_event;
		belle_request_listener.process_auth_requested=belle_request_process_auth_requested;
		belle_request_listener.process_io_error=belle_request_process_io_error;
		belle_request_listener.process_timeout=belle_request_process_timeout;
98

Simon Morlat's avatar
Simon Morlat committed
99
		lc->provisioning_http_listener = belle_http_request_listener_create_from_callbacks(&belle_request_listener, lc);
100

jehan's avatar
jehan committed
101 102 103 104
		request=belle_http_request_create("GET"
											, uri
											, belle_sip_header_create("User-Agent",linphone_core_get_user_agent(lc))
											, NULL);
105

Simon Morlat's avatar
Simon Morlat committed
106
		return belle_http_provider_send_request(lc->http_provider, request, lc->provisioning_http_listener);
107
	} else {
108
		ms_error("Invalid provisioning URI [%s] (missing scheme or host ?)",remote_provisioning_uri);
109 110 111
		if (uri) {
			belle_sip_object_unref(uri);
		}
112
		return -1;
113
	}
114 115
}

116
LinphoneStatus linphone_core_set_provisioning_uri(LinphoneCore *lc, const char *remote_provisioning_uri) {
117 118 119
	belle_generic_uri_t *uri=remote_provisioning_uri?belle_generic_uri_parse(remote_provisioning_uri):NULL;
	if (!remote_provisioning_uri||uri) {
		lp_config_set_string(lc->config,"misc","config-uri",remote_provisioning_uri);
120 121 122
		if (uri) {
			belle_sip_object_unref(uri);
		}
123 124 125 126 127
		return 0;
	}
	ms_error("Invalid provisioning URI [%s] (could not be parsed)",remote_provisioning_uri);
	return -1;

128 129 130 131 132
}

const char*linphone_core_get_provisioning_uri(const LinphoneCore *lc){
	return lp_config_get_string(lc->config,"misc","config-uri",NULL);
}
133 134 135 136

bool_t linphone_core_is_provisioning_transient(LinphoneCore *lc) {
	return lp_config_get_int(lc->config, "misc", "transient_provisioning", 0) == 1;
}