1 /*
2 * Get/put file functions for CUPS.
3 *
4 * Copyright 2007-2014 by Apple Inc.
5 * Copyright 1997-2006 by Easy Software Products.
6 *
7 * These coded instructions, statements, and computer programs are the
8 * property of Apple Inc. and are protected by Federal copyright
9 * law. Distribution and use rights are outlined in the file "LICENSE.txt"
10 * which should have been included with this file. If this file is
11 * missing or damaged, see the license at "http://www.cups.org/".
12 *
13 * This file is subject to the Apple OS-Developed Software exception.
14 */
15
16 /*
17 * Include necessary headers...
18 */
19
20 #include "cups-private.h"
21 #include <fcntl.h>
22 #include <sys/stat.h>
23 #if defined(WIN32) || defined(__EMX__)
24 # include <io.h>
25 #else
26 # include <unistd.h>
27 #endif /* WIN32 || __EMX__ */
28
29
30 /*
31 * 'cupsGetFd()' - Get a file from the server.
32 *
33 * This function returns @code HTTP_STATUS_OK@ when the file is successfully retrieved.
34 *
35 * @since CUPS 1.1.20/macOS 10.4@
36 */
37
38 http_status_t /* O - HTTP status */
cupsGetFd(http_t * http,const char * resource,int fd)39 cupsGetFd(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */
40 const char *resource, /* I - Resource name */
41 int fd) /* I - File descriptor */
42 {
43 ssize_t bytes; /* Number of bytes read */
44 char buffer[8192]; /* Buffer for file */
45 http_status_t status; /* HTTP status from server */
46 char if_modified_since[HTTP_MAX_VALUE];
47 /* If-Modified-Since header */
48
49
50 /*
51 * Range check input...
52 */
53
54 DEBUG_printf(("cupsGetFd(http=%p, resource=\"%s\", fd=%d)", (void *)http, resource, fd));
55
56 if (!resource || fd < 0)
57 {
58 if (http)
59 http->error = EINVAL;
60
61 return (HTTP_STATUS_ERROR);
62 }
63
64 if (!http)
65 if ((http = _cupsConnect()) == NULL)
66 return (HTTP_STATUS_SERVICE_UNAVAILABLE);
67
68 /*
69 * Then send GET requests to the HTTP server...
70 */
71
72 strlcpy(if_modified_since, httpGetField(http, HTTP_FIELD_IF_MODIFIED_SINCE),
73 sizeof(if_modified_since));
74
75 do
76 {
77 if (!_cups_strcasecmp(httpGetField(http, HTTP_FIELD_CONNECTION), "close"))
78 {
79 httpClearFields(http);
80 if (httpReconnect2(http, 30000, NULL))
81 {
82 status = HTTP_STATUS_ERROR;
83 break;
84 }
85 }
86
87 httpClearFields(http);
88 httpSetField(http, HTTP_FIELD_AUTHORIZATION, http->authstring);
89 httpSetField(http, HTTP_FIELD_IF_MODIFIED_SINCE, if_modified_since);
90
91 if (httpGet(http, resource))
92 {
93 if (httpReconnect2(http, 30000, NULL))
94 {
95 status = HTTP_STATUS_ERROR;
96 break;
97 }
98 else
99 {
100 status = HTTP_STATUS_UNAUTHORIZED;
101 continue;
102 }
103 }
104
105 while ((status = httpUpdate(http)) == HTTP_STATUS_CONTINUE);
106
107 if (status == HTTP_STATUS_UNAUTHORIZED)
108 {
109 /*
110 * Flush any error message...
111 */
112
113 httpFlush(http);
114
115 /*
116 * See if we can do authentication...
117 */
118
119 if (cupsDoAuthentication(http, "GET", resource))
120 {
121 status = HTTP_STATUS_CUPS_AUTHORIZATION_CANCELED;
122 break;
123 }
124
125 if (httpReconnect2(http, 30000, NULL))
126 {
127 status = HTTP_STATUS_ERROR;
128 break;
129 }
130
131 continue;
132 }
133 #ifdef HAVE_SSL
134 else if (status == HTTP_STATUS_UPGRADE_REQUIRED)
135 {
136 /* Flush any error message... */
137 httpFlush(http);
138
139 /* Reconnect... */
140 if (httpReconnect2(http, 30000, NULL))
141 {
142 status = HTTP_STATUS_ERROR;
143 break;
144 }
145
146 /* Upgrade with encryption... */
147 httpEncryption(http, HTTP_ENCRYPTION_REQUIRED);
148
149 /* Try again, this time with encryption enabled... */
150 continue;
151 }
152 #endif /* HAVE_SSL */
153 }
154 while (status == HTTP_STATUS_UNAUTHORIZED || status == HTTP_STATUS_UPGRADE_REQUIRED);
155
156 /*
157 * See if we actually got the file or an error...
158 */
159
160 if (status == HTTP_STATUS_OK)
161 {
162 /*
163 * Yes, copy the file...
164 */
165
166 while ((bytes = httpRead2(http, buffer, sizeof(buffer))) > 0)
167 write(fd, buffer, (size_t)bytes);
168 }
169 else
170 {
171 _cupsSetHTTPError(status);
172 httpFlush(http);
173 }
174
175 /*
176 * Return the request status...
177 */
178
179 DEBUG_printf(("1cupsGetFd: Returning %d...", status));
180
181 return (status);
182 }
183
184
185 /*
186 * 'cupsGetFile()' - Get a file from the server.
187 *
188 * This function returns @code HTTP_STATUS_OK@ when the file is successfully retrieved.
189 *
190 * @since CUPS 1.1.20/macOS 10.4@
191 */
192
193 http_status_t /* O - HTTP status */
cupsGetFile(http_t * http,const char * resource,const char * filename)194 cupsGetFile(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */
195 const char *resource, /* I - Resource name */
196 const char *filename) /* I - Filename */
197 {
198 int fd; /* File descriptor */
199 http_status_t status; /* Status */
200
201
202 /*
203 * Range check input...
204 */
205
206 if (!http || !resource || !filename)
207 {
208 if (http)
209 http->error = EINVAL;
210
211 return (HTTP_STATUS_ERROR);
212 }
213
214 /*
215 * Create the file...
216 */
217
218 if ((fd = open(filename, O_WRONLY | O_EXCL | O_TRUNC)) < 0)
219 {
220 /*
221 * Couldn't open the file!
222 */
223
224 http->error = errno;
225
226 return (HTTP_STATUS_ERROR);
227 }
228
229 /*
230 * Get the file...
231 */
232
233 status = cupsGetFd(http, resource, fd);
234
235 /*
236 * If the file couldn't be gotten, then remove the file...
237 */
238
239 close(fd);
240
241 if (status != HTTP_STATUS_OK)
242 unlink(filename);
243
244 /*
245 * Return the HTTP status code...
246 */
247
248 return (status);
249 }
250
251
252 /*
253 * 'cupsPutFd()' - Put a file on the server.
254 *
255 * This function returns @code HTTP_STATUS_CREATED@ when the file is stored
256 * successfully.
257 *
258 * @since CUPS 1.1.20/macOS 10.4@
259 */
260
261 http_status_t /* O - HTTP status */
cupsPutFd(http_t * http,const char * resource,int fd)262 cupsPutFd(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */
263 const char *resource, /* I - Resource name */
264 int fd) /* I - File descriptor */
265 {
266 ssize_t bytes; /* Number of bytes read */
267 int retries; /* Number of retries */
268 char buffer[8192]; /* Buffer for file */
269 http_status_t status; /* HTTP status from server */
270
271
272 /*
273 * Range check input...
274 */
275
276 DEBUG_printf(("cupsPutFd(http=%p, resource=\"%s\", fd=%d)", (void *)http, resource, fd));
277
278 if (!resource || fd < 0)
279 {
280 if (http)
281 http->error = EINVAL;
282
283 return (HTTP_STATUS_ERROR);
284 }
285
286 if (!http)
287 if ((http = _cupsConnect()) == NULL)
288 return (HTTP_STATUS_SERVICE_UNAVAILABLE);
289
290 /*
291 * Then send PUT requests to the HTTP server...
292 */
293
294 retries = 0;
295
296 do
297 {
298 if (!_cups_strcasecmp(httpGetField(http, HTTP_FIELD_CONNECTION), "close"))
299 {
300 httpClearFields(http);
301 if (httpReconnect2(http, 30000, NULL))
302 {
303 status = HTTP_STATUS_ERROR;
304 break;
305 }
306 }
307
308 DEBUG_printf(("2cupsPutFd: starting attempt, authstring=\"%s\"...",
309 http->authstring));
310
311 httpClearFields(http);
312 httpSetField(http, HTTP_FIELD_AUTHORIZATION, http->authstring);
313 httpSetField(http, HTTP_FIELD_TRANSFER_ENCODING, "chunked");
314 httpSetExpect(http, HTTP_STATUS_CONTINUE);
315
316 if (httpPut(http, resource))
317 {
318 if (httpReconnect2(http, 30000, NULL))
319 {
320 status = HTTP_STATUS_ERROR;
321 break;
322 }
323 else
324 {
325 status = HTTP_STATUS_UNAUTHORIZED;
326 continue;
327 }
328 }
329
330 /*
331 * Wait up to 1 second for a 100-continue response...
332 */
333
334 if (httpWait(http, 1000))
335 status = httpUpdate(http);
336 else
337 status = HTTP_STATUS_CONTINUE;
338
339 if (status == HTTP_STATUS_CONTINUE)
340 {
341 /*
342 * Copy the file...
343 */
344
345 lseek(fd, 0, SEEK_SET);
346
347 while ((bytes = read(fd, buffer, sizeof(buffer))) > 0)
348 if (httpCheck(http))
349 {
350 if ((status = httpUpdate(http)) != HTTP_STATUS_CONTINUE)
351 break;
352 }
353 else
354 httpWrite2(http, buffer, (size_t)bytes);
355 }
356
357 if (status == HTTP_STATUS_CONTINUE)
358 {
359 httpWrite2(http, buffer, 0);
360
361 while ((status = httpUpdate(http)) == HTTP_STATUS_CONTINUE);
362 }
363
364 if (status == HTTP_STATUS_ERROR && !retries)
365 {
366 DEBUG_printf(("2cupsPutFd: retry on status %d", status));
367
368 retries ++;
369
370 /* Flush any error message... */
371 httpFlush(http);
372
373 /* Reconnect... */
374 if (httpReconnect2(http, 30000, NULL))
375 {
376 status = HTTP_STATUS_ERROR;
377 break;
378 }
379
380 /* Try again... */
381 continue;
382 }
383
384 DEBUG_printf(("2cupsPutFd: status=%d", status));
385
386 if (status == HTTP_STATUS_UNAUTHORIZED)
387 {
388 /*
389 * Flush any error message...
390 */
391
392 httpFlush(http);
393
394 /*
395 * See if we can do authentication...
396 */
397
398 if (cupsDoAuthentication(http, "PUT", resource))
399 {
400 status = HTTP_STATUS_CUPS_AUTHORIZATION_CANCELED;
401 break;
402 }
403
404 if (httpReconnect2(http, 30000, NULL))
405 {
406 status = HTTP_STATUS_ERROR;
407 break;
408 }
409
410 continue;
411 }
412 #ifdef HAVE_SSL
413 else if (status == HTTP_STATUS_UPGRADE_REQUIRED)
414 {
415 /* Flush any error message... */
416 httpFlush(http);
417
418 /* Reconnect... */
419 if (httpReconnect2(http, 30000, NULL))
420 {
421 status = HTTP_STATUS_ERROR;
422 break;
423 }
424
425 /* Upgrade with encryption... */
426 httpEncryption(http, HTTP_ENCRYPTION_REQUIRED);
427
428 /* Try again, this time with encryption enabled... */
429 continue;
430 }
431 #endif /* HAVE_SSL */
432 }
433 while (status == HTTP_STATUS_UNAUTHORIZED || status == HTTP_STATUS_UPGRADE_REQUIRED ||
434 (status == HTTP_STATUS_ERROR && retries < 2));
435
436 /*
437 * See if we actually put the file or an error...
438 */
439
440 if (status != HTTP_STATUS_CREATED)
441 {
442 _cupsSetHTTPError(status);
443 httpFlush(http);
444 }
445
446 DEBUG_printf(("1cupsPutFd: Returning %d...", status));
447
448 return (status);
449 }
450
451
452 /*
453 * 'cupsPutFile()' - Put a file on the server.
454 *
455 * This function returns @code HTTP_CREATED@ when the file is stored
456 * successfully.
457 *
458 * @since CUPS 1.1.20/macOS 10.4@
459 */
460
461 http_status_t /* O - HTTP status */
cupsPutFile(http_t * http,const char * resource,const char * filename)462 cupsPutFile(http_t *http, /* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */
463 const char *resource, /* I - Resource name */
464 const char *filename) /* I - Filename */
465 {
466 int fd; /* File descriptor */
467 http_status_t status; /* Status */
468
469
470 /*
471 * Range check input...
472 */
473
474 if (!http || !resource || !filename)
475 {
476 if (http)
477 http->error = EINVAL;
478
479 return (HTTP_STATUS_ERROR);
480 }
481
482 /*
483 * Open the local file...
484 */
485
486 if ((fd = open(filename, O_RDONLY)) < 0)
487 {
488 /*
489 * Couldn't open the file!
490 */
491
492 http->error = errno;
493
494 return (HTTP_STATUS_ERROR);
495 }
496
497 /*
498 * Put the file...
499 */
500
501 status = cupsPutFd(http, resource, fd);
502
503 close(fd);
504
505 return (status);
506 }
507