diff src/protocols/icq/filesession.c @ 2086:424a40f12a6c

[gaim-migrate @ 2096] moving protocols from plugins/ to src/protocols. making it so that you can select which protocols are compiled statically. committer: Tailor Script <tailor@pidgin.im>
author Eric Warmenhoven <eric@warmenhoven.org>
date Tue, 31 Jul 2001 01:00:39 +0000
parents
children f0a2a9afdb77
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/protocols/icq/filesession.c	Tue Jul 31 01:00:39 2001 +0000
@@ -0,0 +1,292 @@
+/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+
+/*
+ * $Id: filesession.c 2096 2001-07-31 01:00:39Z warmenhoven $
+ *
+ * Copyright (C) 1998-2001, Denis V. Dmitrienko <denis@null.net> and
+ *                          Bill Soudan <soudan@kde.org>
+ *
+ * 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., 675 Mass Ave, Cambridge, MA 02139, USA.
+ *
+ */
+
+#include <stdlib.h>
+#include <fcntl.h>
+#include <sys/stat.h>
+
+#ifdef _MSVC_
+#include <io.h>
+#define open _open
+#define close _close
+#define read _read
+#define write _write
+#endif
+
+#include "icqlib.h"
+#include "filesession.h"
+#include "stdpackets.h"
+
+icq_FileSession *icq_FileSessionNew(icq_Link *icqlink)
+{
+  icq_FileSession *p=(icq_FileSession *)malloc(sizeof(icq_FileSession));
+
+  if (p)
+  {
+    p->status=0;
+    p->id=0L;
+    p->icqlink=icqlink;
+    p->tcplink=NULL;
+    p->current_fd=-1;
+    p->current_file_num=0;
+    p->current_file_progress=0;
+    p->current_file_size=0;
+    p->files=0L;
+    p->current_speed=100;
+    p->total_bytes=0;
+    p->total_files=0;
+    p->total_transferred_bytes=0;
+    p->working_dir[0]=0;
+    p->user_data=NULL;
+    icq_ListInsert(icqlink->d->icq_FileSessions, 0, p);
+  }
+	
+  return p;
+}
+
+void icq_FileSessionDelete(void *pv)
+{
+  icq_FileSession *p=(icq_FileSession *)pv;
+
+  invoke_callback(p->icqlink, icq_FileNotify)(p, FILE_NOTIFY_CLOSE, 0, 
+    NULL);
+
+  if(p->files) {
+    char **p2=p->files;
+    while(*p2)
+      free(*(p2++));
+    free(p->files);
+    p->files=NULL;
+  }
+
+  if (p->current_fd > -1 ) {
+     close(p->current_fd);
+     p->current_fd=-1;
+  }
+
+  free(p);
+}
+
+int _icq_FindFileSession(void *p, va_list data)
+{
+  icq_FileSession *psession=(icq_FileSession *)p;
+  DWORD uin=va_arg(data, DWORD);
+  unsigned long id=va_arg(data, unsigned long);
+  
+  return (psession->remote_uin == uin) && ( id ? (psession->id == id) : 1 );
+
+}
+
+icq_FileSession *icq_FindFileSession(icq_Link *icqlink, DWORD uin,
+  unsigned long id)
+{
+  return icq_ListTraverse(icqlink->d->icq_FileSessions, _icq_FindFileSession, 
+    uin, id);
+}
+
+void icq_FileSessionSetStatus(icq_FileSession *p, int status)
+{
+  if(status!=p->status)
+  {
+    p->status=status;
+    if(p->id)
+      invoke_callback(p->icqlink, icq_FileNotify)(p, FILE_NOTIFY_STATUS,
+        status, NULL);
+    if (status == FILE_STATUS_SENDING)
+      icq_SocketSetHandler(p->tcplink->socket, ICQ_SOCKET_WRITE,
+        icq_FileSessionSendData, p);
+    else
+      icq_SocketSetHandler(p->tcplink->socket, ICQ_SOCKET_WRITE, NULL, NULL);
+  }
+}
+
+void icq_FileSessionSetHandle(icq_FileSession *p, const char *handle)
+{
+  strncpy(p->remote_handle, handle, 64);
+}
+
+void icq_FileSessionSetCurrentFile(icq_FileSession *p, const char *filename)
+{
+#ifdef _WIN32
+  struct _stat file_status;
+#else
+  struct stat file_status;
+#endif
+  char file[1024];
+
+  strcpy(file, p->working_dir);
+  strcat(file, filename);
+
+  if (p->current_fd>-1) {
+    close(p->current_fd);
+    p->current_fd=-1;
+  }
+
+  strncpy(p->current_file, file, 64);
+  p->current_file_progress=0;
+
+  /* does the file already exist? */
+#ifdef _WIN32
+  if (_stat(file, &file_status)==0) {
+#else
+  if (stat(file, &file_status)==0) {
+#endif
+    p->current_file_progress=file_status.st_size;
+    p->total_transferred_bytes+=file_status.st_size;
+#ifdef _WIN32
+    p->current_fd=open(file, _O_WRONLY | _O_APPEND | _O_BINARY);
+#else
+    p->current_fd=open(file, O_WRONLY | O_APPEND);
+#endif
+  } else {
+#ifdef _WIN32
+    p->current_fd=open(file, _O_WRONLY | _O_CREAT | _O_BINARY, 
+      _S_IREAD|_S_IWRITE);
+#else
+    p->current_fd=open(file, O_WRONLY | O_CREAT, S_IRWXU);
+#endif
+  }
+
+  /* make sure we have a valid filehandle */
+  if (p->current_fd == -1)
+    perror("couldn't open file: ");
+      
+}
+
+void icq_FileSessionPrepareNextFile(icq_FileSession *p)
+{
+  int i=0;
+  char **files=p->files;
+
+  p->current_file_num++;
+
+  while(*files) {
+    i++;
+    if(i==p->current_file_num)
+      break;
+    else
+      files++;
+  }
+
+  if(*files) {
+#ifdef _WIN32
+    struct _stat file_status;
+#else
+    struct stat file_status;
+#endif
+
+    if (p->current_fd>-1) {
+       close(p->current_fd);
+       p->current_fd=-1;
+    }
+
+#ifdef _WIN32
+    if (_stat(*files, &file_status)==0) {
+       char *basename=*files;
+       char *pos=strrchr(basename, '\\');
+#else
+    if (stat(*files, &file_status)==0) {
+       char *basename=*files;
+       char *pos=strrchr(basename, '/');
+#endif
+       if(pos) basename=pos+1;
+       strncpy(p->current_file, basename, 64);
+       p->current_file_progress=0;
+       p->current_file_size=file_status.st_size;
+#ifdef _WIN32
+       p->current_fd=open(*files, _O_RDONLY | _O_BINARY);
+#else
+       p->current_fd=open(*files, O_RDONLY);
+#endif
+    }
+
+    /* make sure we have a valid filehandle */
+    if (p->current_fd == -1)
+       perror("couldn't open file: ");
+  }     
+}        
+
+void icq_FileSessionSendData(icq_FileSession *p)
+{
+  /* for now just send a packet at a time */
+  char buffer[2048];
+  int count=read(p->current_fd, buffer, 2048);
+
+  if(count>0) {
+      icq_Packet *p2=icq_TCPCreateFile06Packet(count, buffer);
+      icq_TCPLinkSend(p->tcplink, p2);
+      p->total_transferred_bytes+=count;
+      p->current_file_progress+=count;
+      icq_FileSessionSetStatus(p, FILE_STATUS_SENDING);
+
+      invoke_callback(p->icqlink, icq_FileNotify)(p, FILE_NOTIFY_DATAPACKET,
+        count, buffer);
+  }
+
+  /* done transmitting if read returns less that 2048 bytes */
+  if(count<2048)
+      icq_FileSessionClose(p);
+
+  return;
+}
+
+/* public */
+
+void icq_FileSessionSetSpeed(icq_FileSession *p, int speed)
+{
+  icq_Packet *packet=icq_TCPCreateFile05Packet(speed);
+
+  icq_TCPLinkSend(p->tcplink, packet);
+}
+
+void icq_FileSessionClose(icq_FileSession *p)
+{
+  icq_TCPLink *plink=p->tcplink;
+
+  /* TODO: handle closing already unallocated filesession? */
+
+  /* if we're attached to a tcplink, unattach so the link doesn't try
+   * to close us, and then close the tcplink */
+  if (plink)
+  {
+    plink->session=0L;
+    icq_TCPLinkClose(plink);
+  }
+
+  icq_ListRemove(p->icqlink->d->icq_FileSessions, p);
+  icq_FileSessionDelete(p);
+}   
+
+void icq_FileSessionSetWorkingDir(icq_FileSession *p, const char *dir)
+{
+  int length = sizeof(p->working_dir);
+  strncpy(p->working_dir, dir, length);
+  p->working_dir[length-1]='\0';
+}  
+
+void icq_FileSessionSetFiles(icq_FileSession *p, char **files)
+{
+  p->files=files;
+}
+