summaryrefslogtreecommitdiff
path: root/src/irc/dcc
diff options
context:
space:
mode:
Diffstat (limited to 'src/irc/dcc')
-rw-r--r--src/irc/dcc/Makefile.am7
-rw-r--r--src/irc/dcc/dcc-autoget.c84
-rw-r--r--src/irc/dcc/dcc-chat.c186
-rw-r--r--src/irc/dcc/dcc-chat.h11
-rw-r--r--src/irc/dcc/dcc-files.c667
-rw-r--r--src/irc/dcc/dcc-get.c385
-rw-r--r--src/irc/dcc/dcc-get.h28
-rw-r--r--src/irc/dcc/dcc-resume.c177
-rw-r--r--src/irc/dcc/dcc-send.c267
-rw-r--r--src/irc/dcc/dcc.c543
-rw-r--r--src/irc/dcc/dcc.h38
11 files changed, 1380 insertions, 1013 deletions
diff --git a/src/irc/dcc/Makefile.am b/src/irc/dcc/Makefile.am
index 91250034..d8274173 100644
--- a/src/irc/dcc/Makefile.am
+++ b/src/irc/dcc/Makefile.am
@@ -9,8 +9,13 @@ INCLUDES = $(GLIB_CFLAGS) \
libirc_dcc_la_SOURCES = \
dcc.c \
dcc-chat.c \
- dcc-files.c
+ dcc-get.c \
+ dcc-send.c \
+ dcc-resume.c \
+ dcc-autoget.c
noinst_HEADERS = \
dcc.h \
+ dcc-chat.h \
+ dcc-get.h \
module.h
diff --git a/src/irc/dcc/dcc-autoget.c b/src/irc/dcc/dcc-autoget.c
new file mode 100644
index 00000000..cb25ecf8
--- /dev/null
+++ b/src/irc/dcc/dcc-autoget.c
@@ -0,0 +1,84 @@
+/*
+ dcc-autoget.c : irssi
+
+ Copyright (C) 1999-2001 Timo Sirainen
+
+ 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; either version 2 of the License, or
+ (at your option) any later version.
+
+ 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.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "module.h"
+#include "signals.h"
+#include "masks.h"
+#include "settings.h"
+
+#include "dcc-get.h"
+
+static void sig_dcc_request(DCC_REC *dcc, const char *nickaddr)
+{
+ struct stat statbuf;
+ const char *masks;
+ char *str, *file;
+ int max_size;
+
+ g_return_if_fail(dcc != NULL);
+ if (dcc->type != DCC_TYPE_GET) return;
+
+ /* check if we want to autoget file offer */
+ if (!settings_get_bool("dcc_autoget") &&
+ !settings_get_bool("dcc_autoresume"))
+ return;
+
+ /* check for lowports */
+ if (dcc->port < 1024 && !settings_get_bool("dcc_autoget_lowports"))
+ return;
+
+ /* check that autoget masks match */
+ masks = settings_get_str("dcc_autoget_masks");
+ if (*masks != '\0' &&
+ !masks_match(SERVER(dcc->server), masks, dcc->nick, nickaddr))
+ return;
+
+ /* check file size limit, NOTE: it's still possible to send a
+ bogus file size and then just send what ever sized file.. */
+ max_size = settings_get_int("dcc_autoget_max_size");
+ if (max_size > 0 && max_size*1024 < dcc->size)
+ return;
+
+ /* ok. but do we want/need to resume? */
+ file = dcc_get_download_path(dcc->arg);
+ str = g_strdup_printf(settings_get_bool("dcc_autoresume") &&
+ stat(file, &statbuf) == 0 ?
+ "RESUME %s %s" : "GET %s %s",
+ dcc->nick, dcc->arg);
+ signal_emit("command dcc", 2, str, dcc->server);
+ g_free(file);
+ g_free(str);
+}
+
+void dcc_autoget_init(void)
+{
+ settings_add_bool("dcc", "dcc_autoget", FALSE);
+ settings_add_bool("dcc", "dcc_autoget_lowports", FALSE);
+ settings_add_bool("dcc", "dcc_autoresume", FALSE);
+ settings_add_int("dcc", "dcc_autoget_max_size", 1000);
+ settings_add_str("dcc", "dcc_autoget_masks", "");
+
+ signal_add_last("dcc request", (SIGNAL_FUNC) sig_dcc_request);
+}
+
+void dcc_autoget_deinit(void)
+{
+ signal_remove("dcc request", (SIGNAL_FUNC) sig_dcc_request);
+}
diff --git a/src/irc/dcc/dcc-chat.c b/src/irc/dcc/dcc-chat.c
index 0084cee8..6fa9b19e 100644
--- a/src/irc/dcc/dcc-chat.c
+++ b/src/irc/dcc/dcc-chat.c
@@ -25,6 +25,7 @@
#include "net-nonblock.h"
#include "net-sendbuffer.h"
#include "line-split.h"
+#include "misc.h"
#include "settings.h"
#include "masks.h"
@@ -34,6 +35,45 @@
#include "dcc.h"
+DCC_REC *dcc_chat_find_id(const char *id)
+{
+ GSList *tmp;
+
+ g_return_val_if_fail(id != NULL, NULL);
+
+ for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next) {
+ DCC_REC *dcc = tmp->data;
+
+ if (dcc->type == DCC_TYPE_CHAT && dcc->chat_id != NULL &&
+ g_strcasecmp(dcc->chat_id, id) == 0)
+ return dcc;
+ }
+
+ return NULL;
+}
+
+static void dcc_chat_set_id(DCC_REC *dcc)
+{
+ char *id;
+ int num;
+
+ if (dcc_chat_find_id(dcc->nick) == NULL) {
+ /* same as nick, good */
+ dcc->chat_id = g_strdup(dcc->nick);
+ return;
+ }
+
+ /* keep adding numbers after nick until some of them isn't found */
+ for (num = 2;; num++) {
+ id = g_strdup_printf("%s%d", dcc->nick, num);
+ if (dcc_chat_find_id(id) == NULL) {
+ dcc->chat_id = id;
+ break;
+ }
+ g_free(id);
+ }
+}
+
/* Send `data' to dcc chat. */
void dcc_chat_send(DCC_REC *dcc, const char *data)
{
@@ -54,7 +94,7 @@ DCC_REC *item_get_dcc(WI_ITEM_REC *item)
if (query == NULL || *query->name != '=')
return NULL;
- return dcc_find_item(DCC_TYPE_CHAT, query->name+1, NULL);
+ return dcc_chat_find_id(query->name+1);
}
/* Send text to DCC chat */
@@ -74,7 +114,7 @@ static void cmd_msg(const char *data)
if (!cmd_get_params(data, &free_arg, 2 | PARAM_FLAG_GETREST, &target, &text))
return;
- dcc = dcc_find_item(DCC_TYPE_CHAT, ++target, NULL);
+ dcc = dcc_chat_find_id(++target);
if (dcc != NULL && dcc->sendbuf != NULL)
dcc_chat_send(dcc, text);
@@ -92,7 +132,7 @@ static void cmd_me(const char *data, IRC_SERVER_REC *server, QUERY_REC *item)
dcc = item_get_dcc((WI_ITEM_REC *) item);
if (dcc == NULL) return;
- str = g_strdup_printf("ACTION %s", data);
+ str = g_strconcat("ACTION ", data, NULL);
dcc_ctcp_message(NULL, dcc->nick, dcc, FALSE, str);
g_free(str);
@@ -116,9 +156,9 @@ static void cmd_action(const char *data, IRC_SERVER_REC *server)
return;
if (*target == '\0' || *text == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
- dcc = dcc_find_item(DCC_TYPE_CHAT, target+1, NULL);
+ dcc = dcc_chat_find_id(target+1);
if (dcc != NULL) {
- str = g_strdup_printf("ACTION %s", text);
+ str = g_strconcat("ACTION ", text, NULL);
dcc_ctcp_message(NULL, dcc->nick, dcc, FALSE, str);
g_free(str);
}
@@ -146,11 +186,11 @@ static void cmd_ctcp(const char *data, IRC_SERVER_REC *server)
return;
}
- dcc = dcc_find_item(DCC_TYPE_CHAT, target+1, NULL);
+ dcc = dcc_chat_find_id(target+1);
if (dcc != NULL) {
g_strup(ctcpcmd);
- str = g_strdup_printf("%s %s", ctcpcmd, ctcpdata);
+ str = g_strconcat(ctcpcmd, " ", ctcpdata, NULL);
dcc_ctcp_message(NULL, dcc->nick, dcc, FALSE, str);
g_free(str);
}
@@ -262,53 +302,64 @@ static void dcc_chat_connect(DCC_REC *dcc)
}
}
-/* SYNTAX: DCC CHAT <nick> */
+/* SYNTAX: DCC CHAT [<nick>] */
static void cmd_dcc_chat(const char *data, IRC_SERVER_REC *server)
{
void *free_arg;
DCC_REC *dcc;
IPADDR own_ip;
GIOChannel *handle;
- char *nick, *str, host[MAX_IP_LEN];
+ char *nick, host[MAX_IP_LEN];
int port;
g_return_if_fail(data != NULL);
if (!cmd_get_params(data, &free_arg, 1, &nick))
return;
- if (*nick == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
- dcc = dcc_find_item(DCC_TYPE_CHAT, nick, NULL);
- if (dcc != NULL) {
- /* found from dcc list - so we're the connecting side.. */
+ if (*nick == '\0') {
+ dcc = dcc_find_request_latest(DCC_TYPE_CHAT);
+ if (dcc != NULL)
+ dcc_chat_connect(dcc);
+ cmd_params_free(free_arg);
+ return;
+ }
+
+ dcc = dcc_chat_find_id(nick);
+ if (dcc != NULL && dcc_is_waiting_user(dcc)) {
+ /* found from dcc chat requests,
+ we're the connecting side */
dcc_chat_connect(dcc);
cmd_params_free(free_arg);
return;
}
- /* send dcc chat request */
+ if (dcc != NULL && dcc_is_listening(dcc) &&
+ dcc->server == server) {
+ /* sending request again even while old request is
+ still waiting, remove it. */
+ dcc_destroy(dcc);
+ }
+
+ /* start listening */
if (server == NULL || !server->connected)
cmd_param_error(CMDERR_NOT_CONNECTED);
- if (net_getsockname(net_sendbuffer_handle(server->handle),
- &own_ip, NULL) == -1)
- cmd_param_error(CMDERR_ERRNO);
-
- port = settings_get_int("dcc_port");
- handle = net_listen(&own_ip, &port);
+ handle = dcc_listen(net_sendbuffer_handle(server->handle),
+ &own_ip, &port);
if (handle == NULL)
cmd_param_error(CMDERR_ERRNO);
- dcc = dcc_create(DCC_TYPE_CHAT, handle, nick, "chat", server, NULL);
+ dcc = dcc_create(DCC_TYPE_CHAT, nick, "chat", server, NULL);
+ dcc_chat_set_id(dcc);
+ dcc->handle = handle;
dcc->tagconn = g_input_add(dcc->handle, G_INPUT_READ,
(GInputFunction) dcc_chat_listen, dcc);
- /* send the request */
+ /* send the chat request */
dcc_make_address(&own_ip, host);
- str = g_strdup_printf("PRIVMSG %s :\001DCC CHAT CHAT %s %d\001",
- nick, host, port);
- irc_send_cmd(server, str);
- g_free(str);
+ irc_send_cmdv(server, "PRIVMSG %s :\001DCC CHAT CHAT %s %d\001",
+ nick, host, port);
cmd_params_free(free_arg);
}
@@ -328,6 +379,63 @@ static void cmd_mircdcc(const char *data, IRC_SERVER_REC *server,
g_strncasecmp(data, "OF", 3) != 0;
}
+#define DCC_AUTOACCEPT_PORT(dcc) \
+ ((dcc)->port >= 1024 || settings_get_bool("dcc_autoaccept_lowport"))
+
+#define DCC_CHAT_AUTOACCEPT(dcc, server, nick, addr) \
+ (DCC_AUTOACCEPT_PORT(dcc) && \
+ masks_match(SERVER(server), \
+ settings_get_str("dcc_autochat_masks"), (nick), (addr)))
+
+
+/* CTCP: DCC CHAT */
+static void ctcp_msg_dcc_chat(IRC_SERVER_REC *server, const char *data,
+ const char *nick, const char *addr,
+ const char *target, DCC_REC *chat)
+{
+ DCC_REC *dcc;
+ char **params;
+ int paramcount;
+ int autoallow = FALSE;
+
+ /* CHAT <unused> <address> <port> */
+ params = g_strsplit(data, " ", -1);
+ paramcount = strarray_length(params);
+
+ if (paramcount < 3) {
+ g_strfreev(params);
+ return;
+ }
+
+ dcc = dcc_find_request(DCC_TYPE_CHAT, nick, NULL);
+ if (dcc != NULL) {
+ if (dcc_is_listening(dcc)) {
+ /* we requested dcc chat, they requested
+ dcc chat from us .. allow it. */
+ dcc_destroy(dcc);
+ autoallow = TRUE;
+ } else {
+ /* we already have one dcc chat request
+ from this nick, remove it. */
+ dcc_destroy(dcc);
+ }
+ }
+
+ dcc = dcc_create(DCC_TYPE_CHAT, nick, params[0], server, chat);
+ dcc_chat_set_id(dcc);
+ dcc->target = g_strdup(target);
+ dcc->port = atoi(params[2]);
+ dcc_get_address(params[1], &dcc->addr);
+ net_ip2host(&dcc->addr, dcc->addrstr);
+
+ signal_emit("dcc request", 2, dcc, addr);
+
+ if (autoallow || DCC_CHAT_AUTOACCEPT(dcc, server, nick, addr))
+ dcc_chat_connect(dcc);
+
+ g_strfreev(params);
+}
+
/* DCC CHAT: text received */
static void dcc_chat_msg(DCC_REC *dcc, const char *msg)
{
@@ -364,31 +472,47 @@ static void dcc_chat_msg(DCC_REC *dcc, const char *msg)
if (ptr != NULL) *ptr++ = '\0'; else ptr = "";
g_strdown(cmd+9);
- if (!signal_emit(cmd, 2, ptr, dcc))
- signal_emit(reply ? "default dcc reply" : "default dcc ctcp", 2, msg, dcc);
+ if (!signal_emit(cmd, 2, ptr, dcc)) {
+ signal_emit(reply ? "default dcc reply" :
+ "default dcc ctcp", 2, msg, dcc);
+ }
g_free(cmd);
signal_stop();
}
-static void dcc_ctcp_redirect(gchar *msg, DCC_REC *dcc)
+static void dcc_ctcp_redirect(const char *msg, DCC_REC *dcc)
{
g_return_if_fail(msg != NULL);
g_return_if_fail(dcc != NULL);
- signal_emit("ctcp msg dcc", 6, dcc->server, msg, dcc->nick, "dcc", dcc->mynick, dcc);
+ signal_emit("ctcp msg dcc", 6, dcc->server, msg,
+ dcc->nick, "dcc", dcc->mynick, dcc);
+}
+
+static void dcc_ctcp_reply_redirect(const char *msg, DCC_REC *dcc)
+{
+ g_return_if_fail(msg != NULL);
+ g_return_if_fail(dcc != NULL);
+
+ signal_emit("ctcp reply dcc", 6, dcc->server, msg,
+ dcc->nick, "dcc", dcc->mynick, dcc);
}
void dcc_chat_init(void)
{
+ settings_add_str("dcc", "dcc_autochat_masks", "");
+
command_bind("msg", NULL, (SIGNAL_FUNC) cmd_msg);
command_bind("me", NULL, (SIGNAL_FUNC) cmd_me);
command_bind("action", NULL, (SIGNAL_FUNC) cmd_action);
command_bind("ctcp", NULL, (SIGNAL_FUNC) cmd_ctcp);
command_bind("dcc chat", NULL, (SIGNAL_FUNC) cmd_dcc_chat);
command_bind("mircdcc", NULL, (SIGNAL_FUNC) cmd_mircdcc);
+ signal_add("ctcp msg dcc chat", (SIGNAL_FUNC) ctcp_msg_dcc_chat);
signal_add_first("dcc chat message", (SIGNAL_FUNC) dcc_chat_msg);
signal_add("dcc ctcp dcc", (SIGNAL_FUNC) dcc_ctcp_redirect);
+ signal_add("dcc reply dcc", (SIGNAL_FUNC) dcc_ctcp_reply_redirect);
}
void dcc_chat_deinit(void)
@@ -399,6 +523,8 @@ void dcc_chat_deinit(void)
command_unbind("ctcp", (SIGNAL_FUNC) cmd_ctcp);
command_unbind("dcc chat", (SIGNAL_FUNC) cmd_dcc_chat);
command_unbind("mircdcc", (SIGNAL_FUNC) cmd_mircdcc);
+ signal_remove("ctcp msg dcc chat", (SIGNAL_FUNC) ctcp_msg_dcc_chat);
signal_remove("dcc chat message", (SIGNAL_FUNC) dcc_chat_msg);
signal_remove("dcc ctcp dcc", (SIGNAL_FUNC) dcc_ctcp_redirect);
+ signal_remove("dcc reply dcc", (SIGNAL_FUNC) dcc_ctcp_reply_redirect);
}
diff --git a/src/irc/dcc/dcc-chat.h b/src/irc/dcc/dcc-chat.h
new file mode 100644
index 00000000..e1ea6f1b
--- /dev/null
+++ b/src/irc/dcc/dcc-chat.h
@@ -0,0 +1,11 @@
+#ifndef __DCC_CHAT_H
+#define __DCC_CHAT_H
+
+#include "dcc.h"
+
+DCC_REC *dcc_chat_find_id(const char *id);
+
+void dcc_chat_init(void);
+void dcc_chat_deinit(void);
+
+#endif
diff --git a/src/irc/dcc/dcc-files.c b/src/irc/dcc/dcc-files.c
deleted file mode 100644
index c284087a..00000000
--- a/src/irc/dcc/dcc-files.c
+++ /dev/null
@@ -1,667 +0,0 @@
-/*
- dcc-files.c : irssi
-
- Copyright (C) 1999-2000 Timo Sirainen
-
- 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; either version 2 of the License, or
- (at your option) any later version.
-
- 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.
-
- You should have received a copy of the GNU General Public License
- along with this program; if not, write to the Free Software
- Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
-*/
-
-#include "module.h"
-#include "signals.h"
-#include "commands.h"
-#include "network.h"
-#include "net-sendbuffer.h"
-#include "line-split.h"
-#include "misc.h"
-#include "settings.h"
-
-#include "masks.h"
-#include "irc.h"
-#include "servers-setup.h"
-
-#include "dcc.h"
-
-static int dcc_file_create_mode;
-
-static char *dcc_get_download_path(const char *fname)
-{
- char *str, *downpath;
-
- downpath = convert_home(settings_get_str("dcc_download_path"));
- str = g_strconcat(downpath, G_DIR_SEPARATOR_S, g_basename(fname), NULL);
- g_free(downpath);
-
- return str;
-}
-
-static void sig_dccget_send(DCC_REC *dcc);
-
-void dcc_get_send_received(DCC_REC *dcc)
-{
- guint32 recd;
-
- recd = (guint32) htonl(dcc->transfd);
- memcpy(dcc->count_buf, &recd, 4);
-
- dcc->count_pos = net_transmit(dcc->handle, dcc->count_buf+dcc->count_pos, 4-dcc->count_pos);
- if (dcc->count_pos == 4) dcc->count_pos = 0;
-
- /* count_pos might be -1 here. if this happens, the
- count_buf should be re-sent.. also, if it's 1, 2 or 3, the
- last 1-3 bytes should be sent later. these happen probably
- never, but I just want to do it right.. :) */
- if (dcc->tagwrite == -1) {
- dcc->tagwrite = g_input_add(dcc->handle, G_INPUT_WRITE,
- (GInputFunction) sig_dccget_send, dcc);
- }
-}
-
-/* input function: DCC GET is free to send data */
-static void sig_dccget_send(DCC_REC *dcc)
-{
- guint32 recd;
- int ret;
-
- if (dcc->count_pos != 0) {
- ret = net_transmit(dcc->handle, dcc->count_buf+dcc->count_pos, 4-dcc->count_pos);
- if (dcc->count_pos <= 0)
- dcc->count_pos = ret;
- else if (ret > 0)
- dcc->count_pos += ret;
-
- if (dcc->count_pos == 4) dcc->count_pos = 0;
-
- }
-
- if (dcc->count_pos == 0) {
- g_source_remove(dcc->tagwrite);
- dcc->tagwrite = -1;
- }
-
- memcpy(&recd, dcc->count_buf, 4);
- if (recd != (guint32) htonl(dcc->transfd))
- dcc_get_send_received(dcc);
-}
-
-/* input function: DCC GET received data */
-static void sig_dccget_receive(DCC_REC *dcc)
-{
- int ret;
-
- g_return_if_fail(dcc != NULL);
-
- for (;;) {
- ret = net_receive(dcc->handle, dcc->databuf, dcc->databufsize);
- if (ret == 0) break;
-
- if (ret < 0) {
- /* socket closed - transmit complete,
- or other side died.. */
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
- return;
- }
-
- write(dcc->fhandle, dcc->databuf, ret);
- dcc->transfd += ret;
- }
-
- /* send number of total bytes received */
- if (dcc->count_pos <= 0)
- dcc_get_send_received(dcc);
-
- signal_emit("dcc transfer update", 1, dcc);
-}
-
-static char *get_rename_file(const char *fname)
-{
- GString *newname;
- struct stat statbuf;
- char *ret;
- int num;
-
- newname = g_string_new(NULL);
- num = 1;
- do {
- g_string_sprintf(newname, "%s.%d", fname, num);
- num++;
- } while (stat(newname->str, &statbuf) == 0);
-
- ret = newname->str;
- g_string_free(newname, FALSE);
- return ret;
-}
-
-/* callback: net_connect() finished for DCC GET */
-static void sig_dccget_connected(DCC_REC *dcc)
-{
- struct stat statbuf;
- char *fname;
-
- g_return_if_fail(dcc != NULL);
-
- if (net_geterror(dcc->handle) != 0) {
- /* error connecting */
- signal_emit("dcc error connect", 1, dcc);
- dcc_destroy(dcc);
- return;
- }
-
- g_source_remove(dcc->tagconn);
-
- g_free_not_null(dcc->file);
- dcc->file = dcc_get_download_path(dcc->arg);
-
- /* if some plugin wants to change the file name/path here.. */
- signal_emit("dcc get receive", 1, dcc);
-
- if (stat(dcc->file, &statbuf) == 0 && dcc->get_type == DCC_GET_RENAME) {
- /* file exists, rename.. */
- fname = get_rename_file(dcc->file);
- g_free(dcc->file);
- dcc->file = fname;
- }
-
- if (dcc->get_type != DCC_GET_RESUME) {
- dcc->fhandle = open(dcc->file, O_WRONLY | O_TRUNC | O_CREAT, dcc_file_create_mode);
- if (dcc->fhandle == -1) {
- signal_emit("dcc error file create", 2, dcc, dcc->file);
- dcc_destroy(dcc);
- return;
- }
- }
-
- dcc->databufsize = settings_get_int("dcc_block_size");
- if (dcc->databufsize <= 0) dcc->databufsize = 2048;
- dcc->databuf = g_malloc(dcc->databufsize);
-
- dcc->starttime = time(NULL);
- dcc->tagread = g_input_add(dcc->handle, G_INPUT_READ,
- (GInputFunction) sig_dccget_receive, dcc);
- signal_emit("dcc connected", 1, dcc);
-}
-
-static void dcc_get_connect(DCC_REC *dcc)
-{
- dcc->handle = net_connect_ip(&dcc->addr, dcc->port,
- source_host_ok ? source_host_ip : NULL);
- if (dcc->handle != NULL) {
- dcc->tagconn = g_input_add(dcc->handle,
- G_INPUT_WRITE | G_INPUT_READ,
- (GInputFunction) sig_dccget_connected, dcc);
- } else {
- /* error connecting */
- signal_emit("dcc error connect", 1, dcc);
- dcc_destroy(dcc);
- }
-}
-
-#define dcc_is_unget(dcc) \
- ((dcc)->type == DCC_TYPE_GET && (dcc)->handle == NULL)
-
-/* SYNTAX: DCC GET <nick> [<file>] */
-static void cmd_dcc_get(const char *data)
-{
- DCC_REC *dcc;
- GSList *tmp, *next;
- char *nick, *fname;
- void *free_arg;
- int found;
-
- g_return_if_fail(data != NULL);
-
- if (!cmd_get_params(data, &free_arg, 2 | PARAM_FLAG_GETREST, &nick, &fname))
- return;
- if (*nick == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
-
- dcc = NULL; found = FALSE;
- for (tmp = dcc_conns; tmp != NULL; tmp = next) {
- dcc = tmp->data;
- next = tmp->next;
-
- if (dcc_is_unget(dcc) && g_strcasecmp(dcc->nick, nick) == 0 &&
- (*fname == '\0' || strcmp(dcc->arg, fname) == 0)) {
- found = TRUE;
- dcc_get_connect(dcc);
- }
- }
-
- if (!found)
- signal_emit("dcc error get not found", 1, nick);
-
- cmd_params_free(free_arg);
-}
-
-static void dcc_resume_send(DCC_REC *dcc, int port)
-{
- char *str;
-
- g_return_if_fail(dcc != NULL);
- g_return_if_fail(dcc->type == DCC_TYPE_SEND);
-
- str = g_strdup_printf("DCC ACCEPT %s %d %lu",
- dcc->arg, port, dcc->transfd);
- dcc_ctcp_message(dcc->server, dcc->nick, dcc->chat, FALSE, str);
- g_free(str);
-}
-
-#define is_resume_type(type) \
- (g_strcasecmp(type, "RESUME") == 0 || \
- g_strcasecmp(type, "ACCEPT") == 0)
-
-#define is_resume_ok(type, dcc) \
- (g_strcasecmp(type, "RESUME") != 0 || \
- ((dcc)->type == DCC_TYPE_SEND && (dcc)->transfd == 0))
-
-#define is_accept_ok(type, dcc) \
- (g_strcasecmp(type, "ACCEPT") != 0 || \
- ((dcc)->type == DCC_TYPE_GET && \
- (dcc)->get_type == DCC_GET_RESUME && (dcc)->handle == NULL))
-
-static void dcc_ctcp_msg(IRC_SERVER_REC *server, const char *data,
- const char *sender, const char *sendaddr,
- const char *target, DCC_REC *chat)
-{
- char *type, *arg, *portstr, *sizestr;
- void *free_arg;
- long size;
- int port;
- DCC_REC *dcc;
-
- g_return_if_fail(data != NULL);
- g_return_if_fail(sender != NULL);
-
- if (!cmd_get_params(data, &free_arg, 4 | PARAM_FLAG_NOQUOTES,
- &type, &arg, &portstr, &sizestr))
- return;
-
- port = atoi(portstr);
- size = atol(sizestr);
-
- dcc = dcc_find_by_port(sender, port);
- if (dcc == NULL || !is_resume_type(type) ||
- !is_resume_ok(type, dcc) || !is_accept_ok(type, dcc)) {
- cmd_params_free(free_arg);
- return;
- }
-
- if (lseek(dcc->fhandle, 0, SEEK_END) == size) {
- /* whole file sent */
- dcc->starttime = time(NULL);
- dcc_reject(dcc, server);
- }
- else if (lseek(dcc->fhandle, size, SEEK_SET) != size) {
- /* error, or trying to seek after end of file */
- dcc_reject(dcc, server);
- } else {
- dcc->transfd = dcc->skipped = size;
-
- if (dcc->type == DCC_TYPE_SEND)
- dcc_resume_send(dcc, port);
- else
- dcc_get_connect(dcc);
- }
-
- cmd_params_free(free_arg);
-}
-
-static void dcc_resume_rec(DCC_REC *dcc)
-{
- char *str;
-
- g_return_if_fail(dcc != NULL);
-
- dcc->file = dcc_get_download_path(dcc->arg);
- dcc->fhandle = open(dcc->file, O_WRONLY, dcc_file_create_mode);
- if (dcc->fhandle == -1) {
- signal_emit("dcc error file not found", 2, dcc, dcc->file);
- return;
- }
-
- dcc->get_type = DCC_GET_RESUME;
-
- dcc->transfd = lseek(dcc->fhandle, 0, SEEK_END);
- if (dcc->transfd < 0) dcc->transfd = 0;
- dcc->skipped = dcc->transfd;
-
- if (dcc->skipped == dcc->size) {
- /* already received whole file */
- dcc->starttime = time(NULL);
- dcc_reject(dcc, NULL);
- return;
- }
-
- str = g_strdup_printf("DCC RESUME %s %d %lu",
- dcc->arg, dcc->port, dcc->transfd);
- dcc_ctcp_message(dcc->server, dcc->nick, dcc->chat, FALSE, str);
- g_free(str);
-}
-
-/* SYNTAX: DCC RESUME <nick> [<file>] */
-static void cmd_dcc_resume(const char *data)
-{
- DCC_REC *dcc;
- GSList *tmp, *next;
- char *nick, *fname;
- void *free_arg;
- int found;
-
- g_return_if_fail(data != NULL);
-
- if (!cmd_get_params(data, &free_arg, 2 | PARAM_FLAG_GETREST, &nick, &fname))
- return;
- if (*nick == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
-
- dcc = NULL; found = FALSE;
- for (tmp = dcc_conns; tmp != NULL; tmp = next) {
- dcc = tmp->data;
-
- next = tmp->next;
- if (dcc_is_unget(dcc) && g_strcasecmp(dcc->nick, nick) == 0 &&
- (*fname == '\0' || strcmp(dcc->arg, fname) == 0)) {
- dcc_resume_rec(dcc);
- found = TRUE;
- }
- }
-
- if (!found)
- signal_emit("dcc error get not found", 1, nick);
-
- cmd_params_free(free_arg);
-}
-
-/* input function: DCC SEND - we're ready to send more data */
-static void dcc_send_data(DCC_REC *dcc)
-{
- int ret;
-
- g_return_if_fail(dcc != NULL);
-
- if (!dcc->fastsend && !dcc->gotalldata) {
- /* haven't received everything we've send there yet.. */
- return;
- }
-
- ret = read(dcc->fhandle, dcc->databuf, dcc->databufsize);
- if (ret <= 0) {
- /* end of file .. or some error .. */
- if (dcc->fastsend) {
- /* no need to call this function anymore..
- in fact it just eats all the cpu.. */
- dcc->waitforend = TRUE;
- g_source_remove(dcc->tagwrite);
- dcc->tagwrite = -1;
- } else {
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
- }
- return;
- }
-
- ret = net_transmit(dcc->handle, dcc->databuf, ret);
- if (ret > 0) dcc->transfd += ret;
- dcc->gotalldata = FALSE;
-
- lseek(dcc->fhandle, dcc->transfd, SEEK_SET);
-
- signal_emit("dcc transfer update", 1, dcc);
-}
-
-/* input function: DCC SEND - received some data */
-static void dcc_send_read_size(DCC_REC *dcc)
-{
- guint32 bytes;
- int ret;
-
- g_return_if_fail(dcc != NULL);
-
- if (dcc->count_pos == 4)
- return;
-
- /* we need to get 4 bytes.. */
- ret = net_receive(dcc->handle, dcc->count_buf+dcc->count_pos, 4-dcc->count_pos);
- if (ret == -1) {
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
- return;
- }
-
- dcc->count_pos += ret;
-
- if (dcc->count_pos != 4)
- return;
-
- memcpy(&bytes, dcc->count_buf, 4);
- bytes = (guint32) ntohl(bytes);
-
- dcc->gotalldata = (long) bytes == dcc->transfd;
- dcc->count_pos = 0;
-
- if (!dcc->fastsend) {
- /* send more data.. */
- dcc_send_data(dcc);
- }
-
- if (dcc->waitforend && dcc->gotalldata) {
- /* file is sent */
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
- }
-}
-
-/* input function: DCC SEND - someone tried to connect to our socket */
-static void dcc_send_init(DCC_REC *dcc)
-{
- GIOChannel *handle;
- IPADDR addr;
- int port;
-
- g_return_if_fail(dcc != NULL);
-
- /* accept connection */
- handle = net_accept(dcc->handle, &addr, &port);
- if (handle == NULL)
- return;
-
- /* TODO: some kind of paranoia check would be nice. it would check
- that the host of the nick who we sent the request matches the
- address who connected us. */
-
- g_source_remove(dcc->tagconn);
- net_disconnect(dcc->handle);
-
- dcc->starttime = time(NULL);
- dcc->fastsend = settings_get_bool("dcc_fast_send");
- dcc->handle = handle;
- memcpy(&dcc->addr, &addr, sizeof(IPADDR));
- net_ip2host(&dcc->addr, dcc->addrstr);
- dcc->port = port;
-
- dcc->databufsize = settings_get_int("dcc_block_size");
- if (dcc->databufsize <= 0) dcc->databufsize = 2048;
- dcc->databuf = g_malloc(dcc->databufsize);
-
- dcc->tagread = g_input_add(handle, G_INPUT_READ,
- (GInputFunction) dcc_send_read_size, dcc);
- dcc->tagwrite = !dcc->fastsend ? -1 :
- g_input_add(handle, G_INPUT_WRITE, (GInputFunction) dcc_send_data, dcc);
-
- signal_emit("dcc connected", 1, dcc);
-
- if (!dcc->fastsend) {
- /* send first block */
- dcc->gotalldata = TRUE;
- dcc_send_data(dcc);
- }
-}
-
-/* SYNTAX: DCC SEND <nick> <file> */
-static void cmd_dcc_send(const char *data, IRC_SERVER_REC *server,
- WI_ITEM_REC *item)
-{
- char *target, *fname, *str, *ptr;
- void *free_arg;
- char host[MAX_IP_LEN];
- int hfile, port;
- long fsize;
- DCC_REC *dcc, *chat;
- IPADDR own_ip;
- GIOChannel *handle, *hlisten;
-
- g_return_if_fail(data != NULL);
-
- if (!cmd_get_params(data, &free_arg, 2 | PARAM_FLAG_GETREST, &target, &fname))
- return;
- if (*target == '\0' || *fname == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
-
- /* if we're in dcc chat, send the request via it. */
- chat = item_get_dcc(item);
-
- if (chat != NULL && (chat->mirc_ctcp || g_strcasecmp(target, chat->nick) != 0))
- chat = NULL;
-
- if ((server == NULL || !server->connected) && chat == NULL)
- cmd_param_error(CMDERR_NOT_CONNECTED);
-
- if (dcc_find_item(DCC_TYPE_SEND, target, fname)) {
- signal_emit("dcc error send exists", 2, target, fname);
- cmd_params_free(free_arg);
- return;
- }
-
- str = convert_home(fname);
- if (!g_path_is_absolute(str)) {
- char *path;
-
- g_free(str);
- path = convert_home(settings_get_str("dcc_upload_path"));
- str = g_strconcat(path, G_DIR_SEPARATOR_S, fname, NULL);
- g_free(path);
- }
-
- hfile = open(str, O_RDONLY);
- g_free(str);
-
- if (hfile == -1) {
- signal_emit("dcc error file not found", 2, target, fname);
- cmd_params_free(free_arg);
- return;
- }
- fsize = lseek(hfile, 0, SEEK_END);
- lseek(hfile, 0, SEEK_SET);
-
- /* get the IP address we use with IRC server */
- handle = chat != NULL ? chat->handle :
- net_sendbuffer_handle(server->handle);
- if (net_getsockname(handle, &own_ip, NULL) == -1) {
- close(hfile);
- cmd_param_error(CMDERR_ERRNO);
- }
-
- /* start listening */
- port = settings_get_int("dcc_port");
- hlisten = net_listen(&own_ip, &port);
- if (hlisten == NULL) {
- close(hfile);
- cmd_param_error(CMDERR_ERRNO);
- }
-
- /* skip path, change all spaces to _ */
- fname = g_strdup(g_basename(fname));
- for (ptr = fname; *ptr != '\0'; ptr++)
- if (*ptr == ' ') *ptr = '_';
-
- dcc = dcc_create(DCC_TYPE_SEND, hlisten, target, fname, server, chat);
- dcc->port = port;
- dcc->size = fsize;
- dcc->fhandle = hfile;
- dcc->tagconn = g_input_add(hlisten, G_INPUT_READ,
- (GInputFunction) dcc_send_init, dcc);
-
- /* send DCC request */
- dcc_make_address(&own_ip, host);
- str = g_strdup_printf("DCC SEND %s %s %d %lu",
- fname, host, port, fsize);
- dcc_ctcp_message(server, target, chat, FALSE, str);
- g_free(str);
-
- g_free(fname);
- cmd_params_free(free_arg);
-}
-
-static void sig_dcc_request(DCC_REC *dcc, const char *nickaddr)
-{
- struct stat statbuf;
- const char *masks;
- char *str, *file;
- int max_size;
-
- g_return_if_fail(dcc != NULL);
- if (dcc->type != DCC_TYPE_GET) return;
-
- /* check if we want to autoget file offer */
- if (!settings_get_bool("dcc_autoget") &&
- !settings_get_bool("dcc_autoresume"))
- return;
-
- /* check that autoget masks match */
- masks = settings_get_str("dcc_autoget_masks");
- if (*masks != '\0' &&
- !masks_match(SERVER(dcc->server), masks, dcc->nick, nickaddr))
- return;
-
- /* check file size limit, FIXME: it's still possible to send a
- bogus file size and then just send what ever sized file.. */
- max_size = settings_get_int("dcc_max_autoget_size");
- if (max_size > 0 && max_size*1024 < dcc->size)
- return;
-
- /* ok. but do we want/need to resume? */
- file = dcc_get_download_path(dcc->arg);
- str = g_strdup_printf(settings_get_bool("dcc_autoresume") &&
- stat(file, &statbuf) == 0 ?
- "RESUME %s %s" : "GET %s %s",
- dcc->nick, dcc->arg);
- signal_emit("command dcc", 2, str, dcc->server);
- g_free(file);
- g_free(str);
-}
-
-static void read_settings(void)
-{
- dcc_file_create_mode = octal2dec(settings_get_int("dcc_file_create_mode"));
-}
-
-void dcc_files_init(void)
-{
- signal_add("ctcp msg dcc", (SIGNAL_FUNC) dcc_ctcp_msg);
- signal_add("setup changed", (SIGNAL_FUNC) read_settings);
- signal_add_last("dcc request", (SIGNAL_FUNC) sig_dcc_request);
- signal_add("irssi init finished", (SIGNAL_FUNC) read_settings);
- command_bind("dcc send", NULL, (SIGNAL_FUNC) cmd_dcc_send);
- command_bind("dcc get", NULL, (SIGNAL_FUNC) cmd_dcc_get);
- command_bind("dcc resume", NULL, (SIGNAL_FUNC) cmd_dcc_resume);
-}
-
-void dcc_files_deinit(void)
-{
- signal_remove("ctcp msg dcc", (SIGNAL_FUNC) dcc_ctcp_msg);
- signal_remove("setup changed", (SIGNAL_FUNC) read_settings);
- signal_remove("dcc request", (SIGNAL_FUNC) sig_dcc_request);
- signal_remove("irssi init finished", (SIGNAL_FUNC) read_settings);
- command_unbind("dcc send", (SIGNAL_FUNC) cmd_dcc_send);
- command_unbind("dcc get", (SIGNAL_FUNC) cmd_dcc_get);
- command_unbind("dcc resume", (SIGNAL_FUNC) cmd_dcc_resume);
-}
diff --git a/src/irc/dcc/dcc-get.c b/src/irc/dcc/dcc-get.c
new file mode 100644
index 00000000..0ebf3e20
--- /dev/null
+++ b/src/irc/dcc/dcc-get.c
@@ -0,0 +1,385 @@
+/*
+ dcc-get.c : irssi
+
+ Copyright (C) 1999-2001 Timo Sirainen
+
+ 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; either version 2 of the License, or
+ (at your option) any later version.
+
+ 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.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "module.h"
+#include "signals.h"
+#include "commands.h"
+#include "network.h"
+#include "misc.h"
+#include "settings.h"
+
+#include "servers-setup.h"
+#include "dcc-get.h"
+
+static int dcc_file_create_mode;
+
+char *dcc_get_download_path(const char *fname)
+{
+ char *str, *downpath;
+
+ downpath = convert_home(settings_get_str("dcc_download_path"));
+ str = g_strconcat(downpath, G_DIR_SEPARATOR_S, g_basename(fname), NULL);
+ g_free(downpath);
+
+ return str;
+}
+
+static char *dcc_get_rename_file(const char *fname)
+{
+ GString *newname;
+ struct stat statbuf;
+ char *ret;
+ int num;
+
+ newname = g_string_new(NULL);
+ num = 1;
+ do {
+ g_string_sprintf(newname, "%s.%d", fname, num);
+ num++;
+ } while (stat(newname->str, &statbuf) == 0);
+
+ ret = newname->str;
+ g_string_free(newname, FALSE);
+ return ret;
+}
+
+static void sig_dccget_send(DCC_REC *dcc);
+
+void dcc_get_send_received(DCC_REC *dcc)
+{
+ guint32 recd;
+
+ recd = (guint32) htonl(dcc->transfd);
+ memcpy(dcc->count_buf, &recd, 4);
+
+ dcc->count_pos =
+ net_transmit(dcc->handle, dcc->count_buf+dcc->count_pos,
+ 4-dcc->count_pos);
+ if (dcc->count_pos == 4) dcc->count_pos = 0;
+
+ /* count_pos might be -1 here. if this happens, the
+ count_buf should be re-sent.. also, if it's 1, 2 or 3, the
+ last 1-3 bytes should be sent later. these happen probably
+ never, but I just want to do it right.. :) */
+ if (dcc->tagwrite == -1) {
+ dcc->tagwrite = g_input_add(dcc->handle, G_INPUT_WRITE,
+ (GInputFunction) sig_dccget_send,
+ dcc);
+ }
+}
+
+/* input function: DCC GET is free to send data */
+static void sig_dccget_send(DCC_REC *dcc)
+{
+ guint32 recd;
+ int ret;
+
+ if (dcc->count_pos != 0) {
+ ret = net_transmit(dcc->handle, dcc->count_buf+dcc->count_pos,
+ 4-dcc->count_pos);
+
+ if (dcc->count_pos <= 0)
+ dcc->count_pos = ret;
+ else if (ret > 0)
+ dcc->count_pos += ret;
+
+ if (dcc->count_pos == 4) dcc->count_pos = 0;
+
+ }
+
+ if (dcc->count_pos == 0) {
+ g_source_remove(dcc->tagwrite);
+ dcc->tagwrite = -1;
+ }
+
+ memcpy(&recd, dcc->count_buf, 4);
+ if (recd != (guint32) htonl(dcc->transfd))
+ dcc_get_send_received(dcc);
+}
+
+/* input function: DCC GET received data */
+static void sig_dccget_receive(DCC_REC *dcc)
+{
+ int ret;
+
+ g_return_if_fail(dcc != NULL);
+
+ for (;;) {
+ ret = net_receive(dcc->handle, dcc->databuf, dcc->databufsize);
+ if (ret == 0) break;
+
+ if (ret < 0) {
+ /* socket closed - transmit complete,
+ or other side died.. */
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ return;
+ }
+
+ write(dcc->fhandle, dcc->databuf, ret);
+ dcc->transfd += ret;
+ }
+
+ /* send number of total bytes received */
+ if (dcc->count_pos <= 0)
+ dcc_get_send_received(dcc);
+
+ signal_emit("dcc transfer update", 1, dcc);
+}
+
+/* callback: net_connect() finished for DCC GET */
+static void sig_dccget_connected(DCC_REC *dcc)
+{
+ struct stat statbuf;
+ char *fname;
+
+ g_return_if_fail(dcc != NULL);
+
+ if (net_geterror(dcc->handle) != 0) {
+ /* error connecting */
+ signal_emit("dcc error connect", 1, dcc);
+ dcc_destroy(dcc);
+ return;
+ }
+
+ g_source_remove(dcc->tagconn);
+
+ g_free_not_null(dcc->file);
+ dcc->file = dcc_get_download_path(dcc->arg);
+
+ /* if some plugin wants to change the file name/path here.. */
+ signal_emit("dcc get receive", 1, dcc);
+
+ if (stat(dcc->file, &statbuf) == 0 &&
+ dcc->get_type == DCC_GET_RENAME) {
+ /* file exists, rename.. */
+ fname = dcc_get_rename_file(dcc->file);
+ g_free(dcc->file);
+ dcc->file = fname;
+ }
+
+ if (dcc->get_type != DCC_GET_RESUME) {
+ dcc->fhandle = open(dcc->file, O_WRONLY | O_TRUNC | O_CREAT, dcc_file_create_mode);
+ if (dcc->fhandle == -1) {
+ signal_emit("dcc error file create", 2, dcc, dcc->file);
+ dcc_destroy(dcc);
+ return;
+ }
+ }
+
+ dcc->databufsize = settings_get_int("dcc_block_size");
+ if (dcc->databufsize <= 0) dcc->databufsize = 2048;
+ dcc->databuf = g_malloc(dcc->databufsize);
+
+ dcc->starttime = time(NULL);
+ dcc->tagread = g_input_add(dcc->handle, G_INPUT_READ,
+ (GInputFunction) sig_dccget_receive, dcc);
+ signal_emit("dcc connected", 1, dcc);
+}
+
+void dcc_get_connect(DCC_REC *dcc)
+{
+ if (dcc->get_type == DCC_GET_DEFAULT) {
+ dcc->get_type = settings_get_bool("dcc_autorename") ?
+ DCC_GET_RENAME : DCC_GET_OVERWRITE;
+ }
+
+
+ dcc->handle = net_connect_ip(&dcc->addr, dcc->port,
+ source_host_ok ? source_host_ip : NULL);
+ if (dcc->handle != NULL) {
+ dcc->tagconn = g_input_add(dcc->handle,
+ G_INPUT_WRITE | G_INPUT_READ,
+ (GInputFunction) sig_dccget_connected, dcc);
+ } else {
+ /* error connecting */
+ signal_emit("dcc error connect", 1, dcc);
+ dcc_destroy(dcc);
+ }
+}
+
+#define get_params_match(params, pos) \
+ ((is_numeric(params[pos], '\0') || is_ipv6_address(params[pos])) && \
+ is_numeric(params[(pos)+1], '\0') && atol(params[(pos)+1]) < 65536 && \
+ is_numeric(params[(pos)+2], '\0'))
+
+/* Return number of parameters in `params' that belong to file name.
+ Normally it's paramcount-3, but I don't think anything forbids of
+ adding some extension where there could be more parameters after
+ file size.
+
+ MIRC sends filenames with spaces quoted ("file name"), but I'd rather
+ not trust that entirely either. At least some clients that don't really
+ understand the problem with spaces in file names sends the file name
+ without any quotes. */
+int get_file_params_count(char **params, int paramcount)
+{
+ int pos, best;
+
+ if (*params[0] == '"') {
+ /* quoted file name? */
+ for (pos = 0; pos < paramcount-3; pos++) {
+ if (params[pos][strlen(params[pos])-1] == '"' &&
+ get_params_match(params, pos+1))
+ return pos+1;
+ }
+ }
+
+ best = paramcount-3;
+ for (pos = paramcount-3; pos > 0; pos--) {
+ if (get_params_match(params, pos))
+ best = pos;
+ }
+
+ return best;
+}
+
+/* CTCP: DCC SEND */
+static void ctcp_msg_dcc_send(IRC_SERVER_REC *server, const char *data,
+ const char *nick, const char *addr,
+ const char *target, DCC_REC *chat)
+{
+ DCC_REC *dcc;
+ IPADDR ip;
+ char **params, *fname;
+ int paramcount, fileparams;
+ int port, len, quoted = FALSE;
+ long size;
+
+ /* SEND <file name> <address> <port> <size> [...] */
+ params = g_strsplit(data, " ", -1);
+ paramcount = strarray_length(params);
+
+ if (paramcount < 4) {
+ signal_emit("dcc error ctcp", 5, "SEND", data,
+ nick, addr, target);
+ g_strfreev(params);
+ return;
+ }
+
+ fileparams = get_file_params_count(params, paramcount);
+
+ dcc_get_address(params[fileparams], &ip);
+ port = atoi(params[fileparams+1]);
+ size = atol(params[fileparams+2]);
+
+ params[fileparams] = NULL;
+ fname = g_strjoinv(" ", params);
+ g_strfreev(params);
+
+ len = strlen(fname);
+ if (len > 1 && *fname == '"' && fname[len-1] == '"') {
+ /* "file name" - MIRC sends filenames with spaces like this */
+ fname[len-1] = '\0';
+ g_memmove(fname, fname+1, len);
+ quoted = TRUE;
+ }
+
+ dcc = dcc_find_request(DCC_TYPE_GET, nick, fname);
+ if (dcc != NULL) {
+ /* same DCC request offered again, remove the old one */
+ dcc_destroy(dcc);
+ }
+
+ dcc = dcc_create(DCC_TYPE_GET, nick, fname, server, chat);
+ dcc->target = g_strdup(target);
+ memcpy(&dcc->addr, &ip, sizeof(ip));
+ net_ip2host(&dcc->addr, dcc->addrstr);
+ dcc->port = port;
+ dcc->size = size;
+ dcc->file_quoted = quoted;
+
+ signal_emit("dcc request", 2, dcc, addr);
+
+ g_free(fname);
+}
+
+/* handle receiving DCC - GET/RESUME. */
+void cmd_dcc_receive(const char *data, DCC_GET_FUNC accept)
+{
+ DCC_REC *dcc;
+ GSList *tmp, *next;
+ char *nick, *fname;
+ void *free_arg;
+ int found;
+
+ g_return_if_fail(data != NULL);
+
+ if (!cmd_get_params(data, &free_arg, 2 | PARAM_FLAG_GETREST,
+ &nick, &fname))
+ return;
+
+ if (*nick == '\0') {
+ dcc = dcc_find_request_latest(DCC_TYPE_GET);
+ if (dcc != NULL)
+ accept(dcc);
+ cmd_params_free(free_arg);
+ return;
+ }
+
+ found = FALSE;
+ for (tmp = dcc_conns; tmp != NULL; tmp = next) {
+ DCC_REC *dcc = tmp->data;
+
+ next = tmp->next;
+ if (dcc_is_waiting_get(dcc) &&
+ g_strcasecmp(dcc->nick, nick) == 0 &&
+ (*fname == '\0' || strcmp(dcc->arg, fname) == 0)) {
+ found = TRUE;
+ accept(dcc);
+ }
+ }
+
+ if (!found)
+ signal_emit("dcc error get not found", 1, nick);
+
+ cmd_params_free(free_arg);
+}
+
+/* SYNTAX: DCC GET [<nick> [<file>]] */
+static void cmd_dcc_get(const char *data)
+{
+ cmd_dcc_receive(data, dcc_get_connect);
+}
+
+static void read_settings(void)
+{
+ dcc_file_create_mode =
+ octal2dec(settings_get_int("dcc_file_create_mode"));
+}
+
+void dcc_get_init(void)
+{
+ settings_add_bool("dcc", "dcc_autorename", FALSE);
+ settings_add_str("dcc", "dcc_download_path", "~");
+ settings_add_int("dcc", "dcc_file_create_mode", 644);
+
+ read_settings();
+ signal_add("ctcp msg dcc send", (SIGNAL_FUNC) ctcp_msg_dcc_send);
+ signal_add("setup changed", (SIGNAL_FUNC) read_settings);
+ command_bind("dcc get", NULL, (SIGNAL_FUNC) cmd_dcc_get);
+}
+
+void dcc_get_deinit(void)
+{
+ signal_remove("ctcp msg dcc send", (SIGNAL_FUNC) ctcp_msg_dcc_send);
+ signal_remove("setup changed", (SIGNAL_FUNC) read_settings);
+ command_unbind("dcc get", (SIGNAL_FUNC) cmd_dcc_get);
+}
diff --git a/src/irc/dcc/dcc-get.h b/src/irc/dcc/dcc-get.h
new file mode 100644
index 00000000..f103b2ae
--- /dev/null
+++ b/src/irc/dcc/dcc-get.h
@@ -0,0 +1,28 @@
+#ifndef __DCC_GET_H
+#define __DCC_GET_H
+
+#include "dcc.h"
+
+enum {
+ DCC_GET_DEFAULT,
+
+ DCC_GET_RENAME,
+ DCC_GET_OVERWRITE,
+ DCC_GET_RESUME
+};
+
+typedef void (*DCC_GET_FUNC) (DCC_REC *);
+
+/* handle receiving DCC - GET/RESUME. */
+void cmd_dcc_receive(const char *data, DCC_GET_FUNC accept);
+
+void dcc_get_connect(DCC_REC *dcc);
+char *dcc_get_download_path(const char *fname);
+
+#define dcc_is_waiting_get(dcc) \
+ ((dcc)->type == DCC_TYPE_GET && dcc_is_waiting_user(dcc))
+
+void dcc_get_init(void);
+void dcc_get_deinit(void);
+
+#endif
diff --git a/src/irc/dcc/dcc-resume.c b/src/irc/dcc/dcc-resume.c
new file mode 100644
index 00000000..3fa884bb
--- /dev/null
+++ b/src/irc/dcc/dcc-resume.c
@@ -0,0 +1,177 @@
+/*
+ dcc-resume.c : irssi
+
+ Copyright (C) 1999-2001 Timo Sirainen
+
+ 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; either version 2 of the License, or
+ (at your option) any later version.
+
+ 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.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "module.h"
+#include "signals.h"
+#include "commands.h"
+#include "network.h"
+#include "misc.h"
+
+#include "dcc-get.h"
+
+static DCC_REC *dcc_resume_find(int type, const char *nick, int port)
+{
+ GSList *tmp;
+
+ for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next) {
+ DCC_REC *dcc = tmp->data;
+
+ if (dcc->type == type && !dcc_is_connected(dcc) &&
+ dcc->port == port && g_strcasecmp(dcc->nick, nick) == 0)
+ return dcc;
+ }
+
+ return NULL;
+}
+
+static int dcc_ctcp_resume_parse(int type, const char *data, const char *nick,
+ DCC_REC **dcc, long *size)
+{
+ char **params;
+ int paramcount;
+ int port;
+
+ /* RESUME|ACCEPT <file name> <port> <size> */
+ params = g_strsplit(data, " ", -1);
+ paramcount = strarray_length(params);
+
+ if (paramcount >= 3) {
+ port = atoi(params[paramcount-2]);
+ *size = atol(params[paramcount-1]);
+
+ type = type == DCC_TYPE_RESUME ? DCC_TYPE_SEND : DCC_TYPE_GET;
+ *dcc = dcc_resume_find(type, nick, port);
+ }
+ g_strfreev(params);
+ return paramcount >= 3;
+}
+
+static int dcc_resume_file_check(DCC_REC *dcc, IRC_SERVER_REC *server,
+ long size)
+{
+ if (lseek(dcc->fhandle, 0, SEEK_END) == dcc->size) {
+ /* whole file sent */
+ dcc->starttime = time(NULL);
+ dcc_reject(dcc, server);
+ } else if (lseek(dcc->fhandle, size, SEEK_SET) != size) {
+ /* error, or trying to seek after end of file */
+ dcc_reject(dcc, server);
+ } else {
+ dcc->transfd = dcc->skipped = size;
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
+/* CTCP: DCC RESUME */
+static void ctcp_msg_dcc_resume(IRC_SERVER_REC *server, const char *data,
+ const char *nick, const char *addr,
+ const char *target, DCC_REC *chat)
+{
+ DCC_REC *dcc;
+ char *str;
+ long size;
+
+ if (!dcc_ctcp_resume_parse(DCC_TYPE_RESUME, data, nick, &dcc, &size))
+ signal_emit("dcc error ctcp", 5, "RESUME", data,
+ nick, addr, target);
+
+ if (dcc != NULL && dcc_resume_file_check(dcc, server, size)) {
+ str = g_strdup_printf(dcc->file_quoted ?
+ "DCC ACCEPT \"%s\" %d %lu" :
+ "DCC ACCEPT %s %d %lu",
+ dcc->arg, dcc->port, dcc->transfd);
+ dcc_ctcp_message(dcc->server, dcc->nick,
+ dcc->chat, FALSE, str);
+ g_free(str);
+ }
+}
+
+/* CTCP: DCC ACCEPT */
+static void ctcp_msg_dcc_accept(IRC_SERVER_REC *server, const char *data,
+ const char *nick, const char *addr,
+ const char *target, DCC_REC *chat)
+{
+ DCC_REC *dcc;
+ long size;
+
+ if (!dcc_ctcp_resume_parse(DCC_TYPE_ACCEPT, data, nick, &dcc, &size) ||
+ (dcc != NULL && dcc->get_type != DCC_GET_RESUME))
+ signal_emit("dcc error ctcp", 5, "ACCEPT", data,
+ nick, addr, target);
+
+ if (dcc != NULL && dcc_resume_file_check(dcc, server, size))
+ dcc_get_connect(dcc);
+}
+
+static void dcc_send_resume(DCC_REC *dcc)
+{
+ char *str;
+
+ g_return_if_fail(dcc != NULL);
+
+ dcc->file = dcc_get_download_path(dcc->arg);
+ dcc->fhandle = open(dcc->file, O_WRONLY);
+ if (dcc->fhandle == -1) {
+ signal_emit("dcc error file not found", 2, dcc, dcc->file);
+ return;
+ }
+
+ dcc->get_type = DCC_GET_RESUME;
+
+ dcc->transfd = lseek(dcc->fhandle, 0, SEEK_END);
+ if (dcc->transfd < 0) dcc->transfd = 0;
+ dcc->skipped = dcc->transfd;
+
+ if (dcc->skipped == dcc->size) {
+ /* already received whole file */
+ dcc->starttime = time(NULL);
+ dcc_reject(dcc, NULL);
+ return;
+ }
+
+ str = g_strdup_printf(dcc->file_quoted ?
+ "DCC RESUME \"%s\" %d %lu" :
+ "DCC RESUME %s %d %lu",
+ dcc->arg, dcc->port, dcc->transfd);
+ dcc_ctcp_message(dcc->server, dcc->nick, dcc->chat, FALSE, str);
+ g_free(str);
+}
+
+/* SYNTAX: DCC RESUME [<nick> [<file>]] */
+static void cmd_dcc_resume(const char *data)
+{
+ cmd_dcc_receive(data, dcc_send_resume);
+}
+
+void dcc_resume_init(void)
+{
+ signal_add("ctcp msg dcc resume", (SIGNAL_FUNC) ctcp_msg_dcc_resume);
+ signal_add("ctcp msg dcc accept", (SIGNAL_FUNC) ctcp_msg_dcc_accept);
+ command_bind("dcc resume", NULL, (SIGNAL_FUNC) cmd_dcc_resume);
+}
+
+void dcc_resume_deinit(void)
+{
+ signal_remove("ctcp msg dcc resume", (SIGNAL_FUNC) ctcp_msg_dcc_resume);
+ signal_remove("ctcp msg dcc accept", (SIGNAL_FUNC) ctcp_msg_dcc_accept);
+ command_unbind("dcc resume", (SIGNAL_FUNC) cmd_dcc_resume);
+}
diff --git a/src/irc/dcc/dcc-send.c b/src/irc/dcc/dcc-send.c
new file mode 100644
index 00000000..2ad66b25
--- /dev/null
+++ b/src/irc/dcc/dcc-send.c
@@ -0,0 +1,267 @@
+/*
+ dcc-send.c : irssi
+
+ Copyright (C) 1999-2001 Timo Sirainen
+
+ 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; either version 2 of the License, or
+ (at your option) any later version.
+
+ 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.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "module.h"
+#include "signals.h"
+#include "commands.h"
+#include "network.h"
+#include "net-sendbuffer.h"
+#include "misc.h"
+#include "settings.h"
+
+#include "dcc.h"
+
+/* input function: DCC SEND - we're ready to send more data */
+static void dcc_send_data(DCC_REC *dcc)
+{
+ int ret;
+
+ g_return_if_fail(dcc != NULL);
+
+ if (!dcc->fastsend && !dcc->gotalldata) {
+ /* haven't received everything we've send there yet.. */
+ return;
+ }
+
+ ret = read(dcc->fhandle, dcc->databuf, dcc->databufsize);
+ if (ret <= 0) {
+ /* end of file .. or some error .. */
+ if (dcc->fastsend) {
+ /* no need to call this function anymore..
+ in fact it just eats all the cpu.. */
+ dcc->waitforend = TRUE;
+ g_source_remove(dcc->tagwrite);
+ dcc->tagwrite = -1;
+ } else {
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ }
+ return;
+ }
+
+ ret = net_transmit(dcc->handle, dcc->databuf, ret);
+ if (ret > 0) dcc->transfd += ret;
+ dcc->gotalldata = FALSE;
+
+ lseek(dcc->fhandle, dcc->transfd, SEEK_SET);
+
+ signal_emit("dcc transfer update", 1, dcc);
+}
+
+/* input function: DCC SEND - received some data */
+static void dcc_send_read_size(DCC_REC *dcc)
+{
+ guint32 bytes;
+ int ret;
+
+ g_return_if_fail(dcc != NULL);
+
+ if (dcc->count_pos == 4)
+ return;
+
+ /* we need to get 4 bytes.. */
+ ret = net_receive(dcc->handle, dcc->count_buf+dcc->count_pos,
+ 4-dcc->count_pos);
+ if (ret == -1) {
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ return;
+ }
+
+ dcc->count_pos += ret;
+
+ if (dcc->count_pos != 4)
+ return;
+
+ memcpy(&bytes, dcc->count_buf, 4);
+ bytes = (guint32) ntohl(bytes);
+
+ dcc->gotalldata = (long) bytes == dcc->transfd;
+ dcc->count_pos = 0;
+
+ if (!dcc->fastsend) {
+ /* send more data.. */
+ dcc_send_data(dcc);
+ }
+
+ if (dcc->waitforend && dcc->gotalldata) {
+ /* file is sent */
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ }
+}
+
+/* input function: DCC SEND - someone tried to connect to our socket */
+static void dcc_send_connected(DCC_REC *dcc)
+{
+ GIOChannel *handle;
+ IPADDR addr;
+ int port;
+
+ g_return_if_fail(dcc != NULL);
+
+ /* accept connection */
+ handle = net_accept(dcc->handle, &addr, &port);
+ if (handle == NULL)
+ return;
+
+ /* TODO: some kind of paranoia check would be nice. it would check
+ that the host of the nick who we sent the request matches the
+ address who connected us. */
+
+ g_source_remove(dcc->tagconn);
+ net_disconnect(dcc->handle);
+
+ dcc->starttime = time(NULL);
+ dcc->fastsend = settings_get_bool("dcc_fast_send");
+ dcc->handle = handle;
+ memcpy(&dcc->addr, &addr, sizeof(IPADDR));
+ net_ip2host(&dcc->addr, dcc->addrstr);
+ dcc->port = port;
+
+ dcc->databufsize = settings_get_int("dcc_block_size");
+ if (dcc->databufsize <= 0) dcc->databufsize = 2048;
+ dcc->databuf = g_malloc(dcc->databufsize);
+
+ dcc->tagread = g_input_add(handle, G_INPUT_READ,
+ (GInputFunction) dcc_send_read_size, dcc);
+ dcc->tagwrite = !dcc->fastsend ? -1 :
+ g_input_add(handle, G_INPUT_WRITE,
+ (GInputFunction) dcc_send_data, dcc);
+
+ signal_emit("dcc connected", 1, dcc);
+
+ if (!dcc->fastsend) {
+ /* send first block */
+ dcc->gotalldata = TRUE;
+ dcc_send_data(dcc);
+ }
+}
+
+static char *dcc_send_get_file(const char *fname)
+{
+ char *str, *path;
+
+ str = convert_home(fname);
+ if (!g_path_is_absolute(str)) {
+ /* full path not given to file, use dcc_upload_path */
+ g_free(str);
+
+ path = convert_home(settings_get_str("dcc_upload_path"));
+ str = g_strconcat(path, G_DIR_SEPARATOR_S, fname, NULL);
+ g_free(path);
+ }
+
+ return str;
+}
+
+/* SYNTAX: DCC SEND <nick> <file> */
+static void cmd_dcc_send(const char *data, IRC_SERVER_REC *server,
+ WI_ITEM_REC *item)
+{
+ char *target, *fname, *str;
+ void *free_arg;
+ char host[MAX_IP_LEN];
+ int hfile, port;
+ long fsize;
+ DCC_REC *dcc, *chat;
+ IPADDR own_ip;
+ GIOChannel *handle;
+
+ g_return_if_fail(data != NULL);
+
+ if (!cmd_get_params(data, &free_arg, 2 | PARAM_FLAG_GETREST,
+ &target, &fname))
+ return;
+ if (*target == '\0' || *fname == '\0')
+ cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
+
+ /* if we're in dcc chat, send the request via it. */
+ chat = item_get_dcc(item);
+
+ if (chat != NULL && (chat->mirc_ctcp ||
+ g_strcasecmp(target, chat->nick) != 0))
+ chat = NULL;
+
+ if ((server == NULL || !server->connected) && chat == NULL)
+ cmd_param_error(CMDERR_NOT_CONNECTED);
+
+ if (dcc_find_request(DCC_TYPE_SEND, target, fname)) {
+ signal_emit("dcc error send exists", 2, target, fname);
+ cmd_params_free(free_arg);
+ return;
+ }
+
+ str = dcc_send_get_file(fname);
+ hfile = open(str, O_RDONLY);
+ g_free(str);
+
+ if (hfile == -1) {
+ signal_emit("dcc error file not found", 2, target, fname);
+ cmd_params_free(free_arg);
+ return;
+ }
+ fsize = lseek(hfile, 0, SEEK_END);
+ lseek(hfile, 0, SEEK_SET);
+
+ /* start listening */
+ handle = dcc_listen(chat != NULL ? chat->handle :
+ net_sendbuffer_handle(server->handle),
+ &own_ip, &port);
+ if (handle == NULL) {
+ close(hfile);
+ cmd_param_error(CMDERR_ERRNO);
+ }
+
+ fname = g_basename(fname);
+
+ dcc = dcc_create(DCC_TYPE_SEND, target, fname, server, chat);
+ dcc->handle = handle;
+ dcc->port = port;
+ dcc->size = fsize;
+ dcc->fhandle = hfile;
+ dcc->file_quoted = strchr(fname, ' ') != NULL;
+ dcc->tagconn = g_input_add(handle, G_INPUT_READ,
+ (GInputFunction) dcc_send_connected, dcc);
+
+ /* send DCC request */
+ dcc_make_address(&own_ip, host);
+ str = g_strdup_printf(dcc->file_quoted ?
+ "DCC SEND \"%s\" %s %d %lu" :
+ "DCC SEND %s %s %d %lu",
+ fname, host, port, fsize);
+ dcc_ctcp_message(server, target, chat, FALSE, str);
+ g_free(str);
+
+ cmd_params_free(free_arg);
+}
+
+void dcc_send_init(void)
+{
+ settings_add_bool("dcc", "dcc_fast_send", TRUE);
+ settings_add_str("dcc", "dcc_upload_path", "~");
+
+ command_bind("dcc send", NULL, (SIGNAL_FUNC) cmd_dcc_send);
+}
+
+void dcc_send_deinit(void)
+{
+ command_unbind("dcc send", (SIGNAL_FUNC) cmd_dcc_send);
+}
diff --git a/src/irc/dcc/dcc.c b/src/irc/dcc/dcc.c
index 1d0fc2dc..8e5cee14 100644
--- a/src/irc/dcc/dcc.c
+++ b/src/irc/dcc/dcc.c
@@ -1,7 +1,7 @@
/*
dcc.c : irssi
- Copyright (C) 1999-2000 Timo Sirainen
+ Copyright (C) 1999-2001 Timo Sirainen
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
@@ -26,16 +26,19 @@
#include "line-split.h"
#include "settings.h"
-#include "masks.h"
#include "irc.h"
-#include "dcc.h"
+#include "dcc-chat.h"
+#include "dcc-get.h"
-void dcc_chat_init(void);
-void dcc_chat_deinit(void);
+void dcc_send_init(void);
+void dcc_send_deinit(void);
-void dcc_files_init(void);
-void dcc_files_deinit(void);
+void dcc_resume_init(void);
+void dcc_resume_deinit(void);
+
+void dcc_autoget_init(void);
+void dcc_autoget_deinit(void);
#define DCC_TYPES 5
@@ -52,7 +55,7 @@ GSList *dcc_conns;
static int dcc_timeouttag;
/* Create new DCC record */
-DCC_REC *dcc_create(int type, GIOChannel *handle, const char *nick, const char *arg,
+DCC_REC *dcc_create(int type, const char *nick, const char *arg,
IRC_SERVER_REC *server, DCC_REC *chat)
{
DCC_REC *dcc;
@@ -67,7 +70,6 @@ DCC_REC *dcc_create(int type, GIOChannel *handle, const char *nick, const char *
dcc->type = type;
dcc->arg = g_strdup(arg);
dcc->nick = g_strdup(nick);
- dcc->handle = handle;
dcc->fhandle = -1;
dcc->tagconn = dcc->tagread = dcc->tagwrite = -1;
dcc->server = server;
@@ -123,6 +125,8 @@ void dcc_destroy(DCC_REC *dcc)
g_free_not_null(dcc->file);
g_free_not_null(dcc->ircnet);
+ g_free_not_null(dcc->chat_id);
+ g_free_not_null(dcc->target);
g_free(dcc->mynick);
g_free(dcc->nick);
g_free(dcc->arg);
@@ -143,36 +147,34 @@ void dcc_make_address(IPADDR *ip, char *host)
}
}
-/* Find DCC record, arg can be NULL */
-DCC_REC *dcc_find_item(int type, const char *nick, const char *arg)
+DCC_REC *dcc_find_request_latest(int type)
{
- DCC_REC *dcc;
- GSList *tmp;
-
- g_return_val_if_fail(nick != NULL, NULL);
+ DCC_REC *latest;
+ GSList *tmp;
+ latest = NULL;
for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next) {
- dcc = tmp->data;
+ DCC_REC *dcc = tmp->data;
- if (dcc->type == type && g_strcasecmp(dcc->nick, nick) == 0 &&
- (arg == NULL || strcmp(dcc->arg, arg) == 0))
- return dcc;
+ if (dcc->type == type && dcc_is_waiting_user(dcc))
+ latest = dcc;
}
- return NULL;
+ return latest;
}
-/* Find DCC record by port # */
-DCC_REC *dcc_find_by_port(const char *nick, int port)
+DCC_REC *dcc_find_request(int type, const char *nick, const char *arg)
{
- DCC_REC *dcc;
GSList *tmp;
+ g_return_val_if_fail(nick != NULL, NULL);
+
for (tmp = dcc_conns; tmp != NULL; tmp = tmp->next) {
- dcc = tmp->data;
+ DCC_REC *dcc = tmp->data;
- if ((dcc->type == DCC_TYPE_GET || dcc->type == DCC_TYPE_SEND) &&
- dcc->port == port && g_strcasecmp(dcc->nick, nick) == 0)
+ if (dcc->type == type && !dcc_is_connected(dcc) &&
+ g_strcasecmp(dcc->nick, nick) == 0 &&
+ (arg == NULL || strcmp(dcc->arg, arg) == 0))
return dcc;
}
@@ -198,6 +200,31 @@ int dcc_str2type(const char *type)
return 0;
}
+GIOChannel *dcc_listen(GIOChannel *interface, IPADDR *ip, int *port)
+{
+ if (net_getsockname(interface, ip, NULL) == -1)
+ return NULL;
+
+ *port = settings_get_int("dcc_port");
+ return net_listen(ip, port);
+}
+
+void dcc_get_address(const char *str, IPADDR *ip)
+{
+ unsigned long addr;
+
+ if (strchr(str, ':') == NULL) {
+ /* normal IPv4 address in 32bit number form */
+ addr = strtoul(str, NULL, 10);
+ ip->family = AF_INET;
+ addr = (unsigned long) ntohl(addr);
+ memcpy(&ip->addr, &addr, 4);
+ } else {
+ /* IPv6 - in standard form */
+ net_host2ip(str, ip);
+ }
+}
+
void dcc_ctcp_message(IRC_SERVER_REC *server, const char *target,
DCC_REC *chat, int notice, const char *msg)
{
@@ -206,15 +233,14 @@ void dcc_ctcp_message(IRC_SERVER_REC *server, const char *target,
if (chat != NULL && chat->sendbuf != NULL) {
/* send it via open DCC chat */
str = g_strdup_printf("%s\001%s\001", chat->mirc_ctcp ? "" :
- notice ? "CTCP_REPLY " : "CTCP_MESSAGE ", msg);
+ notice ? "CTCP_REPLY " :
+ "CTCP_MESSAGE ", msg);
dcc_chat_send(chat, str);
+ g_free(str);
} else {
- str = g_strdup_printf("%s %s :\001%s\001",
- notice ? "NOTICE" : "PRIVMSG", target, msg);
- irc_send_cmd(server, str);
+ irc_send_cmdv(server, "%s %s :\001%s\001",
+ notice ? "NOTICE" : "PRIVMSG", target, msg);
}
-
- g_free(str);
}
/* Server connected, check if there's any open dcc sessions for this ircnet.. */
@@ -252,321 +278,232 @@ static void dcc_server_disconnected(IRC_SERVER_REC *server)
if (dcc->server != server)
continue;
- if (dcc->ircnet == NULL)
- dcc->server = NULL;
- else {
- dcc->server = (IRC_SERVER_REC *) server_find_chatnet(dcc->ircnet);
- if (dcc->server != NULL) {
- g_free(dcc->mynick);
- dcc->mynick = g_strdup(dcc->server->nick);
- }
+ dcc->server = dcc->ircnet == NULL ? NULL :
+ IRC_SERVER(server_find_chatnet(dcc->ircnet));
+ if (dcc->server != NULL) {
+ g_free(dcc->mynick);
+ dcc->mynick = g_strdup(dcc->server->nick);
}
}
}
-static void dcc_get_address(const char *str, IPADDR *ip)
+/* Handle incoming DCC CTCP messages - either from IRC server or DCC chat */
+static void ctcp_msg_dcc(IRC_SERVER_REC *server, const char *data,
+ const char *nick, const char *addr,
+ const char *target, DCC_REC *chat)
{
- unsigned long addr;
+ char *args, *str;
- if (strchr(str, ':') == NULL) {
- /* normal IPv4 address in 32bit number form */
- addr = strtoul(str, NULL, 10);
- ip->family = AF_INET;
- addr = (unsigned long) ntohl(addr);
- memcpy(&ip->addr, &addr, 4);
- } else {
- /* IPv6 - in standard form */
- net_host2ip(str, ip);
+ str = g_strconcat("ctcp msg dcc ", data, NULL);
+ args = strchr(str+13, ' ');
+ if (args != NULL) *args++ = '\0'; else args = "";
+
+ g_strdown(str+13);
+ if (!signal_emit(str, 6, server, args, nick, addr, target, chat)) {
+ signal_emit("default ctcp msg dcc", 6,
+ server, data, nick, addr, target, chat);
}
+ g_free(str);
}
-/* Handle incoming DCC CTCP messages */
-static void dcc_ctcp_msg(IRC_SERVER_REC *server, char *data,
- char *sender, char *sendaddr,
- char *target, DCC_REC *chat)
+/* Handle incoming DCC CTCP replies - either from IRC server or DCC chat */
+static void ctcp_reply_dcc(IRC_SERVER_REC *server, const char *data,
+ const char *nick, const char *addr,
+ const char *target)
{
- char *type, *arg, *addrstr, *portstr, *sizestr, *str;
- void *free_arg;
- const char *masks;
- DCC_REC *dcc, *olddcc;
- long size;
- int dcctype, port;
-
- g_return_if_fail(data != NULL);
- g_return_if_fail(sender != NULL);
-
- if (!cmd_get_params(data, &free_arg, 5 | PARAM_FLAG_NOQUOTES,
- &type, &arg, &addrstr, &portstr, &sizestr))
- return;
-
- if (sscanf(portstr, "%d", &port) != 1) port = 0;
- if (sscanf(sizestr, "%ld", &size) != 1) size = 0;
-
- dcctype = SWAP_SENDGET(dcc_str2type(type));
- olddcc = dcc_find_item(dcctype, sender,
- dcctype == DCC_TYPE_CHAT ? NULL : arg);
- if (olddcc != NULL) {
- /* same DCC request offered again */
- if (olddcc->type == DCC_TYPE_CHAT &&
- olddcc->handle != NULL && olddcc->starttime == 0) {
- /* we requested dcc chat, they requested
- dcc chat from us .. allow it. */
- dcc_destroy(olddcc);
- } else {
- /* if the connection isn't open, update the port,
- otherwise just ignore */
- if (olddcc->handle == NULL)
- olddcc->port = port;
- cmd_params_free(free_arg);
- return;
- }
- }
-
- dcc = dcc_create(dcctype, NULL, sender, arg, server, chat);
- dcc_get_address(addrstr, &dcc->addr);
- net_ip2host(&dcc->addr, dcc->addrstr);
- dcc->port = port;
- dcc->size = size;
-
- switch (dcc->type)
- {
- case DCC_TYPE_GET:
- /* send request */
- signal_emit("dcc request", 2, dcc, sendaddr);
- break;
-
- case DCC_TYPE_CHAT:
- /* send request */
- signal_emit("dcc request", 2, dcc, sendaddr);
-
- masks = settings_get_str("dcc_autochat_masks");
- if (olddcc != NULL ||
- (*masks != '\0' && masks_match(SERVER(server), masks, sender, sendaddr)))
- {
- /* automatically accept chat */
- str = g_strdup_printf("CHAT %s", dcc->nick);
- signal_emit("command dcc", 2, str, server);
- g_free(str);
- }
- break;
-
- case DCC_TYPE_RESUME:
- case DCC_TYPE_ACCEPT:
- /* handle this in dcc-files.c */
- dcc_destroy(dcc);
- break;
-
- default:
- /* unknown DCC command */
- signal_emit("dcc unknown ctcp", 3, data, sender, sendaddr);
- dcc_destroy(dcc);
- break;
- }
-
- cmd_params_free(free_arg);
+ char *args, *str;
+
+ str = g_strconcat("ctcp reply dcc ", data, NULL);
+ args = strchr(str+15, ' ');
+ if (args != NULL) *args++ = '\0'; else args = "";
+
+ g_strdown(str+15);
+ if (!signal_emit(str, 5, server, args, nick, addr, target)) {
+ signal_emit("default ctcp reply dcc", 5,
+ server, data, nick, addr, target);
+ }
+ g_free(str);
}
-/* Handle incoming DCC CTCP replies */
-static void dcc_ctcp_reply(IRC_SERVER_REC *server, char *data,
- char *sender, char *sendaddr)
+/* CTCP REPLY: REJECT */
+static void ctcp_reply_dcc_reject(IRC_SERVER_REC *server, const char *data,
+ const char *nick, const char *addr,
+ DCC_REC *chat)
{
- char *cmd, *subcmd, *args;
- void *free_arg;
- int type;
- DCC_REC *dcc;
-
- g_return_if_fail(data != NULL);
- g_return_if_fail(sender != NULL);
-
- if (!cmd_get_params(data, &free_arg, 3 | PARAM_FLAG_GETREST, &cmd, &subcmd, &args))
- return;
-
- if (g_strcasecmp(cmd, "REJECT") == 0)
- {
- type = dcc_str2type(subcmd);
- dcc = dcc_find_item(type, sender, type == DCC_TYPE_CHAT ? NULL : args);
- if (dcc != NULL)
- {
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
- }
- }
- else
- {
- /* unknown dcc ctcp reply */
- signal_emit("dcc unknown reply", 3, data, sender, sendaddr);
- }
-
- cmd_params_free(free_arg);
+ DCC_REC *dcc;
+ char *typestr, *args;
+ int type;
+
+ typestr = g_strdup(data);
+ args = strchr(typestr, ' ');
+ if (args != NULL) *args++ = '\0'; else args = "";
+
+ type = dcc_str2type(typestr);
+ dcc = dcc_find_request(type, nick,
+ type == DCC_TYPE_CHAT ? NULL : args);
+ if (dcc != NULL) {
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ }
+
+ g_free(typestr);
}
-/* reject DCC request */
+/* Reject a DCC request */
void dcc_reject(DCC_REC *dcc, IRC_SERVER_REC *server)
{
- char *str;
+ g_return_if_fail(dcc != NULL);
- g_return_if_fail(dcc != NULL);
+ if (dcc->server != NULL) server = dcc->server;
+ if (server != NULL && (dcc->type != DCC_TYPE_CHAT ||
+ dcc->starttime == 0)) {
+ signal_emit("dcc rejected", 1, dcc);
+ irc_send_cmdv(server, "NOTICE %s :\001DCC REJECT %s %s\001",
+ dcc->nick, dcc_type2str(SWAP_SENDGET(dcc->type)),
+ dcc->arg);
+ }
- if (dcc->server != NULL) server = dcc->server;
- if (server != NULL && (dcc->type != DCC_TYPE_CHAT || dcc->starttime == 0))
- {
- signal_emit("dcc rejected", 1, dcc);
- str = g_strdup_printf("NOTICE %s :\001DCC REJECT %s %s\001",
- dcc->nick, dcc_type2str(SWAP_SENDGET(dcc->type)), dcc->arg);
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+}
+
+static int dcc_timeout_func(void)
+{
+ GSList *tmp, *next;
+ time_t now;
- irc_send_cmd(server, str);
- g_free(str);
- }
+ now = time(NULL)-settings_get_int("dcc_timeout");
+ for (tmp = dcc_conns; tmp != NULL; tmp = next) {
+ DCC_REC *rec = tmp->data;
- signal_emit("dcc closed", 1, dcc);
- dcc_destroy(dcc);
+ next = tmp->next;
+ if (rec->tagread == -1 && now > rec->created) {
+ /* timed out. */
+ dcc_reject(rec, NULL);
+ }
+ }
+ return 1;
}
-/* SYNTAX: DCC CLOSE <type> <nick> [<file>] */
-static void cmd_dcc_close(char *data, IRC_SERVER_REC *server)
+static void event_no_such_nick(IRC_SERVER_REC *server, char *data)
{
- DCC_REC *dcc;
- GSList *tmp, *next;
- char *type, *nick, *arg;
- void *free_arg;
- gboolean found;
- int itype;
+ char *params, *nick;
+ GSList *tmp, *next;
- g_return_if_fail(data != NULL);
+ g_return_if_fail(data != NULL);
- if (!cmd_get_params(data, &free_arg, 3, &type, &nick, &arg))
- return;
+ params = event_get_params(data, 2, NULL, &nick);
- if (*nick == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
+ /* check if we've send any dcc requests to this nick.. */
+ for (tmp = dcc_conns; tmp != NULL; tmp = next) {
+ DCC_REC *dcc = tmp->data;
- g_strup(type);
- itype = dcc_str2type(type);
- if (itype == 0)
- {
- signal_emit("dcc error unknown type", 1, type);
- cmd_params_free(free_arg);
- return;
- }
-
- dcc = NULL; found = FALSE;
- for (tmp = dcc_conns; tmp != NULL; tmp = next)
- {
- dcc = tmp->data;
- next = tmp->next;
-
- if (dcc->type == itype && g_strcasecmp(nick, dcc->nick) == 0)
- {
- dcc_reject(dcc, server);
- found = TRUE;
- }
- }
-
- if (!found)
- signal_emit("dcc error close not found", 3, type, nick, arg);
-
- cmd_params_free(free_arg);
-}
+ next = tmp->next;
+ if (!dcc_is_connected(dcc) && dcc->server == server &&
+ g_strcasecmp(dcc->nick, nick) == 0) {
+ signal_emit("dcc closed", 1, dcc);
+ dcc_destroy(dcc);
+ }
+ }
-static void cmd_dcc(const char *data, IRC_SERVER_REC *server, void *item)
-{
- command_runsub("dcc", data, server, item);
+ g_free(params);
}
-static int dcc_timeout_func(void)
+/* SYNTAX: DCC CLOSE <type> <nick> [<file>] */
+static void cmd_dcc_close(char *data, IRC_SERVER_REC *server)
{
- GSList *tmp, *next;
- time_t now;
-
- now = time(NULL)-settings_get_int("dcc_timeout");
- for (tmp = dcc_conns; tmp != NULL; tmp = next)
- {
- DCC_REC *rec = tmp->data;
-
- next = tmp->next;
- if (rec->tagread == -1 && now > rec->created)
- {
- /* timed out. */
- dcc_reject(rec, NULL);
- }
- }
- return 1;
-}
+ GSList *tmp, *next;
+ char *typestr, *nick, *arg;
+ void *free_arg;
+ int found, type;
-static void event_no_such_nick(IRC_SERVER_REC *server, char *data)
-{
- char *params, *nick;
- GSList *tmp, *next;
+ g_return_if_fail(data != NULL);
+
+ if (!cmd_get_params(data, &free_arg, 3, &typestr, &nick, &arg))
+ return;
+
+ if (*nick == '\0') cmd_param_error(CMDERR_NOT_ENOUGH_PARAMS);
+
+ g_strup(typestr);
+ type = dcc_str2type(typestr);
+ if (type == 0) {
+ signal_emit("dcc error unknown type", 1, typestr);
+ cmd_params_free(free_arg);
+ return;
+ }
- g_return_if_fail(data != NULL);
+ found = FALSE;
+ for (tmp = dcc_conns; tmp != NULL; tmp = next) {
+ DCC_REC *dcc = tmp->data;
- params = event_get_params(data, 2, NULL, &nick);
+ next = tmp->next;
+ if (dcc->type == type &&
+ g_strcasecmp(dcc->chat_id != NULL ?
+ dcc->chat_id : dcc->nick, nick) == 0 &&
+ (*arg == '\0' || strcmp(dcc->arg, arg) == 0)) {
+ dcc_reject(dcc, server);
+ found = TRUE;
+ }
+ }
- /* check if we've send any dcc requests to this nick.. */
- for (tmp = dcc_conns; tmp != NULL; tmp = next)
- {
- DCC_REC *rec = tmp->data;
+ if (!found) {
+ signal_emit("dcc error close not found", 3,
+ typestr, nick, arg);
+ }
- next = tmp->next;
- if (g_strcasecmp(rec->nick, nick) == 0 && rec->starttime == 0)
- {
- /* timed out. */
- signal_emit("dcc closed", 1, rec);
- dcc_destroy(rec);
- }
- }
+ cmd_params_free(free_arg);
+}
- g_free(params);
+static void cmd_dcc(const char *data, IRC_SERVER_REC *server, void *item)
+{
+ command_runsub("dcc", data, server, item);
}
void irc_dcc_init(void)
{
- dcc_conns = NULL;
- dcc_timeouttag = g_timeout_add(1000, (GSourceFunc) dcc_timeout_func, NULL);
-
- settings_add_bool("dcc", "dcc_autorename", FALSE);
- settings_add_bool("dcc", "dcc_autoget", FALSE);
- settings_add_bool("dcc", "dcc_autoresume", FALSE);
- settings_add_int("dcc", "dcc_max_autoget_size", 1000);
- settings_add_str("dcc", "dcc_download_path", "~");
- settings_add_int("dcc", "dcc_file_create_mode", 644);
- settings_add_str("dcc", "dcc_autoget_masks", "");
- settings_add_str("dcc", "dcc_autochat_masks", "");
-
- settings_add_bool("dcc", "dcc_fast_send", TRUE);
- settings_add_str("dcc", "dcc_upload_path", "~");
-
- settings_add_bool("dcc", "dcc_mirc_ctcp", FALSE);
- settings_add_int("dcc", "dcc_block_size", 2048);
- settings_add_int("dcc", "dcc_port", 0);
- settings_add_int("dcc", "dcc_timeout", 300);
-
- signal_add("server connected", (SIGNAL_FUNC) dcc_server_connected);
- signal_add("server disconnected", (SIGNAL_FUNC) dcc_server_disconnected);
- signal_add("ctcp reply dcc", (SIGNAL_FUNC) dcc_ctcp_reply);
- signal_add("ctcp msg dcc", (SIGNAL_FUNC) dcc_ctcp_msg);
- command_bind("dcc", NULL, (SIGNAL_FUNC) cmd_dcc);
- command_bind("dcc close", NULL, (SIGNAL_FUNC) cmd_dcc_close);
- signal_add("event 401", (SIGNAL_FUNC) event_no_such_nick);
-
- dcc_chat_init();
- dcc_files_init();
+ dcc_conns = NULL;
+ dcc_timeouttag = g_timeout_add(1000, (GSourceFunc) dcc_timeout_func, NULL);
+
+ settings_add_bool("dcc", "dcc_mirc_ctcp", FALSE);
+ settings_add_int("dcc", "dcc_port", 0);
+ settings_add_int("dcc", "dcc_timeout", 300);
+ settings_add_int("dcc", "dcc_block_size", 2048);
+
+ signal_add("server connected", (SIGNAL_FUNC) dcc_server_connected);
+ signal_add("server disconnected", (SIGNAL_FUNC) dcc_server_disconnected);
+ signal_add("ctcp msg dcc", (SIGNAL_FUNC) ctcp_msg_dcc);
+ signal_add("ctcp reply dcc", (SIGNAL_FUNC) ctcp_reply_dcc);
+ signal_add("ctcp reply dcc reject", (SIGNAL_FUNC) ctcp_reply_dcc_reject);
+ command_bind("dcc", NULL, (SIGNAL_FUNC) cmd_dcc);
+ command_bind("dcc close", NULL, (SIGNAL_FUNC) cmd_dcc_close);
+ signal_add("event 401", (SIGNAL_FUNC) event_no_such_nick);
+
+ dcc_chat_init();
+ dcc_get_init();
+ dcc_send_init();
+ dcc_resume_init();
+ dcc_autoget_init();
}
void irc_dcc_deinit(void)
{
- dcc_chat_deinit();
- dcc_files_deinit();
-
- signal_remove("server connected", (SIGNAL_FUNC) dcc_server_connected);
- signal_remove("server disconnected", (SIGNAL_FUNC) dcc_server_disconnected);
- signal_remove("ctcp reply dcc", (SIGNAL_FUNC) dcc_ctcp_reply);
- signal_remove("ctcp msg dcc", (SIGNAL_FUNC) dcc_ctcp_msg);
- command_unbind("dcc", (SIGNAL_FUNC) cmd_dcc);
- command_unbind("dcc close", (SIGNAL_FUNC) cmd_dcc_close);
- signal_remove("event 401", (SIGNAL_FUNC) event_no_such_nick);
-
- g_source_remove(dcc_timeouttag);
-
- while (dcc_conns != NULL)
- dcc_destroy(dcc_conns->data);
+ dcc_chat_deinit();
+ dcc_get_deinit();
+ dcc_send_deinit();
+ dcc_resume_deinit();
+ dcc_autoget_deinit();
+
+ signal_remove("server connected", (SIGNAL_FUNC) dcc_server_connected);
+ signal_remove("server disconnected", (SIGNAL_FUNC) dcc_server_disconnected);
+ signal_remove("ctcp msg dcc", (SIGNAL_FUNC) ctcp_msg_dcc);
+ signal_remove("ctcp reply dcc", (SIGNAL_FUNC) ctcp_reply_dcc);
+ signal_remove("ctcp reply dcc reject", (SIGNAL_FUNC) ctcp_reply_dcc_reject);
+ command_unbind("dcc", (SIGNAL_FUNC) cmd_dcc);
+ command_unbind("dcc close", (SIGNAL_FUNC) cmd_dcc_close);
+ signal_remove("event 401", (SIGNAL_FUNC) event_no_such_nick);
+
+ g_source_remove(dcc_timeouttag);
+
+ while (dcc_conns != NULL)
+ dcc_destroy(dcc_conns->data);
}
diff --git a/src/irc/dcc/dcc.h b/src/irc/dcc/dcc.h
index 88e4effb..eabbef68 100644
--- a/src/irc/dcc/dcc.h
+++ b/src/irc/dcc/dcc.h
@@ -2,6 +2,7 @@
#define __DCC_H
#include "network.h"
+#include "irc-servers.h"
enum {
DCC_TYPE_CHAT = 1,
@@ -11,12 +12,6 @@ enum {
DCC_TYPE_ACCEPT
};
-enum {
- DCC_GET_RENAME = 0, /* this also acts as default */
- DCC_GET_OVERWRITE,
- DCC_GET_RESUME
-};
-
#define SWAP_SENDGET(a) ((a) == DCC_TYPE_SEND ? DCC_TYPE_GET : \
(a) == DCC_TYPE_GET ? DCC_TYPE_SEND : (a))
@@ -25,7 +20,9 @@ typedef struct DCC_REC {
time_t created;
IRC_SERVER_REC *server;
+ char *chat_id; /* unique identifier for dcc chat. usually same as nick. */
char *nick;
+ char *target; /* who the request was sent to - your nick, channel or NULL if you sent the request */
struct DCC_REC *chat; /* if the request came through DCC chat */
@@ -53,6 +50,7 @@ typedef struct DCC_REC {
unsigned int waitforend:1; /* DCC fast send: file is sent, just wait for the replies from the other side */
unsigned int gotalldata:1; /* DCC fast send: got all acks from the other end (needed to make sure the end of transfer works right) */
+ unsigned int file_quoted:1; /* file name was received quoted ("file name") */
unsigned int mirc_ctcp:1; /* DCC chat: Send CTCPs without the CTCP_MESSAGE prefix */
unsigned int connection_lost:1; /* DCC chat: other side closed connection */
unsigned int destroyed:1; /* We're about to destroy this DCC recond */
@@ -72,18 +70,22 @@ extern GSList *dcc_conns;
void dcc_init(void);
void dcc_deinit(void);
-/* Find DCC record, arg can be NULL */
-DCC_REC *dcc_find_item(int type, const char *nick, const char *arg);
-DCC_REC *dcc_find_by_port(const char *nick, int port);
+/* Find waiting DCC requests (non-connected) */
+DCC_REC *dcc_find_request_latest(int type);
+DCC_REC *dcc_find_request(int type, const char *nick, const char *arg);
const char *dcc_type2str(int type);
int dcc_str2type(const char *type);
void dcc_make_address(IPADDR *ip, char *host);
-DCC_REC *dcc_create(int type, GIOChannel *handle, const char *nick,
- const char *arg, IRC_SERVER_REC *server, DCC_REC *chat);
+DCC_REC *dcc_create(int type, const char *nick, const char *arg,
+ IRC_SERVER_REC *server, DCC_REC *chat);
void dcc_destroy(DCC_REC *dcc);
+GIOChannel *dcc_listen(GIOChannel *interface, IPADDR *ip, int *port);
+
+void dcc_get_address(const char *str, IPADDR *ip);
+
/* Send a CTCP message/notify to target. Send the CTCP via DCC chat if
`chat' is specified. */
void dcc_ctcp_message(IRC_SERVER_REC *server, const char *target,
@@ -94,7 +96,19 @@ void dcc_chat_send(DCC_REC *dcc, const char *data);
/* If `item' is a query of a =nick, return DCC chat record of nick */
DCC_REC *item_get_dcc(WI_ITEM_REC *item);
-/* reject DCC request */
+/* Reject a DCC request */
void dcc_reject(DCC_REC *dcc, IRC_SERVER_REC *server);
+/* fully connected? */
+#define dcc_is_connected(dcc) \
+ ((dcc)->starttime != 0)
+
+/* not connected, we're waiting for other side to connect */
+#define dcc_is_listening(dcc) \
+ ((dcc)->handle != NULL && (dcc)->starttime == 0)
+
+/* not connected, waiting for user to accept it */
+#define dcc_is_waiting_user(dcc) \
+ ((dcc)->handle == NULL)
+
#endif