Branch data Line data Source code
1 : : /*
2 : : * http_client - HTTP client
3 : : * Copyright (c) 2009, Jouni Malinen <j@w1.fi>
4 : : *
5 : : * This software may be distributed under the terms of the BSD license.
6 : : * See README for more details.
7 : : */
8 : :
9 : : #include "includes.h"
10 : : #include <fcntl.h>
11 : :
12 : : #include "common.h"
13 : : #include "eloop.h"
14 : : #include "httpread.h"
15 : : #include "http_client.h"
16 : :
17 : :
18 : : #define HTTP_CLIENT_TIMEOUT_SEC 30
19 : :
20 : :
21 : : struct http_client {
22 : : struct sockaddr_in dst;
23 : : int sd;
24 : : struct wpabuf *req;
25 : : size_t req_pos;
26 : : size_t max_response;
27 : :
28 : : void (*cb)(void *ctx, struct http_client *c,
29 : : enum http_client_event event);
30 : : void *cb_ctx;
31 : : struct httpread *hread;
32 : : struct wpabuf body;
33 : : };
34 : :
35 : :
36 : 0 : static void http_client_timeout(void *eloop_data, void *user_ctx)
37 : : {
38 : 0 : struct http_client *c = eloop_data;
39 : 0 : wpa_printf(MSG_DEBUG, "HTTP: Timeout (c=%p)", c);
40 : 0 : c->cb(c->cb_ctx, c, HTTP_CLIENT_TIMEOUT);
41 : 0 : }
42 : :
43 : :
44 : 42 : static void http_client_got_response(struct httpread *handle, void *cookie,
45 : : enum httpread_event e)
46 : : {
47 : 42 : struct http_client *c = cookie;
48 : :
49 : 42 : wpa_printf(MSG_DEBUG, "HTTP: httpread callback: handle=%p cookie=%p "
50 : : "e=%d", handle, cookie, e);
51 : :
52 : 42 : eloop_cancel_timeout(http_client_timeout, c, NULL);
53 [ + - - - ]: 42 : switch (e) {
54 : : case HTTPREAD_EVENT_FILE_READY:
55 [ + - ]: 42 : if (httpread_hdr_type_get(c->hread) == HTTPREAD_HDR_TYPE_REPLY)
56 : : {
57 : 42 : int reply_code = httpread_reply_code_get(c->hread);
58 [ + + ]: 42 : if (reply_code == 200 /* OK */) {
59 : 41 : wpa_printf(MSG_DEBUG, "HTTP: Response OK from "
60 : : "%s:%d",
61 : : inet_ntoa(c->dst.sin_addr),
62 : 41 : ntohs(c->dst.sin_port));
63 : 41 : c->cb(c->cb_ctx, c, HTTP_CLIENT_OK);
64 : : } else {
65 : 1 : wpa_printf(MSG_DEBUG, "HTTP: Error %d from "
66 : : "%s:%d", reply_code,
67 : : inet_ntoa(c->dst.sin_addr),
68 : 1 : ntohs(c->dst.sin_port));
69 : 1 : c->cb(c->cb_ctx, c, HTTP_CLIENT_INVALID_REPLY);
70 : : }
71 : : } else
72 : 0 : c->cb(c->cb_ctx, c, HTTP_CLIENT_INVALID_REPLY);
73 : 42 : break;
74 : : case HTTPREAD_EVENT_TIMEOUT:
75 : 0 : c->cb(c->cb_ctx, c, HTTP_CLIENT_TIMEOUT);
76 : 0 : break;
77 : : case HTTPREAD_EVENT_ERROR:
78 : 0 : c->cb(c->cb_ctx, c, HTTP_CLIENT_FAILED);
79 : 0 : break;
80 : : }
81 : 42 : }
82 : :
83 : :
84 : 44 : static void http_client_tx_ready(int sock, void *eloop_ctx, void *sock_ctx)
85 : : {
86 : 44 : struct http_client *c = eloop_ctx;
87 : : int res;
88 : :
89 : 88 : wpa_printf(MSG_DEBUG, "HTTP: Send client request to %s:%d (%lu of %lu "
90 : : "bytes remaining)",
91 : 44 : inet_ntoa(c->dst.sin_addr), ntohs(c->dst.sin_port),
92 : 44 : (unsigned long) wpabuf_len(c->req),
93 : 44 : (unsigned long) wpabuf_len(c->req) - c->req_pos);
94 : :
95 : 44 : res = send(c->sd, wpabuf_head_u8(c->req) + c->req_pos,
96 : 44 : wpabuf_len(c->req) - c->req_pos, 0);
97 [ + + ]: 44 : if (res < 0) {
98 : 2 : wpa_printf(MSG_DEBUG, "HTTP: Failed to send buffer: %s",
99 : 2 : strerror(errno));
100 : 2 : eloop_unregister_sock(c->sd, EVENT_TYPE_WRITE);
101 : 2 : c->cb(c->cb_ctx, c, HTTP_CLIENT_FAILED);
102 : 2 : return;
103 : : }
104 : :
105 [ - + ]: 42 : if ((size_t) res < wpabuf_len(c->req) - c->req_pos) {
106 : 0 : wpa_printf(MSG_DEBUG, "HTTP: Sent %d of %lu bytes; %lu bytes "
107 : : "remaining",
108 : 0 : res, (unsigned long) wpabuf_len(c->req),
109 : 0 : (unsigned long) wpabuf_len(c->req) - c->req_pos -
110 : : res);
111 : 0 : c->req_pos += res;
112 : 0 : return;
113 : : }
114 : :
115 : 42 : wpa_printf(MSG_DEBUG, "HTTP: Full client request sent to %s:%d",
116 : 42 : inet_ntoa(c->dst.sin_addr), ntohs(c->dst.sin_port));
117 : 42 : eloop_unregister_sock(c->sd, EVENT_TYPE_WRITE);
118 : 42 : wpabuf_free(c->req);
119 : 42 : c->req = NULL;
120 : :
121 : 42 : c->hread = httpread_create(c->sd, http_client_got_response, c,
122 : 42 : c->max_response, HTTP_CLIENT_TIMEOUT_SEC);
123 [ - + ]: 42 : if (c->hread == NULL) {
124 : 0 : c->cb(c->cb_ctx, c, HTTP_CLIENT_FAILED);
125 : 44 : return;
126 : : }
127 : : }
128 : :
129 : :
130 : 44 : struct http_client * http_client_addr(struct sockaddr_in *dst,
131 : : struct wpabuf *req, size_t max_response,
132 : : void (*cb)(void *ctx,
133 : : struct http_client *c,
134 : : enum http_client_event event),
135 : : void *cb_ctx)
136 : : {
137 : : struct http_client *c;
138 : :
139 : 44 : c = os_zalloc(sizeof(*c));
140 [ - + ]: 44 : if (c == NULL)
141 : 0 : return NULL;
142 : 44 : c->sd = -1;
143 : 44 : c->dst = *dst;
144 : 44 : c->max_response = max_response;
145 : 44 : c->cb = cb;
146 : 44 : c->cb_ctx = cb_ctx;
147 : :
148 : 44 : c->sd = socket(AF_INET, SOCK_STREAM, 0);
149 [ - + ]: 44 : if (c->sd < 0) {
150 : 0 : http_client_free(c);
151 : 0 : return NULL;
152 : : }
153 : :
154 [ - + ]: 44 : if (fcntl(c->sd, F_SETFL, O_NONBLOCK) != 0) {
155 : 0 : wpa_printf(MSG_DEBUG, "HTTP: fnctl(O_NONBLOCK) failed: %s",
156 : 0 : strerror(errno));
157 : 0 : http_client_free(c);
158 : 0 : return NULL;
159 : : }
160 : :
161 [ + - ]: 44 : if (connect(c->sd, (struct sockaddr *) dst, sizeof(*dst))) {
162 [ - + ]: 44 : if (errno != EINPROGRESS) {
163 : 0 : wpa_printf(MSG_DEBUG, "HTTP: Failed to connect: %s",
164 : 0 : strerror(errno));
165 : 0 : http_client_free(c);
166 : 0 : return NULL;
167 : : }
168 : :
169 : : /*
170 : : * Continue connecting in the background; eloop will call us
171 : : * once the connection is ready (or failed).
172 : : */
173 : : }
174 : :
175 [ - + ]: 44 : if (eloop_register_sock(c->sd, EVENT_TYPE_WRITE, http_client_tx_ready,
176 : : c, NULL)) {
177 : 0 : http_client_free(c);
178 : 0 : return NULL;
179 : : }
180 : :
181 [ - + ]: 44 : if (eloop_register_timeout(HTTP_CLIENT_TIMEOUT_SEC, 0,
182 : : http_client_timeout, c, NULL)) {
183 : 0 : http_client_free(c);
184 : 0 : return NULL;
185 : : }
186 : :
187 : 44 : c->req = req;
188 : :
189 : 44 : return c;
190 : : }
191 : :
192 : :
193 : 28 : char * http_client_url_parse(const char *url, struct sockaddr_in *dst,
194 : : char **ret_path)
195 : : {
196 : : char *u, *addr, *port, *path;
197 : :
198 : 28 : u = os_strdup(url);
199 [ - + ]: 28 : if (u == NULL)
200 : 0 : return NULL;
201 : :
202 : 28 : os_memset(dst, 0, sizeof(*dst));
203 : 28 : dst->sin_family = AF_INET;
204 : 28 : addr = u + 7;
205 : 28 : path = os_strchr(addr, '/');
206 : 28 : port = os_strchr(addr, ':');
207 [ - + ]: 28 : if (path == NULL) {
208 : 0 : path = "/";
209 : : } else {
210 : 28 : *path = '\0'; /* temporary nul termination for address */
211 [ - + ]: 28 : if (port > path)
212 : 0 : port = NULL;
213 : : }
214 [ + - ]: 28 : if (port)
215 : 28 : *port++ = '\0';
216 : :
217 [ - + ]: 28 : if (inet_aton(addr, &dst->sin_addr) == 0) {
218 : : /* TODO: name lookup */
219 : 0 : wpa_printf(MSG_DEBUG, "HTTP: Unsupported address in URL '%s' "
220 : : "(addr='%s' port='%s')",
221 : : url, addr, port);
222 : 0 : os_free(u);
223 : 0 : return NULL;
224 : : }
225 : :
226 [ + - ]: 28 : if (port)
227 : 28 : dst->sin_port = htons(atoi(port));
228 : : else
229 : 0 : dst->sin_port = htons(80);
230 : :
231 [ + - ]: 28 : if (*path == '\0') {
232 : : /* remove temporary nul termination for address */
233 : 28 : *path = '/';
234 : : }
235 : :
236 : 28 : *ret_path = path;
237 : :
238 : 28 : return u;
239 : : }
240 : :
241 : :
242 : 2 : struct http_client * http_client_url(const char *url,
243 : : struct wpabuf *req, size_t max_response,
244 : : void (*cb)(void *ctx,
245 : : struct http_client *c,
246 : : enum http_client_event event),
247 : : void *cb_ctx)
248 : : {
249 : : struct sockaddr_in dst;
250 : : struct http_client *c;
251 : : char *u, *path;
252 : 2 : struct wpabuf *req_buf = NULL;
253 : :
254 [ - + ]: 2 : if (os_strncmp(url, "http://", 7) != 0)
255 : 0 : return NULL;
256 : 2 : u = http_client_url_parse(url, &dst, &path);
257 [ - + ]: 2 : if (u == NULL)
258 : 0 : return NULL;
259 : :
260 [ + - ]: 2 : if (req == NULL) {
261 : 2 : req_buf = wpabuf_alloc(os_strlen(url) + 1000);
262 [ - + ]: 2 : if (req_buf == NULL) {
263 : 0 : os_free(u);
264 : 0 : return NULL;
265 : : }
266 : 2 : req = req_buf;
267 : 2 : wpabuf_printf(req,
268 : : "GET %s HTTP/1.1\r\n"
269 : : "Cache-Control: no-cache\r\n"
270 : : "Pragma: no-cache\r\n"
271 : : "Accept: text/xml, application/xml\r\n"
272 : : "User-Agent: wpa_supplicant\r\n"
273 : : "Host: %s:%d\r\n"
274 : : "\r\n",
275 : : path, inet_ntoa(dst.sin_addr),
276 : 2 : ntohs(dst.sin_port));
277 : : }
278 : 2 : os_free(u);
279 : :
280 : 2 : c = http_client_addr(&dst, req, max_response, cb, cb_ctx);
281 [ - + ]: 2 : if (c == NULL) {
282 : 0 : wpabuf_free(req_buf);
283 : 0 : return NULL;
284 : : }
285 : :
286 : 2 : return c;
287 : : }
288 : :
289 : :
290 : 52 : void http_client_free(struct http_client *c)
291 : : {
292 [ + + ]: 52 : if (c == NULL)
293 : 52 : return;
294 : 44 : httpread_destroy(c->hread);
295 : 44 : wpabuf_free(c->req);
296 [ + - ]: 44 : if (c->sd >= 0) {
297 : 44 : eloop_unregister_sock(c->sd, EVENT_TYPE_WRITE);
298 : 44 : close(c->sd);
299 : : }
300 : 44 : eloop_cancel_timeout(http_client_timeout, c, NULL);
301 : 44 : os_free(c);
302 : : }
303 : :
304 : :
305 : 8 : struct wpabuf * http_client_get_body(struct http_client *c)
306 : : {
307 [ - + ]: 8 : if (c->hread == NULL)
308 : 0 : return NULL;
309 : 8 : wpabuf_set(&c->body, httpread_data_get(c->hread),
310 : 8 : httpread_length_get(c->hread));
311 : 8 : return &c->body;
312 : : }
313 : :
314 : :
315 : 2 : char * http_client_get_hdr_line(struct http_client *c, const char *tag)
316 : : {
317 [ - + ]: 2 : if (c->hread == NULL)
318 : 0 : return NULL;
319 : 2 : return httpread_hdr_line_get(c->hread, tag);
320 : : }
321 : :
322 : :
323 : 6 : char * http_link_update(char *url, const char *base)
324 : : {
325 : : char *n;
326 : : size_t len;
327 : : const char *pos;
328 : :
329 : : /* RFC 2396, Chapter 5.2 */
330 : : /* TODO: consider adding all cases described in RFC 2396 */
331 : :
332 [ - + ]: 6 : if (url == NULL)
333 : 0 : return NULL;
334 : :
335 [ - + ]: 6 : if (os_strncmp(url, "http://", 7) == 0)
336 : 0 : return url; /* absolute link */
337 : :
338 [ - + ]: 6 : if (os_strncmp(base, "http://", 7) != 0)
339 : 0 : return url; /* unable to handle base URL */
340 : :
341 : 6 : len = os_strlen(url) + 1 + os_strlen(base) + 1;
342 : 6 : n = os_malloc(len);
343 [ - + ]: 6 : if (n == NULL)
344 : 0 : return url; /* failed */
345 : :
346 [ - + ]: 6 : if (url[0] == '/') {
347 : 0 : pos = os_strchr(base + 7, '/');
348 [ # # ]: 0 : if (pos == NULL) {
349 : 0 : os_snprintf(n, len, "%s%s", base, url);
350 : : } else {
351 : 0 : os_memcpy(n, base, pos - base);
352 : 0 : os_memcpy(n + (pos - base), url, os_strlen(url) + 1);
353 : : }
354 : : } else {
355 : 6 : pos = os_strrchr(base + 7, '/');
356 [ - + ]: 6 : if (pos == NULL) {
357 : 0 : os_snprintf(n, len, "%s/%s", base, url);
358 : : } else {
359 : 6 : os_memcpy(n, base, pos - base + 1);
360 : 6 : os_memcpy(n + (pos - base) + 1, url, os_strlen(url) +
361 : : 1);
362 : : }
363 : : }
364 : :
365 : 6 : os_free(url);
366 : :
367 : 6 : return n;
368 : : }
|