2007-09-08 22:41:00 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* netutils/webclient/webclient.c
|
2007-08-26 23:12:17 +00:00
|
|
|
* Implementation of the HTTP client.
|
|
|
|
*
|
2009-03-26 01:31:43 +00:00
|
|
|
* Copyright (C) 2007, 2009 Gregory Nutt. All rights reserved.
|
2007-09-08 22:41:00 +00:00
|
|
|
* Author: Gregory Nutt <spudmonkey@racsa.co.cr>
|
2007-08-26 23:12:17 +00:00
|
|
|
*
|
2007-09-08 22:41:00 +00:00
|
|
|
* Based on uIP which also has a BSD style license:
|
|
|
|
*
|
|
|
|
* Author: Adam Dunkels <adam@dunkels.com>
|
|
|
|
* Copyright (c) 2002, Adam Dunkels.
|
|
|
|
* All rights reserved.
|
2007-08-26 23:12:17 +00:00
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above
|
|
|
|
* copyright notice, this list of conditions and the following
|
|
|
|
* disclaimer in the documentation and/or other materials provided
|
|
|
|
* with the distribution.
|
|
|
|
* 3. The name of the author may not be used to endorse or promote
|
|
|
|
* products derived from this software without specific prior
|
|
|
|
* written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
|
|
|
|
* OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
|
|
|
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
|
|
|
|
* GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
2007-09-08 22:41:00 +00:00
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/* This example shows a HTTP client that is able to download web pages
|
|
|
|
* and files from web servers. It requires a number of callback
|
|
|
|
* functions to be implemented by the module that utilizes the code:
|
2009-03-26 01:31:43 +00:00
|
|
|
* webclient_datahandler().
|
2007-08-26 23:12:17 +00:00
|
|
|
*/
|
|
|
|
|
2007-09-08 22:41:00 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
2009-03-26 02:38:07 +00:00
|
|
|
#ifndef CONFIG_NETUTILS_WEBCLIENT_HOST
|
|
|
|
|
|
|
|
# include <nuttx/config.h>
|
|
|
|
# include <nuttx/compiler.h>
|
|
|
|
# include <debug.h>
|
|
|
|
|
|
|
|
# include <net/uip/uip.h>
|
|
|
|
# include <net/uip/uip-lib.h>
|
|
|
|
# include <net/uip/resolv.h>
|
|
|
|
|
|
|
|
#endif
|
2009-03-26 01:31:43 +00:00
|
|
|
|
2007-08-26 23:12:17 +00:00
|
|
|
#include <sys/types.h>
|
2007-09-02 21:58:35 +00:00
|
|
|
#include <sys/socket.h>
|
2009-03-26 01:31:43 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <net/uip/webclient.h>
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2007-11-22 12:48:40 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* Definitions
|
|
|
|
****************************************************************************/
|
|
|
|
|
2007-08-26 23:12:17 +00:00
|
|
|
#define WEBCLIENT_TIMEOUT 100
|
|
|
|
|
|
|
|
#define WEBCLIENT_STATE_STATUSLINE 0
|
|
|
|
#define WEBCLIENT_STATE_HEADERS 1
|
|
|
|
#define WEBCLIENT_STATE_DATA 2
|
|
|
|
#define WEBCLIENT_STATE_CLOSE 3
|
|
|
|
|
|
|
|
#define HTTPFLAG_NONE 0
|
|
|
|
#define HTTPFLAG_OK 1
|
|
|
|
#define HTTPFLAG_MOVED 2
|
|
|
|
#define HTTPFLAG_ERROR 3
|
|
|
|
|
|
|
|
#define ISO_nl 0x0a
|
|
|
|
#define ISO_cr 0x0d
|
|
|
|
#define ISO_space 0x20
|
|
|
|
|
2007-11-22 12:48:40 +00:00
|
|
|
/****************************************************************************
|
2009-03-26 01:31:43 +00:00
|
|
|
* Private Types
|
2007-11-22 12:48:40 +00:00
|
|
|
****************************************************************************/
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
struct wget_s
|
|
|
|
{
|
|
|
|
ubyte state;
|
|
|
|
ubyte httpflag;
|
|
|
|
|
|
|
|
/* These describe the just-received buffer of data */
|
|
|
|
|
|
|
|
FAR char *buffer; /* user-provided buffer */
|
|
|
|
int buflen; /* Length of the user provided buffer */
|
|
|
|
int offset; /* Offset to the beginning of interesting data */
|
|
|
|
int datend; /* Offset+1 to the last valid byte of data in the buffer */
|
|
|
|
|
|
|
|
/* Buffer HTTP header data and parse line at a time */
|
|
|
|
|
|
|
|
char line[CONFIG_NETUTILS_WEBCLIENT_MAXHTTPLINE];
|
|
|
|
int ndx;
|
|
|
|
|
|
|
|
#ifdef CONFIG_WEBCLIENT_GETMIMETYPE
|
|
|
|
char mimetype[CONFIG_NETUTILS_WEBCLIENT_MAXMIMESIZE];
|
|
|
|
#endif
|
|
|
|
#ifdef CONFIG_WEBCLIENT_GETHOST
|
|
|
|
char host[CONFIG_NETUTILS_WEBCLIENT_MAXHOSTNAME];
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Data
|
|
|
|
****************************************************************************/
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2007-11-22 12:48:40 +00:00
|
|
|
static const char g_http10[] = "HTTP/1.0";
|
|
|
|
static const char g_http11[] = "HTTP/1.1";
|
2009-03-26 01:31:43 +00:00
|
|
|
#ifdef CONFIG_WEBCLIENT_GETMIMETYPE
|
2007-11-22 12:48:40 +00:00
|
|
|
static const char g_httpcontenttype[] = "content-type: ";
|
2009-03-26 01:31:43 +00:00
|
|
|
#endif
|
2007-11-22 12:48:40 +00:00
|
|
|
static const char g_httphost[] = "host: ";
|
2009-03-26 01:31:43 +00:00
|
|
|
#ifdef CONFIG_WEBCLIENT_GETHOST
|
2007-11-22 12:48:40 +00:00
|
|
|
static const char g_httplocation[] = "location: ";
|
2009-03-26 01:31:43 +00:00
|
|
|
#endif
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2007-11-22 12:48:40 +00:00
|
|
|
static const char g_httpget[] = "GET ";
|
2009-03-26 01:31:43 +00:00
|
|
|
#ifdef CONFIG_WEBCLIENT_GETHOST
|
2007-11-22 12:48:40 +00:00
|
|
|
static const char g_httphttp[] = "http://";
|
2009-03-26 01:31:43 +00:00
|
|
|
#endif
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2007-11-22 12:48:40 +00:00
|
|
|
static const char g_httpuseragentfields[] =
|
|
|
|
"Connection: close\r\n"
|
|
|
|
"User-Agent: uIP/1.0 (; http://www.sics.se/~adam/uip/)\r\n\r\n";
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2007-11-22 12:48:40 +00:00
|
|
|
static const char g_http200[] = "200 ";
|
|
|
|
static const char g_http301[] = "301 ";
|
|
|
|
static const char g_http302[] = "302 ";
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2007-11-22 12:48:40 +00:00
|
|
|
static const char g_httpcrnl[] = "\r\n";
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: wget_strcpy
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static char *wget_strcpy(char *dest, const char *src)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
int len = strlen(src);
|
|
|
|
memcpy(dest, src, len);
|
2007-08-26 23:12:17 +00:00
|
|
|
return dest + len;
|
|
|
|
}
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: wget_parsestatus
|
|
|
|
****************************************************************************/
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
static inline int wget_parsestatus(struct wget_s *ws)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
int offset;
|
|
|
|
int ndx;
|
|
|
|
char *dest;
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
offset = ws->offset;
|
|
|
|
ndx = ws->ndx;
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
while (offset < ws->datend)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
ws->line[ndx] = ws->buffer[offset];
|
|
|
|
if (ws->line[ndx] == ISO_nl)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
ws->line[ndx] = '\0';
|
|
|
|
if ((strncmp(ws->line, g_http10, strlen(g_http10)) == 0) ||
|
|
|
|
(strncmp(ws->line, g_http11, strlen(g_http11)) == 0))
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
dest = &(ws->line[9]);
|
|
|
|
ws->httpflag = HTTPFLAG_NONE;
|
|
|
|
|
|
|
|
/* Check for 200 OK */
|
|
|
|
|
|
|
|
if (strncmp(dest, g_http200, strlen(g_http200)) == 0)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
ws->httpflag = HTTPFLAG_OK;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
/* Check for 301 Moved permanently or 302 Found. Location: header line
|
|
|
|
* will contain the new location.
|
|
|
|
*/
|
|
|
|
|
|
|
|
else if (strncmp(dest, g_http301, strlen(g_http301)) == 0 ||
|
|
|
|
strncmp(dest, g_http302, strlen(g_http302)) == 0)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
|
|
|
|
ws->httpflag = HTTPFLAG_MOVED;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
return - ECONNABORTED;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
/* We're done parsing the status line, so start parsing the HTTP headers. */
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
ws->state = WEBCLIENT_STATE_HEADERS;
|
2007-08-26 23:12:17 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
offset++;
|
|
|
|
ndx++;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
ws->offset = offset;
|
|
|
|
ws->ndx = ndx;
|
|
|
|
return OK;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: wget_parsestatus
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static inline int wget_parseheaders(struct wget_s *ws)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
int offset;
|
|
|
|
int ndx;
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
offset = ws->offset;
|
|
|
|
ndx = ws->ndx;
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
while (offset < ws->datend)
|
|
|
|
{
|
|
|
|
ws->line[ndx] = ws->buffer[offset];
|
|
|
|
if (ws->line[ndx] == ISO_nl)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
/* We have an entire HTTP header line in s.line, so
|
2007-08-26 23:12:17 +00:00
|
|
|
* we parse it.
|
|
|
|
*/
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
if (ws->line[0] == ISO_cr)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
|
|
|
/* This was the last header line (i.e., and empty "\r\n"), so
|
|
|
|
* we are done with the headers and proceed with the actual
|
|
|
|
* data.
|
|
|
|
*/
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
ws->state = WEBCLIENT_STATE_DATA;
|
|
|
|
return OK;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
ws->line[ndx] = '\0';
|
2007-08-26 23:12:17 +00:00
|
|
|
|
|
|
|
/* Check for specific HTTP header fields. */
|
2009-03-26 01:31:43 +00:00
|
|
|
|
|
|
|
#ifdef CONFIG_WEBCLIENT_GETMIMETYPE
|
|
|
|
if (strncasecmp(ws->line, g_httpcontenttype, strlen(g_httpcontenttype)) == 0)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
|
|
|
/* Found Content-type field. */
|
2009-03-26 02:38:07 +00:00
|
|
|
|
|
|
|
char *dest = strchr(ws->line, ';');
|
2009-03-26 01:31:43 +00:00
|
|
|
if (dest != NULL)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
*dest = 0;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
2009-03-26 01:31:43 +00:00
|
|
|
strncpy(ws->mimetype, ws->line + strlen(g_httpcontenttype) - 1, sizeof(ws->mimetype));
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
2009-03-26 01:31:43 +00:00
|
|
|
# ifdef CONFIG_WEBCLIENT_GETHOST
|
|
|
|
else
|
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
#ifdef CONFIG_WEBCLIENT_GETHOST
|
|
|
|
if (strncasecmp(ws->line, g_httplocation, strlen(g_httplocation)) == 0)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 02:38:07 +00:00
|
|
|
char *dest = ws->line + strlen(g_httplocation) - 1;
|
2007-08-26 23:12:17 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
if (strncmp(dest, g_httphttp, strlen(g_httphttp)) == 0)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
dest += 7;
|
|
|
|
for(i = 0; i < ws->ndx - 7; ++i)
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
if (*dest == 0 || *dest == '/' || *dest == ' ' || *dest == ':')
|
2007-08-26 23:12:17 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
ws->host[i] = 0;
|
2007-08-26 23:12:17 +00:00
|
|
|
break;
|
|
|
|
}
|
2009-03-26 01:31:43 +00:00
|
|
|
ws->host[i] = *dest;
|
|
|
|
++dest;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
|
|
|
}
|
2009-03-26 01:31:43 +00:00
|
|
|
strncpy(ws->file, dest, sizeof(ws->file));
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
2009-03-26 01:31:43 +00:00
|
|
|
#endif
|
2007-08-26 23:12:17 +00:00
|
|
|
|
|
|
|
/* We're done parsing, so we reset the pointer and start the
|
|
|
|
* next line.
|
|
|
|
*/
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
ndx = 0;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
ndx++;
|
|
|
|
offset++;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
ws->offset = offset;
|
|
|
|
ws->ndx = ndx;
|
|
|
|
return OK;
|
2007-08-26 23:12:17 +00:00
|
|
|
}
|
2007-11-22 12:48:40 +00:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
2008-12-12 18:01:25 +00:00
|
|
|
/****************************************************************************
|
2009-03-26 01:31:43 +00:00
|
|
|
* Name: wget
|
2008-12-12 18:01:25 +00:00
|
|
|
****************************************************************************/
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
int wget(FAR const char *host, uint16 port, FAR const char *file,
|
|
|
|
FAR char *buffer, int buflen, wget_callback_t callback)
|
2007-11-22 12:48:40 +00:00
|
|
|
{
|
2009-03-26 02:38:07 +00:00
|
|
|
#ifndef CONFIG_NETUTILS_WEBCLIENT_HOST
|
2007-11-22 12:48:40 +00:00
|
|
|
static uip_ipaddr_t addr;
|
2009-03-26 02:38:07 +00:00
|
|
|
#endif
|
2007-11-22 12:48:40 +00:00
|
|
|
struct sockaddr_in server;
|
2009-03-26 01:31:43 +00:00
|
|
|
struct wget_s ws;
|
|
|
|
char *dest;
|
2007-11-22 12:48:40 +00:00
|
|
|
int sockfd;
|
2009-03-26 01:31:43 +00:00
|
|
|
int len;
|
|
|
|
int ret = OK;
|
2007-11-22 12:48:40 +00:00
|
|
|
|
|
|
|
/* First check if the host is an IP address. */
|
|
|
|
|
2009-03-26 02:38:07 +00:00
|
|
|
#ifndef CONFIG_NETUTILS_WEBCLIENT_HOST
|
2008-12-12 18:01:25 +00:00
|
|
|
if (!uiplib_ipaddrconv(host, (unsigned char *)addr))
|
2007-11-22 12:48:40 +00:00
|
|
|
{
|
2008-12-12 18:01:25 +00:00
|
|
|
/* 'host' does not point to a avalid address string. Try to resolve
|
|
|
|
* the host name to an IP address.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (resolv_query(host, &server) < 0)
|
2007-11-22 12:48:40 +00:00
|
|
|
{
|
|
|
|
return ERROR;
|
|
|
|
}
|
2008-12-12 18:01:25 +00:00
|
|
|
|
|
|
|
/* Save the host address */
|
|
|
|
|
|
|
|
addr = server.sin_addr.s_addr;
|
2007-11-22 12:48:40 +00:00
|
|
|
}
|
2009-03-26 02:38:07 +00:00
|
|
|
#endif
|
2007-11-22 12:48:40 +00:00
|
|
|
|
|
|
|
/* Create a socket */
|
|
|
|
|
|
|
|
sockfd = socket(AF_INET, SOCK_STREAM, 0);
|
|
|
|
if (sockfd < 0)
|
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
/* socket failed. It will set the errno appropriately */
|
|
|
|
|
|
|
|
ndbg("socket failed: %d\n", errno);
|
2007-11-22 12:48:40 +00:00
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Connect to server. First we have to set some fields in the
|
2008-12-12 18:01:25 +00:00
|
|
|
* 'server' address structure. The system will assign me an arbitrary
|
2007-11-22 12:48:40 +00:00
|
|
|
* local port that is not in use.
|
|
|
|
*/
|
|
|
|
|
|
|
|
server.sin_family = AF_INET;
|
|
|
|
memcpy(&server.sin_addr.s_addr, &host, sizeof(in_addr_t));
|
2009-03-26 01:31:43 +00:00
|
|
|
server.sin_port = htons(port);
|
2007-11-22 12:48:40 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
ret = connect(sockfd, (struct sockaddr *)&server, sizeof(struct sockaddr_in));
|
|
|
|
if (ret < 0)
|
2007-11-22 12:48:40 +00:00
|
|
|
{
|
2009-03-26 01:31:43 +00:00
|
|
|
ndbg("connect failed: %d\n", errno);
|
|
|
|
goto errout;
|
2007-11-22 12:48:40 +00:00
|
|
|
}
|
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
/* Send the GET request */
|
|
|
|
|
|
|
|
dest = (char *)buffer;
|
|
|
|
dest = wget_strcpy(dest, g_httpget);
|
|
|
|
dest = wget_strcpy(dest, file);
|
|
|
|
*dest++ = ISO_space;
|
|
|
|
dest = wget_strcpy(dest, g_http10);
|
|
|
|
dest = wget_strcpy(dest, g_httpcrnl);
|
|
|
|
dest = wget_strcpy(dest, g_httphost);
|
|
|
|
dest = wget_strcpy(dest, host);
|
|
|
|
dest = wget_strcpy(dest, g_httpcrnl);
|
|
|
|
dest = wget_strcpy(dest, g_httpuseragentfields);
|
|
|
|
len = dest - buffer;
|
|
|
|
|
|
|
|
ret = send(sockfd, buffer, len, 0);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
ndbg("send failed: %d\n", errno);
|
|
|
|
goto errout;
|
|
|
|
}
|
2007-11-22 12:48:40 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
/* Now get the response */
|
2007-11-22 12:48:40 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
memset(&ws, 0, sizeof(struct wget_s));
|
|
|
|
ws.state = WEBCLIENT_STATE_STATUSLINE;
|
|
|
|
ws.buffer = buffer;
|
|
|
|
ws.buflen = buflen;
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
ws.datend = recv(sockfd, ws.buffer, ws.buflen, 0);
|
|
|
|
if (ws.datend < 0)
|
|
|
|
{
|
|
|
|
ndbg("recv failed: %d\n", errno);
|
|
|
|
ret = ws.datend;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
else if (ret == 0)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ws.offset = 0;
|
|
|
|
if (ws.state == WEBCLIENT_STATE_STATUSLINE)
|
|
|
|
{
|
|
|
|
ret = wget_parsestatus(&ws);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
goto errout_with_errno;
|
|
|
|
}
|
|
|
|
}
|
2007-11-22 12:48:40 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
if (ws.state == WEBCLIENT_STATE_HEADERS)
|
|
|
|
{
|
|
|
|
ret = wget_parseheaders(&ws);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
goto errout_with_errno;
|
|
|
|
}
|
|
|
|
}
|
2007-11-22 12:48:40 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
/* Let the client decide what to do with the received file */
|
2007-11-22 12:48:40 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
if (ws.state == WEBCLIENT_STATE_DATA && ws.httpflag != HTTPFLAG_MOVED)
|
|
|
|
{
|
|
|
|
callback(&ws.buffer, ws.offset, ws.datend, &buflen);
|
|
|
|
}
|
|
|
|
}
|
2007-11-22 12:48:40 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
close(sockfd);
|
|
|
|
return OK;
|
2007-11-22 12:48:40 +00:00
|
|
|
|
2009-03-26 01:31:43 +00:00
|
|
|
errout_with_errno:
|
|
|
|
errno = -ret;
|
|
|
|
errout:
|
|
|
|
close(sockfd);
|
|
|
|
return ERROR;
|
|
|
|
}
|