appshare: move default to core
[siplcs/bazlsi01.git] / src / core / sipe-ft-lync.c
blobf4207f9b11671632887117c24328d20b07045470
1 /**
2 * @file sipe-ft-lync.c
4 * pidgin-sipe
6 * Copyright (C) 2014-2016 SIPE Project <http://sipe.sourceforge.net/>
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23 #ifdef HAVE_CONFIG_H
24 #include "config.h"
25 #endif
27 #include <glib.h>
29 #include <errno.h>
30 #include <fcntl.h>
31 #include <stdlib.h>
32 #include <string.h>
33 #ifdef HAVE_UNISTD_H
34 #include <unistd.h>
35 #endif
37 #include "sip-transport.h"
38 #include "sipe-backend.h"
39 #include "sipe-common.h"
40 #include "sipe-core.h"
41 #include "sipe-core-private.h"
42 #include "sipe-ft-lync.h"
43 #include "sipe-media.h"
44 #include "sipe-mime.h"
45 #include "sipe-nls.h"
46 #include "sipe-utils.h"
47 #include "sipe-xml.h"
48 #include "sipmsg.h"
50 struct sipe_file_transfer_lync {
51 struct sipe_file_transfer public;
53 gchar *sdp;
54 gchar *file_name;
55 gchar *id;
56 gsize file_size;
57 guint request_id;
59 guint bytes_left_in_chunk;
61 guint8 buffer[2048];
62 guint buffer_len;
63 guint buffer_read_pos;
65 int backend_pipe[2];
66 int backend_pipe_write_source_id;
68 struct sipe_core_private *sipe_private;
69 struct sipe_media_call *call;
71 void (*call_reject_parent_cb)(struct sipe_media_call *call,
72 gboolean local);
74 #define SIPE_FILE_TRANSFER ((struct sipe_file_transfer *) ft_private)
75 #define SIPE_FILE_TRANSFER_PRIVATE ((struct sipe_file_transfer_lync *) ft)
77 typedef enum {
78 SIPE_XDATA_DATA_CHUNK = 0x00,
79 SIPE_XDATA_START_OF_STREAM = 0x01,
80 SIPE_XDATA_END_OF_STREAM = 0x02
81 } SipeXDataMessages;
83 #define XDATA_HEADER_SIZE sizeof (guint8) + sizeof (guint16)
85 static void
86 sipe_file_transfer_lync_free(struct sipe_file_transfer_lync *ft_private)
88 int our_pipe_end;
90 our_pipe_end = sipe_backend_ft_is_incoming(SIPE_FILE_TRANSFER) ? 1 : 0;
92 if (ft_private->backend_pipe[our_pipe_end] != 0) {
93 // Backend is responsible for closing the pipe's other end.
94 close(ft_private->backend_pipe[our_pipe_end]);
97 g_free(ft_private->file_name);
98 g_free(ft_private->sdp);
99 g_free(ft_private->id);
101 if (ft_private->backend_pipe_write_source_id) {
102 g_source_remove(ft_private->backend_pipe_write_source_id);
105 g_free(ft_private);
108 static void
109 send_ms_filetransfer_msg(char *body, struct sipe_file_transfer_lync *ft_private,
110 TransCallback callback)
112 sip_transport_info(sipe_media_get_sipe_core_private(ft_private->call),
113 "Content-Type: application/ms-filetransfer+xml\r\n",
114 body,
115 sipe_media_get_sip_dialog(ft_private->call),
116 callback);
118 g_free(body);
121 static void
122 send_ms_filetransfer_response(struct sipe_file_transfer_lync *ft_private,
123 const gchar *code, const gchar *reason,
124 TransCallback callback)
126 static const gchar *RESPONSE_STR =
127 "<response xmlns=\"http://schemas.microsoft.com/rtc/2009/05/filetransfer\" requestId=\"%d\" code=\"%s\" %s%s%s/>";
129 send_ms_filetransfer_msg(g_strdup_printf(RESPONSE_STR,
130 ft_private->request_id, code,
131 reason ? "reason=\"" : "",
132 reason ? reason : "",
133 reason ? "\"" : ""),
134 ft_private, callback);
137 static void
138 mime_mixed_cb(gpointer user_data, const GSList *fields, const gchar *body,
139 gsize length)
141 struct sipe_file_transfer_lync *ft_private = user_data;
142 const gchar *ctype = sipe_utils_nameval_find(fields, "Content-Type");
144 /* Lync 2010 file transfer */
145 if (g_str_has_prefix(ctype, "application/ms-filetransfer+xml")) {
146 sipe_xml *xml = sipe_xml_parse(body, length);
148 if (xml) {
149 const sipe_xml *node;
151 ft_private->request_id = sipe_xml_int_attribute(xml,
152 "requestId",
153 ft_private->request_id);
155 node = sipe_xml_child(xml, "publishFile/fileInfo/name");
156 if (node) {
157 g_free(ft_private->file_name);
158 ft_private->file_name = sipe_xml_data(node);
161 node = sipe_xml_child(xml, "publishFile/fileInfo/id");
162 if (node) {
163 g_free(ft_private->id);
164 ft_private->id = sipe_xml_data(node);
167 node = sipe_xml_child(xml, "publishFile/fileInfo/size");
168 if (node) {
169 gchar *size_str = sipe_xml_data(node);
170 if (size_str) {
171 ft_private->file_size = atoi(size_str);
172 g_free(size_str);
176 sipe_xml_free(xml);
178 } else if (g_str_has_prefix(ctype, "application/sdp")) {
179 g_free(ft_private->sdp);
180 ft_private->sdp = g_strndup(body, length);
184 static void
185 candidate_pairs_established_cb(struct sipe_media_stream *stream)
187 struct sipe_file_transfer_lync *ft_private;
188 static const gchar *DOWNLOAD_FILE_REQUEST =
189 "<request xmlns=\"http://schemas.microsoft.com/rtc/2009/05/filetransfer\" requestId=\"%d\">"
190 "<downloadFile>"
191 "<fileInfo>"
192 "<id>%s</id>"
193 "<name>%s</name>"
194 "</fileInfo>"
195 "</downloadFile>"
196 "</request>";
198 g_return_if_fail(sipe_strequal(stream->id, "data"));
200 ft_private = sipe_media_stream_get_data(stream);
202 send_ms_filetransfer_response(ft_private, "success", NULL, NULL);
204 send_ms_filetransfer_msg(g_strdup_printf(DOWNLOAD_FILE_REQUEST,
205 ++ft_private->request_id,
206 ft_private->id,
207 ft_private->file_name),
208 ft_private, NULL);
211 static gboolean
212 create_pipe(int pipefd[2])
214 #ifdef _WIN32
215 #error "Pipes not implemented for Windows"
216 /* Those interested in porting the code may use Pidgin's wpurple_input_pipe() in
217 * win32dep.c as an inspiration. */
218 #else
219 if (pipe(pipefd) != 0) {
220 return FALSE;
223 /* @TODO: ignoring potential error return - how to handle? */
224 (void) fcntl(pipefd[0], F_SETFL, fcntl(pipefd[0], F_GETFL) | O_NONBLOCK);
225 (void) fcntl(pipefd[1], F_SETFL, fcntl(pipefd[1], F_GETFL) | O_NONBLOCK);
227 return TRUE;
228 #endif
231 static void
232 xdata_start_of_stream_cb(struct sipe_media_stream *stream,
233 guint8 *buffer, gsize len)
235 struct sipe_file_transfer_lync *ft_private =
236 sipe_media_stream_get_data(stream);
237 struct sipe_backend_fd *fd;
239 buffer[len] = 0;
240 SIPE_DEBUG_INFO("Received new stream for requestId : %s", buffer);
242 if (!create_pipe(ft_private->backend_pipe)) {
243 SIPE_DEBUG_ERROR_NOFORMAT("Couldn't create backend pipe");
244 sipe_backend_ft_cancel_local(SIPE_FILE_TRANSFER);
245 return;
248 fd = sipe_backend_fd_from_int(ft_private->backend_pipe[0]);
249 sipe_backend_ft_start(SIPE_FILE_TRANSFER, fd, NULL, 0);
250 sipe_backend_fd_free(fd);
253 static void
254 xdata_end_of_stream_cb(SIPE_UNUSED_PARAMETER struct sipe_media_stream *stream,
255 guint8 *buffer, gsize len)
257 buffer[len] = 0;
258 SIPE_DEBUG_INFO("Received end of stream for requestId : %s", buffer);
261 static void
262 xdata_got_header_cb(struct sipe_media_stream *stream,
263 guint8 *buffer,
264 SIPE_UNUSED_PARAMETER gsize len)
266 struct sipe_file_transfer_lync *ft_private =
267 sipe_media_stream_get_data(stream);
269 guint8 type = buffer[0];
270 guint16 size = (buffer[1] << 8) + buffer[2]; /* stored as big-endian */
272 switch (type) {
273 case SIPE_XDATA_START_OF_STREAM:
274 sipe_media_stream_read_async(stream,
275 ft_private->buffer, size,
276 xdata_start_of_stream_cb);
277 break;
278 case SIPE_XDATA_DATA_CHUNK:
279 SIPE_DEBUG_INFO("Received new data chunk of size %d",
280 size);
281 ft_private->bytes_left_in_chunk = size;
282 break;
283 /* We'll read the data when read_cb is called again. */
284 case SIPE_XDATA_END_OF_STREAM:
285 sipe_media_stream_read_async(stream,
286 ft_private->buffer, size,
287 xdata_end_of_stream_cb);
288 break;
292 static void
293 read_cb(struct sipe_media_stream *stream)
295 struct sipe_file_transfer_lync *ft_private =
296 sipe_media_stream_get_data(stream);
298 if (ft_private->buffer_read_pos < ft_private->buffer_len) {
299 /* Have data in buffer, write them to the backend. */
301 gpointer buffer;
302 size_t len;
303 ssize_t written;
305 buffer = ft_private->buffer + ft_private->buffer_read_pos;
306 len = ft_private->buffer_len - ft_private->buffer_read_pos;
307 written = write(ft_private->backend_pipe[1], buffer, len);
309 if (written > 0) {
310 ft_private->buffer_read_pos += written;
311 } else if (written < 0 && errno != EAGAIN) {
312 SIPE_DEBUG_ERROR_NOFORMAT("Error while writing into "
313 "backend pipe");
314 sipe_backend_ft_cancel_local(SIPE_FILE_TRANSFER);
315 return;
317 } else if (ft_private->bytes_left_in_chunk != 0) {
318 /* Have data from the sender, replenish our buffer with it. */
320 ft_private->buffer_len = MIN(ft_private->bytes_left_in_chunk,
321 sizeof (ft_private->buffer));
323 ft_private->buffer_len =
324 sipe_backend_media_stream_read(stream,
325 ft_private->buffer,
326 ft_private->buffer_len);
328 ft_private->bytes_left_in_chunk -= ft_private->buffer_len;
329 ft_private->buffer_read_pos = 0;
331 SIPE_DEBUG_INFO("Read %d bytes. %d left in this chunk.",
332 ft_private->buffer_len, ft_private->bytes_left_in_chunk);
333 } else {
334 /* No data available. This is either stream start, beginning of
335 * chunk, or stream end. */
337 sipe_media_stream_read_async(stream, ft_private->buffer,
338 XDATA_HEADER_SIZE,
339 xdata_got_header_cb);
343 static void
344 ft_lync_incoming_init(struct sipe_file_transfer *ft,
345 SIPE_UNUSED_PARAMETER const gchar *filename,
346 SIPE_UNUSED_PARAMETER gsize size,
347 SIPE_UNUSED_PARAMETER const gchar *who)
349 struct sipe_media_call *call = SIPE_FILE_TRANSFER_PRIVATE->call;
351 if (call) {
352 sipe_backend_media_accept(call->backend_private, TRUE);
356 static void
357 ft_lync_request_denied(struct sipe_file_transfer *ft)
359 struct sipe_file_transfer_lync *ft_private = SIPE_FILE_TRANSFER_PRIVATE;
360 struct sipe_media_call *call;
362 g_return_if_fail(ft_private);
364 call = ft_private->call;
366 if (call && call->backend_private) {
367 sipe_backend_media_reject(call->backend_private, TRUE);
371 static struct sipe_file_transfer_lync *
372 ft_private_from_call(struct sipe_media_call *call)
374 struct sipe_media_stream *stream =
375 sipe_core_media_get_stream_by_id(call, "data");
376 g_return_val_if_fail(stream, NULL);
378 return sipe_media_stream_get_data(stream);
381 static void
382 send_transfer_progress(struct sipe_file_transfer_lync *ft_private)
384 static const gchar *FILETRANSFER_PROGRESS =
385 "<notify xmlns=\"http://schemas.microsoft.com/rtc/2009/05/filetransfer\" notifyId=\"%d\">"
386 "<fileTransferProgress>"
387 "<transferId>%d</transferId>"
388 "<bytesReceived>"
389 "<from>0</from>"
390 "<to>%d</to>"
391 "</bytesReceived>"
392 "</fileTransferProgress>"
393 "</notify>";
395 send_ms_filetransfer_msg(g_strdup_printf(FILETRANSFER_PROGRESS,
396 rand(),
397 ft_private->request_id,
398 ft_private->file_size - 1),
399 ft_private, NULL);
402 static gboolean
403 ft_lync_end(struct sipe_file_transfer *ft)
405 send_transfer_progress(SIPE_FILE_TRANSFER_PRIVATE);
407 return TRUE;
410 static void
411 call_reject_cb(struct sipe_media_call *call, gboolean local)
413 struct sipe_file_transfer_lync *ft_private = ft_private_from_call(call);
414 g_return_if_fail(ft_private);
416 if (ft_private->call_reject_parent_cb) {
417 ft_private->call_reject_parent_cb(call, local);
420 if (!local) {
421 sipe_backend_ft_cancel_remote(&ft_private->public);
425 static void
426 ft_lync_incoming_cancelled(struct sipe_file_transfer *ft)
428 static const gchar *FILETRANSFER_CANCEL_REQUEST =
429 "<request xmlns=\"http://schemas.microsoft.com/rtc/2009/05/filetransfer\" requestId=\"%d\"/>"
430 "<cancelTransfer>"
431 "<transferId>%d</transferId>"
432 "<fileInfo>"
433 "<id>%s</id>"
434 "<name>%s</name>"
435 "</fileInfo>"
436 "</cancelTransfer>"
437 "</request>";
439 struct sipe_file_transfer_lync *ft_private = SIPE_FILE_TRANSFER_PRIVATE;
440 struct sipe_media_stream *stream;
442 send_ms_filetransfer_msg(g_strdup_printf(FILETRANSFER_CANCEL_REQUEST,
443 ft_private->request_id + 1,
444 ft_private->request_id,
445 ft_private->id,
446 ft_private->file_name),
447 ft_private,
448 NULL);
450 stream = sipe_core_media_get_stream_by_id(ft_private->call, "data");
451 if (stream) {
452 stream->read_cb = NULL;
455 sipe_backend_media_hangup(ft_private->call->backend_private, FALSE);
458 void
459 process_incoming_invite_ft_lync(struct sipe_core_private *sipe_private,
460 struct sipmsg *msg)
462 struct sipe_file_transfer_lync *ft_private;
463 struct sipe_media_call *call;
464 struct sipe_media_stream *stream;
466 ft_private = g_new0(struct sipe_file_transfer_lync, 1);
467 sipe_mime_parts_foreach(sipmsg_find_header(msg, "Content-Type"),
468 msg->body, mime_mixed_cb, ft_private);
470 if (!ft_private->file_name || !ft_private->file_size || !ft_private->sdp) {
471 sip_transport_response(sipe_private, msg, 488, "Not Acceptable Here", NULL);
472 sipe_file_transfer_lync_free(ft_private);
473 return;
476 /* Replace multipart message body with the selected SDP part and
477 * initialize media session as if invited to a media call. */
478 g_free(msg->body);
479 msg->body = ft_private->sdp;
480 msg->bodylen = strlen(msg->body);
481 ft_private->sdp = NULL;
483 ft_private->call = process_incoming_invite_call(sipe_private, msg);
484 if (!ft_private->call) {
485 sip_transport_response(sipe_private, msg, 500, "Server Internal Error", NULL);
486 sipe_file_transfer_lync_free(ft_private);
487 return;
490 call = ft_private->call;
492 ft_private->public.ft_init = ft_lync_incoming_init;
493 ft_private->public.ft_request_denied = ft_lync_request_denied;
494 ft_private->public.ft_cancelled = ft_lync_incoming_cancelled;
495 ft_private->public.ft_end = ft_lync_end;
497 ft_private->call_reject_parent_cb = call->call_reject_cb;
498 call->call_reject_cb = call_reject_cb;
500 stream = sipe_core_media_get_stream_by_id(call, "data");
501 if (stream) {
502 stream->candidate_pairs_established_cb = candidate_pairs_established_cb;
503 stream->read_cb = read_cb;
504 sipe_media_stream_add_extra_attribute(stream, "recvonly", NULL);
505 sipe_media_stream_set_data(stream, ft_private,
506 (GDestroyNotify)sipe_file_transfer_lync_free);
508 sipe_backend_ft_incoming(SIPE_CORE_PUBLIC, SIPE_FILE_TRANSFER,
509 call->with, ft_private->file_name,
510 ft_private->file_size);
511 } else {
512 sip_transport_response(sipe_private, msg, 500, "Server Internal Error", NULL);
513 sipe_file_transfer_lync_free(ft_private);
514 return;
518 static void
519 process_response_incoming(struct sipe_file_transfer_lync *ft_private,
520 sipe_xml *xml)
522 const gchar *attr;
523 guint request_id = sipe_xml_int_attribute(xml, "requestId", 0);
525 if (request_id != ft_private->request_id) {
526 return;
529 attr = sipe_xml_attribute(xml, "code");
530 if (sipe_strequal(attr, "failure")) {
531 const gchar *reason = sipe_xml_attribute(xml, "reason");
532 if (sipe_strequal(reason, "requestCancelled")) {
533 sipe_backend_ft_cancel_remote(SIPE_FILE_TRANSFER);
538 static void
539 write_chunk(struct sipe_media_stream *stream,
540 guint8 type, guint16 len, const gchar *buffer)
542 guint16 len_be = GUINT16_TO_BE(len);
544 sipe_media_stream_write(stream, &type, sizeof (guint8));
545 sipe_media_stream_write(stream, (guint8 *)&len_be, sizeof (guint16));
546 sipe_media_stream_write(stream, (guint8 *)buffer, len);
549 static gboolean
550 send_file_chunk(SIPE_UNUSED_PARAMETER GIOChannel *source,
551 SIPE_UNUSED_PARAMETER GIOCondition condition,
552 gpointer data)
554 struct sipe_file_transfer_lync *ft_private = data;
555 struct sipe_media_call *call = ft_private->call;
556 struct sipe_media_stream *stream;
557 gssize bytes_read;
559 stream = sipe_core_media_get_stream_by_id(call, "data");
560 if (!stream) {
561 SIPE_DEBUG_ERROR_NOFORMAT("Couldn't find data stream");
562 sipe_backend_ft_cancel_local(SIPE_FILE_TRANSFER);
563 ft_private->backend_pipe_write_source_id = 0;
564 return FALSE; /* G_SOURCE_REMOVE */
567 if (!sipe_media_stream_is_writable(stream)) {
568 return TRUE; /* G_SOURCE_CONTINUE */
571 bytes_read = read(ft_private->backend_pipe[0],
572 ft_private->buffer, sizeof (ft_private->buffer));
573 if (bytes_read > 0) {
574 write_chunk(stream, SIPE_XDATA_DATA_CHUNK,
575 bytes_read, (const gchar *)ft_private->buffer);
576 } else if (bytes_read == 0) {
577 /* EOF, write end of stream */
578 gchar *request_id_str;
580 request_id_str = g_strdup_printf("%u", ft_private->request_id);
581 write_chunk(stream, SIPE_XDATA_END_OF_STREAM,
582 strlen(request_id_str), request_id_str);
583 g_free(request_id_str);
585 return FALSE; /* G_SOURCE_REMOVE */
588 return TRUE; /* G_SOURCE_CONTINUE */
591 static void
592 start_writing(struct sipe_file_transfer_lync *ft_private)
594 struct sipe_media_stream *stream;
595 gchar *request_id_str;
596 struct sipe_backend_fd *fd;
597 GIOChannel *channel;
599 stream = sipe_core_media_get_stream_by_id(ft_private->call, "data");
600 if (!stream) {
601 return;
604 if (!create_pipe(ft_private->backend_pipe)) {
605 SIPE_DEBUG_ERROR_NOFORMAT("Couldn't create backend pipe");
606 sipe_backend_ft_cancel_local(SIPE_FILE_TRANSFER);
607 return;
610 request_id_str = g_strdup_printf("%u", ft_private->request_id);
611 write_chunk(stream, SIPE_XDATA_START_OF_STREAM,
612 strlen(request_id_str), request_id_str);
613 g_free(request_id_str);
615 channel = g_io_channel_unix_new(ft_private->backend_pipe[0]);
616 ft_private->backend_pipe_write_source_id = g_io_add_watch(channel,
617 G_IO_IN | G_IO_HUP,
618 send_file_chunk,
619 ft_private);
620 g_io_channel_unref(channel);
622 fd = sipe_backend_fd_from_int(ft_private->backend_pipe[1]);
623 sipe_backend_ft_start(SIPE_FILE_TRANSFER, fd, NULL, 0);
624 sipe_backend_fd_free(fd);
627 static void
628 process_request(struct sipe_file_transfer_lync *ft_private, sipe_xml *xml)
630 static const gchar *DOWNLOAD_PENDING_RESPONSE =
631 "<response xmlns=\"http://schemas.microsoft.com/rtc/2009/05/filetransfer\" "
632 "requestId=\"%u\" code=\"pending\"/>";
634 if (sipe_xml_child(xml, "downloadFile")) {
635 ft_private->request_id =
636 atoi(sipe_xml_attribute(xml, "requestId"));
638 send_ms_filetransfer_msg(g_strdup_printf(DOWNLOAD_PENDING_RESPONSE,
639 ft_private->request_id),
640 ft_private, NULL);
642 start_writing(ft_private);
646 static void
647 process_notify(struct sipe_file_transfer_lync *ft_private, sipe_xml *xml)
649 static const gchar *DOWNLOAD_SUCCESS_RESPONSE =
650 "<response xmlns=\"http://schemas.microsoft.com/rtc/2009/05/filetransfer\" "
651 "requestId=\"%u\" code=\"success\"/>";
653 const sipe_xml *progress_node = sipe_xml_child(xml, "fileTransferProgress");
655 if (progress_node) {
656 gchar *to_str = sipe_xml_data(sipe_xml_child(progress_node, "bytesReceived/to"));
658 if (atoi(to_str) == (int)(ft_private->file_size - 1)) {
659 send_ms_filetransfer_msg(g_strdup_printf(DOWNLOAD_SUCCESS_RESPONSE,
660 ft_private->request_id),
661 ft_private, NULL);
662 sipe_backend_media_hangup(ft_private->call->backend_private, TRUE);
664 g_free(to_str);
668 void
669 process_incoming_info_ft_lync(struct sipe_core_private *sipe_private,
670 struct sipmsg *msg)
672 struct sipe_media_call *call;
673 struct sipe_file_transfer_lync *ft_private;
674 sipe_xml *xml;
676 call = g_hash_table_lookup(sipe_private->media_calls,
677 sipmsg_find_header(msg, "Call-ID"));
678 if (!call) {
679 return;
682 ft_private = ft_private_from_call(call);
683 if (!ft_private) {
684 return;
687 xml = sipe_xml_parse(msg->body, msg->bodylen);
688 if (!xml) {
689 return;
692 sip_transport_response(sipe_private, msg, 200, "OK", NULL);
694 if (sipe_backend_ft_is_incoming(SIPE_FILE_TRANSFER)) {
695 if (sipe_strequal(sipe_xml_name(xml), "response")) {
696 process_response_incoming(ft_private, xml);
698 } else {
699 if (sipe_strequal(sipe_xml_name(xml), "request")) {
700 process_request(ft_private, xml);
701 } else if (sipe_strequal(sipe_xml_name(xml), "notify")) {
702 process_notify(ft_private, xml);
706 sipe_xml_free(xml);
709 static void
710 append_publish_file_invite(struct sipe_media_call *call,
711 struct sipe_file_transfer_lync *ft_private)
713 static const gchar *PUBLISH_FILE_REQUEST =
714 "Content-Type: application/ms-filetransfer+xml\r\n"
715 "Content-Transfer-Encoding: 7bit\r\n"
716 "Content-Disposition: render; handling=optional\r\n"
717 "\r\n"
718 "<request xmlns=\"http://schemas.microsoft.com/rtc/2009/05/filetransfer\" "
719 "requestId=\"%u\">"
720 "<publishFile>"
721 "<fileInfo>"
722 "<id>{6244F934-2EB1-443F-8E2C-48BA64AF463D}</id>"
723 "<name>%s</name>"
724 "<size>%u</size>"
725 "</fileInfo>"
726 "</publishFile>"
727 "</request>\r\n";
728 gchar *body;
730 ft_private->request_id =
731 ++ft_private->sipe_private->ms_filetransfer_request_id;
733 body = g_strdup_printf(PUBLISH_FILE_REQUEST, ft_private->request_id,
734 ft_private->file_name, ft_private->file_size);
736 sipe_media_add_extra_invite_section(call, "multipart/mixed", body);
739 static void
740 ft_lync_outgoing_init(struct sipe_file_transfer *ft, const gchar *filename,
741 gsize size, SIPE_UNUSED_PARAMETER const gchar *who)
743 struct sipe_core_private *sipe_private =
744 SIPE_FILE_TRANSFER_PRIVATE->sipe_private;
745 struct sipe_file_transfer_lync *ft_private = SIPE_FILE_TRANSFER_PRIVATE;
746 struct sipe_media_call *call;
747 struct sipe_media_stream *stream;
749 ft_private->file_name = g_strdup(filename);
750 ft_private->file_size = size;
752 call = sipe_media_call_new(sipe_private, who, NULL, SIPE_ICE_RFC_5245,
753 SIPE_MEDIA_CALL_NO_UI);
755 ft_private->call = call;
757 ft_private->call_reject_parent_cb = call->call_reject_cb;
758 call->call_reject_cb = call_reject_cb;
760 stream = sipe_media_stream_add(call, "data", SIPE_MEDIA_APPLICATION,
761 SIPE_ICE_RFC_5245, TRUE, 0);
762 if (!stream) {
763 sipe_backend_notify_error(SIPE_CORE_PUBLIC,
764 _("Error occurred"),
765 _("Error creating data stream"));
767 sipe_backend_media_hangup(call->backend_private, FALSE);
768 sipe_backend_ft_cancel_local(ft);
769 return;
772 sipe_media_stream_add_extra_attribute(stream, "sendonly", NULL);
773 sipe_media_stream_add_extra_attribute(stream, "mid", "1");
774 sipe_media_stream_set_data(stream, ft,
775 (GDestroyNotify)sipe_file_transfer_lync_free);
776 append_publish_file_invite(call, ft_private);
779 struct sipe_file_transfer *
780 sipe_file_transfer_lync_new_outgoing(struct sipe_core_private *sipe_private)
782 struct sipe_file_transfer_lync *ft_private;
784 ft_private = g_new0(struct sipe_file_transfer_lync, 1);
786 ft_private->sipe_private = sipe_private;
787 ft_private->public.ft_init = ft_lync_outgoing_init;
789 return SIPE_FILE_TRANSFER;
793 Local Variables:
794 mode: c
795 c-file-style: "bsd"
796 indent-tabs-mode: t
797 tab-width: 8
798 End: