src/protocols/novell/novell.c

branch
gaim
changeset 20470
77693555855f
parent 13071
b98e72d4089a
parent 20469
b2836a24d81e
child 20471
1966704b3e42
equal deleted inserted replaced
13071:b98e72d4089a 20470:77693555855f
1 /*
2 * novell.c
3 *
4 * Copyright (c) 2004 Novell, Inc. All Rights Reserved.
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; version 2 of the License.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18 *
19 */
20
21 #include "internal.h"
22 #include "accountopt.h"
23 #include "debug.h"
24 #include "prpl.h"
25 #include "server.h"
26 #include "nmuser.h"
27 #include "notify.h"
28 #include "util.h"
29 #include "sslconn.h"
30 #include "request.h"
31 #include "network.h"
32 #include "privacy.h"
33 #include "status.h"
34 #include "version.h"
35
36 #define DEFAULT_PORT 8300
37 #define NOVELL_CONNECT_STEPS 4
38 #define NM_ROOT_FOLDER_NAME "GroupWise Messenger"
39
40 #define NOVELL_STATUS_TYPE_AVAILABLE "available"
41 #define NOVELL_STATUS_TYPE_AWAY "away"
42 #define NOVELL_STATUS_TYPE_BUSY "busy"
43 #define NOVELL_STATUS_TYPE_OFFLINE "offline"
44 #define NOVELL_STATUS_TYPE_IDLE "idle"
45 #define NOVELL_STATUS_TYPE_APPEAR_OFFLINE "appearoffline"
46
47 static GaimPlugin *my_protocol = NULL;
48
49 static gboolean
50 _is_disconnect_error(NMERR_T err);
51
52 static gboolean
53 _check_for_disconnect(NMUser * user, NMERR_T err);
54
55 static void
56 _send_message(NMUser * user, NMMessage * message);
57
58 static void
59 _update_buddy_status(NMUser *user, GaimBuddy * buddy, int status, int gmt);
60
61 static void
62 _remove_gaim_buddies(NMUser * user);
63
64 static void
65 _add_contacts_to_gaim_blist(NMUser * user, NMFolder * folder);
66
67 static void
68 _add_gaim_buddies(NMUser * user);
69
70 static void
71 _sync_contact_list(NMUser *user);
72
73 static void
74 _sync_privacy_lists(NMUser *user);
75
76 static void
77 _show_info(GaimConnection * gc, NMUserRecord * user_record);
78
79 const char *
80 _get_conference_name(int id);
81
82 /*******************************************************************************
83 * Response callbacks
84 *******************************************************************************/
85
86 /* Handle login response */
87 static void
88 _login_resp_cb(NMUser * user, NMERR_T ret_code,
89 gpointer resp_data, gpointer user_data)
90 {
91 GaimConnection *gc;
92 const char *alias;
93 NMERR_T rc;
94
95 if (user == NULL)
96 return;
97
98 gc = gaim_account_get_connection(user->client_data);
99 if (gc == NULL)
100 return;
101
102 if (ret_code == NM_OK) {
103
104 /* Set alias for user if not set (use Full Name) */
105 alias = gaim_account_get_alias(user->client_data);
106 if (alias == NULL || *alias == '\0') {
107 alias = nm_user_record_get_full_name(user->user_record);
108
109 if (alias)
110 gaim_account_set_alias(user->client_data, alias);
111 }
112
113 /* Tell Gaim that we are connected */
114 gaim_connection_set_state(gc, GAIM_CONNECTED);
115
116 _sync_contact_list(user);
117
118 rc = nm_send_set_status(user, NM_STATUS_AVAILABLE, NULL, NULL, NULL,
119 NULL);
120 _check_for_disconnect(user, rc);
121
122 } else {
123
124 char *err = g_strdup_printf(_("Login failed (%s)."),
125 nm_error_to_string (ret_code));
126
127 /* Clear the password if it was invalid ... don't want to retry
128 * and get ourselves locked out.
129 */
130 if (ret_code == NMERR_AUTHENTICATION_FAILED ||
131 ret_code == NMERR_CREDENTIALS_MISSING ||
132 ret_code == NMERR_PASSWORD_INVALID) {
133 gaim_account_set_password((GaimAccount*)user->client_data, NULL);
134 }
135 gaim_connection_error(gc, err);
136 g_free(err);
137 }
138 }
139
140 /* Handle getstatus response*/
141 static void
142 _get_status_resp_cb(NMUser * user, NMERR_T ret_code,
143 gpointer resp_data, gpointer user_data)
144 {
145 GaimBuddy *buddy;
146 GSList *buddies;
147 GSList *bnode;
148 NMUserRecord *user_record = (NMUserRecord *) resp_data;
149 int status;
150
151 if (user == NULL || user_record == NULL)
152 return;
153
154 if (ret_code == NM_OK) {
155
156 /* Find all Gaim buddies and update their statuses */
157 const char *name = nm_user_record_get_display_id(user_record);
158
159 if (name) {
160 buddies = gaim_find_buddies((GaimAccount *) user->client_data, name);
161 for (bnode = buddies; bnode; bnode = bnode->next) {
162 buddy = (GaimBuddy *) bnode->data;
163 if (buddy) {
164 status = nm_user_record_get_status(user_record);
165 _update_buddy_status(user, buddy, status, time(0));
166 }
167 }
168 g_slist_free(buddies);
169 }
170
171 } else {
172
173 gaim_debug(GAIM_DEBUG_INFO, "novell",
174 "_get_status_resp_cb(): rc = 0x%X\n", ret_code);
175
176 }
177 }
178
179 /* Show an error if the rename failed */
180 static void
181 _rename_contact_resp_cb(NMUser * user, NMERR_T ret_code,
182 gpointer resp_data, gpointer user_data)
183 {
184 if (ret_code != NM_OK) {
185 gaim_debug(GAIM_DEBUG_INFO, "novell",
186 "_rename_contact_resp_cb(): rc = 0x%X\n", ret_code);
187 }
188 }
189
190 /* Handle the getdetails response and send the message */
191 static void
192 _get_details_resp_send_msg(NMUser * user, NMERR_T ret_code,
193 gpointer resp_data, gpointer user_data)
194 {
195 GaimConversation *gconv;
196 GaimConnection *gc;
197 NMUserRecord *user_record = NULL;
198 NMContact *cntct = NULL;
199 NMConference *conf;
200 NMMessage *msg = user_data;
201 const char *dn = NULL;
202 const char *name;
203
204 if (user == NULL || msg == NULL)
205 return;
206
207 if (ret_code == NM_OK) {
208 user_record = (NMUserRecord *) resp_data;
209 if (user_record) {
210
211 /* Set the title for the conversation */
212 /* XXX - Should this be GAIM_CONV_TYPE_IM? */
213 gconv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_ANY,
214 nm_user_record_get_display_id(user_record),
215 (GaimAccount *) user->client_data);
216 if (gconv) {
217
218 dn = nm_user_record_get_dn(user_record);
219 if (dn) {
220 cntct = nm_find_contact(user, dn);
221 }
222
223 if (cntct) {
224 gaim_conversation_set_title(gconv,
225 nm_contact_get_display_name(cntct));
226 } else {
227
228 /* Not in the contact list, try to user full name */
229 name = (char *) nm_user_record_get_full_name(user_record);
230 if (name)
231 gaim_conversation_set_title(gconv, name);
232 }
233 }
234
235 /* Add the user record to particpant list */
236 conf = nm_message_get_conference(msg);
237 if (conf) {
238 nm_conference_add_participant(conf, user_record);
239 _send_message(user, msg);
240 }
241 }
242
243 } else {
244
245 gc = gaim_account_get_connection(user->client_data);
246 if (gc != NULL) {
247 char *err = g_strdup_printf(_("Unable to send message."
248 " Could not get details for user (%s)."),
249 nm_error_to_string (ret_code));
250
251 gaim_notify_error(gc, NULL, err, NULL);
252 g_free(err);
253 }
254
255 if (msg)
256 nm_release_message(msg);
257 }
258 }
259
260 /* Set up the new GaimBuddy based on the response from getdetails */
261 static void
262 _get_details_resp_setup_buddy(NMUser * user, NMERR_T ret_code,
263 gpointer resp_data, gpointer user_data)
264 {
265 NMUserRecord *user_record;
266 NMContact *contact;
267 GaimBuddy *buddy;
268 const char *alias;
269 NMERR_T rc = NM_OK;
270
271 if (user == NULL || resp_data == NULL || user_data == NULL)
272 return;
273
274 contact = user_data;
275
276 if (ret_code == NM_OK) {
277 user_record = resp_data;
278
279 buddy = nm_contact_get_data(contact);
280
281 nm_contact_set_user_record(contact, user_record);
282
283 /* Set the display id */
284 gaim_blist_rename_buddy(buddy,
285 nm_user_record_get_display_id(user_record));
286
287 alias = gaim_buddy_get_alias(buddy);
288 if (alias == NULL || *alias == '\0' || (strcmp(alias, buddy->name) == 0)) {
289 gaim_blist_alias_buddy(buddy,
290 nm_user_record_get_full_name(user_record));
291
292 /* Tell the server about the new display name */
293 rc = nm_send_rename_contact(user, contact,
294 nm_user_record_get_full_name(user_record),
295 NULL, NULL);
296 _check_for_disconnect(user, rc);
297
298 }
299
300
301 /* Get initial status for the buddy */
302 rc = nm_send_get_status(user, resp_data, _get_status_resp_cb, NULL);
303 _check_for_disconnect(user, rc);
304
305 /* nm_release_contact(contact);*/
306
307 }
308
309 if (contact)
310 nm_release_contact(contact);
311 }
312
313 /* Add the new contact into the GaimBuddy list */
314 static void
315 _create_contact_resp_cb(NMUser * user, NMERR_T ret_code,
316 gpointer resp_data, gpointer user_data)
317 {
318 NMContact *tmp_contact = (NMContact *) user_data;
319 NMContact *new_contact = NULL;
320 NMFolder *folder = NULL;
321 GaimGroup *group;
322 GaimBuddy *buddy;
323 const char *folder_name = NULL;
324 NMERR_T rc = NM_OK;
325
326 if (user == NULL)
327 return;
328
329 if (ret_code == NM_OK) {
330
331 new_contact = (NMContact *) resp_data;
332 if (new_contact == NULL || tmp_contact == NULL)
333 return;
334
335 /* Get the userid and folder name for the new contact */
336 folder = nm_find_folder_by_id(user,
337 nm_contact_get_parent_id(new_contact));
338 if (folder) {
339 folder_name = nm_folder_get_name(folder);
340 }
341
342 if (*folder_name == '\0')
343 folder_name = NM_ROOT_FOLDER_NAME;
344
345 /* Re-add the buddy now that we got the okay from the server */
346 if (folder_name && (group = gaim_find_group(folder_name))) {
347
348 const char *alias = nm_contact_get_display_name(tmp_contact);
349 const char *display_id = nm_contact_get_display_id(new_contact);
350
351 if (display_id == NULL)
352 display_id = nm_contact_get_dn(new_contact);
353
354 if (alias && strcmp(alias, display_id)) {
355
356 /* The user requested an alias, tell the server about it. */
357 rc = nm_send_rename_contact(user, new_contact, alias,
358 _rename_contact_resp_cb, NULL);
359 _check_for_disconnect(user, rc);
360
361 } else {
362
363 alias = "";
364
365 }
366
367 /* Add it to the gaim buddy list if it is not there */
368 buddy = gaim_find_buddy_in_group(user->client_data, display_id, group);
369 if (buddy == NULL) {
370 buddy = gaim_buddy_new(user->client_data, display_id, alias);
371 gaim_blist_add_buddy(buddy, NULL, group, NULL);
372 }
373
374 /* Save the new buddy as part of the contact object */
375 nm_contact_set_data(new_contact, (gpointer) buddy);
376
377 /* We need details for the user before we can setup the
378 * new Gaim buddy. We always call this because the
379 * 'createcontact' response fields do not always contain
380 * everything that we need.
381 */
382 nm_contact_add_ref(new_contact);
383
384 rc = nm_send_get_details(user, nm_contact_get_dn(new_contact),
385 _get_details_resp_setup_buddy, new_contact);
386 _check_for_disconnect(user, rc);
387
388 }
389
390 } else {
391 GaimConnection *gc = gaim_account_get_connection(user->client_data);
392 const char *name = nm_contact_get_dn(tmp_contact);
393 char *err;
394
395 err =
396 g_strdup_printf(_("Unable to add %s to your buddy list (%s)."),
397 name, nm_error_to_string (ret_code));
398 gaim_notify_error(gc, NULL, err, NULL);
399 g_free(err);
400
401 }
402
403 if (tmp_contact)
404 nm_release_contact(tmp_contact);
405 }
406
407 /* Show an error if we failed to send the message */
408 static void
409 _send_message_resp_cb(NMUser * user, NMERR_T ret_code,
410 gpointer resp_data, gpointer user_data)
411 {
412 GaimConnection *gc;
413 char *err = NULL;
414
415 if (user == NULL)
416 return;
417
418 if (ret_code != NM_OK) {
419 gc = gaim_account_get_connection(user->client_data);
420
421 /* TODO: Improve this! message to who or for what conference? */
422 err = g_strdup_printf(_("Unable to send message (%s)."),
423 nm_error_to_string (ret_code));
424 gaim_notify_error(gc, NULL, err, NULL);
425 g_free(err);
426 }
427 }
428
429 /* Show an error if the remove failed */
430 static void
431 _remove_contact_resp_cb(NMUser * user, NMERR_T ret_code,
432 gpointer resp_data, gpointer user_data)
433 {
434 if (ret_code != NM_OK) {
435 /* TODO: Display an error? */
436
437 gaim_debug(GAIM_DEBUG_INFO, "novell",
438 "_remove_contact_resp_cb(): rc = 0x%x\n", ret_code);
439 }
440 }
441
442 /* Show an error if the remove failed */
443 static void
444 _remove_folder_resp_cb(NMUser * user, NMERR_T ret_code,
445 gpointer resp_data, gpointer user_data)
446 {
447 if (ret_code != NM_OK) {
448 /* TODO: Display an error? */
449
450 gaim_debug(GAIM_DEBUG_INFO, "novell",
451 "_remove_folder_resp_cb(): rc = 0x%x\n", ret_code);
452 }
453 }
454
455 /* Show an error if the move failed */
456 static void
457 _move_contact_resp_cb(NMUser * user, NMERR_T ret_code,
458 gpointer resp_data, gpointer user_data)
459 {
460 if (ret_code != NM_OK) {
461 /* TODO: Display an error? */
462
463 gaim_debug(GAIM_DEBUG_INFO, "novell",
464 "_move_contact_resp_cb(): rc = 0x%x\n", ret_code);
465 }
466 }
467
468 /* Show an error if the rename failed */
469 static void
470 _rename_folder_resp_cb(NMUser * user, NMERR_T ret_code,
471 gpointer resp_data, gpointer user_data)
472 {
473 if (ret_code != NM_OK) {
474 /* TODO: Display an error? */
475
476 gaim_debug(GAIM_DEBUG_INFO, "novell",
477 "_rename_folder_resp_cb(): rc = 0x%x\n", ret_code);
478 }
479 }
480
481 static void
482 _sendinvite_resp_cb(NMUser *user, NMERR_T ret_code,
483 gpointer resp_data, gpointer user_data)
484 {
485 char *err;
486 GaimConnection *gc;
487
488 if (user == NULL)
489 return;
490
491 if (ret_code != NM_OK) {
492 gc = gaim_account_get_connection(user->client_data);
493 err = g_strdup_printf(_("Unable to invite user (%s)."), nm_error_to_string(ret_code));
494 gaim_notify_error(gc, NULL, err, NULL);
495 g_free(err);
496
497 gaim_debug(GAIM_DEBUG_INFO, "novell",
498 "_sendinvite_resp_cb(): rc = 0x%x\n", ret_code);
499 }
500
501 }
502
503 /* If the createconf was successful attempt to send the message,
504 * otherwise display an error message to the user.
505 */
506 static void
507 _createconf_resp_send_msg(NMUser * user, NMERR_T ret_code,
508 gpointer resp_data, gpointer user_data)
509 {
510 NMConference *conf;
511 NMMessage *msg = user_data;
512
513 if (user == NULL || msg == NULL)
514 return;
515
516 if (ret_code == NM_OK) {
517 _send_message(user, msg);
518 } else {
519
520 if ((conf = nm_message_get_conference(msg))) {
521
522 GaimConnection *gc = gaim_account_get_connection(user->client_data);
523 const char *name = NULL;
524 char *err;
525 NMUserRecord *ur;
526
527 ur = nm_conference_get_participant(conf, 0);
528 if (ur)
529 name = nm_user_record_get_userid(ur);
530
531 if (name)
532 err = g_strdup_printf(_("Unable to send message to %s."
533 " Could not create the conference (%s)."),
534 name,
535 nm_error_to_string (ret_code));
536 else
537 err = g_strdup_printf(_("Unable to send message."
538 " Could not create the conference (%s)."),
539 nm_error_to_string (ret_code));
540
541 gaim_notify_error(gc, NULL, err, NULL);
542 g_free(err);
543 }
544
545 if (msg)
546 nm_release_message(msg);
547 }
548 }
549
550 /* Move contact to newly created folder */
551 static void
552 _create_folder_resp_move_contact(NMUser * user, NMERR_T ret_code,
553 gpointer resp_data, gpointer user_data)
554 {
555 NMContact *contact = user_data;
556 NMFolder *new_folder;
557 char *folder_name = resp_data;
558 NMERR_T rc = NM_OK;
559
560 if (user == NULL || folder_name == NULL || contact == NULL) {
561
562 if (folder_name)
563 g_free(folder_name);
564
565 return;
566 }
567
568 if (ret_code == NM_OK || ret_code == NMERR_DUPLICATE_FOLDER) {
569 new_folder = nm_find_folder(user, folder_name);
570 if (new_folder) {
571
572 /* Tell the server to move the contact to the new folder */
573 /* rc = nm_send_move_contact(user, contact, new_folder,
574 _move_contact_resp_cb, NULL); */
575
576 rc = nm_send_create_contact(user, new_folder, contact,
577 NULL, NULL);
578
579 _check_for_disconnect(user, rc);
580
581 }
582 } else {
583 GaimConnection *gc = gaim_account_get_connection(user->client_data);
584 char *err = g_strdup_printf(_("Unable to move user %s"
585 " to folder %s in the server side list."
586 " Error while creating folder (%s)."),
587 nm_contact_get_dn(contact),
588 folder_name,
589 nm_error_to_string (ret_code));
590
591 gaim_notify_error(gc, NULL, err, NULL);
592 g_free(err);
593 }
594
595 if (folder_name)
596 g_free(folder_name);
597 }
598
599 /* Add contact to newly create folder */
600 static void
601 _create_folder_resp_add_contact(NMUser * user, NMERR_T ret_code,
602 gpointer resp_data, gpointer user_data)
603 {
604 NMContact *contact = (NMContact *) user_data;
605 NMFolder *folder;
606 char *folder_name = (char *) resp_data;
607 NMERR_T rc = NM_OK;
608
609 if (user == NULL || folder_name == NULL || contact == NULL) {
610
611 if (contact)
612 nm_release_contact(contact);
613
614 if (folder_name)
615 g_free(folder_name);
616
617 return;
618 }
619
620 if (ret_code == NM_OK || ret_code == NMERR_DUPLICATE_FOLDER) {
621 folder = nm_find_folder(user, folder_name);
622 if (folder) {
623
624 rc = nm_send_create_contact(user, folder, contact,
625 _create_contact_resp_cb, contact);
626 _check_for_disconnect(user, rc);
627 }
628 } else {
629 GaimConnection *gc = gaim_account_get_connection(user->client_data);
630 const char *name = nm_contact_get_dn(contact);
631 char *err =
632 g_strdup_printf(_("Unable to add %s to your buddy list."
633 " Error creating folder in server side list (%s)."),
634 name, nm_error_to_string (ret_code));
635
636 gaim_notify_error(gc, NULL, err, NULL);
637
638 nm_release_contact(contact);
639 g_free(err);
640 }
641
642 g_free(folder_name);
643 }
644
645 static void
646 _join_conf_resp_cb(NMUser * user, NMERR_T ret_code,
647 gpointer resp_data, gpointer user_data)
648 {
649 GaimConversation *chat;
650 GaimConnection *gc;
651 NMUserRecord *ur;
652 NMConference *conference = user_data;
653 const char *name, *conf_name;
654 int i, count;
655
656 if (user == NULL || conference == NULL)
657 return;
658
659 gc = gaim_account_get_connection(user->client_data);
660
661 if (ret_code == NM_OK) {
662 conf_name = _get_conference_name(++user->conference_count);
663 chat = serv_got_joined_chat(gc, user->conference_count, conf_name);
664 if (chat) {
665
666 nm_conference_set_data(conference, (gpointer) chat);
667
668 count = nm_conference_get_participant_count(conference);
669 for (i = 0; i < count; i++) {
670 ur = nm_conference_get_participant(conference, i);
671 if (ur) {
672 name = nm_user_record_get_display_id(ur);
673 gaim_conv_chat_add_user(GAIM_CONV_CHAT(chat), name, NULL,
674 GAIM_CBFLAGS_NONE, TRUE);
675 }
676 }
677 }
678 }
679 }
680
681 /* Show info returned by getdetails */
682 static void
683 _get_details_resp_show_info(NMUser * user, NMERR_T ret_code,
684 gpointer resp_data, gpointer user_data)
685 {
686 GaimConnection *gc;
687 NMUserRecord *user_record;
688 char *name;
689 char *err;
690
691 if (user == NULL)
692 return;
693
694 name = user_data;
695
696 if (ret_code == NM_OK) {
697 user_record = (NMUserRecord *) resp_data;
698 if (user_record) {
699 _show_info(gaim_account_get_connection(user->client_data),
700 user_record);
701 }
702 } else {
703 gc = gaim_account_get_connection(user->client_data);
704 err =
705 g_strdup_printf(_("Could not get details for user %s (%s)."),
706 name, nm_error_to_string (ret_code));
707 gaim_notify_error(gc, NULL, err, NULL);
708 g_free(err);
709 }
710
711 if (name)
712 g_free(name);
713 }
714
715 /* Handle get details response add to privacy list */
716 static void
717 _get_details_resp_add_privacy_item(NMUser *user, NMERR_T ret_code,
718 gpointer resp_data, gpointer user_data)
719 {
720 GaimConnection *gc;
721 NMUserRecord *user_record = resp_data;
722 char *err;
723 gboolean allowed = GPOINTER_TO_INT(user_data);
724 const char *display_id;
725
726 if (user == NULL)
727 return;
728
729 gc = gaim_account_get_connection(user->client_data);
730 display_id = nm_user_record_get_display_id(user_record);
731
732 if (ret_code == NM_OK) {
733
734 if (allowed) {
735
736 if (!g_slist_find_custom(gc->account->permit,
737 display_id, (GCompareFunc)nm_utf8_strcasecmp)) {
738 gaim_privacy_permit_add(gc->account, display_id, TRUE);
739 }
740
741 } else {
742
743 if (!g_slist_find_custom(gc->account->permit,
744 display_id, (GCompareFunc)nm_utf8_strcasecmp)) {
745 gaim_privacy_deny_add(gc->account, display_id, TRUE);
746 }
747 }
748
749 } else {
750
751 err = g_strdup_printf(_("Unable to add user to privacy list (%s)."),
752 nm_error_to_string(ret_code));
753 gaim_notify_error(gc, NULL, err, NULL);
754 g_free(err);
755
756 }
757 }
758
759 /* Handle response to create privacy item request */
760 static void
761 _create_privacy_item_deny_resp_cb(NMUser *user, NMERR_T ret_code,
762 gpointer resp_data, gpointer user_data)
763 {
764 GaimConnection *gc;
765 NMUserRecord *user_record;
766 char *who = user_data;
767 char *err;
768 NMERR_T rc = NM_OK;
769 const char *display_id = NULL;
770
771 if (user == NULL)
772 return;
773
774 gc = gaim_account_get_connection(user->client_data);
775
776 if (ret_code == NM_OK) {
777
778 user_record = nm_find_user_record(user, who);
779 if (user_record)
780 display_id = nm_user_record_get_display_id(user_record);
781
782 if (display_id) {
783
784 if (!g_slist_find_custom(gc->account->deny,
785 display_id, (GCompareFunc)nm_utf8_strcasecmp)) {
786
787 gaim_privacy_deny_add(gc->account, display_id, TRUE);
788 }
789
790 } else {
791 rc = nm_send_get_details(user, who,
792 _get_details_resp_add_privacy_item,
793 (gpointer)FALSE);
794 _check_for_disconnect(user, rc);
795 }
796 } else {
797
798 err = g_strdup_printf(_("Unable to add %s to deny list (%s)."),
799 who, nm_error_to_string(ret_code));
800 gaim_notify_error(gc, NULL, err, NULL);
801 g_free(err);
802
803 }
804
805 if (who)
806 g_free(who);
807
808 }
809
810 /* Handle response to create privacy item request */
811 static void
812 _create_privacy_item_permit_resp_cb(NMUser *user, NMERR_T ret_code,
813 gpointer resp_data, gpointer user_data)
814 {
815 GaimConnection *gc;
816 NMUserRecord *user_record;
817 char *who = user_data;
818 char *err;
819 NMERR_T rc = NM_OK;
820 const char *display_id = NULL;
821
822 if (user == NULL)
823 return;
824
825 gc = gaim_account_get_connection(user->client_data);
826
827 if (ret_code == NM_OK) {
828
829 user_record = nm_find_user_record(user, who);
830 if (user_record)
831 display_id = nm_user_record_get_display_id(user_record);
832
833 if (display_id) {
834
835 if (!g_slist_find_custom(gc->account->permit,
836 display_id,
837 (GCompareFunc)nm_utf8_strcasecmp)) {
838
839 gaim_privacy_permit_add(gc->account, display_id, TRUE);
840 }
841
842 } else {
843 rc = nm_send_get_details(user, who,
844 _get_details_resp_add_privacy_item,
845 (gpointer)TRUE);
846 _check_for_disconnect(user, rc);
847 }
848
849 } else {
850
851 err = g_strdup_printf(_("Unable to add %s to permit list (%s)."), who,
852 nm_error_to_string(ret_code));
853 gaim_notify_error(gc, NULL, err, NULL);
854 g_free(err);
855
856 }
857
858 if (who)
859 g_free(who);
860 }
861
862 static void
863 _get_details_send_privacy_create(NMUser *user, NMERR_T ret_code,
864 gpointer resp_data, gpointer user_data)
865 {
866 NMERR_T rc = NM_OK;
867 GaimConnection *gc;
868 NMUserRecord *user_record = resp_data;
869 char *err;
870 gboolean allowed = GPOINTER_TO_INT(user_data);
871 const char *dn, *display_id;
872
873 if (user == NULL)
874 return;
875
876 gc = gaim_account_get_connection(user->client_data);
877 dn = nm_user_record_get_dn(user_record);
878 display_id = nm_user_record_get_display_id(user_record);
879
880 if (ret_code == NM_OK) {
881
882 if (allowed) {
883 rc = nm_send_create_privacy_item(user, dn, TRUE,
884 _create_privacy_item_permit_resp_cb,
885 g_strdup(display_id));
886 _check_for_disconnect(user, rc);
887
888 } else {
889 rc = nm_send_create_privacy_item(user, dn, FALSE,
890 _create_privacy_item_deny_resp_cb,
891 g_strdup(display_id));
892 _check_for_disconnect(user, rc);
893 }
894
895 } else {
896
897 err = g_strdup_printf(_("Unable to add user to privacy list (%s)."),
898 nm_error_to_string(ret_code));
899 gaim_notify_error(gc, NULL, err, NULL);
900 g_free(err);
901
902 }
903 }
904
905 static void
906 _remove_privacy_item_resp_cb(NMUser *user, NMERR_T ret_code,
907 gpointer resp_data, gpointer user_data)
908 {
909 GaimConnection *gc;
910 char *who = user_data;
911 char *err;
912
913 if (user == NULL)
914 return;
915
916 if (ret_code != NM_OK) {
917
918 gc = gaim_account_get_connection(user->client_data);
919 err = g_strdup_printf(_("Unable to remove %s from privacy list (%s)."), who,
920 nm_error_to_string(ret_code));
921 gaim_notify_error(gc, NULL, err, NULL);
922 g_free(err);
923 }
924
925 if (who)
926 g_free(who);
927 }
928
929 static void
930 _set_privacy_default_resp_cb(NMUser *user, NMERR_T ret_code,
931 gpointer resp_data, gpointer user_data)
932 {
933 GaimConnection *gc;
934 char *err;
935
936 if (user == NULL)
937 return;
938
939 if (ret_code != NM_OK) {
940
941 gc = gaim_account_get_connection(user->client_data);
942 err = g_strdup_printf(_("Unable to change server side privacy settings (%s)."),
943 nm_error_to_string(ret_code));
944 gaim_notify_error(gc, NULL, err, NULL);
945 g_free(err);
946
947 }
948 }
949
950 /* Handle get details response add to privacy list */
951 static void
952 _get_details_resp_send_invite(NMUser *user, NMERR_T ret_code,
953 gpointer resp_data, gpointer user_data)
954 {
955 NMERR_T rc = NM_OK;
956 GaimConnection *gc;
957 NMUserRecord *user_record = resp_data;
958 char *err;
959 const char *display_id;
960 GSList *cnode;
961 NMConference *conference;
962 gpointer chat;
963 long id = (long) user_data;
964
965 if (user == NULL)
966 return;
967
968 gc = gaim_account_get_connection(user->client_data);
969 display_id = nm_user_record_get_display_id(user_record);
970
971 if (ret_code == NM_OK) {
972
973 for (cnode = user->conferences; cnode != NULL; cnode = cnode->next) {
974 conference = cnode->data;
975 if (conference && (chat = nm_conference_get_data(conference))) {
976 if (gaim_conv_chat_get_id(GAIM_CONV_CHAT(chat)) == id) {
977 rc = nm_send_conference_invite(user, conference, user_record,
978 NULL, _sendinvite_resp_cb, NULL);
979 _check_for_disconnect(user, rc);
980 break;
981 }
982 }
983 }
984
985 } else {
986
987 err = g_strdup_printf(_("Unable to invite user (%s)."), nm_error_to_string(ret_code));
988 gaim_notify_error(gc, NULL, err, NULL);
989 g_free(err);
990
991 }
992 }
993
994 static void
995 _createconf_resp_send_invite(NMUser * user, NMERR_T ret_code,
996 gpointer resp_data, gpointer user_data)
997 {
998 NMERR_T rc = NM_OK;
999 NMConference *conference = resp_data;
1000 NMUserRecord *user_record = user_data;
1001 GaimConnection *gc;
1002 char *err;
1003
1004 if (user == NULL)
1005 return;
1006
1007
1008
1009 if (ret_code == NM_OK) {
1010 rc = nm_send_conference_invite(user, conference, user_record,
1011 NULL, _sendinvite_resp_cb, NULL);
1012 _check_for_disconnect(user, rc);
1013 } else {
1014 err = g_strdup_printf(_("Unable to create conference (%s)."), nm_error_to_string(ret_code));
1015 gc = gaim_account_get_connection(user->client_data);
1016 gaim_notify_error(gc, NULL, err, NULL);
1017 g_free(err);
1018 }
1019 }
1020
1021 /*******************************************************************************
1022 * Helper functions
1023 ******************************************************************************/
1024
1025 static char *
1026 _user_agent_string()
1027 {
1028
1029 #if !defined(_WIN32)
1030
1031 const char *sysname = "";
1032 const char *release = "";
1033 struct utsname u;
1034
1035 if (uname(&u) == 0) {
1036 sysname = u.sysname;
1037 release = u.release;
1038 } else {
1039 sysname = "Linux";
1040 release = "Unknown";
1041 }
1042
1043 return g_strdup_printf("Gaim/%s (%s; %s)", VERSION, sysname, release);
1044
1045 #else
1046
1047 const char *sysname = "";
1048 OSVERSIONINFO os_info;
1049 SYSTEM_INFO sys_info;
1050
1051 os_info.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
1052 GetVersionEx(&os_info);
1053 GetSystemInfo(&sys_info);
1054
1055 if (os_info.dwPlatformId == VER_PLATFORM_WIN32_NT) {
1056 switch (os_info.dwMajorVersion) {
1057 case 3:
1058 case 4:
1059 sysname = "Windows NT";
1060 break;
1061 case 5:
1062 switch (os_info.dwMinorVersion) {
1063 case 0:
1064 sysname = "Windows 2000";
1065 break;
1066 case 1:
1067 sysname = "Windows XP";
1068 break;
1069 case 2:
1070 sysname = "Windows Server 2003";
1071 break;
1072 default:
1073 sysname = "Windows";
1074 break;
1075 }
1076 break;
1077 default:
1078 sysname = "Windows";
1079 break;
1080 }
1081
1082 } else if (os_info.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS) {
1083 switch (os_info.dwMinorVersion) {
1084 case 0:
1085 sysname = "Windows 95";
1086 break;
1087 case 10:
1088 sysname = "Windows 98";
1089 break;
1090 case 90:
1091 sysname = "Windows ME";
1092 break;
1093 default:
1094 sysname = "Windows";
1095 break;
1096 }
1097 } else {
1098 sysname = "Windows";
1099 }
1100
1101 return g_strdup_printf("Gaim/%s (%s; %ld.%ld)", VERSION, sysname,
1102 os_info.dwMajorVersion, os_info.dwMinorVersion);
1103
1104 #endif
1105
1106
1107 }
1108
1109 static gboolean
1110 _is_disconnect_error(NMERR_T err)
1111 {
1112 return (err == NMERR_TCP_WRITE ||
1113 err == NMERR_TCP_READ || err == NMERR_PROTOCOL);
1114 }
1115
1116 static gboolean
1117 _check_for_disconnect(NMUser * user, NMERR_T err)
1118 {
1119 GaimConnection *gc = gaim_account_get_connection(user->client_data);
1120
1121 if (_is_disconnect_error(err)) {
1122
1123 gaim_connection_error(gc, _("Error communicating with server."
1124 " Closing connection."));
1125 return TRUE;
1126
1127 }
1128
1129 return FALSE;
1130 }
1131
1132 /* Check to see if the conference is instantiated, if so send the message.
1133 * If not send the create conference -- the response handler for the createconf
1134 * will call this function again.
1135 */
1136 static void
1137 _send_message(NMUser * user, NMMessage * message)
1138 {
1139 NMConference *conf;
1140 NMERR_T rc = NM_OK;
1141
1142 conf = nm_message_get_conference(message);
1143 if (conf) {
1144 /* We have a conference make sure that the
1145 server knows about it already. */
1146 if (nm_conference_is_instantiated(conf)) {
1147
1148 /* We have everything that we need...finally! */
1149 rc = nm_send_message(user, message, _send_message_resp_cb);
1150 _check_for_disconnect(user, rc);
1151
1152 nm_release_message(message);
1153
1154 } else {
1155 rc = nm_send_create_conference(user, conf, _createconf_resp_send_msg, message);
1156 _check_for_disconnect(user, rc);
1157 }
1158 }
1159 }
1160
1161 /*
1162 * Update the status of the given buddy in the Gaim buddy list
1163 */
1164 static void
1165 _update_buddy_status(NMUser *user, GaimBuddy * buddy, int novellstatus, int gmt)
1166 {
1167 GaimAccount *account;
1168 const char *status_id;
1169 const char *text = NULL;
1170 const char *dn;
1171 int idle = 0;
1172 gboolean loggedin = TRUE;
1173
1174 account = buddy->account;
1175
1176 switch (novellstatus) {
1177 case NM_STATUS_AVAILABLE:
1178 status_id = NOVELL_STATUS_TYPE_AVAILABLE;
1179 break;
1180 case NM_STATUS_AWAY:
1181 status_id = NOVELL_STATUS_TYPE_AWAY;
1182 break;
1183 case NM_STATUS_BUSY:
1184 status_id = NOVELL_STATUS_TYPE_BUSY;
1185 break;
1186 case NM_STATUS_OFFLINE:
1187 status_id = NOVELL_STATUS_TYPE_OFFLINE;
1188 loggedin = FALSE;
1189 break;
1190 case NM_STATUS_AWAY_IDLE:
1191 status_id = NOVELL_STATUS_TYPE_AWAY;
1192 idle = gmt;
1193 break;
1194 default:
1195 status_id = NOVELL_STATUS_TYPE_OFFLINE;
1196 loggedin = FALSE;
1197 break;
1198 }
1199
1200 /* Get status text for the user */
1201 dn = nm_lookup_dn(user, buddy->name);
1202 if (dn) {
1203 NMUserRecord *user_record = nm_find_user_record(user, dn);
1204 if (user_record) {
1205 text = nm_user_record_get_status_text(user_record);
1206 }
1207 }
1208
1209 gaim_prpl_got_user_status(account, buddy->name, status_id,
1210 "message", text, NULL);
1211 gaim_prpl_got_user_idle(account, buddy->name,
1212 (novellstatus == NM_STATUS_AWAY_IDLE), idle);
1213 }
1214
1215 /* Iterate through the cached Gaim buddy list and remove buddies
1216 * that are not in the server side list.
1217 */
1218 static void
1219 _remove_gaim_buddies(NMUser *user)
1220 {
1221 GaimBlistNode *gnode;
1222 GaimBlistNode *cnode;
1223 GaimBlistNode *bnode;
1224 GaimGroup *group;
1225 GaimBuddy *buddy;
1226 GaimBuddyList *blist;
1227 GSList *rem_list = NULL;
1228 GSList *l;
1229 NMFolder *folder = NULL;
1230 const char *gname = NULL;
1231
1232 if ((blist = gaim_get_blist())) {
1233 for (gnode = blist->root; gnode; gnode = gnode->next) {
1234 if (!GAIM_BLIST_NODE_IS_GROUP(gnode))
1235 continue;
1236 group = (GaimGroup *) gnode;
1237 for (cnode = gnode->child; cnode; cnode = cnode->next) {
1238 if (!GAIM_BLIST_NODE_IS_CONTACT(cnode))
1239 continue;
1240 for (bnode = cnode->child; bnode; bnode = bnode->next) {
1241 if (!GAIM_BLIST_NODE_IS_BUDDY(bnode))
1242 continue;
1243 buddy = (GaimBuddy *) bnode;
1244 if (buddy->account == user->client_data) {
1245 gname = group->name;
1246 if (strcmp(group->name, NM_ROOT_FOLDER_NAME) == 0)
1247 gname = "";
1248 folder = nm_find_folder(user, gname);
1249 if (folder == NULL ||
1250 !nm_folder_find_contact_by_display_id(folder, buddy->name)) {
1251 rem_list = g_slist_append(rem_list, buddy);
1252 }
1253 }
1254 }
1255 }
1256 }
1257
1258 if (rem_list) {
1259 for (l = rem_list; l; l = l->next) {
1260 gaim_blist_remove_buddy(l->data);
1261 }
1262 g_slist_free(rem_list);
1263 }
1264 }
1265 }
1266
1267 /* Add all of the contacts in the given folder to the Gaim buddy list */
1268 static void
1269 _add_contacts_to_gaim_blist(NMUser * user, NMFolder * folder)
1270 {
1271 NMUserRecord *user_record = NULL;
1272 NMContact *contact = NULL;
1273 GaimBuddy *buddy = NULL;
1274 GaimGroup *group;
1275 NMERR_T cnt = 0, i;
1276 const char *text = NULL;
1277 const char *name = NULL;
1278 const char *fname = NULL;
1279 int status = 0;
1280
1281 /* If this is the root folder give it a name. Gaim does not have the concept of
1282 * a root folder.
1283 */
1284 fname = nm_folder_get_name(folder);
1285 if (fname == NULL || *fname == '\0') {
1286 fname = NM_ROOT_FOLDER_NAME;
1287 }
1288
1289 /* Does the Gaim group exist already? */
1290 group = gaim_find_group(fname);
1291 if (group == NULL) {
1292 group = gaim_group_new(fname);
1293 gaim_blist_add_group(group, NULL);
1294 }
1295
1296 /* Get each contact for this folder */
1297 cnt = nm_folder_get_contact_count(folder);
1298 for (i = 0; i < cnt; i++) {
1299 contact = nm_folder_get_contact(folder, i);
1300 if (contact) {
1301
1302 name = nm_contact_get_display_id(contact);
1303 if (name) {
1304
1305 buddy = gaim_find_buddy_in_group(user->client_data, name, group);
1306 if (buddy == NULL) {
1307 /* Add it to the gaim buddy list */
1308 buddy = gaim_buddy_new(user->client_data,
1309 name,
1310 nm_contact_get_display_name(contact));
1311
1312 gaim_blist_add_buddy(buddy, NULL, group, NULL);
1313 }
1314
1315 /* Set the initial status for the buddy */
1316 user_record = nm_contact_get_user_record(contact);
1317 if (user_record) {
1318 status = nm_user_record_get_status(user_record);
1319 text = nm_user_record_get_status_text(user_record);
1320 }
1321 _update_buddy_status(user, buddy, status, time(0));
1322
1323 /* Save the new buddy as part of the contact object */
1324 nm_contact_set_data(contact, (gpointer) buddy);
1325 }
1326
1327 } else {
1328 /* NULL contact. This should not happen, but
1329 * let's break out of the loop.
1330 */
1331 break;
1332 }
1333 }
1334 }
1335
1336 /* Add all of the server side contacts to the Gaim buddy list. */
1337 static void
1338 _add_gaim_buddies(NMUser * user)
1339 {
1340 int cnt = 0, i;
1341 NMFolder *root_folder = NULL;
1342 NMFolder *folder = NULL;
1343
1344 root_folder = nm_get_root_folder(user);
1345 if (root_folder) {
1346
1347 /* Add sub-folders and contacts to sub-folders...
1348 * iterate throught the sub-folders in reverse order
1349 * because Gaim adds the folders to the front -- so we
1350 * want to add the first folder last
1351 */
1352 cnt = nm_folder_get_subfolder_count(root_folder);
1353 for (i = cnt-1; i >= 0; i--) {
1354 folder = nm_folder_get_subfolder(root_folder, i);
1355 if (folder) {
1356 _add_contacts_to_gaim_blist(user, folder);
1357 }
1358 }
1359
1360 /* Add contacts for the root folder */
1361 _add_contacts_to_gaim_blist(user, root_folder);
1362 }
1363 }
1364
1365 static void
1366 _sync_contact_list(NMUser *user)
1367 {
1368 /* Remove all buddies from the local list that are
1369 * not in the server side list and add all buddies
1370 * from the server side list that are not in
1371 * the local list
1372 */
1373 _remove_gaim_buddies(user);
1374 _add_gaim_buddies(user);
1375 user->clist_synched = TRUE;
1376 }
1377
1378 static void
1379 _sync_privacy_lists(NMUser *user)
1380 {
1381 GSList *node = NULL, *rem_list = NULL;
1382 GaimConnection *gc;
1383 const char *name, *dn;
1384 NMUserRecord *user_record;
1385
1386 if (user == NULL)
1387 return;
1388
1389 gc = gaim_account_get_connection(user->client_data);
1390 if (gc == NULL)
1391 return;
1392
1393 /* Set the Gaim privacy setting */
1394 if (user->default_deny) {
1395 if (user->allow_list == NULL) {
1396 gc->account->perm_deny = GAIM_PRIVACY_DENY_ALL;
1397 } else {
1398 gc->account->perm_deny = GAIM_PRIVACY_ALLOW_USERS;
1399 }
1400 } else {
1401 if (user->deny_list == NULL) {
1402 gc->account->perm_deny = GAIM_PRIVACY_ALLOW_ALL;
1403 } else {
1404 gc->account->perm_deny = GAIM_PRIVACY_DENY_USERS;
1405 }
1406 }
1407
1408 /* Add stuff */
1409 for (node = user->allow_list; node; node = node->next) {
1410 user_record = nm_find_user_record(user, (char *)node->data);
1411 if (user_record)
1412 name = nm_user_record_get_display_id(user_record);
1413 else
1414 name =(char *)node->data;
1415
1416 if (!g_slist_find_custom(gc->account->permit,
1417 name, (GCompareFunc)nm_utf8_strcasecmp)) {
1418 gaim_privacy_permit_add(gc->account, name , TRUE);
1419 }
1420 }
1421
1422 for (node = user->deny_list; node; node = node->next) {
1423 user_record = nm_find_user_record(user, (char *)node->data);
1424 if (user_record)
1425 name = nm_user_record_get_display_id(user_record);
1426 else
1427 name =(char *)node->data;
1428
1429 if (!g_slist_find_custom(gc->account->deny,
1430 name, (GCompareFunc)nm_utf8_strcasecmp)) {
1431 gaim_privacy_deny_add(gc->account, name, TRUE);
1432 }
1433 }
1434
1435
1436 /* Remove stuff */
1437 for (node = gc->account->permit; node; node = node->next) {
1438 dn = nm_lookup_dn(user, (char *)node->data);
1439 if (dn != NULL &&
1440 !g_slist_find_custom(user->allow_list,
1441 dn, (GCompareFunc)nm_utf8_strcasecmp)) {
1442 rem_list = g_slist_append(rem_list, node->data);
1443 }
1444 }
1445
1446 if (rem_list) {
1447 for (node = rem_list; node; node = node->next) {
1448 gaim_privacy_permit_remove(gc->account, (char *)node->data, TRUE);
1449 }
1450 g_free(rem_list);
1451 rem_list = NULL;
1452 }
1453
1454 for (node = gc->account->deny; node; node = node->next) {
1455 dn = nm_lookup_dn(user, (char *)node->data);
1456 if (dn != NULL &&
1457 !g_slist_find_custom(user->deny_list,
1458 dn, (GCompareFunc)nm_utf8_strcasecmp)) {
1459 rem_list = g_slist_append(rem_list, node->data);
1460 }
1461 }
1462
1463 if (rem_list) {
1464 for (node = rem_list; node; node = node->next) {
1465 gaim_privacy_deny_remove(gc->account, (char *)node->data, TRUE);
1466 }
1467 g_slist_free(rem_list);
1468 }
1469 }
1470
1471 /* Map known property tags to user-friendly strings */
1472 static const char *
1473 _map_property_tag(const char *tag)
1474 {
1475 if (tag == NULL) return NULL;
1476
1477 if (strcmp(tag, "telephoneNumber") == 0)
1478 return _("Telephone Number");
1479 else if (strcmp(tag, "L") == 0)
1480 return _("Location");
1481 else if (strcmp(tag, "OU") == 0)
1482 return _("Department");
1483 else if (strcmp(tag, "personalTitle") == 0)
1484 return _("Personal Title");
1485 else if (strcmp(tag, "Title") == 0)
1486 return _("Title");
1487 else if (strcmp(tag, "mailstop") == 0)
1488 return _("Mailstop");
1489 else if (strcmp(tag, "Internet EMail Address") == 0)
1490 return _("Email Address");
1491 else
1492 return tag;
1493 }
1494
1495 /* Display a dialog box showing the properties for the given user record */
1496 static void
1497 _show_info(GaimConnection * gc, NMUserRecord * user_record)
1498 {
1499 GString *info_text;
1500 int count, i;
1501 NMProperty *property;
1502 const char *tag, *value;
1503
1504 info_text = g_string_new("");
1505
1506 tag = _("User ID");
1507 value = nm_user_record_get_userid(user_record);
1508 if (value) {
1509 g_string_append_printf(info_text, "<b>%s:</b> %s<br>", tag, value);
1510 }
1511
1512 /* tag = _("DN");
1513 value = nm_user_record_get_dn(user_record);
1514 if (value) {
1515 g_string_append_printf(info_text, "<b>%s:</b> %s<br>",
1516 tag, value);
1517 }
1518 */
1519
1520 tag = _("Full name");
1521 value = nm_user_record_get_full_name(user_record);
1522 if (value) {
1523 g_string_append_printf(info_text, "<b>%s:</b> %s<br>", tag, value);
1524 }
1525
1526 count = nm_user_record_get_property_count(user_record);
1527 for (i = 0; i < count; i++) {
1528 property = nm_user_record_get_property(user_record, i);
1529 if (property) {
1530 tag = _map_property_tag(nm_property_get_tag(property));
1531 value = nm_property_get_value(property);
1532 if (tag && value) {
1533 g_string_append_printf(info_text, "<b>%s:</b> %s<br>",
1534 tag, value);
1535 }
1536 nm_release_property(property);
1537 }
1538 }
1539
1540 gaim_notify_userinfo(gc, nm_user_record_get_userid(user_record),
1541 info_text->str, NULL, NULL);
1542
1543 g_string_free(info_text, TRUE);
1544 }
1545
1546 /* Send a join conference, the first item in the parms list is the
1547 * NMUser object and the second item is the conference to join.
1548 * This callback is passed to gaim_request_action when we ask the
1549 * user if they want to join the conference.
1550 */
1551 static void
1552 _join_conference_cb(GSList * parms)
1553 {
1554 NMUser *user;
1555 NMConference *conference;
1556 NMERR_T rc = NM_OK;
1557
1558 if (parms == NULL || g_slist_length(parms) != 2)
1559 return;
1560
1561 user = g_slist_nth_data(parms, 0);
1562 conference = g_slist_nth_data(parms, 1);
1563
1564 if (user && conference) {
1565 rc = nm_send_join_conference(user, conference,
1566 _join_conf_resp_cb, conference);
1567 _check_for_disconnect(user, rc);
1568 }
1569
1570 g_slist_free(parms);
1571 }
1572
1573 /* Send a reject conference, the first item in the parms list is the
1574 * NMUser object and the second item is the conference to reject.
1575 * This callback is passed to gaim_request_action when we ask the
1576 * user if they want to joing the conference.
1577 */
1578 static void
1579 _reject_conference_cb(GSList * parms)
1580 {
1581 NMUser *user;
1582 NMConference *conference;
1583 NMERR_T rc = NM_OK;
1584
1585 if (parms == NULL || g_slist_length(parms) != 2)
1586 return;
1587
1588 user = g_slist_nth_data(parms, 0);
1589 conference = g_slist_nth_data(parms, 1);
1590
1591 if (user && conference) {
1592 rc = nm_send_reject_conference(user, conference, NULL, NULL);
1593 _check_for_disconnect(user, rc);
1594 }
1595
1596 g_slist_free(parms);
1597 }
1598
1599 static void
1600 _initiate_conference_cb(GaimBlistNode *node, gpointer ignored)
1601 {
1602 GaimBuddy *buddy;
1603 GaimConnection *gc;
1604
1605 NMUser *user;
1606 const char *conf_name;
1607 GaimConversation *chat = NULL;
1608 NMUserRecord *user_record;
1609 NMConference *conference;
1610
1611 g_return_if_fail(GAIM_BLIST_NODE_IS_BUDDY(node));
1612
1613 buddy = (GaimBuddy *) node;
1614 gc = gaim_account_get_connection(buddy->account);
1615
1616 user = gc->proto_data;
1617 if (user == NULL)
1618 return;
1619
1620 /* We should already have a userrecord for the buddy */
1621 user_record = nm_find_user_record(user, buddy->name);
1622 if (user_record == NULL)
1623 return;
1624
1625 conf_name = _get_conference_name(++user->conference_count);
1626 chat = serv_got_joined_chat(gc, user->conference_count, conf_name);
1627 if (chat) {
1628
1629 conference = nm_create_conference(NULL);
1630 nm_conference_set_data(conference, (gpointer) chat);
1631 nm_send_create_conference(user, conference, _createconf_resp_send_invite, user_record);
1632 nm_release_conference(conference);
1633 }
1634 }
1635
1636 const char *
1637 _get_conference_name(int id)
1638 {
1639 static char *name = NULL;
1640
1641 if (name)
1642 g_free(name);
1643
1644 name = g_strdup_printf(_("GroupWise Conference %d"), id);
1645
1646 return name;
1647 }
1648
1649 static void
1650 _show_privacy_locked_error(GaimConnection *gc, NMUser *user)
1651 {
1652 char *err;
1653
1654 err = g_strdup_printf(_("Unable to change server side privacy settings (%s)."),
1655 nm_error_to_string(NMERR_ADMIN_LOCKED));
1656 gaim_notify_error(gc, NULL, err, NULL);
1657 g_free(err);
1658 }
1659
1660 /*******************************************************************************
1661 * Connect and recv callbacks
1662 ******************************************************************************/
1663
1664 static void
1665 novell_ssl_connect_error(GaimSslConnection * gsc,
1666 GaimSslErrorType error, gpointer data)
1667 {
1668 gaim_connection_error((GaimConnection *)data,
1669 _("Unable to make SSL connection to server."));
1670 }
1671
1672 static void
1673 novell_ssl_recv_cb(gpointer data, GaimSslConnection * gsc,
1674 GaimInputCondition condition)
1675 {
1676 GaimConnection *gc = data;
1677 NMUser *user;
1678 NMERR_T rc;
1679
1680 if (gc == NULL)
1681 return;
1682
1683 user = gc->proto_data;
1684 if (user == NULL)
1685 return;
1686
1687 rc = nm_process_new_data(user);
1688 if (rc != NM_OK) {
1689
1690 if (_is_disconnect_error(rc)) {
1691
1692 gaim_connection_error(gc,
1693 _("Error communicating with server."
1694 " Closing connection."));
1695 } else {
1696 gaim_debug(GAIM_DEBUG_INFO, "novell",
1697 "Error processing event or response (%d).\n", rc);
1698 }
1699 }
1700 }
1701
1702 static void
1703 novell_ssl_connected_cb(gpointer data, GaimSslConnection * gsc,
1704 GaimInputCondition cond)
1705 {
1706 GaimConnection *gc = data;
1707 NMUser *user;
1708 NMConn *conn;
1709 NMERR_T rc = 0;
1710 const char *pwd = NULL;
1711 const char *my_addr = NULL;
1712 char *ua = NULL;
1713
1714 if (gc == NULL || gsc == NULL)
1715 return;
1716
1717 user = gc->proto_data;
1718 if ((user == NULL) || (conn = user->conn) == NULL)
1719 return;
1720
1721 conn->ssl_conn = g_new0(NMSSLConn, 1);
1722 conn->ssl_conn->data = gsc;
1723 conn->ssl_conn->read = (nm_ssl_read_cb) gaim_ssl_read;
1724 conn->ssl_conn->write = (nm_ssl_write_cb) gaim_ssl_write;
1725
1726 gaim_connection_update_progress(gc, _("Authenticating..."),
1727 2, NOVELL_CONNECT_STEPS);
1728
1729 my_addr = gaim_network_get_my_ip(gsc->fd);
1730 pwd = gaim_connection_get_password(gc);
1731 ua = _user_agent_string();
1732
1733 rc = nm_send_login(user, pwd, my_addr, ua, _login_resp_cb, NULL);
1734 if (rc == NM_OK) {
1735 conn->connected = TRUE;
1736 gaim_ssl_input_add(gsc, novell_ssl_recv_cb, gc);
1737 } else {
1738 gaim_connection_error(gc, _("Unable to connect to server."));
1739 }
1740
1741 gaim_connection_update_progress(gc, _("Waiting for response..."),
1742 3, NOVELL_CONNECT_STEPS);
1743
1744 g_free(ua);
1745 }
1746
1747 /*******************************************************************************
1748 * Event callback and event handlers
1749 ******************************************************************************/
1750
1751 static void
1752 _evt_receive_message(NMUser * user, NMEvent * event)
1753 {
1754 NMUserRecord *user_record = NULL;
1755 NMContact *contact = NULL;
1756 GaimConversation *gconv;
1757 NMConference *conference;
1758 GaimMessageFlags flags;
1759 char *text = NULL;
1760
1761 text = g_markup_escape_text(nm_event_get_text(event), -1);
1762
1763 conference = nm_event_get_conference(event);
1764 if (conference) {
1765
1766 GaimConversation *chat = nm_conference_get_data(conference);
1767
1768 /* Is this a single person 'conversation' or a conference? */
1769 if (chat == NULL && nm_conference_get_participant_count(conference) == 1) {
1770
1771 user_record = nm_find_user_record(user, nm_event_get_source(event));
1772 if (user_record) {
1773
1774 flags = 0;
1775 if (nm_event_get_type(event) == NMEVT_RECEIVE_AUTOREPLY)
1776 flags |= GAIM_MESSAGE_AUTO_RESP;
1777
1778 serv_got_im(gaim_account_get_connection(user->client_data),
1779 nm_user_record_get_display_id(user_record),
1780 text, flags,
1781 nm_event_get_gmt(event));
1782
1783 gconv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM,
1784 nm_user_record_get_display_id(user_record),
1785 (GaimAccount *) user->client_data);
1786 if (gconv) {
1787
1788 contact = nm_find_contact(user, nm_event_get_source(event));
1789 if (contact) {
1790
1791 gaim_conversation_set_title(
1792 gconv, nm_contact_get_display_name(contact));
1793
1794
1795 } else {
1796
1797 const char *name =
1798 nm_user_record_get_full_name(user_record);
1799
1800 if (name == NULL)
1801 name = nm_user_record_get_userid(user_record);
1802
1803 gaim_conversation_set_title(gconv, name);
1804 }
1805
1806 }
1807
1808 } else {
1809 /* this should not happen, see the event code.
1810 * the event code will get the contact details from
1811 * the server if it does not have them before calling
1812 * the event callback.
1813 */
1814 }
1815
1816 } else if (chat) {
1817
1818 /* get the contact for send if we have one */
1819 NMContact *contact = nm_find_contact(user,
1820 nm_event_get_source(event));
1821
1822 /* get the user record for the sender */
1823 user_record = nm_find_user_record(user, nm_event_get_source(event));
1824 if (user_record) {
1825 const char *name = nm_contact_get_display_name(contact);
1826
1827 if (name == NULL) {
1828 name = nm_user_record_get_full_name(user_record);
1829 if (name == NULL)
1830 name = nm_user_record_get_display_id(user_record);
1831 }
1832
1833 serv_got_chat_in(gaim_account_get_connection(user->client_data),
1834 gaim_conv_chat_get_id(GAIM_CONV_CHAT(chat)),
1835 name, 0, text, nm_event_get_gmt(event));
1836 }
1837 }
1838 }
1839
1840 g_free(text);
1841 }
1842
1843 static void
1844 _evt_conference_left(NMUser * user, NMEvent * event)
1845 {
1846 GaimConversation *chat;
1847 NMConference *conference;
1848
1849 conference = nm_event_get_conference(event);
1850 if (conference) {
1851 chat = nm_conference_get_data(conference);
1852 if (chat) {
1853 NMUserRecord *ur = nm_find_user_record(user,
1854 nm_event_get_source(event));
1855
1856 if (ur)
1857 gaim_conv_chat_remove_user(GAIM_CONV_CHAT(chat),
1858 nm_user_record_get_display_id(ur),
1859 NULL);
1860 }
1861 }
1862 }
1863
1864 static void
1865 _evt_conference_invite_notify(NMUser * user, NMEvent * event)
1866 {
1867 GaimConversation *gconv;
1868 NMConference *conference;
1869 NMUserRecord *user_record = NULL;
1870 char *str = NULL;
1871
1872 user_record = nm_find_user_record(user, nm_event_get_source(event));
1873 conference = nm_event_get_conference(event);
1874 if (user_record && conference) {
1875 gconv = nm_conference_get_data(conference);
1876 str = g_strdup_printf(_("%s has been invited to this conversation."),
1877 nm_user_record_get_display_id(user_record));
1878 gaim_conversation_write(gconv, NULL, str,
1879 GAIM_MESSAGE_SYSTEM, time(NULL));
1880 g_free(str);
1881 }
1882 }
1883
1884 static void
1885 _evt_conference_invite(NMUser * user, NMEvent * event)
1886 {
1887 NMUserRecord *ur;
1888 GaimConnection *gc;
1889 GSList *parms = NULL;
1890 const char *title = NULL;
1891 const char *secondary = NULL;
1892 const char *name = NULL;
1893 char *primary = NULL;
1894 time_t gmt;
1895
1896 ur = nm_find_user_record(user, nm_event_get_source(event));
1897 if (ur)
1898 name = nm_user_record_get_full_name(ur);
1899
1900 if (name == NULL)
1901 name = nm_event_get_source(event);
1902
1903 gmt = nm_event_get_gmt(event);
1904 title = _("Invitation to Conversation");
1905 primary = g_strdup_printf(_("Invitation from: %s\n\nSent: %s"),
1906 name, asctime(localtime(&gmt)));
1907 secondary = _("Would you like to join the conversation?");
1908
1909 /* Set up parms list for the callbacks
1910 * We need to send the NMUser object and
1911 * the NMConference object to the callbacks
1912 */
1913 parms = NULL;
1914 parms = g_slist_append(parms, user);
1915 parms = g_slist_append(parms, nm_event_get_conference(event));
1916
1917 /* Prompt the user */
1918 gc = gaim_account_get_connection(user->client_data);
1919 gaim_request_action(gc, title, primary, secondary,
1920 GAIM_DEFAULT_ACTION_NONE, parms, 2,
1921 _("Yes"), G_CALLBACK(_join_conference_cb),
1922 _("No"), G_CALLBACK(_reject_conference_cb));
1923
1924 g_free(primary);
1925 }
1926
1927
1928 static void
1929 _evt_conference_joined(NMUser * user, NMEvent * event)
1930 {
1931 GaimConversation *chat = NULL;
1932 GaimConnection *gc;
1933 NMConference *conference = NULL;
1934 NMUserRecord *ur = NULL;
1935 const char *name;
1936 const char *conf_name;
1937
1938 gc = gaim_account_get_connection(user->client_data);
1939 if (gc == NULL)
1940 return;
1941
1942 conference = nm_event_get_conference(event);
1943 if (conference) {
1944 chat = nm_conference_get_data(conference);
1945 if (nm_conference_get_participant_count(conference) == 2 && chat == NULL) {
1946 ur = nm_conference_get_participant(conference, 0);
1947 if (ur) {
1948 conf_name = _get_conference_name(++user->conference_count);
1949 chat =
1950 serv_got_joined_chat(gc, user->conference_count, conf_name);
1951 if (chat) {
1952
1953 nm_conference_set_data(conference, (gpointer) chat);
1954
1955 name = nm_user_record_get_display_id(ur);
1956 gaim_conv_chat_add_user(GAIM_CONV_CHAT(chat), name, NULL,
1957 GAIM_CBFLAGS_NONE, TRUE);
1958
1959 }
1960 }
1961 }
1962
1963 if (chat != NULL) {
1964 ur = nm_find_user_record(user, nm_event_get_source(event));
1965 if (ur) {
1966 name = nm_user_record_get_display_id(ur);
1967 if (!gaim_conv_chat_find_user(GAIM_CONV_CHAT(chat), name)) {
1968 gaim_conv_chat_add_user(GAIM_CONV_CHAT(chat), name, NULL,
1969 GAIM_CBFLAGS_NONE, TRUE);
1970 }
1971 }
1972 }
1973 }
1974 }
1975
1976 static void
1977 _evt_status_change(NMUser * user, NMEvent * event)
1978 {
1979 GaimBuddy *buddy = NULL;
1980 GSList *buddies;
1981 GSList *bnode;
1982 NMUserRecord *user_record;
1983 const char *display_id;
1984 int status;
1985
1986 user_record = nm_event_get_user_record(event);
1987 if (user_record) {
1988
1989 /* Retrieve new status */
1990 status = nm_user_record_get_status(user_record);
1991
1992 /* Update status for buddy in all folders */
1993 display_id = nm_user_record_get_display_id(user_record);
1994 buddies = gaim_find_buddies(user->client_data, display_id);
1995 for (bnode = buddies; bnode; bnode = bnode->next) {
1996 buddy = (GaimBuddy *) bnode->data;
1997 if (buddy) {
1998 _update_buddy_status(user, buddy, status, nm_event_get_gmt(event));
1999 }
2000 }
2001
2002 g_slist_free(buddies);
2003
2004 }
2005 }
2006
2007 static void
2008 _evt_user_disconnect(NMUser * user, NMEvent * event)
2009 {
2010 GaimConnection *gc;
2011
2012 gc = gaim_account_get_connection((GaimAccount *) user->client_data);
2013 if (gc)
2014 gaim_connection_error(gc, _("You have been logged out because you"
2015 " logged in at another workstation."));
2016 }
2017
2018 static void
2019 _evt_user_typing(NMUser * user, NMEvent * event)
2020 {
2021 GaimConnection *gc;
2022 NMUserRecord *user_record = NULL;
2023
2024 gc = gaim_account_get_connection((GaimAccount *) user->client_data);
2025 if (gc) {
2026 user_record = nm_find_user_record(user, nm_event_get_source(event));
2027 if (user_record) {
2028 serv_got_typing(gc, nm_user_record_get_display_id(user_record),
2029 30, GAIM_TYPING);
2030 }
2031 }
2032 }
2033
2034 static void
2035 _evt_user_not_typing(NMUser * user, NMEvent * event)
2036 {
2037 GaimConnection *gc;
2038 NMUserRecord *user_record;
2039
2040 gc = gaim_account_get_connection((GaimAccount *) user->client_data);
2041 if (gc) {
2042 user_record = nm_find_user_record(user, nm_event_get_source(event));
2043 if (user_record) {
2044 serv_got_typing_stopped(gc,
2045 nm_user_record_get_display_id(user_record));
2046 }
2047 }
2048 }
2049
2050 static void
2051 _evt_undeliverable_status(NMUser * user, NMEvent * event)
2052 {
2053 NMUserRecord *ur;
2054 GaimConversation *gconv;
2055 char *str;
2056
2057 ur = nm_find_user_record(user, nm_event_get_source(event));
2058 if (ur) {
2059 /* XXX - Should this be GAIM_CONV_TYPE_IM? */
2060 gconv =
2061 gaim_find_conversation_with_account(GAIM_CONV_TYPE_ANY,
2062 nm_user_record_get_display_id(ur),
2063 user->client_data);
2064 if (gconv) {
2065 const char *name = nm_user_record_get_full_name(ur);
2066
2067 if (name == NULL) {
2068 name = nm_user_record_get_display_id(ur);
2069 }
2070 str = g_strdup_printf(_("%s appears to be offline and did not receive"
2071 " the message that you just sent."), name);
2072 gaim_conversation_write(gconv, NULL, str,
2073 GAIM_MESSAGE_SYSTEM, time(NULL));
2074 g_free(str);
2075 }
2076 }
2077 }
2078
2079 static void
2080 _event_callback(NMUser * user, NMEvent * event)
2081 {
2082 if (user == NULL || event == NULL)
2083 return;
2084
2085 switch (nm_event_get_type(event)) {
2086 case NMEVT_STATUS_CHANGE:
2087 _evt_status_change(user, event);
2088 break;
2089 case NMEVT_RECEIVE_AUTOREPLY:
2090 case NMEVT_RECEIVE_MESSAGE:
2091 _evt_receive_message(user, event);
2092 break;
2093 case NMEVT_USER_DISCONNECT:
2094 _evt_user_disconnect(user, event);
2095 break;
2096 case NMEVT_USER_TYPING:
2097 _evt_user_typing(user, event);
2098 break;
2099 case NMEVT_USER_NOT_TYPING:
2100 _evt_user_not_typing(user, event);
2101 break;
2102 case NMEVT_SERVER_DISCONNECT:
2103 /* Nothing to do? */
2104 break;
2105 case NMEVT_INVALID_RECIPIENT:
2106 break;
2107 case NMEVT_UNDELIVERABLE_STATUS:
2108 _evt_undeliverable_status(user, event);
2109 break;
2110 case NMEVT_CONFERENCE_INVITE_NOTIFY:
2111 /* Someone else has been invited to join a
2112 * conference that we are currently a part of
2113 */
2114 _evt_conference_invite_notify(user, event);
2115 break;
2116 case NMEVT_CONFERENCE_INVITE:
2117 /* We have been invited to join a conference */
2118 _evt_conference_invite(user, event);
2119 break;
2120 case NMEVT_CONFERENCE_JOINED:
2121 /* Some one has joined a conference that we
2122 * are a part of
2123 */
2124 _evt_conference_joined(user, event);
2125 break;
2126 case NMEVT_CONFERENCE_LEFT:
2127 /* Someone else has left a conference that we
2128 * are currently a part of
2129 */
2130 _evt_conference_left(user, event);
2131 break;
2132 default:
2133 gaim_debug(GAIM_DEBUG_INFO, "novell",
2134 "_event_callback(): unhandled event, %d\n",
2135 nm_event_get_type(event));
2136 break;
2137 }
2138 }
2139
2140 /*******************************************************************************
2141 * Prpl Ops
2142 ******************************************************************************/
2143
2144 static void
2145 novell_login(GaimAccount * account)
2146 {
2147 GaimConnection *gc;
2148 NMUser *user = NULL;
2149 const char *server;
2150 const char *name;
2151 int port;
2152
2153 if (account == NULL)
2154 return;
2155
2156 gc = gaim_account_get_connection(account);
2157 if (gc == NULL)
2158 return;
2159
2160 server = gaim_account_get_string(account, "server", NULL);
2161 if (server == NULL || *server == '\0') {
2162
2163 /* TODO: Would be nice to prompt if not set!
2164 * gaim_request_fields(gc, _("Server Address"),...);
2165 */
2166
2167 /* ...but for now just error out with a nice message. */
2168 gaim_connection_error(gc, _("Unable to connect to server."
2169 " Please enter the address of the server"
2170 " you wish to connect to."));
2171 return;
2172 }
2173
2174 port = gaim_account_get_int(account, "port", DEFAULT_PORT);
2175 name = gaim_account_get_username(account);
2176
2177 user = nm_initialize_user(name, server, port, account, _event_callback);
2178 if (user) {
2179 /* save user */
2180 gc->proto_data = user;
2181
2182 /* connect to the server */
2183 gaim_connection_update_progress(gc, _("Connecting"),
2184 1, NOVELL_CONNECT_STEPS);
2185
2186 user->conn->use_ssl = TRUE;
2187 if (gaim_ssl_connect(user->client_data, user->conn->addr,
2188 user->conn->port, novell_ssl_connected_cb,
2189 novell_ssl_connect_error, gc) == NULL) {
2190 gaim_connection_error(gc, _("Error."
2191 " SSL support is not installed."));
2192 }
2193 }
2194 }
2195
2196 static void
2197 novell_close(GaimConnection * gc)
2198 {
2199 NMUser *user;
2200 NMConn *conn;
2201
2202 if (gc == NULL)
2203 return;
2204
2205 user = gc->proto_data;
2206 if (user) {
2207 conn = user->conn;
2208 if (conn && conn->ssl_conn) {
2209 gaim_ssl_close(user->conn->ssl_conn->data);
2210 }
2211 nm_deinitialize_user(user);
2212 }
2213 gc->proto_data = NULL;
2214 }
2215
2216 static int
2217 novell_send_im(GaimConnection * gc, const char *name,
2218 const char *message_body, GaimMessageFlags flags)
2219 {
2220 NMUserRecord *user_record = NULL;
2221 NMConference *conf = NULL;
2222 NMMessage *message;
2223 NMUser *user;
2224 const char *dn = NULL;
2225 char *plain;
2226 gboolean done = TRUE, created_conf = FALSE;
2227 NMERR_T rc = NM_OK;
2228
2229 if (gc == NULL || name == NULL ||
2230 message_body == NULL || *message_body == '\0')
2231 return 0;
2232
2233 user = gc->proto_data;
2234 if (user == NULL)
2235 return 0;
2236
2237 /* Create a new message */
2238 plain = gaim_unescape_html(message_body);
2239 message = nm_create_message(plain);
2240 g_free(plain);
2241
2242 /* Need to get the DN for the buddy so we can look up the convo */
2243 dn = nm_lookup_dn(user, name);
2244
2245 /* Do we already know about the sender? */
2246 user_record = nm_find_user_record(user, dn);
2247 if (user_record) {
2248
2249 /* Do we already have an instantiated conference? */
2250 conf = nm_find_conversation(user, dn);
2251 if (conf == NULL) {
2252
2253 /* If not, create a blank conference */
2254 conf = nm_create_conference(NULL);
2255 created_conf = TRUE;
2256
2257 nm_conference_add_participant(conf, user_record);
2258 }
2259
2260 nm_message_set_conference(message, conf);
2261
2262 /* Make sure conference is instantiated */
2263 if (!nm_conference_is_instantiated(conf)) {
2264
2265 /* It is not, so send the createconf. We will
2266 * have to finish sending the message when we
2267 * get the response with the new conference guid.
2268 */
2269 rc = nm_send_create_conference(user, conf, _createconf_resp_send_msg, message);
2270 _check_for_disconnect(user, rc);
2271
2272 done = FALSE;
2273 }
2274
2275 } else {
2276
2277 /* If we don't have details for the user, then we don't have
2278 * a conference yet. So create one and send the getdetails
2279 * to the server. We will have to finish sending the message
2280 * when we get the response from the server.
2281 */
2282 conf = nm_create_conference(NULL);
2283 created_conf = TRUE;
2284
2285 nm_message_set_conference(message, conf);
2286
2287 rc = nm_send_get_details(user, name, _get_details_resp_send_msg, message);
2288 _check_for_disconnect(user, rc);
2289
2290 done = FALSE;
2291 }
2292
2293 if (done) {
2294
2295 /* Did we find everything we needed? */
2296 rc = nm_send_message(user, message, _send_message_resp_cb);
2297 _check_for_disconnect(user, rc);
2298
2299 nm_release_message(message);
2300 }
2301
2302 if (created_conf && conf)
2303 nm_release_conference(conf);
2304
2305 return 1;
2306 }
2307
2308 static int
2309 novell_send_typing(GaimConnection * gc, const char *name, int typing)
2310 {
2311 NMConference *conf = NULL;
2312 NMUser *user;
2313 const char *dn = NULL;
2314 NMERR_T rc = NM_OK;
2315
2316 if (gc == NULL || name == NULL)
2317 return -1;
2318
2319 user = gc->proto_data;
2320 if (user == NULL)
2321 return -1;
2322
2323 /* Need to get the DN for the buddy so we can look up the convo */
2324 dn = nm_lookup_dn(user, name);
2325 if (dn) {
2326
2327 /* Now find the conference in our list */
2328 conf = nm_find_conversation(user, dn);
2329 if (conf) {
2330
2331 rc = nm_send_typing(user, conf,
2332 ((typing == GAIM_TYPING) ? TRUE : FALSE), NULL);
2333 _check_for_disconnect(user, rc);
2334
2335 }
2336
2337 }
2338
2339 return 0;
2340 }
2341
2342 static void
2343 novell_convo_closed(GaimConnection * gc, const char *who)
2344 {
2345 NMUser *user;
2346 NMConference *conf;
2347 const char *dn;
2348 NMERR_T rc = NM_OK;
2349
2350 if (gc == NULL || who == NULL)
2351 return;
2352
2353 user = gc->proto_data;
2354 if (user && (dn = nm_lookup_dn(user, who))) {
2355 conf = nm_find_conversation(user, dn);
2356 if (conf) {
2357 rc = nm_send_leave_conference(user, conf, NULL, NULL);
2358 _check_for_disconnect(user, rc);
2359 }
2360 }
2361 }
2362
2363 static void
2364 novell_chat_leave(GaimConnection * gc, int id)
2365 {
2366 NMConference *conference;
2367 NMUser *user;
2368 GaimConversation *chat;
2369 GSList *cnode;
2370 NMERR_T rc = NM_OK;
2371
2372 if (gc == NULL)
2373 return;
2374
2375 user = gc->proto_data;
2376 if (user == NULL)
2377 return;
2378
2379 for (cnode = user->conferences; cnode != NULL; cnode = cnode->next) {
2380 conference = cnode->data;
2381 if (conference && (chat = nm_conference_get_data(conference))) {
2382 if (gaim_conv_chat_get_id(GAIM_CONV_CHAT(chat)) == id) {
2383 rc = nm_send_leave_conference(user, conference, NULL, NULL);
2384 _check_for_disconnect(user, rc);
2385 break;
2386 }
2387 }
2388 }
2389
2390 serv_got_chat_left(gc, id);
2391 }
2392
2393 static void
2394 novell_chat_invite(GaimConnection *gc, int id,
2395 const char *message, const char *who)
2396 {
2397 NMConference *conference;
2398 NMUser *user;
2399 GaimConversation *chat;
2400 GSList *cnode;
2401 NMERR_T rc = NM_OK;
2402 NMUserRecord *user_record = NULL;
2403
2404 if (gc == NULL)
2405 return;
2406
2407 user = gc->proto_data;
2408 if (user == NULL)
2409 return;
2410
2411 user_record = nm_find_user_record(user, who);
2412 if (user_record == NULL) {
2413 rc = nm_send_get_details(user, who, _get_details_resp_send_invite, GINT_TO_POINTER(id));
2414 _check_for_disconnect(user, rc);
2415 return;
2416 }
2417
2418 for (cnode = user->conferences; cnode != NULL; cnode = cnode->next) {
2419 conference = cnode->data;
2420 if (conference && (chat = nm_conference_get_data(conference))) {
2421 if (gaim_conv_chat_get_id(GAIM_CONV_CHAT(chat)) == id) {
2422 rc = nm_send_conference_invite(user, conference, user_record,
2423 message, _sendinvite_resp_cb, NULL);
2424 _check_for_disconnect(user, rc);
2425 break;
2426 }
2427 }
2428 }
2429 }
2430
2431 static int
2432 novell_chat_send(GaimConnection * gc, int id, const char *text, GaimMessageFlags flags)
2433 {
2434 NMConference *conference;
2435 GaimConversation *chat;
2436 GSList *cnode;
2437 NMMessage *message;
2438 NMUser *user;
2439 NMERR_T rc = NM_OK;
2440 const char *name;
2441 char *str, *plain;
2442
2443 if (gc == NULL || text == NULL)
2444 return -1;
2445
2446 user = gc->proto_data;
2447 if (user == NULL)
2448 return -1;
2449
2450 plain = gaim_unescape_html(text);
2451 message = nm_create_message(plain);
2452 g_free(plain);
2453
2454 for (cnode = user->conferences; cnode != NULL; cnode = cnode->next) {
2455 conference = cnode->data;
2456 if (conference && (chat = nm_conference_get_data(conference))) {
2457 if (gaim_conv_chat_get_id(GAIM_CONV_CHAT(chat)) == id) {
2458
2459 nm_message_set_conference(message, conference);
2460
2461 /* check to see if the conference is instatiated yet */
2462 if (!nm_conference_is_instantiated(conference)) {
2463 nm_message_add_ref(message);
2464 nm_send_create_conference(user, conference, _createconf_resp_send_msg, message);
2465 } else {
2466 rc = nm_send_message(user, message, _send_message_resp_cb);
2467 }
2468
2469 nm_release_message(message);
2470
2471 if (!_check_for_disconnect(user, rc)) {
2472
2473 /* Use the account alias if it is set */
2474 name = gaim_account_get_alias(user->client_data);
2475 if (name == NULL || *name == '\0') {
2476
2477 /* If there is no account alias, try full name */
2478 name = nm_user_record_get_full_name(user->user_record);
2479 if (name == NULL || *name == '\0') {
2480
2481 /* Fall back to the username that we are signed in with */
2482 name = gaim_account_get_username(user->client_data);
2483 }
2484 }
2485
2486 serv_got_chat_in(gc, id, name, 0, text, time(NULL));
2487 return 0;
2488 } else
2489 return -1;
2490
2491 }
2492 }
2493 }
2494
2495
2496 /* The conference was not found, must be closed */
2497 chat = gaim_find_chat(gc, id);
2498 if (chat) {
2499 str = g_strdup_printf(_("This conference has been closed."
2500 " No more messages can be sent."));
2501 gaim_conversation_write(chat, NULL, str, GAIM_MESSAGE_SYSTEM, time(NULL));
2502 g_free(str);
2503 }
2504
2505 if (message)
2506 nm_release_message(message);
2507
2508 return -1;
2509 }
2510
2511 static void
2512 novell_add_buddy(GaimConnection * gc, GaimBuddy *buddy, GaimGroup * group)
2513 {
2514 NMFolder *folder = NULL;
2515 NMContact *contact;
2516 NMUser *user;
2517 NMERR_T rc = NM_OK;
2518 const char *alias, *gname;
2519
2520 if (gc == NULL || buddy == NULL || group == NULL)
2521 return;
2522
2523 user = (NMUser *) gc->proto_data;
2524 if (user == NULL)
2525 return;
2526
2527 /* If we haven't synched the contact list yet, ignore
2528 * the add_buddy calls. Server side list is the master.
2529 */
2530 if (!user->clist_synched)
2531 return;
2532
2533 contact = nm_create_contact();
2534 nm_contact_set_dn(contact, buddy->name);
2535
2536 /* Remove the GaimBuddy (we will add it back after adding it
2537 * to the server side list). Save the alias if there is one.
2538 */
2539 alias = gaim_buddy_get_alias(buddy);
2540 if (alias && strcmp(alias, buddy->name))
2541 nm_contact_set_display_name(contact, alias);
2542
2543 gaim_blist_remove_buddy(buddy);
2544 buddy = NULL;
2545
2546 if (strcmp(group->name, NM_ROOT_FOLDER_NAME) == 0) {
2547 gname = "";
2548 } else {
2549 gname = group->name;
2550 }
2551
2552 folder = nm_find_folder(user, gname);
2553 if (folder) {
2554
2555 /* We have everything that we need, so send the createcontact */
2556 rc = nm_send_create_contact(user, folder, contact,
2557 _create_contact_resp_cb, contact);
2558
2559 } else {
2560
2561 /* Need to create the folder before we can add the contact */
2562 rc = nm_send_create_folder(user, gname,
2563 _create_folder_resp_add_contact, contact);
2564 }
2565
2566 _check_for_disconnect(user, rc);
2567
2568 }
2569
2570 static void
2571 novell_remove_buddy(GaimConnection *gc, GaimBuddy *buddy, GaimGroup *group)
2572 {
2573 NMContact *contact;
2574 NMFolder *folder;
2575 NMUser *user;
2576 const char *dn, *gname;
2577 NMERR_T rc = NM_OK;
2578
2579 if (gc == NULL || buddy == NULL || group == NULL)
2580 return;
2581
2582 user = (NMUser *) gc->proto_data;
2583 if (user && (dn = nm_lookup_dn(user, buddy->name))) {
2584 if (strcmp(group->name, NM_ROOT_FOLDER_NAME) == 0) {
2585 gname = "";
2586 } else {
2587 gname = group->name;
2588 }
2589 folder = nm_find_folder(user, gname);
2590 if (folder) {
2591 contact = nm_folder_find_contact(folder, dn);
2592 if (contact) {
2593
2594 /* Remove the buddy from the contact */
2595 nm_contact_set_data(contact, NULL);
2596
2597 /* Tell the server to remove the contact */
2598 rc = nm_send_remove_contact(user, folder, contact,
2599 _remove_contact_resp_cb, NULL);
2600 _check_for_disconnect(user, rc);
2601 }
2602 }
2603 }
2604 }
2605
2606 static void
2607 novell_remove_group(GaimConnection * gc, GaimGroup *group)
2608 {
2609 NMUser *user;
2610 NMERR_T rc = NM_OK;
2611
2612 if (gc == NULL || group == NULL)
2613 return;
2614
2615 user = (NMUser *) gc->proto_data;
2616 if (user) {
2617 NMFolder *folder = nm_find_folder(user, group->name);
2618
2619 if (folder) {
2620 rc = nm_send_remove_folder(user, folder,
2621 _remove_folder_resp_cb, NULL);
2622 _check_for_disconnect(user, rc);
2623 }
2624 }
2625 }
2626
2627 static void
2628 novell_alias_buddy(GaimConnection * gc, const char *name, const char *alias)
2629 {
2630 NMContact *contact;
2631 NMUser *user;
2632 GList *contacts = NULL;
2633 GList *cnode = NULL;
2634 const char *dn = NULL, *fname = NULL;
2635 NMERR_T rc = NM_OK;
2636
2637 if (gc == NULL || name == NULL || alias == NULL)
2638 return;
2639
2640 user = (NMUser *) gc->proto_data;
2641 if (user && (dn = nm_lookup_dn(user, name))) {
2642
2643 /* Alias all of instances of the contact */
2644 contacts = nm_find_contacts(user, dn);
2645 for (cnode = contacts; cnode != NULL; cnode = cnode->next) {
2646 contact = (NMContact *) cnode->data;
2647 if (contact) {
2648 GaimGroup *group = NULL;
2649 GaimBuddy *buddy;
2650 NMFolder *folder;
2651
2652 /* Alias the Gaim buddy? */
2653 folder = nm_find_folder_by_id(user,
2654 nm_contact_get_parent_id(contact));
2655 if (folder) {
2656 fname = nm_folder_get_name(folder);
2657 if (*fname == '\0') {
2658 fname = NM_ROOT_FOLDER_NAME;
2659 }
2660 group = gaim_find_group(fname);
2661 }
2662
2663 if (group) {
2664 buddy = gaim_find_buddy_in_group(user->client_data,
2665 name, group);
2666 if (buddy && strcmp(buddy->alias, alias))
2667 gaim_blist_alias_buddy(buddy, alias);
2668 }
2669
2670 /* Tell the server to alias the contact */
2671 rc = nm_send_rename_contact(user, contact, alias,
2672 _rename_contact_resp_cb, NULL);
2673 _check_for_disconnect(user, rc);
2674 }
2675 }
2676 if (contacts)
2677 g_list_free(contacts);
2678 }
2679 }
2680
2681 static void
2682 novell_group_buddy(GaimConnection * gc,
2683 const char *name, const char *old_group_name,
2684 const char *new_group_name)
2685 {
2686 NMFolder *old_folder;
2687 NMFolder *new_folder;
2688 NMContact *contact;
2689 NMUser *user;
2690 const char *dn;
2691 NMERR_T rc = NM_OK;
2692
2693 if (gc == NULL || name == NULL ||
2694 old_group_name == NULL || new_group_name == NULL)
2695 return;
2696
2697 user = (NMUser *) gc->proto_data;
2698 if (user && (dn = nm_lookup_dn(user, name))) {
2699
2700 /* Find the old folder */
2701 if (strcmp(old_group_name, NM_ROOT_FOLDER_NAME) == 0) {
2702 old_folder = nm_get_root_folder(user);
2703 if (nm_folder_find_contact(old_folder, dn) == NULL)
2704 old_folder = nm_find_folder(user, old_group_name);
2705 } else {
2706 old_folder = nm_find_folder(user, old_group_name);
2707 }
2708
2709 if (old_folder && (contact = nm_folder_find_contact(old_folder, dn))) {
2710
2711 /* Find the new folder */
2712 new_folder = nm_find_folder(user, new_group_name);
2713 if (new_folder == NULL) {
2714 if (strcmp(new_group_name, NM_ROOT_FOLDER_NAME) == 0)
2715 new_folder = nm_get_root_folder(user);
2716 }
2717
2718 if (new_folder) {
2719
2720 /* Tell the server to move the contact to the new folder */
2721 rc = nm_send_move_contact(user, contact, new_folder,
2722 _move_contact_resp_cb, NULL);
2723
2724 } else {
2725
2726 nm_contact_add_ref(contact);
2727
2728 /* Remove the old contact first */
2729 nm_send_remove_contact(user, old_folder, contact,
2730 _remove_contact_resp_cb, NULL);
2731
2732 /* New folder does not exist yet, so create it */
2733 rc = nm_send_create_folder(user, new_group_name,
2734 _create_folder_resp_move_contact,
2735 contact);
2736 }
2737
2738 _check_for_disconnect(user, rc);
2739 }
2740 }
2741 }
2742
2743 static void
2744 novell_rename_group(GaimConnection * gc, const char *old_name,
2745 GaimGroup *group, GList *moved_buddies)
2746 {
2747 NMERR_T rc = NM_OK;
2748 NMFolder *folder;
2749 NMUser *user;
2750
2751 if (gc == NULL || old_name == NULL || group == NULL || moved_buddies == NULL) {
2752 return;
2753 }
2754
2755 user = gc->proto_data;
2756 if (user) {
2757 /* Does new folder exist already? */
2758 if (nm_find_folder(user, group->name)) {
2759 /* gaim_blist_rename_group() adds the buddies
2760 * to the new group and removes the old group...
2761 * so there is nothing more to do here.
2762 */
2763 return;
2764 }
2765
2766 if (strcmp(old_name, NM_ROOT_FOLDER_NAME) == 0) {
2767 /* Can't rename the root folder ... need to revisit this */
2768 return;
2769 }
2770
2771 folder = nm_find_folder(user, old_name);
2772 if (folder) {
2773 rc = nm_send_rename_folder(user, folder, group->name,
2774 _rename_folder_resp_cb, NULL);
2775 _check_for_disconnect(user, rc);
2776 }
2777 }
2778 }
2779
2780 static void
2781 novell_list_emblems(GaimBuddy * buddy, const char **se, const char **sw, const char **nw, const char **ne)
2782 {
2783 NMUserRecord *user_record = NULL;
2784 GaimConnection *gc;
2785 NMUser *user;
2786 int status = 0;
2787
2788 gc = gaim_account_get_connection(buddy->account);
2789
2790 if (gc == NULL || (user = gc->proto_data) == NULL)
2791 return;
2792
2793 user_record = nm_find_user_record(user, buddy->name);
2794
2795 if (user_record)
2796 status = nm_user_record_get_status(user_record);
2797
2798 switch (status) {
2799 case NM_STATUS_AVAILABLE:
2800 *se = "";
2801 break;
2802 case NM_STATUS_AWAY:
2803 *se = "away";
2804 break;
2805 case NM_STATUS_BUSY:
2806 *se = "occupied";
2807 break;
2808 case NM_STATUS_UNKNOWN:
2809 *se = "error";
2810 break;
2811 }
2812 }
2813
2814 static const char *
2815 novell_list_icon(GaimAccount * account, GaimBuddy * buddy)
2816 {
2817 return "novell";
2818 }
2819
2820 static void
2821 novell_tooltip_text(GaimBuddy * buddy, GString * str, gboolean full)
2822 {
2823 NMUserRecord *user_record = NULL;
2824 GaimConnection *gc;
2825 NMUser *user;
2826 int status = 0;
2827 const char *status_str = NULL;
2828 const char *text = NULL;
2829
2830 if (buddy == NULL)
2831 return;
2832
2833 gc = gaim_account_get_connection(buddy->account);
2834 if (gc == NULL || (user = gc->proto_data) == NULL)
2835 return;
2836
2837 if (GAIM_BUDDY_IS_ONLINE(buddy)) {
2838 user_record = nm_find_user_record(user, buddy->name);
2839 if (user_record) {
2840 status = nm_user_record_get_status(user_record);
2841 text = nm_user_record_get_status_text(user_record);
2842 /* No custom text, so default it ... */
2843 switch (status) {
2844 case NM_STATUS_AVAILABLE:
2845 status_str = _("Available");
2846 break;
2847 case NM_STATUS_AWAY:
2848 status_str = _("Away");
2849 break;
2850 case NM_STATUS_BUSY:
2851 status_str = _("Busy");
2852 break;
2853 case NM_STATUS_AWAY_IDLE:
2854 status_str = _("Idle");
2855 break;
2856 case NM_STATUS_OFFLINE:
2857 status_str = _("Offline");
2858 break;
2859 default:
2860 status_str = _("Unknown");
2861 break;
2862 }
2863
2864 if (text)
2865 g_string_append_printf(str, "\n<b>%s:</b> %s"
2866 "\n<b>%s:</b> %s",
2867 _("Status"), status_str,
2868 _("Message"), text);
2869 else
2870 g_string_append_printf(str, "\n<b>%s:</b> %s",
2871 _("Status"), status_str);
2872 }
2873 }
2874 }
2875
2876 static void
2877 novell_set_idle(GaimConnection * gc, int time)
2878 {
2879 NMUser *user;
2880 NMERR_T rc = NM_OK;
2881 const char *id = NULL;
2882 GaimStatus *status = NULL;
2883
2884 if (gc == NULL)
2885 return;
2886
2887 user = gc->proto_data;
2888 if (user == NULL)
2889 return;
2890
2891 status = gaim_account_get_active_status(gaim_connection_get_account(gc));
2892 id = gaim_status_get_id(status);
2893
2894 /* Only go idle if active status is available */
2895 if (!strcmp(id, NOVELL_STATUS_TYPE_AVAILABLE)) {
2896 if (time > 0) {
2897 rc = nm_send_set_status(user, NM_STATUS_AWAY_IDLE, NULL, NULL, NULL, NULL);
2898 } else {
2899 rc = nm_send_set_status(user, NM_STATUS_AVAILABLE, NULL, NULL, NULL, NULL);
2900 }
2901 }
2902
2903 _check_for_disconnect(user, rc);
2904 }
2905
2906 static void
2907 novell_get_info(GaimConnection * gc, const char *name)
2908 {
2909 NMUserRecord *user_record;
2910 NMUser *user;
2911 NMERR_T rc;
2912
2913 if (gc == NULL || name == NULL)
2914 return;
2915
2916 user = (NMUser *) gc->proto_data;
2917 if (user) {
2918
2919 user_record = nm_find_user_record(user, name);
2920 if (user_record) {
2921
2922 _show_info(gc, user_record);
2923
2924 } else {
2925
2926 rc = nm_send_get_details(user, name,
2927 _get_details_resp_show_info, g_strdup(name));
2928
2929 _check_for_disconnect(user, rc);
2930
2931 }
2932
2933 }
2934 }
2935
2936 static char *
2937 novell_status_text(GaimBuddy * buddy)
2938 {
2939 const char *text = NULL;
2940 const char *dn = NULL;
2941
2942 if (buddy && buddy->account) {
2943 GaimConnection *gc = gaim_account_get_connection(buddy->account);
2944
2945 if (gc && gc->proto_data) {
2946 NMUser *user = gc->proto_data;
2947
2948 dn = nm_lookup_dn(user, buddy->name);
2949 if (dn) {
2950 NMUserRecord *user_record = nm_find_user_record(user, dn);
2951
2952 if (user_record) {
2953 text = nm_user_record_get_status_text(user_record);
2954 if (text)
2955 return g_strdup(text);
2956 }
2957 }
2958 }
2959 }
2960
2961 return NULL;
2962 }
2963
2964 static GList *
2965 novell_status_types(GaimAccount *account)
2966 {
2967 GList *status_types = NULL;
2968 GaimStatusType *type;
2969
2970 g_return_val_if_fail(account != NULL, NULL);
2971
2972 type = gaim_status_type_new_with_attrs(GAIM_STATUS_AVAILABLE, NOVELL_STATUS_TYPE_AVAILABLE,
2973 NULL, TRUE, TRUE, FALSE,
2974 "message", _("Message"), gaim_value_new(GAIM_TYPE_STRING),
2975 NULL);
2976 status_types = g_list_append(status_types, type);
2977
2978 type = gaim_status_type_new_with_attrs(GAIM_STATUS_AWAY, NOVELL_STATUS_TYPE_AWAY,
2979 NULL, TRUE, TRUE, FALSE,
2980 "message", _("Message"), gaim_value_new(GAIM_TYPE_STRING),
2981 NULL);
2982 status_types = g_list_append(status_types, type);
2983
2984 type = gaim_status_type_new_with_attrs(GAIM_STATUS_UNAVAILABLE, NOVELL_STATUS_TYPE_BUSY,
2985 _("Busy"), TRUE, TRUE, FALSE,
2986 "message", _("Message"), gaim_value_new(GAIM_TYPE_STRING),
2987 NULL);
2988 status_types = g_list_append(status_types, type);
2989
2990 type = gaim_status_type_new_full(GAIM_STATUS_INVISIBLE, NOVELL_STATUS_TYPE_APPEAR_OFFLINE,
2991 NULL, TRUE, TRUE, FALSE);
2992 status_types = g_list_append(status_types, type);
2993
2994 type = gaim_status_type_new_full(GAIM_STATUS_OFFLINE, NULL, NULL, FALSE, TRUE, FALSE);
2995 status_types = g_list_append(status_types, type);
2996
2997 return status_types;
2998 }
2999
3000 static void
3001 novell_set_status(GaimAccount *account, GaimStatus *status)
3002 {
3003 GaimConnection *gc;
3004 gboolean connected;
3005 GaimPresence *presence;
3006 GaimStatusType *type;
3007 GaimStatusPrimitive primitive;
3008 NMUser *user;
3009 NMSTATUS_T novellstatus = NM_STATUS_AVAILABLE;
3010 NMERR_T rc = NM_OK;
3011 const char *msg = NULL;
3012 char *text = NULL;
3013
3014 connected = gaim_account_is_connected(account);
3015 presence = gaim_status_get_presence(status);
3016 type = gaim_status_get_type(status);
3017 primitive = gaim_status_type_get_primitive(type);
3018
3019 /*
3020 * We don't have any independent statuses, so we don't need to
3021 * do anything when a status is deactivated (because another
3022 * status is about to be activated).
3023 */
3024 if (!gaim_status_is_active(status))
3025 return;
3026
3027 if (!connected)
3028 return;
3029
3030 gc = gaim_account_get_connection(account);
3031 user = gc->proto_data;
3032 if (user == NULL)
3033 return;
3034
3035 if (primitive == GAIM_STATUS_AVAILABLE) {
3036 novellstatus = NM_STATUS_AVAILABLE;
3037 } else if (primitive == GAIM_STATUS_AWAY) {
3038 novellstatus = NM_STATUS_AWAY;
3039 } else if (primitive == GAIM_STATUS_UNAVAILABLE) {
3040 novellstatus = NM_STATUS_BUSY;
3041 } else if (primitive == GAIM_STATUS_INVISIBLE) {
3042 novellstatus = NM_STATUS_OFFLINE;
3043 } else if (gaim_presence_is_idle(presence)) {
3044 novellstatus = NM_STATUS_AWAY_IDLE;
3045 } else {
3046 novellstatus = NM_STATUS_AVAILABLE;
3047 }
3048
3049 if (primitive == GAIM_STATUS_AWAY || primitive == GAIM_STATUS_AVAILABLE ||
3050 primitive == GAIM_STATUS_UNAVAILABLE) {
3051 msg = gaim_status_get_attr_string(status, "message");
3052 text = g_strdup(msg);
3053
3054 if (primitive == GAIM_STATUS_AVAILABLE)
3055 msg = NULL; /* no auto replies for online status */
3056
3057 /* Don't want newlines in status text */
3058 gaim_util_chrreplace(text, '\n', ' ');
3059 }
3060
3061 rc = nm_send_set_status(user, novellstatus, text, msg, NULL, NULL);
3062 _check_for_disconnect(user, rc);
3063
3064 if (text)
3065 g_free(text);
3066 }
3067
3068 static void
3069 novell_add_permit(GaimConnection *gc, const char *who)
3070 {
3071 NMUser *user;
3072 NMERR_T rc = NM_OK;
3073 const char *name = who;
3074
3075 if (gc == NULL || who == NULL)
3076 return;
3077
3078 user = gc->proto_data;
3079 if (user == NULL)
3080 return;
3081
3082 /* Remove first -- we will add it back in when we get
3083 * the okay from the server
3084 */
3085 gaim_privacy_permit_remove(gc->account, who, TRUE);
3086
3087 if (nm_user_is_privacy_locked(user)) {
3088 _show_privacy_locked_error(gc, user);
3089 _sync_privacy_lists(user);
3090 return;
3091 }
3092
3093 /* Work around for problem with un-typed, dotted contexts */
3094 if (strchr(who, '.')) {
3095 const char *dn = nm_lookup_dn(user, who);
3096 if (dn == NULL) {
3097 rc = nm_send_get_details(user, who, _get_details_send_privacy_create,
3098 (gpointer)TRUE);
3099 _check_for_disconnect(user, rc);
3100 return;
3101 } else {
3102 name = dn;
3103 }
3104 }
3105
3106 rc = nm_send_create_privacy_item(user, name, TRUE,
3107 _create_privacy_item_permit_resp_cb,
3108 g_strdup(who));
3109 _check_for_disconnect(user, rc);
3110 }
3111
3112 static void
3113 novell_add_deny(GaimConnection *gc, const char *who)
3114 {
3115 NMUser *user;
3116 NMERR_T rc = NM_OK;
3117 const char *name = who;
3118
3119 if (gc == NULL || who == NULL)
3120 return;
3121
3122 user = gc->proto_data;
3123 if (user == NULL)
3124 return;
3125
3126 /* Remove first -- we will add it back in when we get
3127 * the okay from the server
3128 */
3129 gaim_privacy_deny_remove(gc->account, who, TRUE);
3130
3131 if (nm_user_is_privacy_locked(user)) {
3132 _show_privacy_locked_error(gc, user);
3133 _sync_privacy_lists(user);
3134 return;
3135 }
3136
3137 /* Work around for problem with un-typed, dotted contexts */
3138 if (strchr(who, '.')) {
3139 const char *dn = nm_lookup_dn(user, who);
3140 if (dn == NULL) {
3141 rc = nm_send_get_details(user, who, _get_details_send_privacy_create,
3142 (gpointer)FALSE);
3143 _check_for_disconnect(user, rc);
3144 return;
3145 } else {
3146 name = dn;
3147 }
3148 }
3149
3150 rc = nm_send_create_privacy_item(user, name, FALSE,
3151 _create_privacy_item_deny_resp_cb,
3152 g_strdup(who));
3153 _check_for_disconnect(user, rc);
3154 }
3155
3156 static void
3157 novell_rem_permit(GaimConnection *gc, const char *who)
3158 {
3159 NMUser *user;
3160 NMERR_T rc = NM_OK;
3161 const char *dn = NULL;
3162
3163 if (gc == NULL || who == NULL)
3164 return;
3165
3166 user = gc->proto_data;
3167 if (user == NULL)
3168 return;
3169
3170 if (nm_user_is_privacy_locked(user)) {
3171 _show_privacy_locked_error(gc, user);
3172 _sync_privacy_lists(user);
3173 return;
3174 }
3175
3176 dn = nm_lookup_dn(user, who);
3177 if (dn == NULL)
3178 dn = who;
3179
3180 rc = nm_send_remove_privacy_item(user, dn, TRUE,
3181 _remove_privacy_item_resp_cb,
3182 g_strdup(who));
3183 _check_for_disconnect(user, rc);
3184 }
3185
3186 static void
3187 novell_rem_deny(GaimConnection *gc, const char *who)
3188 {
3189 NMUser *user;
3190 NMERR_T rc = NM_OK;
3191 const char *dn = NULL;
3192
3193 if (gc == NULL || who == NULL)
3194 return;
3195
3196 user = gc->proto_data;
3197 if (user == NULL)
3198 return;
3199
3200 if (nm_user_is_privacy_locked(user)) {
3201 _show_privacy_locked_error(gc, user);
3202 _sync_privacy_lists(user);
3203 return;
3204 }
3205
3206 dn = nm_lookup_dn(user, who);
3207 if (dn == NULL)
3208 dn = who;
3209
3210 rc = nm_send_remove_privacy_item(user, dn, FALSE,
3211 _remove_privacy_item_resp_cb,
3212 g_strdup(who));
3213 _check_for_disconnect(user, rc);
3214 }
3215
3216 static void
3217 novell_set_permit_deny(GaimConnection *gc)
3218 {
3219 NMERR_T rc = NM_OK;
3220 const char *dn, *name = NULL;
3221 NMUserRecord *user_record = NULL;
3222 GSList *node = NULL, *copy = NULL;
3223 NMUser *user;
3224 int i, j, num_contacts, num_folders;
3225 NMContact *contact;
3226 NMFolder *folder = NULL;
3227
3228 if (gc == NULL)
3229 return;
3230
3231 user = gc->proto_data;
3232 if (user == NULL)
3233 return;
3234
3235 if (user->privacy_synched == FALSE) {
3236 _sync_privacy_lists(user);
3237 user->privacy_synched = TRUE;
3238 return;
3239 }
3240
3241 if (nm_user_is_privacy_locked(user)) {
3242 _show_privacy_locked_error(gc, user);
3243 _sync_privacy_lists(user);
3244 return;
3245 }
3246
3247 switch (gc->account->perm_deny) {
3248
3249 case GAIM_PRIVACY_ALLOW_ALL:
3250 rc = nm_send_set_privacy_default(user, FALSE,
3251 _set_privacy_default_resp_cb, NULL);
3252 _check_for_disconnect(user, rc);
3253
3254 /* clear server side deny list */
3255 if (rc == NM_OK) {
3256 copy = g_slist_copy(user->deny_list);
3257 for (node = copy; node && node->data; node = node->next) {
3258 rc = nm_send_remove_privacy_item(user, (const char *)node->data,
3259 FALSE, NULL, NULL);
3260 if (_check_for_disconnect(user, rc))
3261 break;
3262 }
3263 g_slist_free(copy);
3264 g_slist_free(user->deny_list);
3265 user->deny_list = NULL;
3266 }
3267 break;
3268
3269 case GAIM_PRIVACY_DENY_ALL:
3270 rc = nm_send_set_privacy_default(user, TRUE,
3271 _set_privacy_default_resp_cb, NULL);
3272 _check_for_disconnect(user, rc);
3273
3274 /* clear server side allow list */
3275 if (rc == NM_OK) {
3276 copy = g_slist_copy(user->allow_list);
3277 for (node = copy; node && node->data; node = node->next) {
3278 rc = nm_send_remove_privacy_item(user, (const char *)node->data,
3279 TRUE, NULL, NULL);
3280 if (_check_for_disconnect(user, rc))
3281 break;
3282 }
3283 g_slist_free(copy);
3284 g_slist_free(user->allow_list);
3285 user->allow_list = NULL;
3286 }
3287 break;
3288
3289 case GAIM_PRIVACY_ALLOW_USERS:
3290
3291 rc = nm_send_set_privacy_default(user, TRUE,
3292 _set_privacy_default_resp_cb, NULL);
3293 _check_for_disconnect(user, rc);
3294
3295 /* sync allow lists */
3296 if (rc == NM_OK) {
3297
3298 for (node = user->allow_list; node; node = node->next) {
3299 user_record = nm_find_user_record(user, (char *)node->data);
3300 if (user_record) {
3301 name = nm_user_record_get_display_id(user_record);
3302
3303 if (!g_slist_find_custom(gc->account->permit,
3304 name, (GCompareFunc)nm_utf8_strcasecmp)) {
3305 gaim_privacy_permit_add(gc->account, name , TRUE);
3306 }
3307 }
3308 }
3309
3310 for (node = gc->account->permit; node; node = node->next) {
3311 name = NULL;
3312 dn = nm_lookup_dn(user, (char *)node->data);
3313 if (dn) {
3314 user_record = nm_find_user_record(user, dn);
3315 name = nm_user_record_get_display_id(user_record);
3316
3317 if (!g_slist_find_custom(user->allow_list,
3318 dn, (GCompareFunc)nm_utf8_strcasecmp)) {
3319 rc = nm_send_create_privacy_item(user, dn, TRUE,
3320 _create_privacy_item_deny_resp_cb,
3321 g_strdup(dn));
3322 }
3323 } else {
3324 gaim_privacy_permit_remove(gc->account, (char *)node->data, TRUE);
3325 }
3326 }
3327 }
3328 break;
3329
3330 case GAIM_PRIVACY_DENY_USERS:
3331
3332 /* set to default allow */
3333 rc = nm_send_set_privacy_default(user, FALSE,
3334 _set_privacy_default_resp_cb, NULL);
3335 _check_for_disconnect(user, rc);
3336
3337 /* sync deny lists */
3338 if (rc == NM_OK) {
3339
3340 for (node = user->deny_list; node; node = node->next) {
3341 user_record = nm_find_user_record(user, (char *)node->data);
3342 if (user_record) {
3343 name = nm_user_record_get_display_id(user_record);
3344
3345 if (!g_slist_find_custom(gc->account->deny,
3346 name, (GCompareFunc)nm_utf8_strcasecmp)) {
3347 gaim_privacy_deny_add(gc->account, name , TRUE);
3348 }
3349 }
3350 }
3351
3352 for (node = gc->account->deny; node; node = node->next) {
3353
3354 name = NULL;
3355 dn = nm_lookup_dn(user, (char *)node->data);
3356 if (dn) {
3357 user_record = nm_find_user_record(user, dn);
3358 name = nm_user_record_get_display_id(user_record);
3359
3360 if (!g_slist_find_custom(user->deny_list,
3361 dn, (GCompareFunc)nm_utf8_strcasecmp)) {
3362 rc = nm_send_create_privacy_item(user, dn, FALSE,
3363 _create_privacy_item_deny_resp_cb,
3364 g_strdup(name));
3365 }
3366 } else {
3367 gaim_privacy_deny_remove(gc->account, (char *)node->data, TRUE);
3368 }
3369 }
3370
3371 }
3372 break;
3373
3374 case GAIM_PRIVACY_ALLOW_BUDDYLIST:
3375
3376 /* remove users from allow list that are not in buddy list */
3377 copy = g_slist_copy(user->allow_list);
3378 for (node = copy; node && node->data; node = node->next) {
3379 if (!nm_find_contacts(user, node->data)) {
3380 rc = nm_send_remove_privacy_item(user, (const char *)node->data,
3381 TRUE, NULL, NULL);
3382 if (_check_for_disconnect(user, rc))
3383 return;
3384 }
3385 }
3386 g_slist_free(copy);
3387
3388 /* add all buddies to allow list */
3389 num_contacts = nm_folder_get_contact_count(user->root_folder);
3390 for (i = 0; i < num_contacts; i++) {
3391 contact = nm_folder_get_contact(user->root_folder, i);
3392 dn = nm_contact_get_dn(contact);
3393 if (dn && !g_slist_find_custom(user->allow_list,
3394 dn, (GCompareFunc)nm_utf8_strcasecmp))
3395 {
3396 rc = nm_send_create_privacy_item(user, dn, TRUE,
3397 _create_privacy_item_deny_resp_cb,
3398 g_strdup(dn));
3399 if (_check_for_disconnect(user, rc))
3400 return;
3401 }
3402
3403 }
3404
3405 num_folders = nm_folder_get_subfolder_count(user->root_folder);
3406 for (i = 0; i < num_folders; i++) {
3407 folder = nm_folder_get_subfolder(user->root_folder, i);
3408 num_contacts = nm_folder_get_contact_count(folder);
3409 for (j = 0; j < num_contacts; j++) {
3410 contact = nm_folder_get_contact(folder, j);
3411 dn = nm_contact_get_dn(contact);
3412 if (dn && !g_slist_find_custom(user->allow_list,
3413 dn, (GCompareFunc)nm_utf8_strcasecmp))
3414 {
3415 rc = nm_send_create_privacy_item(user, dn, TRUE,
3416 _create_privacy_item_deny_resp_cb,
3417 g_strdup(dn));
3418 if (_check_for_disconnect(user, rc))
3419 return;
3420 }
3421 }
3422 }
3423
3424 /* set to default deny */
3425 rc = nm_send_set_privacy_default(user, TRUE,
3426 _set_privacy_default_resp_cb, NULL);
3427 if (_check_for_disconnect(user, rc))
3428 break;
3429
3430 break;
3431 }
3432 }
3433
3434 static GList *
3435 novell_blist_node_menu(GaimBlistNode *node)
3436 {
3437 GList *list = NULL;
3438 GaimMenuAction *act;
3439
3440 if(GAIM_BLIST_NODE_IS_BUDDY(node)) {
3441 act = gaim_menu_action_new(_("Initiate _Chat"),
3442 GAIM_CALLBACK(_initiate_conference_cb),
3443 NULL, NULL);
3444 list = g_list_append(list, act);
3445 }
3446
3447 return list;
3448 }
3449
3450 static void
3451 novell_keepalive(GaimConnection *gc)
3452 {
3453 NMUser *user;
3454 NMERR_T rc = NM_OK;
3455
3456 if (gc == NULL)
3457 return;
3458
3459 user = gc->proto_data;
3460 if (user == NULL)
3461 return;
3462
3463 rc = nm_send_keepalive(user, NULL, NULL);
3464 _check_for_disconnect(user, rc);
3465 }
3466
3467 static GaimPluginProtocolInfo prpl_info = {
3468 0,
3469 NULL, /* user_splits */
3470 NULL, /* protocol_options */
3471 NO_BUDDY_ICONS, /* icon_spec */
3472 novell_list_icon, /* list_icon */
3473 novell_list_emblems, /* list_emblems */
3474 novell_status_text, /* status_text */
3475 novell_tooltip_text, /* tooltip_text */
3476 novell_status_types, /* status_types */
3477 novell_blist_node_menu, /* blist_node_menu */
3478 NULL, /* chat_info */
3479 NULL, /* chat_info_defaults */
3480 novell_login, /* login */
3481 novell_close, /* close */
3482 novell_send_im, /* send_im */
3483 NULL, /* set_info */
3484 novell_send_typing, /* send_typing */
3485 novell_get_info, /* get_info */
3486 novell_set_status, /* set_status */
3487 novell_set_idle, /* set_idle */
3488 NULL, /* change_passwd */
3489 novell_add_buddy, /* add_buddy */
3490 NULL, /* add_buddies */
3491 novell_remove_buddy, /* remove_buddy */
3492 NULL, /* remove_buddies */
3493 novell_add_permit, /* add_permit */
3494 novell_add_deny, /* add_deny */
3495 novell_rem_permit, /* rem_permit */
3496 novell_rem_deny, /* rem_deny */
3497 novell_set_permit_deny, /* set_permit_deny */
3498 NULL, /* join_chat */
3499 NULL, /* reject_chat */
3500 NULL, /* get_chat_name */
3501 novell_chat_invite, /* chat_invite */
3502 novell_chat_leave, /* chat_leave */
3503 NULL, /* chat_whisper */
3504 novell_chat_send, /* chat_send */
3505 novell_keepalive, /* keepalive */
3506 NULL, /* register_user */
3507 NULL, /* get_cb_info */
3508 NULL, /* get_cb_away */
3509 novell_alias_buddy, /* alias_buddy */
3510 novell_group_buddy, /* group_buddy */
3511 novell_rename_group, /* rename_group */
3512 NULL, /* buddy_free */
3513 novell_convo_closed, /* convo_closed */
3514 gaim_normalize_nocase, /* normalize */
3515 NULL, /* set_buddy_icon */
3516 novell_remove_group, /* remove_group */
3517 NULL, /* get_cb_real_name */
3518 NULL, /* set_chat_topic */
3519 NULL, /* find_blist_chat */
3520 NULL, /* roomlist_get_list */
3521 NULL, /* roomlist_cancel */
3522 NULL, /* roomlist_expand_category */
3523 NULL, /* can_receive_file */
3524 NULL, /* send_file */
3525 NULL, /* new_xfer */
3526 NULL, /* offline_message */
3527 NULL, /* whiteboard_prpl_ops */
3528 NULL, /* media_prpl_ops */
3529 };
3530
3531 static GaimPluginInfo info = {
3532 GAIM_PLUGIN_MAGIC,
3533 GAIM_MAJOR_VERSION,
3534 GAIM_MINOR_VERSION,
3535 GAIM_PLUGIN_PROTOCOL, /**< type */
3536 NULL, /**< ui_requirement */
3537 0, /**< flags */
3538 NULL, /**< dependencies */
3539 GAIM_PRIORITY_DEFAULT, /**< priority */
3540 "prpl-novell", /**< id */
3541 "GroupWise", /**< name */
3542 VERSION, /**< version */
3543 /** summary */
3544 N_("Novell GroupWise Messenger Protocol Plugin"),
3545 /** description */
3546 N_("Novell GroupWise Messenger Protocol Plugin"),
3547 NULL, /**< author */
3548 GAIM_WEBSITE, /**< homepage */
3549
3550 NULL, /**< load */
3551 NULL, /**< unload */
3552 NULL, /**< destroy */
3553
3554 NULL, /**< ui_info */
3555 &prpl_info, /**< extra_info */
3556 NULL,
3557 NULL
3558 };
3559
3560 static void
3561 init_plugin(GaimPlugin * plugin)
3562 {
3563 GaimAccountOption *option;
3564
3565 option = gaim_account_option_string_new(_("Server address"), "server", NULL);
3566 prpl_info.protocol_options =
3567 g_list_append(prpl_info.protocol_options, option);
3568
3569 option = gaim_account_option_int_new(_("Server port"), "port", DEFAULT_PORT);
3570 prpl_info.protocol_options =
3571 g_list_append(prpl_info.protocol_options, option);
3572
3573 my_protocol = plugin;
3574 }
3575
3576 GAIM_INIT_PLUGIN(novell, init_plugin, info);

mercurial