| |
1 /* |
| |
2 |
| |
3 silcgaim_chat.c |
| |
4 |
| |
5 Author: Pekka Riikonen <priikone@silcnet.org> |
| |
6 |
| |
7 Copyright (C) 2004 Pekka Riikonen |
| |
8 |
| |
9 This program is free software; you can redistribute it and/or modify |
| |
10 it under the terms of the GNU General Public License as published by |
| |
11 the Free Software Foundation; version 2 of the License. |
| |
12 |
| |
13 This program is distributed in the hope that it will be useful, |
| |
14 but WITHOUT ANY WARRANTY; without even the implied warranty of |
| |
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| |
16 GNU General Public License for more details. |
| |
17 |
| |
18 */ |
| |
19 |
| |
20 #include "silcincludes.h" |
| |
21 #include "silcclient.h" |
| |
22 #include "silcgaim.h" |
| |
23 #include "wb.h" |
| |
24 |
| |
25 /***************************** Channel Routines ******************************/ |
| |
26 |
| |
27 GList *silcgaim_chat_info(GaimConnection *gc) |
| |
28 { |
| |
29 GList *ci = NULL; |
| |
30 struct proto_chat_entry *pce; |
| |
31 |
| |
32 pce = g_new0(struct proto_chat_entry, 1); |
| |
33 pce->label = _("_Channel:"); |
| |
34 pce->identifier = "channel"; |
| |
35 pce->required = TRUE; |
| |
36 ci = g_list_append(ci, pce); |
| |
37 |
| |
38 pce = g_new0(struct proto_chat_entry, 1); |
| |
39 pce->label = _("_Passphrase:"); |
| |
40 pce->identifier = "passphrase"; |
| |
41 pce->secret = TRUE; |
| |
42 ci = g_list_append(ci, pce); |
| |
43 |
| |
44 return ci; |
| |
45 } |
| |
46 |
| |
47 GHashTable *silcgaim_chat_info_defaults(GaimConnection *gc, const char *chat_name) |
| |
48 { |
| |
49 GHashTable *defaults; |
| |
50 |
| |
51 defaults = g_hash_table_new_full(g_str_hash, g_str_equal, NULL, g_free); |
| |
52 |
| |
53 if (chat_name != NULL) |
| |
54 g_hash_table_insert(defaults, "channel", g_strdup(chat_name)); |
| |
55 |
| |
56 return defaults; |
| |
57 } |
| |
58 |
| |
59 static void |
| |
60 silcgaim_chat_getinfo(GaimConnection *gc, GHashTable *components); |
| |
61 |
| |
62 static void |
| |
63 silcgaim_chat_getinfo_res(SilcClient client, |
| |
64 SilcClientConnection conn, |
| |
65 SilcChannelEntry *channels, |
| |
66 SilcUInt32 channels_count, |
| |
67 void *context) |
| |
68 { |
| |
69 GHashTable *components = context; |
| |
70 GaimConnection *gc = client->application; |
| |
71 const char *chname; |
| |
72 char tmp[256]; |
| |
73 |
| |
74 chname = g_hash_table_lookup(components, "channel"); |
| |
75 if (!chname) |
| |
76 return; |
| |
77 |
| |
78 if (!channels) { |
| |
79 g_snprintf(tmp, sizeof(tmp), |
| |
80 _("Channel %s does not exist in the network"), chname); |
| |
81 gaim_notify_error(gc, _("Channel Information"), |
| |
82 _("Cannot get channel information"), tmp); |
| |
83 return; |
| |
84 } |
| |
85 |
| |
86 silcgaim_chat_getinfo(gc, components); |
| |
87 } |
| |
88 |
| |
89 |
| |
90 static void |
| |
91 silcgaim_chat_getinfo(GaimConnection *gc, GHashTable *components) |
| |
92 { |
| |
93 SilcGaim sg = gc->proto_data; |
| |
94 const char *chname; |
| |
95 char *buf, tmp[256], *tmp2; |
| |
96 GString *s; |
| |
97 SilcChannelEntry channel; |
| |
98 SilcHashTableList htl; |
| |
99 SilcChannelUser chu; |
| |
100 |
| |
101 if (!components) |
| |
102 return; |
| |
103 |
| |
104 chname = g_hash_table_lookup(components, "channel"); |
| |
105 if (!chname) |
| |
106 return; |
| |
107 channel = silc_client_get_channel(sg->client, sg->conn, |
| |
108 (char *)chname); |
| |
109 if (!channel) { |
| |
110 silc_client_get_channel_resolve(sg->client, sg->conn, |
| |
111 (char *)chname, |
| |
112 silcgaim_chat_getinfo_res, |
| |
113 components); |
| |
114 return; |
| |
115 } |
| |
116 |
| |
117 s = g_string_new(""); |
| |
118 tmp2 = g_markup_escape_text(channel->channel_name, -1); |
| |
119 g_string_append_printf(s, _("<b>Channel Name:</b> %s"), tmp2); |
| |
120 g_free(tmp2); |
| |
121 if (channel->user_list && silc_hash_table_count(channel->user_list)) |
| |
122 g_string_append_printf(s, _("<br><b>User Count:</b> %d"), |
| |
123 (int)silc_hash_table_count(channel->user_list)); |
| |
124 |
| |
125 silc_hash_table_list(channel->user_list, &htl); |
| |
126 while (silc_hash_table_get(&htl, NULL, (void *)&chu)) { |
| |
127 if (chu->mode & SILC_CHANNEL_UMODE_CHANFO) { |
| |
128 tmp2 = g_markup_escape_text(chu->client->nickname, -1); |
| |
129 g_string_append_printf(s, _("<br><b>Channel Founder:</b> %s"), |
| |
130 tmp2); |
| |
131 g_free(tmp2); |
| |
132 break; |
| |
133 } |
| |
134 } |
| |
135 silc_hash_table_list_reset(&htl); |
| |
136 |
| |
137 if (channel->channel_key) |
| |
138 g_string_append_printf(s, _("<br><b>Channel Cipher:</b> %s"), |
| |
139 silc_cipher_get_name(channel->channel_key)); |
| |
140 if (channel->hmac) |
| |
141 /* Definition of HMAC: http://en.wikipedia.org/wiki/HMAC */ |
| |
142 g_string_append_printf(s, _("<br><b>Channel HMAC:</b> %s"), |
| |
143 silc_hmac_get_name(channel->hmac)); |
| |
144 |
| |
145 if (channel->topic) { |
| |
146 tmp2 = g_markup_escape_text(channel->topic, -1); |
| |
147 g_string_append_printf(s, _("<br><b>Channel Topic:</b><br>%s"), tmp2); |
| |
148 g_free(tmp2); |
| |
149 } |
| |
150 |
| |
151 if (channel->mode) { |
| |
152 g_string_append_printf(s, _("<br><b>Channel Modes:</b> ")); |
| |
153 silcgaim_get_chmode_string(channel->mode, tmp, sizeof(tmp)); |
| |
154 g_string_append(s, tmp); |
| |
155 } |
| |
156 |
| |
157 if (channel->founder_key) { |
| |
158 char *fingerprint, *babbleprint; |
| |
159 unsigned char *pk; |
| |
160 SilcUInt32 pk_len; |
| |
161 pk = silc_pkcs_public_key_encode(channel->founder_key, &pk_len); |
| |
162 fingerprint = silc_hash_fingerprint(NULL, pk, pk_len); |
| |
163 babbleprint = silc_hash_babbleprint(NULL, pk, pk_len); |
| |
164 |
| |
165 g_string_append_printf(s, _("<br><b>Founder Key Fingerprint:</b><br>%s"), fingerprint); |
| |
166 g_string_append_printf(s, _("<br><b>Founder Key Babbleprint:</b><br>%s"), babbleprint); |
| |
167 |
| |
168 silc_free(fingerprint); |
| |
169 silc_free(babbleprint); |
| |
170 silc_free(pk); |
| |
171 } |
| |
172 |
| |
173 buf = g_string_free(s, FALSE); |
| |
174 gaim_notify_formatted(gc, NULL, _("Channel Information"), NULL, buf, NULL, NULL); |
| |
175 g_free(buf); |
| |
176 } |
| |
177 |
| |
178 |
| |
179 static void |
| |
180 silcgaim_chat_getinfo_menu(GaimBlistNode *node, gpointer data) |
| |
181 { |
| |
182 GaimChat *chat = (GaimChat *)node; |
| |
183 silcgaim_chat_getinfo(chat->account->gc, chat->components); |
| |
184 } |
| |
185 |
| |
186 |
| |
187 #if 0 /* XXX For now these are not implemented. We need better |
| |
188 listview dialog from Gaim for these. */ |
| |
189 /************************** Channel Invite List ******************************/ |
| |
190 |
| |
191 static void |
| |
192 silcgaim_chat_invitelist(GaimBlistNode *node, gpointer data); |
| |
193 { |
| |
194 |
| |
195 } |
| |
196 |
| |
197 |
| |
198 /**************************** Channel Ban List *******************************/ |
| |
199 |
| |
200 static void |
| |
201 silcgaim_chat_banlist(GaimBlistNode *node, gpointer data); |
| |
202 { |
| |
203 |
| |
204 } |
| |
205 #endif |
| |
206 |
| |
207 |
| |
208 /************************* Channel Authentication ****************************/ |
| |
209 |
| |
210 typedef struct { |
| |
211 SilcGaim sg; |
| |
212 SilcChannelEntry channel; |
| |
213 GaimChat *c; |
| |
214 SilcBuffer pubkeys; |
| |
215 } *SilcGaimChauth; |
| |
216 |
| |
217 static void |
| |
218 silcgaim_chat_chpk_add(void *user_data, const char *name) |
| |
219 { |
| |
220 SilcGaimChauth sgc = (SilcGaimChauth)user_data; |
| |
221 SilcGaim sg = sgc->sg; |
| |
222 SilcClient client = sg->client; |
| |
223 SilcClientConnection conn = sg->conn; |
| |
224 SilcPublicKey public_key; |
| |
225 SilcBuffer chpks, pk, chidp; |
| |
226 unsigned char mode[4]; |
| |
227 SilcUInt32 m; |
| |
228 |
| |
229 /* Load the public key */ |
| |
230 if (!silc_pkcs_load_public_key(name, &public_key, SILC_PKCS_FILE_PEM) && |
| |
231 !silc_pkcs_load_public_key(name, &public_key, SILC_PKCS_FILE_BIN)) { |
| |
232 silcgaim_chat_chauth_show(sgc->sg, sgc->channel, sgc->pubkeys); |
| |
233 silc_buffer_free(sgc->pubkeys); |
| |
234 silc_free(sgc); |
| |
235 gaim_notify_error(client->application, |
| |
236 _("Add Channel Public Key"), |
| |
237 _("Could not load public key"), NULL); |
| |
238 return; |
| |
239 } |
| |
240 |
| |
241 pk = silc_pkcs_public_key_payload_encode(public_key); |
| |
242 chpks = silc_buffer_alloc_size(2); |
| |
243 SILC_PUT16_MSB(1, chpks->head); |
| |
244 chpks = silc_argument_payload_encode_one(chpks, pk->data, |
| |
245 pk->len, 0x00); |
| |
246 silc_buffer_free(pk); |
| |
247 |
| |
248 m = sgc->channel->mode; |
| |
249 m |= SILC_CHANNEL_MODE_CHANNEL_AUTH; |
| |
250 |
| |
251 /* Send CMODE */ |
| |
252 SILC_PUT32_MSB(m, mode); |
| |
253 chidp = silc_id_payload_encode(sgc->channel->id, SILC_ID_CHANNEL); |
| |
254 silc_client_command_send(client, conn, SILC_COMMAND_CMODE, |
| |
255 ++conn->cmd_ident, 3, |
| |
256 1, chidp->data, chidp->len, |
| |
257 2, mode, sizeof(mode), |
| |
258 9, chpks->data, chpks->len); |
| |
259 silc_buffer_free(chpks); |
| |
260 silc_buffer_free(chidp); |
| |
261 silc_buffer_free(sgc->pubkeys); |
| |
262 silc_free(sgc); |
| |
263 } |
| |
264 |
| |
265 static void |
| |
266 silcgaim_chat_chpk_cancel(void *user_data, const char *name) |
| |
267 { |
| |
268 SilcGaimChauth sgc = (SilcGaimChauth)user_data; |
| |
269 silcgaim_chat_chauth_show(sgc->sg, sgc->channel, sgc->pubkeys); |
| |
270 silc_buffer_free(sgc->pubkeys); |
| |
271 silc_free(sgc); |
| |
272 } |
| |
273 |
| |
274 static void |
| |
275 silcgaim_chat_chpk_cb(SilcGaimChauth sgc, GaimRequestFields *fields) |
| |
276 { |
| |
277 SilcGaim sg = sgc->sg; |
| |
278 SilcClient client = sg->client; |
| |
279 SilcClientConnection conn = sg->conn; |
| |
280 GaimRequestField *f; |
| |
281 const GList *list; |
| |
282 SilcPublicKey public_key; |
| |
283 SilcBuffer chpks, pk, chidp; |
| |
284 SilcUInt16 c = 0, ct; |
| |
285 unsigned char mode[4]; |
| |
286 SilcUInt32 m; |
| |
287 |
| |
288 f = gaim_request_fields_get_field(fields, "list"); |
| |
289 if (!gaim_request_field_list_get_selected(f)) { |
| |
290 /* Add new public key */ |
| |
291 gaim_request_file(sg->gc, _("Open Public Key..."), NULL, FALSE, |
| |
292 G_CALLBACK(silcgaim_chat_chpk_add), |
| |
293 G_CALLBACK(silcgaim_chat_chpk_cancel), sgc); |
| |
294 return; |
| |
295 } |
| |
296 |
| |
297 list = gaim_request_field_list_get_items(f); |
| |
298 chpks = silc_buffer_alloc_size(2); |
| |
299 |
| |
300 for (ct = 0; list; list = list->next, ct++) { |
| |
301 public_key = gaim_request_field_list_get_data(f, list->data); |
| |
302 if (gaim_request_field_list_is_selected(f, list->data)) { |
| |
303 /* Delete this public key */ |
| |
304 pk = silc_pkcs_public_key_payload_encode(public_key); |
| |
305 chpks = silc_argument_payload_encode_one(chpks, pk->data, |
| |
306 pk->len, 0x01); |
| |
307 silc_buffer_free(pk); |
| |
308 c++; |
| |
309 } |
| |
310 silc_pkcs_public_key_free(public_key); |
| |
311 } |
| |
312 if (!c) { |
| |
313 silc_buffer_free(chpks); |
| |
314 return; |
| |
315 } |
| |
316 SILC_PUT16_MSB(c, chpks->head); |
| |
317 |
| |
318 m = sgc->channel->mode; |
| |
319 if (ct == c) |
| |
320 m &= ~SILC_CHANNEL_MODE_CHANNEL_AUTH; |
| |
321 |
| |
322 /* Send CMODE */ |
| |
323 SILC_PUT32_MSB(m, mode); |
| |
324 chidp = silc_id_payload_encode(sgc->channel->id, SILC_ID_CHANNEL); |
| |
325 silc_client_command_send(client, conn, SILC_COMMAND_CMODE, |
| |
326 ++conn->cmd_ident, 3, |
| |
327 1, chidp->data, chidp->len, |
| |
328 2, mode, sizeof(mode), |
| |
329 9, chpks->data, chpks->len); |
| |
330 silc_buffer_free(chpks); |
| |
331 silc_buffer_free(chidp); |
| |
332 silc_buffer_free(sgc->pubkeys); |
| |
333 silc_free(sgc); |
| |
334 } |
| |
335 |
| |
336 static void |
| |
337 silcgaim_chat_chauth_ok(SilcGaimChauth sgc, GaimRequestFields *fields) |
| |
338 { |
| |
339 SilcGaim sg = sgc->sg; |
| |
340 GaimRequestField *f; |
| |
341 const char *curpass, *val; |
| |
342 int set; |
| |
343 |
| |
344 f = gaim_request_fields_get_field(fields, "passphrase"); |
| |
345 val = gaim_request_field_string_get_value(f); |
| |
346 curpass = gaim_blist_node_get_string((GaimBlistNode *)sgc->c, "passphrase"); |
| |
347 |
| |
348 if (!val && curpass) |
| |
349 set = 0; |
| |
350 else if (val && !curpass) |
| |
351 set = 1; |
| |
352 else if (val && curpass && strcmp(val, curpass)) |
| |
353 set = 1; |
| |
354 else |
| |
355 set = -1; |
| |
356 |
| |
357 if (set == 1) { |
| |
358 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", |
| |
359 sgc->channel->channel_name, "+a", val, NULL); |
| |
360 gaim_blist_node_set_string((GaimBlistNode *)sgc->c, "passphrase", val); |
| |
361 } else if (set == 0) { |
| |
362 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", |
| |
363 sgc->channel->channel_name, "-a", NULL); |
| |
364 gaim_blist_node_remove_setting((GaimBlistNode *)sgc->c, "passphrase"); |
| |
365 } |
| |
366 |
| |
367 silc_buffer_free(sgc->pubkeys); |
| |
368 silc_free(sgc); |
| |
369 } |
| |
370 |
| |
371 void silcgaim_chat_chauth_show(SilcGaim sg, SilcChannelEntry channel, |
| |
372 SilcBuffer channel_pubkeys) |
| |
373 { |
| |
374 SilcUInt16 argc; |
| |
375 SilcArgumentPayload chpks; |
| |
376 unsigned char *pk; |
| |
377 SilcUInt32 pk_len, type; |
| |
378 char *fingerprint, *babbleprint; |
| |
379 SilcPublicKey pubkey; |
| |
380 SilcPublicKeyIdentifier ident; |
| |
381 char tmp2[1024], t[512]; |
| |
382 GaimRequestFields *fields; |
| |
383 GaimRequestFieldGroup *g; |
| |
384 GaimRequestField *f; |
| |
385 SilcGaimChauth sgc; |
| |
386 const char *curpass = NULL; |
| |
387 |
| |
388 sgc = silc_calloc(1, sizeof(*sgc)); |
| |
389 if (!sgc) |
| |
390 return; |
| |
391 sgc->sg = sg; |
| |
392 sgc->channel = channel; |
| |
393 |
| |
394 fields = gaim_request_fields_new(); |
| |
395 |
| |
396 if (sgc->c) |
| |
397 curpass = gaim_blist_node_get_string((GaimBlistNode *)sgc->c, "passphrase"); |
| |
398 |
| |
399 g = gaim_request_field_group_new(NULL); |
| |
400 f = gaim_request_field_string_new("passphrase", _("Channel Passphrase"), |
| |
401 curpass, FALSE); |
| |
402 gaim_request_field_string_set_masked(f, TRUE); |
| |
403 gaim_request_field_group_add_field(g, f); |
| |
404 gaim_request_fields_add_group(fields, g); |
| |
405 |
| |
406 g = gaim_request_field_group_new(NULL); |
| |
407 f = gaim_request_field_label_new("l1", _("Channel Public Keys List")); |
| |
408 gaim_request_field_group_add_field(g, f); |
| |
409 gaim_request_fields_add_group(fields, g); |
| |
410 |
| |
411 g_snprintf(t, sizeof(t), |
| |
412 _("Channel authentication is used to secure the channel from " |
| |
413 "unauthorized access. The authentication may be based on " |
| |
414 "passphrase and digital signatures. If passphrase is set, it " |
| |
415 "is required to be able to join. If channel public keys are set " |
| |
416 "then only users whose public keys are listed are able to join.")); |
| |
417 |
| |
418 if (!channel_pubkeys) { |
| |
419 f = gaim_request_field_list_new("list", NULL); |
| |
420 gaim_request_field_group_add_field(g, f); |
| |
421 gaim_request_fields(sg->gc, _("Channel Authentication"), |
| |
422 _("Channel Authentication"), t, fields, |
| |
423 _("Add / Remove"), G_CALLBACK(silcgaim_chat_chpk_cb), |
| |
424 _("OK"), G_CALLBACK(silcgaim_chat_chauth_ok), sgc); |
| |
425 return; |
| |
426 } |
| |
427 sgc->pubkeys = silc_buffer_copy(channel_pubkeys); |
| |
428 |
| |
429 g = gaim_request_field_group_new(NULL); |
| |
430 f = gaim_request_field_list_new("list", NULL); |
| |
431 gaim_request_field_group_add_field(g, f); |
| |
432 gaim_request_fields_add_group(fields, g); |
| |
433 |
| |
434 SILC_GET16_MSB(argc, channel_pubkeys->data); |
| |
435 chpks = silc_argument_payload_parse(channel_pubkeys->data + 2, |
| |
436 channel_pubkeys->len - 2, argc); |
| |
437 if (!chpks) |
| |
438 return; |
| |
439 |
| |
440 pk = silc_argument_get_first_arg(chpks, &type, &pk_len); |
| |
441 while (pk) { |
| |
442 fingerprint = silc_hash_fingerprint(NULL, pk + 4, pk_len - 4); |
| |
443 babbleprint = silc_hash_babbleprint(NULL, pk + 4, pk_len - 4); |
| |
444 silc_pkcs_public_key_payload_decode(pk, pk_len, &pubkey); |
| |
445 ident = silc_pkcs_decode_identifier(pubkey->identifier); |
| |
446 |
| |
447 g_snprintf(tmp2, sizeof(tmp2), "%s\n %s\n %s", |
| |
448 ident->realname ? ident->realname : ident->username ? |
| |
449 ident->username : "", fingerprint, babbleprint); |
| |
450 gaim_request_field_list_add(f, tmp2, pubkey); |
| |
451 |
| |
452 silc_free(fingerprint); |
| |
453 silc_free(babbleprint); |
| |
454 silc_pkcs_free_identifier(ident); |
| |
455 pk = silc_argument_get_next_arg(chpks, &type, &pk_len); |
| |
456 } |
| |
457 |
| |
458 gaim_request_field_list_set_multi_select(f, FALSE); |
| |
459 gaim_request_fields(sg->gc, _("Channel Authentication"), |
| |
460 _("Channel Authentication"), t, fields, |
| |
461 _("Add / Remove"), G_CALLBACK(silcgaim_chat_chpk_cb), |
| |
462 _("OK"), G_CALLBACK(silcgaim_chat_chauth_ok), sgc); |
| |
463 |
| |
464 silc_argument_payload_free(chpks); |
| |
465 } |
| |
466 |
| |
467 static void |
| |
468 silcgaim_chat_chauth(GaimBlistNode *node, gpointer data) |
| |
469 { |
| |
470 GaimChat *chat; |
| |
471 GaimConnection *gc; |
| |
472 SilcGaim sg; |
| |
473 |
| |
474 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
475 |
| |
476 chat = (GaimChat *) node; |
| |
477 gc = gaim_account_get_connection(chat->account); |
| |
478 sg = gc->proto_data; |
| |
479 |
| |
480 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", |
| |
481 g_hash_table_lookup(chat->components, "channel"), |
| |
482 "+C", NULL); |
| |
483 } |
| |
484 |
| |
485 |
| |
486 /************************** Channel Private Groups **************************/ |
| |
487 |
| |
488 /* Private groups are "virtual" channels. They are groups inside a channel. |
| |
489 This is implemented by using channel private keys. By knowing a channel |
| |
490 private key user becomes part of that group and is able to talk on that |
| |
491 group. Other users, on the same channel, won't be able to see the |
| |
492 messages of that group. It is possible to have multiple groups inside |
| |
493 a channel - and thus having multiple private keys on the channel. */ |
| |
494 |
| |
495 typedef struct { |
| |
496 SilcGaim sg; |
| |
497 GaimChat *c; |
| |
498 const char *channel; |
| |
499 } *SilcGaimCharPrv; |
| |
500 |
| |
501 static void |
| |
502 silcgaim_chat_prv_add(SilcGaimCharPrv p, GaimRequestFields *fields) |
| |
503 { |
| |
504 SilcGaim sg = p->sg; |
| |
505 char tmp[512]; |
| |
506 GaimRequestField *f; |
| |
507 const char *name, *passphrase, *alias; |
| |
508 GHashTable *comp; |
| |
509 GaimGroup *g; |
| |
510 GaimChat *cn; |
| |
511 |
| |
512 f = gaim_request_fields_get_field(fields, "name"); |
| |
513 name = gaim_request_field_string_get_value(f); |
| |
514 if (!name) { |
| |
515 silc_free(p); |
| |
516 return; |
| |
517 } |
| |
518 f = gaim_request_fields_get_field(fields, "passphrase"); |
| |
519 passphrase = gaim_request_field_string_get_value(f); |
| |
520 f = gaim_request_fields_get_field(fields, "alias"); |
| |
521 alias = gaim_request_field_string_get_value(f); |
| |
522 |
| |
523 /* Add private group to buddy list */ |
| |
524 g_snprintf(tmp, sizeof(tmp), "%s [Private Group]", name); |
| |
525 comp = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free); |
| |
526 g_hash_table_replace(comp, g_strdup("channel"), g_strdup(tmp)); |
| |
527 g_hash_table_replace(comp, g_strdup("passphrase"), g_strdup(passphrase)); |
| |
528 |
| |
529 cn = gaim_chat_new(sg->account, alias, comp); |
| |
530 g = (GaimGroup *)p->c->node.parent; |
| |
531 gaim_blist_add_chat(cn, g, (GaimBlistNode *)p->c); |
| |
532 |
| |
533 /* Associate to a real channel */ |
| |
534 gaim_blist_node_set_string((GaimBlistNode *)cn, "parentch", p->channel); |
| |
535 |
| |
536 /* Join the group */ |
| |
537 silcgaim_chat_join(sg->gc, comp); |
| |
538 |
| |
539 silc_free(p); |
| |
540 } |
| |
541 |
| |
542 static void |
| |
543 silcgaim_chat_prv_cancel(SilcGaimCharPrv p, GaimRequestFields *fields) |
| |
544 { |
| |
545 silc_free(p); |
| |
546 } |
| |
547 |
| |
548 static void |
| |
549 silcgaim_chat_prv(GaimBlistNode *node, gpointer data) |
| |
550 { |
| |
551 GaimChat *chat; |
| |
552 GaimConnection *gc; |
| |
553 SilcGaim sg; |
| |
554 |
| |
555 SilcGaimCharPrv p; |
| |
556 GaimRequestFields *fields; |
| |
557 GaimRequestFieldGroup *g; |
| |
558 GaimRequestField *f; |
| |
559 char tmp[512]; |
| |
560 |
| |
561 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
562 |
| |
563 chat = (GaimChat *) node; |
| |
564 gc = gaim_account_get_connection(chat->account); |
| |
565 sg = gc->proto_data; |
| |
566 |
| |
567 p = silc_calloc(1, sizeof(*p)); |
| |
568 if (!p) |
| |
569 return; |
| |
570 p->sg = sg; |
| |
571 |
| |
572 p->channel = g_hash_table_lookup(chat->components, "channel"); |
| |
573 p->c = gaim_blist_find_chat(sg->account, p->channel); |
| |
574 |
| |
575 fields = gaim_request_fields_new(); |
| |
576 |
| |
577 g = gaim_request_field_group_new(NULL); |
| |
578 f = gaim_request_field_string_new("name", _("Group Name"), |
| |
579 NULL, FALSE); |
| |
580 gaim_request_field_group_add_field(g, f); |
| |
581 |
| |
582 f = gaim_request_field_string_new("passphrase", _("Passphrase"), |
| |
583 NULL, FALSE); |
| |
584 gaim_request_field_string_set_masked(f, TRUE); |
| |
585 gaim_request_field_group_add_field(g, f); |
| |
586 |
| |
587 f = gaim_request_field_string_new("alias", _("Alias"), |
| |
588 NULL, FALSE); |
| |
589 gaim_request_field_group_add_field(g, f); |
| |
590 gaim_request_fields_add_group(fields, g); |
| |
591 |
| |
592 g_snprintf(tmp, sizeof(tmp), |
| |
593 _("Please enter the %s channel private group name and passphrase."), |
| |
594 p->channel); |
| |
595 gaim_request_fields(gc, _("Add Channel Private Group"), NULL, tmp, fields, |
| |
596 _("Add"), G_CALLBACK(silcgaim_chat_prv_add), |
| |
597 _("Cancel"), G_CALLBACK(silcgaim_chat_prv_cancel), p); |
| |
598 } |
| |
599 |
| |
600 |
| |
601 /****************************** Channel Modes ********************************/ |
| |
602 |
| |
603 static void |
| |
604 silcgaim_chat_permanent_reset(GaimBlistNode *node, gpointer data) |
| |
605 { |
| |
606 GaimChat *chat; |
| |
607 GaimConnection *gc; |
| |
608 SilcGaim sg; |
| |
609 |
| |
610 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
611 |
| |
612 chat = (GaimChat *) node; |
| |
613 gc = gaim_account_get_connection(chat->account); |
| |
614 sg = gc->proto_data; |
| |
615 |
| |
616 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", |
| |
617 g_hash_table_lookup(chat->components, "channel"), |
| |
618 "-f", NULL); |
| |
619 } |
| |
620 |
| |
621 static void |
| |
622 silcgaim_chat_permanent(GaimBlistNode *node, gpointer data) |
| |
623 { |
| |
624 GaimChat *chat; |
| |
625 GaimConnection *gc; |
| |
626 SilcGaim sg; |
| |
627 const char *channel; |
| |
628 |
| |
629 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
630 |
| |
631 chat = (GaimChat *) node; |
| |
632 gc = gaim_account_get_connection(chat->account); |
| |
633 sg = gc->proto_data; |
| |
634 |
| |
635 if (!sg->conn) |
| |
636 return; |
| |
637 |
| |
638 /* XXX we should have ability to define which founder |
| |
639 key to use. Now we use the user's own public key |
| |
640 (default key). */ |
| |
641 |
| |
642 /* Call CMODE */ |
| |
643 channel = g_hash_table_lookup(chat->components, "channel"); |
| |
644 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", channel, |
| |
645 "+f", NULL); |
| |
646 } |
| |
647 |
| |
648 typedef struct { |
| |
649 SilcGaim sg; |
| |
650 const char *channel; |
| |
651 } *SilcGaimChatInput; |
| |
652 |
| |
653 static void |
| |
654 silcgaim_chat_ulimit_cb(SilcGaimChatInput s, const char *limit) |
| |
655 { |
| |
656 SilcChannelEntry channel; |
| |
657 int ulimit = 0; |
| |
658 |
| |
659 channel = silc_client_get_channel(s->sg->client, s->sg->conn, |
| |
660 (char *)s->channel); |
| |
661 if (!channel) |
| |
662 return; |
| |
663 if (limit) |
| |
664 ulimit = atoi(limit); |
| |
665 |
| |
666 if (!limit || !(*limit) || *limit == '0') { |
| |
667 if (limit && ulimit == channel->user_limit) { |
| |
668 silc_free(s); |
| |
669 return; |
| |
670 } |
| |
671 silc_client_command_call(s->sg->client, s->sg->conn, NULL, "CMODE", |
| |
672 s->channel, "-l", NULL); |
| |
673 |
| |
674 silc_free(s); |
| |
675 return; |
| |
676 } |
| |
677 |
| |
678 if (ulimit == channel->user_limit) { |
| |
679 silc_free(s); |
| |
680 return; |
| |
681 } |
| |
682 |
| |
683 /* Call CMODE */ |
| |
684 silc_client_command_call(s->sg->client, s->sg->conn, NULL, "CMODE", |
| |
685 s->channel, "+l", limit, NULL); |
| |
686 |
| |
687 silc_free(s); |
| |
688 } |
| |
689 |
| |
690 static void |
| |
691 silcgaim_chat_ulimit(GaimBlistNode *node, gpointer data) |
| |
692 { |
| |
693 GaimChat *chat; |
| |
694 GaimConnection *gc; |
| |
695 SilcGaim sg; |
| |
696 |
| |
697 SilcGaimChatInput s; |
| |
698 SilcChannelEntry channel; |
| |
699 const char *ch; |
| |
700 char tmp[32]; |
| |
701 |
| |
702 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
703 |
| |
704 chat = (GaimChat *) node; |
| |
705 gc = gaim_account_get_connection(chat->account); |
| |
706 sg = gc->proto_data; |
| |
707 |
| |
708 if (!sg->conn) |
| |
709 return; |
| |
710 |
| |
711 ch = g_strdup(g_hash_table_lookup(chat->components, "channel")); |
| |
712 channel = silc_client_get_channel(sg->client, sg->conn, (char *)ch); |
| |
713 if (!channel) |
| |
714 return; |
| |
715 |
| |
716 s = silc_calloc(1, sizeof(*s)); |
| |
717 if (!s) |
| |
718 return; |
| |
719 s->channel = ch; |
| |
720 s->sg = sg; |
| |
721 g_snprintf(tmp, sizeof(tmp), "%d", (int)channel->user_limit); |
| |
722 gaim_request_input(gc, _("User Limit"), NULL, |
| |
723 _("Set user limit on channel. Set to zero to reset user limit."), |
| |
724 tmp, FALSE, FALSE, NULL, |
| |
725 _("OK"), G_CALLBACK(silcgaim_chat_ulimit_cb), |
| |
726 _("Cancel"), G_CALLBACK(silcgaim_chat_ulimit_cb), s); |
| |
727 } |
| |
728 |
| |
729 static void |
| |
730 silcgaim_chat_resettopic(GaimBlistNode *node, gpointer data) |
| |
731 { |
| |
732 GaimChat *chat; |
| |
733 GaimConnection *gc; |
| |
734 SilcGaim sg; |
| |
735 |
| |
736 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
737 |
| |
738 chat = (GaimChat *) node; |
| |
739 gc = gaim_account_get_connection(chat->account); |
| |
740 sg = gc->proto_data; |
| |
741 |
| |
742 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", |
| |
743 g_hash_table_lookup(chat->components, "channel"), |
| |
744 "-t", NULL); |
| |
745 } |
| |
746 |
| |
747 static void |
| |
748 silcgaim_chat_settopic(GaimBlistNode *node, gpointer data) |
| |
749 { |
| |
750 GaimChat *chat; |
| |
751 GaimConnection *gc; |
| |
752 SilcGaim sg; |
| |
753 |
| |
754 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
755 |
| |
756 chat = (GaimChat *) node; |
| |
757 gc = gaim_account_get_connection(chat->account); |
| |
758 sg = gc->proto_data; |
| |
759 |
| |
760 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", |
| |
761 g_hash_table_lookup(chat->components, "channel"), |
| |
762 "+t", NULL); |
| |
763 } |
| |
764 |
| |
765 static void |
| |
766 silcgaim_chat_resetprivate(GaimBlistNode *node, gpointer data) |
| |
767 { |
| |
768 GaimChat *chat; |
| |
769 GaimConnection *gc; |
| |
770 SilcGaim sg; |
| |
771 |
| |
772 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
773 |
| |
774 chat = (GaimChat *) node; |
| |
775 gc = gaim_account_get_connection(chat->account); |
| |
776 sg = gc->proto_data; |
| |
777 |
| |
778 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", |
| |
779 g_hash_table_lookup(chat->components, "channel"), |
| |
780 "-p", NULL); |
| |
781 } |
| |
782 |
| |
783 static void |
| |
784 silcgaim_chat_setprivate(GaimBlistNode *node, gpointer data) |
| |
785 { |
| |
786 GaimChat *chat; |
| |
787 GaimConnection *gc; |
| |
788 SilcGaim sg; |
| |
789 |
| |
790 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
791 |
| |
792 chat = (GaimChat *) node; |
| |
793 gc = gaim_account_get_connection(chat->account); |
| |
794 sg = gc->proto_data; |
| |
795 |
| |
796 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", |
| |
797 g_hash_table_lookup(chat->components, "channel"), |
| |
798 "+p", NULL); |
| |
799 } |
| |
800 |
| |
801 static void |
| |
802 silcgaim_chat_resetsecret(GaimBlistNode *node, gpointer data) |
| |
803 { |
| |
804 GaimChat *chat; |
| |
805 GaimConnection *gc; |
| |
806 SilcGaim sg; |
| |
807 |
| |
808 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
809 |
| |
810 chat = (GaimChat *) node; |
| |
811 gc = gaim_account_get_connection(chat->account); |
| |
812 sg = gc->proto_data; |
| |
813 |
| |
814 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", |
| |
815 g_hash_table_lookup(chat->components, "channel"), |
| |
816 "-s", NULL); |
| |
817 } |
| |
818 |
| |
819 static void |
| |
820 silcgaim_chat_setsecret(GaimBlistNode *node, gpointer data) |
| |
821 { |
| |
822 GaimChat *chat; |
| |
823 GaimConnection *gc; |
| |
824 SilcGaim sg; |
| |
825 |
| |
826 g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); |
| |
827 |
| |
828 chat = (GaimChat *) node; |
| |
829 gc = gaim_account_get_connection(chat->account); |
| |
830 sg = gc->proto_data; |
| |
831 |
| |
832 silc_client_command_call(sg->client, sg->conn, NULL, "CMODE", |
| |
833 g_hash_table_lookup(chat->components, "channel"), |
| |
834 "+s", NULL); |
| |
835 } |
| |
836 |
| |
837 typedef struct { |
| |
838 SilcGaim sg; |
| |
839 SilcChannelEntry channel; |
| |
840 } *SilcGaimChatWb; |
| |
841 |
| |
842 static void |
| |
843 silcgaim_chat_wb(GaimBlistNode *node, gpointer data) |
| |
844 { |
| |
845 SilcGaimChatWb wb = data; |
| |
846 silcgaim_wb_init_ch(wb->sg, wb->channel); |
| |
847 silc_free(wb); |
| |
848 } |
| |
849 |
| |
850 GList *silcgaim_chat_menu(GaimChat *chat) |
| |
851 { |
| |
852 GHashTable *components = chat->components; |
| |
853 GaimConnection *gc = gaim_account_get_connection(chat->account); |
| |
854 SilcGaim sg = gc->proto_data; |
| |
855 SilcClientConnection conn = sg->conn; |
| |
856 const char *chname = NULL; |
| |
857 SilcChannelEntry channel = NULL; |
| |
858 SilcChannelUser chu = NULL; |
| |
859 SilcUInt32 mode = 0; |
| |
860 |
| |
861 GList *m = NULL; |
| |
862 GaimMenuAction *act; |
| |
863 |
| |
864 if (components) |
| |
865 chname = g_hash_table_lookup(components, "channel"); |
| |
866 if (chname) |
| |
867 channel = silc_client_get_channel(sg->client, sg->conn, |
| |
868 (char *)chname); |
| |
869 if (channel) { |
| |
870 chu = silc_client_on_channel(channel, conn->local_entry); |
| |
871 if (chu) |
| |
872 mode = chu->mode; |
| |
873 } |
| |
874 |
| |
875 if (strstr(chname, "[Private Group]")) |
| |
876 return NULL; |
| |
877 |
| |
878 act = gaim_menu_action_new(_("Get Info"), |
| |
879 GAIM_CALLBACK(silcgaim_chat_getinfo_menu), |
| |
880 NULL, NULL); |
| |
881 m = g_list_append(m, act); |
| |
882 |
| |
883 #if 0 /* XXX For now these are not implemented. We need better |
| |
884 listview dialog from Gaim for these. */ |
| |
885 if (mode & SILC_CHANNEL_UMODE_CHANOP) { |
| |
886 act = gaim_menu_action_new(_("Invite List"), |
| |
887 GAIM_CALLBACK(silcgaim_chat_invitelist), |
| |
888 NULL, NULL); |
| |
889 m = g_list_append(m, act); |
| |
890 |
| |
891 act = gaim_menu_action_new(_("Ban List"), |
| |
892 GAIM_CALLBACK(silcgaim_chat_banlist), |
| |
893 NULL, NULL); |
| |
894 m = g_list_append(m, act); |
| |
895 } |
| |
896 #endif |
| |
897 |
| |
898 if (chu) { |
| |
899 act = gaim_menu_action_new(_("Add Private Group"), |
| |
900 GAIM_CALLBACK(silcgaim_chat_prv), |
| |
901 NULL, NULL); |
| |
902 m = g_list_append(m, act); |
| |
903 } |
| |
904 |
| |
905 if (mode & SILC_CHANNEL_UMODE_CHANFO) { |
| |
906 act = gaim_menu_action_new(_("Channel Authentication"), |
| |
907 GAIM_CALLBACK(silcgaim_chat_chauth), |
| |
908 NULL, NULL); |
| |
909 m = g_list_append(m, act); |
| |
910 |
| |
911 if (channel->mode & SILC_CHANNEL_MODE_FOUNDER_AUTH) { |
| |
912 act = gaim_menu_action_new(_("Reset Permanent"), |
| |
913 GAIM_CALLBACK(silcgaim_chat_permanent_reset), |
| |
914 NULL, NULL); |
| |
915 m = g_list_append(m, act); |
| |
916 } else { |
| |
917 act = gaim_menu_action_new(_("Set Permanent"), |
| |
918 GAIM_CALLBACK(silcgaim_chat_permanent), |
| |
919 NULL, NULL); |
| |
920 m = g_list_append(m, act); |
| |
921 } |
| |
922 } |
| |
923 |
| |
924 if (mode & SILC_CHANNEL_UMODE_CHANOP) { |
| |
925 act = gaim_menu_action_new(_("Set User Limit"), |
| |
926 GAIM_CALLBACK(silcgaim_chat_ulimit), |
| |
927 NULL, NULL); |
| |
928 m = g_list_append(m, act); |
| |
929 |
| |
930 if (channel->mode & SILC_CHANNEL_MODE_TOPIC) { |
| |
931 act = gaim_menu_action_new(_("Reset Topic Restriction"), |
| |
932 GAIM_CALLBACK(silcgaim_chat_resettopic), |
| |
933 NULL, NULL); |
| |
934 m = g_list_append(m, act); |
| |
935 } else { |
| |
936 act = gaim_menu_action_new(_("Set Topic Restriction"), |
| |
937 GAIM_CALLBACK(silcgaim_chat_settopic), |
| |
938 NULL, NULL); |
| |
939 m = g_list_append(m, act); |
| |
940 } |
| |
941 |
| |
942 if (channel->mode & SILC_CHANNEL_MODE_PRIVATE) { |
| |
943 act = gaim_menu_action_new(_("Reset Private Channel"), |
| |
944 GAIM_CALLBACK(silcgaim_chat_resetprivate), |
| |
945 NULL, NULL); |
| |
946 m = g_list_append(m, act); |
| |
947 } else { |
| |
948 act = gaim_menu_action_new(_("Set Private Channel"), |
| |
949 GAIM_CALLBACK(silcgaim_chat_setprivate), |
| |
950 NULL, NULL); |
| |
951 m = g_list_append(m, act); |
| |
952 } |
| |
953 |
| |
954 if (channel->mode & SILC_CHANNEL_MODE_SECRET) { |
| |
955 act = gaim_menu_action_new(_("Reset Secret Channel"), |
| |
956 GAIM_CALLBACK(silcgaim_chat_resetsecret), |
| |
957 NULL, NULL); |
| |
958 m = g_list_append(m, act); |
| |
959 } else { |
| |
960 act = gaim_menu_action_new(_("Set Secret Channel"), |
| |
961 GAIM_CALLBACK(silcgaim_chat_setsecret), |
| |
962 NULL, NULL); |
| |
963 m = g_list_append(m, act); |
| |
964 } |
| |
965 } |
| |
966 |
| |
967 if (channel) { |
| |
968 SilcGaimChatWb wb; |
| |
969 wb = silc_calloc(1, sizeof(*wb)); |
| |
970 wb->sg = sg; |
| |
971 wb->channel = channel; |
| |
972 act = gaim_menu_action_new(_("Draw On Whiteboard"), |
| |
973 GAIM_CALLBACK(silcgaim_chat_wb), |
| |
974 (void *)wb, NULL); |
| |
975 m = g_list_append(m, act); |
| |
976 } |
| |
977 |
| |
978 return m; |
| |
979 } |
| |
980 |
| |
981 |
| |
982 /******************************* Joining Etc. ********************************/ |
| |
983 |
| |
984 void silcgaim_chat_join_done(SilcClient client, |
| |
985 SilcClientConnection conn, |
| |
986 SilcClientEntry *clients, |
| |
987 SilcUInt32 clients_count, |
| |
988 void *context) |
| |
989 { |
| |
990 GaimConnection *gc = client->application; |
| |
991 SilcGaim sg = gc->proto_data; |
| |
992 SilcChannelEntry channel = context; |
| |
993 GaimConversation *convo; |
| |
994 SilcUInt32 retry = SILC_PTR_TO_32(channel->context); |
| |
995 SilcHashTableList htl; |
| |
996 SilcChannelUser chu; |
| |
997 GList *users = NULL, *flags = NULL; |
| |
998 char tmp[256]; |
| |
999 |
| |
1000 if (!clients && retry < 1) { |
| |
1001 /* Resolving users failed, try again. */ |
| |
1002 channel->context = SILC_32_TO_PTR(retry + 1); |
| |
1003 silc_client_get_clients_by_channel(client, conn, channel, |
| |
1004 silcgaim_chat_join_done, channel); |
| |
1005 return; |
| |
1006 } |
| |
1007 |
| |
1008 /* Add channel to Gaim */ |
| |
1009 channel->context = SILC_32_TO_PTR(++sg->channel_ids); |
| |
1010 serv_got_joined_chat(gc, sg->channel_ids, channel->channel_name); |
| |
1011 convo = gaim_find_conversation_with_account(GAIM_CONV_TYPE_CHAT, |
| |
1012 channel->channel_name, sg->account); |
| |
1013 if (!convo) |
| |
1014 return; |
| |
1015 |
| |
1016 /* Add all users to channel */ |
| |
1017 silc_hash_table_list(channel->user_list, &htl); |
| |
1018 while (silc_hash_table_get(&htl, NULL, (void *)&chu)) { |
| |
1019 GaimConvChatBuddyFlags f = GAIM_CBFLAGS_NONE; |
| |
1020 if (!chu->client->nickname) |
| |
1021 continue; |
| |
1022 chu->context = SILC_32_TO_PTR(sg->channel_ids); |
| |
1023 |
| |
1024 if (chu->mode & SILC_CHANNEL_UMODE_CHANFO) |
| |
1025 f |= GAIM_CBFLAGS_FOUNDER; |
| |
1026 if (chu->mode & SILC_CHANNEL_UMODE_CHANOP) |
| |
1027 f |= GAIM_CBFLAGS_OP; |
| |
1028 users = g_list_append(users, g_strdup(chu->client->nickname)); |
| |
1029 flags = g_list_append(flags, GINT_TO_POINTER(f)); |
| |
1030 |
| |
1031 if (chu->mode & SILC_CHANNEL_UMODE_CHANFO) { |
| |
1032 if (chu->client == conn->local_entry) |
| |
1033 g_snprintf(tmp, sizeof(tmp), |
| |
1034 _("You are channel founder on <I>%s</I>"), |
| |
1035 channel->channel_name); |
| |
1036 else |
| |
1037 g_snprintf(tmp, sizeof(tmp), |
| |
1038 _("Channel founder on <I>%s</I> is <I>%s</I>"), |
| |
1039 channel->channel_name, chu->client->nickname); |
| |
1040 |
| |
1041 gaim_conversation_write(convo, NULL, tmp, |
| |
1042 GAIM_MESSAGE_SYSTEM, time(NULL)); |
| |
1043 |
| |
1044 } |
| |
1045 } |
| |
1046 silc_hash_table_list_reset(&htl); |
| |
1047 |
| |
1048 gaim_conv_chat_add_users(GAIM_CONV_CHAT(convo), users, NULL, flags, FALSE); |
| |
1049 g_list_free(users); |
| |
1050 g_list_free(flags); |
| |
1051 |
| |
1052 /* Set topic */ |
| |
1053 if (channel->topic) |
| |
1054 gaim_conv_chat_set_topic(GAIM_CONV_CHAT(convo), NULL, channel->topic); |
| |
1055 |
| |
1056 /* Set nick */ |
| |
1057 gaim_conv_chat_set_nick(GAIM_CONV_CHAT(convo), conn->local_entry->nickname); |
| |
1058 } |
| |
1059 |
| |
1060 char *silcgaim_get_chat_name(GHashTable *data) |
| |
1061 { |
| |
1062 return g_strdup(g_hash_table_lookup(data, "channel")); |
| |
1063 } |
| |
1064 |
| |
1065 void silcgaim_chat_join(GaimConnection *gc, GHashTable *data) |
| |
1066 { |
| |
1067 SilcGaim sg = gc->proto_data; |
| |
1068 SilcClient client = sg->client; |
| |
1069 SilcClientConnection conn = sg->conn; |
| |
1070 const char *channel, *passphrase, *parentch; |
| |
1071 |
| |
1072 if (!conn) |
| |
1073 return; |
| |
1074 |
| |
1075 channel = g_hash_table_lookup(data, "channel"); |
| |
1076 passphrase = g_hash_table_lookup(data, "passphrase"); |
| |
1077 |
| |
1078 /* Check if we are joining a private group. Handle it |
| |
1079 purely locally as it's not a real channel */ |
| |
1080 if (strstr(channel, "[Private Group]")) { |
| |
1081 SilcChannelEntry channel_entry; |
| |
1082 SilcChannelPrivateKey key; |
| |
1083 GaimChat *c; |
| |
1084 SilcGaimPrvgrp grp; |
| |
1085 |
| |
1086 c = gaim_blist_find_chat(sg->account, channel); |
| |
1087 parentch = gaim_blist_node_get_string((GaimBlistNode *)c, "parentch"); |
| |
1088 if (!parentch) |
| |
1089 return; |
| |
1090 |
| |
1091 channel_entry = silc_client_get_channel(sg->client, sg->conn, |
| |
1092 (char *)parentch); |
| |
1093 if (!channel_entry || |
| |
1094 !silc_client_on_channel(channel_entry, sg->conn->local_entry)) { |
| |
1095 char tmp[512]; |
| |
1096 g_snprintf(tmp, sizeof(tmp), |
| |
1097 _("You have to join the %s channel before you are " |
| |
1098 "able to join the private group"), parentch); |
| |
1099 gaim_notify_error(gc, _("Join Private Group"), |
| |
1100 _("Cannot join private group"), tmp); |
| |
1101 return; |
| |
1102 } |
| |
1103 |
| |
1104 /* Add channel private key */ |
| |
1105 if (!silc_client_add_channel_private_key(client, conn, |
| |
1106 channel_entry, channel, |
| |
1107 NULL, NULL, |
| |
1108 (unsigned char *)passphrase, |
| |
1109 strlen(passphrase), &key)) |
| |
1110 return; |
| |
1111 |
| |
1112 /* Join the group */ |
| |
1113 grp = silc_calloc(1, sizeof(*grp)); |
| |
1114 if (!grp) |
| |
1115 return; |
| |
1116 grp->id = ++sg->channel_ids + SILCGAIM_PRVGRP; |
| |
1117 grp->chid = SILC_PTR_TO_32(channel_entry->context); |
| |
1118 grp->parentch = parentch; |
| |
1119 grp->channel = channel; |
| |
1120 grp->key = key; |
| |
1121 sg->grps = g_list_append(sg->grps, grp); |
| |
1122 serv_got_joined_chat(gc, grp->id, channel); |
| |
1123 return; |
| |
1124 } |
| |
1125 |
| |
1126 /* XXX We should have other properties here as well: |
| |
1127 1. whether to try to authenticate to the channel |
| |
1128 1a. with default key, |
| |
1129 1b. with specific key. |
| |
1130 2. whether to try to authenticate to become founder. |
| |
1131 2a. with default key, |
| |
1132 2b. with specific key. |
| |
1133 |
| |
1134 Since now such variety is not possible in the join dialog |
| |
1135 we always use -founder and -auth options, which try to |
| |
1136 do both 1 and 2 with default keys. */ |
| |
1137 |
| |
1138 /* Call JOIN */ |
| |
1139 if ((passphrase != NULL) && (*passphrase != '\0')) |
| |
1140 silc_client_command_call(client, conn, NULL, "JOIN", |
| |
1141 channel, passphrase, "-auth", "-founder", NULL); |
| |
1142 else |
| |
1143 silc_client_command_call(client, conn, NULL, "JOIN", |
| |
1144 channel, "-auth", "-founder", NULL); |
| |
1145 } |
| |
1146 |
| |
1147 void silcgaim_chat_invite(GaimConnection *gc, int id, const char *msg, |
| |
1148 const char *name) |
| |
1149 { |
| |
1150 SilcGaim sg = gc->proto_data; |
| |
1151 SilcClient client = sg->client; |
| |
1152 SilcClientConnection conn = sg->conn; |
| |
1153 SilcHashTableList htl; |
| |
1154 SilcChannelUser chu; |
| |
1155 gboolean found = FALSE; |
| |
1156 |
| |
1157 if (!conn) |
| |
1158 return; |
| |
1159 |
| |
1160 /* See if we are inviting on a private group. Invite |
| |
1161 to the actual channel */ |
| |
1162 if (id > SILCGAIM_PRVGRP) { |
| |
1163 GList *l; |
| |
1164 SilcGaimPrvgrp prv; |
| |
1165 |
| |
1166 for (l = sg->grps; l; l = l->next) |
| |
1167 if (((SilcGaimPrvgrp)l->data)->id == id) |
| |
1168 break; |
| |
1169 if (!l) |
| |
1170 return; |
| |
1171 prv = l->data; |
| |
1172 id = prv->chid; |
| |
1173 } |
| |
1174 |
| |
1175 /* Find channel by id */ |
| |
1176 silc_hash_table_list(conn->local_entry->channels, &htl); |
| |
1177 while (silc_hash_table_get(&htl, NULL, (void *)&chu)) { |
| |
1178 if (SILC_PTR_TO_32(chu->channel->context) == id ) { |
| |
1179 found = TRUE; |
| |
1180 break; |
| |
1181 } |
| |
1182 } |
| |
1183 silc_hash_table_list_reset(&htl); |
| |
1184 if (!found) |
| |
1185 return; |
| |
1186 |
| |
1187 /* Call INVITE */ |
| |
1188 silc_client_command_call(client, conn, NULL, "INVITE", |
| |
1189 chu->channel->channel_name, |
| |
1190 name, NULL); |
| |
1191 } |
| |
1192 |
| |
1193 void silcgaim_chat_leave(GaimConnection *gc, int id) |
| |
1194 { |
| |
1195 SilcGaim sg = gc->proto_data; |
| |
1196 SilcClient client = sg->client; |
| |
1197 SilcClientConnection conn = sg->conn; |
| |
1198 SilcHashTableList htl; |
| |
1199 SilcChannelUser chu; |
| |
1200 gboolean found = FALSE; |
| |
1201 GList *l; |
| |
1202 SilcGaimPrvgrp prv; |
| |
1203 |
| |
1204 if (!conn) |
| |
1205 return; |
| |
1206 |
| |
1207 /* See if we are leaving a private group */ |
| |
1208 if (id > SILCGAIM_PRVGRP) { |
| |
1209 SilcChannelEntry channel; |
| |
1210 |
| |
1211 for (l = sg->grps; l; l = l->next) |
| |
1212 if (((SilcGaimPrvgrp)l->data)->id == id) |
| |
1213 break; |
| |
1214 if (!l) |
| |
1215 return; |
| |
1216 prv = l->data; |
| |
1217 channel = silc_client_get_channel(sg->client, sg->conn, |
| |
1218 (char *)prv->parentch); |
| |
1219 if (!channel) |
| |
1220 return; |
| |
1221 silc_client_del_channel_private_key(client, conn, |
| |
1222 channel, prv->key); |
| |
1223 silc_free(prv); |
| |
1224 sg->grps = g_list_remove(sg->grps, prv); |
| |
1225 serv_got_chat_left(gc, id); |
| |
1226 return; |
| |
1227 } |
| |
1228 |
| |
1229 /* Find channel by id */ |
| |
1230 silc_hash_table_list(conn->local_entry->channels, &htl); |
| |
1231 while (silc_hash_table_get(&htl, NULL, (void *)&chu)) { |
| |
1232 if (SILC_PTR_TO_32(chu->channel->context) == id ) { |
| |
1233 found = TRUE; |
| |
1234 break; |
| |
1235 } |
| |
1236 } |
| |
1237 silc_hash_table_list_reset(&htl); |
| |
1238 if (!found) |
| |
1239 return; |
| |
1240 |
| |
1241 /* Call LEAVE */ |
| |
1242 silc_client_command_call(client, conn, NULL, "LEAVE", |
| |
1243 chu->channel->channel_name, NULL); |
| |
1244 |
| |
1245 serv_got_chat_left(gc, id); |
| |
1246 |
| |
1247 /* Leave from private groups on this channel as well */ |
| |
1248 for (l = sg->grps; l; l = l->next) |
| |
1249 if (((SilcGaimPrvgrp)l->data)->chid == id) { |
| |
1250 prv = l->data; |
| |
1251 silc_client_del_channel_private_key(client, conn, |
| |
1252 chu->channel, |
| |
1253 prv->key); |
| |
1254 serv_got_chat_left(gc, prv->id); |
| |
1255 silc_free(prv); |
| |
1256 sg->grps = g_list_remove(sg->grps, prv); |
| |
1257 if (!sg->grps) |
| |
1258 break; |
| |
1259 } |
| |
1260 } |
| |
1261 |
| |
1262 int silcgaim_chat_send(GaimConnection *gc, int id, const char *msg, GaimMessageFlags msgflags) |
| |
1263 { |
| |
1264 SilcGaim sg = gc->proto_data; |
| |
1265 SilcClient client = sg->client; |
| |
1266 SilcClientConnection conn = sg->conn; |
| |
1267 SilcHashTableList htl; |
| |
1268 SilcChannelUser chu; |
| |
1269 SilcChannelEntry channel = NULL; |
| |
1270 SilcChannelPrivateKey key = NULL; |
| |
1271 SilcUInt32 flags; |
| |
1272 int ret; |
| |
1273 char *msg2, *tmp; |
| |
1274 gboolean found = FALSE; |
| |
1275 gboolean sign = gaim_account_get_bool(sg->account, "sign-verify", FALSE); |
| |
1276 |
| |
1277 if (!msg || !conn) |
| |
1278 return 0; |
| |
1279 |
| |
1280 flags = SILC_MESSAGE_FLAG_UTF8; |
| |
1281 |
| |
1282 tmp = msg2 = gaim_unescape_html(msg); |
| |
1283 |
| |
1284 if (!g_ascii_strncasecmp(msg2, "/me ", 4)) |
| |
1285 { |
| |
1286 msg2 += 4; |
| |
1287 if (!*msg2) { |
| |
1288 g_free(tmp); |
| |
1289 return 0; |
| |
1290 } |
| |
1291 flags |= SILC_MESSAGE_FLAG_ACTION; |
| |
1292 } else if (strlen(msg) > 1 && msg[0] == '/') { |
| |
1293 if (!silc_client_command_call(client, conn, msg + 1)) |
| |
1294 gaim_notify_error(gc, _("Call Command"), _("Cannot call command"), |
| |
1295 _("Unknown command")); |
| |
1296 g_free(tmp); |
| |
1297 return 0; |
| |
1298 } |
| |
1299 |
| |
1300 |
| |
1301 if (sign) |
| |
1302 flags |= SILC_MESSAGE_FLAG_SIGNED; |
| |
1303 |
| |
1304 /* Get the channel private key if we are sending on |
| |
1305 private group */ |
| |
1306 if (id > SILCGAIM_PRVGRP) { |
| |
1307 GList *l; |
| |
1308 SilcGaimPrvgrp prv; |
| |
1309 |
| |
1310 for (l = sg->grps; l; l = l->next) |
| |
1311 if (((SilcGaimPrvgrp)l->data)->id == id) |
| |
1312 break; |
| |
1313 if (!l) { |
| |
1314 g_free(tmp); |
| |
1315 return 0; |
| |
1316 } |
| |
1317 prv = l->data; |
| |
1318 channel = silc_client_get_channel(sg->client, sg->conn, |
| |
1319 (char *)prv->parentch); |
| |
1320 if (!channel) { |
| |
1321 g_free(tmp); |
| |
1322 return 0; |
| |
1323 } |
| |
1324 key = prv->key; |
| |
1325 } |
| |
1326 |
| |
1327 if (!channel) { |
| |
1328 /* Find channel by id */ |
| |
1329 silc_hash_table_list(conn->local_entry->channels, &htl); |
| |
1330 while (silc_hash_table_get(&htl, NULL, (void *)&chu)) { |
| |
1331 if (SILC_PTR_TO_32(chu->channel->context) == id ) { |
| |
1332 found = TRUE; |
| |
1333 break; |
| |
1334 } |
| |
1335 } |
| |
1336 silc_hash_table_list_reset(&htl); |
| |
1337 if (!found) { |
| |
1338 g_free(tmp); |
| |
1339 return 0; |
| |
1340 } |
| |
1341 channel = chu->channel; |
| |
1342 } |
| |
1343 |
| |
1344 /* Send channel message */ |
| |
1345 ret = silc_client_send_channel_message(client, conn, channel, key, |
| |
1346 flags, (unsigned char *)msg2, |
| |
1347 strlen(msg2), TRUE); |
| |
1348 if (ret) { |
| |
1349 serv_got_chat_in(gc, id, gaim_connection_get_display_name(gc), 0, msg, |
| |
1350 time(NULL)); |
| |
1351 } |
| |
1352 g_free(tmp); |
| |
1353 |
| |
1354 return ret; |
| |
1355 } |
| |
1356 |
| |
1357 void silcgaim_chat_set_topic(GaimConnection *gc, int id, const char *topic) |
| |
1358 { |
| |
1359 SilcGaim sg = gc->proto_data; |
| |
1360 SilcClient client = sg->client; |
| |
1361 SilcClientConnection conn = sg->conn; |
| |
1362 SilcHashTableList htl; |
| |
1363 SilcChannelUser chu; |
| |
1364 gboolean found = FALSE; |
| |
1365 |
| |
1366 if (!conn) |
| |
1367 return; |
| |
1368 |
| |
1369 /* See if setting topic on private group. Set it |
| |
1370 on the actual channel */ |
| |
1371 if (id > SILCGAIM_PRVGRP) { |
| |
1372 GList *l; |
| |
1373 SilcGaimPrvgrp prv; |
| |
1374 |
| |
1375 for (l = sg->grps; l; l = l->next) |
| |
1376 if (((SilcGaimPrvgrp)l->data)->id == id) |
| |
1377 break; |
| |
1378 if (!l) |
| |
1379 return; |
| |
1380 prv = l->data; |
| |
1381 id = prv->chid; |
| |
1382 } |
| |
1383 |
| |
1384 /* Find channel by id */ |
| |
1385 silc_hash_table_list(conn->local_entry->channels, &htl); |
| |
1386 while (silc_hash_table_get(&htl, NULL, (void *)&chu)) { |
| |
1387 if (SILC_PTR_TO_32(chu->channel->context) == id ) { |
| |
1388 found = TRUE; |
| |
1389 break; |
| |
1390 } |
| |
1391 } |
| |
1392 silc_hash_table_list_reset(&htl); |
| |
1393 if (!found) |
| |
1394 return; |
| |
1395 |
| |
1396 /* Call TOPIC */ |
| |
1397 silc_client_command_call(client, conn, NULL, "TOPIC", |
| |
1398 chu->channel->channel_name, topic, NULL); |
| |
1399 } |
| |
1400 |
| |
1401 GaimRoomlist *silcgaim_roomlist_get_list(GaimConnection *gc) |
| |
1402 { |
| |
1403 SilcGaim sg = gc->proto_data; |
| |
1404 SilcClient client = sg->client; |
| |
1405 SilcClientConnection conn = sg->conn; |
| |
1406 GList *fields = NULL; |
| |
1407 GaimRoomlistField *f; |
| |
1408 |
| |
1409 if (!conn) |
| |
1410 return NULL; |
| |
1411 |
| |
1412 if (sg->roomlist) |
| |
1413 gaim_roomlist_unref(sg->roomlist); |
| |
1414 |
| |
1415 sg->roomlist_canceled = FALSE; |
| |
1416 |
| |
1417 sg->roomlist = gaim_roomlist_new(gaim_connection_get_account(gc)); |
| |
1418 f = gaim_roomlist_field_new(GAIM_ROOMLIST_FIELD_STRING, "", "channel", TRUE); |
| |
1419 fields = g_list_append(fields, f); |
| |
1420 f = gaim_roomlist_field_new(GAIM_ROOMLIST_FIELD_INT, |
| |
1421 _("Users"), "users", FALSE); |
| |
1422 fields = g_list_append(fields, f); |
| |
1423 f = gaim_roomlist_field_new(GAIM_ROOMLIST_FIELD_STRING, |
| |
1424 _("Topic"), "topic", FALSE); |
| |
1425 fields = g_list_append(fields, f); |
| |
1426 gaim_roomlist_set_fields(sg->roomlist, fields); |
| |
1427 |
| |
1428 /* Call LIST */ |
| |
1429 silc_client_command_call(client, conn, "LIST"); |
| |
1430 |
| |
1431 gaim_roomlist_set_in_progress(sg->roomlist, TRUE); |
| |
1432 |
| |
1433 return sg->roomlist; |
| |
1434 } |
| |
1435 |
| |
1436 void silcgaim_roomlist_cancel(GaimRoomlist *list) |
| |
1437 { |
| |
1438 GaimConnection *gc = gaim_account_get_connection(list->account); |
| |
1439 SilcGaim sg; |
| |
1440 |
| |
1441 if (!gc) |
| |
1442 return; |
| |
1443 sg = gc->proto_data; |
| |
1444 |
| |
1445 gaim_roomlist_set_in_progress(list, FALSE); |
| |
1446 if (sg->roomlist == list) { |
| |
1447 gaim_roomlist_unref(sg->roomlist); |
| |
1448 sg->roomlist = NULL; |
| |
1449 sg->roomlist_canceled = TRUE; |
| |
1450 } |
| |
1451 } |