LCOV - code coverage report
Current view: top level - src/wps - http_client.c (source / functions) Hit Total Coverage
Test: wpa_supplicant hwsim test run 1388240082 Lines: 124 180 68.9 %
Date: 2013-12-28 Functions: 9 10 90.0 %
Branches: 38 74 51.4 %

           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                 :         26 : static void http_client_got_response(struct httpread *handle, void *cookie,
      45                 :            :                                      enum httpread_event e)
      46                 :            : {
      47                 :         26 :         struct http_client *c = cookie;
      48                 :            : 
      49                 :         26 :         wpa_printf(MSG_DEBUG, "HTTP: httpread callback: handle=%p cookie=%p "
      50                 :            :                    "e=%d", handle, cookie, e);
      51                 :            : 
      52                 :         26 :         eloop_cancel_timeout(http_client_timeout, c, NULL);
      53   [ +  -  -  - ]:         26 :         switch (e) {
      54                 :            :         case HTTPREAD_EVENT_FILE_READY:
      55         [ +  - ]:         26 :                 if (httpread_hdr_type_get(c->hread) == HTTPREAD_HDR_TYPE_REPLY)
      56                 :            :                 {
      57                 :         26 :                         int reply_code = httpread_reply_code_get(c->hread);
      58         [ +  + ]:         26 :                         if (reply_code == 200 /* OK */) {
      59                 :         25 :                                 wpa_printf(MSG_DEBUG, "HTTP: Response OK from "
      60                 :            :                                            "%s:%d",
      61                 :            :                                            inet_ntoa(c->dst.sin_addr),
      62                 :         25 :                                            ntohs(c->dst.sin_port));
      63                 :         25 :                                 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                 :         26 :                 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                 :         26 : }
      82                 :            : 
      83                 :            : 
      84                 :         28 : static void http_client_tx_ready(int sock, void *eloop_ctx, void *sock_ctx)
      85                 :            : {
      86                 :         28 :         struct http_client *c = eloop_ctx;
      87                 :            :         int res;
      88                 :            : 
      89                 :         56 :         wpa_printf(MSG_DEBUG, "HTTP: Send client request to %s:%d (%lu of %lu "
      90                 :            :                    "bytes remaining)",
      91                 :         28 :                    inet_ntoa(c->dst.sin_addr), ntohs(c->dst.sin_port),
      92                 :         28 :                    (unsigned long) wpabuf_len(c->req),
      93                 :         28 :                    (unsigned long) wpabuf_len(c->req) - c->req_pos);
      94                 :            : 
      95                 :         28 :         res = send(c->sd, wpabuf_head_u8(c->req) + c->req_pos,
      96                 :         28 :                    wpabuf_len(c->req) - c->req_pos, 0);
      97         [ +  + ]:         28 :         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         [ -  + ]:         26 :         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                 :         26 :         wpa_printf(MSG_DEBUG, "HTTP: Full client request sent to %s:%d",
     116                 :         26 :                    inet_ntoa(c->dst.sin_addr), ntohs(c->dst.sin_port));
     117                 :         26 :         eloop_unregister_sock(c->sd, EVENT_TYPE_WRITE);
     118                 :         26 :         wpabuf_free(c->req);
     119                 :         26 :         c->req = NULL;
     120                 :            : 
     121                 :         26 :         c->hread = httpread_create(c->sd, http_client_got_response, c,
     122                 :         26 :                                    c->max_response, HTTP_CLIENT_TIMEOUT_SEC);
     123         [ -  + ]:         26 :         if (c->hread == NULL) {
     124                 :          0 :                 c->cb(c->cb_ctx, c, HTTP_CLIENT_FAILED);
     125                 :         28 :                 return;
     126                 :            :         }
     127                 :            : }
     128                 :            : 
     129                 :            : 
     130                 :         28 : 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                 :         28 :         c = os_zalloc(sizeof(*c));
     140         [ -  + ]:         28 :         if (c == NULL)
     141                 :          0 :                 return NULL;
     142                 :         28 :         c->sd = -1;
     143                 :         28 :         c->dst = *dst;
     144                 :         28 :         c->max_response = max_response;
     145                 :         28 :         c->cb = cb;
     146                 :         28 :         c->cb_ctx = cb_ctx;
     147                 :            : 
     148                 :         28 :         c->sd = socket(AF_INET, SOCK_STREAM, 0);
     149         [ -  + ]:         28 :         if (c->sd < 0) {
     150                 :          0 :                 http_client_free(c);
     151                 :          0 :                 return NULL;
     152                 :            :         }
     153                 :            : 
     154         [ -  + ]:         28 :         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         [ +  - ]:         28 :         if (connect(c->sd, (struct sockaddr *) dst, sizeof(*dst))) {
     162         [ -  + ]:         28 :                 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         [ -  + ]:         28 :         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         [ -  + ]:         28 :         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                 :         28 :         c->req = req;
     188                 :            : 
     189                 :         28 :         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                 :         36 : void http_client_free(struct http_client *c)
     291                 :            : {
     292         [ +  + ]:         36 :         if (c == NULL)
     293                 :         36 :                 return;
     294                 :         28 :         httpread_destroy(c->hread);
     295                 :         28 :         wpabuf_free(c->req);
     296         [ +  - ]:         28 :         if (c->sd >= 0) {
     297                 :         28 :                 eloop_unregister_sock(c->sd, EVENT_TYPE_WRITE);
     298                 :         28 :                 close(c->sd);
     299                 :            :         }
     300                 :         28 :         eloop_cancel_timeout(http_client_timeout, c, NULL);
     301                 :         28 :         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                 :            : }

Generated by: LCOV version 1.9