]> git.sesse.net Git - ffmpeg/blob - libavformat/http.c
* UINTX -> uintx_t INTX -> intx_t
[ffmpeg] / libavformat / http.c
1 /*
2  * HTTP protocol for ffmpeg client
3  * Copyright (c) 2000, 2001 Fabrice Bellard.
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library; if not, write to the Free Software
17  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
18  */
19 #include "avformat.h"
20 #include <unistd.h>
21 #include <ctype.h>
22 #include <sys/types.h>
23 #include <sys/socket.h>
24 #include <netinet/in.h>
25 #ifndef __BEOS__
26 # include <arpa/inet.h>
27 #else
28 # include "barpainet.h"
29 #endif
30 #include <netdb.h>
31
32
33 /* XXX: POST protocol is not completly implemented because ffmpeg use
34    only a subset of it */
35
36 //#define DEBUG
37
38 /* used for protocol handling */
39 #define BUFFER_SIZE 1024
40 #define URL_SIZE    4096
41
42 typedef struct {
43     URLContext *hd;
44     unsigned char buffer[BUFFER_SIZE], *buf_ptr, *buf_end;
45     int line_count;
46     int http_code;
47     char location[URL_SIZE];
48 } HTTPContext;
49
50 static int http_connect(URLContext *h, const char *path, const char *hoststr);
51 static int http_write(URLContext *h, uint8_t *buf, int size);
52
53
54 /* return non zero if error */
55 static int http_open(URLContext *h, const char *uri, int flags)
56 {
57     const char *path, *proxy_path;
58     char hostname[1024], hoststr[1024];
59     char path1[1024];
60     char buf[1024];
61     int port, use_proxy, err;
62     HTTPContext *s;
63     URLContext *hd = NULL;
64
65     h->is_streamed = 1;
66
67     s = av_malloc(sizeof(HTTPContext));
68     if (!s) {
69         return -ENOMEM;
70     }
71     h->priv_data = s;
72
73     proxy_path = getenv("http_proxy");
74     use_proxy = (proxy_path != NULL) && !getenv("no_proxy") && 
75         strstart(proxy_path, "http://", NULL);
76
77     /* fill the dest addr */
78  redo:
79     /* needed in any case to build the host string */
80     url_split(NULL, 0, hostname, sizeof(hostname), &port, 
81               path1, sizeof(path1), uri);
82     if (port > 0) {
83         snprintf(hoststr, sizeof(hoststr), "%s:%d", hostname, port);
84     } else {
85         pstrcpy(hoststr, sizeof(hoststr), hostname);
86     }
87
88     if (use_proxy) {
89         url_split(NULL, 0, hostname, sizeof(hostname), &port, 
90                   NULL, 0, proxy_path);
91         path = uri;
92     } else {
93         if (path1[0] == '\0')
94             path = "/";
95         else
96             path = path1;
97     }
98     if (port < 0)
99         port = 80;
100
101     snprintf(buf, sizeof(buf), "tcp://%s:%d", hostname, port);
102     err = url_open(&hd, buf, URL_RDWR);
103     if (err < 0)
104         goto fail;
105
106     s->hd = hd;
107     if (http_connect(h, path, hoststr) < 0)
108         goto fail;
109     if (s->http_code == 303 && s->location[0] != '\0') {
110         /* url moved, get next */
111         uri = s->location;
112         url_close(hd);
113         goto redo;
114     }
115     return 0;
116  fail:
117     if (hd)
118         url_close(hd);
119     av_free(s);
120     return -EIO;
121 }
122
123 static int http_getc(HTTPContext *s)
124 {
125     int len;
126     if (s->buf_ptr >= s->buf_end) {
127         len = url_read(s->hd, s->buffer, BUFFER_SIZE);
128         if (len < 0) {
129             return -EIO;
130         } else if (len == 0) {
131             return -1;
132         } else {
133             s->buf_ptr = s->buffer;
134             s->buf_end = s->buffer + len;
135         }
136     }
137     return *s->buf_ptr++;
138 }
139
140 static int process_line(HTTPContext *s, char *line, int line_count)
141 {
142     char *tag, *p;
143     
144     /* end of header */
145     if (line[0] == '\0')
146         return 0;
147
148     p = line;
149     if (line_count == 0) {
150         while (!isspace(*p) && *p != '\0')
151             p++;
152         while (isspace(*p))
153             p++;
154         s->http_code = strtol(p, NULL, 10);
155 #ifdef DEBUG
156         printf("http_code=%d\n", s->http_code);
157 #endif
158     } else {
159         while (*p != '\0' && *p != ':')
160             p++;
161         if (*p != ':') 
162             return 1;
163         
164         *p = '\0';
165         tag = line;
166         p++;
167         while (isspace(*p))
168             p++;
169         if (!strcmp(tag, "Location")) {
170             strcpy(s->location, p);
171         }
172     }
173     return 1;
174 }
175
176 static int http_connect(URLContext *h, const char *path, const char *hoststr)
177 {
178     HTTPContext *s = h->priv_data;
179     int post, err, ch;
180     char line[1024], *q;
181
182
183     /* send http header */
184     post = h->flags & URL_WRONLY;
185
186     snprintf(s->buffer, sizeof(s->buffer),
187              "%s %s HTTP/1.0\n"
188              "User-Agent: FFmpeg %s\n"
189              "Accept: */*\n"
190              "Host: %s\n"
191              "\n",
192              post ? "POST" : "GET",
193              path,
194              FFMPEG_VERSION,
195              hoststr);
196     
197     if (http_write(h, s->buffer, strlen(s->buffer)) < 0)
198         return -EIO;
199         
200     /* init input buffer */
201     s->buf_ptr = s->buffer;
202     s->buf_end = s->buffer;
203     s->line_count = 0;
204     s->location[0] = '\0';
205     if (post) {
206         sleep(1);
207         return 0;
208     }
209     
210     /* wait for header */
211     q = line;
212     for(;;) {
213         ch = http_getc(s);
214         if (ch < 0)
215             return -EIO;
216         if (ch == '\n') {
217             /* process line */
218             if (q > line && q[-1] == '\r')
219                 q--;
220             *q = '\0';
221 #ifdef DEBUG
222             printf("header='%s'\n", line);
223 #endif
224             err = process_line(s, line, s->line_count);
225             if (err < 0)
226                 return err;
227             if (err == 0)
228                 return 0;
229             s->line_count++;
230             q = line;
231         } else {
232             if ((q - line) < sizeof(line) - 1)
233                 *q++ = ch;
234         }
235     }
236 }
237
238
239 static int http_read(URLContext *h, uint8_t *buf, int size)
240 {
241     HTTPContext *s = h->priv_data;
242     int size1, len;
243
244     size1 = size;
245     while (size > 0) {
246         /* read bytes from input buffer first */
247         len = s->buf_end - s->buf_ptr;
248         if (len > 0) {
249             if (len > size)
250                 len = size;
251             memcpy(buf, s->buf_ptr, len);
252             s->buf_ptr += len;
253         } else {
254             len = url_read (s->hd, buf, size);
255             if (len < 0) {
256                 return len;
257             } else if (len == 0) {
258                 break;
259             }
260         }
261         size -= len;
262         buf += len;
263     }
264     return size1 - size;
265 }
266
267 /* used only when posting data */
268 static int http_write(URLContext *h, uint8_t *buf, int size)
269 {
270     HTTPContext *s = h->priv_data;
271     return url_write(s->hd, buf, size);
272 }
273
274 static int http_close(URLContext *h)
275 {
276     HTTPContext *s = h->priv_data;
277     url_close(s->hd);
278     av_free(s);
279     return 0;
280 }
281
282 URLProtocol http_protocol = {
283     "http",
284     http_open,
285     http_read,
286     http_write,
287     NULL, /* seek */
288     http_close,
289 };
290