Replace functions which called once with their bodies
[pidgin-git.git] / libpurple / protocols / oscar / oft.c
blob6d506e60f451d15c98f27afbccdce017df96f747
1 /*
2 * Purple's oscar protocol plugin
3 * This file is the legal property of its developers.
4 * Please see the AUTHORS file distributed alongside this file.
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA
22 * I feel like this is a good place to explain OFT, so I'm going to
23 * do just that. Each OFT packet has a header type. I guess this
24 * is pretty similar to the subtype of a SNAC packet. The type
25 * basically tells the other client the meaning of the OFT packet.
26 * There are two distinct types of file transfer, which I usually
27 * call "sendfile" and "getfile." Sendfile is when you send a file
28 * to another AIM user. Getfile is when you share a group of files,
29 * and other users request that you send them the files.
31 * A typical sendfile file transfer goes like this:
32 * 1) Sender sends a channel 2 ICBM telling the other user that
33 * we want to send them a file. At the same time, we open a
34 * listener socket (this should be done before sending the
35 * ICBM) on some port, and wait for them to connect to us.
36 * The ICBM we sent should contain our IP address and the port
37 * number that we're listening on.
38 * 2) The receiver connects to the sender on the given IP address
39 * and port. After the connection is established, the receiver
40 * sends an ICBM signifying that we are ready and waiting.
41 * 3) The sender sends an OFT PROMPT message over the OFT
42 * connection.
43 * 4) The receiver of the file sends back an exact copy of this
44 * OFT packet, except the cookie is filled in with the cookie
45 * from the ICBM. I think this might be an attempt to verify
46 * that the user that is connected is actually the guy that
47 * we sent the ICBM to. Oh, I've been calling this the ACK.
48 * 5) The sender starts sending raw data across the connection
49 * until the entire file has been sent.
50 * 6) The receiver knows the file is finished because the sender
51 * sent the file size in an earlier OFT packet. So then the
52 * receiver sends the DONE thingy (after filling in the
53 * "received" checksum and size) and closes the connection.
56 #include "oscar.h"
57 #include "peer.h"
59 #include "util.h"
61 #define CHECKSUM_BUFFER_SIZE 256 * 1024
63 struct _ChecksumData
65 PeerConnection *conn;
66 PurpleXfer *xfer;
67 GSourceFunc callback;
68 size_t size;
69 guint32 checksum;
70 size_t total;
71 FILE *file;
72 guint8 buffer[CHECKSUM_BUFFER_SIZE];
73 guint timer;
76 void
77 peer_oft_checksum_destroy(ChecksumData *checksum_data)
79 checksum_data->conn->checksum_data = NULL;
80 fclose(checksum_data->file);
81 if (checksum_data->timer > 0)
82 g_source_remove(checksum_data->timer);
83 g_free(checksum_data);
86 /**
87 * Calculate oft checksum of buffer
89 * Prevcheck should be 0xFFFF0000 when starting a checksum of a file. The
90 * checksum is kind of a rolling checksum thing, so each time you get bytes
91 * of a file you just call this puppy and it updates the checksum. You can
92 * calculate the checksum of an entire file by calling this in a while or a
93 * for loop, or something.
95 * Thanks to Graham Booker for providing this improved checksum routine,
96 * which is simpler and should be more accurate than Josh Myer's original
97 * code. -- wtm
99 * This algorithm works every time I have tried it. The other fails
100 * sometimes. So, AOL who thought this up? It has got to be the weirdest
101 * checksum I have ever seen.
103 * @param buffer Buffer of data to checksum. Man I'd like to buff her...
104 * @param bufsize Size of buffer.
105 * @param prevchecksum Previous checksum.
106 * @param odd Whether an odd number of bytes have been processed before this call
108 static guint32
109 peer_oft_checksum_chunk(const guint8 *buffer, int bufferlen, guint32 prevchecksum, int odd)
111 guint32 checksum, oldchecksum;
112 int i = 0;
113 unsigned short val;
115 checksum = (prevchecksum >> 16) & 0xffff;
116 if (odd)
119 * This is one hell of a hack, but it should always work.
120 * Essentially, I am reindexing the array so that index 1
121 * is the first element. Since the odd and even bytes are
122 * detected by the index number.
124 i = 1;
125 bufferlen++;
126 buffer--;
128 for (; i < bufferlen; i++)
130 oldchecksum = checksum;
131 if (i & 1)
132 val = buffer[i];
133 else
134 val = buffer[i] << 8;
135 checksum -= val;
137 * The following appears to be necessary.... It happens
138 * every once in a while and the checksum doesn't fail.
140 if (checksum > oldchecksum)
141 checksum--;
143 checksum = ((checksum & 0x0000ffff) + (checksum >> 16));
144 checksum = ((checksum & 0x0000ffff) + (checksum >> 16));
145 return checksum << 16;
148 static gboolean
149 peer_oft_checksum_file_piece(gpointer data)
151 ChecksumData *checksum_data;
152 gboolean repeat;
154 checksum_data = data;
155 repeat = FALSE;
157 if (checksum_data->total < checksum_data->size)
159 size_t bytes = MIN(CHECKSUM_BUFFER_SIZE,
160 checksum_data->size - checksum_data->total);
162 bytes = fread(checksum_data->buffer, 1, bytes, checksum_data->file);
163 if (bytes != 0)
165 checksum_data->checksum = peer_oft_checksum_chunk(checksum_data->buffer, bytes, checksum_data->checksum, checksum_data->total & 1);
166 checksum_data->total += bytes;
167 repeat = TRUE;
171 if (!repeat)
173 purple_debug_info("oscar", "Checksum of %s calculated\n",
174 purple_xfer_get_local_filename(checksum_data->xfer));
175 if (checksum_data->callback != NULL)
176 checksum_data->callback(checksum_data);
177 peer_oft_checksum_destroy(checksum_data);
180 return repeat;
184 * Calculate oft checksum of a file in a series of calls to
185 * peer_oft_checksum_file_piece(). We do it this way because
186 * calculating the checksum on large files can take a long time,
187 * and we want to return control to the UI so that the application
188 * doesn't appear completely frozen.
190 * @param conn The connection used for this file transfer.
191 * @param xfer The file transfer needing this checksum.
192 * @param callback The function to call upon calculation of the checksum.
193 * @param size The maximum size to check.
196 static void
197 peer_oft_checksum_file(PeerConnection *conn, PurpleXfer *xfer, GSourceFunc callback, size_t size)
199 ChecksumData *checksum_data;
201 purple_debug_info("oscar", "Calculating checksum of %s\n",
202 purple_xfer_get_local_filename(xfer));
204 checksum_data = g_malloc0(sizeof(ChecksumData));
205 checksum_data->conn = conn;
206 checksum_data->xfer = xfer;
207 checksum_data->callback = callback;
208 checksum_data->size = size;
209 checksum_data->checksum = 0xffff0000;
210 checksum_data->file = g_fopen(purple_xfer_get_local_filename(xfer), "rb");
212 if (checksum_data->file == NULL)
214 purple_debug_error("oscar", "Unable to open %s for checksumming: %s\n",
215 purple_xfer_get_local_filename(xfer), g_strerror(errno));
216 callback(checksum_data);
217 g_free(checksum_data);
219 else
221 checksum_data->timer = g_timeout_add(10,
222 peer_oft_checksum_file_piece, checksum_data);
223 conn->checksum_data = checksum_data;
227 static void
228 peer_oft_copy_xfer_data(PeerConnection *conn, OftFrame *frame)
230 g_free(conn->xferdata.name);
232 memcpy(&(conn->xferdata), frame, sizeof(OftFrame));
233 conn->xferdata.name = g_memdup(frame->name, frame->name_length);
237 * Free any OFT related data.
239 void
240 peer_oft_close(PeerConnection *conn)
243 * If cancelled by local user, and we're receiving a file, and
244 * we're not connected/ready then send an ICBM cancel message.
246 if ((purple_xfer_get_status(conn->xfer) == PURPLE_XFER_STATUS_CANCEL_LOCAL) &&
247 !conn->ready)
249 aim_im_sendch2_cancel(conn);
252 if (conn->sending_data_timer != 0)
254 g_source_remove(conn->sending_data_timer);
255 conn->sending_data_timer = 0;
260 * Write the given OftFrame to a ByteStream and send it out
261 * on the established PeerConnection.
263 static void
264 peer_oft_send(PeerConnection *conn, OftFrame *frame)
266 size_t length;
267 ByteStream bs;
269 length = 192 + frame->name_length;
270 byte_stream_new(&bs, length);
271 byte_stream_putraw(&bs, conn->magic, 4);
272 byte_stream_put16(&bs, length);
273 byte_stream_put16(&bs, frame->type);
274 byte_stream_putraw(&bs, frame->cookie, 8);
275 byte_stream_put16(&bs, frame->encrypt);
276 byte_stream_put16(&bs, frame->compress);
277 byte_stream_put16(&bs, frame->totfiles);
278 byte_stream_put16(&bs, frame->filesleft);
279 byte_stream_put16(&bs, frame->totparts);
280 byte_stream_put16(&bs, frame->partsleft);
281 byte_stream_put32(&bs, frame->totsize);
282 byte_stream_put32(&bs, frame->size);
283 byte_stream_put32(&bs, frame->modtime);
284 byte_stream_put32(&bs, frame->checksum);
285 byte_stream_put32(&bs, frame->rfrcsum);
286 byte_stream_put32(&bs, frame->rfsize);
287 byte_stream_put32(&bs, frame->cretime);
288 byte_stream_put32(&bs, frame->rfcsum);
289 byte_stream_put32(&bs, frame->nrecvd);
290 byte_stream_put32(&bs, frame->recvcsum);
291 byte_stream_putraw(&bs, frame->idstring, 32);
292 byte_stream_put8(&bs, frame->flags);
293 byte_stream_put8(&bs, frame->lnameoffset);
294 byte_stream_put8(&bs, frame->lsizeoffset);
295 byte_stream_putraw(&bs, frame->dummy, 69);
296 byte_stream_putraw(&bs, frame->macfileinfo, 16);
297 byte_stream_put16(&bs, frame->nencode);
298 byte_stream_put16(&bs, frame->nlanguage);
300 * The name can be more than 64 characters, but if it is less than
301 * 64 characters it is padded with NULLs.
303 byte_stream_putraw(&bs, frame->name, frame->name_length);
305 peer_connection_send(conn, &bs);
307 byte_stream_destroy(&bs);
310 void
311 peer_oft_send_prompt(PeerConnection *conn)
313 conn->xferdata.type = PEER_TYPE_PROMPT;
314 peer_oft_send(conn, &conn->xferdata);
317 static void
318 peer_oft_send_ack(PeerConnection *conn)
320 conn->xferdata.type = PEER_TYPE_ACK;
322 /* Fill in the cookie */
323 memcpy(conn->xferdata.cookie, conn->cookie, 8);
325 peer_oft_send(conn, &conn->xferdata);
328 static void
329 peer_oft_send_resume_accept(PeerConnection *conn)
331 conn->xferdata.type = PEER_TYPE_RESUMEACCEPT;
333 /* Fill in the cookie */
334 memcpy(conn->xferdata.cookie, conn->cookie, 8);
336 peer_oft_send(conn, &conn->xferdata);
339 static void
340 peer_oft_send_done(PeerConnection *conn)
342 conn->xferdata.type = PEER_TYPE_DONE;
343 conn->xferdata.rfrcsum = 0xffff0000;
344 conn->xferdata.nrecvd = purple_xfer_get_bytes_sent(conn->xfer);
345 peer_oft_send(conn, &conn->xferdata);
349 * This function exists so that we don't remove the outgoing
350 * data watcher while we're still sending data. In most cases
351 * any data we're sending will be instantly wisked away to a TCP
352 * buffer maintained by our operating system... but we want to
353 * make sure the core doesn't start sending file data while
354 * we're still sending OFT frame data. That would be bad.
356 static gboolean
357 start_transfer_when_done_sending_data(gpointer data)
359 PeerConnection *conn;
361 conn = data;
363 if (purple_circular_buffer_get_max_read(conn->buffer_outgoing) == 0)
365 int fd = conn->fd;
366 conn->sending_data_timer = 0;
367 conn->fd = -1;
368 purple_xfer_start(conn->xfer, fd, NULL, 0);
369 return FALSE;
372 return TRUE;
376 * This function is similar to the above function, except instead
377 * of starting the xfer it will destroy the connection. This is
378 * used when you want to send one final message across the peer
379 * connection, and then close everything.
381 static gboolean
382 destroy_connection_when_done_sending_data(gpointer data)
384 PeerConnection *conn;
386 conn = data;
388 if (purple_circular_buffer_get_max_read(conn->buffer_outgoing) == 0)
390 conn->sending_data_timer = 0;
391 peer_connection_destroy(conn, conn->disconnect_reason, NULL);
392 return FALSE;
395 return TRUE;
399 * This is called when a buddy sends us some file info. This happens when they
400 * are sending a file to you, and you have just established a connection to them.
401 * You should send them the exact same info except use the real cookie. We also
402 * get like totally ready to like, receive the file, kay?
404 static void
405 peer_oft_recv_frame_prompt(PeerConnection *conn, OftFrame *frame)
407 /* Record the file information and send an ack */
408 peer_oft_copy_xfer_data(conn, frame);
409 peer_oft_send_ack(conn);
411 /* Remove our watchers and use the file transfer watchers in the core */
412 purple_input_remove(conn->watcher_incoming);
413 conn->watcher_incoming = 0;
414 conn->sending_data_timer = g_timeout_add(100,
415 start_transfer_when_done_sending_data, conn);
419 * We are sending a file to someone else. They have just acknowledged our
420 * prompt, so we want to start sending data like there's no tomorrow.
422 static void
423 peer_oft_recv_frame_ack(PeerConnection *conn, OftFrame *frame)
425 if (memcmp(conn->cookie, frame->cookie, 8) != 0)
427 purple_debug_info("oscar", "Received an incorrect cookie. "
428 "Closing connection.\n");
429 peer_connection_destroy(conn, OSCAR_DISCONNECT_INVALID_DATA, NULL);
430 return;
433 /* Remove our watchers and use the file transfer watchers in the core */
434 purple_input_remove(conn->watcher_incoming);
435 conn->watcher_incoming = 0;
436 conn->sending_data_timer = g_timeout_add(100,
437 start_transfer_when_done_sending_data, conn);
440 static gboolean
441 peer_oft_recv_frame_resume_checksum_calculated_cb(gpointer data)
443 ChecksumData *checksum_data;
444 PeerConnection *conn;
446 checksum_data = data;
447 conn = checksum_data->conn;
449 /* Check the checksums here. If not match, don't allow resume */
450 if (checksum_data->checksum != conn->xferdata.recvcsum || checksum_data->total != conn->xferdata.nrecvd)
452 /* Reset internal structure */
453 conn->xferdata.recvcsum = 0xffff0000;
454 conn->xferdata.rfrcsum = 0xffff0000;
455 conn->xferdata.nrecvd = 0;
457 else
458 /* Accept the change */
459 purple_xfer_set_bytes_sent(checksum_data->xfer, conn->xferdata.nrecvd);
461 peer_oft_send_resume_accept(conn);
463 return FALSE;
467 * We are sending a file to someone else. They have just acknowledged our
468 * prompt and are asking to resume, so we accept their resume and await
469 * a resume ack.
471 static void
472 peer_oft_recv_frame_resume(PeerConnection *conn, OftFrame *frame)
474 if (memcmp(conn->cookie, frame->cookie, 8) != 0)
476 purple_debug_info("oscar", "Received an incorrect cookie. "
477 "Closing connection.\n");
478 peer_connection_destroy(conn, OSCAR_DISCONNECT_INVALID_DATA, NULL);
479 return;
482 /* Copy resume data into internal structure */
483 conn->xferdata.recvcsum = frame->recvcsum;
484 conn->xferdata.rfrcsum = frame->rfrcsum;
485 conn->xferdata.nrecvd = frame->nrecvd;
487 peer_oft_checksum_file(conn, conn->xfer,
488 peer_oft_recv_frame_resume_checksum_calculated_cb,
489 frame->nrecvd);
493 * We just sent a file to someone. They said they got it and everything,
494 * so we can close our direct connection and what not.
496 static void
497 peer_oft_recv_frame_done(PeerConnection *conn, OftFrame *frame)
500 * The core ft code sets the xfer to completed automatically if we've
501 * sent all bytes to the other user. But this function can be called
502 * even if we haven't sent all bytes to the other user (in the case
503 * where the user already has this file on their computer and the
504 * checksum matches).
506 if (!purple_xfer_is_completed(conn->xfer))
507 purple_xfer_set_completed(conn->xfer, TRUE);
509 purple_input_remove(conn->watcher_incoming);
510 conn->watcher_incoming = 0;
511 purple_xfer_set_fd(conn->xfer, conn->fd);
512 conn->fd = -1;
513 conn->disconnect_reason = OSCAR_DISCONNECT_DONE;
514 peer_connection_schedule_destroy(conn, conn->disconnect_reason, NULL);
518 * Handle an incoming OftFrame. If there is a payload associated
519 * with this frame, then we remove the old watcher and add the
520 * OFT watcher to read in the payload.
522 void
523 peer_oft_recv_frame(PeerConnection *conn, ByteStream *bs)
525 OftFrame frame;
527 frame.type = byte_stream_get16(bs);
528 byte_stream_getrawbuf(bs, frame.cookie, 8);
529 frame.encrypt = byte_stream_get16(bs);
530 frame.compress = byte_stream_get16(bs);
531 frame.totfiles = byte_stream_get16(bs);
532 frame.filesleft = byte_stream_get16(bs);
533 frame.totparts = byte_stream_get16(bs);
534 frame.partsleft = byte_stream_get16(bs);
535 frame.totsize = byte_stream_get32(bs);
536 frame.size = byte_stream_get32(bs);
537 frame.modtime = byte_stream_get32(bs);
538 frame.checksum = byte_stream_get32(bs);
539 frame.rfrcsum = byte_stream_get32(bs);
540 frame.rfsize = byte_stream_get32(bs);
541 frame.cretime = byte_stream_get32(bs);
542 frame.rfcsum = byte_stream_get32(bs);
543 frame.nrecvd = byte_stream_get32(bs);
544 frame.recvcsum = byte_stream_get32(bs);
545 byte_stream_getrawbuf(bs, frame.idstring, 32);
546 frame.flags = byte_stream_get8(bs);
547 frame.lnameoffset = byte_stream_get8(bs);
548 frame.lsizeoffset = byte_stream_get8(bs);
549 byte_stream_getrawbuf(bs, frame.dummy, 69);
550 byte_stream_getrawbuf(bs, frame.macfileinfo, 16);
551 frame.nencode = byte_stream_get16(bs);
552 frame.nlanguage = byte_stream_get16(bs);
553 frame.name_length = bs->len - 186;
554 frame.name = byte_stream_getraw(bs, frame.name_length);
556 purple_debug_info("oscar", "Incoming OFT frame from %s with "
557 "type=0x%04x\n", conn->bn, frame.type);
559 /* TODOFT: peer_oft_dirconvert_fromstupid(frame->name); */
561 switch(frame.type)
563 case PEER_TYPE_PROMPT:
564 peer_oft_recv_frame_prompt(conn, &frame);
565 break;
566 case PEER_TYPE_ACK:
567 case PEER_TYPE_RESUMEACK:
568 peer_oft_recv_frame_ack(conn, &frame);
569 break;
570 case PEER_TYPE_RESUME:
571 peer_oft_recv_frame_resume(conn, &frame);
572 break;
573 case PEER_TYPE_DONE:
574 peer_oft_recv_frame_done(conn, &frame);
575 break;
576 default:
577 break;
580 g_free(frame.name);
583 /*******************************************************************/
584 /* Begin PurpleXfer callbacks for use when receiving a file */
585 /*******************************************************************/
587 void
588 peer_oft_recvcb_init(PurpleXfer *xfer)
590 PeerConnection *conn;
592 conn = oscar_xfer_get_peer_connection(OSCAR_XFER(xfer));
593 conn->flags |= PEER_CONNECTION_FLAG_APPROVED;
594 peer_connection_trynext(conn);
597 void
598 peer_oft_recvcb_end(PurpleXfer *xfer)
600 PeerConnection *conn;
602 conn = oscar_xfer_get_peer_connection(OSCAR_XFER(xfer));
604 /* Tell the other person that we've received everything */
605 conn->fd = purple_xfer_get_fd(conn->xfer);
606 purple_xfer_set_fd(conn->xfer, -1);
607 peer_oft_send_done(conn);
609 conn->disconnect_reason = OSCAR_DISCONNECT_DONE;
610 conn->sending_data_timer = g_timeout_add(100,
611 destroy_connection_when_done_sending_data, conn);
614 void
615 peer_oft_recvcb_ack_recv(PurpleXfer *xfer, const guchar *buffer, size_t size)
617 PeerConnection *conn;
619 /* Update our rolling checksum. Like Walmart, yo. */
620 conn = oscar_xfer_get_peer_connection(OSCAR_XFER(xfer));
621 conn->xferdata.recvcsum = peer_oft_checksum_chunk(buffer,
622 size, conn->xferdata.recvcsum, purple_xfer_get_bytes_sent(xfer) & 1);
625 /*******************************************************************/
626 /* End PurpleXfer callbacks for use when receiving a file */
627 /*******************************************************************/
629 /*******************************************************************/
630 /* Begin PurpleXfer callbacks for use when sending a file */
631 /*******************************************************************/
633 static gboolean
634 peer_oft_checksum_calculated_cb(gpointer data)
636 ChecksumData *checksum_data;
637 PeerConnection *conn;
639 checksum_data = data;
640 conn = checksum_data->conn;
642 conn->xferdata.checksum = checksum_data->checksum;
644 /* Start the connection process */
645 peer_connection_trynext(checksum_data->conn);
647 return FALSE;
650 void
651 peer_oft_sendcb_init(PurpleXfer *xfer)
653 PeerConnection *conn;
654 goffset size;
656 conn = oscar_xfer_get_peer_connection(OSCAR_XFER(xfer));
657 conn->flags |= PEER_CONNECTION_FLAG_APPROVED;
659 /* Make sure the file size can be represented in 32 bits */
660 size = purple_xfer_get_size(xfer);
661 if (size > G_MAXUINT32)
663 gchar *tmp, *size1, *size2;
664 size1 = g_format_size(size);
665 size2 = g_format_size(G_MAXUINT32);
666 tmp = g_strdup_printf(_("File %s is %s, which is larger than "
667 "the maximum size of %s."),
668 purple_xfer_get_local_filename(xfer), size1, size2);
669 purple_xfer_error(purple_xfer_get_xfer_type(xfer),
670 purple_xfer_get_account(xfer), purple_xfer_get_remote_user(xfer), tmp);
671 g_free(size1);
672 g_free(size2);
673 g_free(tmp);
674 peer_connection_destroy(conn, OSCAR_DISCONNECT_LOCAL_CLOSED, NULL);
675 return;
678 /* Keep track of file transfer info */
679 conn->xferdata.totfiles = 1;
680 conn->xferdata.filesleft = 1;
681 conn->xferdata.totparts = 1;
682 conn->xferdata.partsleft = 1;
683 conn->xferdata.totsize = size;
684 conn->xferdata.size = size;
685 conn->xferdata.checksum = 0xffff0000;
686 conn->xferdata.rfrcsum = 0xffff0000;
687 conn->xferdata.rfcsum = 0xffff0000;
688 conn->xferdata.recvcsum = 0xffff0000;
689 strncpy((gchar *)conn->xferdata.idstring, "Cool FileXfer", 31);
690 conn->xferdata.modtime = 0;
691 conn->xferdata.cretime = 0;
692 purple_xfer_set_filename(xfer, g_path_get_basename(purple_xfer_get_local_filename(xfer)));
693 conn->xferdata.name_length = MAX(64, strlen(purple_xfer_get_filename(xfer)) + 1);
694 conn->xferdata.name = (guchar *)g_strndup(purple_xfer_get_filename(xfer), conn->xferdata.name_length - 1);
696 peer_oft_checksum_file(conn, xfer,
697 peer_oft_checksum_calculated_cb, G_MAXUINT32);
701 * AIM file transfers aren't really meant to be thought
702 * of as a transferring just a single file. The rendezvous
703 * establishes a connection between two computers, and then
704 * those computers can use the same connection for transferring
705 * multiple files. So we don't want the Purple core up and closing
706 * the socket all willy-nilly. We want to do that in the oscar
707 * protocol, whenever one side or the other says they're finished
708 * using the connection. There might be a better way to intercept
709 * the socket from the core...
711 void
712 peer_oft_sendcb_ack(PurpleXfer *xfer, const guchar *buffer, size_t size)
714 PeerConnection *conn;
716 conn = oscar_xfer_get_peer_connection(OSCAR_XFER(xfer));
719 * If we're done sending, intercept the socket from the core ft code
720 * and wait for the other guy to send the "done" OFT packet.
722 if (purple_xfer_get_bytes_remaining(xfer) <= 0)
724 purple_input_remove(purple_xfer_get_watcher(xfer));
725 conn->fd = purple_xfer_get_fd(xfer);
726 purple_xfer_set_fd(xfer, -1);
727 conn->watcher_incoming = purple_input_add(conn->fd,
728 PURPLE_INPUT_READ, peer_connection_recv_cb, conn);
732 /*******************************************************************/
733 /* End PurpleXfer callbacks for use when sending a file */
734 /*******************************************************************/
736 /*******************************************************************/
737 /* Begin PurpleXfer callbacks for use when sending and receiving */
738 /*******************************************************************/
740 void
741 peer_oft_cb_generic_cancel(PurpleXfer *xfer)
743 PeerConnection *conn;
745 conn = oscar_xfer_get_peer_connection(OSCAR_XFER(xfer));
747 if (conn == NULL)
748 return;
750 peer_connection_destroy(conn, OSCAR_DISCONNECT_LOCAL_CLOSED, NULL);
753 /*******************************************************************/
754 /* End PurpleXfer callbacks for use when sending and receiving */
755 /*******************************************************************/
757 #ifdef TODOFT
759 * This little area in oscar.c is the nexus of file transfer code,
760 * so I wrote a little explanation of what happens. I am such a
761 * ninja.
763 * The series of events for a file send is:
764 * -Create xfer and call purple_xfer_request (this happens in oscar_ask_sendfile)
765 * -User chooses a file and oscar_xfer_init is called. It establishes a
766 * listening socket, then asks the remote user to connect to us (and
767 * gives them the file name, port, IP, etc.)
768 * -They connect to us and we send them an PEER_TYPE_PROMPT (this happens
769 * in peer_oft_recv_frame_established)
770 * -They send us an PEER_TYPE_ACK and then we start sending data
771 * -When we finish, they send us an PEER_TYPE_DONE and they close the
772 * connection.
773 * -We get drunk because file transfer kicks ass.
775 * The series of events for a file receive is:
776 * -Create xfer and call purple_xfer request (this happens in incomingim_chan2)
777 * -Purple user selects file to name and location to save file to and
778 * oscar_xfer_init is called
779 * -It connects to the remote user using the IP they gave us earlier
780 * -After connecting, they send us an PEER_TYPE_PROMPT. In reply, we send
781 * them an PEER_TYPE_ACK.
782 * -They begin to send us lots of raw data.
783 * -When they finish sending data we send an PEER_TYPE_DONE and then close
784 * the connection.
786 * Update August 2005:
787 * The series of events for transfers has been seriously complicated by the addition
788 * of transfer redirects and proxied connections. I could throw a whole lot of words
789 * at trying to explain things here, but it probably wouldn't do much good. To get
790 * a better idea of what happens, take a look at the diagrams and documentation
791 * from my Summer of Code project. -- Jonathan Clark
795 * Convert the directory separator from / (0x2f) to ^A (0x01)
797 * @param name The filename to convert.
799 static void
800 peer_oft_dirconvert_tostupid(char *name)
802 while (name[0]) {
803 if (name[0] == 0x01)
804 name[0] = G_DIR_SEPARATOR;
805 name++;
810 * Convert the directory separator from ^A (0x01) to / (0x2f)
812 * @param name The filename to convert.
814 static void
815 peer_oft_dirconvert_fromstupid(char *name)
817 while (name[0]) {
818 if (name[0] == G_DIR_SEPARATOR)
819 name[0] = 0x01;
820 name++;
823 #endif