14ca1ab94SDag-Erling Smørgrav /*- 24ca1ab94SDag-Erling Smørgrav * Copyright (c) 1998 Dag-Erling Co�dan Sm�rgrav 34ca1ab94SDag-Erling Smørgrav * All rights reserved. 44ca1ab94SDag-Erling Smørgrav * 54ca1ab94SDag-Erling Smørgrav * Redistribution and use in source and binary forms, with or without 64ca1ab94SDag-Erling Smørgrav * modification, are permitted provided that the following conditions 74ca1ab94SDag-Erling Smørgrav * are met: 84ca1ab94SDag-Erling Smørgrav * 1. Redistributions of source code must retain the above copyright 94ca1ab94SDag-Erling Smørgrav * notice, this list of conditions and the following disclaimer 104ca1ab94SDag-Erling Smørgrav * in this position and unchanged. 114ca1ab94SDag-Erling Smørgrav * 2. Redistributions in binary form must reproduce the above copyright 124ca1ab94SDag-Erling Smørgrav * notice, this list of conditions and the following disclaimer in the 134ca1ab94SDag-Erling Smørgrav * documentation and/or other materials provided with the distribution. 144ca1ab94SDag-Erling Smørgrav * 3. The name of the author may not be used to endorse or promote products 154ca1ab94SDag-Erling Smørgrav * derived from this software without specific prior written permission 164ca1ab94SDag-Erling Smørgrav * 174ca1ab94SDag-Erling Smørgrav * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 184ca1ab94SDag-Erling Smørgrav * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 194ca1ab94SDag-Erling Smørgrav * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 204ca1ab94SDag-Erling Smørgrav * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 214ca1ab94SDag-Erling Smørgrav * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 224ca1ab94SDag-Erling Smørgrav * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 234ca1ab94SDag-Erling Smørgrav * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 244ca1ab94SDag-Erling Smørgrav * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 254ca1ab94SDag-Erling Smørgrav * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 264ca1ab94SDag-Erling Smørgrav * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 274ca1ab94SDag-Erling Smørgrav * 287f3dea24SPeter Wemm * $FreeBSD$ 294ca1ab94SDag-Erling Smørgrav */ 304ca1ab94SDag-Erling Smørgrav 31f62e5228SDag-Erling Smørgrav /* 32f62e5228SDag-Erling Smørgrav * The base64 code in this file is based on code from MIT fetch, which 33f62e5228SDag-Erling Smørgrav * has the following copyright and license: 34f62e5228SDag-Erling Smørgrav * 35f62e5228SDag-Erling Smørgrav *- 36f62e5228SDag-Erling Smørgrav * Copyright 1997 Massachusetts Institute of Technology 37f62e5228SDag-Erling Smørgrav * 38f62e5228SDag-Erling Smørgrav * Permission to use, copy, modify, and distribute this software and 39f62e5228SDag-Erling Smørgrav * its documentation for any purpose and without fee is hereby 40f62e5228SDag-Erling Smørgrav * granted, provided that both the above copyright notice and this 41f62e5228SDag-Erling Smørgrav * permission notice appear in all copies, that both the above 42f62e5228SDag-Erling Smørgrav * copyright notice and this permission notice appear in all 43f62e5228SDag-Erling Smørgrav * supporting documentation, and that the name of M.I.T. not be used 44f62e5228SDag-Erling Smørgrav * in advertising or publicity pertaining to distribution of the 45f62e5228SDag-Erling Smørgrav * software without specific, written prior permission. M.I.T. makes 46f62e5228SDag-Erling Smørgrav * no representations about the suitability of this software for any 47f62e5228SDag-Erling Smørgrav * purpose. It is provided "as is" without express or implied 48f62e5228SDag-Erling Smørgrav * warranty. 49f62e5228SDag-Erling Smørgrav * 50f62e5228SDag-Erling Smørgrav * THIS SOFTWARE IS PROVIDED BY M.I.T. ``AS IS''. M.I.T. DISCLAIMS 51f62e5228SDag-Erling Smørgrav * ALL EXPRESS OR IMPLIED WARRANTIES WITH REGARD TO THIS SOFTWARE, 52f62e5228SDag-Erling Smørgrav * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 53f62e5228SDag-Erling Smørgrav * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT 54f62e5228SDag-Erling Smørgrav * SHALL M.I.T. BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 55f62e5228SDag-Erling Smørgrav * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 56f62e5228SDag-Erling Smørgrav * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF 57f62e5228SDag-Erling Smørgrav * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND 58f62e5228SDag-Erling Smørgrav * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 59f62e5228SDag-Erling Smørgrav * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT 60f62e5228SDag-Erling Smørgrav * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 61f62e5228SDag-Erling Smørgrav * SUCH DAMAGE. */ 62f62e5228SDag-Erling Smørgrav 634ca1ab94SDag-Erling Smørgrav #include <sys/param.h> 6428c645cfSHajimu UMEMOTO #include <sys/socket.h> 654ca1ab94SDag-Erling Smørgrav 664ca1ab94SDag-Erling Smørgrav #include <err.h> 674ca1ab94SDag-Erling Smørgrav #include <ctype.h> 6860245e42SDag-Erling Smørgrav #include <locale.h> 69e6182307SDag-Erling Smørgrav #include <netdb.h> 70f62e5228SDag-Erling Smørgrav #include <stdarg.h> 714ca1ab94SDag-Erling Smørgrav #include <stdio.h> 724ca1ab94SDag-Erling Smørgrav #include <stdlib.h> 734ca1ab94SDag-Erling Smørgrav #include <string.h> 7460245e42SDag-Erling Smørgrav #include <time.h> 754ca1ab94SDag-Erling Smørgrav #include <unistd.h> 764ca1ab94SDag-Erling Smørgrav 774ca1ab94SDag-Erling Smørgrav #include "fetch.h" 78842a95ccSDag-Erling Smørgrav #include "common.h" 790fba3a00SDag-Erling Smørgrav #include "httperr.h" 804ca1ab94SDag-Erling Smørgrav 814ca1ab94SDag-Erling Smørgrav extern char *__progname; 824ca1ab94SDag-Erling Smørgrav 834ca1ab94SDag-Erling Smørgrav #define ENDL "\r\n" 844ca1ab94SDag-Erling Smørgrav 853d2a8471SDag-Erling Smørgrav #define HTTP_OK 200 863d2a8471SDag-Erling Smørgrav #define HTTP_PARTIAL 206 873d2a8471SDag-Erling Smørgrav 884ca1ab94SDag-Erling Smørgrav struct cookie 894ca1ab94SDag-Erling Smørgrav { 904ca1ab94SDag-Erling Smørgrav FILE *real_f; 914ca1ab94SDag-Erling Smørgrav #define ENC_NONE 0 924ca1ab94SDag-Erling Smørgrav #define ENC_CHUNKED 1 934ca1ab94SDag-Erling Smørgrav int encoding; /* 1 = chunked, 0 = none */ 944ca1ab94SDag-Erling Smørgrav #define HTTPCTYPELEN 59 954ca1ab94SDag-Erling Smørgrav char content_type[HTTPCTYPELEN+1]; 964ca1ab94SDag-Erling Smørgrav char *buf; 974ca1ab94SDag-Erling Smørgrav int b_cur, eof; 984ca1ab94SDag-Erling Smørgrav unsigned b_len, chunksize; 994ca1ab94SDag-Erling Smørgrav }; 1004ca1ab94SDag-Erling Smørgrav 101f62e5228SDag-Erling Smørgrav /* 102f62e5228SDag-Erling Smørgrav * Send a formatted line; optionally echo to terminal 103f62e5228SDag-Erling Smørgrav */ 104f62e5228SDag-Erling Smørgrav static int 105f62e5228SDag-Erling Smørgrav _http_cmd(FILE *f, char *fmt, ...) 106f62e5228SDag-Erling Smørgrav { 107f62e5228SDag-Erling Smørgrav va_list ap; 108f62e5228SDag-Erling Smørgrav 109f62e5228SDag-Erling Smørgrav va_start(ap, fmt); 110f62e5228SDag-Erling Smørgrav vfprintf(f, fmt, ap); 111f62e5228SDag-Erling Smørgrav #ifndef NDEBUG 112f62e5228SDag-Erling Smørgrav fprintf(stderr, "\033[1m>>> "); 113f62e5228SDag-Erling Smørgrav vfprintf(stderr, fmt, ap); 114f62e5228SDag-Erling Smørgrav fprintf(stderr, "\033[m"); 115f62e5228SDag-Erling Smørgrav #endif 116f62e5228SDag-Erling Smørgrav va_end(ap); 117f62e5228SDag-Erling Smørgrav 118f62e5228SDag-Erling Smørgrav return 0; /* XXX */ 119f62e5228SDag-Erling Smørgrav } 120f62e5228SDag-Erling Smørgrav 121f62e5228SDag-Erling Smørgrav /* 122f62e5228SDag-Erling Smørgrav * Fill the input buffer, do chunk decoding on the fly 123f62e5228SDag-Erling Smørgrav */ 1244ca1ab94SDag-Erling Smørgrav static char * 1254ca1ab94SDag-Erling Smørgrav _http_fillbuf(struct cookie *c) 1264ca1ab94SDag-Erling Smørgrav { 1274ca1ab94SDag-Erling Smørgrav char *ln; 1284ca1ab94SDag-Erling Smørgrav unsigned int len; 1294ca1ab94SDag-Erling Smørgrav 1304ca1ab94SDag-Erling Smørgrav if (c->eof) 1314ca1ab94SDag-Erling Smørgrav return NULL; 1324ca1ab94SDag-Erling Smørgrav 1334ca1ab94SDag-Erling Smørgrav if (c->encoding == ENC_NONE) { 1344ca1ab94SDag-Erling Smørgrav c->buf = fgetln(c->real_f, &(c->b_len)); 1354ca1ab94SDag-Erling Smørgrav c->b_cur = 0; 1364ca1ab94SDag-Erling Smørgrav } else if (c->encoding == ENC_CHUNKED) { 1374ca1ab94SDag-Erling Smørgrav if (c->chunksize == 0) { 1384ca1ab94SDag-Erling Smørgrav ln = fgetln(c->real_f, &len); 1396efb30c8SDag-Erling Smørgrav if (len <= 2) 1406efb30c8SDag-Erling Smørgrav return NULL; 1414ca1ab94SDag-Erling Smørgrav DEBUG(fprintf(stderr, "\033[1m_http_fillbuf(): new chunk: " 1424ca1ab94SDag-Erling Smørgrav "%*.*s\033[m\n", (int)len-2, (int)len-2, ln)); 1434ca1ab94SDag-Erling Smørgrav sscanf(ln, "%x", &(c->chunksize)); 1444ca1ab94SDag-Erling Smørgrav if (!c->chunksize) { 1454ca1ab94SDag-Erling Smørgrav DEBUG(fprintf(stderr, "\033[1m_http_fillbuf(): " 1464ca1ab94SDag-Erling Smørgrav "end of last chunk\033[m\n")); 1474ca1ab94SDag-Erling Smørgrav c->eof = 1; 1484ca1ab94SDag-Erling Smørgrav return NULL; 1494ca1ab94SDag-Erling Smørgrav } 1504ca1ab94SDag-Erling Smørgrav DEBUG(fprintf(stderr, "\033[1m_http_fillbuf(): " 1514ca1ab94SDag-Erling Smørgrav "new chunk: %X\033[m\n", c->chunksize)); 1524ca1ab94SDag-Erling Smørgrav } 1534ca1ab94SDag-Erling Smørgrav c->buf = fgetln(c->real_f, &(c->b_len)); 1544ca1ab94SDag-Erling Smørgrav if (c->b_len > c->chunksize) 1554ca1ab94SDag-Erling Smørgrav c->b_len = c->chunksize; 1564ca1ab94SDag-Erling Smørgrav c->chunksize -= c->b_len; 1574ca1ab94SDag-Erling Smørgrav c->b_cur = 0; 1584ca1ab94SDag-Erling Smørgrav } 1594ca1ab94SDag-Erling Smørgrav else return NULL; /* unknown encoding */ 1604ca1ab94SDag-Erling Smørgrav return c->buf; 1614ca1ab94SDag-Erling Smørgrav } 1624ca1ab94SDag-Erling Smørgrav 163f62e5228SDag-Erling Smørgrav /* 164f62e5228SDag-Erling Smørgrav * Read function 165f62e5228SDag-Erling Smørgrav */ 1664ca1ab94SDag-Erling Smørgrav static int 1674ca1ab94SDag-Erling Smørgrav _http_readfn(struct cookie *c, char *buf, int len) 1684ca1ab94SDag-Erling Smørgrav { 1694ca1ab94SDag-Erling Smørgrav int l, pos = 0; 1704ca1ab94SDag-Erling Smørgrav while (len) { 1714ca1ab94SDag-Erling Smørgrav /* empty buffer */ 1724ca1ab94SDag-Erling Smørgrav if (!c->buf || (c->b_cur == c->b_len)) 1734ca1ab94SDag-Erling Smørgrav if (!_http_fillbuf(c)) 1744ca1ab94SDag-Erling Smørgrav break; 1754ca1ab94SDag-Erling Smørgrav 1764ca1ab94SDag-Erling Smørgrav l = c->b_len - c->b_cur; 1774ca1ab94SDag-Erling Smørgrav if (len < l) l = len; 1784ca1ab94SDag-Erling Smørgrav memcpy(buf + pos, c->buf + c->b_cur, l); 1794ca1ab94SDag-Erling Smørgrav c->b_cur += l; 1804ca1ab94SDag-Erling Smørgrav pos += l; 1814ca1ab94SDag-Erling Smørgrav len -= l; 1824ca1ab94SDag-Erling Smørgrav } 1834ca1ab94SDag-Erling Smørgrav 1844ca1ab94SDag-Erling Smørgrav if (ferror(c->real_f)) 1854ca1ab94SDag-Erling Smørgrav return -1; 1864ca1ab94SDag-Erling Smørgrav else return pos; 1874ca1ab94SDag-Erling Smørgrav } 1884ca1ab94SDag-Erling Smørgrav 189f62e5228SDag-Erling Smørgrav /* 190f62e5228SDag-Erling Smørgrav * Write function 191f62e5228SDag-Erling Smørgrav */ 1924ca1ab94SDag-Erling Smørgrav static int 1934ca1ab94SDag-Erling Smørgrav _http_writefn(struct cookie *c, const char *buf, int len) 1944ca1ab94SDag-Erling Smørgrav { 1954ca1ab94SDag-Erling Smørgrav size_t r = fwrite(buf, 1, (size_t)len, c->real_f); 1964ca1ab94SDag-Erling Smørgrav return r ? r : -1; 1974ca1ab94SDag-Erling Smørgrav } 1984ca1ab94SDag-Erling Smørgrav 199f62e5228SDag-Erling Smørgrav /* 200f62e5228SDag-Erling Smørgrav * Close function 201f62e5228SDag-Erling Smørgrav */ 2024ca1ab94SDag-Erling Smørgrav static int 2034ca1ab94SDag-Erling Smørgrav _http_closefn(struct cookie *c) 2044ca1ab94SDag-Erling Smørgrav { 2054ca1ab94SDag-Erling Smørgrav int r = fclose(c->real_f); 2064ca1ab94SDag-Erling Smørgrav free(c); 2074ca1ab94SDag-Erling Smørgrav return (r == EOF) ? -1 : 0; 2084ca1ab94SDag-Erling Smørgrav } 2094ca1ab94SDag-Erling Smørgrav 210f62e5228SDag-Erling Smørgrav /* 211f62e5228SDag-Erling Smørgrav * Extract content type from cookie 212f62e5228SDag-Erling Smørgrav */ 2134ca1ab94SDag-Erling Smørgrav char * 2144ca1ab94SDag-Erling Smørgrav fetchContentType(FILE *f) 2154ca1ab94SDag-Erling Smørgrav { 2164ca1ab94SDag-Erling Smørgrav /* 2174ca1ab94SDag-Erling Smørgrav * We have no way of making sure this really *is* one of our cookies, 2184ca1ab94SDag-Erling Smørgrav * so just check for a null pointer and hope for the best. 2194ca1ab94SDag-Erling Smørgrav */ 2204ca1ab94SDag-Erling Smørgrav return f->_cookie ? (((struct cookie *)f->_cookie)->content_type) : NULL; 2214ca1ab94SDag-Erling Smørgrav } 2224ca1ab94SDag-Erling Smørgrav 223f62e5228SDag-Erling Smørgrav /* 224f62e5228SDag-Erling Smørgrav * Base64 encoding 225f62e5228SDag-Erling Smørgrav */ 226f62e5228SDag-Erling Smørgrav int 227f62e5228SDag-Erling Smørgrav _http_base64(char *dst, char *src, int l) 228f62e5228SDag-Erling Smørgrav { 229f62e5228SDag-Erling Smørgrav static const char base64[] = 230f62e5228SDag-Erling Smørgrav "ABCDEFGHIJKLMNOPQRSTUVWXYZ" 231f62e5228SDag-Erling Smørgrav "abcdefghijklmnopqrstuvwxyz" 232f62e5228SDag-Erling Smørgrav "0123456789+/"; 233f62e5228SDag-Erling Smørgrav int t, r = 0; 234f62e5228SDag-Erling Smørgrav 235f62e5228SDag-Erling Smørgrav while (l >= 3) { 236f62e5228SDag-Erling Smørgrav t = (src[0] << 16) | (src[1] << 8) | src[2]; 237f62e5228SDag-Erling Smørgrav dst[0] = base64[(t >> 18) & 0x3f]; 238f62e5228SDag-Erling Smørgrav dst[1] = base64[(t >> 12) & 0x3f]; 239f62e5228SDag-Erling Smørgrav dst[2] = base64[(t >> 6) & 0x3f]; 240f62e5228SDag-Erling Smørgrav dst[3] = base64[(t >> 0) & 0x3f]; 241f62e5228SDag-Erling Smørgrav src += 3; l -= 3; 242f62e5228SDag-Erling Smørgrav dst += 4; r += 4; 243f62e5228SDag-Erling Smørgrav } 244f62e5228SDag-Erling Smørgrav 245f62e5228SDag-Erling Smørgrav switch (l) { 246f62e5228SDag-Erling Smørgrav case 2: 247f62e5228SDag-Erling Smørgrav t = (src[0] << 16) | (src[1] << 8); 248f62e5228SDag-Erling Smørgrav dst[0] = base64[(t >> 18) & 0x3f]; 249f62e5228SDag-Erling Smørgrav dst[1] = base64[(t >> 12) & 0x3f]; 250f62e5228SDag-Erling Smørgrav dst[2] = base64[(t >> 6) & 0x3f]; 251f62e5228SDag-Erling Smørgrav dst[3] = '='; 252f62e5228SDag-Erling Smørgrav dst += 4; 253f62e5228SDag-Erling Smørgrav r += 4; 254f62e5228SDag-Erling Smørgrav break; 255f62e5228SDag-Erling Smørgrav case 1: 256f62e5228SDag-Erling Smørgrav t = src[0] << 16; 257f62e5228SDag-Erling Smørgrav dst[0] = base64[(t >> 18) & 0x3f]; 258f62e5228SDag-Erling Smørgrav dst[1] = base64[(t >> 12) & 0x3f]; 259f62e5228SDag-Erling Smørgrav dst[2] = dst[3] = '='; 260f62e5228SDag-Erling Smørgrav dst += 4; 261f62e5228SDag-Erling Smørgrav r += 4; 262f62e5228SDag-Erling Smørgrav break; 263f62e5228SDag-Erling Smørgrav case 0: 264f62e5228SDag-Erling Smørgrav break; 265f62e5228SDag-Erling Smørgrav } 266f62e5228SDag-Erling Smørgrav 267f62e5228SDag-Erling Smørgrav *dst = 0; 268f62e5228SDag-Erling Smørgrav return r; 269f62e5228SDag-Erling Smørgrav } 270f62e5228SDag-Erling Smørgrav 271f62e5228SDag-Erling Smørgrav /* 272f62e5228SDag-Erling Smørgrav * Encode username and password 273f62e5228SDag-Erling Smørgrav */ 274f62e5228SDag-Erling Smørgrav char * 275f62e5228SDag-Erling Smørgrav _http_auth(char *usr, char *pwd) 276f62e5228SDag-Erling Smørgrav { 277f62e5228SDag-Erling Smørgrav int len, lu, lp; 278f62e5228SDag-Erling Smørgrav char *str, *s; 279f62e5228SDag-Erling Smørgrav 280f62e5228SDag-Erling Smørgrav lu = strlen(usr); 281f62e5228SDag-Erling Smørgrav lp = strlen(pwd); 282f62e5228SDag-Erling Smørgrav 283f62e5228SDag-Erling Smørgrav len = (lu * 4 + 2) / 3 /* user name, round up */ 284f62e5228SDag-Erling Smørgrav + 1 /* colon */ 285f62e5228SDag-Erling Smørgrav + (lp * 4 + 2) / 3 /* password, round up */ 286f62e5228SDag-Erling Smørgrav + 1; /* null */ 287f62e5228SDag-Erling Smørgrav 288f62e5228SDag-Erling Smørgrav if ((s = str = (char *)malloc(len)) == NULL) 289f62e5228SDag-Erling Smørgrav return NULL; 290f62e5228SDag-Erling Smørgrav 291f62e5228SDag-Erling Smørgrav s += _http_base64(s, usr, lu); 292f62e5228SDag-Erling Smørgrav *s++ = ':'; 293f62e5228SDag-Erling Smørgrav s += _http_base64(s, pwd, lp); 294f62e5228SDag-Erling Smørgrav *s = 0; 295f62e5228SDag-Erling Smørgrav 296f62e5228SDag-Erling Smørgrav return str; 297f62e5228SDag-Erling Smørgrav } 298f62e5228SDag-Erling Smørgrav 299f62e5228SDag-Erling Smørgrav /* 30060245e42SDag-Erling Smørgrav * Connect to server or proxy 301f62e5228SDag-Erling Smørgrav */ 3024ca1ab94SDag-Erling Smørgrav FILE * 30360245e42SDag-Erling Smørgrav _http_connect(struct url *URL, char *flags) 3044ca1ab94SDag-Erling Smørgrav { 30560245e42SDag-Erling Smørgrav int direct, sd = -1, verbose; 30628c645cfSHajimu UMEMOTO #ifdef INET6 30728c645cfSHajimu UMEMOTO int af = AF_UNSPEC; 30828c645cfSHajimu UMEMOTO #else 30928c645cfSHajimu UMEMOTO int af = AF_INET; 31028c645cfSHajimu UMEMOTO #endif 3114ca1ab94SDag-Erling Smørgrav size_t len; 31260245e42SDag-Erling Smørgrav char *px; 31360245e42SDag-Erling Smørgrav FILE *f; 3144ca1ab94SDag-Erling Smørgrav 315f5f109a0SDag-Erling Smørgrav direct = (flags && strchr(flags, 'd')); 316f5f109a0SDag-Erling Smørgrav verbose = (flags && strchr(flags, 'v')); 31728c645cfSHajimu UMEMOTO if ((flags && strchr(flags, '4'))) 31828c645cfSHajimu UMEMOTO af = AF_INET; 31928c645cfSHajimu UMEMOTO else if ((flags && strchr(flags, '6'))) 32028c645cfSHajimu UMEMOTO af = AF_INET6; 3210fba3a00SDag-Erling Smørgrav 3224ca1ab94SDag-Erling Smørgrav /* check port */ 323e6182307SDag-Erling Smørgrav if (!URL->port) { 324e6182307SDag-Erling Smørgrav struct servent *se; 325e6182307SDag-Erling Smørgrav 326c97925adSHajimu UMEMOTO if (strcasecmp(URL->scheme, "ftp") == 0) 327c97925adSHajimu UMEMOTO if ((se = getservbyname("ftp", "tcp")) != NULL) 328c97925adSHajimu UMEMOTO URL->port = ntohs(se->s_port); 329c97925adSHajimu UMEMOTO else 330c97925adSHajimu UMEMOTO URL->port = 21; 331c97925adSHajimu UMEMOTO else 332e6182307SDag-Erling Smørgrav if ((se = getservbyname("http", "tcp")) != NULL) 333e6182307SDag-Erling Smørgrav URL->port = ntohs(se->s_port); 334e6182307SDag-Erling Smørgrav else 335e6182307SDag-Erling Smørgrav URL->port = 80; 336e6182307SDag-Erling Smørgrav } 3374ca1ab94SDag-Erling Smørgrav 3384ca1ab94SDag-Erling Smørgrav /* attempt to connect to proxy server */ 339f5f109a0SDag-Erling Smørgrav if (!direct && (px = getenv("HTTP_PROXY")) != NULL) { 3403b7a6740SDag-Erling Smørgrav char host[MAXHOSTNAMELEN]; 341e6182307SDag-Erling Smørgrav int port = 0; 3424ca1ab94SDag-Erling Smørgrav 3434ca1ab94SDag-Erling Smørgrav /* measure length */ 34428c645cfSHajimu UMEMOTO #ifdef INET6 34528c645cfSHajimu UMEMOTO if (px[0] != '[' || 34628c645cfSHajimu UMEMOTO (len = strcspn(px, "]")) >= strlen(px) || 34728c645cfSHajimu UMEMOTO (px[++len] != '\0' && px[len] != ':')) 34828c645cfSHajimu UMEMOTO #endif 3494ca1ab94SDag-Erling Smørgrav len = strcspn(px, ":"); 3504ca1ab94SDag-Erling Smørgrav 351f5f109a0SDag-Erling Smørgrav /* get port (XXX atoi is a little too tolerant perhaps?) */ 352e6182307SDag-Erling Smørgrav if (px[len] == ':') { 353e6182307SDag-Erling Smørgrav if (strspn(px+len+1, "0123456789") != strlen(px+len+1) 354e6182307SDag-Erling Smørgrav || strlen(px+len+1) > 5) { 355e6182307SDag-Erling Smørgrav /* XXX we should emit some kind of warning */ 356e6182307SDag-Erling Smørgrav } 3574ca1ab94SDag-Erling Smørgrav port = atoi(px+len+1); 358e6182307SDag-Erling Smørgrav if (port < 1 || port > 65535) { 359e6182307SDag-Erling Smørgrav /* XXX we should emit some kind of warning */ 360e6182307SDag-Erling Smørgrav } 361e6182307SDag-Erling Smørgrav } 362e6182307SDag-Erling Smørgrav if (!port) { 363e6182307SDag-Erling Smørgrav #if 0 364e6182307SDag-Erling Smørgrav /* 365e6182307SDag-Erling Smørgrav * commented out, since there is currently no service name 366e6182307SDag-Erling Smørgrav * for HTTP proxies 367e6182307SDag-Erling Smørgrav */ 368e6182307SDag-Erling Smørgrav struct servent *se; 369e6182307SDag-Erling Smørgrav 370e6182307SDag-Erling Smørgrav if ((se = getservbyname("xxxx", "tcp")) != NULL) 371e6182307SDag-Erling Smørgrav port = ntohs(se->s_port); 372e6182307SDag-Erling Smørgrav else 373e6182307SDag-Erling Smørgrav #endif 374e6182307SDag-Erling Smørgrav port = 3128; 375e6182307SDag-Erling Smørgrav } 3764ca1ab94SDag-Erling Smørgrav 3774ca1ab94SDag-Erling Smørgrav /* get host name */ 37828c645cfSHajimu UMEMOTO #ifdef INET6 37928c645cfSHajimu UMEMOTO if (len > 1 && px[0] == '[' && px[len - 1] == ']') { 38028c645cfSHajimu UMEMOTO px++; 38128c645cfSHajimu UMEMOTO len -= 2; 38228c645cfSHajimu UMEMOTO } 38328c645cfSHajimu UMEMOTO #endif 3844ca1ab94SDag-Erling Smørgrav if (len >= MAXHOSTNAMELEN) 3854ca1ab94SDag-Erling Smørgrav len = MAXHOSTNAMELEN - 1; 3864ca1ab94SDag-Erling Smørgrav strncpy(host, px, len); 3874ca1ab94SDag-Erling Smørgrav host[len] = 0; 3884ca1ab94SDag-Erling Smørgrav 3894ca1ab94SDag-Erling Smørgrav /* connect */ 39028c645cfSHajimu UMEMOTO sd = _fetch_connect(host, port, af, verbose); 3914ca1ab94SDag-Erling Smørgrav } 3924ca1ab94SDag-Erling Smørgrav 3934ca1ab94SDag-Erling Smørgrav /* if no proxy is configured or could be contacted, try direct */ 394ecc91352SDag-Erling Smørgrav if (sd == -1) { 395c97925adSHajimu UMEMOTO if (strcasecmp(URL->scheme, "ftp") == 0) 396c97925adSHajimu UMEMOTO goto ouch; 39728c645cfSHajimu UMEMOTO if ((sd = _fetch_connect(URL->host, URL->port, af, verbose)) == -1) 3984ca1ab94SDag-Erling Smørgrav goto ouch; 3994ca1ab94SDag-Erling Smørgrav } 4004ca1ab94SDag-Erling Smørgrav 4014ca1ab94SDag-Erling Smørgrav /* reopen as stream */ 4028e3986eaSDag-Erling Smørgrav if ((f = fdopen(sd, "r+")) == NULL) 4034ca1ab94SDag-Erling Smørgrav goto ouch; 40460245e42SDag-Erling Smørgrav 40560245e42SDag-Erling Smørgrav return f; 40660245e42SDag-Erling Smørgrav 40760245e42SDag-Erling Smørgrav ouch: 40860245e42SDag-Erling Smørgrav if (sd >= 0) 40960245e42SDag-Erling Smørgrav close(sd); 41060245e42SDag-Erling Smørgrav _http_seterr(999); /* XXX do this properly RSN */ 41160245e42SDag-Erling Smørgrav return NULL; 41260245e42SDag-Erling Smørgrav } 41360245e42SDag-Erling Smørgrav 41460245e42SDag-Erling Smørgrav /* 41560245e42SDag-Erling Smørgrav * Send a HEAD or GET request 41660245e42SDag-Erling Smørgrav */ 41760245e42SDag-Erling Smørgrav int 41860245e42SDag-Erling Smørgrav _http_request(FILE *f, char *op, struct url *URL, char *flags) 41960245e42SDag-Erling Smørgrav { 42060245e42SDag-Erling Smørgrav int e, verbose; 42160245e42SDag-Erling Smørgrav char *ln, *p; 42260245e42SDag-Erling Smørgrav size_t len; 42328c645cfSHajimu UMEMOTO char *host; 42428c645cfSHajimu UMEMOTO #ifdef INET6 42528c645cfSHajimu UMEMOTO char hbuf[MAXHOSTNAMELEN + 1]; 42628c645cfSHajimu UMEMOTO #endif 42760245e42SDag-Erling Smørgrav 42860245e42SDag-Erling Smørgrav verbose = (flags && strchr(flags, 'v')); 4294ca1ab94SDag-Erling Smørgrav 43028c645cfSHajimu UMEMOTO host = URL->host; 43128c645cfSHajimu UMEMOTO #ifdef INET6 43228c645cfSHajimu UMEMOTO if (strchr(URL->host, ':')) { 43328c645cfSHajimu UMEMOTO snprintf(hbuf, sizeof(hbuf), "[%s]", URL->host); 43428c645cfSHajimu UMEMOTO host = hbuf; 43528c645cfSHajimu UMEMOTO } 43628c645cfSHajimu UMEMOTO #endif 43728c645cfSHajimu UMEMOTO 4384ca1ab94SDag-Erling Smørgrav /* send request (proxies require absolute form, so use that) */ 4390fba3a00SDag-Erling Smørgrav if (verbose) 440c97925adSHajimu UMEMOTO _fetch_info("requesting %s://%s:%d%s", 44128c645cfSHajimu UMEMOTO URL->scheme, host, URL->port, URL->doc); 44260245e42SDag-Erling Smørgrav _http_cmd(f, "%s %s://%s:%d%s HTTP/1.1" ENDL, 44328c645cfSHajimu UMEMOTO op, URL->scheme, host, URL->port, URL->doc); 4444ca1ab94SDag-Erling Smørgrav 4454ca1ab94SDag-Erling Smørgrav /* start sending headers away */ 4464ca1ab94SDag-Erling Smørgrav if (URL->user[0] || URL->pwd[0]) { 447f62e5228SDag-Erling Smørgrav char *auth_str = _http_auth(URL->user, URL->pwd); 448f62e5228SDag-Erling Smørgrav if (!auth_str) 44960245e42SDag-Erling Smørgrav return 999; /* XXX wrong */ 450f62e5228SDag-Erling Smørgrav _http_cmd(f, "Authorization: Basic %s" ENDL, auth_str); 451f62e5228SDag-Erling Smørgrav free(auth_str); 4524ca1ab94SDag-Erling Smørgrav } 45328c645cfSHajimu UMEMOTO _http_cmd(f, "Host: %s:%d" ENDL, host, URL->port); 454f62e5228SDag-Erling Smørgrav _http_cmd(f, "User-Agent: %s " _LIBFETCH_VER ENDL, __progname); 4553d2a8471SDag-Erling Smørgrav if (URL->offset) 4563d2a8471SDag-Erling Smørgrav _http_cmd(f, "Range: bytes=%lld-" ENDL, URL->offset); 457f62e5228SDag-Erling Smørgrav _http_cmd(f, "Connection: close" ENDL ENDL); 4584ca1ab94SDag-Erling Smørgrav 4594ca1ab94SDag-Erling Smørgrav /* get response */ 4604ca1ab94SDag-Erling Smørgrav if ((ln = fgetln(f, &len)) == NULL) 46160245e42SDag-Erling Smørgrav return 999; 4624ca1ab94SDag-Erling Smørgrav DEBUG(fprintf(stderr, "response: [\033[1m%*.*s\033[m]\n", 4634ca1ab94SDag-Erling Smørgrav (int)len-2, (int)len-2, ln)); 4644ca1ab94SDag-Erling Smørgrav 4654ca1ab94SDag-Erling Smørgrav /* we can't use strchr() and friends since ln isn't NUL-terminated */ 4664ca1ab94SDag-Erling Smørgrav p = ln; 4674ca1ab94SDag-Erling Smørgrav while ((p < ln + len) && !isspace(*p)) 4684ca1ab94SDag-Erling Smørgrav p++; 4694ca1ab94SDag-Erling Smørgrav while ((p < ln + len) && !isdigit(*p)) 4704ca1ab94SDag-Erling Smørgrav p++; 4714ca1ab94SDag-Erling Smørgrav if (!isdigit(*p)) 47260245e42SDag-Erling Smørgrav return 999; 47360245e42SDag-Erling Smørgrav 4743b7a6740SDag-Erling Smørgrav e = atoi(p); 4753b7a6740SDag-Erling Smørgrav DEBUG(fprintf(stderr, "code: [\033[1m%d\033[m]\n", e)); 47660245e42SDag-Erling Smørgrav return e; 47760245e42SDag-Erling Smørgrav } 47860245e42SDag-Erling Smørgrav 47960245e42SDag-Erling Smørgrav /* 48060245e42SDag-Erling Smørgrav * Check a header line 48160245e42SDag-Erling Smørgrav */ 48260245e42SDag-Erling Smørgrav char * 48360245e42SDag-Erling Smørgrav _http_match(char *str, char *hdr) 48460245e42SDag-Erling Smørgrav { 48560245e42SDag-Erling Smørgrav while (*str && *hdr && tolower(*str++) == tolower(*hdr++)) 48660245e42SDag-Erling Smørgrav /* nothing */; 48760245e42SDag-Erling Smørgrav if (*str || *hdr != ':') 48860245e42SDag-Erling Smørgrav return NULL; 48960245e42SDag-Erling Smørgrav while (*hdr && isspace(*++hdr)) 49060245e42SDag-Erling Smørgrav /* nothing */; 49160245e42SDag-Erling Smørgrav return hdr; 49260245e42SDag-Erling Smørgrav } 49360245e42SDag-Erling Smørgrav 49460245e42SDag-Erling Smørgrav /* 49560245e42SDag-Erling Smørgrav * Retrieve a file by HTTP 49660245e42SDag-Erling Smørgrav */ 49760245e42SDag-Erling Smørgrav FILE * 49860245e42SDag-Erling Smørgrav fetchGetHTTP(struct url *URL, char *flags) 49960245e42SDag-Erling Smørgrav { 500f8f4130bSDag-Erling Smørgrav int e, enc = ENC_NONE, i; 50160245e42SDag-Erling Smørgrav struct cookie *c; 50260245e42SDag-Erling Smørgrav char *ln, *p, *q; 50360245e42SDag-Erling Smørgrav FILE *f, *cf; 50460245e42SDag-Erling Smørgrav size_t len; 50560245e42SDag-Erling Smørgrav off_t pos = 0; 50660245e42SDag-Erling Smørgrav 50760245e42SDag-Erling Smørgrav /* allocate cookie */ 50860245e42SDag-Erling Smørgrav if ((c = calloc(1, sizeof *c)) == NULL) 50960245e42SDag-Erling Smørgrav return NULL; 51060245e42SDag-Erling Smørgrav 51160245e42SDag-Erling Smørgrav /* connect */ 51260245e42SDag-Erling Smørgrav if ((f = _http_connect(URL, flags)) == NULL) { 51360245e42SDag-Erling Smørgrav free(c); 51460245e42SDag-Erling Smørgrav return NULL; 51560245e42SDag-Erling Smørgrav } 51660245e42SDag-Erling Smørgrav c->real_f = f; 51760245e42SDag-Erling Smørgrav 51860245e42SDag-Erling Smørgrav e = _http_request(f, "GET", URL, flags); 5194ca1ab94SDag-Erling Smørgrav 5204ca1ab94SDag-Erling Smørgrav /* add code to handle redirects later */ 5213d2a8471SDag-Erling Smørgrav if (e != (URL->offset ? HTTP_PARTIAL : HTTP_OK)) { 5223b7a6740SDag-Erling Smørgrav _http_seterr(e); 5234ca1ab94SDag-Erling Smørgrav goto fouch; 5248e3986eaSDag-Erling Smørgrav } 5254ca1ab94SDag-Erling Smørgrav 5264ca1ab94SDag-Erling Smørgrav /* browse through header */ 5274ca1ab94SDag-Erling Smørgrav while (1) { 5284ca1ab94SDag-Erling Smørgrav if ((ln = fgetln(f, &len)) == NULL) 5294ca1ab94SDag-Erling Smørgrav goto fouch; 5304ca1ab94SDag-Erling Smørgrav if ((ln[0] == '\r') || (ln[0] == '\n')) 5314ca1ab94SDag-Erling Smørgrav break; 53260245e42SDag-Erling Smørgrav while (isspace(ln[len-1])) 53360245e42SDag-Erling Smørgrav --len; 53460245e42SDag-Erling Smørgrav ln[len] = '\0'; /* XXX */ 53560245e42SDag-Erling Smørgrav DEBUG(fprintf(stderr, "header: [\033[1m%s\033[m]\n", ln)); 53660245e42SDag-Erling Smørgrav if ((p = _http_match("Transfer-Encoding", ln)) != NULL) { 53760245e42SDag-Erling Smørgrav for (q = p; *q && !isspace(*q); q++) 5384ca1ab94SDag-Erling Smørgrav /* VOID */ ; 5394ca1ab94SDag-Erling Smørgrav *q = 0; 5404ca1ab94SDag-Erling Smørgrav if (strcasecmp(p, "chunked") == 0) 5414ca1ab94SDag-Erling Smørgrav enc = ENC_CHUNKED; 54260245e42SDag-Erling Smørgrav DEBUG(fprintf(stderr, "transfer encoding: [\033[1m%s\033[m]\n", p)); 54360245e42SDag-Erling Smørgrav } else if ((p = _http_match("Content-Type", ln)) != NULL) { 54460245e42SDag-Erling Smørgrav for (i = 0; *p && i < HTTPCTYPELEN; p++, i++) 54560245e42SDag-Erling Smørgrav c->content_type[i] = *p; 5464ca1ab94SDag-Erling Smørgrav do c->content_type[i--] = 0; while (isspace(c->content_type[i])); 54760245e42SDag-Erling Smørgrav DEBUG(fprintf(stderr, "content type: [\033[1m%s\033[m]\n", 5484ca1ab94SDag-Erling Smørgrav c->content_type)); 54960245e42SDag-Erling Smørgrav } else if ((p = _http_match("Content-Range", ln)) != NULL) { 55060245e42SDag-Erling Smørgrav if (strncasecmp(p, "bytes ", 6) != 0) 5513d2a8471SDag-Erling Smørgrav goto fouch; 55260245e42SDag-Erling Smørgrav p += 6; 55360245e42SDag-Erling Smørgrav while (*p && isdigit(*p)) 5543d2a8471SDag-Erling Smørgrav pos = pos * 10 + (*p++ - '0'); 5553d2a8471SDag-Erling Smørgrav /* XXX wouldn't hurt to be slightly more paranoid here */ 55660245e42SDag-Erling Smørgrav DEBUG(fprintf(stderr, "content range: [\033[1m%lld-\033[m]\n", pos)); 5573d2a8471SDag-Erling Smørgrav if (pos > URL->offset) 5583d2a8471SDag-Erling Smørgrav goto fouch; 5594ca1ab94SDag-Erling Smørgrav } 5604ca1ab94SDag-Erling Smørgrav } 5614ca1ab94SDag-Erling Smørgrav 5624ca1ab94SDag-Erling Smørgrav /* only body remains */ 5634ca1ab94SDag-Erling Smørgrav c->encoding = enc; 5644ca1ab94SDag-Erling Smørgrav cf = funopen(c, 5654ca1ab94SDag-Erling Smørgrav (int (*)(void *, char *, int))_http_readfn, 5664ca1ab94SDag-Erling Smørgrav (int (*)(void *, const char *, int))_http_writefn, 5674ca1ab94SDag-Erling Smørgrav (fpos_t (*)(void *, fpos_t, int))NULL, 5684ca1ab94SDag-Erling Smørgrav (int (*)(void *))_http_closefn); 5694ca1ab94SDag-Erling Smørgrav if (cf == NULL) 5704ca1ab94SDag-Erling Smørgrav goto fouch; 571e6182307SDag-Erling Smørgrav 5723d2a8471SDag-Erling Smørgrav while (pos < URL->offset) 5733d2a8471SDag-Erling Smørgrav if (fgetc(cf) == EOF) 5743d2a8471SDag-Erling Smørgrav goto cfouch; 5753d2a8471SDag-Erling Smørgrav 5764ca1ab94SDag-Erling Smørgrav return cf; 5774ca1ab94SDag-Erling Smørgrav 5784ca1ab94SDag-Erling Smørgrav fouch: 5794ca1ab94SDag-Erling Smørgrav fclose(f); 5804ca1ab94SDag-Erling Smørgrav free(c); 5810fba3a00SDag-Erling Smørgrav _http_seterr(999); /* XXX do this properly RSN */ 5824ca1ab94SDag-Erling Smørgrav return NULL; 5833d2a8471SDag-Erling Smørgrav cfouch: 5843d2a8471SDag-Erling Smørgrav fclose(cf); 5853d2a8471SDag-Erling Smørgrav _http_seterr(999); /* XXX do this properly RSN */ 5863d2a8471SDag-Erling Smørgrav return NULL; 5874ca1ab94SDag-Erling Smørgrav } 5884ca1ab94SDag-Erling Smørgrav 5894ca1ab94SDag-Erling Smørgrav FILE * 590d8acd8dcSDag-Erling Smørgrav fetchPutHTTP(struct url *URL, char *flags) 5914ca1ab94SDag-Erling Smørgrav { 5924ca1ab94SDag-Erling Smørgrav warnx("fetchPutHTTP(): not implemented"); 5934ca1ab94SDag-Erling Smørgrav return NULL; 5944ca1ab94SDag-Erling Smørgrav } 595d8acd8dcSDag-Erling Smørgrav 596d8acd8dcSDag-Erling Smørgrav /* 597d8acd8dcSDag-Erling Smørgrav * Get an HTTP document's metadata 598d8acd8dcSDag-Erling Smørgrav */ 599d8acd8dcSDag-Erling Smørgrav int 60060245e42SDag-Erling Smørgrav fetchStatHTTP(struct url *URL, struct url_stat *us, char *flags) 601d8acd8dcSDag-Erling Smørgrav { 602f8f4130bSDag-Erling Smørgrav int e; 60360245e42SDag-Erling Smørgrav size_t len; 60460245e42SDag-Erling Smørgrav char *ln, *p; 60560245e42SDag-Erling Smørgrav FILE *f; 60660245e42SDag-Erling Smørgrav 607f8f4130bSDag-Erling Smørgrav us->size = -1; 608f8f4130bSDag-Erling Smørgrav us->atime = us->mtime = 0; 60960245e42SDag-Erling Smørgrav 61060245e42SDag-Erling Smørgrav /* connect */ 61160245e42SDag-Erling Smørgrav if ((f = _http_connect(URL, flags)) == NULL) 61260245e42SDag-Erling Smørgrav return -1; 61360245e42SDag-Erling Smørgrav 61460245e42SDag-Erling Smørgrav if ((e = _http_request(f, "HEAD", URL, flags)) != HTTP_OK) { 61560245e42SDag-Erling Smørgrav _http_seterr(e); 61660245e42SDag-Erling Smørgrav goto ouch; 61760245e42SDag-Erling Smørgrav } 61860245e42SDag-Erling Smørgrav 61960245e42SDag-Erling Smørgrav while (1) { 62060245e42SDag-Erling Smørgrav if ((ln = fgetln(f, &len)) == NULL) 62160245e42SDag-Erling Smørgrav goto fouch; 62260245e42SDag-Erling Smørgrav if ((ln[0] == '\r') || (ln[0] == '\n')) 62360245e42SDag-Erling Smørgrav break; 62460245e42SDag-Erling Smørgrav while (isspace(ln[len-1])) 62560245e42SDag-Erling Smørgrav --len; 62660245e42SDag-Erling Smørgrav ln[len] = '\0'; /* XXX */ 62760245e42SDag-Erling Smørgrav DEBUG(fprintf(stderr, "header: [\033[1m%s\033[m]\n", ln)); 62860245e42SDag-Erling Smørgrav if ((p = _http_match("Last-Modified", ln)) != NULL) { 62960245e42SDag-Erling Smørgrav struct tm tm; 63060245e42SDag-Erling Smørgrav char locale[64]; 63160245e42SDag-Erling Smørgrav 63260245e42SDag-Erling Smørgrav strncpy(locale, setlocale(LC_TIME, NULL), sizeof locale); 63360245e42SDag-Erling Smørgrav setlocale(LC_TIME, "C"); 63460245e42SDag-Erling Smørgrav strptime(p, "%a, %d %b %Y %H:%M:%S GMT", &tm); 63560245e42SDag-Erling Smørgrav /* XXX should add support for date-2 and date-3 */ 63660245e42SDag-Erling Smørgrav setlocale(LC_TIME, locale); 63760245e42SDag-Erling Smørgrav us->atime = us->mtime = timegm(&tm); 63860245e42SDag-Erling Smørgrav DEBUG(fprintf(stderr, "last modified: [\033[1m%04d-%02d-%02d " 63960245e42SDag-Erling Smørgrav "%02d:%02d:%02d\033[m]\n", 64060245e42SDag-Erling Smørgrav tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday, 64160245e42SDag-Erling Smørgrav tm.tm_hour, tm.tm_min, tm.tm_sec)); 64260245e42SDag-Erling Smørgrav } else if ((p = _http_match("Content-Length", ln)) != NULL) { 64360245e42SDag-Erling Smørgrav us->size = 0; 64460245e42SDag-Erling Smørgrav while (*p && isdigit(*p)) 64560245e42SDag-Erling Smørgrav us->size = us->size * 10 + (*p++ - '0'); 64660245e42SDag-Erling Smørgrav DEBUG(fprintf(stderr, "content length: [\033[1m%lld\033[m]\n", us->size)); 64760245e42SDag-Erling Smørgrav } 64860245e42SDag-Erling Smørgrav } 64960245e42SDag-Erling Smørgrav 650f8f4130bSDag-Erling Smørgrav fclose(f); 65160245e42SDag-Erling Smørgrav return 0; 65260245e42SDag-Erling Smørgrav ouch: 65360245e42SDag-Erling Smørgrav _http_seterr(999); /* XXX do this properly RSN */ 65460245e42SDag-Erling Smørgrav fouch: 65560245e42SDag-Erling Smørgrav fclose(f); 656d8acd8dcSDag-Erling Smørgrav return -1; 657d8acd8dcSDag-Erling Smørgrav } 658ce71b736SDag-Erling Smørgrav 659ce71b736SDag-Erling Smørgrav /* 660ce71b736SDag-Erling Smørgrav * List a directory 661ce71b736SDag-Erling Smørgrav */ 662ce71b736SDag-Erling Smørgrav struct url_ent * 663ce71b736SDag-Erling Smørgrav fetchListHTTP(struct url *url, char *flags) 664ce71b736SDag-Erling Smørgrav { 665ce71b736SDag-Erling Smørgrav warnx("fetchListHTTP(): not implemented"); 666ce71b736SDag-Erling Smørgrav return NULL; 667ce71b736SDag-Erling Smørgrav } 668