8849
|
1 /*
|
|
2
|
|
3 silcgaim_ft.c
|
|
4
|
|
5 Author: Pekka Riikonen <priikone@silcnet.org>
|
|
6
|
|
7 Copyright (C) 2004 Pekka Riikonen
|
|
8
|
|
9 This program is free software; you can redistribute it and/or modify
|
|
10 it under the terms of the GNU General Public License as published by
|
|
11 the Free Software Foundation; version 2 of the License.
|
|
12
|
|
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.
|
|
17
|
|
18 */
|
|
19
|
|
20 #include "silcincludes.h"
|
|
21 #include "silcclient.h"
|
|
22 #include "silcgaim.h"
|
|
23
|
|
24 /****************************** File Transfer ********************************/
|
|
25
|
|
26 /* This implements the secure file transfer protocol (SFTP) using the SILC
|
|
27 SFTP library implementation. The API we use from the SILC Toolkit is the
|
|
28 SILC Client file transfer API, as it provides a simple file transfer we
|
|
29 need in this case. We could use the SILC SFTP API directly, but it would
|
|
30 be an overkill since we'd effectively re-implement the file transfer what
|
|
31 the SILC Client's file transfer API already provides.
|
|
32
|
|
33 From Gaim we do NOT use the FT API to do the transfer as it is very limiting.
|
|
34 In fact it does not suite to file transfers like SFTP at all. For example,
|
|
35 it assumes that read operations are synchronous what they are not in SFTP.
|
|
36 It also assumes that the file transfer socket is to be handled by the Gaim
|
|
37 eventloop, and this naturally is something we don't want to do in case of
|
|
38 SILC Toolkit. The FT API suites well to purely stream based file transfers
|
|
39 like HTTP GET and similar.
|
|
40
|
|
41 For this reason, we directly access the Gaim GKT FT API and hack the FT
|
|
42 API to merely provide the user interface experience and all the magic
|
|
43 is done in the SILC Toolkit. Ie. we update the statistics information in
|
|
44 the FT API for user interface, and that's it. A bit dirty but until the
|
|
45 FT API gets better this is the way to go. Good thing that FT API allowed
|
|
46 us to do this. */
|
|
47
|
|
48 typedef struct {
|
|
49 SilcGaim sg;
|
|
50 SilcClientEntry client_entry;
|
|
51 SilcUInt32 session_id;
|
|
52 char *hostname;
|
|
53 SilcUInt16 port;
|
|
54 GaimXfer *xfer;
|
|
55
|
|
56 SilcClientFileName completion;
|
|
57 void *completion_context;
|
|
58 } *SilcGaimXfer;
|
|
59
|
|
60 static void
|
|
61 silcgaim_ftp_monitor(SilcClient client,
|
|
62 SilcClientConnection conn,
|
|
63 SilcClientMonitorStatus status,
|
|
64 SilcClientFileError error,
|
|
65 SilcUInt64 offset,
|
|
66 SilcUInt64 filesize,
|
|
67 SilcClientEntry client_entry,
|
|
68 SilcUInt32 session_id,
|
|
69 const char *filepath,
|
|
70 void *context)
|
|
71 {
|
|
72 SilcGaimXfer xfer = context;
|
|
73 GaimConnection *gc = xfer->sg->gc;
|
|
74 char tmp[256];
|
|
75
|
|
76 if (status == SILC_CLIENT_FILE_MONITOR_CLOSED) {
|
|
77 gaim_xfer_unref(xfer->xfer);
|
|
78 silc_free(xfer);
|
|
79 return;
|
|
80 }
|
|
81
|
|
82 if (status == SILC_CLIENT_FILE_MONITOR_KEY_AGREEMENT)
|
|
83 return;
|
|
84
|
|
85 if (status == SILC_CLIENT_FILE_MONITOR_ERROR) {
|
|
86 if (error == SILC_CLIENT_FILE_NO_SUCH_FILE) {
|
|
87 g_snprintf(tmp, sizeof(tmp), "No such file %s",
|
|
88 filepath ? filepath : "[N/A]");
|
|
89 gaim_notify_error(gc, _("Secure File Transfer"),
|
|
90 _("Error during file transfer"), tmp);
|
|
91 } else if (error == SILC_CLIENT_FILE_PERMISSION_DENIED) {
|
|
92 gaim_notify_error(gc, _("Secure File Transfer"),
|
|
93 _("Error during file transfer"),
|
|
94 _("Permission denied"));
|
|
95 } else if (error == SILC_CLIENT_FILE_KEY_AGREEMENT_FAILED) {
|
|
96 gaim_notify_error(gc, _("Secure File Transfer"),
|
|
97 _("Error during file transfer"),
|
|
98 _("Key agreement failed"));
|
|
99 } else if (error == SILC_CLIENT_FILE_UNKNOWN_SESSION) {
|
|
100 gaim_notify_error(gc, _("Secure File Transfer"),
|
|
101 _("Error during file transfer"),
|
|
102 _("File transfer sessions does not exist"));
|
|
103 } else {
|
|
104 gaim_notify_error(gc, _("Secure File Transfer"),
|
|
105 _("Error during file transfer"), NULL);
|
|
106 }
|
|
107 silc_client_file_close(client, conn, session_id);
|
|
108 gaim_xfer_unref(xfer->xfer);
|
|
109 silc_free(xfer);
|
|
110 return;
|
|
111 }
|
|
112
|
|
113 /* Update file transfer UI */
|
|
114 if (!offset && filesize)
|
|
115 gaim_xfer_set_size(xfer->xfer, filesize);
|
|
116 if (offset && filesize) {
|
|
117 xfer->xfer->bytes_sent = offset;
|
|
118 xfer->xfer->bytes_remaining = filesize - offset;
|
|
119 }
|
|
120 gaim_xfer_update_progress(xfer->xfer);
|
|
121
|
|
122 if (status == SILC_CLIENT_FILE_MONITOR_SEND ||
|
|
123 status == SILC_CLIENT_FILE_MONITOR_RECEIVE) {
|
|
124 if (offset == filesize) {
|
|
125 /* Download finished */
|
|
126 gaim_xfer_set_completed(xfer->xfer, TRUE);
|
|
127 silc_client_file_close(client, conn, session_id);
|
|
128 }
|
|
129 }
|
|
130 }
|
|
131
|
|
132 static void
|
|
133 silcgaim_ftp_cancel(GaimXfer *x)
|
|
134 {
|
|
135 SilcGaimXfer xfer = x->data;
|
|
136 xfer->xfer->status = GAIM_XFER_STATUS_CANCEL_LOCAL;
|
|
137 gaim_xfer_update_progress(xfer->xfer);
|
|
138 silc_client_file_close(xfer->sg->client, xfer->sg->conn, xfer->session_id);
|
|
139 }
|
|
140
|
|
141 static void
|
|
142 silcgaim_ftp_ask_name_cancel(GaimXfer *x)
|
|
143 {
|
|
144 SilcGaimXfer xfer = x->data;
|
|
145
|
|
146 /* Cancel the transmission */
|
|
147 xfer->completion(NULL, xfer->completion_context);
|
|
148 silc_client_file_close(xfer->sg->client, xfer->sg->conn, xfer->session_id);
|
|
149 }
|
|
150
|
|
151 static void
|
|
152 silcgaim_ftp_ask_name_ok(GaimXfer *x)
|
|
153 {
|
|
154 SilcGaimXfer xfer = x->data;
|
|
155 const char *name;
|
|
156
|
|
157 name = gaim_xfer_get_local_filename(x);
|
|
158 unlink(name);
|
|
159 xfer->completion(name, xfer->completion_context);
|
|
160 }
|
|
161
|
|
162 static void
|
|
163 silcgaim_ftp_ask_name(SilcClient client,
|
|
164 SilcClientConnection conn,
|
|
165 SilcUInt32 session_id,
|
|
166 const char *remote_filename,
|
|
167 SilcClientFileName completion,
|
|
168 void *completion_context,
|
|
169 void *context)
|
|
170 {
|
|
171 SilcGaimXfer xfer = context;
|
|
172
|
|
173 xfer->completion = completion;
|
|
174 xfer->completion_context = completion_context;
|
|
175
|
|
176 gaim_xfer_set_init_fnc(xfer->xfer, silcgaim_ftp_ask_name_ok);
|
|
177 gaim_xfer_set_request_denied_fnc(xfer->xfer, silcgaim_ftp_ask_name_cancel);
|
|
178
|
|
179 /* Request to save the file */
|
|
180 gaim_xfer_set_filename(xfer->xfer, remote_filename);
|
|
181 gaim_xfer_request(xfer->xfer);
|
|
182 }
|
|
183
|
|
184 static void
|
|
185 silcgaim_ftp_request_result(GaimXfer *x)
|
|
186 {
|
|
187 SilcGaimXfer xfer = x->data;
|
|
188 SilcClientFileError status;
|
|
189 GaimConnection *gc = xfer->sg->gc;
|
|
190
|
|
191 if (gaim_xfer_get_status(x) != GAIM_XFER_STATUS_ACCEPTED)
|
|
192 return;
|
|
193
|
|
194 /* Start the file transfer */
|
|
195 status = silc_client_file_receive(xfer->sg->client, xfer->sg->conn,
|
|
196 silcgaim_ftp_monitor, xfer,
|
|
197 NULL, xfer->session_id,
|
|
198 silcgaim_ftp_ask_name, xfer);
|
|
199 switch (status) {
|
|
200 case SILC_CLIENT_FILE_OK:
|
|
201 return;
|
|
202 break;
|
|
203
|
|
204 case SILC_CLIENT_FILE_UNKNOWN_SESSION:
|
|
205 gaim_notify_error(gc, _("Secure File Transfer"),
|
|
206 _("No file transfer session active"), NULL);
|
|
207 break;
|
|
208
|
|
209 case SILC_CLIENT_FILE_ALREADY_STARTED:
|
|
210 gaim_notify_error(gc, _("Secure File Transfer"),
|
|
211 _("File transfer already started"), NULL);
|
|
212 break;
|
|
213
|
|
214 case SILC_CLIENT_FILE_KEY_AGREEMENT_FAILED:
|
|
215 gaim_notify_error(gc, _("Secure File Transfer"),
|
|
216 _("Could not perform key agreement for file transfer"),
|
|
217 NULL);
|
|
218 break;
|
|
219
|
|
220 default:
|
|
221 gaim_notify_error(gc, _("Secure File Transfer"),
|
|
222 _("Could not start the file transfer"), NULL);
|
|
223 break;
|
|
224 }
|
|
225
|
|
226 /* Error */
|
|
227 gaim_xfer_unref(xfer->xfer);
|
|
228 g_free(xfer->hostname);
|
|
229 silc_free(xfer);
|
|
230 }
|
|
231
|
|
232 static void
|
|
233 silcgaim_ftp_request_denied(GaimXfer *x)
|
|
234 {
|
|
235
|
|
236 }
|
|
237
|
|
238 void silcgaim_ftp_request(SilcClient client, SilcClientConnection conn,
|
|
239 SilcClientEntry client_entry, SilcUInt32 session_id,
|
|
240 const char *hostname, SilcUInt16 port)
|
|
241 {
|
|
242 GaimConnection *gc = client->application;
|
|
243 SilcGaim sg = gc->proto_data;
|
|
244 SilcGaimXfer xfer;
|
|
245
|
|
246 xfer = silc_calloc(1, sizeof(*xfer));
|
|
247 if (!xfer) {
|
|
248 silc_client_file_close(sg->client, sg->conn, xfer->session_id);
|
|
249 return;
|
|
250 }
|
|
251
|
|
252 xfer->sg = sg;
|
|
253 xfer->client_entry = client_entry;
|
|
254 xfer->session_id = session_id;
|
|
255 xfer->hostname = g_strdup(hostname);
|
|
256 xfer->port = port;
|
|
257 xfer->xfer = gaim_xfer_new(xfer->sg->account, GAIM_XFER_RECEIVE,
|
|
258 xfer->client_entry->nickname);
|
|
259 if (!xfer->xfer) {
|
|
260 silc_client_file_close(xfer->sg->client, xfer->sg->conn, xfer->session_id);
|
|
261 g_free(xfer->hostname);
|
|
262 silc_free(xfer);
|
|
263 return;
|
|
264 }
|
|
265 gaim_xfer_set_init_fnc(xfer->xfer, silcgaim_ftp_request_result);
|
|
266 gaim_xfer_set_request_denied_fnc(xfer->xfer, silcgaim_ftp_request_denied);
|
|
267 gaim_xfer_set_cancel_recv_fnc(xfer->xfer, silcgaim_ftp_cancel);
|
|
268 xfer->xfer->remote_ip = g_strdup(hostname);
|
|
269 xfer->xfer->remote_port = port;
|
|
270 xfer->xfer->data = xfer;
|
|
271
|
|
272 /* File transfer request */
|
|
273 gaim_xfer_request(xfer->xfer);
|
|
274 }
|
|
275
|
|
276 static void
|
|
277 silcgaim_ftp_send_cancel(GaimXfer *x)
|
|
278 {
|
|
279 SilcGaimXfer xfer = x->data;
|
|
280 silc_client_file_close(xfer->sg->client, xfer->sg->conn, xfer->session_id);
|
|
281 gaim_xfer_unref(xfer->xfer);
|
|
282 g_free(xfer->hostname);
|
|
283 silc_free(xfer);
|
|
284 }
|
|
285
|
|
286 static void
|
|
287 silcgaim_ftp_send(GaimXfer *x)
|
|
288 {
|
|
289 SilcGaimXfer xfer = x->data;
|
|
290 const char *name;
|
8910
|
291 char *local_ip = NULL, *remote_ip = NULL;
|
8849
|
292 gboolean local = TRUE;
|
|
293
|
|
294 name = gaim_xfer_get_local_filename(x);
|
|
295
|
|
296 /* Do the same magic what we do with key agreement (see silcgaim_buddy.c)
|
|
297 to see if we are behind NAT. */
|
|
298 if (silc_net_check_local_by_sock(xfer->sg->conn->sock->sock,
|
|
299 NULL, &local_ip)) {
|
|
300 /* Check if the IP is private */
|
|
301 if (silcgaim_ip_is_private(local_ip)) {
|
|
302 local = FALSE;
|
|
303 /* Local IP is private, resolve the remote server IP to see whether
|
|
304 we are talking to Internet or just on LAN. */
|
|
305 if (silc_net_check_host_by_sock(xfer->sg->conn->sock->sock, NULL,
|
|
306 &remote_ip))
|
|
307 if (silcgaim_ip_is_private(remote_ip))
|
|
308 /* We assume we are in LAN. Let's provide the connection point. */
|
|
309 local = TRUE;
|
|
310 }
|
|
311 }
|
|
312
|
|
313 if (local && !local_ip)
|
|
314 local_ip = silc_net_localip();
|
|
315
|
|
316 /* Send the file */
|
|
317 silc_client_file_send(xfer->sg->client, xfer->sg->conn,
|
|
318 silcgaim_ftp_monitor, xfer,
|
|
319 local_ip, 0, !local, xfer->client_entry,
|
|
320 name, &xfer->session_id);
|
|
321
|
|
322 silc_free(local_ip);
|
|
323 silc_free(remote_ip);
|
|
324 }
|
|
325
|
|
326 static void
|
|
327 silcgaim_ftp_send_file_resolved(SilcClient client,
|
|
328 SilcClientConnection conn,
|
|
329 SilcClientEntry *clients,
|
|
330 SilcUInt32 clients_count,
|
|
331 void *context)
|
|
332 {
|
|
333 GaimConnection *gc = client->application;
|
|
334 char tmp[256];
|
|
335
|
|
336 if (!clients) {
|
|
337 g_snprintf(tmp, sizeof(tmp),
|
|
338 _("User %s is not present in the network"),
|
|
339 (const char *)context);
|
|
340 gaim_notify_error(gc, _("Secure File Transfer"),
|
|
341 _("Cannot send file"), tmp);
|
|
342 silc_free(context);
|
|
343 return;
|
|
344 }
|
|
345
|
9466
|
346 silcgaim_ftp_send_file(client->application, (const char *)context, NULL);
|
8849
|
347 silc_free(context);
|
|
348 }
|
|
349
|
9466
|
350 void silcgaim_ftp_send_file(GaimConnection *gc, const char *name, const char *file)
|
8849
|
351 {
|
|
352 SilcGaim sg = gc->proto_data;
|
|
353 SilcClient client = sg->client;
|
|
354 SilcClientConnection conn = sg->conn;
|
|
355 SilcClientEntry *clients;
|
|
356 SilcUInt32 clients_count;
|
|
357 SilcGaimXfer xfer;
|
|
358 char *nickname;
|
|
359
|
|
360 if (!name)
|
|
361 return;
|
|
362
|
|
363 if (!silc_parse_userfqdn(name, &nickname, NULL))
|
|
364 return;
|
|
365
|
9353
|
366 #ifndef _WIN32
|
8849
|
367 silc_debug = TRUE;
|
|
368 silc_log_set_debug_string("*client*,*ftp*");
|
|
369 #endif
|
|
370
|
|
371 /* Find client entry */
|
|
372 clients = silc_client_get_clients_local(client, conn, nickname, name,
|
|
373 &clients_count);
|
|
374 if (!clients) {
|
|
375 silc_client_get_clients(client, conn, nickname, NULL,
|
|
376 silcgaim_ftp_send_file_resolved,
|
|
377 strdup(name));
|
|
378 silc_free(nickname);
|
|
379 return;
|
|
380 }
|
|
381
|
|
382 xfer = silc_calloc(1, sizeof(*xfer));
|
|
383 if (!xfer)
|
|
384 return;
|
|
385 xfer->sg = sg;
|
|
386 xfer->client_entry = clients[0];
|
|
387 xfer->xfer = gaim_xfer_new(xfer->sg->account, GAIM_XFER_SEND,
|
|
388 xfer->client_entry->nickname);
|
|
389 if (!xfer->xfer) {
|
|
390 silc_client_file_close(xfer->sg->client, xfer->sg->conn, xfer->session_id);
|
|
391 g_free(xfer->hostname);
|
|
392 silc_free(xfer);
|
|
393 return;
|
|
394 }
|
|
395 gaim_xfer_set_init_fnc(xfer->xfer, silcgaim_ftp_send);
|
|
396 gaim_xfer_set_request_denied_fnc(xfer->xfer, silcgaim_ftp_request_denied);
|
|
397 gaim_xfer_set_cancel_send_fnc(xfer->xfer, silcgaim_ftp_send_cancel);
|
|
398 xfer->xfer->data = xfer;
|
|
399
|
|
400 /* Choose file to send */
|
9466
|
401 if (file)
|
|
402 gaim_xfer_request_accepted(xfer->xfer, file);
|
|
403 else
|
|
404 gaim_xfer_request(xfer->xfer);
|
8849
|
405
|
|
406 silc_free(clients);
|
|
407 silc_free(nickname);
|
|
408 }
|