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
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"
46 #include "sipe-utils.h"
50 struct sipe_file_transfer_lync
{
51 struct sipe_file_transfer
public;
59 guint bytes_left_in_chunk
;
63 guint buffer_read_pos
;
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
,
74 #define SIPE_FILE_TRANSFER ((struct sipe_file_transfer *) ft_private)
75 #define SIPE_FILE_TRANSFER_PRIVATE ((struct sipe_file_transfer_lync *) ft)
78 SIPE_XDATA_DATA_CHUNK
= 0x00,
79 SIPE_XDATA_START_OF_STREAM
= 0x01,
80 SIPE_XDATA_END_OF_STREAM
= 0x02
83 #define XDATA_HEADER_SIZE sizeof (guint8) + sizeof (guint16)
86 sipe_file_transfer_lync_free(struct sipe_file_transfer_lync
*ft_private
)
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
);
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",
115 sipe_media_get_sip_dialog(ft_private
->call
),
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
: "",
134 ft_private
, callback
);
138 mime_mixed_cb(gpointer user_data
, const GSList
*fields
, const gchar
*body
,
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
);
149 const sipe_xml
*node
;
151 ft_private
->request_id
= sipe_xml_int_attribute(xml
,
153 ft_private
->request_id
);
155 node
= sipe_xml_child(xml
, "publishFile/fileInfo/name");
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");
163 g_free(ft_private
->id
);
164 ft_private
->id
= sipe_xml_data(node
);
167 node
= sipe_xml_child(xml
, "publishFile/fileInfo/size");
169 gchar
*size_str
= sipe_xml_data(node
);
171 ft_private
->file_size
= atoi(size_str
);
178 } else if (g_str_has_prefix(ctype
, "application/sdp")) {
179 g_free(ft_private
->sdp
);
180 ft_private
->sdp
= g_strndup(body
, length
);
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\">"
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
,
207 ft_private
->file_name
),
212 create_pipe(int pipefd
[2])
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. */
219 if (pipe(pipefd
) != 0) {
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
);
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
;
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
);
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
);
254 xdata_end_of_stream_cb(SIPE_UNUSED_PARAMETER
struct sipe_media_stream
*stream
,
255 guint8
*buffer
, gsize len
)
258 SIPE_DEBUG_INFO("Received end of stream for requestId : %s", buffer
);
262 xdata_got_header_cb(struct sipe_media_stream
*stream
,
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 */
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
);
278 case SIPE_XDATA_DATA_CHUNK
:
279 SIPE_DEBUG_INFO("Received new data chunk of size %d",
281 ft_private
->bytes_left_in_chunk
= size
;
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
);
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. */
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
);
310 ft_private
->buffer_read_pos
+= written
;
311 } else if (written
< 0 && errno
!= EAGAIN
) {
312 SIPE_DEBUG_ERROR_NOFORMAT("Error while writing into "
314 sipe_backend_ft_cancel_local(SIPE_FILE_TRANSFER
);
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
,
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
);
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
,
339 xdata_got_header_cb
);
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
;
352 sipe_backend_media_accept(call
->backend_private
, TRUE
);
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
);
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>"
392 "</fileTransferProgress>"
395 send_ms_filetransfer_msg(g_strdup_printf(FILETRANSFER_PROGRESS
,
397 ft_private
->request_id
,
398 ft_private
->file_size
- 1),
403 ft_lync_end(struct sipe_file_transfer
*ft
)
405 send_transfer_progress(SIPE_FILE_TRANSFER_PRIVATE
);
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
);
421 sipe_backend_ft_cancel_remote(&ft_private
->public);
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\"/>"
431 "<transferId>%d</transferId>"
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
,
446 ft_private
->file_name
),
450 stream
= sipe_core_media_get_stream_by_id(ft_private
->call
, "data");
452 stream
->read_cb
= NULL
;
455 sipe_backend_media_hangup(ft_private
->call
->backend_private
, FALSE
);
459 process_incoming_invite_ft_lync(struct sipe_core_private
*sipe_private
,
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
);
476 /* Replace multipart message body with the selected SDP part and
477 * initialize media session as if invited to a media call. */
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
);
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");
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
);
512 sip_transport_response(sipe_private
, msg
, 500, "Server Internal Error", NULL
);
513 sipe_file_transfer_lync_free(ft_private
);
519 process_response_incoming(struct sipe_file_transfer_lync
*ft_private
,
523 guint request_id
= sipe_xml_int_attribute(xml
, "requestId", 0);
525 if (request_id
!= ft_private
->request_id
) {
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
);
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
);
550 send_file_chunk(SIPE_UNUSED_PARAMETER GIOChannel
*source
,
551 SIPE_UNUSED_PARAMETER GIOCondition condition
,
554 struct sipe_file_transfer_lync
*ft_private
= data
;
555 struct sipe_media_call
*call
= ft_private
->call
;
556 struct sipe_media_stream
*stream
;
559 stream
= sipe_core_media_get_stream_by_id(call
, "data");
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 */
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
;
599 stream
= sipe_core_media_get_stream_by_id(ft_private
->call
, "data");
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
);
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
,
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
);
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
),
642 start_writing(ft_private
);
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");
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
),
662 sipe_backend_media_hangup(ft_private
->call
->backend_private
, TRUE
);
669 process_incoming_info_ft_lync(struct sipe_core_private
*sipe_private
,
672 struct sipe_media_call
*call
;
673 struct sipe_file_transfer_lync
*ft_private
;
676 call
= g_hash_table_lookup(sipe_private
->media_calls
,
677 sipmsg_find_header(msg
, "Call-ID"));
682 ft_private
= ft_private_from_call(call
);
687 xml
= sipe_xml_parse(msg
->body
, msg
->bodylen
);
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
);
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
);
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"
718 "<request xmlns=\"http://schemas.microsoft.com/rtc/2009/05/filetransfer\" "
722 "<id>{6244F934-2EB1-443F-8E2C-48BA64AF463D}</id>"
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
);
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);
763 sipe_backend_notify_error(SIPE_CORE_PUBLIC
,
765 _("Error creating data stream"));
767 sipe_backend_media_hangup(call
->backend_private
, FALSE
);
768 sipe_backend_ft_cancel_local(ft
);
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
;