/* command_reply.c Author: Pekka Riikonen Copyright (C) 1997 - 2002 Pekka Riikonen This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; version 2 of the License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. */ /* $Id$ */ /* * Command reply functions are "the otherside" of the command functions. * Reply to a command sent by server is handled by these functions. * * The arguments received from server are also passed to the calling * application through command_reply client operation. The arguments are * exactly same and in same order as the server sent it. However, ID's are * not sent to the application. Instead, corresponding ID entry is sent * to the application. For example, instead of sending Client ID the * corresponding SilcClientEntry is sent to the application. The case is * same with for example Channel ID's. This way application has all the * necessary data already in hand without redundant searching. If ID is * received but ID entry does not exist, NULL is sent. */ #include "silcincludes.h" #include "silcclient.h" #include "client_internal.h" const SilcCommandStatusMessage silc_command_status_messages[] = { { STAT(NO_SUCH_NICK), "There was no such nickname" }, { STAT(NO_SUCH_CHANNEL), "There was no such channel" }, { STAT(NO_SUCH_SERVER), "There was no such server" }, { STAT(TOO_MANY_TARGETS), "Duplicate recipients. No message delivered" }, { STAT(NO_RECIPIENT), "No recipient given" }, { STAT(UNKNOWN_COMMAND), "Unknown command" }, { STAT(WILDCARDS), "Unknown command" }, { STAT(NO_CLIENT_ID), "No Client ID given" }, { STAT(NO_CHANNEL_ID), "No Channel ID given" }, { STAT(NO_SERVER_ID), "No Server ID given" }, { STAT(BAD_CLIENT_ID), "Bad Client ID" }, { STAT(BAD_CHANNEL_ID), "Bad Channel ID" }, { STAT(NO_SUCH_CLIENT_ID), "There is no such client" }, { STAT(NO_SUCH_CHANNEL_ID),"There is no such channel" }, { STAT(NICKNAME_IN_USE), "Nickname already exists" }, { STAT(NOT_ON_CHANNEL), "You are not on that channel" }, { STAT(USER_NOT_ON_CHANNEL),"They are not on the channel" }, { STAT(USER_ON_CHANNEL), "User already on the channel" }, { STAT(NOT_REGISTERED), "You have not registered" }, { STAT(NOT_ENOUGH_PARAMS), "Not enough parameters" }, { STAT(TOO_MANY_PARAMS), "Too many parameters" }, { STAT(PERM_DENIED), "Permission denied" }, { STAT(BANNED_FROM_SERVER),"You are banned from this server" }, { STAT(BAD_PASSWORD), "Cannot join channel. Incorrect password" }, { STAT(CHANNEL_IS_FULL), "Cannot join channel. Channel is full" }, { STAT(NOT_INVITED), "Cannot join channel. You have not been invited" }, { STAT(BANNED_FROM_CHANNEL), "Cannot join channel. You have been banned" }, { STAT(UNKNOWN_MODE), "Unknown mode" }, { STAT(NOT_YOU), "Cannot change mode for other users" }, { STAT(NO_CHANNEL_PRIV), "Permission denied. You are not channel operator" }, { STAT(NO_CHANNEL_FOPRIV),"Permission denied. You are not channel founder" }, { STAT(NO_SERVER_PRIV), "Permission denied. You are not server operator" }, { STAT(NO_ROUTER_PRIV), "Permission denied. You are not SILC operator" }, { STAT(BAD_NICKNAME), "Bad nickname" }, { STAT(BAD_CHANNEL), "Bad channel name" }, { STAT(AUTH_FAILED), "Authentication failed" }, { STAT(UNKNOWN_ALGORITHM), "Unsupported algorithm" }, { STAT(NO_SUCH_SERVER_ID), "No such Server ID" }, { 0, NULL } }; /* Command reply operation that is called at the end of all command replys. Usage: COMMAND_REPLY((ARGS, argument1, argument2, etc...)), */ #define COMMAND_REPLY(args) cmd->client->internal->ops->command_reply args #define ARGS cmd->client, cmd->sock->user_data, \ cmd->payload, TRUE, silc_command_get(cmd->payload), cmd->status /* Error reply to application. Usage: COMMAND_REPLY_ERROR; */ #define COMMAND_REPLY_ERROR cmd->client->internal->ops-> \ command_reply(cmd->client, cmd->sock->user_data, cmd->payload, \ FALSE, silc_command_get(cmd->payload), cmd->status) #define SAY cmd->client->internal->ops->say /* All functions that call the COMMAND_CHECK_STATUS macro must have out: goto label. */ #define COMMAND_CHECK_STATUS \ do { \ SILC_LOG_DEBUG(("Start")); \ if (!silc_command_get_status(cmd->payload, NULL, NULL)) { \ COMMAND_REPLY_ERROR; \ goto out; \ } \ } while(0) /* Process received command reply. */ void silc_client_command_reply_process(SilcClient client, SilcSocketConnection sock, SilcPacketContext *packet) { SilcBuffer buffer = packet->buffer; SilcClientCommand cmd; SilcClientCommandReplyContext ctx; SilcCommandPayload payload; SilcCommand command; SilcCommandCb reply = NULL; /* Get command reply payload from packet */ payload = silc_command_payload_parse(buffer->data, buffer->len); if (!payload) { /* Silently ignore bad reply packet */ SILC_LOG_DEBUG(("Bad command reply packet")); return; } /* Allocate command reply context. This must be free'd by the command reply routine receiving it. */ ctx = silc_calloc(1, sizeof(*ctx)); ctx->client = client; ctx->sock = sock; ctx->payload = payload; ctx->args = silc_command_get_args(ctx->payload); ctx->packet = packet; ctx->ident = silc_command_get_ident(ctx->payload); silc_command_get_status(ctx->payload, &ctx->status, &ctx->error); /* Check for pending commands and mark to be exeucted */ silc_client_command_pending_check(sock->user_data, ctx, silc_command_get(ctx->payload), ctx->ident); /* Execute command reply */ command = silc_command_get(ctx->payload); /* Try to find matching the command identifier */ silc_list_start(client->internal->commands); while ((cmd = silc_list_get(client->internal->commands)) != SILC_LIST_END) { if (cmd->cmd == command && !cmd->ident) reply = cmd->reply; if (cmd->cmd == command && cmd->ident == ctx->ident) { (*cmd->reply)((void *)ctx, NULL); break; } } if (cmd == SILC_LIST_END) { if (reply) /* No specific identifier for command reply, call first one found */ (*reply)(ctx, NULL); else silc_free(ctx); } } /* Returns status message string */ char *silc_client_command_status_message(SilcCommandStatus status) { int i; for (i = 0; silc_command_status_messages[i].message; i++) { if (silc_command_status_messages[i].status == status) break; } if (silc_command_status_messages[i].message == NULL) return NULL; return silc_command_status_messages[i].message; } /* Free command reply context and its internals. */ void silc_client_command_reply_free(SilcClientCommandReplyContext cmd) { if (cmd) { silc_command_payload_free(cmd->payload); silc_free(cmd); } } static void silc_client_command_reply_whois_save(SilcClientCommandReplyContext cmd, SilcCommandStatus status, bool notify) { SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcClientID *client_id; SilcClientEntry client_entry = NULL; int argc; SilcUInt32 len; unsigned char *id_data, *tmp; char *nickname = NULL, *username = NULL; char *realname = NULL; SilcUInt32 idle = 0, mode = 0; SilcBuffer channels = NULL; unsigned char *fingerprint; SilcUInt32 fingerprint_len; argc = silc_argument_get_arg_num(cmd->args); id_data = silc_argument_get_arg_type(cmd->args, 2, &len); if (!id_data) { if (notify) COMMAND_REPLY_ERROR; return; } client_id = silc_id_payload_parse_id(id_data, len, NULL); if (!client_id) { if (notify) COMMAND_REPLY_ERROR; return; } nickname = silc_argument_get_arg_type(cmd->args, 3, &len); username = silc_argument_get_arg_type(cmd->args, 4, &len); realname = silc_argument_get_arg_type(cmd->args, 5, &len); if (!nickname || !username || !realname) { if (notify) COMMAND_REPLY_ERROR; return; } tmp = silc_argument_get_arg_type(cmd->args, 6, &len); if (tmp) { channels = silc_buffer_alloc(len); silc_buffer_pull_tail(channels, SILC_BUFFER_END(channels)); silc_buffer_put(channels, tmp, len); } tmp = silc_argument_get_arg_type(cmd->args, 7, &len); if (tmp) SILC_GET32_MSB(mode, tmp); tmp = silc_argument_get_arg_type(cmd->args, 8, &len); if (tmp) SILC_GET32_MSB(idle, tmp); fingerprint = silc_argument_get_arg_type(cmd->args, 9, &fingerprint_len); /* Check if we have this client cached already. */ client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); if (!client_entry) { SILC_LOG_DEBUG(("Adding new client entry")); client_entry = silc_client_add_client(cmd->client, conn, nickname, username, realname, client_id, mode); } else { silc_client_update_client(cmd->client, conn, client_entry, nickname, username, realname, mode); silc_free(client_id); } if (fingerprint && !client_entry->fingerprint) { client_entry->fingerprint = silc_memdup(fingerprint, fingerprint_len); client_entry->fingerprint_len = fingerprint_len; } if (client_entry->status & SILC_CLIENT_STATUS_RESOLVING) client_entry->status &= ~SILC_CLIENT_STATUS_RESOLVING; /* Notify application */ if (!cmd->callback && notify) COMMAND_REPLY((ARGS, client_entry, nickname, username, realname, channels, mode, idle, fingerprint)); if (channels) silc_buffer_free(channels); } /* Received reply for WHOIS command. This maybe called several times for one WHOIS command as server may reply with list of results. */ SILC_CLIENT_CMD_REPLY_FUNC(whois) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; COMMAND_CHECK_STATUS; /* Save WHOIS info */ silc_client_command_reply_whois_save(cmd, cmd->status, TRUE); /* Pending callbacks are not executed if this was an list entry */ if (cmd->status != SILC_STATUS_OK && cmd->status != SILC_STATUS_LIST_END) { silc_client_command_reply_free(cmd); return; } out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_WHOIS); /* If we received notify for invalid ID we'll remove the ID if we have it cached. */ if (cmd->error == SILC_STATUS_ERR_NO_SUCH_CLIENT_ID) { SilcClientEntry client_entry; SilcUInt32 tmp_len; unsigned char *tmp = silc_argument_get_arg_type(silc_command_get_args(cmd->payload), 2, &tmp_len); if (tmp) { SilcClientID *client_id = silc_id_payload_parse_id(tmp, tmp_len, NULL); if (client_id) { client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); if (client_entry) silc_client_del_client(cmd->client, conn, client_entry); silc_free(client_id); } } } silc_client_command_reply_free(cmd); } /* Received reply for WHOWAS command. */ SILC_CLIENT_CMD_REPLY_FUNC(whowas) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcClientID *client_id; SilcClientEntry client_entry = NULL; SilcUInt32 len; unsigned char *id_data; char *nickname, *username; char *realname = NULL; COMMAND_CHECK_STATUS; id_data = silc_argument_get_arg_type(cmd->args, 2, &len); if (!id_data) { COMMAND_REPLY_ERROR; goto out; } client_id = silc_id_payload_parse_id(id_data, len, NULL); if (!client_id) { COMMAND_REPLY_ERROR; goto out; } /* Get the client entry, if exists */ client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); silc_free(client_id); nickname = silc_argument_get_arg_type(cmd->args, 3, &len); username = silc_argument_get_arg_type(cmd->args, 4, &len); realname = silc_argument_get_arg_type(cmd->args, 5, &len); if (!nickname || !username) { COMMAND_REPLY_ERROR; goto out; } /* Notify application. We don't save any history information to any cache. Just pass the data to the application for displaying on the screen. */ COMMAND_REPLY((ARGS, client_entry, nickname, username, realname)); /* Pending callbacks are not executed if this was an list entry */ if (cmd->status != SILC_STATUS_OK && cmd->status != SILC_STATUS_LIST_END) { silc_client_command_reply_free(cmd); return; } out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_WHOWAS); silc_client_command_reply_free(cmd); } static void silc_client_command_reply_identify_save(SilcClientCommandReplyContext cmd, SilcCommandStatus status, bool notify) { SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcClient client = cmd->client; SilcClientID *client_id = NULL; SilcServerID *server_id = NULL; SilcChannelID *channel_id = NULL; SilcClientEntry client_entry; SilcServerEntry server_entry; SilcChannelEntry channel_entry; int argc; SilcUInt32 len; unsigned char *id_data; char *name = NULL, *info = NULL; SilcIDPayload idp = NULL; SilcIdType id_type; argc = silc_argument_get_arg_num(cmd->args); id_data = silc_argument_get_arg_type(cmd->args, 2, &len); if (!id_data) { if (notify) COMMAND_REPLY_ERROR; return; } idp = silc_id_payload_parse(id_data, len); if (!idp) { if (notify) COMMAND_REPLY_ERROR; return; } name = silc_argument_get_arg_type(cmd->args, 3, &len); info = silc_argument_get_arg_type(cmd->args, 4, &len); id_type = silc_id_payload_get_type(idp); switch (id_type) { case SILC_ID_CLIENT: client_id = silc_id_payload_get_id(idp); SILC_LOG_DEBUG(("Received client information")); /* Check if we have this client cached already. */ client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); if (!client_entry) { SILC_LOG_DEBUG(("Adding new client entry")); client_entry = silc_client_add_client(cmd->client, conn, name, info, NULL, silc_id_dup(client_id, id_type), 0); } else { silc_client_update_client(cmd->client, conn, client_entry, name, info, NULL, 0); } if (client_entry->status & SILC_CLIENT_STATUS_RESOLVING) client_entry->status &= ~SILC_CLIENT_STATUS_RESOLVING; /* Notify application */ if (notify) COMMAND_REPLY((ARGS, client_entry, name, info)); break; case SILC_ID_SERVER: server_id = silc_id_payload_get_id(idp); SILC_LOG_DEBUG(("Received server information")); /* Check if we have this server cached already. */ server_entry = silc_client_get_server_by_id(cmd->client, conn, server_id); if (!server_entry) { SILC_LOG_DEBUG(("Adding new server entry")); server_entry = silc_client_add_server(cmd->client, conn, name, info, silc_id_dup(server_id, id_type)); if (!server_entry) { if (notify) COMMAND_REPLY_ERROR; return; } } /* Notify application */ if (notify) COMMAND_REPLY((ARGS, server_entry, name, info)); break; case SILC_ID_CHANNEL: channel_id = silc_id_payload_get_id(idp); SILC_LOG_DEBUG(("Received channel information")); /* Check if we have this channel cached already. */ channel_entry = silc_client_get_channel_by_id(client, conn, channel_id); if (!channel_entry) { if (!name) break; /* Add new channel entry */ channel_entry = silc_client_add_channel(client, conn, name, 0, channel_id); channel_id = NULL; } /* Notify application */ if (notify) COMMAND_REPLY((ARGS, channel_entry, name, info)); break; } silc_id_payload_free(idp); silc_free(client_id); silc_free(server_id); silc_free(channel_id); } /* Received reply for IDENTIFY command. This maybe called several times for one IDENTIFY command as server may reply with list of results. This is totally silent and does not print anything on screen. */ SILC_CLIENT_CMD_REPLY_FUNC(identify) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; COMMAND_CHECK_STATUS; /* Save IDENTIFY info */ silc_client_command_reply_identify_save(cmd, cmd->status, TRUE); /* Pending callbacks are not executed if this was an list entry */ if (cmd->status != SILC_STATUS_OK && cmd->status != SILC_STATUS_LIST_END) { silc_client_command_reply_free(cmd); return; } out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_IDENTIFY); /* If we received notify for invalid ID we'll remove the ID if we have it cached. */ if (cmd->error == SILC_STATUS_ERR_NO_SUCH_CLIENT_ID) { SilcClientEntry client_entry; SilcUInt32 tmp_len; unsigned char *tmp = silc_argument_get_arg_type(silc_command_get_args(cmd->payload), 2, &tmp_len); if (tmp) { SilcClientID *client_id = silc_id_payload_parse_id(tmp, tmp_len, NULL); if (client_id) { client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); if (client_entry) silc_client_del_client(cmd->client, conn, client_entry); silc_free(client_id); } } } silc_client_command_reply_free(cmd); } /* Received reply for command NICK. If everything went without errors we just received our new Client ID. */ SILC_CLIENT_CMD_REPLY_FUNC(nick) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcIDPayload idp; unsigned char *tmp; SilcUInt32 argc, len; SILC_LOG_DEBUG(("Start")); if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "Cannot set nickname: %s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } argc = silc_argument_get_arg_num(cmd->args); if (argc < 2 || argc > 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "Cannot set nickname: bad reply to command"); COMMAND_REPLY_ERROR; goto out; } /* Take received Client ID */ tmp = silc_argument_get_arg_type(cmd->args, 2, &len); idp = silc_id_payload_parse(tmp, len); if (!idp) { COMMAND_REPLY_ERROR; goto out; } silc_client_receive_new_id(cmd->client, cmd->sock, idp); /* Notify application */ SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_NICK); COMMAND_REPLY((ARGS, conn->local_entry)); silc_client_command_reply_free(cmd); return; out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_NICK); silc_client_command_reply_free(cmd); } /* Received reply to the LIST command. */ SILC_CLIENT_CMD_REPLY_FUNC(list) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; unsigned char *tmp, *name, *topic; SilcUInt32 usercount = 0, len; SilcChannelID *channel_id = NULL; SilcChannelEntry channel_entry; COMMAND_CHECK_STATUS; tmp = silc_argument_get_arg_type(cmd->args, 2, &len); if (!tmp) { COMMAND_REPLY_ERROR; goto out; } channel_id = silc_id_payload_parse_id(tmp, len, NULL); if (!channel_id) { COMMAND_REPLY_ERROR; goto out; } name = silc_argument_get_arg_type(cmd->args, 3, NULL); if (!name) { COMMAND_REPLY_ERROR; goto out; } topic = silc_argument_get_arg_type(cmd->args, 4, NULL); tmp = silc_argument_get_arg_type(cmd->args, 5, NULL); if (tmp) SILC_GET32_MSB(usercount, tmp); /* Check whether the channel exists, and add it to cache if it doesn't. */ channel_entry = silc_client_get_channel_by_id(cmd->client, conn, channel_id); if (!channel_entry) { /* Add new channel entry */ channel_entry = silc_client_add_channel(cmd->client, conn, name, 0, channel_id); if (!channel_entry) { COMMAND_REPLY_ERROR; goto out; } channel_id = NULL; } /* Notify application */ COMMAND_REPLY((ARGS, channel_entry, name, topic, usercount)); /* Pending callbacks are not executed if this was an list entry */ if (cmd->status != SILC_STATUS_OK && cmd->status != SILC_STATUS_LIST_END) { silc_client_command_reply_free(cmd); return; } out: silc_free(channel_id); SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_LIST); silc_client_command_reply_free(cmd); } /* Received reply to topic command. */ SILC_CLIENT_CMD_REPLY_FUNC(topic) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcChannelEntry channel; SilcChannelID *channel_id = NULL; unsigned char *tmp; char *topic; SilcUInt32 argc, len; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } argc = silc_argument_get_arg_num(cmd->args); if (argc < 1 || argc > 3) { COMMAND_REPLY_ERROR; goto out; } /* Take Channel ID */ tmp = silc_argument_get_arg_type(cmd->args, 2, &len); if (!tmp) goto out; /* Take topic */ topic = silc_argument_get_arg_type(cmd->args, 3, NULL); if (!topic) goto out; channel_id = silc_id_payload_parse_id(tmp, len, NULL); if (!channel_id) goto out; /* Get the channel entry */ channel = silc_client_get_channel_by_id(cmd->client, conn, channel_id); if (!channel) { silc_free(channel_id); COMMAND_REPLY_ERROR; goto out; } /* Notify application */ COMMAND_REPLY((ARGS, channel, topic)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_TOPIC); silc_client_command_reply_free(cmd); } /* Received reply to invite command. */ SILC_CLIENT_CMD_REPLY_FUNC(invite) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcChannelEntry channel; SilcChannelID *channel_id; unsigned char *tmp; SilcUInt32 len; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Take Channel ID */ tmp = silc_argument_get_arg_type(cmd->args, 2, &len); if (!tmp) goto out; channel_id = silc_id_payload_parse_id(tmp, len, NULL); if (!channel_id) goto out; /* Get the channel entry */ channel = silc_client_get_channel_by_id(cmd->client, conn, channel_id); if (!channel) { silc_free(channel_id); COMMAND_REPLY_ERROR; goto out; } /* Get the invite list */ tmp = silc_argument_get_arg_type(cmd->args, 3, &len); /* Notify application */ COMMAND_REPLY((ARGS, channel, tmp)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_INVITE); silc_client_command_reply_free(cmd); } /* Received reply to the KILL command. */ SILC_CLIENT_CMD_REPLY_FUNC(kill) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Notify application */ COMMAND_REPLY((ARGS)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_KILL); silc_client_command_reply_free(cmd); } /* Received reply to INFO command. We receive the server ID and some information about the server user requested. */ SILC_CLIENT_CMD_REPLY_FUNC(info) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; unsigned char *tmp; SilcServerEntry server; SilcServerID *server_id = NULL; char *server_name, *server_info; SilcUInt32 len; SILC_LOG_DEBUG(("Start")); if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Get server ID */ tmp = silc_argument_get_arg_type(cmd->args, 2, &len); if (!tmp) goto out; server_id = silc_id_payload_parse_id(tmp, len, NULL); if (!server_id) goto out; /* Get server name */ server_name = silc_argument_get_arg_type(cmd->args, 3, NULL); if (!server_name) goto out; /* Get server info */ server_info = silc_argument_get_arg_type(cmd->args, 4, NULL); if (!server_info) goto out; /* See whether we have this server cached. If not create it. */ server = silc_client_get_server_by_id(cmd->client, conn, server_id); if (!server) { SILC_LOG_DEBUG(("New server entry")); server = silc_client_add_server(cmd->client, conn, server_name, server_info, silc_id_dup(server_id, SILC_ID_SERVER)); if (!server) goto out; } /* Notify application */ COMMAND_REPLY((ARGS, server, server->server_name, server->server_info)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_INFO); silc_free(server_id); silc_client_command_reply_free(cmd); } /* Received reply to PING command. The reply time is shown to user. */ SILC_CLIENT_CMD_REPLY_FUNC(ping) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; void *id; int i; time_t diff, curtime; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } curtime = time(NULL); id = silc_id_str2id(cmd->packet->src_id, cmd->packet->src_id_len, cmd->packet->src_id_type); if (!id || !conn->ping) { COMMAND_REPLY_ERROR; goto out; } for (i = 0; i < conn->ping_count; i++) { if (!conn->ping[i].dest_id) continue; if (SILC_ID_SERVER_COMPARE(conn->ping[i].dest_id, id)) { diff = curtime - conn->ping[i].start_time; SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Ping reply from %s: %d second%s", conn->ping[i].dest_name, diff, diff == 1 ? "" : "s"); conn->ping[i].start_time = 0; silc_free(conn->ping[i].dest_id); conn->ping[i].dest_id = NULL; silc_free(conn->ping[i].dest_name); conn->ping[i].dest_name = NULL; break; } } silc_free(id); /* Notify application */ COMMAND_REPLY((ARGS)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_PING); silc_client_command_reply_free(cmd); } /* Received reply for JOIN command. */ SILC_CLIENT_CMD_REPLY_FUNC(join) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcChannelEntry channel; SilcChannelUser chu; SilcChannelID *channel_id; SilcUInt32 argc, mode = 0, len, list_count; char *topic, *tmp, *channel_name = NULL, *hmac; SilcBuffer keyp = NULL, client_id_list = NULL, client_mode_list = NULL; int i; SILC_LOG_DEBUG(("Start")); if (cmd->error != SILC_STATUS_OK) { if (cmd->error != SILC_STATUS_ERR_USER_ON_CHANNEL) SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } argc = silc_argument_get_arg_num(cmd->args); if (argc < 7 || argc > 14) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "Cannot join channel: Bad reply packet"); COMMAND_REPLY_ERROR; goto out; } /* Get channel name */ tmp = silc_argument_get_arg_type(cmd->args, 2, NULL); if (!tmp) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "Cannot join channel: Bad reply packet"); COMMAND_REPLY_ERROR; goto out; } channel_name = tmp; /* Get Channel ID */ tmp = silc_argument_get_arg_type(cmd->args, 3, &len); if (!tmp) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "Cannot join channel: Bad reply packet"); COMMAND_REPLY_ERROR; goto out; } channel_id = silc_id_payload_parse_id(tmp, len, NULL); if (!channel_id) { COMMAND_REPLY_ERROR; goto out; } /* Get channel mode */ tmp = silc_argument_get_arg_type(cmd->args, 5, NULL); if (tmp) SILC_GET32_MSB(mode, tmp); /* Get channel key */ tmp = silc_argument_get_arg_type(cmd->args, 7, &len); if (tmp) { keyp = silc_buffer_alloc(len); silc_buffer_pull_tail(keyp, SILC_BUFFER_END(keyp)); silc_buffer_put(keyp, tmp, len); } /* Get topic */ topic = silc_argument_get_arg_type(cmd->args, 10, NULL); /* Check whether we have this channel entry already. */ channel = silc_client_get_channel(cmd->client, conn, channel_name); if (channel) { if (!SILC_ID_CHANNEL_COMPARE(channel->id, channel_id)) silc_client_replace_channel_id(cmd->client, conn, channel, channel_id); } else { /* Create new channel entry */ channel = silc_client_add_channel(cmd->client, conn, channel_name, mode, channel_id); } conn->current_channel = channel; /* Get hmac */ hmac = silc_argument_get_arg_type(cmd->args, 11, NULL); if (hmac) { if (!silc_hmac_alloc(hmac, NULL, &channel->hmac)) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "Cannot join channel: Unsupported HMAC `%s'", hmac); COMMAND_REPLY_ERROR; goto out; } } /* Get the list count */ tmp = silc_argument_get_arg_type(cmd->args, 12, &len); if (!tmp) goto out; SILC_GET32_MSB(list_count, tmp); /* Get Client ID list */ tmp = silc_argument_get_arg_type(cmd->args, 13, &len); if (!tmp) goto out; client_id_list = silc_buffer_alloc(len); silc_buffer_pull_tail(client_id_list, len); silc_buffer_put(client_id_list, tmp, len); /* Get client mode list */ tmp = silc_argument_get_arg_type(cmd->args, 14, &len); if (!tmp) goto out; client_mode_list = silc_buffer_alloc(len); silc_buffer_pull_tail(client_mode_list, len); silc_buffer_put(client_mode_list, tmp, len); /* Add clients we received in the reply to the channel */ for (i = 0; i < list_count; i++) { SilcUInt16 idp_len; SilcUInt32 mode; SilcClientID *client_id; SilcClientEntry client_entry; /* Client ID */ SILC_GET16_MSB(idp_len, client_id_list->data + 2); idp_len += 4; client_id = silc_id_payload_parse_id(client_id_list->data, idp_len, NULL); if (!client_id) continue; /* Mode */ SILC_GET32_MSB(mode, client_mode_list->data); /* Check if we have this client cached already. */ client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); if (!client_entry) { /* No, we don't have it, add entry for it. */ client_entry = silc_client_add_client(cmd->client, conn, NULL, NULL, NULL, silc_id_dup(client_id, SILC_ID_CLIENT), 0); } /* Join client to the channel */ if (!silc_client_on_channel(channel, client_entry)) { chu = silc_calloc(1, sizeof(*chu)); chu->client = client_entry; chu->channel = channel; chu->mode = mode; silc_hash_table_add(channel->user_list, client_entry, chu); silc_hash_table_add(client_entry->channels, channel, chu); } silc_free(client_id); silc_buffer_pull(client_id_list, idp_len); silc_buffer_pull(client_mode_list, 4); } silc_buffer_push(client_id_list, client_id_list->data - client_id_list->head); silc_buffer_push(client_mode_list, client_mode_list->data - client_mode_list->head); /* Save channel key */ if (keyp && !(channel->mode & SILC_CHANNEL_MODE_PRIVKEY)) silc_client_save_channel_key(cmd->client, conn, keyp, channel); /* Notify application */ COMMAND_REPLY((ARGS, channel_name, channel, mode, 0, keyp ? keyp->head : NULL, NULL, NULL, topic, hmac, list_count, client_id_list, client_mode_list)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_JOIN); silc_client_command_reply_free(cmd); if (keyp) silc_buffer_free(keyp); if (client_id_list) silc_buffer_free(client_id_list); if (client_mode_list) silc_buffer_free(client_mode_list); } /* Received reply for MOTD command */ SILC_CLIENT_CMD_REPLY_FUNC(motd) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcUInt32 argc, i; char *motd = NULL, *cp, line[256]; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; return; } argc = silc_argument_get_arg_num(cmd->args); if (argc > 3) { COMMAND_REPLY_ERROR; goto out; } if (argc == 3) { motd = silc_argument_get_arg_type(cmd->args, 3, NULL); if (!motd) { COMMAND_REPLY_ERROR; goto out; } i = 0; cp = motd; while(cp[i] != 0) { if (cp[i++] == '\n') { memset(line, 0, sizeof(line)); strncat(line, cp, i - 1); cp += i; if (i == 2) line[0] = ' '; SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "%s", line); if (!strlen(cp)) break; i = 0; } } } /* Notify application */ COMMAND_REPLY((ARGS, motd)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_MOTD); silc_client_command_reply_free(cmd); } /* Received reply tot he UMODE command. Save the current user mode */ SILC_CLIENT_CMD_REPLY_FUNC(umode) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; unsigned char *tmp; SilcUInt32 mode; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } tmp = silc_argument_get_arg_type(cmd->args, 2, NULL); if (!tmp) { COMMAND_REPLY_ERROR; goto out; } SILC_GET32_MSB(mode, tmp); conn->local_entry->mode = mode; /* Notify application */ COMMAND_REPLY((ARGS, mode)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_UMODE); silc_client_command_reply_free(cmd); } /* Received reply for CMODE command. */ SILC_CLIENT_CMD_REPLY_FUNC(cmode) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; unsigned char *tmp; SilcUInt32 mode; SilcChannelID *channel_id; SilcChannelEntry channel; SilcUInt32 len; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Take Channel ID */ tmp = silc_argument_get_arg_type(cmd->args, 2, &len); if (!tmp) goto out; channel_id = silc_id_payload_parse_id(tmp, len, NULL); if (!channel_id) goto out; /* Get the channel entry */ channel = silc_client_get_channel_by_id(cmd->client, conn, channel_id); if (!channel) { silc_free(channel_id); COMMAND_REPLY_ERROR; goto out; } /* Get channel mode */ tmp = silc_argument_get_arg_type(cmd->args, 3, NULL); if (!tmp) { silc_free(channel_id); COMMAND_REPLY_ERROR; goto out; } /* Save the mode */ SILC_GET32_MSB(mode, tmp); channel->mode = mode; /* Notify application */ COMMAND_REPLY((ARGS, channel, mode)); silc_free(channel_id); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_CMODE); silc_client_command_reply_free(cmd); } /* Received reply for CUMODE command */ SILC_CLIENT_CMD_REPLY_FUNC(cumode) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcClientID *client_id; SilcChannelID *channel_id; SilcClientEntry client_entry; SilcChannelEntry channel; SilcChannelUser chu; unsigned char *modev, *tmp, *id; SilcUInt32 len, mode; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Get channel mode */ modev = silc_argument_get_arg_type(cmd->args, 2, NULL); if (!modev) { COMMAND_REPLY_ERROR; goto out; } /* Take Channel ID */ tmp = silc_argument_get_arg_type(cmd->args, 3, &len); if (!tmp) goto out; channel_id = silc_id_payload_parse_id(tmp, len, NULL); if (!channel_id) goto out; /* Get the channel entry */ channel = silc_client_get_channel_by_id(cmd->client, conn, channel_id); if (!channel) { silc_free(channel_id); COMMAND_REPLY_ERROR; goto out; } /* Get Client ID */ id = silc_argument_get_arg_type(cmd->args, 4, &len); if (!id) { silc_free(channel_id); COMMAND_REPLY_ERROR; goto out; } client_id = silc_id_payload_parse_id(id, len, NULL); if (!client_id) { silc_free(channel_id); COMMAND_REPLY_ERROR; goto out; } /* Get client entry */ client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); if (!client_entry) { silc_free(channel_id); silc_free(client_id); COMMAND_REPLY_ERROR; goto out; } /* Save the mode */ SILC_GET32_MSB(mode, modev); chu = silc_client_on_channel(channel, client_entry); if (chu) chu->mode = mode; /* Notify application */ COMMAND_REPLY((ARGS, mode, channel, client_entry)); silc_free(client_id); silc_free(channel_id); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_CUMODE); silc_client_command_reply_free(cmd); } SILC_CLIENT_CMD_REPLY_FUNC(kick) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Notify application */ COMMAND_REPLY((ARGS)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_KICK); silc_client_command_reply_free(cmd); } SILC_CLIENT_CMD_REPLY_FUNC(silcoper) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Notify application */ COMMAND_REPLY((ARGS)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_SILCOPER); silc_client_command_reply_free(cmd); } SILC_CLIENT_CMD_REPLY_FUNC(oper) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Notify application */ COMMAND_REPLY((ARGS)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_OPER); silc_client_command_reply_free(cmd); } SILC_CLIENT_CMD_REPLY_FUNC(ban) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcChannelEntry channel; SilcChannelID *channel_id; unsigned char *tmp; SilcUInt32 len; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Take Channel ID */ tmp = silc_argument_get_arg_type(cmd->args, 2, &len); if (!tmp) goto out; channel_id = silc_id_payload_parse_id(tmp, len, NULL); if (!channel_id) goto out; /* Get the channel entry */ channel = silc_client_get_channel_by_id(cmd->client, conn, channel_id); if (!channel) { silc_free(channel_id); COMMAND_REPLY_ERROR; goto out; } /* Get the ban list */ tmp = silc_argument_get_arg_type(cmd->args, 3, &len); /* Notify application */ COMMAND_REPLY((ARGS, channel, tmp)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_BAN); silc_client_command_reply_free(cmd); } /* Reply to LEAVE command. */ SILC_CLIENT_CMD_REPLY_FUNC(leave) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcChannelID *channel_id; SilcChannelEntry channel = NULL; unsigned char *tmp; SilcUInt32 len; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* From protocol version 1.1 we get the channel ID of the left channel */ tmp = silc_argument_get_arg_type(cmd->args, 2, &len); if (tmp) { channel_id = silc_id_payload_parse_id(tmp, len, NULL); if (!channel_id) goto out; /* Get the channel entry */ channel = silc_client_get_channel_by_id(cmd->client, conn, channel_id); if (!channel) { silc_free(channel_id); COMMAND_REPLY_ERROR; goto out; } silc_free(channel_id); } /* Notify application */ COMMAND_REPLY((ARGS, channel)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_LEAVE); silc_client_command_reply_free(cmd); } /* Channel resolving callback for USERS command reply. */ static void silc_client_command_reply_users_cb(SilcClient client, SilcClientConnection conn, SilcChannelEntry *channels, SilcUInt32 channels_count, void *context) { if (!channels_count) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; cmd->status = cmd->error = SILC_STATUS_ERR_NO_SUCH_CHANNEL; SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_USERS); silc_client_command_reply_free(cmd); return; } silc_client_command_reply_users(context, NULL); } /* Reply to USERS command. Received list of client ID's and theirs modes on the channel we requested. */ SILC_CLIENT_CMD_REPLY_FUNC(users) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcChannelEntry channel; SilcClientEntry client_entry; SilcChannelUser chu; SilcChannelID *channel_id = NULL; SilcBuffer client_id_list = NULL; SilcBuffer client_mode_list = NULL; unsigned char *tmp; SilcUInt32 tmp_len, list_count; int i; unsigned char **res_argv = NULL; SilcUInt32 *res_argv_lens = NULL, *res_argv_types = NULL, res_argc = 0; SILC_LOG_DEBUG(("Start")); if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Get channel ID */ tmp = silc_argument_get_arg_type(cmd->args, 2, &tmp_len); if (!tmp) { COMMAND_REPLY_ERROR; goto out; } channel_id = silc_id_payload_parse_id(tmp, tmp_len, NULL); if (!channel_id) { COMMAND_REPLY_ERROR; goto out; } /* Get the list count */ tmp = silc_argument_get_arg_type(cmd->args, 3, &tmp_len); if (!tmp) { COMMAND_REPLY_ERROR; goto out; } SILC_GET32_MSB(list_count, tmp); /* Get Client ID list */ tmp = silc_argument_get_arg_type(cmd->args, 4, &tmp_len); if (!tmp) { COMMAND_REPLY_ERROR; goto out; } client_id_list = silc_buffer_alloc(tmp_len); silc_buffer_pull_tail(client_id_list, tmp_len); silc_buffer_put(client_id_list, tmp, tmp_len); /* Get client mode list */ tmp = silc_argument_get_arg_type(cmd->args, 5, &tmp_len); if (!tmp) { COMMAND_REPLY_ERROR; goto out; } client_mode_list = silc_buffer_alloc(tmp_len); silc_buffer_pull_tail(client_mode_list, tmp_len); silc_buffer_put(client_mode_list, tmp, tmp_len); /* Get channel entry */ channel = silc_client_get_channel_by_id(cmd->client, conn, channel_id); if (!channel) { /* Resolve the channel from server */ silc_client_get_channel_by_id_resolve(cmd->client, conn, channel_id, silc_client_command_reply_users_cb, cmd); silc_free(channel_id); if (client_id_list) silc_buffer_free(client_id_list); if (client_mode_list) silc_buffer_free(client_mode_list); return; } /* Cache the received Client ID's and modes. */ for (i = 0; i < list_count; i++) { SilcUInt16 idp_len; SilcUInt32 mode; SilcClientID *client_id; /* Client ID */ SILC_GET16_MSB(idp_len, client_id_list->data + 2); idp_len += 4; client_id = silc_id_payload_parse_id(client_id_list->data, idp_len, NULL); if (!client_id) continue; /* Mode */ SILC_GET32_MSB(mode, client_mode_list->data); /* Check if we have this client cached already. */ client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); if (!client_entry || !client_entry->username || !client_entry->realname) { if (client_entry) { if (client_entry->status & SILC_CLIENT_STATUS_RESOLVING) { silc_buffer_pull(client_id_list, idp_len); silc_buffer_pull(client_mode_list, 4); continue; } client_entry->status |= SILC_CLIENT_STATUS_RESOLVING; } /* No we don't have it (or it is incomplete in information), query it from the server. Assemble argument table that will be sent for the WHOIS command later. */ res_argv = silc_realloc(res_argv, sizeof(*res_argv) * (res_argc + 1)); res_argv_lens = silc_realloc(res_argv_lens, sizeof(*res_argv_lens) * (res_argc + 1)); res_argv_types = silc_realloc(res_argv_types, sizeof(*res_argv_types) * (res_argc + 1)); res_argv[res_argc] = client_id_list->data; res_argv_lens[res_argc] = idp_len; res_argv_types[res_argc] = res_argc + 3; res_argc++; } else { if (!silc_client_on_channel(channel, client_entry)) { chu = silc_calloc(1, sizeof(*chu)); chu->client = client_entry; chu->mode = mode; chu->channel = channel; silc_hash_table_add(channel->user_list, client_entry, chu); silc_hash_table_add(client_entry->channels, channel, chu); } } silc_free(client_id); silc_buffer_pull(client_id_list, idp_len); silc_buffer_pull(client_mode_list, 4); } /* Query the client information from server if the list included clients that we don't know about. */ if (res_argc) { SilcBuffer res_cmd; /* Send the WHOIS command to server */ silc_client_command_register(cmd->client, SILC_COMMAND_WHOIS, NULL, NULL, silc_client_command_reply_whois_i, 0, ++conn->cmd_ident); res_cmd = silc_command_payload_encode(SILC_COMMAND_WHOIS, res_argc, res_argv, res_argv_lens, res_argv_types, conn->cmd_ident); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, res_cmd->data, res_cmd->len, TRUE); /* Register pending command callback. After we've received the WHOIS command reply we will reprocess this command reply by re-calling this USERS command reply callback. */ silc_client_command_pending(conn, SILC_COMMAND_WHOIS, conn->cmd_ident, silc_client_command_reply_users, cmd); silc_buffer_free(res_cmd); silc_free(channel_id); silc_free(res_argv); silc_free(res_argv_lens); silc_free(res_argv_types); if (client_id_list) silc_buffer_free(client_id_list); if (client_mode_list) silc_buffer_free(client_mode_list); return; } silc_buffer_push(client_id_list, (client_id_list->data - client_id_list->head)); silc_buffer_push(client_mode_list, (client_mode_list->data - client_mode_list->head)); /* Notify application */ COMMAND_REPLY((ARGS, channel, list_count, client_id_list, client_mode_list)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_USERS); silc_client_command_reply_free(cmd); silc_free(channel_id); if (client_id_list) silc_buffer_free(client_id_list); if (client_mode_list) silc_buffer_free(client_mode_list); } /* Received command reply to GETKEY command. WE've received the remote client's public key. */ SILC_CLIENT_CMD_REPLY_FUNC(getkey) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SilcIDPayload idp = NULL; SilcClientID *client_id = NULL; SilcClientEntry client_entry; SilcServerID *server_id = NULL; SilcServerEntry server_entry; SilcSKEPKType type; unsigned char *tmp, *pk; SilcUInt32 len; SilcUInt16 pk_len; SilcIdType id_type; SilcPublicKey public_key = NULL; SILC_LOG_DEBUG(("Start")); if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } tmp = silc_argument_get_arg_type(cmd->args, 2, &len); if (!tmp) { COMMAND_REPLY_ERROR; goto out; } idp = silc_id_payload_parse(tmp, len); if (!idp) { COMMAND_REPLY_ERROR; goto out; } /* Get the public key payload */ tmp = silc_argument_get_arg_type(cmd->args, 3, &len); if (tmp) { /* Decode the public key */ SILC_GET16_MSB(pk_len, tmp); SILC_GET16_MSB(type, tmp + 2); pk = tmp + 4; if (type == SILC_SKE_PK_TYPE_SILC) if (!silc_pkcs_public_key_decode(pk, pk_len, &public_key)) public_key = NULL; } id_type = silc_id_payload_get_type(idp); if (id_type == SILC_ID_CLIENT) { /* Received client's public key */ client_id = silc_id_payload_get_id(idp); client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); if (!client_entry) { COMMAND_REPLY_ERROR; goto out; } /* Notify application */ COMMAND_REPLY((ARGS, id_type, client_entry, public_key)); } else if (id_type == SILC_ID_SERVER) { /* Received server's public key */ server_id = silc_id_payload_get_id(idp); server_entry = silc_client_get_server_by_id(cmd->client, conn, server_id); if (!server_entry) { COMMAND_REPLY_ERROR; goto out; } /* Notify application */ COMMAND_REPLY((ARGS, id_type, server_entry, public_key)); } out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_GETKEY); if (idp) silc_id_payload_free(idp); if (public_key) silc_pkcs_public_key_free(public_key); silc_free(client_id); silc_free(server_id); silc_client_command_reply_free(cmd); } SILC_CLIENT_CMD_REPLY_FUNC(quit) { silc_client_command_reply_free(context); } /****************************************************************************** Internal command reply functions ******************************************************************************/ SILC_CLIENT_CMD_REPLY_FUNC(whois_i) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SILC_LOG_DEBUG(("Start")); if (cmd->error != SILC_STATUS_OK) goto out; /* Save WHOIS info */ silc_client_command_reply_whois_save(cmd, cmd->status, FALSE); /* Pending callbacks are not executed if this was an list entry */ if (cmd->status != SILC_STATUS_OK && cmd->status != SILC_STATUS_LIST_END) { silc_client_command_reply_free(cmd); return; } out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_WHOIS); /* If we received notify for invalid ID we'll remove the ID if we have it cached. */ if (cmd->error == SILC_STATUS_ERR_NO_SUCH_CLIENT_ID) { SilcClientEntry client_entry; SilcUInt32 tmp_len; unsigned char *tmp = silc_argument_get_arg_type(silc_command_get_args(cmd->payload), 2, &tmp_len); if (tmp) { SilcClientID *client_id = silc_id_payload_parse_id(tmp, tmp_len, NULL); if (client_id) { client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); if (client_entry) silc_client_del_client(cmd->client, conn, client_entry); silc_free(client_id); } } } /* Unregister this command reply */ silc_client_command_unregister(cmd->client, SILC_COMMAND_WHOIS, NULL, silc_client_command_reply_whois_i, cmd->ident); silc_client_command_reply_free(cmd); } SILC_CLIENT_CMD_REPLY_FUNC(identify_i) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; SILC_LOG_DEBUG(("Start")); if (cmd->error != SILC_STATUS_OK) goto out; /* Save IDENTIFY info */ silc_client_command_reply_identify_save(cmd, cmd->status, FALSE); /* Pending callbacks are not executed if this was an list entry */ if (cmd->status != SILC_STATUS_OK && cmd->status != SILC_STATUS_LIST_END) { silc_client_command_reply_free(cmd); return; } out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_IDENTIFY); /* If we received notify for invalid ID we'll remove the ID if we have it cached. */ if (cmd->error == SILC_STATUS_ERR_NO_SUCH_CLIENT_ID) { SilcClientEntry client_entry; SilcUInt32 tmp_len; unsigned char *tmp = silc_argument_get_arg_type(silc_command_get_args(cmd->payload), 2, &tmp_len); if (tmp) { SilcClientID *client_id = silc_id_payload_parse_id(tmp, tmp_len, NULL); if (client_id) { client_entry = silc_client_get_client_by_id(cmd->client, conn, client_id); if (client_entry) silc_client_del_client(cmd->client, conn, client_entry); silc_free(client_id); } } } /* Unregister this command reply */ silc_client_command_unregister(cmd->client, SILC_COMMAND_IDENTIFY, NULL, silc_client_command_reply_identify_i, cmd->ident); silc_client_command_reply_free(cmd); } SILC_CLIENT_CMD_REPLY_FUNC(info_i) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; unsigned char *tmp; SilcServerEntry server; SilcServerID *server_id = NULL; char *server_name, *server_info; SilcUInt32 len; SILC_LOG_DEBUG(("Start")); if (cmd->error != SILC_STATUS_OK) goto out; /* Get server ID */ tmp = silc_argument_get_arg_type(cmd->args, 2, &len); if (!tmp) goto out; server_id = silc_id_payload_parse_id(tmp, len, NULL); if (!server_id) goto out; /* Get server name */ server_name = silc_argument_get_arg_type(cmd->args, 3, NULL); if (!server_name) goto out; /* Get server info */ server_info = silc_argument_get_arg_type(cmd->args, 4, NULL); if (!server_info) goto out; /* See whether we have this server cached. If not create it. */ server = silc_client_get_server_by_id(cmd->client, conn, server_id); if (!server) { SILC_LOG_DEBUG(("New server entry")); silc_client_add_server(cmd->client, conn, server_name, server_info, silc_id_dup(server_id, SILC_ID_SERVER)); } out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_INFO); silc_free(server_id); silc_client_command_reply_free(cmd); } /* Private range commands, specific to this implementation (and compatible with SILC Server). */ SILC_CLIENT_CMD_REPLY_FUNC(connect) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Notify application */ COMMAND_REPLY((ARGS)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_PRIV_CONNECT); silc_client_command_reply_free(cmd); } SILC_CLIENT_CMD_REPLY_FUNC(close) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Notify application */ COMMAND_REPLY((ARGS)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_PRIV_CLOSE); silc_client_command_reply_free(cmd); } SILC_CLIENT_CMD_REPLY_FUNC(shutdown) { SilcClientCommandReplyContext cmd = (SilcClientCommandReplyContext)context; SilcClientConnection conn = (SilcClientConnection)cmd->sock->user_data; if (cmd->error != SILC_STATUS_OK) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_client_command_status_message(cmd->error)); COMMAND_REPLY_ERROR; goto out; } /* Notify application */ COMMAND_REPLY((ARGS)); out: SILC_CLIENT_PENDING_EXEC(cmd, SILC_COMMAND_PRIV_SHUTDOWN); silc_client_command_reply_free(cmd); }