| 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 } |
|