/* command.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$ */ #include "silcincludes.h" #include "silcclient.h" #include "client_internal.h" #define SILC_NOT_CONNECTED(x, c) \ x->internal->ops->say((x), (c), SILC_CLIENT_MESSAGE_ERROR, \ "You are not connected to a server, use /SERVER to connect"); /* Command operation that is called at the end of all commands. Usage: COMMAND(status); */ #define COMMAND(status) cmd->client->internal->ops->command(cmd->client, \ cmd->conn, cmd, TRUE, cmd->command->cmd, (status)) /* Error to application. Usage: COMMAND_ERROR(status); */ #define COMMAND_ERROR(status) \ cmd->client->internal->ops->command(cmd->client, \ cmd->conn, cmd, FALSE, cmd->command->cmd, (status)) #define SAY cmd->client->internal->ops->say /* Generic function to send any command. The arguments must be sent already encoded into correct form and in correct order. */ void silc_client_command_send(SilcClient client, SilcClientConnection conn, SilcCommand command, SilcUInt16 ident, SilcUInt32 argc, ...) { SilcBuffer packet; va_list ap; va_start(ap, argc); packet = silc_command_payload_encode_vap(command, ident, argc, ap); silc_client_packet_send(client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, packet->data, packet->len, TRUE); silc_buffer_free(packet); } /* Finds and returns a pointer to the command list. Return NULL if the command is not found. */ SilcClientCommand silc_client_command_find(SilcClient client, const char *name) { SilcClientCommand cmd; silc_list_start(client->internal->commands); while ((cmd = silc_list_get(client->internal->commands)) != SILC_LIST_END) { if (cmd->name && !strcmp(cmd->name, name)) return cmd; } return NULL; } /* Calls the command (executes it). Application can call this after it has allocated the SilcClientCommandContext with the function silc_client_command_alloc and found the command from the client library by calling silc_client_command_find. This will execute the command. */ void silc_client_command_call(SilcClientCommand command, SilcClientCommandContext cmd) { (*command->command)((void *)cmd, NULL); } /* Add new pending command to be executed when reply to a command has been received. The `reply_cmd' is the command that will call the `callback' with `context' when reply has been received. It can be SILC_COMMAND_NONE to match any command with the `ident'. If `ident' is non-zero the `callback' will be executed when received reply with command identifier `ident'. If there already exists pending command for the specified command, ident, callback and context this function has no effect. */ void silc_client_command_pending(SilcClientConnection conn, SilcCommand reply_cmd, SilcUInt16 ident, SilcCommandCb callback, void *context) { SilcClientCommandPending *reply; /* Check whether identical pending already exists for same command, ident, callback and callback context. If it does then it would be error to register it again. */ silc_dlist_start(conn->pending_commands); while ((reply = silc_dlist_get(conn->pending_commands)) != SILC_LIST_END) { if (reply->reply_cmd == reply_cmd && reply->ident == ident && reply->callback == callback && reply->context == context) return; } reply = silc_calloc(1, sizeof(*reply)); reply->reply_cmd = reply_cmd; reply->ident = ident; reply->context = context; reply->callback = callback; silc_dlist_add(conn->pending_commands, reply); } /* Deletes pending command by reply command type. */ void silc_client_command_pending_del(SilcClientConnection conn, SilcCommand reply_cmd, SilcUInt16 ident) { SilcClientCommandPending *r; silc_dlist_start(conn->pending_commands); while ((r = silc_dlist_get(conn->pending_commands)) != SILC_LIST_END) { if (r->reply_cmd == reply_cmd && r->ident == ident) { silc_dlist_del(conn->pending_commands, r); break; } } } /* Checks for pending commands and marks callbacks to be called from the command reply function. */ SilcClientCommandPendingCallbacks silc_client_command_pending_check(SilcClientConnection conn, SilcClientCommandReplyContext ctx, SilcCommand command, SilcUInt16 ident, SilcUInt32 *callbacks_count) { SilcClientCommandPending *r; SilcClientCommandPendingCallbacks callbacks = NULL; int i = 0; silc_dlist_start(conn->pending_commands); while ((r = silc_dlist_get(conn->pending_commands)) != SILC_LIST_END) { if ((r->reply_cmd == command || r->reply_cmd == SILC_COMMAND_NONE) && r->ident == ident) { callbacks = silc_realloc(callbacks, sizeof(*callbacks) * (i + 1)); callbacks[i].context = r->context; callbacks[i].callback = r->callback; ctx->ident = ident; i++; } } *callbacks_count = i; return callbacks; } /* Allocate Command Context */ SilcClientCommandContext silc_client_command_alloc(void) { SilcClientCommandContext ctx = silc_calloc(1, sizeof(*ctx)); ctx->users++; return ctx; } /* Free command context and its internals */ void silc_client_command_free(SilcClientCommandContext ctx) { ctx->users--; SILC_LOG_DEBUG(("Command context %p refcnt %d->%d", ctx, ctx->users + 1, ctx->users)); if (ctx->users < 1) { int i; for (i = 0; i < ctx->argc; i++) silc_free(ctx->argv[i]); silc_free(ctx->argv_lens); silc_free(ctx->argv_types); silc_free(ctx); } } /* Duplicate Command Context by adding reference counter. The context won't be free'd untill it hits zero. */ SilcClientCommandContext silc_client_command_dup(SilcClientCommandContext ctx) { ctx->users++; SILC_LOG_DEBUG(("Command context %p refcnt %d->%d", ctx, ctx->users - 1, ctx->users)); return ctx; } /* Command WHOIS. This command is used to query information about specific user. */ SILC_CLIENT_CMD_FUNC(whois) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; unsigned char count[4]; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } /* Given without arguments fetches client's own information */ if (cmd->argc < 2) { buffer = silc_id_payload_encode(cmd->conn->local_id, SILC_ID_CLIENT); silc_client_command_send(cmd->client, cmd->conn, SILC_COMMAND_WHOIS, ++conn->cmd_ident, 1, 3, buffer->data, buffer->len); silc_buffer_free(buffer); goto out; } if (cmd->argc == 2) { buffer = silc_command_payload_encode_va(SILC_COMMAND_WHOIS, ++conn->cmd_ident, 1, 1, cmd->argv[1], cmd->argv_lens[1]); } else { int c = atoi(cmd->argv[2]); memset(count, 0, sizeof(count)); SILC_PUT32_MSB(c, count); buffer = silc_command_payload_encode_va(SILC_COMMAND_WHOIS, ++conn->cmd_ident, 2, 1, cmd->argv[1], cmd->argv_lens[1], 2, count, sizeof(count)); } silc_client_packet_send(cmd->client, cmd->conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Command WHOWAS. This command is used to query history information about specific user that used to exist in the network. */ SILC_CLIENT_CMD_FUNC(whowas) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; unsigned char count[4]; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2 || cmd->argc > 3) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /WHOWAS [@] []"); COMMAND_ERROR((cmd->argc < 2 ? SILC_STATUS_ERR_NOT_ENOUGH_PARAMS : SILC_STATUS_ERR_TOO_MANY_PARAMS)); goto out; } if (cmd->argc == 2) { buffer = silc_command_payload_encode_va(SILC_COMMAND_WHOWAS, ++conn->cmd_ident, 1, 1, cmd->argv[1], cmd->argv_lens[1]); } else { int c = atoi(cmd->argv[2]); memset(count, 0, sizeof(count)); SILC_PUT32_MSB(c, count); buffer = silc_command_payload_encode_va(SILC_COMMAND_WHOWAS, ++conn->cmd_ident, 2, 1, cmd->argv[1], cmd->argv_lens[1], 2, count, sizeof(count)); } silc_client_packet_send(cmd->client, cmd->conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Command IDENTIFY. This command is used to query information about specific user, especially ID's. NOTE: This command is used only internally by the client library and application MUST NOT call this command directly. */ SILC_CLIENT_CMD_FUNC(identify) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; unsigned char count[4]; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2 || cmd->argc > 3) goto out; if (cmd->argc == 2) { buffer = silc_command_payload_encode_va(SILC_COMMAND_IDENTIFY, ++conn->cmd_ident, 1, 1, cmd->argv[1], cmd->argv_lens[1]); } else { int c = atoi(cmd->argv[2]); memset(count, 0, sizeof(count)); SILC_PUT32_MSB(c, count); buffer = silc_command_payload_encode_va(SILC_COMMAND_IDENTIFY, ++conn->cmd_ident, 2, 1, cmd->argv[1], cmd->argv_lens[1], 4, count, sizeof(count)); } silc_client_packet_send(cmd->client, cmd->conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); out: silc_client_command_free(cmd); } /* Pending callbcak that will be called after the NICK command was replied by the server. This sets the nickname if there were no errors. */ SILC_CLIENT_CMD_FUNC(nick_change) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcClientCommandReplyContext reply = (SilcClientCommandReplyContext)context2; SilcStatus status; silc_command_get_status(reply->payload, &status, NULL); if (status == SILC_STATUS_OK) { /* Set the nickname */ silc_idcache_del_by_context(conn->client_cache, conn->local_entry); if (conn->nickname) silc_free(conn->nickname); conn->nickname = strdup(cmd->argv[1]); conn->local_entry->nickname = conn->nickname; silc_client_nickname_format(cmd->client, conn, conn->local_entry); silc_idcache_add(conn->client_cache, strdup(cmd->argv[1]), conn->local_entry->id, conn->local_entry, 0, NULL); COMMAND(SILC_STATUS_OK); } else { COMMAND_ERROR(status); } silc_client_command_free(cmd); } /* Command NICK. Shows current nickname/sets new nickname on current window. */ SILC_CLIENT_CMD_FUNC(nick) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /NICK "); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (!strcmp(conn->nickname, cmd->argv[1])) goto out; /* Show current nickname */ if (cmd->argc < 2) { if (cmd->conn) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Your nickname is %s on server %s", conn->nickname, conn->remote_host); } else { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Your nickname is %s", conn->nickname); } COMMAND(SILC_STATUS_OK); goto out; } if (cmd->argv_lens[1] > 128) cmd->argv_lens[1] = 128; /* Send the NICK command */ buffer = silc_command_payload_encode(SILC_COMMAND_NICK, 1, &cmd->argv[1], &cmd->argv_lens[1], &cmd->argv_types[1], ++cmd->conn->cmd_ident); silc_client_packet_send(cmd->client, cmd->conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); /* Register pending callback that will actually set the new nickname if there were no errors returned by the server. */ silc_client_command_pending(conn, SILC_COMMAND_NICK, cmd->conn->cmd_ident, silc_client_command_nick_change, silc_client_command_dup(cmd)); cmd->pending = TRUE; out: silc_client_command_free(cmd); } /* Command LIST. Lists channels on the current server. */ SILC_CLIENT_CMD_FUNC(list) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcIDCacheEntry id_cache = NULL; SilcChannelEntry channel; SilcBuffer buffer, idp = NULL; char *name; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc == 2) { name = cmd->argv[1]; /* Get the Channel ID of the channel */ if (silc_idcache_find_by_name_one(conn->channel_cache, name, &id_cache)) { channel = (SilcChannelEntry)id_cache->context; idp = silc_id_payload_encode(id_cache->id, SILC_ID_CHANNEL); } } if (!idp) buffer = silc_command_payload_encode_va(SILC_COMMAND_LIST, ++conn->cmd_ident, 0); else buffer = silc_command_payload_encode_va(SILC_COMMAND_LIST, ++conn->cmd_ident, 1, 1, idp->data, idp->len); silc_client_packet_send(cmd->client, cmd->conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); if (idp) silc_buffer_free(idp); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Command TOPIC. Sets/shows topic on a channel. */ SILC_CLIENT_CMD_FUNC(topic) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcIDCacheEntry id_cache = NULL; SilcChannelEntry channel; SilcBuffer buffer, idp; char *name; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2 || cmd->argc > 3) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /TOPIC []"); COMMAND_ERROR((cmd->argc < 2 ? SILC_STATUS_ERR_NOT_ENOUGH_PARAMS : SILC_STATUS_ERR_TOO_MANY_PARAMS)); goto out; } if (cmd->argv[1][0] == '*') { if (!conn->current_channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on any channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } name = conn->current_channel->channel_name; } else { name = cmd->argv[1]; } if (!conn->current_channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on that channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } /* Get the Channel ID of the channel */ if (!silc_idcache_find_by_name_one(conn->channel_cache, name, &id_cache)) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on that channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } channel = (SilcChannelEntry)id_cache->context; /* Send TOPIC command to the server */ idp = silc_id_payload_encode(id_cache->id, SILC_ID_CHANNEL); if (cmd->argc > 2) buffer = silc_command_payload_encode_va(SILC_COMMAND_TOPIC, ++conn->cmd_ident, 2, 1, idp->data, idp->len, 2, cmd->argv[2], strlen(cmd->argv[2])); else buffer = silc_command_payload_encode_va(SILC_COMMAND_TOPIC, ++conn->cmd_ident, 1, 1, idp->data, idp->len); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(idp); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Command INVITE. Invites specific client to join a channel. This is also used to mange the invite list of the channel. */ SILC_CLIENT_CMD_FUNC(invite) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClient client = cmd->client; SilcClientConnection conn = cmd->conn; SilcClientEntry client_entry = NULL; SilcChannelEntry channel; SilcBuffer buffer, clidp, chidp; SilcUInt32 type = 0; char *nickname = NULL, *name; char *invite = NULL; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /INVITE [[@server>]" "[+|-[[@[![@hostname>]]]]]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argv[1][0] == '*') { if (!conn->current_channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on any channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } channel = conn->current_channel; } else { name = cmd->argv[1]; channel = silc_client_get_channel(cmd->client, conn, name); if (!channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are on that channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } } /* Parse the typed nickname. */ if (cmd->argc == 3) { if (cmd->argv[2][0] != '+' && cmd->argv[2][0] != '-') { if (client->internal->params->nickname_parse) client->internal->params->nickname_parse(cmd->argv[2], &nickname); else nickname = strdup(cmd->argv[2]); /* Find client entry */ client_entry = silc_idlist_get_client(client, conn, nickname, cmd->argv[2], TRUE); if (!client_entry) { if (cmd->pending) { COMMAND_ERROR(SILC_STATUS_ERR_NO_SUCH_NICK); goto out; } /* Client entry not found, it was requested thus mark this to be pending command. */ silc_client_command_pending(conn, SILC_COMMAND_IDENTIFY, conn->cmd_ident, silc_client_command_invite, silc_client_command_dup(cmd)); cmd->pending = 1; goto out; } } else { invite = cmd->argv[2]; invite++; if (cmd->argv[2][0] == '+') type = 3; else type = 4; } } /* Send the command */ chidp = silc_id_payload_encode(channel->id, SILC_ID_CHANNEL); if (client_entry) { clidp = silc_id_payload_encode(client_entry->id, SILC_ID_CLIENT); buffer = silc_command_payload_encode_va(SILC_COMMAND_INVITE, ++conn->cmd_ident, 3, 1, chidp->data, chidp->len, 2, clidp->data, clidp->len, type, invite, invite ? strlen(invite) : 0); silc_buffer_free(clidp); } else { buffer = silc_command_payload_encode_va(SILC_COMMAND_INVITE, ++conn->cmd_ident, 2, 1, chidp->data, chidp->len, type, invite, invite ? strlen(invite) : 0); } silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(chidp); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_free(nickname); silc_client_command_free(cmd); } typedef struct { SilcClient client; SilcClientConnection conn; } *QuitInternal; SILC_TASK_CALLBACK(silc_client_command_quit_cb) { QuitInternal q = (QuitInternal)context; /* Close connection */ q->client->internal->ops->disconnect(q->client, q->conn); silc_client_close_connection(q->client, q->conn->sock->user_data); silc_free(q); } /* Command QUIT. Closes connection with current server. */ SILC_CLIENT_CMD_FUNC(quit) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcBuffer buffer; QuitInternal q; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc > 1) buffer = silc_command_payload_encode(SILC_COMMAND_QUIT, cmd->argc - 1, &cmd->argv[1], &cmd->argv_lens[1], &cmd->argv_types[1], 0); else buffer = silc_command_payload_encode(SILC_COMMAND_QUIT, 0, NULL, NULL, NULL, 0); silc_client_packet_send(cmd->client, cmd->conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); q = silc_calloc(1, sizeof(*q)); q->client = cmd->client; q->conn = cmd->conn; /* Sleep for a while */ sleep(2); /* We quit the connection with little timeout */ silc_schedule_task_add(cmd->client->schedule, cmd->conn->sock->sock, silc_client_command_quit_cb, (void *)q, 1, 0, SILC_TASK_TIMEOUT, SILC_TASK_PRI_NORMAL); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Timeout callback to remove the killed client from cache */ SILC_TASK_CALLBACK(silc_client_command_kill_remove_later) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClient client = cmd->client; SilcClientConnection conn = cmd->conn; SilcClientEntry target; char *nickname = NULL; /* Parse the typed nickname. */ if (client->internal->params->nickname_parse) client->internal->params->nickname_parse(cmd->argv[1], &nickname); else nickname = strdup(cmd->argv[1]); /* Get the target client */ target = silc_idlist_get_client(cmd->client, conn, nickname, cmd->argv[1], FALSE); if (target) /* Remove the client from all channels and free it */ silc_client_del_client(client, conn, target); silc_free(nickname); silc_client_command_free(cmd); } /* Kill command's pending command callback to actually remove the killed client from our local cache. */ SILC_CLIENT_CMD_FUNC(kill_remove) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientCommandReplyContext reply = (SilcClientCommandReplyContext)context2; SilcStatus status; silc_command_get_status(reply->payload, &status, NULL); if (status == SILC_STATUS_OK) { /* Remove with timeout */ silc_schedule_task_add(cmd->client->schedule, cmd->conn->sock->sock, silc_client_command_kill_remove_later, context, 1, 0, SILC_TASK_TIMEOUT, SILC_TASK_PRI_NORMAL); return; } silc_client_command_free(cmd); } /* Command KILL. Router operator can use this command to remove an client fromthe SILC Network. */ SILC_CLIENT_CMD_FUNC(kill) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClient client = cmd->client; SilcClientConnection conn = cmd->conn; SilcBuffer buffer, idp; SilcClientEntry target; char *nickname = NULL; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /KILL []"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } /* Parse the typed nickname. */ if (client->internal->params->nickname_parse) client->internal->params->nickname_parse(cmd->argv[1], &nickname); else nickname = strdup(cmd->argv[1]); /* Get the target client */ target = silc_idlist_get_client(cmd->client, conn, nickname, cmd->argv[1], TRUE); if (!target) { if (cmd->pending) { COMMAND_ERROR(SILC_STATUS_ERR_NO_SUCH_NICK); goto out; } /* Client entry not found, it was requested thus mark this to be pending command. */ silc_client_command_pending(conn, SILC_COMMAND_IDENTIFY, conn->cmd_ident, silc_client_command_kill, silc_client_command_dup(cmd)); cmd->pending = 1; goto out; } /* Send the KILL command to the server */ idp = silc_id_payload_encode(target->id, SILC_ID_CLIENT); if (cmd->argc == 2) buffer = silc_command_payload_encode_va(SILC_COMMAND_KILL, ++conn->cmd_ident, 1, 1, idp->data, idp->len); else buffer = silc_command_payload_encode_va(SILC_COMMAND_KILL, ++conn->cmd_ident, 2, 1, idp->data, idp->len, 2, cmd->argv[2], strlen(cmd->argv[2])); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(idp); /* Notify application */ COMMAND(SILC_STATUS_OK); /* Register a pending callback that will actually remove the killed client from our cache. */ silc_client_command_pending(conn, SILC_COMMAND_KILL, conn->cmd_ident, silc_client_command_kill_remove, silc_client_command_dup(cmd)); out: silc_free(nickname); silc_client_command_free(cmd); } /* Command INFO. Request information about specific server. If specific server is not provided the current server is used. */ SILC_CLIENT_CMD_FUNC(info) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; char *name = NULL; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc == 2) name = strdup(cmd->argv[1]); /* Send the command */ if (name) buffer = silc_command_payload_encode_va(SILC_COMMAND_INFO, 0, 1, 1, name, strlen(name)); else buffer = silc_command_payload_encode(SILC_COMMAND_INFO, 0, NULL, NULL, NULL, 0); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); if (name) silc_free(name); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Command PING. Sends ping to server. This is used to test the communication channel. */ SILC_CLIENT_CMD_FUNC(ping) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; void *id; int i; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } /* Send the command */ buffer = silc_command_payload_encode_va(SILC_COMMAND_PING, 0, 1, 1, conn->remote_id_data, silc_id_get_len(conn->remote_id, SILC_ID_SERVER)); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); id = silc_id_str2id(conn->remote_id_data, conn->remote_id_data_len, SILC_ID_SERVER); if (!id) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } /* Start counting time */ for (i = 0; i < conn->ping_count; i++) { if (conn->ping[i].dest_id == NULL) { conn->ping[i].start_time = time(NULL); conn->ping[i].dest_id = id; conn->ping[i].dest_name = strdup(conn->remote_host); break; } } if (i >= conn->ping_count) { i = conn->ping_count; conn->ping = silc_realloc(conn->ping, sizeof(*conn->ping) * (i + 1)); conn->ping[i].start_time = time(NULL); conn->ping[i].dest_id = id; conn->ping[i].dest_name = strdup(conn->remote_host); conn->ping_count++; } /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Command JOIN. Joins to a channel. */ SILC_CLIENT_CMD_FUNC(join) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcChannelEntry channel; SilcBuffer buffer, idp, auth = NULL; char *name, *passphrase = NULL, *pu8, *cipher = NULL, *hmac = NULL; int i, passphrase_len = 0; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } /* See if we have joined to the requested channel already */ channel = silc_client_get_channel(cmd->client, conn, cmd->argv[1]); if (channel && silc_client_on_channel(channel, conn->local_entry)) goto out; idp = silc_id_payload_encode(conn->local_id, SILC_ID_CLIENT); if (cmd->argv_lens[1] > 256) cmd->argv_lens[1] = 256; name = cmd->argv[1]; for (i = 2; i < cmd->argc; i++) { if (!strcasecmp(cmd->argv[i], "-cipher") && cmd->argc > i + 1) { cipher = cmd->argv[i + 1]; i++; } else if (!strcasecmp(cmd->argv[i], "-hmac") && cmd->argc > i + 1) { hmac = cmd->argv[i + 1]; i++; } else if (!strcasecmp(cmd->argv[i], "-founder") && cmd->argc > i + 1) { if (!strcasecmp(cmd->argv[i + 1], "-pubkey")) { auth = silc_auth_public_key_auth_generate(cmd->client->public_key, cmd->client->private_key, cmd->client->rng, conn->hash, conn->local_id, SILC_ID_CLIENT); } else { auth = silc_auth_payload_encode(SILC_AUTH_PASSWORD, NULL, 0, cmd->argv[i + 1], cmd->argv_lens[i + 1]); } i++; } else { /* Passphrases must be UTF-8 encoded, so encode if it is not */ if (!silc_utf8_valid(cmd->argv[i], cmd->argv_lens[i])) { passphrase_len = silc_utf8_encoded_len(cmd->argv[i], cmd->argv_lens[i], 0); pu8 = silc_calloc(passphrase_len, sizeof(*pu8)); passphrase_len = silc_utf8_encode(cmd->argv[i], cmd->argv_lens[i], 0, pu8, passphrase_len); passphrase = pu8; } else { passphrase = strdup(cmd->argv[i]); passphrase_len = cmd->argv_lens[i]; } } } /* Send JOIN command to the server */ buffer = silc_command_payload_encode_va(SILC_COMMAND_JOIN, 0, 6, 1, name, strlen(name), 2, idp->data, idp->len, 3, passphrase, passphrase_len, 4, cipher, cipher ? strlen(cipher) : 0, 5, hmac, hmac ? strlen(hmac) : 0, 6, auth ? auth->data : NULL, auth ? auth->len : 0); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(idp); if (auth) silc_buffer_free(auth); silc_free(passphrase); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* MOTD command. Requests motd from server. */ SILC_CLIENT_CMD_FUNC(motd) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 1 || cmd->argc > 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /MOTD []"); COMMAND_ERROR((cmd->argc < 1 ? SILC_STATUS_ERR_NOT_ENOUGH_PARAMS : SILC_STATUS_ERR_TOO_MANY_PARAMS)); goto out; } /* Send TOPIC command to the server */ if (cmd->argc == 1) buffer = silc_command_payload_encode_va(SILC_COMMAND_MOTD, 0, 1, 1, conn->remote_host, strlen(conn->remote_host)); else buffer = silc_command_payload_encode_va(SILC_COMMAND_MOTD, 0, 1, 1, cmd->argv[1], cmd->argv_lens[1]); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* UMODE. Set/unset user mode in SILC. This is used mainly to unset the modes as client cannot set itself server/router operator privileges. */ SILC_CLIENT_CMD_FUNC(umode) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer, idp; unsigned char *cp, modebuf[4]; SilcUInt32 mode, add, len; int i; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /UMODE +|-"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } mode = conn->local_entry->mode; /* Are we adding or removing mode */ if (cmd->argv[1][0] == '-') add = FALSE; else add = TRUE; /* Parse mode */ cp = cmd->argv[1] + 1; len = strlen(cp); for (i = 0; i < len; i++) { switch(cp[i]) { case 'a': if (add) { mode = 0; mode |= SILC_UMODE_SERVER_OPERATOR; mode |= SILC_UMODE_ROUTER_OPERATOR; mode |= SILC_UMODE_GONE; mode |= SILC_UMODE_INDISPOSED; mode |= SILC_UMODE_BUSY; mode |= SILC_UMODE_PAGE; mode |= SILC_UMODE_HYPER; mode |= SILC_UMODE_ROBOT; mode |= SILC_UMODE_BLOCK_PRIVMSG; mode |= SILC_UMODE_REJECT_WATCHING; } else { mode = SILC_UMODE_NONE; } break; case 's': if (add) mode |= SILC_UMODE_SERVER_OPERATOR; else mode &= ~SILC_UMODE_SERVER_OPERATOR; break; case 'r': if (add) mode |= SILC_UMODE_ROUTER_OPERATOR; else mode &= ~SILC_UMODE_ROUTER_OPERATOR; break; case 'g': if (add) mode |= SILC_UMODE_GONE; else mode &= ~SILC_UMODE_GONE; break; case 'i': if (add) mode |= SILC_UMODE_INDISPOSED; else mode &= ~SILC_UMODE_INDISPOSED; break; case 'b': if (add) mode |= SILC_UMODE_BUSY; else mode &= ~SILC_UMODE_BUSY; break; case 'p': if (add) mode |= SILC_UMODE_PAGE; else mode &= ~SILC_UMODE_PAGE; break; case 'h': if (add) mode |= SILC_UMODE_HYPER; else mode &= ~SILC_UMODE_HYPER; break; case 't': if (add) mode |= SILC_UMODE_ROBOT; else mode &= ~SILC_UMODE_ROBOT; break; case 'P': if (add) mode |= SILC_UMODE_BLOCK_PRIVMSG; else mode &= ~SILC_UMODE_BLOCK_PRIVMSG; break; case 'w': if (add) mode |= SILC_UMODE_REJECT_WATCHING; else mode &= ~SILC_UMODE_REJECT_WATCHING; break; default: COMMAND_ERROR(SILC_STATUS_ERR_UNKNOWN_MODE); goto out; break; } } idp = silc_id_payload_encode(conn->local_id, SILC_ID_CLIENT); SILC_PUT32_MSB(mode, modebuf); /* Send the command packet. We support sending only one mode at once that requires an argument. */ buffer = silc_command_payload_encode_va(SILC_COMMAND_UMODE, ++conn->cmd_ident, 2, 1, idp->data, idp->len, 2, modebuf, sizeof(modebuf)); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(idp); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* CMODE command. Sets channel mode. Modes that does not require any arguments can be set several at once. Those modes that require argument must be set separately (unless set with modes that does not require arguments). */ SILC_CLIENT_CMD_FUNC(cmode) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcChannelEntry channel; SilcBuffer buffer, chidp, auth = NULL; unsigned char *name, *cp, modebuf[4], tmp[4], *arg = NULL; SilcUInt32 mode, add, type, len, arg_len = 0; int i; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 3) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /CMODE +|- [{ }]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argv[1][0] == '*') { if (!conn->current_channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on any channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } channel = conn->current_channel; } else { name = cmd->argv[1]; channel = silc_client_get_channel(cmd->client, conn, name); if (!channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are on that channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } } mode = channel->mode; /* Are we adding or removing mode */ if (cmd->argv[2][0] == '-') add = FALSE; else add = TRUE; /* Argument type to be sent to server */ type = 0; /* Parse mode */ cp = cmd->argv[2] + 1; len = strlen(cp); for (i = 0; i < len; i++) { switch(cp[i]) { case 'p': if (add) mode |= SILC_CHANNEL_MODE_PRIVATE; else mode &= ~SILC_CHANNEL_MODE_PRIVATE; break; case 's': if (add) mode |= SILC_CHANNEL_MODE_SECRET; else mode &= ~SILC_CHANNEL_MODE_SECRET; break; case 'k': if (add) mode |= SILC_CHANNEL_MODE_PRIVKEY; else mode &= ~SILC_CHANNEL_MODE_PRIVKEY; break; case 'i': if (add) mode |= SILC_CHANNEL_MODE_INVITE; else mode &= ~SILC_CHANNEL_MODE_INVITE; break; case 't': if (add) mode |= SILC_CHANNEL_MODE_TOPIC; else mode &= ~SILC_CHANNEL_MODE_TOPIC; break; case 'm': if (add) mode |= SILC_CHANNEL_MODE_SILENCE_USERS; else mode &= ~SILC_CHANNEL_MODE_SILENCE_USERS; break; case 'M': if (add) mode |= SILC_CHANNEL_MODE_SILENCE_OPERS; else mode &= ~SILC_CHANNEL_MODE_SILENCE_OPERS; break; case 'l': if (add) { int ll; mode |= SILC_CHANNEL_MODE_ULIMIT; type = 3; if (cmd->argc < 4) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /CMODE +|- [{ }]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } ll = atoi(cmd->argv[3]); SILC_PUT32_MSB(ll, tmp); arg = tmp; arg_len = 4; } else { mode &= ~SILC_CHANNEL_MODE_ULIMIT; } break; case 'a': if (add) { mode |= SILC_CHANNEL_MODE_PASSPHRASE; type = 4; if (cmd->argc < 4) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /CMODE +|- [{ }]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } arg = cmd->argv[3]; arg_len = cmd->argv_lens[3]; } else { mode &= ~SILC_CHANNEL_MODE_PASSPHRASE; } break; case 'c': if (add) { mode |= SILC_CHANNEL_MODE_CIPHER; type = 5; if (cmd->argc < 4) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /CMODE +|- [{ }]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } arg = cmd->argv[3]; arg_len = cmd->argv_lens[3]; } else { mode &= ~SILC_CHANNEL_MODE_CIPHER; } break; case 'h': if (add) { mode |= SILC_CHANNEL_MODE_HMAC; type = 6; if (cmd->argc < 4) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /CMODE +|- [{ }]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } arg = cmd->argv[3]; arg_len = cmd->argv_lens[3]; } else { mode &= ~SILC_CHANNEL_MODE_HMAC; } break; case 'f': if (add) { mode |= SILC_CHANNEL_MODE_FOUNDER_AUTH; type = 7; if (cmd->argc < 4) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /CMODE +|- [{ }]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (!strcasecmp(cmd->argv[3], "-pubkey")) { auth = silc_auth_public_key_auth_generate(cmd->client->public_key, cmd->client->private_key, cmd->client->rng, conn->hash, conn->local_id, SILC_ID_CLIENT); } else { auth = silc_auth_payload_encode(SILC_AUTH_PASSWORD, NULL, 0, cmd->argv[3], cmd->argv_lens[3]); } arg = auth->data; arg_len = auth->len; } else { mode &= ~SILC_CHANNEL_MODE_FOUNDER_AUTH; } break; default: COMMAND_ERROR(SILC_STATUS_ERR_UNKNOWN_MODE); goto out; break; } } chidp = silc_id_payload_encode(channel->id, SILC_ID_CHANNEL); SILC_PUT32_MSB(mode, modebuf); /* Send the command packet. We support sending only one mode at once that requires an argument. */ if (type && arg) { buffer = silc_command_payload_encode_va(SILC_COMMAND_CMODE, 0, 3, 1, chidp->data, chidp->len, 2, modebuf, sizeof(modebuf), type, arg, arg_len); } else { buffer = silc_command_payload_encode_va(SILC_COMMAND_CMODE, 0, 2, 1, chidp->data, chidp->len, 2, modebuf, sizeof(modebuf)); } silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(chidp); if (auth) silc_buffer_free(auth); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* CUMODE command. Changes client's mode on a channel. */ SILC_CLIENT_CMD_FUNC(cumode) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClient client = cmd->client; SilcClientConnection conn = cmd->conn; SilcChannelEntry channel; SilcChannelUser chu; SilcClientEntry client_entry; SilcBuffer buffer, clidp, chidp, auth = NULL; unsigned char *name, *cp, modebuf[4]; SilcUInt32 mode = 0, add, len; char *nickname = NULL; int i; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 4) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /CUMODE +|- [@]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argv[1][0] == '*') { if (!conn->current_channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on any channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } channel = conn->current_channel; } else { name = cmd->argv[1]; channel = silc_client_get_channel(cmd->client, conn, name); if (!channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are on that channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } } /* Parse the typed nickname. */ if (client->internal->params->nickname_parse) client->internal->params->nickname_parse(cmd->argv[3], &nickname); else nickname = strdup(cmd->argv[3]); /* Find client entry */ client_entry = silc_idlist_get_client(cmd->client, conn, nickname, cmd->argv[3], TRUE); if (!client_entry) { if (cmd->pending) { COMMAND_ERROR(SILC_STATUS_ERR_NO_SUCH_NICK); goto out; } /* Client entry not found, it was requested thus mark this to be pending command. */ silc_client_command_pending(conn, SILC_COMMAND_IDENTIFY, conn->cmd_ident, silc_client_command_cumode, silc_client_command_dup(cmd)); cmd->pending = 1; goto out; } /* Get the current mode */ chu = silc_client_on_channel(channel, client_entry); if (chu) mode = chu->mode; /* Are we adding or removing mode */ if (cmd->argv[2][0] == '-') add = FALSE; else add = TRUE; /* Parse mode */ cp = cmd->argv[2] + 1; len = strlen(cp); for (i = 0; i < len; i++) { switch(cp[i]) { case 'a': if (add) { mode |= SILC_CHANNEL_UMODE_CHANFO; mode |= SILC_CHANNEL_UMODE_CHANOP; mode |= SILC_CHANNEL_UMODE_BLOCK_MESSAGES; mode |= SILC_CHANNEL_UMODE_BLOCK_MESSAGES_USERS; mode |= SILC_CHANNEL_UMODE_BLOCK_MESSAGES_ROBOTS; } else { mode = SILC_CHANNEL_UMODE_NONE; } break; case 'f': if (add) { if (cmd->argc == 5) { if (!strcasecmp(cmd->argv[4], "-pubkey")) { auth = silc_auth_public_key_auth_generate(cmd->client->public_key, cmd->client->private_key, cmd->client->rng, conn->hash, conn->local_id, SILC_ID_CLIENT); } else { auth = silc_auth_payload_encode(SILC_AUTH_PASSWORD, NULL, 0, cmd->argv[4], cmd->argv_lens[4]); } } mode |= SILC_CHANNEL_UMODE_CHANFO; } else { mode &= ~SILC_CHANNEL_UMODE_CHANFO; } break; case 'o': if (add) mode |= SILC_CHANNEL_UMODE_CHANOP; else mode &= ~SILC_CHANNEL_UMODE_CHANOP; break; case 'b': if (add) mode |= SILC_CHANNEL_UMODE_BLOCK_MESSAGES; else mode &= ~SILC_CHANNEL_UMODE_BLOCK_MESSAGES; break; case 'u': if (add) mode |= SILC_CHANNEL_UMODE_BLOCK_MESSAGES_USERS; else mode &= ~SILC_CHANNEL_UMODE_BLOCK_MESSAGES_USERS; break; case 'r': if (add) mode |= SILC_CHANNEL_UMODE_BLOCK_MESSAGES_ROBOTS; else mode &= ~SILC_CHANNEL_UMODE_BLOCK_MESSAGES_ROBOTS; break; default: COMMAND_ERROR(SILC_STATUS_ERR_UNKNOWN_MODE); goto out; break; } } chidp = silc_id_payload_encode(channel->id, SILC_ID_CHANNEL); SILC_PUT32_MSB(mode, modebuf); clidp = silc_id_payload_encode(client_entry->id, SILC_ID_CLIENT); /* Send the command packet. We support sending only one mode at once that requires an argument. */ buffer = silc_command_payload_encode_va(SILC_COMMAND_CUMODE, 0, auth ? 4 : 3, 1, chidp->data, chidp->len, 2, modebuf, 4, 3, clidp->data, clidp->len, 4, auth ? auth->data : NULL, auth ? auth->len : 0); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(chidp); silc_buffer_free(clidp); if (auth) silc_buffer_free(auth); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_free(nickname); silc_client_command_free(cmd); } /* KICK command. Kicks a client out of channel. */ SILC_CLIENT_CMD_FUNC(kick) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClient client = cmd->client; SilcClientConnection conn = cmd->conn; SilcIDCacheEntry id_cache = NULL; SilcChannelEntry channel; SilcBuffer buffer, idp, idp2; SilcClientEntry target; char *name; char *nickname = NULL; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 3) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /KICK []"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argv[1][0] == '*') { if (!conn->current_channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on any channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } name = conn->current_channel->channel_name; } else { name = cmd->argv[1]; } if (!conn->current_channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on that channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } /* Get the Channel ID of the channel */ if (!silc_idcache_find_by_name_one(conn->channel_cache, name, &id_cache)) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on that channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } channel = (SilcChannelEntry)id_cache->context; /* Parse the typed nickname. */ if (client->internal->params->nickname_parse) client->internal->params->nickname_parse(cmd->argv[2], &nickname); else nickname = strdup(cmd->argv[2]); /* Get the target client */ target = silc_idlist_get_client(cmd->client, conn, nickname, cmd->argv[2], FALSE); if (!target) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "No such client: %s", cmd->argv[2]); COMMAND_ERROR(SILC_STATUS_ERR_NO_SUCH_NICK); goto out; } /* Send KICK command to the server */ idp = silc_id_payload_encode(id_cache->id, SILC_ID_CHANNEL); idp2 = silc_id_payload_encode(target->id, SILC_ID_CLIENT); if (cmd->argc == 3) buffer = silc_command_payload_encode_va(SILC_COMMAND_KICK, 0, 2, 1, idp->data, idp->len, 2, idp2->data, idp2->len); else buffer = silc_command_payload_encode_va(SILC_COMMAND_KICK, 0, 3, 1, idp->data, idp->len, 2, idp2->data, idp2->len, 3, cmd->argv[3], strlen(cmd->argv[3])); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(idp); silc_buffer_free(idp2); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_free(nickname); silc_client_command_free(cmd); } static void silc_client_command_oper_send(unsigned char *data, SilcUInt32 data_len, void *context) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer, auth; if (cmd->argc >= 3) { /* Encode the public key authentication payload */ auth = silc_auth_public_key_auth_generate(cmd->client->public_key, cmd->client->private_key, cmd->client->rng, conn->hash, conn->local_id, SILC_ID_CLIENT); } else { /* Encode the password authentication payload */ auth = silc_auth_payload_encode(SILC_AUTH_PASSWORD, NULL, 0, data, data_len); } buffer = silc_command_payload_encode_va(SILC_COMMAND_OPER, 0, 2, 1, cmd->argv[1], strlen(cmd->argv[1]), 2, auth->data, auth->len); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(auth); /* Notify application */ COMMAND(SILC_STATUS_OK); } /* OPER command. Used to obtain server operator privileges. */ SILC_CLIENT_CMD_FUNC(oper) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /OPER [-pubkey]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argc < 3) { /* Get passphrase */ cmd->client->internal->ops->ask_passphrase(cmd->client, conn, silc_client_command_oper_send, context); return; } silc_client_command_oper_send(NULL, 0, context); out: silc_client_command_free(cmd); } static void silc_client_command_silcoper_send(unsigned char *data, SilcUInt32 data_len, void *context) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer, auth; if (cmd->argc >= 3) { /* Encode the public key authentication payload */ auth = silc_auth_public_key_auth_generate(cmd->client->public_key, cmd->client->private_key, cmd->client->rng, conn->hash, conn->local_id, SILC_ID_CLIENT); } else { /* Encode the password authentication payload */ auth = silc_auth_payload_encode(SILC_AUTH_PASSWORD, NULL, 0, data, data_len); } buffer = silc_command_payload_encode_va(SILC_COMMAND_SILCOPER, 0, 2, 1, cmd->argv[1], strlen(cmd->argv[1]), 2, auth->data, auth->len); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(auth); /* Notify application */ COMMAND(SILC_STATUS_OK); } /* SILCOPER command. Used to obtain router operator privileges. */ SILC_CLIENT_CMD_FUNC(silcoper) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /SILCOPER [-pubkey]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argc < 3) { /* Get passphrase */ cmd->client->internal->ops->ask_passphrase(cmd->client, conn, silc_client_command_silcoper_send, context); return; } silc_client_command_silcoper_send(NULL, 0, context); out: silc_client_command_free(cmd); } /* Command BAN. This is used to manage the ban list of the channel. */ SILC_CLIENT_CMD_FUNC(ban) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcChannelEntry channel; SilcBuffer buffer, chidp; int type = 0; char *name, *ban = NULL; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /BAN " "[+|-[[@[![@hostname>]]]]]"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argv[1][0] == '*') { if (!conn->current_channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on any channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } channel = conn->current_channel; } else { name = cmd->argv[1]; channel = silc_client_get_channel(cmd->client, conn, name); if (!channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are noton that channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } } if (cmd->argc == 3) { if (cmd->argv[2][0] == '+') type = 2; else type = 3; ban = cmd->argv[2]; ban++; } chidp = silc_id_payload_encode(channel->id, SILC_ID_CHANNEL); /* Send the command */ buffer = silc_command_payload_encode_va(SILC_COMMAND_BAN, ++conn->cmd_ident, 2, 1, chidp->data, chidp->len, type, ban, ban ? strlen(ban) : 0); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(chidp); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Command DETACH. This is used to detach from the server */ SILC_CLIENT_CMD_FUNC(detach) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } buffer = silc_command_payload_encode_va(SILC_COMMAND_DETACH, ++conn->cmd_ident, 0); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Command WATCH. */ SILC_CLIENT_CMD_FUNC(watch) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer, idp = NULL; int type = 0; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 3) { COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } idp = silc_id_payload_encode(conn->local_id, SILC_ID_CLIENT); if (!strcasecmp(cmd->argv[1], "-add")) { type = 2; } else if (!strcasecmp(cmd->argv[1], "-del")) { type = 3; } else { COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } buffer = silc_command_payload_encode_va(SILC_COMMAND_WATCH, ++conn->cmd_ident, 2, 1, idp->data, idp->len, type, cmd->argv[2], cmd->argv_lens[2]); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); /* Notify application */ COMMAND(SILC_STATUS_OK); out: if (idp) silc_buffer_free(idp); silc_client_command_free(cmd); } /* LEAVE command. Leaves a channel. Client removes itself from a channel. */ SILC_CLIENT_CMD_FUNC(leave) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcChannelEntry channel; SilcChannelUser chu; SilcBuffer buffer, idp; char *name; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc != 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /LEAVE "); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argv[1][0] == '*') { if (!conn->current_channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on any channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } name = conn->current_channel->channel_name; } else { name = cmd->argv[1]; } /* Get the channel entry */ channel = silc_client_get_channel(cmd->client, conn, name); if (!channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on that channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } /* Remove us from channel */ chu = silc_client_on_channel(channel, conn->local_entry); if (chu) { silc_hash_table_del(chu->client->channels, chu->channel); silc_hash_table_del(chu->channel->user_list, chu->client); silc_free(chu); } /* Send LEAVE command to the server */ idp = silc_id_payload_encode(channel->id, SILC_ID_CHANNEL); buffer = silc_command_payload_encode_va(SILC_COMMAND_LEAVE, 0, 1, 1, idp->data, idp->len); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(idp); /* Notify application */ COMMAND(SILC_STATUS_OK); if (conn->current_channel == channel) conn->current_channel = NULL; silc_client_del_channel(cmd->client, cmd->conn, channel); out: silc_client_command_free(cmd); } /* Command USERS. Requests the USERS of the clients joined on requested channel. */ SILC_CLIENT_CMD_FUNC(users) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; char *name; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc != 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /USERS "); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argv[1][0] == '*') { if (!conn->current_channel) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "You are not on any channel"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ON_CHANNEL); goto out; } name = conn->current_channel->channel_name; } else { name = cmd->argv[1]; } /* Send USERS command to the server */ buffer = silc_command_payload_encode_va(SILC_COMMAND_USERS, ++conn->cmd_ident, 1, 2, name, strlen(name)); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Command GETKEY. Used to fetch remote client's public key. */ SILC_CLIENT_CMD_FUNC(getkey) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcClient client = cmd->client; SilcClientEntry client_entry = NULL; SilcServerEntry server_entry = NULL; char *nickname = NULL; SilcBuffer idp, buffer; SILC_LOG_DEBUG(("Start")); if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { client->internal->ops->say(client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /GETKEY "); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } /* Parse the typed nickname. */ if (client->internal->params->nickname_parse) client->internal->params->nickname_parse(cmd->argv[1], &nickname); else nickname = strdup(cmd->argv[1]); /* Find client entry */ client_entry = silc_idlist_get_client(client, conn, nickname, cmd->argv[1], FALSE); if (!client_entry) { /* Check whether user requested server actually */ server_entry = silc_client_get_server(client, conn, cmd->argv[1]); if (!server_entry) { /* No. what ever user wants we don't have it, so resolve it. We will first try to resolve the client, and if that fails then we'll try to resolve the server. */ if (!cmd->pending) { /* This will send the IDENTIFY command for nickname */ silc_idlist_get_client(client, conn, nickname, cmd->argv[1], TRUE); silc_client_command_pending(conn, SILC_COMMAND_IDENTIFY, conn->cmd_ident, silc_client_command_getkey, silc_client_command_dup(cmd)); cmd->pending = 1; goto out; } else { SilcClientCommandReplyContext reply = (SilcClientCommandReplyContext)context2; SilcStatus error; /* If nickname was not found, then resolve the server. */ silc_command_get_status(reply->payload, NULL, &error); if (error == SILC_STATUS_ERR_NO_SUCH_NICK) { /* This sends the IDENTIFY command to resolve the server. */ silc_client_command_register(client, SILC_COMMAND_IDENTIFY, NULL, NULL, silc_client_command_reply_identify_i, 0, ++conn->cmd_ident); silc_client_command_send(client, conn, SILC_COMMAND_IDENTIFY, conn->cmd_ident, 1, 2, cmd->argv[1], cmd->argv_lens[1]); silc_client_command_pending(conn, SILC_COMMAND_IDENTIFY, conn->cmd_ident, silc_client_command_getkey, silc_client_command_dup(cmd)); goto out; } /* If server was not found, then we've resolved both nickname and server and did not find anybody. */ if (error == SILC_STATUS_ERR_NO_SUCH_SERVER) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_get_status_message(SILC_STATUS_ERR_NO_SUCH_NICK)); SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_ERROR, "%s", silc_get_status_message(error)); COMMAND_ERROR(SILC_STATUS_ERR_NO_SUCH_NICK); goto out; } COMMAND_ERROR(error); goto out; } } idp = silc_id_payload_encode(server_entry->server_id, SILC_ID_SERVER); } else { idp = silc_id_payload_encode(client_entry->id, SILC_ID_CLIENT); } buffer = silc_command_payload_encode_va(SILC_COMMAND_GETKEY, 0, 1, 1, idp->data, idp->len); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); silc_buffer_free(idp); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_free(nickname); silc_client_command_free(cmd); } /* Register a new command indicated by the `command' to the SILC client. The `name' is optional command name. If provided the command may be searched using the silc_client_command_find by that name. The `command_function' is the function to be called when the command is executed, and the `command_reply_function' is the function to be called after the server has sent reply back to the command. The `ident' is optional identifier for the command. If non-zero the `command_reply_function' for the command type `command' will be called only if the command reply sent by server includes the command identifier `ident'. Application usually does not need it and set it to zero value. */ bool silc_client_command_register(SilcClient client, SilcCommand command, const char *name, SilcCommandCb command_function, SilcCommandCb command_reply_function, SilcUInt8 max_args, SilcUInt16 ident) { SilcClientCommand cmd; cmd = silc_calloc(1, sizeof(*cmd)); cmd->cmd = command; cmd->command = command_function; cmd->reply = command_reply_function; cmd->name = name ? strdup(name) : NULL; cmd->max_args = max_args; cmd->ident = ident; silc_list_add(client->internal->commands, cmd); return TRUE; } /* Unregister a command indicated by the `command' with command function `command_function' and command reply function `command_reply_function'. Returns TRUE if the command was found and unregistered. */ bool silc_client_command_unregister(SilcClient client, SilcCommand command, SilcCommandCb command_function, SilcCommandCb command_reply_function, SilcUInt16 ident) { SilcClientCommand cmd; silc_list_start(client->internal->commands); while ((cmd = silc_list_get(client->internal->commands)) != SILC_LIST_END) { if (cmd->cmd == command && cmd->command == command_function && cmd->reply == command_reply_function && cmd->ident == ident) { silc_list_del(client->internal->commands, cmd); silc_free(cmd->name); silc_free(cmd); return TRUE; } } return FALSE; } /* Private range commands, specific to this implementation (and compatible with SILC Server). */ /* CONNECT command. Connects the server to another server. */ SILC_CLIENT_CMD_FUNC(connect) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; unsigned char port[4]; SilcUInt32 tmp; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /CONNECT []"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argc == 3) { tmp = atoi(cmd->argv[2]); SILC_PUT32_MSB(tmp, port); } if (cmd->argc == 3) buffer = silc_command_payload_encode_va(SILC_COMMAND_PRIV_CONNECT, 0, 2, 1, cmd->argv[1], strlen(cmd->argv[1]), 2, port, 4); else buffer = silc_command_payload_encode_va(SILC_COMMAND_PRIV_CONNECT, 0, 1, 1, cmd->argv[1], strlen(cmd->argv[1])); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* CLOSE command. Close server connection to the remote server */ SILC_CLIENT_CMD_FUNC(close) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; SilcClientConnection conn = cmd->conn; SilcBuffer buffer; unsigned char port[4]; SilcUInt32 tmp; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } if (cmd->argc < 2) { SAY(cmd->client, conn, SILC_CLIENT_MESSAGE_INFO, "Usage: /CLOSE []"); COMMAND_ERROR(SILC_STATUS_ERR_NOT_ENOUGH_PARAMS); goto out; } if (cmd->argc == 3) { tmp = atoi(cmd->argv[2]); SILC_PUT32_MSB(tmp, port); } if (cmd->argc == 3) buffer = silc_command_payload_encode_va(SILC_COMMAND_PRIV_CLOSE, 0, 2, 1, cmd->argv[1], strlen(cmd->argv[1]), 2, port, 4); else buffer = silc_command_payload_encode_va(SILC_COMMAND_PRIV_CLOSE, 0, 1, 1, cmd->argv[1], strlen(cmd->argv[1])); silc_client_packet_send(cmd->client, conn->sock, SILC_PACKET_COMMAND, NULL, 0, NULL, NULL, buffer->data, buffer->len, TRUE); silc_buffer_free(buffer); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* SHUTDOWN command. Shutdowns the server. */ SILC_CLIENT_CMD_FUNC(shutdown) { SilcClientCommandContext cmd = (SilcClientCommandContext)context; if (!cmd->conn) { SILC_NOT_CONNECTED(cmd->client, cmd->conn); COMMAND_ERROR(SILC_STATUS_ERR_NOT_REGISTERED); goto out; } /* Send the command */ silc_client_command_send(cmd->client, cmd->conn, SILC_COMMAND_PRIV_SHUTDOWN, 0, 0); /* Notify application */ COMMAND(SILC_STATUS_OK); out: silc_client_command_free(cmd); } /* Register all default commands provided by the client library for the application. */ void silc_client_commands_register(SilcClient client) { silc_list_init(client->internal->commands, struct SilcClientCommandStruct, next); SILC_CLIENT_CMD(whois, WHOIS, "WHOIS", 3); SILC_CLIENT_CMD(whowas, WHOWAS, "WHOWAS", 3); SILC_CLIENT_CMD(identify, IDENTIFY, "IDENTIFY", 3); SILC_CLIENT_CMD(nick, NICK, "NICK", 2); SILC_CLIENT_CMD(list, LIST, "LIST", 2); SILC_CLIENT_CMD(topic, TOPIC, "TOPIC", 3); SILC_CLIENT_CMD(invite, INVITE, "INVITE", 3); SILC_CLIENT_CMD(quit, QUIT, "QUIT", 2); SILC_CLIENT_CMD(kill, KILL, "KILL", 3); SILC_CLIENT_CMD(info, INFO, "INFO", 2); SILC_CLIENT_CMD(ping, PING, "PING", 2); SILC_CLIENT_CMD(oper, OPER, "OPER", 3); SILC_CLIENT_CMD(join, JOIN, "JOIN", 9); SILC_CLIENT_CMD(motd, MOTD, "MOTD", 2); SILC_CLIENT_CMD(umode, UMODE, "UMODE", 2); SILC_CLIENT_CMD(cmode, CMODE, "CMODE", 4); SILC_CLIENT_CMD(cumode, CUMODE, "CUMODE", 5); SILC_CLIENT_CMD(kick, KICK, "KICK", 4); SILC_CLIENT_CMD(ban, BAN, "BAN", 3); SILC_CLIENT_CMD(detach, DETACH, "DETACH", 0); SILC_CLIENT_CMD(watch, WATCH, "WATCH", 3); SILC_CLIENT_CMD(silcoper, SILCOPER, "SILCOPER", 3); SILC_CLIENT_CMD(leave, LEAVE, "LEAVE", 2); SILC_CLIENT_CMD(users, USERS, "USERS", 2); SILC_CLIENT_CMD(getkey, GETKEY, "GETKEY", 2); SILC_CLIENT_CMD(connect, PRIV_CONNECT, "CONNECT", 3); SILC_CLIENT_CMD(close, PRIV_CLOSE, "CLOSE", 3); SILC_CLIENT_CMD(shutdown, PRIV_SHUTDOWN, "SHUTDOWN", 1); } /* Unregister all commands. */ void silc_client_commands_unregister(SilcClient client) { SILC_CLIENT_CMDU(whois, WHOIS, "WHOIS"); SILC_CLIENT_CMDU(whowas, WHOWAS, "WHOWAS"); SILC_CLIENT_CMDU(identify, IDENTIFY, "IDENTIFY"); SILC_CLIENT_CMDU(nick, NICK, "NICK"); SILC_CLIENT_CMDU(list, LIST, "LIST"); SILC_CLIENT_CMDU(topic, TOPIC, "TOPIC"); SILC_CLIENT_CMDU(invite, INVITE, "INVITE"); SILC_CLIENT_CMDU(quit, QUIT, "QUIT"); SILC_CLIENT_CMDU(kill, KILL, "KILL"); SILC_CLIENT_CMDU(info, INFO, "INFO"); SILC_CLIENT_CMDU(ping, PING, "PING"); SILC_CLIENT_CMDU(oper, OPER, "OPER"); SILC_CLIENT_CMDU(join, JOIN, "JOIN"); SILC_CLIENT_CMDU(motd, MOTD, "MOTD"); SILC_CLIENT_CMDU(umode, UMODE, "UMODE"); SILC_CLIENT_CMDU(cmode, CMODE, "CMODE"); SILC_CLIENT_CMDU(cumode, CUMODE, "CUMODE"); SILC_CLIENT_CMDU(kick, KICK, "KICK"); SILC_CLIENT_CMDU(ban, BAN, "BAN"); SILC_CLIENT_CMDU(detach, DETACH, "DETACH"); SILC_CLIENT_CMDU(watch, WATCH, "WATCH"); SILC_CLIENT_CMDU(silcoper, SILCOPER, "SILCOPER"); SILC_CLIENT_CMDU(leave, LEAVE, "LEAVE"); SILC_CLIENT_CMDU(users, USERS, "USERS"); SILC_CLIENT_CMDU(getkey, GETKEY, "GETKEY"); SILC_CLIENT_CMDU(connect, PRIV_CONNECT, "CONNECT"); SILC_CLIENT_CMDU(close, PRIV_CLOSE, "CLOSE"); SILC_CLIENT_CMDU(shutdown, PRIV_SHUTDOWN, "SHUTDOWN"); }