core: add SIPE version to debug log
[siplcs.git] / src / core / sipe-core.c
blob789f3f00b77086ae2f12bb3ce818e8d45ade1294
1 /**
2 * @file sipe-core.c
4 * pidgin-sipe
6 * Copyright (C) 2010-2013 SIPE Project <http://sipe.sourceforge.net/>
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23 * Some notes on the history of this project/code/copyrights:
25 * - the project is called SIPE, but originally the code was only written
26 * for the libpurple framework, i.e. Pidgin. Hence the package name is
27 * "pidgin-sipe".
29 * - in the beginning almost all of the code was located in a module
30 * called "sipe.c". During the effort to remove the libpurple
31 * dependencies from the SIPE core, thousands of lines of code got
32 * shifted out of sipe.c, mostly to newly created modules and sipe.c
33 * ceased to exist.
35 * - it would have been tedious to track down the original author or
36 * copyright and preserve them for each line of code that was moved.
37 * Therefore the new modules started with a fresh copyright notice
38 * (like the one above).
40 * - the original copyright notices from sipe.c have been moved to this
41 * file (see below) and *MUST* be preserved!
43 * - if necessary the author of a line of code in question can still be
44 * reconstructed from the git repository information.
45 * See also "man git-blame"
47 * - if you think your copyright should be restored for a piece of code,
48 * then please contact the SIPE project to fix the source files ASAP.
50 *------------------- Copyright notices from "sipe.c" ---------------
51 * Copyright (C) 2010-11 SIPE Project <http://sipe.sourceforge.net/>
52 * Copyright (C) 2009-10 pier11 <pier11@operamail.com>
53 * Copyright (C) 2008 Novell, Inc.
54 * Copyright (C) 2007-09 Anibal Avelar <debianmx@gmail.com>
55 * Copyright (C) 2005 Thomas Butter <butter@uni-mannheim.de>
57 * ***
58 * Thanks to Google's Summer of Code Program and the helpful mentors
59 * ***
60 *------------------- Copyright notices from "sipe.c" ---------------
63 #ifdef HAVE_CONFIG_H
64 #include "config.h"
65 #endif
67 #include <stdlib.h>
68 #include <string.h>
70 #include <glib.h>
72 #include "sipe-common.h"
73 #include "sip-csta.h"
74 #include "sip-sec.h"
75 #include "sip-transport.h"
76 #include "sipe-backend.h"
77 #include "sipe-buddy.h"
78 #include "sipe-cal.h"
79 #include "sipe-certificate.h"
80 #include "sipe-chat.h"
81 #include "sipe-conf.h"
82 #include "sipe-core.h"
83 #include "sipe-core-private.h"
84 #include "sipe-crypt.h"
85 #include "sipe-group.h"
86 #include "sipe-groupchat.h"
87 #include "sipe-media.h"
88 #include "sipe-mime.h"
89 #include "sipe-nls.h"
90 #include "sipe-ocs2007.h"
91 #include "sipe-schedule.h"
92 #include "sipe-session.h"
93 #include "sipe-status.h"
94 #include "sipe-subscriptions.h"
95 #include "sipe-svc.h"
96 #include "sipe-utils.h"
97 #include "sipe-webticket.h"
99 /* locale_dir is unused if ENABLE_NLS is not defined */
100 void sipe_core_init(SIPE_UNUSED_PARAMETER const char *locale_dir)
102 srand(time(NULL));
103 sip_sec_init();
105 #ifdef ENABLE_NLS
106 SIPE_DEBUG_INFO("bindtextdomain = %s",
107 bindtextdomain(PACKAGE_NAME, locale_dir));
108 SIPE_DEBUG_INFO("bind_textdomain_codeset = %s",
109 bind_textdomain_codeset(PACKAGE_NAME, "UTF-8"));
110 textdomain(PACKAGE_NAME);
111 #endif
112 /* Initialization for crypto backend (production mode) */
113 sipe_crypto_init(TRUE);
114 sipe_mime_init();
115 sipe_status_init();
118 void sipe_core_destroy(void)
120 sipe_chat_destroy();
121 sipe_status_shutdown();
122 sipe_mime_shutdown();
123 sipe_crypto_shutdown();
124 sip_sec_destroy();
127 gchar *sipe_core_about(void)
129 return g_strdup_printf(
131 * Non-translatable parts, like markup, are hard-coded
132 * into the format string. This requires more translatable
133 * texts but it makes the translations less error prone.
135 "<b><font size=\"+1\">SIPE " PACKAGE_VERSION " </font></b><br/>"
136 "<br/>"
137 /* 1 */ "%s:<br/>"
138 "<li> - Microsoft Lync Server 2010</li><br/>"
139 "<li> - Microsoft Office Communications Server 2007 R2</li><br/>"
140 "<li> - Microsoft Office Communications Server 2007</li><br/>"
141 "<li> - Microsoft Live Communications Server 2005</li><br/>"
142 "<li> - Microsoft Live Communications Server 2003</li><br/>"
143 "<li> - Reuters Messaging</li><br/>"
144 "<br/>"
145 /* 2 */ "%s: <a href=\"" PACKAGE_URL "\">" PACKAGE_URL "</a><br/>"
146 /* 3,4 */ "%s: <a href=\"http://sourceforge.net/projects/sipe/forums/forum/688534\">%s</a><br/>"
147 /* 5,6 */ "%s: <a href=\"" PACKAGE_BUGREPORT "\">%s</a><br/>"
148 /* 7 */ "%s: <a href=\"" SIPE_TRANSLATIONS_URL "\">Transifex.net</a><br/>"
149 /* 8 */ "%s: GPLv2+<br/>"
150 "<br/>"
151 /* 9 */ "%s:<br/>"
152 " - CERN<br/>"
153 " - Reuters Messaging network<br/>"
154 " - Deutsche Bank<br/>"
155 " - Merrill Lynch<br/>"
156 " - Wachovia<br/>"
157 " - Intel<br/>"
158 " - Nokia<br/>"
159 " - HP<br/>"
160 " - Symantec<br/>"
161 " - Accenture<br/>"
162 " - Capgemini<br/>"
163 " - Siemens<br/>"
164 " - Alcatel-Lucent<br/>"
165 " - BT<br/>"
166 "<br/>"
167 /* 10,11 */ "%s<a href=\"" SIPE_TRANSLATIONS_URL "\">Transifex.net</a>%s.<br/>"
168 "<br/>"
169 /* 12 */ "<b>%s:</b><br/>"
170 " - Anibal Avelar<br/>"
171 " - Gabriel Burt<br/>"
172 " - Stefan Becker<br/>"
173 " - pier11<br/>"
174 " - Jakub Adam<br/>"
175 " - Tomáš Hrabčík<br/>"
176 "<br/>"
177 /* 13 */ "%s<br/>"
179 /* The next 13 texts make up the SIPE about note text */
180 /* About note, part 1/13: introduction */
181 _("A third-party plugin implementing extended version of SIP/SIMPLE used by various products"),
182 /* About note, part 2/13: home page URL (label) */
183 _("Home Page"),
184 /* About note, part 3/13: support forum URL (label) */
185 _("Support"),
186 /* About note, part 4/13: support forum name (hyperlink text) */
187 _("Help Forum"),
188 /* About note, part 5/13: bug tracker URL (label) */
189 _("Report Problems"),
190 /* About note, part 6/13: bug tracker URL (hyperlink text) */
191 _("Bug Tracker"),
192 /* About note, part 7/13: translation service URL (label) */
193 _("Translations"),
194 /* About note, part 8/13: license type (label) */
195 _("License"),
196 /* About note, part 9/13: known users */
197 _("We support users in such organizations as"),
198 /* About note, part 10/13: translation request, text before Transifex.net URL */
199 /* append a space if text is not empty */
200 _("Please help us to translate SIPE to your native language here at "),
201 /* About note, part 11/13: translation request, text after Transifex.net URL */
202 /* start with a space if text is not empty */
203 _(" using convenient web interface"),
204 /* About note, part 12/13: author list (header) */
205 _("Authors"),
206 /* About note, part 13/13: Localization credit */
207 /* PLEASE NOTE: do *NOT* simply translate the english original */
208 /* but write something similar to the following sentence: */
209 /* "Localization for <language name> (<language code>): <name>" */
210 _("Original texts in English (en): SIPE developers")
214 static guint sipe_ht_hash_nick(const char *nick)
216 char *lc = g_utf8_strdown(nick, -1);
217 guint bucket = g_str_hash(lc);
218 g_free(lc);
220 return bucket;
223 static gboolean sipe_ht_equals_nick(const char *nick1, const char *nick2)
225 char *nick1_norm = NULL;
226 char *nick2_norm = NULL;
227 gboolean equal;
229 if (nick1 == NULL && nick2 == NULL) return TRUE;
230 if (nick1 == NULL || nick2 == NULL ||
231 !g_utf8_validate(nick1, -1, NULL) ||
232 !g_utf8_validate(nick2, -1, NULL)) return FALSE;
234 nick1_norm = g_utf8_casefold(nick1, -1);
235 nick2_norm = g_utf8_casefold(nick2, -1);
236 equal = g_utf8_collate(nick1_norm, nick2_norm) == 0;
237 g_free(nick2_norm);
238 g_free(nick1_norm);
240 return equal;
243 struct sipe_core_public *sipe_core_allocate(const gchar *signin_name,
244 gboolean sso,
245 const gchar *login_domain,
246 const gchar *login_account,
247 const gchar *password,
248 const gchar *email,
249 const gchar *email_url,
250 const gchar **errmsg)
252 struct sipe_core_private *sipe_private;
253 gchar **user_domain;
255 SIPE_DEBUG_INFO("sipe_core_allocate: SIPE version " PACKAGE_VERSION " signin_name '%s'", signin_name);
257 /* ensure that sign-in name doesn't contain invalid characters */
258 if (strpbrk(signin_name, "\t\v\r\n") != NULL) {
259 *errmsg = _("SIP Exchange user name contains invalid characters");
260 return NULL;
263 /* ensure that sign-in name format is name@domain */
264 if (!strchr(signin_name, '@') ||
265 g_str_has_prefix(signin_name, "@") ||
266 g_str_has_suffix(signin_name, "@")) {
267 *errmsg = _("User name should be a valid SIP URI\nExample: user@company.com");
268 return NULL;
271 /* ensure that Login & Password are valid when SSO is not selected */
272 if (!sso && (is_empty(login_account) || is_empty(password))) {
273 *errmsg = _("Login and password are required when Single Sign-On is not enabled");
274 return NULL;
277 /* ensure that email format is name@domain (if provided) */
278 if (!is_empty(email) &&
279 (!strchr(email, '@') ||
280 g_str_has_prefix(email, "@") ||
281 g_str_has_suffix(email, "@")))
283 *errmsg = _("Email address should be valid if provided\nExample: user@company.com");
284 return NULL;
287 /* ensure that user name doesn't contain spaces */
288 user_domain = g_strsplit(signin_name, "@", 2);
289 SIPE_DEBUG_INFO("sipe_core_allocate: user '%s' domain '%s'", user_domain[0], user_domain[1]);
290 if (strchr(user_domain[0], ' ') != NULL) {
291 g_strfreev(user_domain);
292 *errmsg = _("SIP Exchange user name contains whitespace");
293 return NULL;
296 /* ensure that email_url is in proper format if enabled (if provided).
297 * Example (Exchange): https://server.company.com/EWS/Exchange.asmx
298 * Example (Domino) : https://[domino_server]/[mail_database_name].nsf
300 if (!is_empty(email_url)) {
301 char *tmp = g_ascii_strdown(email_url, -1);
302 if (!g_str_has_prefix(tmp, "https://"))
304 g_free(tmp);
305 g_strfreev(user_domain);
306 *errmsg = _("Email services URL should be valid if provided\n"
307 "Example: https://exchange.corp.com/EWS/Exchange.asmx\n"
308 "Example: https://domino.corp.com/maildatabase.nsf");
309 return NULL;
311 g_free(tmp);
314 sipe_private = g_new0(struct sipe_core_private, 1);
315 SIPE_CORE_PRIVATE_FLAG_UNSET(SUBSCRIBED_BUDDIES);
316 SIPE_CORE_PRIVATE_FLAG_UNSET(INITIAL_PUBLISH);
317 SIPE_CORE_PRIVATE_FLAG_UNSET(SSO);
318 if (sso)
319 SIPE_CORE_PRIVATE_FLAG_SET(SSO);
320 sipe_private->username = g_strdup(signin_name);
321 sipe_private->email = is_empty(email) ? g_strdup(signin_name) : g_strdup(email);
322 sipe_private->authdomain = sso ? NULL : g_strdup(login_domain);
323 sipe_private->authuser = sso ? NULL : g_strdup(login_account);
324 sipe_private->password = sso ? NULL : g_strdup(password);
325 sipe_private->public.sip_name = g_strdup(user_domain[0]);
326 sipe_private->public.sip_domain = g_strdup(user_domain[1]);
327 g_strfreev(user_domain);
329 sipe_private->buddies = g_hash_table_new((GHashFunc)sipe_ht_hash_nick, (GEqualFunc)sipe_ht_equals_nick);
330 sipe_private->our_publications = g_hash_table_new_full(g_str_hash, g_str_equal,
331 g_free, (GDestroyNotify)g_hash_table_destroy);
332 sipe_subscriptions_init(sipe_private);
333 sipe_status_set_activity(sipe_private, SIPE_ACTIVITY_UNSET);
335 return((struct sipe_core_public *)sipe_private);
338 void sipe_core_connection_cleanup(struct sipe_core_private *sipe_private)
340 g_free(sipe_private->epid);
341 sipe_private->epid = NULL;
343 sip_transport_disconnect(sipe_private);
345 sipe_schedule_cancel_all(sipe_private);
347 if (sipe_private->allowed_events) {
348 GSList *entry = sipe_private->allowed_events;
349 while (entry) {
350 g_free(entry->data);
351 entry = entry->next;
354 g_slist_free(sipe_private->allowed_events);
356 sipe_ocs2007_free(sipe_private);
358 sipe_core_buddy_menu_free(SIPE_CORE_PUBLIC);
360 if (sipe_private->contact)
361 g_free(sipe_private->contact);
362 sipe_private->contact = NULL;
363 if (sipe_private->register_callid)
364 g_free(sipe_private->register_callid);
365 sipe_private->register_callid = NULL;
367 if (sipe_private->focus_factory_uri)
368 g_free(sipe_private->focus_factory_uri);
369 sipe_private->focus_factory_uri = NULL;
371 if (sipe_private->calendar) {
372 sipe_cal_calendar_free(sipe_private->calendar);
374 sipe_private->calendar = NULL;
376 sipe_groupchat_free(sipe_private);
379 void sipe_core_deallocate(struct sipe_core_public *sipe_public)
381 struct sipe_core_private *sipe_private = SIPE_CORE_PRIVATE;
383 #ifdef HAVE_VV
384 if (sipe_private->media_call) {
385 sipe_media_handle_going_offline(sipe_private->media_call);
387 #endif
389 /* leave all conversations */
390 if (sipe_private->sessions) {
391 GSList *entry;
392 while ((entry = sipe_private->sessions) != NULL) {
393 sipe_session_close(sipe_private, entry->data);
397 sipe_conf_cancel_unaccepted(sipe_private, NULL);
399 if (sipe_private->csta) {
400 sip_csta_close(sipe_private);
403 /* pending service requests must be cancelled first */
404 sipe_svc_free(sipe_private);
405 sipe_webticket_free(sipe_private);
407 if (sipe_backend_connection_is_valid(SIPE_CORE_PUBLIC)) {
408 sipe_subscriptions_unsubscribe(sipe_private);
409 sip_transport_deregister(sipe_private);
412 sipe_core_connection_cleanup(sipe_private);
413 sipe_certificate_free(sipe_private);
415 g_free(sipe_private->public.sip_name);
416 g_free(sipe_private->public.sip_domain);
417 g_free(sipe_private->username);
418 g_free(sipe_private->email);
419 g_free(sipe_private->password);
420 g_free(sipe_private->authdomain);
421 g_free(sipe_private->authuser);
422 g_free(sipe_private->status);
423 g_free(sipe_private->note);
424 g_free(sipe_private->ocs2005_user_states);
426 sipe_buddy_free_all(sipe_private);
427 g_hash_table_destroy(sipe_private->buddies);
428 g_hash_table_destroy(sipe_private->our_publications);
429 g_hash_table_destroy(sipe_private->user_state_publications);
430 sipe_subscriptions_destroy(sipe_private);
432 if (sipe_private->groups) {
433 GSList *entry;
434 while ((entry = sipe_private->groups) != NULL)
435 sipe_group_free(sipe_private, entry->data);
438 if (sipe_private->our_publication_keys) {
439 GSList *entry = sipe_private->our_publication_keys;
440 while (entry) {
441 g_free(entry->data);
442 entry = entry->next;
445 g_slist_free(sipe_private->our_publication_keys);
447 #ifdef HAVE_VV
448 g_free(sipe_private->test_call_bot_uri);
449 g_free(sipe_private->mras_uri);
450 g_free(sipe_private->media_relay_username);
451 g_free(sipe_private->media_relay_password);
452 sipe_media_relay_list_free(sipe_private->media_relays);
453 #endif
455 g_free(sipe_private->addressbook_uri);
456 g_free(sipe_private->dlx_uri);
457 g_free(sipe_private);
461 Local Variables:
462 mode: c
463 c-file-style: "bsd"
464 indent-tabs-mode: t
465 tab-width: 8
466 End: