summaryrefslogtreecommitdiff
path: root/src/irc/dcc/dcc-files.c
diff options
context:
space:
mode:
authorTimo Sirainen <cras@irssi.org>2001-01-18 02:30:59 +0000
committercras <cras@dbcabf3a-b0e7-0310-adc4-f8d773084564>2001-01-18 02:30:59 +0000
commitce6e5a12f92f5375da6bcf499be666210856dbf3 (patch)
tree172f342986f4f86aaf1ccf7f99331c642ef617f5 /src/irc/dcc/dcc-files.c
parentbabf7c77ac9344c8b12ca2ce87096e7ed079b687 (diff)
downloadirssi-ce6e5a12f92f5375da6bcf499be666210856dbf3.zip
DCC cleanups - half rewrite. New features: file names with spaces work
properly, you can have multiple dcc chats with same people (or more useful, same nick in different ircnets), /DCC CHAT|GET|RESUME with no arguments accepts the last request, notifies if dcc request was sent to channel, warns about connecting to lowports, /SET dcc_autoget_lowports specifies if autogetting should work with lowports, complains of invalid dcc ctcps instead of ignoring. And fixed /SET dcc_autorename OFF which didn't work before. git-svn-id: http://svn.irssi.org/repos/irssi/trunk@1135 dbcabf3a-b0e7-0310-adc4-f8d773084564
Diffstat (limited to 'src/irc/dcc/dcc-files.c')
-rw-r--r--src/irc/dcc/dcc-files.c667
1 files changed, 0 insertions, 667 deletions
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);
-}