2006-01-22 05:08:50 +00:00
/*
2009-01-27 22:34:36 +00:00
* Copyright ( c ) 2002 - 2007 Niels Provos < provos @ citi . umich . edu >
2012-02-10 17:29:53 -05:00
* Copyright ( c ) 2007 - 2012 Niels Provos and Nick Mathewson
2006-01-22 05:08:50 +00:00
*
2006-02-27 02:27:37 +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 .
2006-01-22 05:08:50 +00:00
*/
2010-07-07 16:45:03 -04:00
# include "event2/event-config.h"
2011-01-02 08:43:45 -07:00
# include "evconfig-private.h"
2009-11-06 21:46:57 +00:00
2016-01-31 11:31:00 +00:00
# define member_size(type, member) sizeof(((type *)0)->member)
2012-02-29 15:07:31 -05:00
# ifdef EVENT__HAVE_SYS_PARAM_H
2007-11-07 21:01:26 +00:00
# include <sys/param.h>
# endif
2012-02-29 15:07:31 -05:00
# ifdef EVENT__HAVE_SYS_TYPES_H
2007-11-07 21:01:26 +00:00
# include <sys/types.h>
# endif
2015-08-26 10:06:09 +02:00
# ifdef HAVE_SYS_IOCCOM_H
# include <sys/ioccom.h>
# endif
# ifdef EVENT__HAVE_SYS_RESOURCE_H
# include <sys/resource.h>
# endif
2012-02-29 15:07:31 -05:00
# ifdef EVENT__HAVE_SYS_TIME_H
2006-01-22 05:08:50 +00:00
# include <sys/time.h>
# endif
2015-08-26 10:06:09 +02:00
# ifdef EVENT__HAVE_SYS_WAIT_H
# include <sys/wait.h>
2006-01-22 05:08:50 +00:00
# endif
2006-11-22 01:21:10 +00:00
2011-05-25 19:50:56 -04:00
# ifndef _WIN32
2006-01-22 05:08:50 +00:00
# include <sys/socket.h>
# include <sys/stat.h>
2019-05-12 15:19:31 +03:00
# else /* _WIN32 */
2010-03-08 13:46:48 -05:00
# include <winsock2.h>
2009-11-16 22:25:46 +00:00
# include <ws2tcpip.h>
2019-05-12 15:19:31 +03:00
# endif /* _WIN32 */
# ifdef EVENT__HAVE_SYS_UN_H
# include <sys/un.h>
# endif
# ifdef EVENT__HAVE_AFUNIX_H
# include <afunix.h>
2006-11-22 01:21:10 +00:00
# endif
2006-01-22 05:08:50 +00:00
# include <sys/queue.h>
2012-02-29 15:07:31 -05:00
# ifdef EVENT__HAVE_NETINET_IN_H
2006-01-22 05:08:50 +00:00
# include <netinet/in.h>
2009-11-03 20:40:48 +00:00
# endif
2012-02-29 15:07:31 -05:00
# ifdef EVENT__HAVE_ARPA_INET_H
2011-02-13 01:22:25 -05:00
# include <arpa/inet.h>
# endif
2012-02-29 15:07:31 -05:00
# ifdef EVENT__HAVE_NETDB_H
2006-01-22 05:08:50 +00:00
# include <netdb.h>
2006-11-22 01:21:10 +00:00
# endif
2006-01-22 05:08:50 +00:00
2011-05-25 19:50:56 -04:00
# ifdef _WIN32
2007-11-07 21:01:26 +00:00
# include <winsock2.h>
# endif
2024-02-18 08:56:46 -05:00
# include <ctype.h>
2006-01-22 05:08:50 +00:00
# include <errno.h>
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
2011-05-25 19:50:56 -04:00
# ifndef _WIN32
2006-01-22 05:08:50 +00:00
# include <syslog.h>
2019-05-12 15:19:31 +03:00
# endif /* !_WIN32 */
2006-01-22 05:08:50 +00:00
# include <signal.h>
2012-02-29 15:07:31 -05:00
# ifdef EVENT__HAVE_UNISTD_H
2006-01-22 05:08:50 +00:00
# include <unistd.h>
2007-11-07 21:01:26 +00:00
# endif
2012-02-29 15:07:31 -05:00
# ifdef EVENT__HAVE_FCNTL_H
2006-01-22 05:08:50 +00:00
# include <fcntl.h>
2007-11-07 21:01:26 +00:00
# endif
2006-01-22 05:08:50 +00:00
# undef timeout_pending
# undef timeout_initialized
2007-01-03 07:11:17 +00:00
# include "strlcpy-internal.h"
2008-05-08 06:15:04 +00:00
# include "event2/http.h"
2008-05-08 15:55:09 +00:00
# include "event2/event.h"
# include "event2/buffer.h"
# include "event2/bufferevent.h"
2008-05-08 06:15:04 +00:00
# include "event2/http_struct.h"
2009-01-02 04:56:56 +00:00
# include "event2/http_compat.h"
2008-05-08 15:55:09 +00:00
# include "event2/util.h"
2022-09-12 22:16:56 +03:00
# include "event2/ws.h"
2009-12-30 00:41:03 -05:00
# include "event2/listener.h"
2009-01-13 20:26:37 +00:00
# include "log-internal.h"
2009-01-29 15:09:24 +00:00
# include "util-internal.h"
2006-06-10 22:28:21 +00:00
# include "http-internal.h"
2007-11-25 17:14:19 +00:00
# include "mm-internal.h"
2011-03-07 21:55:47 -05:00
# include "bufferevent-internal.h"
2006-01-22 05:08:50 +00:00
2012-02-29 15:07:31 -05:00
# ifndef EVENT__HAVE_GETNAMEINFO
2008-06-24 22:38:37 +00:00
# define NI_MAXSERV 32
# define NI_MAXHOST 1025
2009-11-16 22:25:46 +00:00
# ifndef NI_NUMERICHOST
2008-06-24 22:38:37 +00:00
# define NI_NUMERICHOST 1
2009-11-16 22:25:46 +00:00
# endif
# ifndef NI_NUMERICSERV
2008-06-24 22:38:37 +00:00
# define NI_NUMERICSERV 2
2009-11-16 22:25:46 +00:00
# endif
2008-06-24 22:38:37 +00:00
2009-01-13 21:39:32 +00:00
static int
2009-01-27 21:10:31 +00:00
fake_getnameinfo ( const struct sockaddr * sa , size_t salen , char * host ,
2008-06-24 22:38:37 +00:00
size_t hostlen , char * serv , size_t servlen , int flags )
{
2010-02-18 17:41:15 -05:00
struct sockaddr_in * sin = ( struct sockaddr_in * ) sa ;
if ( serv ! = NULL ) {
char tmpserv [ 16 ] ;
evutil_snprintf ( tmpserv , sizeof ( tmpserv ) ,
" %d " , ntohs ( sin - > sin_port ) ) ;
if ( strlcpy ( serv , tmpserv , servlen ) > = servlen )
return ( - 1 ) ;
}
if ( host ! = NULL ) {
if ( flags & NI_NUMERICHOST ) {
if ( strlcpy ( host , inet_ntoa ( sin - > sin_addr ) ,
hostlen ) > = hostlen )
return ( - 1 ) ;
else
return ( 0 ) ;
} else {
struct hostent * hp ;
hp = gethostbyaddr ( ( char * ) & sin - > sin_addr ,
sizeof ( struct in_addr ) , AF_INET ) ;
if ( hp = = NULL )
return ( - 2 ) ;
if ( strlcpy ( host , hp - > h_name , hostlen ) > = hostlen )
return ( - 1 ) ;
else
return ( 0 ) ;
}
}
return ( 0 ) ;
2008-06-24 22:38:37 +00:00
}
# endif
2010-11-02 15:19:12 -04:00
# define REQ_VERSION_BEFORE(req, major_v, minor_v) \
( ( req ) - > major < ( major_v ) | | \
( ( req ) - > major = = ( major_v ) & & ( req ) - > minor < ( minor_v ) ) )
# define REQ_VERSION_ATLEAST(req, major_v, minor_v) \
( ( req ) - > major > ( major_v ) | | \
( ( req ) - > major = = ( major_v ) & & ( req ) - > minor > = ( minor_v ) ) )
2007-02-09 07:49:55 +00:00
# ifndef MIN
# define MIN(a,b) (((a)<(b))?(a):(b))
# endif
2024-10-29 11:43:02 +00:00
/** The request obj owns the evhttp connection and needs to free it */
# define EVHTTP_REQ_OWN_CONNECTION 0x0001
/** The request object is owned by the user; the user must free it */
# define EVHTTP_USER_OWNED 0x0004
/** The request will be used again upstack; freeing must be deferred */
# define EVHTTP_REQ_DEFER_FREE 0x0008
/** The request should be freed upstack */
# define EVHTTP_REQ_NEEDS_FREE 0x0010
2006-01-22 05:08:50 +00:00
extern int debug ;
2019-09-18 23:12:59 +03:00
static evutil_socket_t create_bind_socket_nonblock ( struct evutil_addrinfo * , int reuse ) ;
2008-06-20 06:52:13 +00:00
static evutil_socket_t bind_socket ( const char * , ev_uint16_t , int reuse ) ;
2009-07-30 17:01:21 +00:00
static void name_from_addr ( struct sockaddr * , ev_socklen_t , char * * , char * * ) ;
2020-10-27 01:40:34 +03:00
static struct evhttp_uri * evhttp_uri_parse_authority ( char * source_uri , unsigned flags ) ;
2006-11-18 03:05:26 +00:00
static int evhttp_associate_new_request_with_connection (
struct evhttp_connection * evcon ) ;
2006-11-23 06:32:20 +00:00
static void evhttp_connection_start_detectclose (
struct evhttp_connection * evcon ) ;
static void evhttp_connection_stop_detectclose (
struct evhttp_connection * evcon ) ;
static void evhttp_request_dispatch ( struct evhttp_connection * evcon ) ;
2008-06-26 00:40:57 +00:00
static void evhttp_read_firstline ( struct evhttp_connection * evcon ,
struct evhttp_request * req ) ;
static void evhttp_read_header ( struct evhttp_connection * evcon ,
2009-04-10 05:43:45 +00:00
struct evhttp_request * req ) ;
static int evhttp_add_header_internal ( struct evkeyvalq * headers ,
const char * key , const char * value ) ;
2010-05-26 12:50:59 -04:00
static const char * evhttp_response_phrase_internal ( int code ) ;
2022-08-01 10:16:18 +02:00
static void evhttp_get_request ( struct evhttp * , evutil_socket_t , struct sockaddr * , ev_socklen_t , struct bufferevent * bev ) ;
2010-08-09 13:25:50 -04:00
static void evhttp_write_buffer ( struct evhttp_connection * ,
void ( * ) ( struct evhttp_connection * , void * ) , void * ) ;
static void evhttp_make_header ( struct evhttp_connection * , struct evhttp_request * ) ;
2016-01-08 13:36:20 -08:00
static int evhttp_method_may_have_body_ ( struct evhttp_connection * , enum evhttp_cmd_type ) ;
2006-01-22 05:08:50 +00:00
2008-04-29 04:52:50 +00:00
/* callbacks for bufferevent */
static void evhttp_read_cb ( struct bufferevent * , void * ) ;
static void evhttp_write_cb ( struct bufferevent * , void * ) ;
static void evhttp_error_cb ( struct bufferevent * bufev , short what , void * arg ) ;
2016-01-08 13:36:20 -08:00
static int evhttp_find_vhost ( struct evhttp * http , struct evhttp * * outhttp , const char * hostname ) ;
static const char * evhttp_method_ ( struct evhttp_connection * evcon ,
enum evhttp_cmd_type type , ev_uint16_t * flags ) ;
2006-01-22 05:08:50 +00:00
2012-02-29 15:07:31 -05:00
# ifndef EVENT__HAVE_STRSEP
2007-10-03 04:14:54 +00:00
/* strsep replacement for platforms that lack it. Only works if
* del is one character long . */
2006-11-22 01:21:10 +00:00
static char *
strsep ( char * * s , const char * del )
{
char * d , * tok ;
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( strlen ( del ) = = 1 ) ;
2006-11-22 01:21:10 +00:00
if ( ! s | | ! * s )
return NULL ;
tok = * s ;
d = strstr ( tok , del ) ;
2007-10-03 04:14:54 +00:00
if ( d ) {
* d = ' \0 ' ;
* s = d + 1 ;
} else
2006-11-22 01:21:10 +00:00
* s = NULL ;
return tok ;
}
# endif
2011-05-23 18:01:24 -04:00
static size_t
html_replace ( const char ch , const char * * escaped )
2006-01-22 05:08:50 +00:00
{
switch ( ch ) {
case ' < ' :
2011-05-23 18:01:24 -04:00
* escaped = " < " ;
return 4 ;
2006-01-22 05:08:50 +00:00
case ' > ' :
2011-05-23 18:01:24 -04:00
* escaped = " > " ;
return 4 ;
2006-01-22 05:08:50 +00:00
case ' " ' :
2011-05-23 18:01:24 -04:00
* escaped = " " " ;
return 6 ;
2006-01-22 05:08:50 +00:00
case ' \' ' :
2011-05-23 18:01:24 -04:00
* escaped = " ' " ;
return 6 ;
2006-01-22 05:08:50 +00:00
case ' & ' :
2011-05-23 18:01:24 -04:00
* escaped = " & " ;
return 5 ;
2006-01-22 05:08:50 +00:00
default :
break ;
}
2011-05-23 18:01:24 -04:00
return 1 ;
2006-01-22 05:08:50 +00:00
}
/*
* Replaces < , > , " , ' and & with <, >, ",
* & # 03 9 ; and & amp ; correspondingly .
*
* The returned string needs to be freed by the caller .
*/
char *
evhttp_htmlescape ( const char * html )
{
2010-11-01 14:16:39 -04:00
size_t i ;
2011-05-23 18:01:24 -04:00
size_t new_size = 0 , old_size = 0 ;
2006-01-22 05:08:50 +00:00
char * escaped_html , * p ;
2008-02-26 20:24:29 +00:00
2011-05-23 18:01:24 -04:00
if ( html = = NULL )
return ( NULL ) ;
old_size = strlen ( html ) ;
for ( i = 0 ; i < old_size ; + + i ) {
const char * replaced = NULL ;
const size_t replace_size = html_replace ( html [ i ] , & replaced ) ;
if ( replace_size > EV_SIZE_MAX - new_size ) {
event_warn ( " %s: html_replace overflow " , __func__ ) ;
return ( NULL ) ;
}
new_size + = replace_size ;
}
2006-01-22 05:08:50 +00:00
2011-05-23 18:01:24 -04:00
if ( new_size = = EV_SIZE_MAX )
return ( NULL ) ;
2008-04-25 01:18:08 +00:00
p = escaped_html = mm_malloc ( new_size + 1 ) ;
2009-11-19 23:08:50 +00:00
if ( escaped_html = = NULL ) {
2011-05-23 18:01:24 -04:00
event_warn ( " %s: malloc(%lu) " , __func__ ,
( unsigned long ) ( new_size + 1 ) ) ;
2009-11-19 23:08:50 +00:00
return ( NULL ) ;
}
2007-02-16 08:48:55 +00:00
for ( i = 0 ; i < old_size ; + + i ) {
2011-05-23 18:01:24 -04:00
const char * replaced = & html [ i ] ;
const size_t len = html_replace ( html [ i ] , & replaced ) ;
2010-05-14 14:36:49 -04:00
memcpy ( p , replaced , len ) ;
p + = len ;
2006-01-22 05:08:50 +00:00
}
* p = ' \0 ' ;
return ( escaped_html ) ;
}
2010-08-09 13:25:50 -04:00
/** Given an evhttp_cmd_type, returns a constant string containing the
2016-01-08 13:36:20 -08:00
* equivalent HTTP command , or NULL if the evhttp_cmd_type is
2010-08-09 13:25:50 -04:00
* unrecognized . */
2007-11-07 07:33:16 +00:00
static const char *
2016-01-08 13:36:20 -08:00
evhttp_method_ ( struct evhttp_connection * evcon ,
enum evhttp_cmd_type type , ev_uint16_t * flags )
2006-01-22 05:08:50 +00:00
{
2016-01-08 13:36:20 -08:00
struct evhttp_ext_method ext_method ;
const char * method = NULL ;
ev_uint16_t tmp_flags = EVHTTP_METHOD_HAS_BODY ;
2006-01-22 05:08:50 +00:00
switch ( type ) {
case EVHTTP_REQ_GET :
method = " GET " ;
break ;
case EVHTTP_REQ_POST :
method = " POST " ;
break ;
case EVHTTP_REQ_HEAD :
method = " HEAD " ;
2016-01-08 13:36:20 -08:00
tmp_flags & = ~ EVHTTP_METHOD_HAS_BODY ;
2006-01-22 05:08:50 +00:00
break ;
2008-02-25 07:49:22 +00:00
case EVHTTP_REQ_PUT :
method = " PUT " ;
break ;
case EVHTTP_REQ_DELETE :
method = " DELETE " ;
break ;
2010-11-04 11:25:35 -04:00
case EVHTTP_REQ_OPTIONS :
method = " OPTIONS " ;
break ;
case EVHTTP_REQ_TRACE :
method = " TRACE " ;
2016-01-08 13:36:20 -08:00
tmp_flags & = ~ EVHTTP_METHOD_HAS_BODY ;
2010-11-04 11:25:35 -04:00
break ;
case EVHTTP_REQ_CONNECT :
method = " CONNECT " ;
break ;
case EVHTTP_REQ_PATCH :
method = " PATCH " ;
break ;
2019-03-13 10:51:55 +03:00
case EVHTTP_REQ_PROPFIND :
method = " PROPFIND " ;
break ;
case EVHTTP_REQ_PROPPATCH :
method = " PROPPATCH " ;
break ;
case EVHTTP_REQ_MKCOL :
method = " MKCOL " ;
break ;
case EVHTTP_REQ_LOCK :
method = " LOCK " ;
break ;
case EVHTTP_REQ_UNLOCK :
method = " UNLOCK " ;
break ;
case EVHTTP_REQ_COPY :
method = " COPY " ;
break ;
case EVHTTP_REQ_MOVE :
method = " MOVE " ;
break ;
2006-01-22 05:08:50 +00:00
default :
2016-01-08 13:36:20 -08:00
/* setup the structure to allow for the cmp.
*
* if the cmp function is set , it has the ability to
* modify method and flags . Other fields will be
* ignored .
*
* NOTE : the flags returned are OR ' d with the current
* flags .
*/
tmp_flags = 0 ;
ext_method . method = NULL ;
ext_method . type = type ;
ext_method . flags = tmp_flags ;
if ( evcon - > ext_method_cmp ! = NULL & &
evcon - > ext_method_cmp ( & ext_method ) = = 0 ) {
if ( ext_method . type ! = type ) {
event_debug ( ( " %s: callback modified type from %u to %u, not allowed " ,
__func__ , type , ext_method . type ) ) ;
return NULL ;
}
method = ext_method . method ;
tmp_flags | = ext_method . flags ;
}
2006-01-22 05:08:50 +00:00
break ;
}
2016-01-08 13:36:20 -08:00
event_debug ( ( " %s: type=%04x => '%s' flags=%04x " ,
__func__ , ( int ) type , method , tmp_flags ) ) ;
if ( flags )
* flags = tmp_flags ;
2006-01-22 05:08:50 +00:00
return ( method ) ;
}
2008-07-02 06:08:16 +00:00
/**
* Determines if a response should have a body .
* Follows the rules in RFC 2616 section 4.3 .
2010-02-18 17:41:15 -05:00
* @ return 1 if the response MUST have a body ; 0 if the response MUST NOT have
* a body .
2008-07-02 06:08:16 +00:00
*/
static int
evhttp_response_needs_body ( struct evhttp_request * req )
{
return ( req - > response_code ! = HTTP_NOCONTENT & &
req - > response_code ! = HTTP_NOTMODIFIED & &
( req - > response_code < 100 | | req - > response_code > = 200 ) & &
2020-01-12 13:43:18 +03:00
req - > type ! = EVHTTP_REQ_CONNECT & &
2008-07-02 06:08:16 +00:00
req - > type ! = EVHTTP_REQ_HEAD ) ;
}
2010-08-09 13:25:50 -04:00
/** Helper: called after we've added some data to an evcon's bufferevent's
* output buffer . Sets the evconn ' s writing - is - done callback , and puts
* the bufferevent into writing mode .
*/
static void
2006-07-17 00:33:57 +00:00
evhttp_write_buffer ( struct evhttp_connection * evcon ,
void ( * cb ) ( struct evhttp_connection * , void * ) , void * arg )
2006-01-22 05:08:50 +00:00
{
event_debug ( ( " %s: preparing to write buffer \n " , __func__ ) ) ;
/* Set call back */
2006-07-17 00:33:57 +00:00
evcon - > cb = cb ;
evcon - > cb_arg = arg ;
2006-01-22 05:08:50 +00:00
2010-11-04 14:05:08 -04:00
/* Disable the read callback: we don't actually care about data;
Fix crashing http server when callback do not reply in place
General http callback looks like:
static void http_cb(struct evhttp_request *req, void *arg)
{
evhttp_send_reply(req, HTTP_OK, "Everything is fine", NULL);
}
And they will work fine becuase in this case http will write request
first, and during write preparation it will disable *read callback* (in
evhttp_write_buffer()), but if we don't reply immediately, for example:
static void http_cb(struct evhttp_request *req, void *arg)
{
return;
}
This will leave connection in incorrect state, and if another request
will be written to the same connection libevent will abort with:
[err] ../http.c: illegal connection state 7
Because it thinks that read for now is not possible, since there were no
write.
Fix this by disabling EV_READ entirely. We couldn't just reset callbacks
because this will leave EOF detection, which we don't need, since user
hasn't replied to callback yet.
Reported-by: Cory Fields <cory@coryfields.com>
2017-10-23 00:13:37 +03:00
* we only care about close detection . ( We don ' t disable reading - -
* EV_READ , since we * do * want to learn about any close events . ) */
2010-11-04 14:05:08 -04:00
bufferevent_setcb ( evcon - > bufev ,
NULL , /*read*/
evhttp_write_cb ,
evhttp_error_cb ,
evcon ) ;
2013-09-03 14:46:47 -07:00
Fix crashing http server when callback do not reply in place
General http callback looks like:
static void http_cb(struct evhttp_request *req, void *arg)
{
evhttp_send_reply(req, HTTP_OK, "Everything is fine", NULL);
}
And they will work fine becuase in this case http will write request
first, and during write preparation it will disable *read callback* (in
evhttp_write_buffer()), but if we don't reply immediately, for example:
static void http_cb(struct evhttp_request *req, void *arg)
{
return;
}
This will leave connection in incorrect state, and if another request
will be written to the same connection libevent will abort with:
[err] ../http.c: illegal connection state 7
Because it thinks that read for now is not possible, since there were no
write.
Fix this by disabling EV_READ entirely. We couldn't just reset callbacks
because this will leave EOF detection, which we don't need, since user
hasn't replied to callback yet.
Reported-by: Cory Fields <cory@coryfields.com>
2017-10-23 00:13:37 +03:00
bufferevent_enable ( evcon - > bufev , EV_READ | EV_WRITE ) ;
2006-01-22 05:08:50 +00:00
}
2010-11-29 18:25:04 -08:00
static void
evhttp_send_continue_done ( struct evhttp_connection * evcon , void * arg )
{
bufferevent_disable ( evcon - > bufev , EV_WRITE ) ;
}
static void
evhttp_send_continue ( struct evhttp_connection * evcon ,
struct evhttp_request * req )
{
bufferevent_enable ( evcon - > bufev , EV_WRITE ) ;
evbuffer_add_printf ( bufferevent_get_output ( evcon - > bufev ) ,
" HTTP/%d.%d 100 Continue \r \n \r \n " ,
req - > major , req - > minor ) ;
evcon - > cb = evhttp_send_continue_done ;
evcon - > cb_arg = NULL ;
bufferevent_setcb ( evcon - > bufev ,
evhttp_read_cb ,
evhttp_write_cb ,
evhttp_error_cb ,
evcon ) ;
}
2010-08-09 13:25:50 -04:00
/** Helper: returns true iff evconn is in any connected state. */
2008-06-26 00:40:57 +00:00
static int
evhttp_connected ( struct evhttp_connection * evcon )
{
switch ( evcon - > state ) {
case EVCON_DISCONNECTED :
case EVCON_CONNECTING :
return ( 0 ) ;
case EVCON_IDLE :
case EVCON_READING_FIRSTLINE :
case EVCON_READING_HEADERS :
case EVCON_READING_BODY :
case EVCON_READING_TRAILER :
case EVCON_WRITING :
default :
return ( 1 ) ;
}
}
2010-08-09 13:25:50 -04:00
/* Create the headers needed for an outgoing HTTP request, adds them to
* the request ' s header list , and writes the request line to the
* connection ' s output buffer .
2006-02-27 02:27:37 +00:00
*/
2006-02-13 02:22:48 +00:00
static void
2006-07-17 00:33:57 +00:00
evhttp_make_header_request ( struct evhttp_connection * evcon ,
struct evhttp_request * req )
2006-02-13 02:22:48 +00:00
{
const char * method ;
2022-07-09 17:17:38 +03:00
/* NOTE: some version of GCC reports a warning that flags may be uninitialized, hence assignment */
ev_uint16_t flags = 0 ;
2009-01-27 21:10:31 +00:00
2006-02-13 02:22:48 +00:00
/* Generate request line */
2016-01-08 13:36:20 -08:00
if ( ! ( method = evhttp_method_ ( evcon , req - > type , & flags ) ) ) {
2016-06-24 18:07:39 -07:00
method = " NULL " ;
}
2008-05-08 22:51:39 +00:00
evbuffer_add_printf ( bufferevent_get_output ( evcon - > bufev ) ,
2008-04-29 04:52:50 +00:00
" %s %s HTTP/%d.%d \r \n " ,
2006-02-13 02:22:48 +00:00
method , req - > uri , req - > major , req - > minor ) ;
2016-01-08 13:36:20 -08:00
/* Add the content length on a request if missing
* Always add it for POST and PUT requests as clients expect it */
if ( ( flags & EVHTTP_METHOD_HAS_BODY ) & &
( evbuffer_get_length ( req - > output_buffer ) > 0 | |
req - > type = = EVHTTP_REQ_POST | | req - > type = = EVHTTP_REQ_PUT ) & &
evhttp_find_header ( req - > output_headers , " Content-Length " ) = = NULL ) {
2010-10-25 15:49:42 -04:00
char size [ 22 ] ;
2011-06-08 17:18:03 -04:00
evutil_snprintf ( size , sizeof ( size ) , EV_SIZE_FMT ,
EV_SIZE_ARG ( evbuffer_get_length ( req - > output_buffer ) ) ) ;
2006-02-13 02:22:48 +00:00
evhttp_add_header ( req - > output_headers , " Content-Length " , size ) ;
}
}
2010-08-09 13:25:50 -04:00
/** Return true if the list of headers in 'headers', intepreted with respect
* to flags , means that we should send a " connection: close " when the request
* is done . */
2006-11-18 03:05:26 +00:00
static int
2007-02-15 02:16:07 +00:00
evhttp_is_connection_close ( int flags , struct evkeyvalq * headers )
2006-11-18 03:05:26 +00:00
{
2018-03-28 19:30:38 -07:00
const char * connection = evhttp_find_header ( headers , " Connection " ) ;
return ( connection ! = NULL & & evutil_ascii_strcasecmp ( connection , " close " ) = = 0 ) ;
2006-11-18 03:05:26 +00:00
}
2018-03-28 19:30:38 -07:00
2015-04-24 19:04:51 +03:00
static int
evhttp_is_request_connection_close ( struct evhttp_request * req )
{
2020-01-12 15:29:48 +03:00
if ( req - > type = = EVHTTP_REQ_CONNECT )
return 0 ;
2015-04-24 19:04:51 +03:00
return
evhttp_is_connection_close ( req - > flags , req - > input_headers ) | |
evhttp_is_connection_close ( req - > flags , req - > output_headers ) ;
}
2006-11-18 03:05:26 +00:00
2010-08-09 13:25:50 -04:00
/* Return true iff 'headers' contains 'Connection: keep-alive' */
2007-02-13 06:25:16 +00:00
static int
evhttp_is_connection_keepalive ( struct evkeyvalq * headers )
{
const char * connection = evhttp_find_header ( headers , " Connection " ) ;
2009-01-27 21:10:31 +00:00
return ( connection ! = NULL
2009-07-28 19:41:57 +00:00
& & evutil_ascii_strncasecmp ( connection , " keep-alive " , 10 ) = = 0 ) ;
2007-02-13 06:25:16 +00:00
}
2010-08-09 13:25:50 -04:00
/* Add a correct "Date" header to headers, unless it already has one. */
2007-11-29 06:08:24 +00:00
static void
evhttp_maybe_add_date_header ( struct evkeyvalq * headers )
{
if ( evhttp_find_header ( headers , " Date " ) = = NULL ) {
char date [ 50 ] ;
2021-02-05 18:54:30 +03:00
if ( ( signed ) sizeof ( date ) > evutil_date_rfc1123 ( date , sizeof ( date ) , NULL ) ) {
2007-11-29 06:08:24 +00:00
evhttp_add_header ( headers , " Date " , date ) ;
}
}
}
2010-08-09 13:25:50 -04:00
/* Add a "Content-Length" header with value 'content_length' to headers,
* unless it already has a content - length or transfer - encoding header . */
2007-11-29 06:08:24 +00:00
static void
evhttp_maybe_add_content_length_header ( struct evkeyvalq * headers ,
2011-06-08 17:18:03 -04:00
size_t content_length )
2007-11-29 06:08:24 +00:00
{
if ( evhttp_find_header ( headers , " Transfer-Encoding " ) = = NULL & &
2023-03-27 04:35:00 +08:00
evhttp_find_header ( headers , " Content-Length " ) = = NULL ) {
2010-10-25 15:49:42 -04:00
char len [ 22 ] ;
2011-06-08 17:18:03 -04:00
evutil_snprintf ( len , sizeof ( len ) , EV_SIZE_FMT ,
EV_SIZE_ARG ( content_length ) ) ;
2007-11-29 06:08:24 +00:00
evhttp_add_header ( headers , " Content-Length " , len ) ;
}
}
2006-02-27 02:27:37 +00:00
/*
2010-08-09 13:25:50 -04:00
* Create the headers needed for an HTTP reply in req - > output_headers ,
* and write the first HTTP response for req line to evcon .
2006-02-27 02:27:37 +00:00
*/
2006-02-13 02:22:48 +00:00
static void
2006-07-17 00:33:57 +00:00
evhttp_make_header_response ( struct evhttp_connection * evcon ,
struct evhttp_request * req )
2006-02-13 02:22:48 +00:00
{
2008-05-15 01:53:48 +00:00
int is_keepalive = evhttp_is_connection_keepalive ( req - > input_headers ) ;
2023-03-05 23:59:21 +08:00
int need_body = evhttp_response_needs_body ( req ) ;
2008-05-08 22:51:39 +00:00
evbuffer_add_printf ( bufferevent_get_output ( evcon - > bufev ) ,
2008-04-29 04:52:50 +00:00
" HTTP/%d.%d %d %s \r \n " ,
2006-02-13 02:22:48 +00:00
req - > major , req - > minor , req - > response_code ,
req - > response_code_line ) ;
2008-05-15 01:53:48 +00:00
if ( req - > major = = 1 ) {
2010-11-02 15:19:12 -04:00
if ( req - > minor > = 1 )
2008-05-15 01:53:48 +00:00
evhttp_maybe_add_date_header ( req - > output_headers ) ;
2007-02-11 07:58:39 +00:00
2008-05-15 01:53:48 +00:00
/*
* if the protocol is 1.0 ; and the connection was keep - alive
* we need to add a keep - alive header , too .
2006-12-02 21:28:39 +00:00
*/
2008-05-15 01:53:48 +00:00
if ( req - > minor = = 0 & & is_keepalive )
evhttp_add_header ( req - > output_headers ,
" Connection " , " keep-alive " ) ;
2023-03-05 23:59:21 +08:00
if ( ( req - > minor > = 1 | | is_keepalive ) & & need_body ) {
2009-01-27 21:10:31 +00:00
/*
2008-05-15 01:53:48 +00:00
* we need to add the content length if the
* user did not give it , this is required for
* persistent connections to work .
*/
evhttp_maybe_add_content_length_header (
req - > output_headers ,
2011-06-08 17:18:03 -04:00
evbuffer_get_length ( req - > output_buffer ) ) ;
2008-05-15 01:53:48 +00:00
}
2007-11-29 06:08:24 +00:00
}
/* Potentially add headers for unidentified content. */
2023-03-05 23:59:21 +08:00
if ( need_body ) {
2006-12-02 21:28:39 +00:00
if ( evhttp_find_header ( req - > output_headers ,
2013-09-28 20:03:28 +02:00
" Content-Type " ) = = NULL
2013-10-10 16:10:50 -04:00
& & evcon - > http_server - > default_content_type ) {
2006-12-02 21:28:39 +00:00
evhttp_add_header ( req - > output_headers ,
2013-09-28 20:03:28 +02:00
" Content-Type " ,
2013-10-10 16:10:50 -04:00
evcon - > http_server - > default_content_type ) ;
2006-12-02 21:28:39 +00:00
}
2006-11-18 03:05:26 +00:00
}
/* if the request asked for a close, we send a close, too */
2007-02-15 02:16:07 +00:00
if ( evhttp_is_connection_close ( req - > flags , req - > input_headers ) ) {
2006-11-18 03:05:26 +00:00
evhttp_remove_header ( req - > output_headers , " Connection " ) ;
2018-03-28 19:30:38 -07:00
evhttp_add_header ( req - > output_headers , " Connection " , " close " ) ;
2006-11-17 07:45:42 +00:00
}
2006-02-13 02:22:48 +00:00
}
2016-03-11 13:08:28 +03:00
enum expect { NO , CONTINUE , OTHER } ;
static enum expect evhttp_have_expect ( struct evhttp_request * req , int input )
{
const char * expect ;
struct evkeyvalq * h = input ? req - > input_headers : req - > output_headers ;
2016-03-25 10:21:48 +03:00
if ( ! ( req - > kind = = EVHTTP_REQUEST ) | | ! REQ_VERSION_ATLEAST ( req , 1 , 1 ) )
2016-03-11 13:08:28 +03:00
return NO ;
expect = evhttp_find_header ( h , " Expect " ) ;
if ( ! expect )
return NO ;
return ! evutil_ascii_strcasecmp ( expect , " 100-continue " ) ? CONTINUE : OTHER ;
}
2010-08-09 13:25:50 -04:00
/** Generate all headers appropriate for sending the http request in req (or
* the response , if we ' re sending a response ) , and write them to evcon ' s
* bufferevent . Also writes all data from req - > output_buffer */
static void
2006-07-17 00:33:57 +00:00
evhttp_make_header ( struct evhttp_connection * evcon , struct evhttp_request * req )
2006-01-22 05:08:50 +00:00
{
struct evkeyval * header ;
2008-05-08 22:51:39 +00:00
struct evbuffer * output = bufferevent_get_output ( evcon - > bufev ) ;
2009-01-27 21:10:31 +00:00
2006-02-13 02:22:48 +00:00
/*
* Depending if this is a HTTP request or response , we might need to
* add some new headers or remove existing headers .
*/
2006-01-22 05:08:50 +00:00
if ( req - > kind = = EVHTTP_REQUEST ) {
2006-07-17 00:33:57 +00:00
evhttp_make_header_request ( evcon , req ) ;
2006-01-22 05:08:50 +00:00
} else {
2006-07-17 00:33:57 +00:00
evhttp_make_header_response ( evcon , req ) ;
2006-01-22 05:08:50 +00:00
}
TAILQ_FOREACH ( header , req - > output_headers , next ) {
2008-04-29 04:52:50 +00:00
evbuffer_add_printf ( output , " %s: %s \r \n " ,
2006-01-22 05:08:50 +00:00
header - > key , header - > value ) ;
}
2008-04-29 04:52:50 +00:00
evbuffer_add ( output , " \r \n " , 2 ) ;
2006-07-17 00:33:57 +00:00
2016-03-11 13:08:28 +03:00
if ( evhttp_have_expect ( req , 0 ) ! = CONTINUE & &
evbuffer_get_length ( req - > output_buffer ) ) {
2006-07-17 00:33:57 +00:00
/*
* For a request , we add the POST data , for a reply , this
* is the regular data .
*/
2008-04-29 04:52:50 +00:00
evbuffer_add_buffer ( output , req - > output_buffer ) ;
2006-01-22 05:08:50 +00:00
}
}
2009-11-04 20:17:32 +00:00
void
evhttp_connection_set_max_headers_size ( struct evhttp_connection * evcon ,
ev_ssize_t new_max_headers_size )
{
if ( new_max_headers_size < 0 )
evcon - > max_headers_size = EV_SIZE_MAX ;
else
evcon - > max_headers_size = new_max_headers_size ;
}
void
evhttp_connection_set_max_body_size ( struct evhttp_connection * evcon ,
ev_ssize_t new_max_body_size )
{
if ( new_max_body_size < 0 )
evcon - > max_body_size = EV_UINT64_MAX ;
else
evcon - > max_body_size = new_max_body_size ;
}
2006-11-22 05:03:02 +00:00
static int
evhttp_connection_incoming_fail ( struct evhttp_request * req ,
2013-03-21 13:55:40 +04:00
enum evhttp_request_error error )
2006-11-22 05:03:02 +00:00
{
2016-02-15 03:26:40 +03:00
switch ( error ) {
case EVREQ_HTTP_DATA_TOO_LONG :
req - > response_code = HTTP_ENTITYTOOLARGE ;
break ;
default :
req - > response_code = HTTP_BADREQUEST ;
}
2006-11-22 05:03:02 +00:00
switch ( error ) {
2013-03-21 13:55:40 +04:00
case EVREQ_HTTP_TIMEOUT :
case EVREQ_HTTP_EOF :
2009-01-27 21:10:31 +00:00
/*
2006-12-06 03:38:41 +00:00
* these are cases in which we probably should just
* close the connection and not send a reply . this
* case may happen when a browser keeps a persistent
2010-02-03 14:34:56 -08:00
* connection open and we timeout on the read . when
* the request is still being used for sending , we
2022-10-23 13:47:23 +02:00
* need to disassociate it from the connection here .
2006-11-22 05:03:02 +00:00
*/
2010-02-03 14:34:56 -08:00
if ( ! req - > userdone ) {
/* remove it so that it will not be freed */
TAILQ_REMOVE ( & req - > evcon - > requests , req , next ) ;
/* indicate that this request no longer has a
* connection object
*/
req - > evcon = NULL ;
}
2006-11-22 05:03:02 +00:00
return ( - 1 ) ;
2013-03-21 13:55:40 +04:00
case EVREQ_HTTP_INVALID_HEADER :
case EVREQ_HTTP_BUFFER_ERROR :
case EVREQ_HTTP_REQUEST_CANCEL :
case EVREQ_HTTP_DATA_TOO_LONG :
2006-11-22 05:03:02 +00:00
default : /* xxx: probably should just error on default */
/* the callback looks at the uri to determine errors */
if ( req - > uri ) {
2008-04-25 01:18:08 +00:00
mm_free ( req - > uri ) ;
2006-11-22 05:03:02 +00:00
req - > uri = NULL ;
}
2010-11-05 11:17:07 -07:00
if ( req - > uri_elems ) {
evhttp_uri_free ( req - > uri_elems ) ;
req - > uri_elems = NULL ;
}
2006-12-06 03:38:41 +00:00
2009-01-27 21:10:31 +00:00
/*
2006-12-06 03:38:41 +00:00
* the callback needs to send a reply , once the reply has
* been send , the connection should get freed .
*/
2006-11-22 05:03:02 +00:00
( * req - > cb ) ( req , req - > cb_arg ) ;
}
2009-01-27 21:10:31 +00:00
2006-11-22 05:03:02 +00:00
return ( 0 ) ;
}
2015-09-09 19:21:51 +03:00
/* Free connection ownership of which can be acquired by user using
* evhttp_request_own ( ) . */
static inline void
evhttp_request_free_auto ( struct evhttp_request * req )
{
2016-02-15 03:53:25 +03:00
if ( ! ( req - > flags & EVHTTP_USER_OWNED ) )
2015-09-09 19:21:51 +03:00
evhttp_request_free ( req ) ;
}
2015-08-18 03:03:51 +03:00
static void
evhttp_request_free_ ( struct evhttp_connection * evcon , struct evhttp_request * req )
{
TAILQ_REMOVE ( & evcon - > requests , req , next ) ;
2015-09-09 19:21:51 +03:00
evhttp_request_free_auto ( req ) ;
2015-08-18 03:03:51 +03:00
}
2019-03-04 06:53:42 +03:00
static void
evhttp_set_timeout_tv_ ( struct timeval * tv , const struct timeval * timeout , int def )
{
if ( timeout = = NULL & & def ! = - 1 ) {
tv - > tv_sec = def ;
tv - > tv_usec = 0 ;
return ;
}
if ( timeout ) {
* tv = * timeout ;
} else {
evutil_timerclear ( tv ) ;
}
}
static void
evhttp_set_timeout_ ( struct timeval * tv , int timeout , int def )
{
if ( timeout = = - 1 ) {
timeout = def ;
}
if ( timeout = = - 1 ) {
evutil_timerclear ( tv ) ;
} else {
struct timeval timeout_tv ;
timeout_tv . tv_sec = timeout ;
timeout_tv . tv_usec = 0 ;
* tv = timeout_tv ;
}
}
2010-08-09 13:25:50 -04:00
/* Called when evcon has experienced a (non-recoverable? -NM) error, as
* given in error . If it ' s an outgoing connection , reset the connection ,
* retry any pending requests , and inform the user . If it ' s incoming ,
* delegates to evhttp_connection_incoming_fail ( ) . */
2006-01-22 05:08:50 +00:00
void
2012-02-29 15:07:33 -05:00
evhttp_connection_fail_ ( struct evhttp_connection * evcon ,
2013-03-21 13:55:40 +04:00
enum evhttp_request_error error )
2006-01-22 05:08:50 +00:00
{
2012-11-16 11:51:42 -05:00
const int errsave = EVUTIL_SOCKET_ERROR ( ) ;
2006-07-17 00:33:57 +00:00
struct evhttp_request * req = TAILQ_FIRST ( & evcon - > requests ) ;
2007-09-02 01:33:38 +00:00
void ( * cb ) ( struct evhttp_request * , void * ) ;
void * cb_arg ;
2013-03-21 13:55:40 +04:00
void ( * error_cb ) ( enum evhttp_request_error , void * ) ;
void * error_cb_arg ;
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( req ! = NULL ) ;
2009-01-27 21:10:31 +00:00
2008-04-29 04:52:50 +00:00
bufferevent_disable ( evcon - > bufev , EV_READ | EV_WRITE ) ;
2017-05-19 20:11:52 -07:00
error_cb = req - > error_cb ;
error_cb_arg = req - > cb_arg ;
2006-12-06 03:38:41 +00:00
if ( evcon - > flags & EVHTTP_CON_INCOMING ) {
2009-01-27 21:10:31 +00:00
/*
2006-12-06 03:38:41 +00:00
* for incoming requests , there are two different
* failure cases . it ' s either a network level error
* or an http layer error . for problems on the network
* layer like timeouts we just drop the connections .
* For HTTP problems , we might have to send back a
* reply before the connection can be freed .
2006-11-16 08:49:26 +00:00
*/
2006-12-06 03:38:41 +00:00
if ( evhttp_connection_incoming_fail ( req , error ) = = - 1 )
evhttp_connection_free ( evcon ) ;
2017-05-19 20:11:52 -07:00
if ( error_cb ! = NULL )
error_cb ( error , error_cb_arg ) ;
2006-12-06 03:38:41 +00:00
return ;
2006-07-17 00:33:57 +00:00
}
2008-05-12 03:12:09 +00:00
/* when the request was canceled, the callback is not executed */
2013-03-21 13:55:40 +04:00
if ( error ! = EVREQ_HTTP_REQUEST_CANCEL ) {
2008-05-12 03:12:09 +00:00
/* save the callback for later; the cb might free our object */
cb = req - > cb ;
cb_arg = req - > cb_arg ;
} else {
cb = NULL ;
cb_arg = NULL ;
}
2006-11-16 08:49:26 +00:00
2008-05-12 03:12:09 +00:00
/* do not fail all requests; the next request is going to get
* send over a new connection . when a user cancels a request ,
* all other pending requests should be processed as normal
*/
2015-08-18 03:03:51 +03:00
evhttp_request_free_ ( evcon , req ) ;
2006-12-06 03:38:41 +00:00
/* reset the connection */
2019-09-04 00:56:20 +03:00
evhttp_connection_reset_ ( evcon , 1 ) ;
2009-01-27 21:10:31 +00:00
2006-07-17 00:33:57 +00:00
/* We are trying the next request that was queued on us */
if ( TAILQ_FIRST ( & evcon - > requests ) ! = NULL )
2012-02-29 15:07:33 -05:00
evhttp_connection_connect_ ( evcon ) ;
2020-05-19 11:45:43 +03:00
else
if ( ( evcon - > flags & EVHTTP_CON_OUTGOING ) & &
( evcon - > flags & EVHTTP_CON_AUTOFREE ) ) {
evhttp_connection_free ( evcon ) ;
}
2007-09-02 01:33:38 +00:00
2012-11-13 21:22:39 -08:00
/* The call to evhttp_connection_reset_ overwrote errno.
* Let ' s restore the original errno , so that the user ' s
* callback can have a better idea of what the error was .
*/
2012-11-16 11:51:42 -05:00
EVUTIL_SET_SOCKET_ERROR ( errsave ) ;
2012-11-13 21:22:39 -08:00
2007-09-02 01:33:38 +00:00
/* inform the user */
2013-03-21 13:55:40 +04:00
if ( error_cb ! = NULL )
error_cb ( error , error_cb_arg ) ;
2007-09-02 01:33:38 +00:00
if ( cb ! = NULL )
( * cb ) ( NULL , cb_arg ) ;
2006-01-22 05:08:50 +00:00
}
2010-08-09 13:25:50 -04:00
/* Bufferevent callback: invoked when any data has been written from an
* http connection ' s bufferevent */
2008-04-29 04:52:50 +00:00
static void
evhttp_write_cb ( struct bufferevent * bufev , void * arg )
2006-01-22 05:08:50 +00:00
{
2006-07-17 00:33:57 +00:00
struct evhttp_connection * evcon = arg ;
2006-01-22 05:08:50 +00:00
/* Activate our call back */
2006-12-09 02:58:12 +00:00
if ( evcon - > cb ! = NULL )
( * evcon - > cb ) ( evcon , evcon - > cb_arg ) ;
2006-07-17 00:33:57 +00:00
}
2008-06-26 00:40:57 +00:00
/**
* Advance the connection state .
* - If this is an outgoing connection , we ' ve just processed the response ;
* idle or close the connection .
* - If this is an incoming connection , we ' ve just processed the request ;
* respond .
*/
2007-11-07 07:33:16 +00:00
static void
2006-07-17 00:33:57 +00:00
evhttp_connection_done ( struct evhttp_connection * evcon )
{
struct evhttp_request * req = TAILQ_FIRST ( & evcon - > requests ) ;
2007-05-12 06:23:52 +00:00
int con_outgoing = evcon - > flags & EVHTTP_CON_OUTGOING ;
2014-10-26 01:18:10 -04:00
int free_evcon = 0 ;
2007-11-07 07:33:16 +00:00
2007-05-12 06:23:52 +00:00
if ( con_outgoing ) {
2008-06-26 00:40:57 +00:00
/* idle or close the connection */
2015-04-24 19:04:51 +03:00
int need_close = evhttp_is_request_connection_close ( req ) ;
2006-07-17 00:33:57 +00:00
TAILQ_REMOVE ( & evcon - > requests , req , next ) ;
req - > evcon = NULL ;
2008-06-26 00:40:57 +00:00
evcon - > state = EVCON_IDLE ;
2006-11-17 07:45:42 +00:00
/* check if we got asked to close the connection */
2006-11-23 06:32:20 +00:00
if ( need_close )
2019-09-04 00:56:20 +03:00
evhttp_connection_reset_ ( evcon , 1 ) ;
2006-11-17 07:45:42 +00:00
2006-07-17 00:33:57 +00:00
if ( TAILQ_FIRST ( & evcon - > requests ) ! = NULL ) {
/*
* We have more requests ; reset the connection
2008-06-26 00:40:57 +00:00
* and deal with the next request .
2006-07-17 00:33:57 +00:00
*/
2008-06-26 00:40:57 +00:00
if ( ! evhttp_connected ( evcon ) )
2012-02-29 15:07:33 -05:00
evhttp_connection_connect_ ( evcon ) ;
2006-11-23 06:32:20 +00:00
else
evhttp_request_dispatch ( evcon ) ;
} else if ( ! need_close ) {
/*
* The connection is going to be persistent , but we
* need to detect if the other side closes it .
*/
evhttp_connection_start_detectclose ( evcon ) ;
2014-10-26 01:18:10 -04:00
} else if ( ( evcon - > flags & EVHTTP_CON_AUTOFREE ) ) {
/*
* If we have no more requests that need completion
* and we ' re not waiting for the connection to close
*/
free_evcon = 1 ;
2006-07-17 00:33:57 +00:00
}
2008-06-26 00:40:57 +00:00
} else {
/*
* incoming connection - we need to leave the request on the
* connection so that we can reply to it .
*/
evcon - > state = EVCON_WRITING ;
2006-07-17 00:33:57 +00:00
}
/* notify the user of the request */
2006-01-22 05:08:50 +00:00
( * req - > cb ) ( req , req - > cb_arg ) ;
2006-07-17 00:33:57 +00:00
2015-09-09 19:21:51 +03:00
/* if this was an outgoing request, we own and it's done. so free it. */
if ( con_outgoing ) {
evhttp_request_free_auto ( req ) ;
2006-07-17 00:33:57 +00:00
}
2014-10-26 01:18:10 -04:00
/* If this was the last request of an outgoing connection and we're
* not waiting to receive a connection close event and we want to
* automatically free the connection . We check to ensure our request
* list is empty one last time just in case our callback added a
* new request .
*/
if ( free_evcon & & TAILQ_FIRST ( & evcon - > requests ) = = NULL ) {
evhttp_connection_free ( evcon ) ;
}
2006-01-22 05:08:50 +00:00
}
2006-12-18 15:26:19 +00:00
/*
* Handles reading from a chunked request .
2008-06-02 05:45:26 +00:00
* return ALL_DATA_READ :
* all data has been read
* return MORE_DATA_EXPECTED :
* more data is expected
* return DATA_CORRUPTED :
* data is corrupted
2009-10-16 13:19:57 +00:00
* return REQUEST_CANCELED :
2008-06-02 05:45:26 +00:00
* request was canceled by the user calling evhttp_cancel_request
2009-11-04 20:17:32 +00:00
* return DATA_TOO_LONG :
* ran over the maximum limit
2006-12-18 15:26:19 +00:00
*/
2008-06-26 00:40:57 +00:00
static enum message_read_status
2006-12-18 15:26:19 +00:00
evhttp_handle_chunked_read ( struct evhttp_request * req , struct evbuffer * buf )
{
2011-05-25 12:58:59 -04:00
if ( req = = NULL | | buf = = NULL ) {
return DATA_CORRUPTED ;
}
while ( 1 ) {
size_t buflen ;
if ( ( buflen = evbuffer_get_length ( buf ) ) = = 0 ) {
break ;
}
/* evbuffer_get_length returns size_t, but len variable is ssize_t,
* check for overflow conditions */
2011-05-25 18:52:07 -04:00
if ( buflen > EV_SSIZE_MAX ) {
2011-05-25 12:58:59 -04:00
return DATA_CORRUPTED ;
}
2006-12-18 15:26:19 +00:00
if ( req - > ntoread < 0 ) {
/* Read chunk size */
2008-07-02 04:22:48 +00:00
ev_int64_t ntoread ;
2007-11-25 21:32:26 +00:00
char * p = evbuffer_readln ( buf , NULL , EVBUFFER_EOL_CRLF ) ;
2006-12-18 15:26:19 +00:00
char * endp ;
int error ;
2024-02-18 08:56:46 -05:00
size_t len_p ;
2006-12-18 15:26:19 +00:00
if ( p = = NULL )
break ;
2024-02-18 08:56:46 -05:00
len_p = strlen ( p ) ;
2007-02-14 06:10:32 +00:00
/* the last chunk is on a new line? */
2024-02-18 08:56:46 -05:00
if ( len_p = = 0 ) {
2008-04-25 01:18:08 +00:00
mm_free ( p ) ;
2007-02-14 06:10:32 +00:00
continue ;
2007-07-31 00:32:00 +00:00
}
2024-02-18 08:56:46 -05:00
/* strtoll(,,16) lets through whitespace, 0x, +, and - prefixes, but HTTP doesn't. */
error = isspace ( p [ 0 ] ) | |
p [ 0 ] = = ' - ' | |
p [ 0 ] = = ' + ' | |
2024-02-18 19:26:47 +01:00
( len_p > = 2 & & p [ 0 ] = = ' 0 ' & & ( p [ 1 ] = = ' x ' | | p [ 1 ] = = ' X ' ) ) ;
2024-02-18 08:56:46 -05:00
if ( error ) {
mm_free ( p ) ;
return ( DATA_CORRUPTED ) ;
}
2008-07-02 04:22:48 +00:00
ntoread = evutil_strtoll ( p , & endp , 16 ) ;
error = ( * p = = ' \0 ' | |
( * endp ! = ' \0 ' & & * endp ! = ' ' ) | |
ntoread < 0 ) ;
2008-04-25 01:18:08 +00:00
mm_free ( p ) ;
2006-12-18 15:26:19 +00:00
if ( error ) {
/* could not get chunk size */
2008-06-02 05:45:26 +00:00
return ( DATA_CORRUPTED ) ;
2006-12-18 15:26:19 +00:00
}
2011-05-25 12:58:59 -04:00
/* ntoread is signed int64, body_size is unsigned size_t, check for under/overflow conditions */
2011-05-27 15:08:25 -04:00
if ( ( ev_uint64_t ) ntoread > EV_SIZE_MAX - req - > body_size ) {
2011-05-25 18:52:07 -04:00
return DATA_CORRUPTED ;
2011-05-25 12:58:59 -04:00
}
2009-11-05 21:22:23 +00:00
if ( req - > body_size + ( size_t ) ntoread > req - > evcon - > max_body_size ) {
2010-02-18 17:08:50 -05:00
/* failed body length test */
2009-11-04 20:17:32 +00:00
event_debug ( ( " Request body is too long " ) ) ;
return ( DATA_TOO_LONG ) ;
}
2011-05-25 18:52:07 -04:00
2009-11-05 21:22:23 +00:00
req - > body_size + = ( size_t ) ntoread ;
2008-07-02 04:22:48 +00:00
req - > ntoread = ntoread ;
2006-12-18 15:26:19 +00:00
if ( req - > ntoread = = 0 ) {
/* Last chunk */
2008-06-02 05:45:26 +00:00
return ( ALL_DATA_READ ) ;
2006-12-18 15:26:19 +00:00
}
2007-02-14 06:10:32 +00:00
continue ;
}
2011-05-25 12:58:59 -04:00
/* req->ntoread is signed int64, len is ssize_t, based on arch,
* ssize_t could only be 32 b , check for these conditions */
2011-05-25 18:52:07 -04:00
if ( req - > ntoread > EV_SSIZE_MAX ) {
2011-05-25 12:58:59 -04:00
return DATA_CORRUPTED ;
}
2007-02-14 06:10:32 +00:00
/* don't have enough to complete a chunk; wait for more */
2011-05-27 15:08:25 -04:00
if ( req - > ntoread > 0 & & buflen < ( ev_uint64_t ) req - > ntoread )
2008-06-02 05:45:26 +00:00
return ( MORE_DATA_EXPECTED ) ;
2007-02-14 06:10:32 +00:00
/* Completed chunk */
2009-05-01 00:54:14 +00:00
evbuffer_remove_buffer ( buf , req - > input_buffer , ( size_t ) req - > ntoread ) ;
2007-02-14 06:10:32 +00:00
req - > ntoread = - 1 ;
if ( req - > chunk_cb ! = NULL ) {
2008-06-02 05:45:26 +00:00
req - > flags | = EVHTTP_REQ_DEFER_FREE ;
2007-02-14 06:10:32 +00:00
( * req - > chunk_cb ) ( req , req - > cb_arg ) ;
evbuffer_drain ( req - > input_buffer ,
2009-04-17 06:56:09 +00:00
evbuffer_get_length ( req - > input_buffer ) ) ;
2008-06-02 05:45:26 +00:00
req - > flags & = ~ EVHTTP_REQ_DEFER_FREE ;
if ( ( req - > flags & EVHTTP_REQ_NEEDS_FREE ) ! = 0 ) {
return ( REQUEST_CANCELED ) ;
}
2006-12-18 15:26:19 +00:00
}
}
2008-06-02 05:45:26 +00:00
return ( MORE_DATA_EXPECTED ) ;
2006-12-18 15:26:19 +00:00
}
2008-06-26 00:40:57 +00:00
static void
evhttp_read_trailer ( struct evhttp_connection * evcon , struct evhttp_request * req )
{
struct evbuffer * buf = bufferevent_get_input ( evcon - > bufev ) ;
2012-02-29 15:07:33 -05:00
switch ( evhttp_parse_headers_ ( req , buf ) ) {
2008-06-26 00:40:57 +00:00
case DATA_CORRUPTED :
2009-11-04 20:17:32 +00:00
case DATA_TOO_LONG :
2013-03-21 13:55:40 +04:00
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_DATA_TOO_LONG ) ;
2008-06-26 00:40:57 +00:00
break ;
case ALL_DATA_READ :
bufferevent_disable ( evcon - > bufev , EV_READ ) ;
evhttp_connection_done ( evcon ) ;
break ;
case MORE_DATA_EXPECTED :
2009-04-23 06:27:58 +00:00
case REQUEST_CANCELED : /* ??? */
2008-06-26 00:40:57 +00:00
default :
break ;
}
}
2016-02-15 00:12:54 +03:00
static void
2016-02-15 03:26:40 +03:00
evhttp_lingering_close ( struct evhttp_connection * evcon ,
struct evhttp_request * req )
2016-02-15 00:12:54 +03:00
{
struct evbuffer * buf = bufferevent_get_input ( evcon - > bufev ) ;
size_t n = evbuffer_get_length ( buf ) ;
if ( n > ( size_t ) req - > ntoread )
n = ( size_t ) req - > ntoread ;
req - > ntoread - = n ;
req - > body_size + = n ;
2016-12-06 11:51:18 +03:00
event_debug ( ( " Request body is too long, left " EV_I64_FMT ,
EV_I64_ARG ( req - > ntoread ) ) ) ;
2016-02-15 00:12:54 +03:00
evbuffer_drain ( buf , n ) ;
if ( ! req - > ntoread )
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_DATA_TOO_LONG ) ;
}
2016-02-15 03:26:40 +03:00
static void
evhttp_lingering_fail ( struct evhttp_connection * evcon ,
struct evhttp_request * req )
{
if ( evcon - > flags & EVHTTP_CON_LINGERING_CLOSE )
evhttp_lingering_close ( evcon , req ) ;
else
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_DATA_TOO_LONG ) ;
}
2016-02-15 00:12:54 +03:00
2007-11-07 07:33:16 +00:00
static void
2007-01-04 18:05:17 +00:00
evhttp_read_body ( struct evhttp_connection * evcon , struct evhttp_request * req )
{
2008-05-08 22:51:39 +00:00
struct evbuffer * buf = bufferevent_get_input ( evcon - > bufev ) ;
2009-01-27 21:10:31 +00:00
2007-01-04 18:05:17 +00:00
if ( req - > chunked ) {
2008-06-02 05:45:26 +00:00
switch ( evhttp_handle_chunked_read ( req , buf ) ) {
case ALL_DATA_READ :
2007-01-04 18:05:17 +00:00
/* finished last chunk */
2008-06-26 00:40:57 +00:00
evcon - > state = EVCON_READING_TRAILER ;
evhttp_read_trailer ( evcon , req ) ;
2007-01-04 18:05:17 +00:00
return ;
2008-06-02 05:45:26 +00:00
case DATA_CORRUPTED :
2007-01-04 18:05:17 +00:00
/* corrupted data */
2024-02-18 15:24:58 +01:00
evhttp_connection_fail_ ( evcon ,
EVREQ_HTTP_INVALID_HEADER ) ;
return ;
case DATA_TOO_LONG :
2012-02-29 15:07:33 -05:00
evhttp_connection_fail_ ( evcon ,
2013-03-21 13:55:40 +04:00
EVREQ_HTTP_DATA_TOO_LONG ) ;
2007-01-04 18:05:17 +00:00
return ;
2008-06-02 05:45:26 +00:00
case REQUEST_CANCELED :
/* request canceled */
2015-09-09 19:21:51 +03:00
evhttp_request_free_auto ( req ) ;
2008-06-02 05:45:26 +00:00
return ;
case MORE_DATA_EXPECTED :
default :
break ;
2007-01-04 18:05:17 +00:00
}
} else if ( req - > ntoread < 0 ) {
/* Read until connection close. */
2011-05-25 18:52:07 -04:00
if ( ( size_t ) ( req - > body_size + evbuffer_get_length ( buf ) ) < req - > body_size ) {
2013-03-21 13:55:40 +04:00
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_INVALID_HEADER ) ;
2011-05-25 14:31:09 -04:00
return ;
}
2009-11-04 20:17:32 +00:00
req - > body_size + = evbuffer_get_length ( buf ) ;
2010-10-25 16:09:11 -04:00
evbuffer_add_buffer ( req - > input_buffer , buf ) ;
2011-05-25 14:31:09 -04:00
} else if ( req - > chunk_cb ! = NULL | | evbuffer_get_length ( buf ) > = ( size_t ) req - > ntoread ) {
/* XXX: the above get_length comparison has to be fixed for overflow conditions! */
2008-06-02 05:45:26 +00:00
/* We've postponed moving the data until now, but we're
* about to use it . */
2010-10-25 16:00:47 -04:00
size_t n = evbuffer_get_length ( buf ) ;
2011-05-25 14:31:09 -04:00
2010-10-25 16:00:47 -04:00
if ( n > ( size_t ) req - > ntoread )
n = ( size_t ) req - > ntoread ;
req - > ntoread - = n ;
req - > body_size + = n ;
evbuffer_remove_buffer ( buf , req - > input_buffer , n ) ;
2008-06-02 05:45:26 +00:00
}
2011-02-21 23:25:13 -05:00
if ( req - > body_size > req - > evcon - > max_body_size | |
( ! req - > chunked & & req - > ntoread > = 0 & &
( size_t ) req - > ntoread > req - > evcon - > max_body_size ) ) {
2011-05-25 14:31:09 -04:00
/* XXX: The above casted comparison must checked for overflow */
2010-02-18 17:08:50 -05:00
/* failed body length test */
2016-02-15 00:12:54 +03:00
2016-02-15 03:26:40 +03:00
evhttp_lingering_fail ( evcon , req ) ;
2009-11-04 20:17:32 +00:00
return ;
}
2009-04-17 06:56:09 +00:00
if ( evbuffer_get_length ( req - > input_buffer ) > 0 & & req - > chunk_cb ! = NULL ) {
2008-06-02 05:45:26 +00:00
req - > flags | = EVHTTP_REQ_DEFER_FREE ;
( * req - > chunk_cb ) ( req , req - > cb_arg ) ;
req - > flags & = ~ EVHTTP_REQ_DEFER_FREE ;
evbuffer_drain ( req - > input_buffer ,
2009-04-17 06:56:09 +00:00
evbuffer_get_length ( req - > input_buffer ) ) ;
2008-06-02 05:45:26 +00:00
if ( ( req - > flags & EVHTTP_REQ_NEEDS_FREE ) ! = 0 ) {
2015-09-09 19:21:51 +03:00
evhttp_request_free_auto ( req ) ;
2008-06-02 05:45:26 +00:00
return ;
}
}
2016-02-15 00:12:54 +03:00
if ( ! req - > ntoread ) {
2008-04-29 04:52:50 +00:00
bufferevent_disable ( evcon - > bufev , EV_READ ) ;
2007-01-04 18:05:17 +00:00
/* Completed content length */
evhttp_connection_done ( evcon ) ;
return ;
}
}
2010-10-25 21:53:15 -04:00
# define get_deferred_queue(evcon) \
2012-04-06 11:05:35 -04:00
( ( evcon ) - > base )
2010-10-25 21:53:15 -04:00
2006-01-22 05:08:50 +00:00
/*
2008-04-29 04:52:50 +00:00
* Gets called when more data becomes available
2006-01-22 05:08:50 +00:00
*/
2008-04-29 04:52:50 +00:00
static void
evhttp_read_cb ( struct bufferevent * bufev , void * arg )
2006-01-22 05:08:50 +00:00
{
2006-07-17 00:33:57 +00:00
struct evhttp_connection * evcon = arg ;
struct evhttp_request * req = TAILQ_FIRST ( & evcon - > requests ) ;
2006-01-22 05:08:50 +00:00
2010-10-25 21:53:15 -04:00
/* Cancel if it's pending. */
2012-02-29 15:07:33 -05:00
event_deferred_cb_cancel_ ( get_deferred_queue ( evcon ) ,
2010-10-25 21:53:15 -04:00
& evcon - > read_more_deferred_cb ) ;
2008-06-26 00:40:57 +00:00
switch ( evcon - > state ) {
case EVCON_READING_FIRSTLINE :
evhttp_read_firstline ( evcon , req ) ;
/* note the request may have been freed in
* evhttp_read_body */
break ;
case EVCON_READING_HEADERS :
evhttp_read_header ( evcon , req ) ;
/* note the request may have been freed in
* evhttp_read_body */
break ;
case EVCON_READING_BODY :
evhttp_read_body ( evcon , req ) ;
/* note the request may have been freed in
* evhttp_read_body */
break ;
case EVCON_READING_TRAILER :
evhttp_read_trailer ( evcon , req ) ;
break ;
2011-03-23 12:05:33 +03:00
case EVCON_IDLE :
{
2011-04-26 23:42:01 -04:00
# ifdef USE_DEBUG
2011-03-23 12:05:33 +03:00
struct evbuffer * input ;
size_t total_len ;
input = bufferevent_get_input ( evcon - > bufev ) ;
total_len = evbuffer_get_length ( input ) ;
2011-06-08 17:18:03 -04:00
event_debug ( ( " %s: read " EV_SIZE_FMT
" bytes in EVCON_IDLE state, "
" resetting connection " ,
__func__ , EV_SIZE_ARG ( total_len ) ) ) ;
2011-04-26 23:42:01 -04:00
# endif
2011-03-23 12:05:33 +03:00
2019-09-04 00:56:20 +03:00
evhttp_connection_reset_ ( evcon , 1 ) ;
2011-03-23 12:05:33 +03:00
}
break ;
2008-06-26 00:40:57 +00:00
case EVCON_DISCONNECTED :
case EVCON_CONNECTING :
case EVCON_WRITING :
default :
event_errx ( 1 , " %s: illegal connection state %d " ,
__func__ , evcon - > state ) ;
}
2006-01-22 05:08:50 +00:00
}
2010-10-25 21:53:15 -04:00
static void
2012-04-06 11:05:35 -04:00
evhttp_deferred_read_cb ( struct event_callback * cb , void * data )
2010-10-25 21:53:15 -04:00
{
struct evhttp_connection * evcon = data ;
2019-01-27 15:16:39 +03:00
struct bufferevent * bev = evcon - > bufev ;
if ( bev - > readcb )
( bev - > readcb ) ( evcon - > bufev , evcon ) ;
2010-10-25 21:53:15 -04:00
}
2007-11-07 07:33:16 +00:00
static void
2006-07-17 00:33:57 +00:00
evhttp_write_connectioncb ( struct evhttp_connection * evcon , void * arg )
2006-01-22 05:08:50 +00:00
{
/* This is after writing the request to the server */
2006-07-17 00:33:57 +00:00
struct evhttp_request * req = TAILQ_FIRST ( & evcon - > requests ) ;
2016-03-11 13:08:28 +03:00
struct evbuffer * output = bufferevent_get_output ( evcon - > bufev ) ;
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( req ! = NULL ) ;
2006-01-22 05:08:50 +00:00
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( evcon - > state = = EVCON_WRITING ) ;
2008-06-26 00:40:57 +00:00
2016-03-11 13:08:28 +03:00
/* We need to wait until we've written all of our output data before we can
* continue */
if ( evbuffer_get_length ( output ) > 0 )
return ;
2014-11-24 02:32:23 -05:00
2006-01-22 05:08:50 +00:00
/* We are done writing our header and are now expecting the response */
req - > kind = EVHTTP_RESPONSE ;
2012-02-29 15:07:33 -05:00
evhttp_start_read_ ( evcon ) ;
2006-01-22 05:08:50 +00:00
}
/*
* Clean up a connection object
*/
void
evhttp_connection_free ( struct evhttp_connection * evcon )
{
2006-12-06 03:38:41 +00:00
struct evhttp_request * req ;
2006-12-09 02:58:12 +00:00
/* notify interested parties that this connection is going down */
2019-09-04 00:56:20 +03:00
if ( evhttp_connected ( evcon ) & & evcon - > closecb ! = NULL )
( * evcon - > closecb ) ( evcon , evcon - > closecb_arg ) ;
2006-12-09 02:58:12 +00:00
2010-02-03 14:34:56 -08:00
/* remove all requests that might be queued on this
* connection . for server connections , this should be empty .
* because it gets dequeued either in evhttp_connection_done or
2012-02-29 15:07:33 -05:00
* evhttp_connection_fail_ .
2010-02-03 14:34:56 -08:00
*/
2006-12-06 03:38:41 +00:00
while ( ( req = TAILQ_FIRST ( & evcon - > requests ) ) ! = NULL ) {
2015-08-18 03:03:51 +03:00
evhttp_request_free_ ( evcon , req ) ;
2006-12-06 03:38:41 +00:00
}
2006-11-18 03:05:26 +00:00
if ( evcon - > http_server ! = NULL ) {
struct evhttp * http = evcon - > http_server ;
TAILQ_REMOVE ( & http - > connections , evcon , next ) ;
2018-01-30 15:39:41 -08:00
http - > connection_cnt - - ;
2006-11-18 03:05:26 +00:00
}
2010-01-25 13:38:07 -05:00
if ( event_initialized ( & evcon - > retry_ev ) ) {
2008-04-29 04:52:50 +00:00
event_del ( & evcon - > retry_ev ) ;
2010-01-25 13:38:07 -05:00
event_debug_unassign ( & evcon - > retry_ev ) ;
}
2008-04-29 04:52:50 +00:00
2012-02-29 15:07:33 -05:00
event_deferred_cb_cancel_ ( get_deferred_queue ( evcon ) ,
2010-10-25 21:53:15 -04:00
& evcon - > read_more_deferred_cb ) ;
2018-11-13 21:26:12 +03:00
if ( evcon - > bufev ! = NULL ) {
bufferevent_free ( evcon - > bufev ) ;
}
2016-03-22 23:36:19 +03:00
2007-09-07 02:49:46 +00:00
if ( evcon - > bind_address ! = NULL )
2008-04-25 01:18:08 +00:00
mm_free ( evcon - > bind_address ) ;
2007-09-07 02:49:46 +00:00
2006-01-22 05:08:50 +00:00
if ( evcon - > address ! = NULL )
2008-04-25 01:18:08 +00:00
mm_free ( evcon - > address ) ;
2006-01-22 05:08:50 +00:00
2021-08-12 01:40:27 +03:00
# ifndef _WIN32
2016-01-31 11:31:00 +00:00
if ( evcon - > unixsocket ! = NULL )
mm_free ( evcon - > unixsocket ) ;
2021-08-12 01:40:27 +03:00
# endif
2016-01-31 11:31:00 +00:00
2008-04-25 01:18:08 +00:00
mm_free ( evcon ) ;
2006-01-22 05:08:50 +00:00
}
2014-10-26 01:18:10 -04:00
void
evhttp_connection_free_on_completion ( struct evhttp_connection * evcon ) {
evcon - > flags | = EVHTTP_CON_AUTOFREE ;
}
2007-09-07 02:49:46 +00:00
void
evhttp_connection_set_local_address ( struct evhttp_connection * evcon ,
const char * address )
{
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( evcon - > state = = EVCON_DISCONNECTED ) ;
2007-09-07 02:49:46 +00:00
if ( evcon - > bind_address )
2008-04-25 01:18:08 +00:00
mm_free ( evcon - > bind_address ) ;
if ( ( evcon - > bind_address = mm_strdup ( address ) ) = = NULL )
2010-12-18 02:40:22 -02:00
event_warn ( " %s: strdup " , __func__ ) ;
2007-09-07 02:49:46 +00:00
}
2008-11-16 23:22:14 +00:00
void
evhttp_connection_set_local_port ( struct evhttp_connection * evcon ,
ev_uint16_t port )
{
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( evcon - > state = = EVCON_DISCONNECTED ) ;
2008-11-16 23:22:14 +00:00
evcon - > bind_port = port ;
}
2007-09-07 02:49:46 +00:00
2006-11-23 06:32:20 +00:00
static void
2006-07-17 00:33:57 +00:00
evhttp_request_dispatch ( struct evhttp_connection * evcon )
{
struct evhttp_request * req = TAILQ_FIRST ( & evcon - > requests ) ;
2009-01-27 21:10:31 +00:00
2006-07-17 00:33:57 +00:00
/* this should not usually happy but it's possible */
if ( req = = NULL )
return ;
http: fix connection retries when there more then one request for connection
We should not attemp to establishe the connection if there is retry
timer active, since otherwise there will be a bug.
Imagine next situation:
con = evhttp_connection_base_new()
evhttp_connection_set_retries(con, 2)
req = evhttp_request_new()
evhttp_make_request(con, req, ...)
# failed during connecting, and timer for 2 second scheduler (retry_ev)
Then another request scheduled for this evcon:
evhttp_make_request(con, req, ...)
# got request from server,
# and now it tries to read the response from the server
# (req.kind == EVHTTP_RESPONSE)
#
# but at this point retry_ev scheduled,
# and it schedules the connect again,
# and after the connect will succeeed, it will pick request with
# EVHTTP_RESPONSE for sending and this is completelly wrong and will
# fail in evhttp_make_header_response() since there is no
# "http_server" for this evcon
This was a long standing issue, that I came across few years ago
firstly, bad only now I had time to dig into it (but right now it was
pretty simple, by limiting amount of CPU for the process and using rr
for debug to go back and forth).
2018-12-07 21:46:27 +03:00
EVUTIL_ASSERT ( req - > kind = = EVHTTP_REQUEST ) ;
2006-11-23 06:32:20 +00:00
/* delete possible close detection events */
evhttp_connection_stop_detectclose ( evcon ) ;
2009-01-27 21:10:31 +00:00
2006-07-17 00:33:57 +00:00
/* we assume that the connection is connected already */
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( evcon - > state = = EVCON_IDLE ) ;
2008-06-26 00:40:57 +00:00
evcon - > state = EVCON_WRITING ;
2006-07-17 00:33:57 +00:00
/* Create the header from the store arguments */
evhttp_make_header ( evcon , req ) ;
evhttp_write_buffer ( evcon , evhttp_write_connectioncb , NULL ) ;
}
2019-09-04 00:56:20 +03:00
/** Hard-reset our connection state
*
* This will :
* - reset fd
* - clears out buffers
* - call closecb
*/
static void
evhttp_connection_reset_hard_ ( struct evhttp_connection * evcon )
2006-07-17 00:33:57 +00:00
{
2009-01-16 00:25:54 +00:00
struct evbuffer * tmp ;
2016-03-11 19:52:32 +03:00
int err ;
2009-01-16 00:25:54 +00:00
2011-03-07 21:55:47 -05:00
/* XXXX This is not actually an optimal fix. Instead we ought to have
2021-03-23 09:02:39 +03:00
an API for " stop connecting " , or use bufferevent_replacefd to turn off
2011-03-07 21:55:47 -05:00
connecting . But for Libevent 2.0 , this seems like a minimal change
least likely to disrupt the rest of the bufferevent and http code .
Why is this here ? If the fd is set in the bufferevent , and the
bufferevent is connecting , then you can ' t actually stop the
bufferevent from trying to connect with bufferevent_disable ( ) . The
connect will never trigger , since we close the fd , but the timeout
2012-02-29 15:07:33 -05:00
might . That caused an assertion failure in evhttp_connection_fail_ .
2011-03-07 21:55:47 -05:00
*/
2012-02-29 15:07:33 -05:00
bufferevent_disable_hard_ ( evcon - > bufev , EV_READ | EV_WRITE ) ;
2006-07-17 00:33:57 +00:00
2019-09-04 00:56:20 +03:00
/* inform interested parties about connection close */
if ( evhttp_connected ( evcon ) & & evcon - > closecb ! = NULL )
( * evcon - > closecb ) ( evcon , evcon - > closecb_arg ) ;
2010-10-05 13:06:32 -04:00
2019-09-04 00:56:20 +03:00
/** FIXME: manipulating with fd is unwanted */
2021-03-23 09:02:39 +03:00
err = bufferevent_replacefd ( evcon - > bufev , - 1 ) ;
2018-11-14 00:20:20 +03:00
EVUTIL_ASSERT ( ! err & & " setfd " ) ;
2009-01-16 00:25:54 +00:00
/* we need to clean up any buffered data */
tmp = bufferevent_get_output ( evcon - > bufev ) ;
2016-03-11 19:52:32 +03:00
err = evbuffer_drain ( tmp , - 1 ) ;
EVUTIL_ASSERT ( ! err & & " drain output " ) ;
2009-01-16 00:25:54 +00:00
tmp = bufferevent_get_input ( evcon - > bufev ) ;
2016-03-11 19:52:32 +03:00
err = evbuffer_drain ( tmp , - 1 ) ;
EVUTIL_ASSERT ( ! err & & " drain input " ) ;
2019-09-04 00:56:20 +03:00
}
2009-01-16 00:25:54 +00:00
2019-09-04 00:56:20 +03:00
/** Reset our connection state
*
* This will :
* - disables reading / writing
* - puts us in DISCONNECTED state
*
* @ param hard - hard reset will ( @ see evhttp_connection_reset_hard_ ( ) )
*/
void
evhttp_connection_reset_ ( struct evhttp_connection * evcon , int hard )
{
bufferevent_setcb ( evcon - > bufev , NULL , NULL , NULL , NULL ) ;
2016-03-11 20:17:51 +03:00
2019-09-04 00:56:20 +03:00
if ( hard ) {
evhttp_connection_reset_hard_ ( evcon ) ;
}
evcon - > flags & = ~ EVHTTP_CON_READING_ERROR ;
2006-07-17 00:33:57 +00:00
evcon - > state = EVCON_DISCONNECTED ;
2006-11-23 06:32:20 +00:00
}
static void
evhttp_connection_start_detectclose ( struct evhttp_connection * evcon )
{
evcon - > flags | = EVHTTP_CON_CLOSEDETECT ;
2008-12-19 21:31:43 +00:00
bufferevent_enable ( evcon - > bufev , EV_READ ) ;
2006-11-23 06:32:20 +00:00
}
static void
evhttp_connection_stop_detectclose ( struct evhttp_connection * evcon )
{
2010-11-29 22:44:18 -05:00
evcon - > flags & = ~ EVHTTP_CON_CLOSEDETECT ;
2008-12-19 21:31:43 +00:00
bufferevent_disable ( evcon - > bufev , EV_READ ) ;
2006-07-17 00:33:57 +00:00
}
2006-12-09 01:41:57 +00:00
static void
2007-11-25 21:53:06 +00:00
evhttp_connection_retry ( evutil_socket_t fd , short what , void * arg )
2006-12-09 01:41:57 +00:00
{
struct evhttp_connection * evcon = arg ;
evcon - > state = EVCON_DISCONNECTED ;
2012-02-29 15:07:33 -05:00
evhttp_connection_connect_ ( evcon ) ;
2006-12-09 01:41:57 +00:00
}
2008-04-29 04:52:50 +00:00
static void
evhttp_connection_cb_cleanup ( struct evhttp_connection * evcon )
{
2011-03-24 15:52:34 +03:00
struct evcon_requestq requests ;
2020-05-19 01:02:30 +03:00
EVUTIL_ASSERT ( evcon - > flags & EVHTTP_CON_OUTGOING ) ;
2011-03-24 15:52:34 +03:00
2019-09-04 00:56:20 +03:00
evhttp_connection_reset_ ( evcon , 1 ) ;
2020-05-19 01:02:30 +03:00
2008-04-29 04:52:50 +00:00
if ( evcon - > retry_max < 0 | | evcon - > retry_cnt < evcon - > retry_max ) {
2012-01-24 14:34:04 -05:00
struct timeval tv_retry = evcon - > initial_retry_timeout ;
int i ;
2008-05-02 16:28:25 +00:00
evtimer_assign ( & evcon - > retry_ev , evcon - > base , evhttp_connection_retry , evcon ) ;
2010-01-22 16:14:49 -05:00
/* XXXX handle failure from evhttp_add_event */
2012-01-24 14:34:04 -05:00
for ( i = 0 ; i < evcon - > retry_cnt ; + + i ) {
tv_retry . tv_usec * = 2 ;
if ( tv_retry . tv_usec > 1000000 ) {
tv_retry . tv_usec - = 1000000 ;
tv_retry . tv_sec + = 1 ;
}
tv_retry . tv_sec * = 2 ;
if ( tv_retry . tv_sec > 3600 ) {
tv_retry . tv_sec = 3600 ;
tv_retry . tv_usec = 0 ;
}
}
event_add ( & evcon - > retry_ev , & tv_retry ) ;
2008-04-29 04:52:50 +00:00
evcon - > retry_cnt + + ;
return ;
}
2011-03-24 15:52:34 +03:00
/*
* User callback can do evhttp_make_request ( ) on the same
* evcon so new request will be added to evcon - > requests . To
* avoid freeing it prematurely we iterate over the copy of
* the queue .
*/
TAILQ_INIT ( & requests ) ;
2008-04-29 04:52:50 +00:00
while ( TAILQ_FIRST ( & evcon - > requests ) ! = NULL ) {
struct evhttp_request * request = TAILQ_FIRST ( & evcon - > requests ) ;
TAILQ_REMOVE ( & evcon - > requests , request , next ) ;
2011-03-24 15:52:34 +03:00
TAILQ_INSERT_TAIL ( & requests , request , next ) ;
}
/* for now, we just signal all requests by executing their callbacks */
while ( TAILQ_FIRST ( & requests ) ! = NULL ) {
struct evhttp_request * request = TAILQ_FIRST ( & requests ) ;
TAILQ_REMOVE ( & requests , request , next ) ;
2008-04-29 04:52:50 +00:00
request - > evcon = NULL ;
/* we might want to set an error here */
request - > cb ( request , request - > cb_arg ) ;
2015-09-09 19:21:51 +03:00
evhttp_request_free_auto ( request ) ;
2008-04-29 04:52:50 +00:00
}
2020-05-19 01:02:30 +03:00
if ( TAILQ_FIRST ( & evcon - > requests ) = = NULL
& & ( evcon - > flags & EVHTTP_CON_AUTOFREE ) ) {
evhttp_connection_free ( evcon ) ;
}
2008-04-29 04:52:50 +00:00
}
http: read server response even after server closed the connection
Otherwise if we will try to write more data than server can accept
(see `evhttp_set_max_body_size()` for libevent server) we will get `EPIPE` and
will not try to read server's response which must contain 400 error for now
(which is not strictly correct though, it must 413).
```
$ strace regress --no-fork http/data_length_constraints
...
connect(10, {sa_family=AF_INET, sin_port=htons(43988), sin_addr=inet_addr("127.0.0.1")}, 16) = -1 EINPROGRESS (Operation now in progress)
...
writev(10, [{"POST / HTTP/1.1\r\nHost: somehost\r"..., 60}, {"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16324}], 2) = 16384
epoll_wait(5, [{EPOLLOUT, {u32=10, u64=10}}, {EPOLLIN, {u32=11, u64=11}}], 32, 50000) = 2
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = 16384
ioctl(11, FIONREAD, [32768]) = 0
readv(11, [{"POST / HTTP/1.1\r\nHost: somehost\r"..., 4096}], 1) = 4096
epoll_ctl(5, EPOLL_CTL_DEL, 11, 0x7fff09d41e50) = 0
epoll_ctl(5, EPOLL_CTL_ADD, 11, {EPOLLOUT, {u32=11, u64=11}}) = 0
epoll_wait(5, [{EPOLLOUT, {u32=10, u64=10}}, {EPOLLOUT, {u32=11, u64=11}}], 32, 50000) = 2
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = 16384
writev(11, [{"HTTP/1.1 400 Bad Request\r\nConten"..., 129}, {"<HTML><HEAD>\n<TITLE>400 Bad Requ"..., 94}], 2) = 223
epoll_ctl(5, EPOLL_CTL_DEL, 11, 0x7fff09d42080) = 0
shutdown(11, SHUT_WR) = 0
close(11) = 0
epoll_wait(5, [{EPOLLOUT|EPOLLERR|EPOLLHUP, {u32=10, u64=10}}], 32, 50000) = 1
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = -1 EPIPE (Broken pipe)
--- SIGPIPE {si_signo=SIGPIPE, si_code=SI_USER, si_pid=13954, si_uid=1000} ---
epoll_ctl(5, EPOLL_CTL_DEL, 10, 0x7fff09d42010) = 0
shutdown(10, SHUT_WR) = -1 ENOTCONN (Transport endpoint is not connected)
close(10) = 0
write(1, "\n FAIL ../test/regress_http.c:3"..., 37
```
Careful reader can ask why it send error even when it didn't read
`evcon->max_body_size`, and the answer will be checks for `evcon->max_body_size
against `Content-Length` header, which contains ~8MB (-2 bytes).
And also if we will not drain the output buffer than we will send buffer that
we didn't send in previous request and instead of sending method via
`evhttp_make_header()`.
Fixes: http/data_length_constraints
Refs: #321
v2: do this only under EVHTTP_CON_READ_ON_WRITE_ERROR flag
2016-02-10 14:43:18 +03:00
static void
evhttp_connection_read_on_write_error ( struct evhttp_connection * evcon ,
struct evhttp_request * req )
{
struct evbuffer * buf ;
2016-03-11 20:17:51 +03:00
/** Second time, we can't read anything */
if ( evcon - > flags & EVHTTP_CON_READING_ERROR ) {
evcon - > flags & = ~ EVHTTP_CON_READING_ERROR ;
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_EOF ) ;
return ;
}
http: read server response even after server closed the connection
Otherwise if we will try to write more data than server can accept
(see `evhttp_set_max_body_size()` for libevent server) we will get `EPIPE` and
will not try to read server's response which must contain 400 error for now
(which is not strictly correct though, it must 413).
```
$ strace regress --no-fork http/data_length_constraints
...
connect(10, {sa_family=AF_INET, sin_port=htons(43988), sin_addr=inet_addr("127.0.0.1")}, 16) = -1 EINPROGRESS (Operation now in progress)
...
writev(10, [{"POST / HTTP/1.1\r\nHost: somehost\r"..., 60}, {"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16324}], 2) = 16384
epoll_wait(5, [{EPOLLOUT, {u32=10, u64=10}}, {EPOLLIN, {u32=11, u64=11}}], 32, 50000) = 2
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = 16384
ioctl(11, FIONREAD, [32768]) = 0
readv(11, [{"POST / HTTP/1.1\r\nHost: somehost\r"..., 4096}], 1) = 4096
epoll_ctl(5, EPOLL_CTL_DEL, 11, 0x7fff09d41e50) = 0
epoll_ctl(5, EPOLL_CTL_ADD, 11, {EPOLLOUT, {u32=11, u64=11}}) = 0
epoll_wait(5, [{EPOLLOUT, {u32=10, u64=10}}, {EPOLLOUT, {u32=11, u64=11}}], 32, 50000) = 2
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = 16384
writev(11, [{"HTTP/1.1 400 Bad Request\r\nConten"..., 129}, {"<HTML><HEAD>\n<TITLE>400 Bad Requ"..., 94}], 2) = 223
epoll_ctl(5, EPOLL_CTL_DEL, 11, 0x7fff09d42080) = 0
shutdown(11, SHUT_WR) = 0
close(11) = 0
epoll_wait(5, [{EPOLLOUT|EPOLLERR|EPOLLHUP, {u32=10, u64=10}}], 32, 50000) = 1
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = -1 EPIPE (Broken pipe)
--- SIGPIPE {si_signo=SIGPIPE, si_code=SI_USER, si_pid=13954, si_uid=1000} ---
epoll_ctl(5, EPOLL_CTL_DEL, 10, 0x7fff09d42010) = 0
shutdown(10, SHUT_WR) = -1 ENOTCONN (Transport endpoint is not connected)
close(10) = 0
write(1, "\n FAIL ../test/regress_http.c:3"..., 37
```
Careful reader can ask why it send error even when it didn't read
`evcon->max_body_size`, and the answer will be checks for `evcon->max_body_size
against `Content-Length` header, which contains ~8MB (-2 bytes).
And also if we will not drain the output buffer than we will send buffer that
we didn't send in previous request and instead of sending method via
`evhttp_make_header()`.
Fixes: http/data_length_constraints
Refs: #321
v2: do this only under EVHTTP_CON_READ_ON_WRITE_ERROR flag
2016-02-10 14:43:18 +03:00
req - > kind = EVHTTP_RESPONSE ;
buf = bufferevent_get_output ( evcon - > bufev ) ;
evbuffer_unfreeze ( buf , 1 ) ;
evbuffer_drain ( buf , evbuffer_get_length ( buf ) ) ;
evbuffer_freeze ( buf , 1 ) ;
2016-03-11 20:17:51 +03:00
evhttp_start_read_ ( evcon ) ;
2016-03-11 19:58:05 +03:00
evcon - > flags | = EVHTTP_CON_READING_ERROR ;
http: read server response even after server closed the connection
Otherwise if we will try to write more data than server can accept
(see `evhttp_set_max_body_size()` for libevent server) we will get `EPIPE` and
will not try to read server's response which must contain 400 error for now
(which is not strictly correct though, it must 413).
```
$ strace regress --no-fork http/data_length_constraints
...
connect(10, {sa_family=AF_INET, sin_port=htons(43988), sin_addr=inet_addr("127.0.0.1")}, 16) = -1 EINPROGRESS (Operation now in progress)
...
writev(10, [{"POST / HTTP/1.1\r\nHost: somehost\r"..., 60}, {"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16324}], 2) = 16384
epoll_wait(5, [{EPOLLOUT, {u32=10, u64=10}}, {EPOLLIN, {u32=11, u64=11}}], 32, 50000) = 2
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = 16384
ioctl(11, FIONREAD, [32768]) = 0
readv(11, [{"POST / HTTP/1.1\r\nHost: somehost\r"..., 4096}], 1) = 4096
epoll_ctl(5, EPOLL_CTL_DEL, 11, 0x7fff09d41e50) = 0
epoll_ctl(5, EPOLL_CTL_ADD, 11, {EPOLLOUT, {u32=11, u64=11}}) = 0
epoll_wait(5, [{EPOLLOUT, {u32=10, u64=10}}, {EPOLLOUT, {u32=11, u64=11}}], 32, 50000) = 2
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = 16384
writev(11, [{"HTTP/1.1 400 Bad Request\r\nConten"..., 129}, {"<HTML><HEAD>\n<TITLE>400 Bad Requ"..., 94}], 2) = 223
epoll_ctl(5, EPOLL_CTL_DEL, 11, 0x7fff09d42080) = 0
shutdown(11, SHUT_WR) = 0
close(11) = 0
epoll_wait(5, [{EPOLLOUT|EPOLLERR|EPOLLHUP, {u32=10, u64=10}}], 32, 50000) = 1
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = -1 EPIPE (Broken pipe)
--- SIGPIPE {si_signo=SIGPIPE, si_code=SI_USER, si_pid=13954, si_uid=1000} ---
epoll_ctl(5, EPOLL_CTL_DEL, 10, 0x7fff09d42010) = 0
shutdown(10, SHUT_WR) = -1 ENOTCONN (Transport endpoint is not connected)
close(10) = 0
write(1, "\n FAIL ../test/regress_http.c:3"..., 37
```
Careful reader can ask why it send error even when it didn't read
`evcon->max_body_size`, and the answer will be checks for `evcon->max_body_size
against `Content-Length` header, which contains ~8MB (-2 bytes).
And also if we will not drain the output buffer than we will send buffer that
we didn't send in previous request and instead of sending method via
`evhttp_make_header()`.
Fixes: http/data_length_constraints
Refs: #321
v2: do this only under EVHTTP_CON_READ_ON_WRITE_ERROR flag
2016-02-10 14:43:18 +03:00
}
2008-04-29 04:52:50 +00:00
static void
evhttp_error_cb ( struct bufferevent * bufev , short what , void * arg )
{
struct evhttp_connection * evcon = arg ;
2008-06-26 00:40:57 +00:00
struct evhttp_request * req = TAILQ_FIRST ( & evcon - > requests ) ;
2008-04-29 04:52:50 +00:00
switch ( evcon - > state ) {
case EVCON_CONNECTING :
2010-11-30 00:05:54 -05:00
if ( what & BEV_EVENT_TIMEOUT ) {
2012-11-02 11:44:29 -04:00
event_debug ( ( " %s: connection timeout for \" %s:%d \" on "
EV_SOCK_FMT ,
2008-04-29 04:52:50 +00:00
__func__ , evcon - > address , evcon - > port ,
2019-09-04 00:56:20 +03:00
EV_SOCK_ARG ( bufferevent_getfd ( bufev ) ) ) ) ;
2008-04-29 04:52:50 +00:00
evhttp_connection_cb_cleanup ( evcon ) ;
return ;
}
break ;
2008-06-26 00:40:57 +00:00
case EVCON_READING_BODY :
if ( ! req - > chunked & & req - > ntoread < 0
2009-05-13 20:36:56 +00:00
& & what = = ( BEV_EVENT_READING | BEV_EVENT_EOF ) ) {
2008-04-29 04:52:50 +00:00
/* EOF on read can be benign */
evhttp_connection_done ( evcon ) ;
return ;
}
break ;
case EVCON_DISCONNECTED :
2008-06-26 00:40:57 +00:00
case EVCON_IDLE :
case EVCON_READING_FIRSTLINE :
case EVCON_READING_HEADERS :
case EVCON_READING_TRAILER :
case EVCON_WRITING :
2008-04-29 04:52:50 +00:00
default :
break ;
}
2008-12-19 21:31:43 +00:00
/* when we are in close detect mode, a read error means that
* the other side closed their connection .
*/
if ( evcon - > flags & EVHTTP_CON_CLOSEDETECT ) {
evcon - > flags & = ~ EVHTTP_CON_CLOSEDETECT ;
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( evcon - > http_server = = NULL ) ;
2008-12-19 22:41:07 +00:00
/* For connections from the client, we just
* reset the connection so that it becomes
* disconnected .
2009-01-27 21:10:31 +00:00
*/
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( evcon - > state = = EVCON_IDLE ) ;
2019-09-04 00:56:20 +03:00
evhttp_connection_reset_ ( evcon , 1 ) ;
2014-10-26 01:18:10 -04:00
/*
* If we have no more requests that need completion
* and we want to auto - free the connection when all
* requests have been completed .
*/
if ( TAILQ_FIRST ( & evcon - > requests ) = = NULL
& & ( evcon - > flags & EVHTTP_CON_OUTGOING )
& & ( evcon - > flags & EVHTTP_CON_AUTOFREE ) ) {
evhttp_connection_free ( evcon ) ;
}
2008-12-19 21:31:43 +00:00
return ;
}
2009-05-13 20:36:56 +00:00
if ( what & BEV_EVENT_TIMEOUT ) {
2013-03-21 13:55:40 +04:00
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_TIMEOUT ) ;
2009-05-13 20:36:56 +00:00
} else if ( what & ( BEV_EVENT_EOF | BEV_EVENT_ERROR ) ) {
http: read server response even after server closed the connection
Otherwise if we will try to write more data than server can accept
(see `evhttp_set_max_body_size()` for libevent server) we will get `EPIPE` and
will not try to read server's response which must contain 400 error for now
(which is not strictly correct though, it must 413).
```
$ strace regress --no-fork http/data_length_constraints
...
connect(10, {sa_family=AF_INET, sin_port=htons(43988), sin_addr=inet_addr("127.0.0.1")}, 16) = -1 EINPROGRESS (Operation now in progress)
...
writev(10, [{"POST / HTTP/1.1\r\nHost: somehost\r"..., 60}, {"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16324}], 2) = 16384
epoll_wait(5, [{EPOLLOUT, {u32=10, u64=10}}, {EPOLLIN, {u32=11, u64=11}}], 32, 50000) = 2
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = 16384
ioctl(11, FIONREAD, [32768]) = 0
readv(11, [{"POST / HTTP/1.1\r\nHost: somehost\r"..., 4096}], 1) = 4096
epoll_ctl(5, EPOLL_CTL_DEL, 11, 0x7fff09d41e50) = 0
epoll_ctl(5, EPOLL_CTL_ADD, 11, {EPOLLOUT, {u32=11, u64=11}}) = 0
epoll_wait(5, [{EPOLLOUT, {u32=10, u64=10}}, {EPOLLOUT, {u32=11, u64=11}}], 32, 50000) = 2
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = 16384
writev(11, [{"HTTP/1.1 400 Bad Request\r\nConten"..., 129}, {"<HTML><HEAD>\n<TITLE>400 Bad Requ"..., 94}], 2) = 223
epoll_ctl(5, EPOLL_CTL_DEL, 11, 0x7fff09d42080) = 0
shutdown(11, SHUT_WR) = 0
close(11) = 0
epoll_wait(5, [{EPOLLOUT|EPOLLERR|EPOLLHUP, {u32=10, u64=10}}], 32, 50000) = 1
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = -1 EPIPE (Broken pipe)
--- SIGPIPE {si_signo=SIGPIPE, si_code=SI_USER, si_pid=13954, si_uid=1000} ---
epoll_ctl(5, EPOLL_CTL_DEL, 10, 0x7fff09d42010) = 0
shutdown(10, SHUT_WR) = -1 ENOTCONN (Transport endpoint is not connected)
close(10) = 0
write(1, "\n FAIL ../test/regress_http.c:3"..., 37
```
Careful reader can ask why it send error even when it didn't read
`evcon->max_body_size`, and the answer will be checks for `evcon->max_body_size
against `Content-Length` header, which contains ~8MB (-2 bytes).
And also if we will not drain the output buffer than we will send buffer that
we didn't send in previous request and instead of sending method via
`evhttp_make_header()`.
Fixes: http/data_length_constraints
Refs: #321
v2: do this only under EVHTTP_CON_READ_ON_WRITE_ERROR flag
2016-02-10 14:43:18 +03:00
if ( what & BEV_EVENT_WRITING & &
evcon - > flags & EVHTTP_CON_READ_ON_WRITE_ERROR ) {
evhttp_connection_read_on_write_error ( evcon , req ) ;
return ;
}
2019-01-29 01:09:44 +03:00
if ( what & BEV_EVENT_READING & &
evcon - > flags & EVHTTP_CON_READ_ON_WRITE_ERROR & &
evbuffer_get_length ( bufferevent_get_input ( bufev ) ) ) {
event_deferred_cb_schedule_ ( get_deferred_queue ( evcon ) ,
& evcon - > read_more_deferred_cb ) ;
return ;
}
2013-03-21 13:55:40 +04:00
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_EOF ) ;
2011-09-12 10:46:17 -04:00
} else if ( what = = BEV_EVENT_CONNECTED ) {
2008-04-29 04:52:50 +00:00
} else {
2013-03-21 13:55:40 +04:00
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_BUFFER_ERROR ) ;
2008-04-29 04:52:50 +00:00
}
}
2006-01-22 05:08:50 +00:00
/*
2009-12-30 00:11:27 -05:00
* Event callback for asynchronous connection attempt .
2006-01-22 05:08:50 +00:00
*/
2006-11-23 06:32:20 +00:00
static void
2009-12-30 00:11:27 -05:00
evhttp_connection_cb ( struct bufferevent * bufev , short what , void * arg )
2006-01-22 05:08:50 +00:00
{
struct evhttp_connection * evcon = arg ;
2013-01-23 02:45:32 +04:00
2009-12-30 00:11:27 -05:00
if ( ! ( what & BEV_EVENT_CONNECTED ) ) {
/* some operating systems return ECONNREFUSED immediately
* when connecting to a local address . the cleanup is going
* to reschedule this function call .
*/
2011-05-25 19:50:56 -04:00
# ifndef _WIN32
2009-12-30 00:11:27 -05:00
if ( errno = = ECONNREFUSED )
goto cleanup ;
2010-01-23 20:07:05 -05:00
# endif
2009-12-30 00:11:27 -05:00
evhttp_error_cb ( bufev , what , arg ) ;
return ;
}
2006-01-22 05:08:50 +00:00
/* We are connected to the server now */
2012-11-02 11:44:29 -04:00
event_debug ( ( " %s: connected to \" %s:%d \" on " EV_SOCK_FMT " \n " ,
__func__ , evcon - > address , evcon - > port ,
2019-09-04 00:56:20 +03:00
EV_SOCK_ARG ( bufferevent_getfd ( bufev ) ) ) ) ;
2006-01-22 05:08:50 +00:00
2006-12-09 01:41:57 +00:00
/* Reset the retry count as we were successful in connecting */
evcon - > retry_cnt = 0 ;
2008-06-26 00:40:57 +00:00
evcon - > state = EVCON_IDLE ;
2008-04-29 04:52:50 +00:00
/* reset the bufferevent cbs */
bufferevent_setcb ( evcon - > bufev ,
2008-06-26 00:40:57 +00:00
evhttp_read_cb ,
2008-04-29 04:52:50 +00:00
evhttp_write_cb ,
evhttp_error_cb ,
evcon ) ;
2019-03-04 06:53:42 +03:00
bufferevent_set_timeouts ( evcon - > bufev ,
& evcon - > timeout_read , & evcon - > timeout_write ) ;
2006-07-17 00:33:57 +00:00
/* try to start requests that have queued up on this connection */
evhttp_request_dispatch ( evcon ) ;
2006-01-22 05:08:50 +00:00
return ;
cleanup :
2008-04-29 04:52:50 +00:00
evhttp_connection_cb_cleanup ( evcon ) ;
2006-01-22 05:08:50 +00:00
}
/*
* Check if we got a valid response code .
*/
2007-11-07 07:33:16 +00:00
static int
2006-01-22 05:08:50 +00:00
evhttp_valid_response_code ( int code )
{
if ( code = = 0 )
return ( 0 ) ;
return ( 1 ) ;
}
2010-11-09 10:14:32 -05:00
static int
evhttp_parse_http_version ( const char * version , struct evhttp_request * req )
{
2024-01-18 16:42:52 -05:00
char major , minor ;
2010-11-09 10:14:32 -05:00
char ch ;
2024-01-18 16:42:52 -05:00
int n = sscanf ( version , " HTTP/%c.%c%c " , & major , & minor , & ch ) ;
if ( n ! = 2 | | major > ' 1 ' | | major < ' 0 ' | | minor > ' 9 ' | | minor < ' 0 ' ) {
2010-11-09 10:14:32 -05:00
event_debug ( ( " %s: bad version %s on message %p from %s " ,
2022-08-28 15:27:04 +03:00
__func__ , version , ( void * ) req , req - > remote_host ) ) ;
2010-11-09 10:14:32 -05:00
return ( - 1 ) ;
}
2024-01-18 16:42:52 -05:00
req - > major = major - ' 0 ' ;
req - > minor = minor - ' 0 ' ;
2010-11-09 10:14:32 -05:00
return ( 0 ) ;
}
2006-01-22 05:08:50 +00:00
/* Parses the status line of a web server */
2007-11-07 07:33:16 +00:00
static int
2006-01-22 05:08:50 +00:00
evhttp_parse_response_line ( struct evhttp_request * req , char * line )
{
char * protocol ;
char * number ;
2010-05-02 12:51:35 +02:00
const char * readable = " " ;
2006-01-22 05:08:50 +00:00
protocol = strsep ( & line , " " ) ;
if ( line = = NULL )
return ( - 1 ) ;
number = strsep ( & line , " " ) ;
2010-04-28 21:33:13 -07:00
if ( line ! = NULL )
readable = line ;
2006-01-22 05:08:50 +00:00
2010-11-09 10:14:32 -05:00
if ( evhttp_parse_http_version ( protocol , req ) < 0 )
2006-01-22 05:08:50 +00:00
return ( - 1 ) ;
req - > response_code = atoi ( number ) ;
if ( ! evhttp_valid_response_code ( req - > response_code ) ) {
2007-09-02 01:33:38 +00:00
event_debug ( ( " %s: bad response code \" %s \" " ,
__func__ , number ) ) ;
2006-01-22 05:08:50 +00:00
return ( - 1 ) ;
}
2016-03-11 20:40:52 +03:00
if ( req - > response_code_line ! = NULL )
mm_free ( req - > response_code_line ) ;
2009-11-19 23:08:50 +00:00
if ( ( req - > response_code_line = mm_strdup ( readable ) ) = = NULL ) {
event_warn ( " %s: strdup " , __func__ ) ;
return ( - 1 ) ;
}
2006-01-22 05:08:50 +00:00
return ( 0 ) ;
}
/* Parse the first line of a HTTP request */
2007-11-07 07:33:16 +00:00
static int
2018-10-22 23:56:19 +03:00
evhttp_parse_request_line ( struct evhttp_request * req , char * line , size_t len )
2006-01-22 05:08:50 +00:00
{
2018-10-22 23:56:19 +03:00
char * eos = line + len ;
2006-01-22 05:08:50 +00:00
char * method ;
char * uri ;
char * version ;
2011-05-06 08:48:54 -04:00
size_t method_len ;
2019-04-08 22:27:33 +03:00
enum evhttp_cmd_type type = 0 ;
2006-01-22 05:08:50 +00:00
2018-10-22 23:56:19 +03:00
while ( eos > line & & * ( eos - 1 ) = = ' ' ) {
* ( eos - 1 ) = ' \0 ' ;
- - eos ;
- - len ;
}
2018-10-22 23:56:50 +03:00
if ( len < strlen ( " GET / HTTP/1.0 " ) )
return - 1 ;
2018-10-22 23:56:19 +03:00
2006-01-22 05:08:50 +00:00
/* Parse the request line */
method = strsep ( & line , " " ) ;
2018-10-22 23:52:46 +03:00
if ( ! line )
return - 1 ;
uri = line ;
version = strrchr ( uri , ' ' ) ;
if ( ! version | | uri = = version )
return - 1 ;
* version = ' \0 ' ;
version + + ;
2006-01-22 05:08:50 +00:00
2011-05-06 08:48:54 -04:00
method_len = ( uri - method ) - 1 ;
2006-01-22 05:08:50 +00:00
/* First line */
2011-05-06 08:48:54 -04:00
switch ( method_len ) {
case 3 :
/* The length of the method string is 3, meaning it can only be one of two methods: GET or PUT */
/* Since both GET and PUT share the same character 'T' at the end,
* if the string doesn ' t have ' T ' , we can immediately determine this
* is an invalid HTTP method */
if ( method [ 2 ] ! = ' T ' ) {
break ;
}
switch ( * method ) {
case ' G ' :
/* This first byte is 'G', so make sure the next byte is
* ' E ' , if it isn ' t then this isn ' t a valid method */
if ( method [ 1 ] = = ' E ' ) {
type = EVHTTP_REQ_GET ;
}
break ;
case ' P ' :
/* First byte is P, check second byte for 'U', if not,
* we know it ' s an invalid method */
if ( method [ 1 ] = = ' U ' ) {
type = EVHTTP_REQ_PUT ;
}
break ;
default :
break ;
}
break ;
case 4 :
2019-03-13 10:51:55 +03:00
/* The method length is 4 bytes, leaving only the methods POST, HEAD, LOCK, COPY and MOVE */
2011-05-06 08:48:54 -04:00
switch ( * method ) {
case ' P ' :
if ( method [ 3 ] = = ' T ' & & method [ 2 ] = = ' S ' & & method [ 1 ] = = ' O ' ) {
type = EVHTTP_REQ_POST ;
}
break ;
case ' H ' :
if ( method [ 3 ] = = ' D ' & & method [ 2 ] = = ' A ' & & method [ 1 ] = = ' E ' ) {
type = EVHTTP_REQ_HEAD ;
}
break ;
2019-03-13 10:51:55 +03:00
case ' L ' :
if ( method [ 3 ] = = ' K ' & & method [ 2 ] = = ' C ' & & method [ 1 ] = = ' O ' ) {
type = EVHTTP_REQ_LOCK ;
}
break ;
case ' C ' :
if ( method [ 3 ] = = ' Y ' & & method [ 2 ] = = ' P ' & & method [ 1 ] = = ' O ' ) {
type = EVHTTP_REQ_COPY ;
}
break ;
case ' M ' :
if ( method [ 3 ] = = ' E ' & & method [ 2 ] = = ' V ' & & method [ 1 ] = = ' O ' ) {
type = EVHTTP_REQ_MOVE ;
}
break ;
2011-05-06 08:48:54 -04:00
default :
break ;
}
2011-08-01 10:30:32 -04:00
break ;
2011-05-06 08:48:54 -04:00
case 5 :
2019-03-13 10:51:55 +03:00
/* Method length is 5 bytes, which can only encompass PATCH, TRACE and MKCOL */
2011-05-06 08:48:54 -04:00
switch ( * method ) {
case ' P ' :
if ( method [ 4 ] = = ' H ' & & method [ 3 ] = = ' C ' & & method [ 2 ] = = ' T ' & & method [ 1 ] = = ' A ' ) {
type = EVHTTP_REQ_PATCH ;
}
break ;
case ' T ' :
if ( method [ 4 ] = = ' E ' & & method [ 3 ] = = ' C ' & & method [ 2 ] = = ' A ' & & method [ 1 ] = = ' R ' ) {
type = EVHTTP_REQ_TRACE ;
}
2019-03-13 10:51:55 +03:00
break ;
case ' M ' :
if ( method [ 4 ] = = ' L ' & & method [ 3 ] = = ' O ' & & method [ 2 ] = = ' C ' & & method [ 1 ] = = ' K ' ) {
type = EVHTTP_REQ_MKCOL ;
}
2011-05-06 08:48:54 -04:00
break ;
default :
break ;
}
break ;
case 6 :
2019-03-13 10:51:55 +03:00
/* Method length is 6, only valid methods 6 bytes in length is DELETE and UNLOCK */
switch ( * method ) {
case ' D ' :
if ( method [ 5 ] = = ' E ' & & method [ 4 ] = = ' T ' & & method [ 3 ] = = ' E ' & &
method [ 2 ] = = ' L ' & & method [ 1 ] = = ' E ' ) {
type = EVHTTP_REQ_DELETE ;
}
break ;
case ' U ' :
if ( method [ 5 ] = = ' K ' & & method [ 4 ] = = ' C ' & & method [ 3 ] = = ' O ' & &
method [ 2 ] = = ' L ' & & method [ 1 ] = = ' N ' ) {
type = EVHTTP_REQ_UNLOCK ;
}
break ;
default :
break ;
2011-05-06 08:48:54 -04:00
}
break ;
case 7 :
/* Method length is 7, only valid methods are "OPTIONS" and "CONNECT" */
switch ( * method ) {
case ' O ' :
if ( method [ 6 ] = = ' S ' & & method [ 5 ] = = ' N ' & & method [ 4 ] = = ' O ' & &
method [ 3 ] = = ' I ' & & method [ 2 ] = = ' T ' & & method [ 1 ] = = ' P ' ) {
type = EVHTTP_REQ_OPTIONS ;
}
break ;
case ' C ' :
if ( method [ 6 ] = = ' T ' & & method [ 5 ] = = ' C ' & & method [ 4 ] = = ' E ' & &
method [ 3 ] = = ' N ' & & method [ 2 ] = = ' N ' & & method [ 1 ] = = ' O ' ) {
type = EVHTTP_REQ_CONNECT ;
}
break ;
default :
break ;
}
2019-03-13 10:51:55 +03:00
break ;
case 8 :
/* Method length is 8, only valid method 8 bytes in length is PROPFIND */
/* If the first byte isn't 'P' then it's invalid */
if ( * method ! = ' P ' ) {
break ;
}
if ( method [ 7 ] = = ' D ' & & method [ 6 ] = = ' N ' & & method [ 5 ] = = ' I ' & &
method [ 4 ] = = ' F ' & & method [ 3 ] = = ' P ' & & method [ 2 ] = = ' O ' & &
method [ 1 ] = = ' R ' ) {
type = EVHTTP_REQ_PROPFIND ;
}
break ;
case 9 :
/* Method length is 9, only valid method 9 bytes in length is PROPPATCH */
/* If the first byte isn't 'P' then it's invalid */
if ( * method ! = ' P ' ) {
break ;
}
if ( method [ 8 ] = = ' H ' & & method [ 7 ] = = ' C ' & & method [ 6 ] = = ' T ' & &
method [ 5 ] = = ' A ' & & method [ 4 ] = = ' P ' & & method [ 3 ] = = ' P ' & &
method [ 2 ] = = ' O ' & & method [ 1 ] = = ' R ' ) {
type = EVHTTP_REQ_PROPPATCH ;
}
2011-05-06 08:48:54 -04:00
break ;
} /* switch */
2016-01-08 13:36:20 -08:00
if ( ! type ) {
/* check extended methods, we only care about the
* type set by the cmp function if the cmp function
* returns a 0 value .
*/
struct evhttp_ext_method ext_method ;
ext_method . method = method ;
ext_method . type = 0 ;
2019-09-16 23:24:32 +08:00
ext_method . flags = 0 ;
2016-01-08 13:36:20 -08:00
if ( req - > evcon - > ext_method_cmp & &
req - > evcon - > ext_method_cmp ( & ext_method ) = = 0 ) {
2019-09-16 23:24:32 +08:00
/* make sure the other fields in ext_method are
2016-01-08 13:36:20 -08:00
* not changed by the callback .
*/
2019-09-16 23:24:32 +08:00
if ( strcmp ( ext_method . method , method ) ! = 0 ) {
event_warn ( " %s: modifying the 'method' field of ext_method_cmp's "
" parameter is not allowed " , __func__ ) ;
return - 1 ;
}
if ( ext_method . flags ! = 0 ) {
event_warn ( " %s: modifying the 'flags' field of ext_method_cmp's "
" parameter is not allowed " , __func__ ) ;
return - 1 ;
}
2016-01-08 13:36:20 -08:00
type = ext_method . type ;
}
}
2019-04-08 22:27:33 +03:00
if ( ! type ) {
event_debug ( ( " %s: bad method %s on request %p from %s " ,
2022-08-28 15:27:04 +03:00
__func__ , method , ( void * ) req , req - > remote_host ) ) ;
2016-01-08 13:36:20 -08:00
/* No error yet; we'll give a better error later when
* we see that req - > type is unsupported . */
2006-01-22 05:08:50 +00:00
}
2019-04-08 22:27:33 +03:00
2011-05-06 08:48:54 -04:00
req - > type = type ;
2006-01-22 05:08:50 +00:00
2010-11-09 10:14:32 -05:00
if ( evhttp_parse_http_version ( version , req ) < 0 )
2018-10-22 23:25:01 +03:00
return - 1 ;
2006-01-22 05:08:50 +00:00
2008-04-25 01:18:08 +00:00
if ( ( req - > uri = mm_strdup ( uri ) ) = = NULL ) {
2009-11-19 23:08:50 +00:00
event_debug ( ( " %s: mm_strdup " , __func__ ) ) ;
2018-10-22 23:25:01 +03:00
return - 1 ;
2006-01-22 05:08:50 +00:00
}
2017-12-02 12:53:57 -08:00
if ( type = = EVHTTP_REQ_CONNECT ) {
2020-10-27 01:40:34 +03:00
if ( ( req - > uri_elems = evhttp_uri_parse_authority ( req - > uri , 0 ) ) = = NULL ) {
2017-12-02 12:53:57 -08:00
return - 1 ;
}
} else {
if ( ( req - > uri_elems = evhttp_uri_parse_with_flags ( req - > uri ,
EVHTTP_URI_NONCONFORMANT ) ) = = NULL ) {
return - 1 ;
}
2010-11-05 11:17:07 -07:00
}
2018-10-22 23:25:01 +03:00
return 0 ;
2006-01-22 05:08:50 +00:00
}
2006-02-27 02:27:37 +00:00
const char *
2006-11-18 03:05:26 +00:00
evhttp_find_header ( const struct evkeyvalq * headers , const char * key )
2006-01-22 05:08:50 +00:00
{
struct evkeyval * header ;
TAILQ_FOREACH ( header , headers , next ) {
2009-07-28 19:41:57 +00:00
if ( evutil_ascii_strcasecmp ( header - > key , key ) = = 0 )
2006-01-22 05:08:50 +00:00
return ( header - > value ) ;
}
return ( NULL ) ;
}
void
evhttp_clear_headers ( struct evkeyvalq * headers )
{
struct evkeyval * header ;
for ( header = TAILQ_FIRST ( headers ) ;
header ! = NULL ;
header = TAILQ_FIRST ( headers ) ) {
TAILQ_REMOVE ( headers , header , next ) ;
2008-04-25 01:18:08 +00:00
mm_free ( header - > key ) ;
mm_free ( header - > value ) ;
mm_free ( header ) ;
2006-01-22 05:08:50 +00:00
}
}
2006-02-27 02:27:37 +00:00
/*
* Returns 0 , if the header was successfully removed .
* Returns - 1 , if the header could not be found .
*/
int
2006-01-22 05:08:50 +00:00
evhttp_remove_header ( struct evkeyvalq * headers , const char * key )
{
struct evkeyval * header ;
TAILQ_FOREACH ( header , headers , next ) {
2009-07-28 19:41:57 +00:00
if ( evutil_ascii_strcasecmp ( header - > key , key ) = = 0 )
2006-01-22 05:08:50 +00:00
break ;
}
if ( header = = NULL )
2006-02-27 02:27:37 +00:00
return ( - 1 ) ;
2006-01-22 05:08:50 +00:00
/* Free and remove the header that we found */
TAILQ_REMOVE ( headers , header , next ) ;
2008-04-25 01:18:08 +00:00
mm_free ( header - > key ) ;
mm_free ( header - > value ) ;
mm_free ( header ) ;
2006-02-27 02:27:37 +00:00
return ( 0 ) ;
2006-01-22 05:08:50 +00:00
}
2009-04-10 05:43:45 +00:00
static int
evhttp_header_is_valid_value ( const char * value )
{
const char * p = value ;
while ( ( p = strpbrk ( p , " \r \n " ) ) ! = NULL ) {
/* we really expect only one new line */
p + = strspn ( p , " \r \n " ) ;
/* we expect a space or tab for continuation */
if ( * p ! = ' ' & & * p ! = ' \t ' )
return ( 0 ) ;
}
return ( 1 ) ;
}
2006-01-22 05:08:50 +00:00
int
2007-07-30 21:27:33 +00:00
evhttp_add_header ( struct evkeyvalq * headers ,
const char * key , const char * value )
2006-01-22 05:08:50 +00:00
{
2007-08-06 20:53:33 +00:00
event_debug ( ( " %s: key: %s val: %s \n " , __func__ , key , value ) ) ;
2006-01-22 05:08:50 +00:00
2024-07-26 22:53:30 +02:00
/* RFC 9110 defines field-names as case-sensitive non-empty strings made of the following characters */
// field-name = 1*tchar
// tchar = "!" / "#" / "$" / "%" / "&" / "'" / "*" / "+" / "-" / "." / "^" / "_" / "`" / "|" / "~" / 0-9 / A-Z / a-z
/* For simplicity, we'll reject field-names containing the documented most dangerous characters */
// "Field values containing CR, LF, or NUL characters are invalid and dangerous, due to the varying ways that implementations might parse and interpret those characters; a recipient of CR, LF, or NUL within a field value MUST either reject the message or replace each of those characters with SP before further processing or forwarding of that message."
if ( strchr ( key , ' \r ' ) ! = NULL | | strchr ( key , ' \n ' ) ! = NULL | | key [ 0 ] = = ' \0 ' ) {
2007-07-30 21:27:33 +00:00
/* drop illegal headers */
2009-04-10 05:43:45 +00:00
event_debug ( ( " %s: dropping illegal header key \n " , __func__ ) ) ;
return ( - 1 ) ;
}
2010-02-18 17:46:56 -05:00
2009-04-10 05:43:45 +00:00
if ( ! evhttp_header_is_valid_value ( value ) ) {
2009-10-16 13:19:57 +00:00
event_debug ( ( " %s: dropping illegal header value \n " , __func__ ) ) ;
2007-07-30 21:27:33 +00:00
return ( - 1 ) ;
}
2009-04-10 05:43:45 +00:00
return ( evhttp_add_header_internal ( headers , key , value ) ) ;
}
static int
evhttp_add_header_internal ( struct evkeyvalq * headers ,
const char * key , const char * value )
{
struct evkeyval * header = mm_calloc ( 1 , sizeof ( struct evkeyval ) ) ;
2006-01-22 05:08:50 +00:00
if ( header = = NULL ) {
event_warn ( " %s: calloc " , __func__ ) ;
return ( - 1 ) ;
}
2008-04-25 01:18:08 +00:00
if ( ( header - > key = mm_strdup ( key ) ) = = NULL ) {
mm_free ( header ) ;
2006-01-22 05:08:50 +00:00
event_warn ( " %s: strdup " , __func__ ) ;
return ( - 1 ) ;
}
2008-04-25 01:18:08 +00:00
if ( ( header - > value = mm_strdup ( value ) ) = = NULL ) {
mm_free ( header - > key ) ;
mm_free ( header ) ;
2006-01-22 05:08:50 +00:00
event_warn ( " %s: strdup " , __func__ ) ;
return ( - 1 ) ;
}
TAILQ_INSERT_TAIL ( headers , header , next ) ;
return ( 0 ) ;
}
/*
* Parses header lines from a request or a response into the specified
* request object given an event buffer .
*
* Returns
2008-06-26 00:40:57 +00:00
* DATA_CORRUPTED on error
* MORE_DATA_EXPECTED when we need to read more headers
* ALL_DATA_READ when all headers have been read .
2006-01-22 05:08:50 +00:00
*/
2008-06-26 00:40:57 +00:00
enum message_read_status
2012-02-29 15:07:33 -05:00
evhttp_parse_firstline_ ( struct evhttp_request * req , struct evbuffer * buffer )
2008-06-26 00:40:57 +00:00
{
char * line ;
enum message_read_status status = ALL_DATA_READ ;
2018-10-22 23:25:01 +03:00
size_t len ;
2009-11-04 20:17:32 +00:00
/* XXX try */
2018-10-22 23:25:01 +03:00
line = evbuffer_readln ( buffer , & len , EVBUFFER_EOL_CRLF ) ;
2009-11-04 20:17:32 +00:00
if ( line = = NULL ) {
if ( req - > evcon ! = NULL & &
evbuffer_get_length ( buffer ) > req - > evcon - > max_headers_size )
return ( DATA_TOO_LONG ) ;
else
return ( MORE_DATA_EXPECTED ) ;
}
2018-10-22 23:25:01 +03:00
if ( req - > evcon ! = NULL & & len > req - > evcon - > max_headers_size ) {
2009-11-04 20:17:32 +00:00
mm_free ( line ) ;
return ( DATA_TOO_LONG ) ;
}
2018-10-22 23:25:01 +03:00
req - > headers_size = len ;
2008-06-26 00:40:57 +00:00
switch ( req - > kind ) {
case EVHTTP_REQUEST :
2018-10-22 23:56:19 +03:00
if ( evhttp_parse_request_line ( req , line , len ) = = - 1 )
2008-06-26 00:40:57 +00:00
status = DATA_CORRUPTED ;
break ;
case EVHTTP_RESPONSE :
if ( evhttp_parse_response_line ( req , line ) = = - 1 )
status = DATA_CORRUPTED ;
break ;
default :
status = DATA_CORRUPTED ;
}
mm_free ( line ) ;
return ( status ) ;
}
2008-06-29 01:30:06 +00:00
static int
2012-11-16 11:38:53 -05:00
evhttp_append_to_last_header ( struct evkeyvalq * headers , char * line )
2008-06-29 01:30:06 +00:00
{
struct evkeyval * header = TAILQ_LAST ( headers , evkeyvalq ) ;
char * newval ;
size_t old_len , line_len ;
if ( header = = NULL )
return ( - 1 ) ;
old_len = strlen ( header - > value ) ;
2012-11-16 11:38:53 -05:00
/* Strip space from start and end of line. */
while ( * line = = ' ' | | * line = = ' \t ' )
+ + line ;
evutil_rtrim_lws_ ( line ) ;
2008-06-29 01:30:06 +00:00
line_len = strlen ( line ) ;
2012-11-16 11:38:53 -05:00
newval = mm_realloc ( header - > value , old_len + line_len + 2 ) ;
2008-06-29 01:30:06 +00:00
if ( newval = = NULL )
return ( - 1 ) ;
2012-11-16 11:38:53 -05:00
newval [ old_len ] = ' ' ;
memcpy ( newval + old_len + 1 , line , line_len + 1 ) ;
2008-06-29 01:30:06 +00:00
header - > value = newval ;
return ( 0 ) ;
}
2008-06-26 00:40:57 +00:00
enum message_read_status
2012-02-29 15:07:33 -05:00
evhttp_parse_headers_ ( struct evhttp_request * req , struct evbuffer * buffer )
2006-01-22 05:08:50 +00:00
{
2009-11-04 20:17:32 +00:00
enum message_read_status errcode = DATA_CORRUPTED ;
2007-08-25 18:47:22 +00:00
char * line ;
2008-06-26 00:40:57 +00:00
enum message_read_status status = MORE_DATA_EXPECTED ;
2006-01-22 05:08:50 +00:00
struct evkeyvalq * headers = req - > input_headers ;
2018-10-22 23:25:01 +03:00
size_t len ;
while ( ( line = evbuffer_readln ( buffer , & len , EVBUFFER_EOL_CRLF ) )
2008-06-26 00:40:57 +00:00
! = NULL ) {
2006-01-22 05:08:50 +00:00
char * skey , * svalue ;
2018-10-22 23:25:01 +03:00
req - > headers_size + = len ;
2009-11-04 20:17:32 +00:00
if ( req - > evcon ! = NULL & &
req - > headers_size > req - > evcon - > max_headers_size ) {
errcode = DATA_TOO_LONG ;
goto error ;
}
2007-08-25 18:47:22 +00:00
if ( * line = = ' \0 ' ) { /* Last header - Done */
2008-06-26 00:40:57 +00:00
status = ALL_DATA_READ ;
2008-04-25 01:18:08 +00:00
mm_free ( line ) ;
2006-01-22 05:08:50 +00:00
break ;
}
2008-06-29 01:30:06 +00:00
/* Check if this is a continuation line */
if ( * line = = ' ' | | * line = = ' \t ' ) {
if ( evhttp_append_to_last_header ( headers , line ) = = - 1 )
goto error ;
2008-12-19 21:31:43 +00:00
mm_free ( line ) ;
2008-06-29 01:30:06 +00:00
continue ;
}
2006-01-22 05:08:50 +00:00
/* Processing of header lines */
2008-06-26 00:40:57 +00:00
svalue = line ;
skey = strsep ( & svalue , " : " ) ;
if ( svalue = = NULL )
goto error ;
2006-01-22 05:08:50 +00:00
2008-06-26 00:40:57 +00:00
svalue + = strspn ( svalue , " " ) ;
2012-11-16 11:29:34 -05:00
evutil_rtrim_lws_ ( svalue ) ;
2006-01-22 05:08:50 +00:00
2008-06-26 00:40:57 +00:00
if ( evhttp_add_header ( headers , skey , svalue ) = = - 1 )
goto error ;
2006-01-22 05:08:50 +00:00
2008-04-25 01:18:08 +00:00
mm_free ( line ) ;
2006-01-22 05:08:50 +00:00
}
2009-11-04 20:17:32 +00:00
if ( status = = MORE_DATA_EXPECTED ) {
2011-03-14 04:13:55 +01:00
if ( req - > evcon ! = NULL & &
req - > headers_size + evbuffer_get_length ( buffer ) > req - > evcon - > max_headers_size )
2009-11-04 20:17:32 +00:00
return ( DATA_TOO_LONG ) ;
}
2008-06-26 00:40:57 +00:00
return ( status ) ;
2007-08-25 18:47:22 +00:00
error :
2008-04-25 01:18:08 +00:00
mm_free ( line ) ;
2009-11-04 20:17:32 +00:00
return ( errcode ) ;
2006-01-22 05:08:50 +00:00
}
2006-12-18 15:26:19 +00:00
static int
evhttp_get_body_length ( struct evhttp_request * req )
2006-01-22 05:08:50 +00:00
{
2006-12-18 15:26:19 +00:00
struct evkeyvalq * headers = req - > input_headers ;
2006-02-27 02:27:37 +00:00
const char * content_length ;
const char * connection ;
2006-01-22 05:08:50 +00:00
content_length = evhttp_find_header ( headers , " Content-Length " ) ;
connection = evhttp_find_header ( headers , " Connection " ) ;
2009-01-27 21:10:31 +00:00
2006-01-22 05:08:50 +00:00
if ( content_length = = NULL & & connection = = NULL )
req - > ntoread = - 1 ;
else if ( content_length = = NULL & &
2009-07-28 19:41:57 +00:00
evutil_ascii_strcasecmp ( connection , " Close " ) ! = 0 ) {
2017-10-29 22:53:41 +03:00
req - > ntoread = 0 ;
2006-12-18 15:26:19 +00:00
} else if ( content_length = = NULL ) {
2006-01-22 05:08:50 +00:00
req - > ntoread = - 1 ;
2006-12-18 15:26:19 +00:00
} else {
char * endp ;
2008-07-02 04:22:48 +00:00
ev_int64_t ntoread = evutil_strtoll ( content_length , & endp , 10 ) ;
if ( * content_length = = ' \0 ' | | * endp ! = ' \0 ' | | ntoread < 0 ) {
event_debug ( ( " %s: illegal content length: %s " ,
__func__ , content_length ) ) ;
2006-12-18 15:26:19 +00:00
return ( - 1 ) ;
}
2008-07-02 04:22:48 +00:00
req - > ntoread = ntoread ;
2006-12-18 15:26:19 +00:00
}
2009-01-27 21:10:31 +00:00
2011-06-08 17:18:03 -04:00
event_debug ( ( " %s: bytes to read: " EV_I64_FMT " (in buffer " EV_SIZE_FMT " ) \n " ,
__func__ , EV_I64_ARG ( req - > ntoread ) ,
EV_SIZE_ARG ( evbuffer_get_length ( bufferevent_get_input ( req - > evcon - > bufev ) ) ) ) ) ;
2006-01-22 05:08:50 +00:00
2006-12-18 15:26:19 +00:00
return ( 0 ) ;
}
2010-11-23 20:31:28 -05:00
static int
2016-01-08 13:36:20 -08:00
evhttp_method_may_have_body_ ( struct evhttp_connection * evcon , enum evhttp_cmd_type type )
2010-11-23 20:31:28 -05:00
{
2022-07-09 17:17:38 +03:00
/* NOTE: some version of GCC reports a warning that flags may be uninitialized, hence assignment */
ev_uint16_t flags = 0 ;
2016-01-08 13:36:20 -08:00
evhttp_method_ ( evcon , type , & flags ) ;
return ( flags & EVHTTP_METHOD_HAS_BODY ) ? 1 : 0 ;
2010-11-23 20:31:28 -05:00
}
2006-12-18 15:26:19 +00:00
static void
evhttp_get_body ( struct evhttp_connection * evcon , struct evhttp_request * req )
{
const char * xfer_enc ;
2009-01-27 21:10:31 +00:00
2006-12-18 15:26:19 +00:00
/* If this is a request without a body, then we are done */
2008-02-25 07:49:22 +00:00
if ( req - > kind = = EVHTTP_REQUEST & &
2016-01-08 13:36:20 -08:00
! evhttp_method_may_have_body_ ( evcon , req - > type ) ) {
2006-07-17 00:33:57 +00:00
evhttp_connection_done ( evcon ) ;
2006-01-22 05:08:50 +00:00
return ;
}
2008-06-26 00:40:57 +00:00
evcon - > state = EVCON_READING_BODY ;
2006-12-18 15:26:19 +00:00
xfer_enc = evhttp_find_header ( req - > input_headers , " Transfer-Encoding " ) ;
2009-07-28 19:41:57 +00:00
if ( xfer_enc ! = NULL & & evutil_ascii_strcasecmp ( xfer_enc , " chunked " ) = = 0 ) {
2006-12-18 15:26:19 +00:00
req - > chunked = 1 ;
req - > ntoread = - 1 ;
} else {
if ( evhttp_get_body_length ( req ) = = - 1 ) {
2016-03-11 13:08:28 +03:00
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_INVALID_HEADER ) ;
2006-12-18 15:26:19 +00:00
return ;
}
2010-11-23 20:31:28 -05:00
if ( req - > kind = = EVHTTP_REQUEST & & req - > ntoread < 1 ) {
/* An incoming request with no content-length and no
* transfer - encoding has no body . */
evhttp_connection_done ( evcon ) ;
return ;
}
2006-12-18 15:26:19 +00:00
}
2010-11-29 18:25:04 -08:00
/* Should we send a 100 Continue status line? */
2016-03-11 13:08:28 +03:00
switch ( evhttp_have_expect ( req , 1 ) ) {
case CONTINUE :
2010-11-29 18:25:04 -08:00
/* XXX It would be nice to do some sanity
checking here . Does the resource exist ?
Should the resource accept post requests ? If
no , we should respond with an error . For
now , just optimistically tell the client to
send their message body . */
2011-05-25 14:31:09 -04:00
if ( req - > ntoread > 0 ) {
/* ntoread is ev_int64_t, max_body_size is ev_uint64_t */
2016-03-11 13:08:28 +03:00
if ( ( req - > evcon - > max_body_size < = EV_INT64_MAX ) & &
( ev_uint64_t ) req - > ntoread > req - > evcon - > max_body_size ) {
2016-02-15 03:26:40 +03:00
evhttp_lingering_fail ( evcon , req ) ;
2011-05-25 14:31:09 -04:00
return ;
}
2010-12-07 11:43:52 -05:00
}
2010-11-29 18:25:04 -08:00
if ( ! evbuffer_get_length ( bufferevent_get_input ( evcon - > bufev ) ) )
evhttp_send_continue ( evcon , req ) ;
2016-03-11 13:08:28 +03:00
break ;
case OTHER :
evhttp_send_error ( req , HTTP_EXPECTATIONFAILED , NULL ) ;
return ;
case NO : break ;
2010-11-29 18:25:04 -08:00
}
2007-01-04 18:05:17 +00:00
evhttp_read_body ( evcon , req ) ;
2008-06-02 05:45:26 +00:00
/* note the request may have been freed in evhttp_read_body */
2006-01-22 05:08:50 +00:00
}
2008-04-29 04:52:50 +00:00
static void
2008-06-26 00:40:57 +00:00
evhttp_read_firstline ( struct evhttp_connection * evcon ,
struct evhttp_request * req )
2006-01-22 05:08:50 +00:00
{
2008-06-26 00:40:57 +00:00
enum message_read_status res ;
2012-02-29 15:07:33 -05:00
res = evhttp_parse_firstline_ ( req , bufferevent_get_input ( evcon - > bufev ) ) ;
2009-11-04 20:17:32 +00:00
if ( res = = DATA_CORRUPTED | | res = = DATA_TOO_LONG ) {
2008-06-26 00:40:57 +00:00
/* Error while reading, terminate */
2012-11-02 11:44:29 -04:00
event_debug ( ( " %s: bad header lines on " EV_SOCK_FMT " \n " ,
2019-09-04 00:56:20 +03:00
__func__ , EV_SOCK_ARG ( bufferevent_getfd ( evcon - > bufev ) ) ) ) ;
2013-03-21 13:55:40 +04:00
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_INVALID_HEADER ) ;
2008-06-26 00:40:57 +00:00
return ;
} else if ( res = = MORE_DATA_EXPECTED ) {
/* Need more header lines */
return ;
}
2008-08-19 11:26:47 +00:00
evcon - > state = EVCON_READING_HEADERS ;
2008-06-26 00:40:57 +00:00
evhttp_read_header ( evcon , req ) ;
}
static void
evhttp_read_header ( struct evhttp_connection * evcon ,
struct evhttp_request * req )
{
enum message_read_status res ;
2019-09-04 00:56:20 +03:00
evutil_socket_t fd = bufferevent_getfd ( evcon - > bufev ) ;
2006-01-22 05:08:50 +00:00
2012-02-29 15:07:33 -05:00
res = evhttp_parse_headers_ ( req , bufferevent_get_input ( evcon - > bufev ) ) ;
2009-11-04 20:17:32 +00:00
if ( res = = DATA_CORRUPTED | | res = = DATA_TOO_LONG ) {
2006-01-22 05:08:50 +00:00
/* Error while reading, terminate */
2012-11-02 11:44:29 -04:00
event_debug ( ( " %s: bad header lines on " EV_SOCK_FMT " \n " ,
__func__ , EV_SOCK_ARG ( fd ) ) ) ;
2013-03-21 13:55:40 +04:00
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_INVALID_HEADER ) ;
2006-01-22 05:08:50 +00:00
return ;
2008-06-26 00:40:57 +00:00
} else if ( res = = MORE_DATA_EXPECTED ) {
2006-01-22 05:08:50 +00:00
/* Need more header lines */
return ;
}
2013-11-18 16:06:16 +01:00
/* Callback can shut down connection with negative return value */
if ( req - > header_cb ! = NULL ) {
if ( ( * req - > header_cb ) ( req , req - > cb_arg ) < 0 ) {
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_EOF ) ;
return ;
}
}
2006-01-22 05:08:50 +00:00
/* Done reading headers, do the real work */
switch ( req - > kind ) {
case EVHTTP_REQUEST :
2012-11-02 11:44:29 -04:00
event_debug ( ( " %s: checking for post data on " EV_SOCK_FMT " \n " ,
__func__ , EV_SOCK_ARG ( fd ) ) ) ;
2006-07-17 00:33:57 +00:00
evhttp_get_body ( evcon , req ) ;
2008-06-02 05:45:26 +00:00
/* note the request may have been freed in evhttp_get_body */
2006-01-22 05:08:50 +00:00
break ;
case EVHTTP_RESPONSE :
2010-11-29 18:25:04 -08:00
/* Start over if we got a 100 Continue response. */
if ( req - > response_code = = 100 ) {
2016-03-11 13:08:28 +03:00
struct evbuffer * output = bufferevent_get_output ( evcon - > bufev ) ;
evbuffer_add_buffer ( output , req - > output_buffer ) ;
evhttp_start_write_ ( evcon ) ;
2010-11-29 18:25:04 -08:00
return ;
}
2008-07-02 06:08:16 +00:00
if ( ! evhttp_response_needs_body ( req ) ) {
2006-12-06 04:12:11 +00:00
event_debug ( ( " %s: skipping body for code %d \n " ,
__func__ , req - > response_code ) ) ;
evhttp_connection_done ( evcon ) ;
} else {
2012-11-02 11:44:29 -04:00
event_debug ( ( " %s: start of read body for %s on "
EV_SOCK_FMT " \n " ,
__func__ , req - > remote_host , EV_SOCK_ARG ( fd ) ) ) ;
2006-12-06 04:12:11 +00:00
evhttp_get_body ( evcon , req ) ;
2008-06-02 05:45:26 +00:00
/* note the request may have been freed in
* evhttp_get_body */
2006-12-06 04:12:11 +00:00
}
2006-01-22 05:08:50 +00:00
break ;
default :
2012-11-01 17:38:34 -04:00
event_warnx ( " %s: bad header on " EV_SOCK_FMT , __func__ ,
EV_SOCK_ARG ( fd ) ) ;
2013-03-21 13:55:40 +04:00
evhttp_connection_fail_ ( evcon , EVREQ_HTTP_INVALID_HEADER ) ;
2006-01-22 05:08:50 +00:00
break ;
}
2008-06-02 05:45:26 +00:00
/* request may have been freed above */
2006-01-22 05:08:50 +00:00
}
/*
* Creates a TCP connection to the specified port and executes a callback
2009-10-16 13:19:57 +00:00
* when finished . Failure or success is indicate by the passed connection
2006-01-22 05:08:50 +00:00
* object .
2006-02-13 02:22:48 +00:00
*
* Although this interface accepts a hostname , it is intended to take
* only numeric hostnames so that non - blocking DNS resolution can
* happen elsewhere .
2006-01-22 05:08:50 +00:00
*/
2008-05-08 06:15:04 +00:00
struct evhttp_connection *
2014-12-13 19:42:42 +01:00
evhttp_connection_new ( const char * address , ev_uint16_t port )
2008-05-08 06:15:04 +00:00
{
2010-01-14 15:42:07 -08:00
return ( evhttp_connection_base_new ( NULL , NULL , address , port ) ) ;
2008-05-08 06:15:04 +00:00
}
2024-11-02 21:39:57 +00:00
/* We were passed a bev with file descriptor set.
* Assume that this is an already - open connection that we
* can start sending requests on .
*/
static int
evhttp_connection_set_existing_ ( struct evhttp_connection * evcon , struct bufferevent * bev )
{
evcon - > state = EVCON_IDLE ;
evcon - > flags | = EVHTTP_CON_OUTGOING ;
return 0 ;
}
2016-01-31 11:31:00 +00:00
static struct evhttp_connection *
evhttp_connection_new_ ( struct event_base * base , struct bufferevent * bev )
2006-01-22 05:08:50 +00:00
{
2016-01-31 11:31:00 +00:00
struct evhttp_connection * evcon ;
2006-01-22 05:08:50 +00:00
2008-04-25 01:18:08 +00:00
if ( ( evcon = mm_calloc ( 1 , sizeof ( struct evhttp_connection ) ) ) = = NULL ) {
2006-01-22 05:08:50 +00:00
event_warn ( " %s: calloc failed " , __func__ ) ;
goto error ;
}
2009-11-04 20:17:32 +00:00
evcon - > max_headers_size = EV_SIZE_MAX ;
evcon - > max_body_size = EV_SIZE_MAX ;
2019-03-04 06:53:42 +03:00
evcon - > timeout_connect . tv_sec = HTTP_CONNECT_TIMEOUT ;
evcon - > timeout_read . tv_sec = HTTP_READ_TIMEOUT ;
evcon - > timeout_write . tv_sec = HTTP_WRITE_TIMEOUT ;
evcon - > initial_retry_timeout . tv_sec = HTTP_INITIAL_RETRY_TIMEOUT ;
2006-12-09 01:41:57 +00:00
evcon - > retry_cnt = evcon - > retry_max = 0 ;
2006-11-22 06:54:28 +00:00
2011-09-12 10:46:17 -04:00
if ( bev = = NULL ) {
2019-09-04 00:56:20 +03:00
if ( ! ( bev = bufferevent_socket_new ( base , - 1 , BEV_OPT_CLOSE_ON_FREE ) ) ) {
2011-09-12 10:57:37 -04:00
event_warn ( " %s: bufferevent_socket_new failed " , __func__ ) ;
2011-09-12 10:46:17 -04:00
goto error ;
}
}
2011-09-12 10:57:37 -04:00
bufferevent_setcb ( bev , evhttp_read_cb , evhttp_write_cb , evhttp_error_cb , evcon ) ;
evcon - > bufev = bev ;
2006-07-17 00:33:57 +00:00
evcon - > state = EVCON_DISCONNECTED ;
TAILQ_INIT ( & evcon - > requests ) ;
2008-05-13 03:51:10 +00:00
if ( base ! = NULL ) {
evcon - > base = base ;
2011-09-12 10:57:37 -04:00
if ( bufferevent_get_base ( bev ) ! = base )
bufferevent_base_set ( base , evcon - > bufev ) ;
2008-05-13 03:51:10 +00:00
}
Restore our priority-inversion-prevention code with deferreds
Back when deferred_cb stuff had its own queue, the queue was always
executed, but we never ran more than 16 callbacks per iteration.
That made for two problems:
1: Because deferred_cb stuff would always run, and had no priority,
it could cause priority inversion.
2: It doesn't respect the max_dispatch_interval code.
Then, when I refactored deferred_cb to be a special case of
event_callback, that solved the above issues, but made for two more
issues:
3: Because deferred_cb stuff would always get the default priority,
it could could low-priority bufferevents to get too much priority.
4: With code like bufferevent_pair, it's easy to get into a
situation where two deferreds keep adding one another, preventing
the event loop from ever actually scanning for more events.
This commit fixes the above by giving deferreds a better notion of
priorities, and by limiting the number of deferreds that can be
added to the _current_ loop iteration's active queues. (Extra
deferreds are put into the active_later state.)
That isn't an all-purpose priority inversion solution, of course: for
that, you may need to mess around with max_dispatch_interval.
2012-05-09 11:06:06 -04:00
event_deferred_cb_init_ (
2012-04-06 04:33:19 -04:00
& evcon - > read_more_deferred_cb ,
Restore our priority-inversion-prevention code with deferreds
Back when deferred_cb stuff had its own queue, the queue was always
executed, but we never ran more than 16 callbacks per iteration.
That made for two problems:
1: Because deferred_cb stuff would always run, and had no priority,
it could cause priority inversion.
2: It doesn't respect the max_dispatch_interval code.
Then, when I refactored deferred_cb to be a special case of
event_callback, that solved the above issues, but made for two more
issues:
3: Because deferred_cb stuff would always get the default priority,
it could could low-priority bufferevents to get too much priority.
4: With code like bufferevent_pair, it's easy to get into a
situation where two deferreds keep adding one another, preventing
the event loop from ever actually scanning for more events.
This commit fixes the above by giving deferreds a better notion of
priorities, and by limiting the number of deferreds that can be
added to the _current_ loop iteration's active queues. (Extra
deferreds are put into the active_later state.)
That isn't an all-purpose priority inversion solution, of course: for
that, you may need to mess around with max_dispatch_interval.
2012-05-09 11:06:06 -04:00
bufferevent_get_priority ( bev ) ,
2010-10-25 21:53:15 -04:00
evhttp_deferred_read_cb , evcon ) ;
2014-03-21 17:32:09 +04:00
evcon - > ai_family = AF_UNSPEC ;
2010-01-14 15:42:07 -08:00
2006-07-17 00:33:57 +00:00
return ( evcon ) ;
2009-01-27 21:10:31 +00:00
2006-07-17 00:33:57 +00:00
error :
if ( evcon ! = NULL )
evhttp_connection_free ( evcon ) ;
return ( NULL ) ;
}
2024-11-02 21:39:57 +00:00
struct evhttp_connection *
evhttp_connection_base_bufferevent_reuse_new ( struct event_base * base , struct evdns_base * dnsbase , struct bufferevent * bev )
{
struct evhttp_connection * evcon = NULL ;
if ( bev = = NULL )
goto error ;
evcon = evhttp_connection_new_ ( base , bev ) ;
if ( evcon = = NULL )
goto error ;
if ( evhttp_connection_set_existing_ ( evcon , bev ) )
goto error ;
evcon - > dns_base = dnsbase ;
evcon - > address = NULL ;
evcon - > port = 0 ;
# ifndef _WIN32
evcon - > unixsocket = NULL ;
# endif
return ( evcon ) ;
error :
if ( evcon ! = NULL )
evhttp_connection_free ( evcon ) ;
return ( NULL ) ;
}
2016-01-31 11:31:00 +00:00
# ifndef _WIN32
struct evhttp_connection *
evhttp_connection_base_bufferevent_unix_new ( struct event_base * base , struct bufferevent * bev , const char * unixsocket )
{
struct evhttp_connection * evcon ;
if ( strlen ( unixsocket ) > = member_size ( struct sockaddr_un , sun_path ) ) {
event_warn ( " %s: unix socket too long " , __func__ ) ;
return NULL ;
}
evcon = evhttp_connection_new_ ( base , bev ) ;
if ( evcon = = NULL )
goto error ;
if ( ( evcon - > unixsocket = mm_strdup ( unixsocket ) ) = = NULL ) {
event_warn ( " %s: strdup failed " , __func__ ) ;
goto error ;
}
evcon - > ai_family = AF_UNIX ;
return ( evcon ) ;
error :
if ( evcon ! = NULL )
evhttp_connection_free ( evcon ) ;
return ( NULL ) ;
}
# endif
struct evhttp_connection *
evhttp_connection_base_bufferevent_new ( struct event_base * base , struct evdns_base * dnsbase , struct bufferevent * bev ,
const char * address , unsigned short port )
{
struct evhttp_connection * evcon ;
event_debug ( ( " Attempting connection to %s:%d \n " , address , port ) ) ;
evcon = evhttp_connection_new_ ( base , bev ) ;
if ( evcon = = NULL )
goto error ;
if ( ( evcon - > address = mm_strdup ( address ) ) = = NULL ) {
event_warn ( " %s: strdup failed " , __func__ ) ;
goto error ;
}
evcon - > port = port ;
evcon - > dns_base = dnsbase ;
return ( evcon ) ;
error :
if ( evcon ! = NULL )
evhttp_connection_free ( evcon ) ;
return ( NULL ) ;
}
2011-09-12 10:46:17 -04:00
struct bufferevent * evhttp_connection_get_bufferevent ( struct evhttp_connection * evcon )
{
return evcon - > bufev ;
}
2013-07-24 20:50:05 +00:00
struct evhttp *
evhttp_connection_get_server ( struct evhttp_connection * evcon )
{
return evcon - > http_server ;
}
2011-09-12 10:46:17 -04:00
struct evhttp_connection *
evhttp_connection_base_new ( struct event_base * base , struct evdns_base * dnsbase ,
2014-12-13 19:42:42 +01:00
const char * address , ev_uint16_t port )
2011-09-12 10:46:17 -04:00
{
return evhttp_connection_base_bufferevent_new ( base , dnsbase , NULL , address , port ) ;
}
2014-03-21 17:32:09 +04:00
void evhttp_connection_set_family ( struct evhttp_connection * evcon ,
int family )
{
evcon - > ai_family = family ;
}
2015-01-01 06:27:31 +03:00
int evhttp_connection_set_flags ( struct evhttp_connection * evcon ,
int flags )
{
2016-02-15 02:59:40 +03:00
int avail_flags = 0 ;
avail_flags | = EVHTTP_CON_REUSE_CONNECTED_ADDR ;
http: read server response even after server closed the connection
Otherwise if we will try to write more data than server can accept
(see `evhttp_set_max_body_size()` for libevent server) we will get `EPIPE` and
will not try to read server's response which must contain 400 error for now
(which is not strictly correct though, it must 413).
```
$ strace regress --no-fork http/data_length_constraints
...
connect(10, {sa_family=AF_INET, sin_port=htons(43988), sin_addr=inet_addr("127.0.0.1")}, 16) = -1 EINPROGRESS (Operation now in progress)
...
writev(10, [{"POST / HTTP/1.1\r\nHost: somehost\r"..., 60}, {"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16324}], 2) = 16384
epoll_wait(5, [{EPOLLOUT, {u32=10, u64=10}}, {EPOLLIN, {u32=11, u64=11}}], 32, 50000) = 2
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = 16384
ioctl(11, FIONREAD, [32768]) = 0
readv(11, [{"POST / HTTP/1.1\r\nHost: somehost\r"..., 4096}], 1) = 4096
epoll_ctl(5, EPOLL_CTL_DEL, 11, 0x7fff09d41e50) = 0
epoll_ctl(5, EPOLL_CTL_ADD, 11, {EPOLLOUT, {u32=11, u64=11}}) = 0
epoll_wait(5, [{EPOLLOUT, {u32=10, u64=10}}, {EPOLLOUT, {u32=11, u64=11}}], 32, 50000) = 2
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = 16384
writev(11, [{"HTTP/1.1 400 Bad Request\r\nConten"..., 129}, {"<HTML><HEAD>\n<TITLE>400 Bad Requ"..., 94}], 2) = 223
epoll_ctl(5, EPOLL_CTL_DEL, 11, 0x7fff09d42080) = 0
shutdown(11, SHUT_WR) = 0
close(11) = 0
epoll_wait(5, [{EPOLLOUT|EPOLLERR|EPOLLHUP, {u32=10, u64=10}}], 32, 50000) = 1
writev(10, [{"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"..., 16384}], 1) = -1 EPIPE (Broken pipe)
--- SIGPIPE {si_signo=SIGPIPE, si_code=SI_USER, si_pid=13954, si_uid=1000} ---
epoll_ctl(5, EPOLL_CTL_DEL, 10, 0x7fff09d42010) = 0
shutdown(10, SHUT_WR) = -1 ENOTCONN (Transport endpoint is not connected)
close(10) = 0
write(1, "\n FAIL ../test/regress_http.c:3"..., 37
```
Careful reader can ask why it send error even when it didn't read
`evcon->max_body_size`, and the answer will be checks for `evcon->max_body_size
against `Content-Length` header, which contains ~8MB (-2 bytes).
And also if we will not drain the output buffer than we will send buffer that
we didn't send in previous request and instead of sending method via
`evhttp_make_header()`.
Fixes: http/data_length_constraints
Refs: #321
v2: do this only under EVHTTP_CON_READ_ON_WRITE_ERROR flag
2016-02-10 14:43:18 +03:00
avail_flags | = EVHTTP_CON_READ_ON_WRITE_ERROR ;
2015-01-01 06:27:31 +03:00
2016-02-15 02:59:40 +03:00
if ( flags & ~ avail_flags | | flags > EVHTTP_CON_PUBLIC_FLAGS_END )
return 1 ;
evcon - > flags & = ~ avail_flags ;
2015-01-01 06:27:31 +03:00
2016-02-15 02:59:40 +03:00
evcon - > flags | = flags ;
2015-01-01 06:27:31 +03:00
return 0 ;
}
2016-01-08 13:36:20 -08:00
void
evhttp_connection_set_ext_method_cmp ( struct evhttp_connection * evcon ,
evhttp_ext_method_cb cmp )
{
evcon - > ext_method_cmp = cmp ;
}
2008-05-13 03:51:10 +00:00
void
evhttp_connection_set_base ( struct evhttp_connection * evcon ,
2007-08-19 02:41:23 +00:00
struct event_base * base )
{
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( evcon - > base = = NULL ) ;
EVUTIL_ASSERT ( evcon - > state = = EVCON_DISCONNECTED ) ;
2007-08-19 02:41:23 +00:00
evcon - > base = base ;
2008-04-29 04:52:50 +00:00
bufferevent_base_set ( base , evcon - > bufev ) ;
2007-08-19 02:41:23 +00:00
}
2006-11-22 06:54:28 +00:00
void
evhttp_connection_set_timeout ( struct evhttp_connection * evcon ,
2019-03-04 06:53:42 +03:00
int timeout )
2006-11-22 06:54:28 +00:00
{
2019-03-04 06:53:42 +03:00
if ( timeout ! = - 1 ) {
evcon - > flags | = EVHTTP_CON_TIMEOUT_ADJUSTED ;
} else {
evcon - > flags & = ~ EVHTTP_CON_TIMEOUT_ADJUSTED ;
2011-02-22 17:52:50 -05:00
}
2019-03-04 06:53:42 +03:00
evhttp_set_timeout_ ( & evcon - > timeout_read , timeout , HTTP_READ_TIMEOUT ) ;
evhttp_set_timeout_ ( & evcon - > timeout_write , timeout , HTTP_WRITE_TIMEOUT ) ;
bufferevent_set_timeouts ( evcon - > bufev ,
& evcon - > timeout_read , & evcon - > timeout_write ) ;
2011-02-22 17:52:50 -05:00
}
void
evhttp_connection_set_timeout_tv ( struct evhttp_connection * evcon ,
const struct timeval * tv )
{
if ( tv ) {
2019-03-04 06:53:42 +03:00
evcon - > flags | = EVHTTP_CON_TIMEOUT_ADJUSTED ;
2011-02-22 17:52:50 -05:00
} else {
2019-03-04 06:53:42 +03:00
evcon - > flags & = ~ EVHTTP_CON_TIMEOUT_ADJUSTED ;
2011-02-22 17:52:50 -05:00
}
2019-03-04 06:53:42 +03:00
evhttp_set_timeout_tv_ ( & evcon - > timeout_read , tv , HTTP_READ_TIMEOUT ) ;
evhttp_set_timeout_tv_ ( & evcon - > timeout_write , tv , HTTP_WRITE_TIMEOUT ) ;
bufferevent_set_timeouts ( evcon - > bufev ,
& evcon - > timeout_read , & evcon - > timeout_write ) ;
}
void evhttp_connection_set_connect_timeout_tv ( struct evhttp_connection * evcon ,
const struct timeval * tv )
{
evcon - > flags | = EVHTTP_CON_TIMEOUT_ADJUSTED ;
evhttp_set_timeout_tv_ ( & evcon - > timeout_connect , tv , - 1 ) ;
if ( evcon - > state = = EVCON_CONNECTING )
bufferevent_set_timeouts ( evcon - > bufev ,
& evcon - > timeout_connect , & evcon - > timeout_connect ) ;
}
void evhttp_connection_set_read_timeout_tv ( struct evhttp_connection * evcon ,
const struct timeval * tv )
{
evcon - > flags | = EVHTTP_CON_TIMEOUT_ADJUSTED ;
evhttp_set_timeout_tv_ ( & evcon - > timeout_read , tv , - 1 ) ;
if ( evcon - > state ! = EVCON_CONNECTING )
bufferevent_set_timeouts ( evcon - > bufev ,
& evcon - > timeout_read , & evcon - > timeout_write ) ;
}
void evhttp_connection_set_write_timeout_tv ( struct evhttp_connection * evcon ,
const struct timeval * tv )
{
evcon - > flags | = EVHTTP_CON_TIMEOUT_ADJUSTED ;
evhttp_set_timeout_tv_ ( & evcon - > timeout_write , tv , - 1 ) ;
if ( evcon - > state ! = EVCON_CONNECTING )
bufferevent_set_timeouts ( evcon - > bufev ,
& evcon - > timeout_read , & evcon - > timeout_write ) ;
2006-11-22 06:54:28 +00:00
}
2012-01-24 14:34:04 -05:00
void
evhttp_connection_set_initial_retry_tv ( struct evhttp_connection * evcon ,
const struct timeval * tv )
{
if ( tv ) {
evcon - > initial_retry_timeout = * tv ;
} else {
evutil_timerclear ( & evcon - > initial_retry_timeout ) ;
evcon - > initial_retry_timeout . tv_sec = 2 ;
}
}
2006-12-09 01:41:57 +00:00
void
evhttp_connection_set_retries ( struct evhttp_connection * evcon ,
int retry_max )
{
evcon - > retry_max = retry_max ;
}
2006-12-09 02:58:12 +00:00
void
evhttp_connection_set_closecb ( struct evhttp_connection * evcon ,
void ( * cb ) ( struct evhttp_connection * , void * ) , void * cbarg )
{
evcon - > closecb = cb ;
evcon - > closecb_arg = cbarg ;
}
void
evhttp_connection_get_peer ( struct evhttp_connection * evcon ,
2020-09-08 15:38:16 +02:00
const char * * address , ev_uint16_t * port )
2006-12-09 02:58:12 +00:00
{
* address = evcon - > address ;
* port = evcon - > port ;
}
http: implement new evhttp_connection_get_addr() api.
Basically tcp final handshake looks like this:
(C - client, S - server)
ACK[C] - FIN/ACK[S] - FIN/ACK[S] - ACK [C]
However there are servers, that didn't close connection like this,
while it is still _considered_ as valid, and using libevent http layer
we can do requests to such servers.
Modified handshake:
(C - client, S - server)
ACK[C] - RST/ACK[S] - RST/ACK[S]
And in this case we can't extract IP address from socket, because it is
already closed, and getpeername() will return: "transport endpoint is not connected".
So we need to store address that we are connecting to, after we know it,
and that is what this patch do.
I have reproduced it, however it have some extra packages.
(I will try to fix it)
https://github.com/azat/nfq-examples/blob/master/nfqnl_rst_fin.c
2013-10-01 19:12:13 +04:00
const struct sockaddr *
evhttp_connection_get_addr ( struct evhttp_connection * evcon )
{
2014-11-15 21:46:11 +03:00
return bufferevent_socket_get_conn_address_ ( evcon - > bufev ) ;
http: implement new evhttp_connection_get_addr() api.
Basically tcp final handshake looks like this:
(C - client, S - server)
ACK[C] - FIN/ACK[S] - FIN/ACK[S] - ACK [C]
However there are servers, that didn't close connection like this,
while it is still _considered_ as valid, and using libevent http layer
we can do requests to such servers.
Modified handshake:
(C - client, S - server)
ACK[C] - RST/ACK[S] - RST/ACK[S]
And in this case we can't extract IP address from socket, because it is
already closed, and getpeername() will return: "transport endpoint is not connected".
So we need to store address that we are connecting to, after we know it,
and that is what this patch do.
I have reproduced it, however it have some extra packages.
(I will try to fix it)
https://github.com/azat/nfq-examples/blob/master/nfqnl_rst_fin.c
2013-10-01 19:12:13 +04:00
}
2006-07-17 00:33:57 +00:00
int
2012-02-29 15:07:33 -05:00
evhttp_connection_connect_ ( struct evhttp_connection * evcon )
2006-07-17 00:33:57 +00:00
{
2013-02-14 09:54:56 -08:00
int old_state = evcon - > state ;
2014-11-12 20:23:46 +03:00
const char * address = evcon - > address ;
const struct sockaddr * sa = evhttp_connection_get_addr ( evcon ) ;
int ret ;
2013-02-14 09:54:56 -08:00
2006-07-17 00:33:57 +00:00
if ( evcon - > state = = EVCON_CONNECTING )
return ( 0 ) ;
2009-01-27 21:10:31 +00:00
2019-09-04 00:56:20 +03:00
/* Do not do hard reset, since this will reset the fd, but someone may
* change some options for it ( i . e . setsockopt ( ) , # 875 )
*
* However don ' t think that this options will be preserved for all
* connection lifetime , they will be reseted in the following cases :
* - evhttp_connection_set_local_address ( )
* - evhttp_connection_set_local_port ( )
* - evhttp_connection_set_retries ( )
* */
evhttp_connection_reset_ ( evcon , 0 ) ;
2006-07-17 00:33:57 +00:00
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( ! ( evcon - > flags & EVHTTP_CON_INCOMING ) ) ;
2006-07-17 00:33:57 +00:00
evcon - > flags | = EVHTTP_CON_OUTGOING ;
2009-01-27 21:10:31 +00:00
2013-01-23 02:45:32 +04:00
if ( evcon - > bind_address | | evcon - > bind_port ) {
2019-09-04 00:56:20 +03:00
int fd = bind_socket ( evcon - > bind_address , evcon - > bind_port ,
0 /*reuse*/ ) ;
if ( fd = = - 1 ) {
2013-01-23 02:45:32 +04:00
event_debug ( ( " %s: failed to bind to \" %s \" " ,
__func__ , evcon - > bind_address ) ) ;
return ( - 1 ) ;
}
2021-03-23 09:02:39 +03:00
if ( bufferevent_replacefd ( evcon - > bufev , fd ) )
2018-11-14 00:20:20 +03:00
return ( - 1 ) ;
2007-09-07 02:49:46 +00:00
}
2006-01-22 05:08:50 +00:00
/* Set up a callback for successful connection setup */
2009-01-27 21:10:31 +00:00
bufferevent_setcb ( evcon - > bufev ,
2008-04-29 04:52:50 +00:00
NULL /* evhttp_read_cb */ ,
2009-12-30 00:11:27 -05:00
NULL /* evhttp_write_cb */ ,
2008-04-29 04:52:50 +00:00
evhttp_connection_cb ,
2009-12-30 00:11:27 -05:00
evcon ) ;
2019-03-04 06:53:42 +03:00
bufferevent_set_timeouts ( evcon - > bufev ,
& evcon - > timeout_connect , & evcon - > timeout_connect ) ;
2008-04-29 04:52:50 +00:00
/* make sure that we get a write callback */
2018-11-14 00:20:20 +03:00
if ( bufferevent_enable ( evcon - > bufev , EV_WRITE ) )
return ( - 1 ) ;
2006-01-22 05:08:50 +00:00
2013-02-14 09:54:56 -08:00
evcon - > state = EVCON_CONNECTING ;
2015-01-01 06:27:31 +03:00
if ( evcon - > flags & EVHTTP_CON_REUSE_CONNECTED_ADDR & &
sa & &
( sa - > sa_family = = AF_INET | | sa - > sa_family = = AF_INET6 ) ) {
2015-08-19 23:00:49 +03:00
int socklen = sizeof ( struct sockaddr_in ) ;
if ( sa - > sa_family = = AF_INET6 ) {
2014-11-12 20:23:46 +03:00
socklen = sizeof ( struct sockaddr_in6 ) ;
}
ret = bufferevent_socket_connect ( evcon - > bufev , sa , socklen ) ;
2016-01-31 11:31:00 +00:00
}
# ifndef _WIN32
else if ( evcon - > unixsocket ) {
struct sockaddr_un sockaddr ;
sockaddr . sun_family = AF_UNIX ;
strcpy ( sockaddr . sun_path , evcon - > unixsocket ) ;
ret = bufferevent_socket_connect ( evcon - > bufev , ( const struct sockaddr * ) & sockaddr , sizeof ( sockaddr ) ) ;
}
# endif
else {
2014-11-12 20:23:46 +03:00
ret = bufferevent_socket_connect_hostname ( evcon - > bufev ,
evcon - > dns_base , evcon - > ai_family , address , evcon - > port ) ;
}
if ( ret < 0 ) {
2013-02-14 09:54:56 -08:00
evcon - > state = old_state ;
2019-09-04 00:56:20 +03:00
event_sock_warn ( bufferevent_getfd ( evcon - > bufev ) , " %s: connection to \" %s \" failed " ,
2009-12-30 00:11:27 -05:00
__func__ , evcon - > address ) ;
/* some operating systems return ECONNREFUSED immediately
* when connecting to a local address . the cleanup is going
* to reschedule this function call .
*/
evhttp_connection_cb_cleanup ( evcon ) ;
return ( 0 ) ;
}
2006-07-17 00:33:57 +00:00
return ( 0 ) ;
2006-01-22 05:08:50 +00:00
}
/*
2006-02-13 02:22:48 +00:00
* Starts an HTTP request on the provided evhttp_connection object .
2006-07-17 00:33:57 +00:00
* If the connection object is not connected to the web server already ,
* this will start the connection .
2006-01-22 05:08:50 +00:00
*/
int
2006-02-13 02:22:48 +00:00
evhttp_make_request ( struct evhttp_connection * evcon ,
2006-01-22 05:08:50 +00:00
struct evhttp_request * req ,
enum evhttp_cmd_type type , const char * uri )
{
/* We are making a request */
req - > kind = EVHTTP_REQUEST ;
req - > type = type ;
if ( req - > uri ! = NULL )
2008-04-25 01:18:08 +00:00
mm_free ( req - > uri ) ;
2009-11-19 23:08:50 +00:00
if ( ( req - > uri = mm_strdup ( uri ) ) = = NULL ) {
event_warn ( " %s: strdup " , __func__ ) ;
2015-09-09 19:21:51 +03:00
evhttp_request_free_auto ( req ) ;
2009-11-19 23:08:50 +00:00
return ( - 1 ) ;
}
2006-01-22 05:08:50 +00:00
/* Set the protocol version if it is not supplied */
if ( ! req - > major & & ! req - > minor ) {
req - > major = 1 ;
req - > minor = 1 ;
}
2009-01-27 21:10:31 +00:00
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( req - > evcon = = NULL ) ;
2006-07-17 00:33:57 +00:00
req - > evcon = evcon ;
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( ! ( req - > flags & EVHTTP_REQ_OWN_CONNECTION ) ) ;
2009-01-27 21:10:31 +00:00
2013-01-18 20:25:41 -08:00
TAILQ_INSERT_TAIL ( & evcon - > requests , req , next ) ;
2011-05-20 23:23:44 -04:00
http: fix connection retries when there more then one request for connection
We should not attemp to establishe the connection if there is retry
timer active, since otherwise there will be a bug.
Imagine next situation:
con = evhttp_connection_base_new()
evhttp_connection_set_retries(con, 2)
req = evhttp_request_new()
evhttp_make_request(con, req, ...)
# failed during connecting, and timer for 2 second scheduler (retry_ev)
Then another request scheduled for this evcon:
evhttp_make_request(con, req, ...)
# got request from server,
# and now it tries to read the response from the server
# (req.kind == EVHTTP_RESPONSE)
#
# but at this point retry_ev scheduled,
# and it schedules the connect again,
# and after the connect will succeeed, it will pick request with
# EVHTTP_RESPONSE for sending and this is completelly wrong and will
# fail in evhttp_make_header_response() since there is no
# "http_server" for this evcon
This was a long standing issue, that I came across few years ago
firstly, bad only now I had time to dig into it (but right now it was
pretty simple, by limiting amount of CPU for the process and using rr
for debug to go back and forth).
2018-12-07 21:46:27 +03:00
/* We do not want to conflict with retry_ev */
if ( evcon - > retry_cnt )
return ( 0 ) ;
2006-07-17 00:33:57 +00:00
/* If the connection object is not connected; make it so */
2011-03-31 19:11:10 +04:00
if ( ! evhttp_connected ( evcon ) ) {
2012-02-29 15:07:33 -05:00
int res = evhttp_connection_connect_ ( evcon ) ;
2013-01-18 20:25:41 -08:00
/* evhttp_connection_fail_(), which is called through
* evhttp_connection_connect_ ( ) , assumes that req lies in
* evcon - > requests . Thus , enqueue the request in advance and
* remove it in the error case . */
if ( res ! = 0 )
TAILQ_REMOVE ( & evcon - > requests , req , next ) ;
2011-03-31 19:11:10 +04:00
http: fix connection retries when there more then one request for connection
We should not attemp to establishe the connection if there is retry
timer active, since otherwise there will be a bug.
Imagine next situation:
con = evhttp_connection_base_new()
evhttp_connection_set_retries(con, 2)
req = evhttp_request_new()
evhttp_make_request(con, req, ...)
# failed during connecting, and timer for 2 second scheduler (retry_ev)
Then another request scheduled for this evcon:
evhttp_make_request(con, req, ...)
# got request from server,
# and now it tries to read the response from the server
# (req.kind == EVHTTP_RESPONSE)
#
# but at this point retry_ev scheduled,
# and it schedules the connect again,
# and after the connect will succeeed, it will pick request with
# EVHTTP_RESPONSE for sending and this is completelly wrong and will
# fail in evhttp_make_header_response() since there is no
# "http_server" for this evcon
This was a long standing issue, that I came across few years ago
firstly, bad only now I had time to dig into it (but right now it was
pretty simple, by limiting amount of CPU for the process and using rr
for debug to go back and forth).
2018-12-07 21:46:27 +03:00
return ( res ) ;
2011-03-31 19:11:10 +04:00
}
2006-07-17 00:33:57 +00:00
/*
* If it ' s connected already and we are the first in the queue ,
* then we can dispatch this request immediately . Otherwise , it
* will be dispatched once the pending requests are completed .
*/
if ( TAILQ_FIRST ( & evcon - > requests ) = = req )
evhttp_request_dispatch ( evcon ) ;
2006-01-22 05:08:50 +00:00
2006-07-17 00:33:57 +00:00
return ( 0 ) ;
2006-01-22 05:08:50 +00:00
}
2008-05-12 03:12:09 +00:00
void
evhttp_cancel_request ( struct evhttp_request * req )
{
struct evhttp_connection * evcon = req - > evcon ;
if ( evcon ! = NULL ) {
/* We need to remove it from the connection */
if ( TAILQ_FIRST ( & evcon - > requests ) = = req ) {
/* it's currently being worked on, so reset
* the connection .
*/
2012-02-29 15:07:33 -05:00
evhttp_connection_fail_ ( evcon ,
2013-03-21 13:55:40 +04:00
EVREQ_HTTP_REQUEST_CANCEL ) ;
2008-05-12 03:12:09 +00:00
/* connection fail freed the request */
return ;
} else {
/* otherwise, we can just remove it from the
* queue
*/
TAILQ_REMOVE ( & evcon - > requests , req , next ) ;
}
}
2015-09-09 19:21:51 +03:00
evhttp_request_free_auto ( req ) ;
2008-05-12 03:12:09 +00:00
}
2006-01-22 05:08:50 +00:00
/*
* Reads data from file descriptor into request structure
* Request structure needs to be set up correctly .
*/
void
2012-02-29 15:07:33 -05:00
evhttp_start_read_ ( struct evhttp_connection * evcon )
2006-01-22 05:08:50 +00:00
{
2008-04-29 04:52:50 +00:00
bufferevent_disable ( evcon - > bufev , EV_WRITE ) ;
bufferevent_enable ( evcon - > bufev , EV_READ ) ;
2015-09-08 16:40:55 +03:00
2008-06-26 00:40:57 +00:00
evcon - > state = EVCON_READING_FIRSTLINE ;
2010-11-04 14:05:08 -04:00
/* Reset the bufferevent callbacks */
bufferevent_setcb ( evcon - > bufev ,
evhttp_read_cb ,
evhttp_write_cb ,
evhttp_error_cb ,
evcon ) ;
2010-10-25 21:53:15 -04:00
/* If there's still data pending, process it next time through the
2024-04-30 15:59:58 +08:00
* loop . Don ' t do it now ; that could get recursive . */
2010-10-25 21:53:15 -04:00
if ( evbuffer_get_length ( bufferevent_get_input ( evcon - > bufev ) ) ) {
2012-02-29 15:07:33 -05:00
event_deferred_cb_schedule_ ( get_deferred_queue ( evcon ) ,
2010-10-25 21:53:15 -04:00
& evcon - > read_more_deferred_cb ) ;
}
2006-01-22 05:08:50 +00:00
}
2016-03-11 13:08:28 +03:00
void
evhttp_start_write_ ( struct evhttp_connection * evcon )
{
bufferevent_disable ( evcon - > bufev , EV_WRITE ) ;
bufferevent_enable ( evcon - > bufev , EV_READ ) ;
evcon - > state = EVCON_WRITING ;
evhttp_write_buffer ( evcon , evhttp_write_connectioncb , NULL ) ;
}
2007-11-07 07:33:16 +00:00
static void
2006-07-17 00:33:57 +00:00
evhttp_send_done ( struct evhttp_connection * evcon , void * arg )
2006-01-22 05:08:50 +00:00
{
2006-11-18 07:30:21 +00:00
int need_close ;
2006-07-17 00:33:57 +00:00
struct evhttp_request * req = TAILQ_FIRST ( & evcon - > requests ) ;
TAILQ_REMOVE ( & evcon - > requests , req , next ) ;
2014-01-05 20:35:46 -05:00
if ( req - > on_complete_cb ! = NULL ) {
req - > on_complete_cb ( req , req - > on_complete_cb_arg ) ;
}
2007-02-13 06:25:16 +00:00
need_close =
2010-11-02 15:19:12 -04:00
( REQ_VERSION_BEFORE ( req , 1 , 1 ) & &
2015-04-24 19:04:51 +03:00
! evhttp_is_connection_keepalive ( req - > input_headers ) ) | |
evhttp_is_request_connection_close ( req ) ;
2006-11-18 03:05:26 +00:00
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( req - > flags & EVHTTP_REQ_OWN_CONNECTION ) ;
2006-01-22 05:08:50 +00:00
evhttp_request_free ( req ) ;
2006-11-18 03:05:26 +00:00
if ( need_close ) {
evhttp_connection_free ( evcon ) ;
return ;
2009-01-27 21:10:31 +00:00
}
2006-11-18 03:05:26 +00:00
2006-11-23 06:32:20 +00:00
/* we have a persistent connection; try to accept another request. */
2008-12-19 21:31:43 +00:00
if ( evhttp_associate_new_request_with_connection ( evcon ) = = - 1 ) {
2006-11-18 03:05:26 +00:00
evhttp_connection_free ( evcon ) ;
2008-12-19 21:31:43 +00:00
}
2006-01-22 05:08:50 +00:00
}
/*
* Returns an error page .
*/
void
evhttp_send_error ( struct evhttp_request * req , int error , const char * reason )
{
2016-02-09 18:01:00 +00:00
# define ERR_FORMAT "<html><head>" \
" <title>%d %s</title> " \
" </head><body> " \
" <h1>%d %s</h1>%s " \
" </body></html> "
2006-01-22 05:08:50 +00:00
struct evbuffer * buf = evbuffer_new ( ) ;
2016-02-09 18:01:00 +00:00
struct evhttp * http = req - > evcon - > http_server ;
2009-11-19 23:08:50 +00:00
if ( buf = = NULL ) {
/* if we cannot allocate memory; we just drop the connection */
evhttp_connection_free ( req - > evcon ) ;
return ;
}
2006-11-18 08:51:12 +00:00
2012-02-29 15:07:33 -05:00
evhttp_response_code_ ( req , error , reason ) ;
2006-01-22 05:08:50 +00:00
2016-02-09 18:01:00 +00:00
/* Output error using callback for connection's evhttp, if available */
if ( ( http - > errorcb = = NULL ) | |
( ( * http - > errorcb ) ( req , buf , error , reason , http - > errorcbarg ) < 0 ) )
{
const char * heading = evhttp_response_phrase_internal ( error ) ;
evbuffer_drain ( buf , evbuffer_get_length ( buf ) ) ;
evbuffer_add_printf ( buf , ERR_FORMAT ,
error , heading , error , heading ,
( reason ? reason : " " ) ) ;
}
2006-01-22 05:08:50 +00:00
2012-02-29 15:07:33 -05:00
evhttp_send_page_ ( req , buf ) ;
2006-01-22 05:08:50 +00:00
evbuffer_free ( buf ) ;
2007-12-06 18:12:56 +00:00
# undef ERR_FORMAT
2006-01-22 05:08:50 +00:00
}
2016-02-09 18:01:00 +00:00
static void
evhttp_send_notfound ( struct evhttp_request * req , const char * url )
{
# define REASON_FORMAT "<p>The requested URL %s was not found on this server.< / p>"
char * escaped_url = NULL ;
char * reason = NULL ;
size_t reason_len ;
url = ( url ! = NULL ? url : req - > uri ) ;
if ( url ! = NULL )
escaped_url = evhttp_htmlescape ( url ) ;
if ( escaped_url ! = NULL ) {
reason_len = strlen ( REASON_FORMAT ) + strlen ( escaped_url ) + 1 ;
reason = mm_malloc ( reason_len ) ;
}
if ( ( escaped_url ! = NULL ) & & ( reason ! = NULL ) )
evutil_snprintf ( reason , reason_len , REASON_FORMAT , escaped_url ) ;
evhttp_send_error ( req , HTTP_NOTFOUND , reason ) ;
if ( reason ! = NULL )
mm_free ( reason ) ;
if ( escaped_url ! = NULL )
mm_free ( escaped_url ) ;
# undef REASON_FORMAT
}
2006-01-22 05:08:50 +00:00
/* Requires that headers and response code are already set up */
2007-06-16 03:23:15 +00:00
static inline void
2006-01-22 05:08:50 +00:00
evhttp_send ( struct evhttp_request * req , struct evbuffer * databuf )
{
2006-07-17 00:33:57 +00:00
struct evhttp_connection * evcon = req - > evcon ;
2006-01-22 05:08:50 +00:00
2010-02-03 16:54:18 -08:00
if ( evcon = = NULL ) {
evhttp_request_free ( req ) ;
return ;
}
2009-10-26 20:00:43 +00:00
EVUTIL_ASSERT ( TAILQ_FIRST ( & evcon - > requests ) = = req ) ;
2006-01-22 05:08:50 +00:00
2010-02-03 14:34:56 -08:00
/* we expect no more calls form the user on this request */
req - > userdone = 1 ;
2006-11-16 07:36:20 +00:00
/* xxx: not sure if we really should expose the data buffer this way */
2006-12-09 01:33:03 +00:00
if ( databuf ! = NULL )
evbuffer_add_buffer ( req - > output_buffer , databuf ) ;
2009-01-27 21:10:31 +00:00
2006-07-17 00:33:57 +00:00
/* Adds headers to the response */
evhttp_make_header ( evcon , req ) ;
2006-01-22 05:08:50 +00:00
2006-07-17 00:33:57 +00:00
evhttp_write_buffer ( evcon , evhttp_send_done , NULL ) ;
2006-01-22 05:08:50 +00:00
}
void
evhttp_send_reply ( struct evhttp_request * req , int code , const char * reason ,
struct evbuffer * databuf )
{
2012-02-29 15:07:33 -05:00
evhttp_response_code_ ( req , code , reason ) ;
2009-01-27 21:10:31 +00:00
2006-01-22 05:08:50 +00:00
evhttp_send ( req , databuf ) ;
}
2006-12-09 02:58:12 +00:00
void
evhttp_send_reply_start ( struct evhttp_request * req , int code ,
const char * reason )
{
2012-02-29 15:07:33 -05:00
evhttp_response_code_ ( req , code , reason ) ;
2017-09-26 19:23:29 +03:00
if ( req - > evcon = = NULL )
return ;
2008-07-02 06:08:16 +00:00
if ( evhttp_find_header ( req - > output_headers , " Content-Length " ) = = NULL & &
2010-11-02 15:19:12 -04:00
REQ_VERSION_ATLEAST ( req , 1 , 1 ) & &
2008-07-02 06:08:16 +00:00
evhttp_response_needs_body ( req ) ) {
/*
* prefer HTTP / 1.1 chunked encoding to closing the connection ;
* note RFC 2616 section 4.4 forbids it with Content - Length :
* and it ' s not necessary then anyway .
*/
2006-12-18 15:26:19 +00:00
evhttp_add_header ( req - > output_headers , " Transfer-Encoding " ,
" chunked " ) ;
req - > chunked = 1 ;
2010-11-02 13:50:57 -04:00
} else {
req - > chunked = 0 ;
2006-12-18 15:26:19 +00:00
}
2006-12-09 02:58:12 +00:00
evhttp_make_header ( req - > evcon , req ) ;
evhttp_write_buffer ( req - > evcon , NULL , NULL ) ;
}
void
2009-05-02 20:40:11 +02:00
evhttp_send_reply_chunk_with_cb ( struct evhttp_request * req , struct evbuffer * databuf ,
void ( * cb ) ( struct evhttp_connection * , void * ) , void * arg )
2006-12-09 02:58:12 +00:00
{
2010-02-03 14:34:56 -08:00
struct evhttp_connection * evcon = req - > evcon ;
struct evbuffer * output ;
if ( evcon = = NULL )
return ;
output = bufferevent_get_output ( evcon - > bufev ) ;
2009-04-17 06:56:09 +00:00
if ( evbuffer_get_length ( databuf ) = = 0 )
2008-07-02 06:08:16 +00:00
return ;
if ( ! evhttp_response_needs_body ( req ) )
return ;
2006-12-18 15:26:19 +00:00
if ( req - > chunked ) {
2008-04-29 04:52:50 +00:00
evbuffer_add_printf ( output , " %x \r \n " ,
2009-04-17 06:56:09 +00:00
( unsigned ) evbuffer_get_length ( databuf ) ) ;
2006-12-18 15:26:19 +00:00
}
2008-04-29 04:52:50 +00:00
evbuffer_add_buffer ( output , databuf ) ;
2008-02-16 16:56:34 +00:00
if ( req - > chunked ) {
2008-04-29 04:52:50 +00:00
evbuffer_add ( output , " \r \n " , 2 ) ;
2008-02-16 16:56:34 +00:00
}
2009-05-02 20:40:11 +02:00
evhttp_write_buffer ( evcon , cb , arg ) ;
2006-12-09 02:58:12 +00:00
}
2022-09-12 22:16:56 +03:00
struct bufferevent *
evhttp_start_ws_ ( struct evhttp_request * req )
{
struct evhttp_connection * evcon = req - > evcon ;
struct bufferevent * bufev ;
evhttp_response_code_ ( req , HTTP_SWITCH_PROTOCOLS , " Switching Protocols " ) ;
if ( req - > evcon = = NULL )
return NULL ;
evhttp_make_header ( req - > evcon , req ) ;
evhttp_write_buffer ( req - > evcon , NULL , NULL ) ;
TAILQ_REMOVE ( & evcon - > requests , req , next ) ;
bufev = evcon - > bufev ;
evcon - > bufev = NULL ;
evcon - > closecb = NULL ;
evhttp_request_free ( req ) ;
evhttp_connection_free ( evcon ) ;
return bufev ;
}
2009-05-02 20:40:11 +02:00
void
evhttp_send_reply_chunk ( struct evhttp_request * req , struct evbuffer * databuf )
{
evhttp_send_reply_chunk_with_cb ( req , databuf , NULL , NULL ) ;
}
2006-12-09 02:58:12 +00:00
void
2006-12-18 15:26:19 +00:00
evhttp_send_reply_end ( struct evhttp_request * req )
2006-12-09 02:58:12 +00:00
{
struct evhttp_connection * evcon = req - > evcon ;
2010-05-30 03:17:48 +02:00
struct evbuffer * output ;
2006-12-09 02:58:12 +00:00
2010-02-03 14:34:56 -08:00
if ( evcon = = NULL ) {
evhttp_request_free ( req ) ;
return ;
}
2010-05-30 03:17:48 +02:00
output = bufferevent_get_output ( evcon - > bufev ) ;
2010-02-03 14:34:56 -08:00
/* we expect no more calls form the user on this request */
req - > userdone = 1 ;
2006-12-18 15:26:19 +00:00
if ( req - > chunked ) {
2008-04-29 04:52:50 +00:00
evbuffer_add ( output , " 0 \r \n \r \n " , 5 ) ;
2006-12-18 15:26:19 +00:00
evhttp_write_buffer ( req - > evcon , evhttp_send_done , NULL ) ;
req - > chunked = 0 ;
2009-04-17 06:56:09 +00:00
} else if ( evbuffer_get_length ( output ) = = 0 ) {
2006-12-09 02:58:12 +00:00
/* let the connection know that we are done with the request */
evhttp_send_done ( evcon , NULL ) ;
} else {
/* make the callback execute after all data has been written */
evcon - > cb = evhttp_send_done ;
evcon - > cb_arg = NULL ;
}
}
2010-05-26 12:50:59 -04:00
static const char * informational_phrases [ ] = {
/* 100 */ " Continue " ,
/* 101 */ " Switching Protocols "
} ;
static const char * success_phrases [ ] = {
/* 200 */ " OK " ,
/* 201 */ " Created " ,
/* 202 */ " Accepted " ,
/* 203 */ " Non-Authoritative Information " ,
/* 204 */ " No Content " ,
/* 205 */ " Reset Content " ,
/* 206 */ " Partial Content "
} ;
static const char * redirection_phrases [ ] = {
/* 300 */ " Multiple Choices " ,
/* 301 */ " Moved Permanently " ,
/* 302 */ " Found " ,
/* 303 */ " See Other " ,
/* 304 */ " Not Modified " ,
/* 305 */ " Use Proxy " ,
/* 307 */ " Temporary Redirect "
} ;
static const char * client_error_phrases [ ] = {
/* 400 */ " Bad Request " ,
/* 401 */ " Unauthorized " ,
/* 402 */ " Payment Required " ,
/* 403 */ " Forbidden " ,
/* 404 */ " Not Found " ,
/* 405 */ " Method Not Allowed " ,
/* 406 */ " Not Acceptable " ,
/* 407 */ " Proxy Authentication Required " ,
/* 408 */ " Request Time-out " ,
/* 409 */ " Conflict " ,
/* 410 */ " Gone " ,
/* 411 */ " Length Required " ,
/* 412 */ " Precondition Failed " ,
/* 413 */ " Request Entity Too Large " ,
/* 414 */ " Request-URI Too Large " ,
/* 415 */ " Unsupported Media Type " ,
/* 416 */ " Requested range not satisfiable " ,
/* 417 */ " Expectation Failed "
} ;
static const char * server_error_phrases [ ] = {
/* 500 */ " Internal Server Error " ,
/* 501 */ " Not Implemented " ,
/* 502 */ " Bad Gateway " ,
/* 503 */ " Service Unavailable " ,
/* 504 */ " Gateway Time-out " ,
/* 505 */ " HTTP Version not supported "
} ;
struct response_class {
const char * name ;
size_t num_responses ;
const char * * responses ;
} ;
# ifndef MEMBERSOF
# define MEMBERSOF(x) (sizeof(x) / sizeof(x[0]))
# endif
static const struct response_class response_classes [ ] = {
/* 1xx */ { " Informational " , MEMBERSOF ( informational_phrases ) , informational_phrases } ,
/* 2xx */ { " Success " , MEMBERSOF ( success_phrases ) , success_phrases } ,
/* 3xx */ { " Redirection " , MEMBERSOF ( redirection_phrases ) , redirection_phrases } ,
/* 4xx */ { " Client Error " , MEMBERSOF ( client_error_phrases ) , client_error_phrases } ,
/* 5xx */ { " Server Error " , MEMBERSOF ( server_error_phrases ) , server_error_phrases }
} ;
static const char *
evhttp_response_phrase_internal ( int code )
{
int klass = code / 100 - 1 ;
int subcode = code % 100 ;
/* Unknown class - can't do any better here */
2010-09-23 22:45:55 -04:00
if ( klass < 0 | | klass > = ( int ) MEMBERSOF ( response_classes ) )
2010-05-26 12:50:59 -04:00
return " Unknown Status Class " ;
/* Unknown sub-code, return class name at least */
2010-09-23 22:45:55 -04:00
if ( subcode > = ( int ) response_classes [ klass ] . num_responses )
2010-05-26 12:50:59 -04:00
return response_classes [ klass ] . name ;
return response_classes [ klass ] . responses [ subcode ] ;
}
2006-01-22 05:08:50 +00:00
void
2012-02-29 15:07:33 -05:00
evhttp_response_code_ ( struct evhttp_request * req , int code , const char * reason )
2006-01-22 05:08:50 +00:00
{
req - > kind = EVHTTP_RESPONSE ;
req - > response_code = code ;
if ( req - > response_code_line ! = NULL )
2008-04-25 01:18:08 +00:00
mm_free ( req - > response_code_line ) ;
2010-05-26 12:50:59 -04:00
if ( reason = = NULL )
reason = evhttp_response_phrase_internal ( code ) ;
2008-04-25 01:18:08 +00:00
req - > response_code_line = mm_strdup ( reason ) ;
2010-12-18 01:07:27 -02:00
if ( req - > response_code_line = = NULL ) {
event_warn ( " %s: strdup " , __func__ ) ;
/* XXX what else can we do? */
}
2006-01-22 05:08:50 +00:00
}
void
2012-02-29 15:07:33 -05:00
evhttp_send_page_ ( struct evhttp_request * req , struct evbuffer * databuf )
2006-01-22 05:08:50 +00:00
{
2006-11-17 06:06:17 +00:00
if ( ! req - > major | | ! req - > minor ) {
req - > major = 1 ;
req - > minor = 1 ;
}
2009-01-27 21:10:31 +00:00
2006-01-22 05:08:50 +00:00
if ( req - > kind ! = EVHTTP_RESPONSE )
2012-02-29 15:07:33 -05:00
evhttp_response_code_ ( req , 200 , " OK " ) ;
2006-01-22 05:08:50 +00:00
evhttp_clear_headers ( req - > output_headers ) ;
evhttp_add_header ( req - > output_headers , " Content-Type " , " text/html " ) ;
evhttp_add_header ( req - > output_headers , " Connection " , " close " ) ;
evhttp_send ( req , databuf ) ;
}
2006-12-18 15:26:19 +00:00
static const char uri_chars [ 256 ] = {
2010-10-08 12:57:11 -04:00
/* 0 */
2006-12-18 15:26:19 +00:00
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
2010-10-08 12:57:11 -04:00
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 1 , 1 , 0 ,
1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 0 , 0 , 0 , 0 , 0 , 0 ,
2006-12-18 15:26:19 +00:00
/* 64 */
2010-10-08 12:57:11 -04:00
0 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 ,
2006-12-18 15:26:19 +00:00
1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 0 , 0 , 0 , 0 , 1 ,
0 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 ,
1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 0 , 0 , 0 , 1 , 0 ,
/* 128 */
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
/* 192 */
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
} ;
2010-10-19 11:26:59 -04:00
# define CHAR_IS_UNRESERVED(c) \
( uri_chars [ ( unsigned char ) ( c ) ] )
2006-12-18 15:26:19 +00:00
/*
2010-10-08 12:57:11 -04:00
* Helper functions to encode / decode a string for inclusion in a URI .
2006-12-18 15:26:19 +00:00
* The returned string must be freed by the caller .
*/
char *
2010-10-08 13:05:13 -04:00
evhttp_uriencode ( const char * uri , ev_ssize_t len , int space_as_plus )
2006-12-18 15:26:19 +00:00
{
struct evbuffer * buf = evbuffer_new ( ) ;
2010-10-08 13:05:13 -04:00
const char * p , * end ;
2018-01-09 21:44:57 +03:00
char * result = NULL ;
2006-12-18 15:26:19 +00:00
2018-01-09 21:44:57 +03:00
if ( ! buf ) {
goto out ;
2016-08-14 14:00:02 -07:00
}
2009-11-19 23:08:50 +00:00
2016-08-24 17:16:32 +08:00
if ( len > = 0 ) {
2016-08-14 14:00:02 -07:00
if ( uri + len < uri ) {
2018-01-09 21:44:57 +03:00
goto out ;
2016-08-14 14:00:02 -07:00
}
end = uri + len ;
} else {
size_t slen = strlen ( uri ) ;
if ( slen > = EV_SSIZE_MAX ) {
/* we don't want to mix signed and unsigned */
2018-01-09 21:44:57 +03:00
goto out ;
2016-08-14 14:00:02 -07:00
}
if ( uri + slen < uri ) {
2018-01-09 21:44:57 +03:00
goto out ;
2016-08-14 14:00:02 -07:00
}
end = uri + slen ;
}
2010-10-08 13:05:13 -04:00
for ( p = uri ; p < end ; p + + ) {
2010-10-19 11:26:59 -04:00
if ( CHAR_IS_UNRESERVED ( * p ) ) {
2006-12-18 15:26:19 +00:00
evbuffer_add ( buf , p , 1 ) ;
2010-10-08 13:05:13 -04:00
} else if ( * p = = ' ' & & space_as_plus ) {
evbuffer_add ( buf , " + " , 1 ) ;
2006-12-18 15:26:19 +00:00
} else {
2008-05-12 00:40:04 +00:00
evbuffer_add_printf ( buf , " %%%02X " , ( unsigned char ) ( * p ) ) ;
2006-12-18 15:26:19 +00:00
}
}
2016-08-14 14:00:02 -07:00
2010-10-08 13:05:13 -04:00
evbuffer_add ( buf , " " , 1 ) ; /* NUL-terminator. */
result = mm_malloc ( evbuffer_get_length ( buf ) ) ;
2016-08-14 14:00:02 -07:00
2012-07-17 11:08:25 -04:00
if ( result )
evbuffer_remove ( buf , result , evbuffer_get_length ( buf ) ) ;
2016-08-14 14:00:02 -07:00
2018-01-09 21:44:57 +03:00
out :
if ( buf )
evbuffer_free ( buf ) ;
return result ;
2010-10-08 13:05:13 -04:00
}
char *
evhttp_encode_uri ( const char * str )
{
return evhttp_uriencode ( str , - 1 , 0 ) ;
2006-12-18 15:26:19 +00:00
}
2009-04-10 05:43:45 +00:00
/*
2010-10-08 13:05:13 -04:00
* @ param decode_plus_ctl : if 1 , we decode plus into space . If 0 , we don ' t .
* If - 1 , when true we transform plus to space only after we ' ve seen
* a ? . - 1 is deprecated .
* @ return the number of bytes written to ' ret ' .
2009-04-10 05:43:45 +00:00
*/
2013-03-01 12:00:24 +04:00
int
2009-04-10 05:43:45 +00:00
evhttp_decode_uri_internal (
2010-10-08 13:05:13 -04:00
const char * uri , size_t length , char * ret , int decode_plus_ctl )
2006-12-09 01:33:03 +00:00
{
2008-05-10 07:32:05 +00:00
char c ;
2010-10-08 13:05:13 -04:00
int j ;
int decode_plus = ( decode_plus_ctl = = 1 ) ? 1 : 0 ;
2009-05-01 00:54:14 +00:00
unsigned i ;
2008-05-10 07:32:05 +00:00
for ( i = j = 0 ; i < length ; i + + ) {
2006-12-18 15:26:19 +00:00
c = uri [ i ] ;
2006-12-09 01:33:03 +00:00
if ( c = = ' ? ' ) {
2010-10-08 13:05:13 -04:00
if ( decode_plus_ctl < 0 )
decode_plus = 1 ;
} else if ( c = = ' + ' & & decode_plus ) {
2006-12-09 01:33:03 +00:00
c = ' ' ;
2013-02-28 23:10:02 +04:00
} else if ( ( i + 2 ) < length & & c = = ' % ' & &
2013-02-28 17:19:44 +04:00
EVUTIL_ISXDIGIT_ ( uri [ i + 1 ] ) & & EVUTIL_ISXDIGIT_ ( uri [ i + 2 ] ) ) {
2009-04-23 05:40:06 +00:00
char tmp [ 3 ] ;
tmp [ 0 ] = uri [ i + 1 ] ;
tmp [ 1 ] = uri [ i + 2 ] ;
tmp [ 2 ] = ' \0 ' ;
2006-12-09 01:33:03 +00:00
c = ( char ) strtol ( tmp , NULL , 16 ) ;
i + = 2 ;
}
ret [ j + + ] = c ;
}
ret [ j ] = ' \0 ' ;
2008-05-10 07:32:05 +00:00
return ( j ) ;
}
2010-10-08 13:05:13 -04:00
/* deprecated */
2008-05-10 07:32:05 +00:00
char *
evhttp_decode_uri ( const char * uri )
{
char * ret ;
2009-01-27 21:10:31 +00:00
2009-11-19 23:08:50 +00:00
if ( ( ret = mm_malloc ( strlen ( uri ) + 1 ) ) = = NULL ) {
event_warn ( " %s: malloc(%lu) " , __func__ ,
2008-05-10 07:32:05 +00:00
( unsigned long ) ( strlen ( uri ) + 1 ) ) ;
2009-11-19 23:08:50 +00:00
return ( NULL ) ;
}
2008-05-10 07:32:05 +00:00
2009-04-10 05:43:45 +00:00
evhttp_decode_uri_internal ( uri , strlen ( uri ) ,
2010-10-08 13:05:13 -04:00
ret , - 1 /*always_decode_plus*/ ) ;
return ( ret ) ;
}
char *
evhttp_uridecode ( const char * uri , int decode_plus , size_t * size_out )
{
char * ret ;
int n ;
if ( ( ret = mm_malloc ( strlen ( uri ) + 1 ) ) = = NULL ) {
event_warn ( " %s: malloc(%lu) " , __func__ ,
( unsigned long ) ( strlen ( uri ) + 1 ) ) ;
return ( NULL ) ;
}
n = evhttp_decode_uri_internal ( uri , strlen ( uri ) ,
ret , ! ! decode_plus /*always_decode_plus*/ ) ;
if ( size_out ) {
EVUTIL_ASSERT ( n > = 0 ) ;
* size_out = ( size_t ) n ;
}
2009-01-27 21:10:31 +00:00
2006-12-09 01:33:03 +00:00
return ( ret ) ;
}
2009-01-27 21:10:31 +00:00
/*
2006-01-22 05:08:50 +00:00
* Helper function to parse out arguments in a query .
* The arguments are separated by key and value .
*/
2010-10-26 10:38:30 -04:00
static int
evhttp_parse_query_impl ( const char * str , struct evkeyvalq * headers ,
2018-10-27 17:21:35 +03:00
int is_whole_uri , unsigned flags )
2006-01-22 05:08:50 +00:00
{
2010-10-19 13:15:48 -04:00
char * line = NULL ;
2006-01-22 05:08:50 +00:00
char * p ;
2010-10-19 13:15:48 -04:00
const char * query_part ;
2010-10-06 11:48:52 -04:00
int result = - 1 ;
2010-10-19 13:15:48 -04:00
struct evhttp_uri * uri = NULL ;
2006-01-22 05:08:50 +00:00
TAILQ_INIT ( headers ) ;
2010-10-19 13:15:48 -04:00
if ( is_whole_uri ) {
uri = evhttp_uri_parse ( str ) ;
if ( ! uri )
goto error ;
2010-10-21 14:41:12 -04:00
query_part = evhttp_uri_get_query ( uri ) ;
2010-10-19 13:15:48 -04:00
} else {
query_part = str ;
}
2006-02-13 02:22:48 +00:00
/* No arguments - we are done */
2010-10-19 13:15:48 -04:00
if ( ! query_part | | ! strlen ( query_part ) ) {
result = 0 ;
goto done ;
}
2006-01-22 05:08:50 +00:00
2010-10-19 13:15:48 -04:00
if ( ( line = mm_strdup ( query_part ) ) = = NULL ) {
2009-11-19 23:08:50 +00:00
event_warn ( " %s: strdup " , __func__ ) ;
2010-10-19 13:15:48 -04:00
goto error ;
2009-11-19 23:08:50 +00:00
}
2006-01-22 05:08:50 +00:00
2023-03-02 13:59:32 +08:00
p = line ;
2006-01-22 05:08:50 +00:00
while ( p ! = NULL & & * p ! = ' \0 ' ) {
2009-04-11 04:18:49 +00:00
char * key , * value , * decoded_value ;
2020-06-25 09:08:31 +03:00
int err ;
2006-01-22 05:08:50 +00:00
2023-03-02 13:59:32 +08:00
value = strsep ( & p , " & " ) ;
2006-01-22 05:08:50 +00:00
key = strsep ( & value , " = " ) ;
2018-10-27 17:21:35 +03:00
if ( flags & EVHTTP_URI_QUERY_NONCONFORMANT ) {
if ( value = = NULL )
2019-03-13 00:20:25 +03:00
value = ( char * ) " " ;
2018-10-27 17:21:35 +03:00
if ( * key = = ' \0 ' )
continue ;
} else {
if ( value = = NULL | | * key = = ' \0 ' )
goto error ;
2010-10-06 11:48:52 -04:00
}
2006-01-22 05:08:50 +00:00
2009-11-19 23:08:50 +00:00
if ( ( decoded_value = mm_malloc ( strlen ( value ) + 1 ) ) = = NULL ) {
event_warn ( " %s: mm_malloc " , __func__ ) ;
2010-10-06 11:48:52 -04:00
goto error ;
2009-11-19 23:08:50 +00:00
}
2009-04-11 04:18:49 +00:00
evhttp_decode_uri_internal ( value , strlen ( value ) ,
decoded_value , 1 /*always_decode_plus*/ ) ;
event_debug ( ( " Query Param: %s -> %s \n " , key , decoded_value ) ) ;
2018-10-27 17:21:35 +03:00
if ( flags & EVHTTP_URI_QUERY_LAST_VAL )
evhttp_remove_header ( headers , key ) ;
2020-06-25 09:08:31 +03:00
err = evhttp_add_header_internal ( headers , key , decoded_value ) ;
2009-04-11 04:18:49 +00:00
mm_free ( decoded_value ) ;
2020-06-25 09:08:31 +03:00
if ( err )
goto error ;
2006-01-22 05:08:50 +00:00
}
2010-10-06 11:48:52 -04:00
result = 0 ;
goto done ;
error :
evhttp_clear_headers ( headers ) ;
done :
2010-10-19 13:15:48 -04:00
if ( line )
mm_free ( line ) ;
if ( uri )
evhttp_uri_free ( uri ) ;
2010-10-06 11:48:52 -04:00
return result ;
2006-01-22 05:08:50 +00:00
}
2010-10-26 10:38:30 -04:00
int
2010-10-06 12:35:38 -04:00
evhttp_parse_query ( const char * uri , struct evkeyvalq * headers )
{
2018-10-27 17:21:35 +03:00
return evhttp_parse_query_impl ( uri , headers , 1 , 0 ) ;
2010-10-26 10:38:30 -04:00
}
int
evhttp_parse_query_str ( const char * uri , struct evkeyvalq * headers )
{
2018-10-27 17:21:35 +03:00
return evhttp_parse_query_impl ( uri , headers , 0 , 0 ) ;
}
int
evhttp_parse_query_str_flags ( const char * uri , struct evkeyvalq * headers , unsigned flags )
{
return evhttp_parse_query_impl ( uri , headers , 0 , flags ) ;
2010-10-06 12:35:38 -04:00
}
2007-03-06 06:26:10 +00:00
static struct evhttp_cb *
evhttp_dispatch_callback ( struct httpcbq * callbacks , struct evhttp_request * req )
2006-01-22 05:08:50 +00:00
{
struct evhttp_cb * cb ;
2007-08-05 02:15:10 +00:00
size_t offset = 0 ;
2008-05-10 07:32:05 +00:00
char * translated ;
2010-11-05 11:17:07 -07:00
const char * path ;
2010-12-09 11:43:12 -05:00
2006-01-22 05:08:50 +00:00
/* Test for different URLs */
2010-11-05 11:17:07 -07:00
path = evhttp_uri_get_path ( req - > uri_elems ) ;
offset = strlen ( path ) ;
2008-05-12 00:56:19 +00:00
if ( ( translated = mm_malloc ( offset + 1 ) ) = = NULL )
2008-05-10 07:32:05 +00:00
return ( NULL ) ;
2010-11-05 11:17:07 -07:00
evhttp_decode_uri_internal ( path , offset , translated ,
0 /* decode_plus */ ) ;
2007-08-05 02:15:10 +00:00
2007-03-06 06:26:10 +00:00
TAILQ_FOREACH ( cb , callbacks , next ) {
2010-11-05 11:17:07 -07:00
if ( ! strcmp ( cb - > what , translated ) ) {
2008-05-12 00:56:19 +00:00
mm_free ( translated ) ;
2007-03-06 06:26:10 +00:00
return ( cb ) ;
2008-05-12 00:56:19 +00:00
}
2007-03-06 06:26:10 +00:00
}
2008-05-12 00:56:19 +00:00
mm_free ( translated ) ;
2007-03-06 06:26:10 +00:00
return ( NULL ) ;
}
2008-05-10 05:58:17 +00:00
static int
prefix_suffix_match ( const char * pattern , const char * name , int ignorecase )
{
char c ;
2009-01-27 21:10:31 +00:00
2008-05-10 05:58:17 +00:00
while ( 1 ) {
switch ( c = * pattern + + ) {
case ' \0 ' :
return * name = = ' \0 ' ;
case ' * ' :
while ( * name ! = ' \0 ' ) {
if ( prefix_suffix_match ( pattern , name ,
ignorecase ) )
return ( 1 ) ;
+ + name ;
}
return ( 0 ) ;
default :
if ( c ! = * name ) {
if ( ! ignorecase | |
2012-02-29 15:07:33 -05:00
EVUTIL_TOLOWER_ ( c ) ! = EVUTIL_TOLOWER_ ( * name ) )
2008-05-10 05:58:17 +00:00
return ( 0 ) ;
}
+ + name ;
}
}
/* NOTREACHED */
}
2010-11-05 11:17:07 -07:00
/*
Search the vhost hierarchy beginning with http for a server alias
matching hostname . If a match is found , and outhttp is non - null ,
outhttp is set to the matching http object and 1 is returned .
*/
static int
evhttp_find_alias ( struct evhttp * http , struct evhttp * * outhttp ,
const char * hostname )
{
struct evhttp_server_alias * alias ;
struct evhttp * vhost ;
TAILQ_FOREACH ( alias , & http - > aliases , next ) {
/* XXX Do we need to handle IP addresses? */
if ( ! evutil_ascii_strcasecmp ( alias - > alias , hostname ) ) {
if ( outhttp )
* outhttp = http ;
return 1 ;
}
}
/* XXX It might be good to avoid recursion here, but I don't
see a way to do that w / o a list . */
TAILQ_FOREACH ( vhost , & http - > virtualhosts , next_vhost ) {
if ( evhttp_find_alias ( vhost , outhttp , hostname ) )
return 1 ;
}
return 0 ;
}
/*
Attempts to find the best http object to handle a request for a hostname .
All aliases for the root http object and vhosts are searched for an exact
match . Then , the vhost hierarchy is traversed again for a matching
pattern .
If an alias or vhost is matched , 1 is returned , and outhttp , if non - null ,
is set with the best matching http object . If there are no matches , the
root http object is stored in outhttp and 0 is returned .
*/
static int
evhttp_find_vhost ( struct evhttp * http , struct evhttp * * outhttp ,
const char * hostname )
{
struct evhttp * vhost ;
struct evhttp * oldhttp ;
int match_found = 0 ;
if ( evhttp_find_alias ( http , outhttp , hostname ) )
return 1 ;
do {
oldhttp = http ;
TAILQ_FOREACH ( vhost , & http - > virtualhosts , next_vhost ) {
if ( prefix_suffix_match ( vhost - > vhost_pattern ,
hostname , 1 /* ignorecase */ ) ) {
http = vhost ;
match_found = 1 ;
break ;
}
}
} while ( oldhttp ! = http ) ;
if ( outhttp )
* outhttp = http ;
return match_found ;
}
2007-11-07 07:33:16 +00:00
static void
2007-03-06 06:26:10 +00:00
evhttp_handle_request ( struct evhttp_request * req , void * arg )
{
struct evhttp * http = arg ;
struct evhttp_cb * cb = NULL ;
2008-05-05 07:17:05 +00:00
const char * hostname ;
2007-03-06 06:26:10 +00:00
2010-04-23 18:59:22 -07:00
/* we have a new request on which the user needs to take action */
req - > userdone = 0 ;
2017-11-04 19:13:28 +03:00
bufferevent_disable ( req - > evcon - > bufev , EV_READ ) ;
2019-04-08 22:27:33 +03:00
if ( req - > uri = = NULL ) {
2016-02-15 03:26:40 +03:00
evhttp_send_error ( req , req - > response_code , NULL ) ;
2007-03-06 06:26:10 +00:00
return ;
}
2010-11-04 11:25:35 -04:00
if ( ( http - > allowed_methods & req - > type ) = = 0 ) {
2010-11-04 11:53:34 -04:00
event_debug ( ( " Rejecting disallowed method %x (allowed: %x) \n " ,
( unsigned ) req - > type , ( unsigned ) http - > allowed_methods ) ) ;
evhttp_send_error ( req , HTTP_NOTIMPLEMENTED , NULL ) ;
2010-11-04 11:25:35 -04:00
return ;
}
2008-05-05 07:17:05 +00:00
/* handle potential virtual hosts */
2010-11-05 11:17:07 -07:00
hostname = evhttp_request_get_host ( req ) ;
2008-05-05 07:17:05 +00:00
if ( hostname ! = NULL ) {
2010-11-05 11:17:07 -07:00
evhttp_find_vhost ( http , & http , hostname ) ;
2008-05-05 07:17:05 +00:00
}
2007-03-06 06:26:10 +00:00
if ( ( cb = evhttp_dispatch_callback ( & http - > callbacks , req ) ) ! = NULL ) {
( * cb - > cb ) ( req , cb - > cbarg ) ;
return ;
2006-01-22 05:08:50 +00:00
}
/* Generic call back */
if ( http - > gencb ) {
( * http - > gencb ) ( req , http - > gencbarg ) ;
return ;
2016-02-09 18:01:00 +00:00
} else
evhttp_send_notfound ( req , NULL ) ;
2006-01-22 05:08:50 +00:00
}
2010-08-09 13:25:50 -04:00
/* Listener callback when a connection arrives at a server. */
2006-01-22 05:08:50 +00:00
static void
2009-12-30 00:41:03 -05:00
accept_socket_cb ( struct evconnlistener * listener , evutil_socket_t nfd , struct sockaddr * peer_sa , int peer_socklen , void * arg )
2006-01-22 05:08:50 +00:00
{
2022-08-01 10:16:18 +02:00
struct evhttp_bound_socket * bound = arg ;
struct evhttp * http = bound - > http ;
2007-02-09 07:49:55 +00:00
2022-08-01 10:16:18 +02:00
struct bufferevent * bev = NULL ;
if ( bound - > bevcb )
bev = bound - > bevcb ( http - > base , bound - > bevcbarg ) ;
evhttp_get_request ( http , nfd , peer_sa , peer_socklen , bev ) ;
2006-01-22 05:08:50 +00:00
}
2007-08-19 02:41:23 +00:00
int
2008-05-12 00:40:04 +00:00
evhttp_bind_socket ( struct evhttp * http , const char * address , ev_uint16_t port )
2009-08-16 19:22:04 +00:00
{
struct evhttp_bound_socket * bound =
evhttp_bind_socket_with_handle ( http , address , port ) ;
if ( bound = = NULL )
return ( - 1 ) ;
return ( 0 ) ;
}
struct evhttp_bound_socket *
evhttp_bind_socket_with_handle ( struct evhttp * http , const char * address , ev_uint16_t port )
2006-01-22 05:08:50 +00:00
{
2007-11-25 21:53:06 +00:00
evutil_socket_t fd ;
2009-08-16 19:22:15 +00:00
struct evhttp_bound_socket * bound ;
2018-12-28 04:42:20 +00:00
int serrno ;
2006-01-22 05:08:50 +00:00
2008-06-20 06:52:13 +00:00
if ( ( fd = bind_socket ( address , port , 1 /*reuse*/ ) ) = = - 1 )
2009-08-16 19:22:04 +00:00
return ( NULL ) ;
2006-01-22 05:08:50 +00:00
2008-02-26 03:49:00 +00:00
if ( listen ( fd , 128 ) = = - 1 ) {
2018-12-28 04:42:20 +00:00
serrno = EVUTIL_SOCKET_ERROR ( ) ;
2008-09-05 16:29:56 +00:00
event_sock_warn ( fd , " %s: listen " , __func__ ) ;
2010-04-14 15:42:57 -04:00
evutil_closesocket ( fd ) ;
2018-12-28 04:42:20 +00:00
EVUTIL_SET_SOCKET_ERROR ( serrno ) ;
2009-08-16 19:22:04 +00:00
return ( NULL ) ;
2006-01-22 05:08:50 +00:00
}
2009-08-16 19:22:15 +00:00
bound = evhttp_accept_socket_with_handle ( http , fd ) ;
2009-01-27 21:10:31 +00:00
2009-08-16 19:22:04 +00:00
if ( bound ! = NULL ) {
2008-02-26 03:12:07 +00:00
event_debug ( ( " Bound to port %d - Awaiting connections ... " ,
port ) ) ;
2009-08-16 19:22:04 +00:00
return ( bound ) ;
}
2008-02-26 03:12:07 +00:00
2009-08-16 19:22:04 +00:00
return ( NULL ) ;
2008-02-26 03:12:07 +00:00
}
int
evhttp_accept_socket ( struct evhttp * http , evutil_socket_t fd )
2009-08-16 19:22:04 +00:00
{
struct evhttp_bound_socket * bound =
evhttp_accept_socket_with_handle ( http , fd ) ;
if ( bound = = NULL )
return ( - 1 ) ;
return ( 0 ) ;
}
2011-10-24 01:31:55 +00:00
void
evhttp_foreach_bound_socket ( struct evhttp * http ,
evhttp_bound_socket_foreach_fn * function ,
void * argument )
{
struct evhttp_bound_socket * bound ;
TAILQ_FOREACH ( bound , & http - > sockets , next )
function ( bound , argument ) ;
}
2009-08-16 19:22:04 +00:00
struct evhttp_bound_socket *
evhttp_accept_socket_with_handle ( struct evhttp * http , evutil_socket_t fd )
2008-02-26 03:12:07 +00:00
{
2008-05-04 22:21:29 +00:00
struct evhttp_bound_socket * bound ;
2009-12-30 00:41:03 -05:00
struct evconnlistener * listener ;
const int flags =
LEV_OPT_REUSEABLE | LEV_OPT_CLOSE_ON_EXEC | LEV_OPT_CLOSE_ON_FREE ;
2008-05-04 22:21:29 +00:00
2010-10-25 11:50:51 -04:00
listener = evconnlistener_new ( http - > base , NULL , NULL ,
2009-12-30 00:41:03 -05:00
flags ,
0 , /* Backlog is '0' because we already said 'listen' */
fd ) ;
2010-10-25 11:50:51 -04:00
if ( ! listener )
return ( NULL ) ;
bound = evhttp_bind_listener ( http , listener ) ;
if ( ! bound ) {
evconnlistener_free ( listener ) ;
2009-08-16 19:22:04 +00:00
return ( NULL ) ;
2008-05-04 22:21:29 +00:00
}
2010-10-25 11:50:51 -04:00
return ( bound ) ;
}
struct evhttp_bound_socket *
evhttp_bind_listener ( struct evhttp * http , struct evconnlistener * listener )
{
struct evhttp_bound_socket * bound ;
bound = mm_malloc ( sizeof ( struct evhttp_bound_socket ) ) ;
if ( bound = = NULL )
return ( NULL ) ;
2008-05-04 22:21:29 +00:00
2010-10-25 11:50:51 -04:00
bound - > listener = listener ;
2022-08-01 10:16:18 +02:00
bound - > bevcb = NULL ;
bound - > http = http ;
2008-05-04 22:21:29 +00:00
TAILQ_INSERT_TAIL ( & http - > sockets , bound , next ) ;
2008-05-02 16:28:25 +00:00
2022-08-01 10:16:18 +02:00
evconnlistener_set_cb ( listener , accept_socket_cb , bound ) ;
2010-10-25 11:50:51 -04:00
return bound ;
2006-01-22 05:08:50 +00:00
}
2010-11-05 11:17:07 -07:00
evutil_socket_t
evhttp_bound_socket_get_fd ( struct evhttp_bound_socket * bound )
2009-08-16 19:21:57 +00:00
{
2009-12-30 00:41:03 -05:00
return evconnlistener_get_fd ( bound - > listener ) ;
2009-08-16 19:21:57 +00:00
}
2010-10-25 11:50:51 -04:00
struct evconnlistener *
evhttp_bound_socket_get_listener ( struct evhttp_bound_socket * bound )
{
return bound - > listener ;
}
2022-08-01 10:16:18 +02:00
void
evhttp_bound_set_bevcb ( struct evhttp_bound_socket * bound ,
struct bufferevent * ( * cb ) ( struct event_base * , void * ) , void * cbarg )
{
bound - > bevcb = cb ;
bound - > bevcbarg = cbarg ;
}
2009-08-16 19:22:10 +00:00
void
evhttp_del_accept_socket ( struct evhttp * http , struct evhttp_bound_socket * bound )
{
TAILQ_REMOVE ( & http - > sockets , bound , next ) ;
2009-12-30 00:41:03 -05:00
evconnlistener_free ( bound - > listener ) ;
2009-08-16 19:22:10 +00:00
mm_free ( bound ) ;
}
2007-08-19 02:41:23 +00:00
static struct evhttp *
2007-11-07 07:33:16 +00:00
evhttp_new_object ( void )
2006-01-22 05:08:50 +00:00
{
2007-08-19 02:41:23 +00:00
struct evhttp * http = NULL ;
2006-01-22 05:08:50 +00:00
2008-04-25 01:18:08 +00:00
if ( ( http = mm_calloc ( 1 , sizeof ( struct evhttp ) ) ) = = NULL ) {
2006-01-22 05:08:50 +00:00
event_warn ( " %s: calloc " , __func__ ) ;
return ( NULL ) ;
}
2019-03-04 06:53:42 +03:00
evutil_timerclear ( & http - > timeout_read ) ;
evutil_timerclear ( & http - > timeout_write ) ;
2009-11-04 20:17:32 +00:00
evhttp_set_max_headers_size ( http , EV_SIZE_MAX ) ;
evhttp_set_max_body_size ( http , EV_SIZE_MAX ) ;
2013-09-28 20:03:28 +02:00
evhttp_set_default_content_type ( http , " text/html; charset=ISO-8859-1 " ) ;
2010-12-09 11:47:54 -05:00
evhttp_set_allowed_methods ( http ,
EVHTTP_REQ_GET |
EVHTTP_REQ_POST |
EVHTTP_REQ_HEAD |
EVHTTP_REQ_PUT |
EVHTTP_REQ_DELETE ) ;
2006-11-23 06:32:20 +00:00
2008-05-04 22:21:29 +00:00
TAILQ_INIT ( & http - > sockets ) ;
2006-01-22 05:08:50 +00:00
TAILQ_INIT ( & http - > callbacks ) ;
2006-11-18 03:05:26 +00:00
TAILQ_INIT ( & http - > connections ) ;
2022-09-12 22:16:56 +03:00
TAILQ_INIT ( & http - > ws_sessions ) ;
2008-05-05 07:17:05 +00:00
TAILQ_INIT ( & http - > virtualhosts ) ;
2010-11-05 11:17:07 -07:00
TAILQ_INIT ( & http - > aliases ) ;
2006-01-22 05:08:50 +00:00
2007-08-19 02:41:23 +00:00
return ( http ) ;
}
struct evhttp *
evhttp_new ( struct event_base * base )
{
2011-05-27 18:40:31 -04:00
struct evhttp * http = NULL ;
2007-08-19 02:41:23 +00:00
2011-05-27 18:40:31 -04:00
http = evhttp_new_object ( ) ;
if ( http = = NULL )
return ( NULL ) ;
2007-08-19 02:41:23 +00:00
http - > base = base ;
return ( http ) ;
}
/*
* Start a web server on the specified address and port .
*/
struct evhttp *
2014-12-13 19:42:42 +01:00
evhttp_start ( const char * address , ev_uint16_t port )
2007-08-19 02:41:23 +00:00
{
2011-05-27 18:40:31 -04:00
struct evhttp * http = NULL ;
2007-08-19 02:41:23 +00:00
2011-05-27 18:40:31 -04:00
http = evhttp_new_object ( ) ;
if ( http = = NULL )
return ( NULL ) ;
2007-08-19 02:41:23 +00:00
if ( evhttp_bind_socket ( http , address , port ) = = - 1 ) {
2008-04-25 01:18:08 +00:00
mm_free ( http ) ;
2006-01-22 05:08:50 +00:00
return ( NULL ) ;
}
return ( http ) ;
}
void
evhttp_free ( struct evhttp * http )
{
struct evhttp_cb * http_cb ;
2006-11-18 03:05:26 +00:00
struct evhttp_connection * evcon ;
2022-09-12 22:16:56 +03:00
struct evws_connection * evws ;
2008-05-04 22:21:29 +00:00
struct evhttp_bound_socket * bound ;
2008-05-05 07:17:05 +00:00
struct evhttp * vhost ;
2010-11-05 11:17:07 -07:00
struct evhttp_server_alias * alias ;
2006-01-22 05:08:50 +00:00
/* Remove the accepting part */
2008-05-04 22:21:29 +00:00
while ( ( bound = TAILQ_FIRST ( & http - > sockets ) ) ! = NULL ) {
TAILQ_REMOVE ( & http - > sockets , bound , next ) ;
2009-12-30 00:41:03 -05:00
evconnlistener_free ( bound - > listener ) ;
2008-05-04 22:21:29 +00:00
mm_free ( bound ) ;
}
2006-01-22 05:08:50 +00:00
2006-11-18 03:05:26 +00:00
while ( ( evcon = TAILQ_FIRST ( & http - > connections ) ) ! = NULL ) {
/* evhttp_connection_free removes the connection */
evhttp_connection_free ( evcon ) ;
}
2022-09-12 22:16:56 +03:00
while ( ( evws = TAILQ_FIRST ( & http - > ws_sessions ) ) ! = NULL ) {
evws_connection_free ( evws ) ;
}
2006-01-22 05:08:50 +00:00
while ( ( http_cb = TAILQ_FIRST ( & http - > callbacks ) ) ! = NULL ) {
TAILQ_REMOVE ( & http - > callbacks , http_cb , next ) ;
2008-04-25 01:18:08 +00:00
mm_free ( http_cb - > what ) ;
mm_free ( http_cb ) ;
2006-01-22 05:08:50 +00:00
}
2008-05-05 07:17:05 +00:00
while ( ( vhost = TAILQ_FIRST ( & http - > virtualhosts ) ) ! = NULL ) {
2010-08-09 13:25:50 -04:00
TAILQ_REMOVE ( & http - > virtualhosts , vhost , next_vhost ) ;
2008-05-05 07:17:05 +00:00
evhttp_free ( vhost ) ;
}
if ( http - > vhost_pattern ! = NULL )
mm_free ( http - > vhost_pattern ) ;
2009-01-27 21:10:31 +00:00
2010-11-05 11:17:07 -07:00
while ( ( alias = TAILQ_FIRST ( & http - > aliases ) ) ! = NULL ) {
TAILQ_REMOVE ( & http - > aliases , alias , next ) ;
mm_free ( alias - > alias ) ;
mm_free ( alias ) ;
}
2008-04-25 01:18:08 +00:00
mm_free ( http ) ;
2006-01-22 05:08:50 +00:00
}
2008-05-05 07:17:05 +00:00
int
evhttp_add_virtual_host ( struct evhttp * http , const char * pattern ,
struct evhttp * vhost )
{
/* a vhost can only be a vhost once and should not have bound sockets */
if ( vhost - > vhost_pattern ! = NULL | |
TAILQ_FIRST ( & vhost - > sockets ) ! = NULL )
return ( - 1 ) ;
vhost - > vhost_pattern = mm_strdup ( pattern ) ;
if ( vhost - > vhost_pattern = = NULL )
return ( - 1 ) ;
2010-08-09 13:25:50 -04:00
TAILQ_INSERT_TAIL ( & http - > virtualhosts , vhost , next_vhost ) ;
2008-05-05 07:17:05 +00:00
return ( 0 ) ;
}
int
evhttp_remove_virtual_host ( struct evhttp * http , struct evhttp * vhost )
{
if ( vhost - > vhost_pattern = = NULL )
return ( - 1 ) ;
2010-08-09 13:25:50 -04:00
TAILQ_REMOVE ( & http - > virtualhosts , vhost , next_vhost ) ;
2008-05-05 07:17:05 +00:00
mm_free ( vhost - > vhost_pattern ) ;
vhost - > vhost_pattern = NULL ;
return ( 0 ) ;
}
2010-11-05 11:17:07 -07:00
int
evhttp_add_server_alias ( struct evhttp * http , const char * alias )
{
struct evhttp_server_alias * evalias ;
evalias = mm_calloc ( 1 , sizeof ( * evalias ) ) ;
if ( ! evalias )
return - 1 ;
evalias - > alias = mm_strdup ( alias ) ;
if ( ! evalias - > alias ) {
mm_free ( evalias ) ;
return - 1 ;
}
TAILQ_INSERT_TAIL ( & http - > aliases , evalias , next ) ;
return 0 ;
}
int
evhttp_remove_server_alias ( struct evhttp * http , const char * alias )
{
struct evhttp_server_alias * evalias ;
TAILQ_FOREACH ( evalias , & http - > aliases , next ) {
if ( evutil_ascii_strcasecmp ( evalias - > alias , alias ) = = 0 ) {
TAILQ_REMOVE ( & http - > aliases , evalias , next ) ;
mm_free ( evalias - > alias ) ;
mm_free ( evalias ) ;
return 0 ;
2010-12-09 11:43:12 -05:00
}
2010-11-05 11:17:07 -07:00
}
2010-12-09 11:43:12 -05:00
return - 1 ;
2010-11-05 11:17:07 -07:00
}
2006-11-23 06:32:20 +00:00
void
2019-03-04 06:53:42 +03:00
evhttp_set_timeout ( struct evhttp * http , int timeout )
2006-11-23 06:32:20 +00:00
{
2019-03-04 06:53:42 +03:00
evhttp_set_timeout_ ( & http - > timeout_read , timeout , - 1 ) ;
evhttp_set_timeout_ ( & http - > timeout_write , timeout , - 1 ) ;
2011-02-22 17:52:50 -05:00
}
void
evhttp_set_timeout_tv ( struct evhttp * http , const struct timeval * tv )
{
2019-03-04 06:53:42 +03:00
evhttp_set_timeout_tv_ ( & http - > timeout_read , tv , - 1 ) ;
evhttp_set_timeout_tv_ ( & http - > timeout_write , tv , - 1 ) ;
}
void
evhttp_set_read_timeout_tv ( struct evhttp * http , const struct timeval * tv )
{
evhttp_set_timeout_tv_ ( & http - > timeout_read , tv , - 1 ) ;
}
void
evhttp_set_write_timeout_tv ( struct evhttp * http , const struct timeval * tv )
{
evhttp_set_timeout_tv_ ( & http - > timeout_write , tv , - 1 ) ;
2006-11-23 06:32:20 +00:00
}
2016-02-15 00:12:54 +03:00
int evhttp_set_flags ( struct evhttp * http , int flags )
{
int avail_flags = 0 ;
avail_flags | = EVHTTP_SERVER_LINGERING_CLOSE ;
if ( flags & ~ avail_flags )
return 1 ;
http - > flags & = ~ avail_flags ;
http - > flags | = flags ;
return 0 ;
}
2009-11-05 15:57:22 +00:00
void
evhttp_set_max_headers_size ( struct evhttp * http , ev_ssize_t max_headers_size )
{
if ( max_headers_size < 0 )
http - > default_max_headers_size = EV_SIZE_MAX ;
else
http - > default_max_headers_size = max_headers_size ;
2009-11-04 20:17:32 +00:00
}
2009-11-05 15:57:22 +00:00
void
evhttp_set_max_body_size ( struct evhttp * http , ev_ssize_t max_body_size )
{
if ( max_body_size < 0 )
http - > default_max_body_size = EV_UINT64_MAX ;
else
http - > default_max_body_size = max_body_size ;
2009-11-04 20:17:32 +00:00
}
2018-01-30 15:39:41 -08:00
void
evhttp_set_max_connections ( struct evhttp * http , int max_connections )
{
if ( max_connections < 0 )
http - > connection_max = 0 ;
else
http - > connection_max = max_connections ;
}
int
evhttp_get_connection_count ( struct evhttp * http )
{
return http - > connection_cnt ;
}
2013-09-28 20:03:28 +02:00
void
evhttp_set_default_content_type ( struct evhttp * http ,
const char * content_type ) {
http - > default_content_type = content_type ;
}
2010-11-04 11:25:35 -04:00
void
2019-04-02 15:43:35 +02:00
evhttp_set_allowed_methods ( struct evhttp * http , ev_uint32_t methods )
2010-11-04 11:25:35 -04:00
{
http - > allowed_methods = methods ;
}
2016-01-08 13:36:20 -08:00
void
evhttp_set_ext_method_cmp ( struct evhttp * http , evhttp_ext_method_cb cmp )
{
http - > ext_method_cmp = cmp ;
}
2009-04-17 01:03:07 +00:00
int
2006-01-22 05:08:50 +00:00
evhttp_set_cb ( struct evhttp * http , const char * uri ,
void ( * cb ) ( struct evhttp_request * , void * ) , void * cbarg )
{
struct evhttp_cb * http_cb ;
2009-04-17 01:03:07 +00:00
TAILQ_FOREACH ( http_cb , & http - > callbacks , next ) {
if ( strcmp ( http_cb - > what , uri ) = = 0 )
return ( - 1 ) ;
}
2009-11-19 23:08:50 +00:00
if ( ( http_cb = mm_calloc ( 1 , sizeof ( struct evhttp_cb ) ) ) = = NULL ) {
event_warn ( " %s: calloc " , __func__ ) ;
return ( - 2 ) ;
}
2006-01-22 05:08:50 +00:00
2008-04-25 01:18:08 +00:00
http_cb - > what = mm_strdup ( uri ) ;
2010-12-18 01:07:27 -02:00
if ( http_cb - > what = = NULL ) {
event_warn ( " %s: strdup " , __func__ ) ;
mm_free ( http_cb ) ;
return ( - 3 ) ;
}
2006-01-22 05:08:50 +00:00
http_cb - > cb = cb ;
http_cb - > cbarg = cbarg ;
TAILQ_INSERT_TAIL ( & http - > callbacks , http_cb , next ) ;
2009-04-17 01:03:07 +00:00
return ( 0 ) ;
2006-01-22 05:08:50 +00:00
}
2007-05-23 05:20:59 +00:00
int
evhttp_del_cb ( struct evhttp * http , const char * uri )
{
struct evhttp_cb * http_cb ;
TAILQ_FOREACH ( http_cb , & http - > callbacks , next ) {
if ( strcmp ( http_cb - > what , uri ) = = 0 )
break ;
}
if ( http_cb = = NULL )
return ( - 1 ) ;
TAILQ_REMOVE ( & http - > callbacks , http_cb , next ) ;
2008-04-25 01:18:08 +00:00
mm_free ( http_cb - > what ) ;
mm_free ( http_cb ) ;
2007-05-23 05:20:59 +00:00
return ( 0 ) ;
}
2006-01-22 05:08:50 +00:00
void
evhttp_set_gencb ( struct evhttp * http ,
void ( * cb ) ( struct evhttp_request * , void * ) , void * cbarg )
{
http - > gencb = cb ;
http - > gencbarg = cbarg ;
}
2011-09-12 10:46:17 -04:00
void
evhttp_set_bevcb ( struct evhttp * http ,
struct bufferevent * ( * cb ) ( struct event_base * , void * ) , void * cbarg )
{
http - > bevcb = cb ;
http - > bevcbarg = cbarg ;
}
2017-12-01 01:29:32 +00:00
void
evhttp_set_newreqcb ( struct evhttp * http ,
int ( * cb ) ( struct evhttp_request * , void * ) , void * cbarg )
{
http - > newreqcb = cb ;
http - > newreqcbarg = cbarg ;
}
2016-02-09 18:01:00 +00:00
void
evhttp_set_errorcb ( struct evhttp * http ,
int ( * cb ) ( struct evhttp_request * , struct evbuffer * , int , const char * , void * ) ,
void * cbarg )
{
http - > errorcb = cb ;
http - > errorcbarg = cbarg ;
}
2017-12-01 01:29:32 +00:00
2006-01-22 05:08:50 +00:00
/*
* Request related functions
*/
struct evhttp_request *
evhttp_request_new ( void ( * cb ) ( struct evhttp_request * , void * ) , void * arg )
{
struct evhttp_request * req = NULL ;
/* Allocate request structure */
2008-04-25 01:18:08 +00:00
if ( ( req = mm_calloc ( 1 , sizeof ( struct evhttp_request ) ) ) = = NULL ) {
2006-01-22 05:08:50 +00:00
event_warn ( " %s: calloc " , __func__ ) ;
goto error ;
}
2009-11-04 20:17:32 +00:00
req - > headers_size = 0 ;
req - > body_size = 0 ;
2006-01-22 05:08:50 +00:00
req - > kind = EVHTTP_RESPONSE ;
2008-04-25 01:18:08 +00:00
req - > input_headers = mm_calloc ( 1 , sizeof ( struct evkeyvalq ) ) ;
2006-01-22 05:08:50 +00:00
if ( req - > input_headers = = NULL ) {
event_warn ( " %s: calloc " , __func__ ) ;
goto error ;
}
TAILQ_INIT ( req - > input_headers ) ;
2008-04-25 01:18:08 +00:00
req - > output_headers = mm_calloc ( 1 , sizeof ( struct evkeyvalq ) ) ;
2006-01-22 05:08:50 +00:00
if ( req - > output_headers = = NULL ) {
event_warn ( " %s: calloc " , __func__ ) ;
goto error ;
}
TAILQ_INIT ( req - > output_headers ) ;
2006-07-17 00:33:57 +00:00
if ( ( req - > input_buffer = evbuffer_new ( ) ) = = NULL ) {
event_warn ( " %s: evbuffer_new " , __func__ ) ;
goto error ;
}
if ( ( req - > output_buffer = evbuffer_new ( ) ) = = NULL ) {
event_warn ( " %s: evbuffer_new " , __func__ ) ;
goto error ;
}
2006-01-22 05:08:50 +00:00
req - > cb = cb ;
req - > cb_arg = arg ;
return ( req ) ;
error :
if ( req ! = NULL )
evhttp_request_free ( req ) ;
return ( NULL ) ;
}
void
evhttp_request_free ( struct evhttp_request * req )
{
2008-06-02 05:45:26 +00:00
if ( ( req - > flags & EVHTTP_REQ_DEFER_FREE ) ! = 0 ) {
req - > flags | = EVHTTP_REQ_NEEDS_FREE ;
return ;
}
2006-01-22 05:08:50 +00:00
if ( req - > remote_host ! = NULL )
2008-04-25 01:18:08 +00:00
mm_free ( req - > remote_host ) ;
2006-01-22 05:08:50 +00:00
if ( req - > uri ! = NULL )
2008-04-25 01:18:08 +00:00
mm_free ( req - > uri ) ;
2010-11-05 11:17:07 -07:00
if ( req - > uri_elems ! = NULL )
evhttp_uri_free ( req - > uri_elems ) ;
2006-01-22 05:08:50 +00:00
if ( req - > response_code_line ! = NULL )
2008-04-25 01:18:08 +00:00
mm_free ( req - > response_code_line ) ;
2010-11-05 11:17:07 -07:00
if ( req - > host_cache ! = NULL )
mm_free ( req - > host_cache ) ;
2010-12-09 11:43:12 -05:00
2006-01-22 05:08:50 +00:00
evhttp_clear_headers ( req - > input_headers ) ;
2008-04-25 01:18:08 +00:00
mm_free ( req - > input_headers ) ;
2006-01-22 05:08:50 +00:00
evhttp_clear_headers ( req - > output_headers ) ;
2008-04-25 01:18:08 +00:00
mm_free ( req - > output_headers ) ;
2006-01-22 05:08:50 +00:00
2006-07-17 00:33:57 +00:00
if ( req - > input_buffer ! = NULL )
evbuffer_free ( req - > input_buffer ) ;
if ( req - > output_buffer ! = NULL )
evbuffer_free ( req - > output_buffer ) ;
2008-04-25 01:18:08 +00:00
mm_free ( req ) ;
2006-01-22 05:08:50 +00:00
}
2007-12-28 00:36:47 +00:00
void
evhttp_request_own ( struct evhttp_request * req )
{
req - > flags | = EVHTTP_USER_OWNED ;
}
int
evhttp_request_is_owned ( struct evhttp_request * req )
{
return ( req - > flags & EVHTTP_USER_OWNED ) ! = 0 ;
}
2010-02-03 14:34:56 -08:00
struct evhttp_connection *
evhttp_request_get_connection ( struct evhttp_request * req )
{
return req - > evcon ;
}
2010-10-21 12:19:28 -04:00
struct event_base *
evhttp_connection_get_base ( struct evhttp_connection * conn )
{
return conn - > base ;
}
2010-02-03 14:34:56 -08:00
2007-02-14 06:10:32 +00:00
void
evhttp_request_set_chunked_cb ( struct evhttp_request * req ,
void ( * cb ) ( struct evhttp_request * , void * ) )
{
req - > chunk_cb = cb ;
}
2013-11-18 16:06:16 +01:00
void
evhttp_request_set_header_cb ( struct evhttp_request * req ,
int ( * cb ) ( struct evhttp_request * , void * ) )
{
req - > header_cb = cb ;
}
2013-03-21 13:55:40 +04:00
void
evhttp_request_set_error_cb ( struct evhttp_request * req ,
void ( * cb ) ( enum evhttp_request_error , void * ) )
{
req - > error_cb = cb ;
}
2014-01-05 20:35:46 -05:00
void
evhttp_request_set_on_complete_cb ( struct evhttp_request * req ,
2014-01-06 20:36:31 -05:00
void ( * cb ) ( struct evhttp_request * , void * ) , void * cb_arg )
2014-01-05 20:35:46 -05:00
{
req - > on_complete_cb = cb ;
2014-01-06 20:36:31 -05:00
req - > on_complete_cb_arg = cb_arg ;
2014-01-05 20:35:46 -05:00
}
2006-02-27 02:27:37 +00:00
/*
* Allows for inspection of the request URI
*/
const char *
2010-10-18 13:58:02 -04:00
evhttp_request_get_uri ( const struct evhttp_request * req ) {
2006-02-27 02:27:37 +00:00
if ( req - > uri = = NULL )
2022-08-28 15:27:04 +03:00
event_debug ( ( " %s: request %p has no uri \n " , __func__ , ( void * ) req ) ) ;
2006-02-27 02:27:37 +00:00
return ( req - > uri ) ;
}
2010-11-05 11:17:07 -07:00
const struct evhttp_uri *
evhttp_request_get_evhttp_uri ( const struct evhttp_request * req ) {
if ( req - > uri_elems = = NULL )
event_debug ( ( " %s: request %p has no uri elems \n " ,
2022-08-28 15:27:04 +03:00
__func__ , ( void * ) req ) ) ;
2010-11-05 11:17:07 -07:00
return ( req - > uri_elems ) ;
}
const char *
evhttp_request_get_host ( struct evhttp_request * req )
{
const char * host = NULL ;
if ( req - > host_cache )
return req - > host_cache ;
if ( req - > uri_elems )
host = evhttp_uri_get_host ( req - > uri_elems ) ;
if ( ! host & & req - > input_headers ) {
const char * p ;
size_t len ;
2010-12-09 11:43:12 -05:00
2010-11-05 11:17:07 -07:00
host = evhttp_find_header ( req - > input_headers , " Host " ) ;
/* The Host: header may include a port. Remove it here
2010-12-09 11:47:54 -05:00
to be consistent with uri_elems case above . */
2010-11-05 11:17:07 -07:00
if ( host ) {
p = host + strlen ( host ) - 1 ;
2012-02-29 15:07:33 -05:00
while ( p > host & & EVUTIL_ISDIGIT_ ( * p ) )
2010-11-05 11:17:07 -07:00
- - p ;
if ( p > host & & * p = = ' : ' ) {
len = p - host ;
req - > host_cache = mm_malloc ( len + 1 ) ;
if ( ! req - > host_cache ) {
event_warn ( " %s: malloc " , __func__ ) ;
return NULL ;
}
memcpy ( req - > host_cache , host , len ) ;
req - > host_cache [ len ] = ' \0 ' ;
host = req - > host_cache ;
}
}
}
2010-12-09 11:43:12 -05:00
2010-11-05 11:17:07 -07:00
return host ;
}
2010-10-18 13:58:02 -04:00
enum evhttp_cmd_type
evhttp_request_get_command ( const struct evhttp_request * req ) {
return ( req - > type ) ;
}
2010-11-03 15:12:08 -04:00
int
evhttp_request_get_response_code ( const struct evhttp_request * req )
{
return req - > response_code ;
}
2012-10-10 13:16:02 -04:00
const char *
evhttp_request_get_response_code_line ( const struct evhttp_request * req )
{
return req - > response_code_line ;
}
2008-05-11 16:22:35 +00:00
/** Returns the input headers */
struct evkeyvalq * evhttp_request_get_input_headers ( struct evhttp_request * req )
{
return ( req - > input_headers ) ;
}
/** Returns the output headers */
struct evkeyvalq * evhttp_request_get_output_headers ( struct evhttp_request * req )
{
return ( req - > output_headers ) ;
}
/** Returns the input buffer */
struct evbuffer * evhttp_request_get_input_buffer ( struct evhttp_request * req )
{
return ( req - > input_buffer ) ;
}
/** Returns the output buffer */
struct evbuffer * evhttp_request_get_output_buffer ( struct evhttp_request * req )
{
return ( req - > output_buffer ) ;
}
2006-01-22 05:08:50 +00:00
/*
* Takes a file descriptor to read a request from .
* The callback is executed once the whole request has been read .
*/
2006-11-18 03:05:26 +00:00
static struct evhttp_connection *
evhttp_get_request_connection (
2007-08-19 02:41:23 +00:00
struct evhttp * http ,
2022-08-01 10:16:18 +02:00
evutil_socket_t fd , struct sockaddr * sa , ev_socklen_t salen ,
struct bufferevent * bev )
2006-01-22 05:08:50 +00:00
{
2006-07-17 00:33:57 +00:00
struct evhttp_connection * evcon ;
2006-01-22 05:08:50 +00:00
2019-05-12 15:19:31 +03:00
# ifdef EVENT__HAVE_STRUCT_SOCKADDR_UN
if ( sa - > sa_family = = AF_UNIX ) {
2019-05-25 23:41:38 +03:00
struct sockaddr_un * sa_un = ( struct sockaddr_un * ) sa ;
sa_un - > sun_path [ 0 ] = ' \0 ' ;
2019-05-12 15:19:31 +03:00
}
# endif
2016-01-31 11:31:00 +00:00
# ifndef _WIN32
if ( sa - > sa_family = = AF_UNIX ) {
struct sockaddr_un * sockaddr = ( struct sockaddr_un * ) sa ;
event_debug ( ( " %s: new request from unix socket on "
EV_SOCK_FMT " \n " , __func__ , EV_SOCK_ARG ( fd ) ) ) ;
/* we need a connection object to put the http request on */
2022-08-01 10:16:18 +02:00
if ( ! bev & & http - > bevcb ! = NULL ) {
2016-01-31 11:31:00 +00:00
bev = ( * http - > bevcb ) ( http - > base , http - > bevcbarg ) ;
}
evcon = evhttp_connection_base_bufferevent_unix_new ( http - > base ,
bev , sockaddr - > sun_path ) ;
2008-04-18 13:46:13 +00:00
}
2016-01-31 11:31:00 +00:00
else
# endif
{
char * hostname = NULL , * portname = NULL ;
name_from_addr ( sa , salen , & hostname , & portname ) ;
if ( hostname = = NULL | | portname = = NULL ) {
if ( hostname ) mm_free ( hostname ) ;
if ( portname ) mm_free ( portname ) ;
return ( NULL ) ;
}
2008-04-17 19:19:36 +00:00
2016-01-31 11:31:00 +00:00
event_debug ( ( " %s: new request from %s:%s on " EV_SOCK_FMT " \n " ,
__func__ , hostname , portname , EV_SOCK_ARG ( fd ) ) ) ;
2006-01-22 05:08:50 +00:00
2016-01-31 11:31:00 +00:00
/* we need a connection object to put the http request on */
2022-08-01 10:16:18 +02:00
if ( ! bev & & http - > bevcb ! = NULL ) {
2016-01-31 11:31:00 +00:00
bev = ( * http - > bevcb ) ( http - > base , http - > bevcbarg ) ;
}
evcon = evhttp_connection_base_bufferevent_new (
http - > base , NULL , bev , hostname , atoi ( portname ) ) ;
mm_free ( hostname ) ;
mm_free ( portname ) ;
2011-09-12 10:46:17 -04:00
}
2008-04-17 15:50:28 +00:00
if ( evcon = = NULL )
2006-11-18 03:05:26 +00:00
return ( NULL ) ;
2007-08-19 02:41:23 +00:00
2009-11-05 21:22:23 +00:00
evcon - > max_headers_size = http - > default_max_headers_size ;
evcon - > max_body_size = http - > default_max_body_size ;
2016-02-15 00:12:54 +03:00
if ( http - > flags & EVHTTP_SERVER_LINGERING_CLOSE )
evcon - > flags | = EVHTTP_CON_LINGERING_CLOSE ;
2010-02-18 17:41:15 -05:00
2006-07-17 00:33:57 +00:00
evcon - > flags | = EVHTTP_CON_INCOMING ;
2008-06-26 00:40:57 +00:00
evcon - > state = EVCON_READING_FIRSTLINE ;
2009-01-27 21:10:31 +00:00
2021-03-23 09:02:39 +03:00
if ( bufferevent_replacefd ( evcon - > bufev , fd ) )
2018-11-14 00:20:20 +03:00
goto err ;
if ( bufferevent_enable ( evcon - > bufev , EV_READ ) )
goto err ;
if ( bufferevent_disable ( evcon - > bufev , EV_WRITE ) )
goto err ;
2018-02-11 16:28:58 -08:00
bufferevent_socket_set_conn_address_ ( evcon - > bufev , sa , salen ) ;
2008-04-29 04:52:50 +00:00
2006-11-18 03:05:26 +00:00
return ( evcon ) ;
2018-11-14 00:20:20 +03:00
err :
evhttp_connection_free ( evcon ) ;
return ( NULL ) ;
2006-11-18 03:05:26 +00:00
}
static int
evhttp_associate_new_request_with_connection ( struct evhttp_connection * evcon )
{
struct evhttp * http = evcon - > http_server ;
struct evhttp_request * req ;
if ( ( req = evhttp_request_new ( evhttp_handle_request , http ) ) = = NULL )
return ( - 1 ) ;
2016-01-31 11:31:00 +00:00
if ( evcon - > address ! = NULL ) {
if ( ( req - > remote_host = mm_strdup ( evcon - > address ) ) = = NULL ) {
event_warn ( " %s: strdup " , __func__ ) ;
evhttp_request_free ( req ) ;
return ( - 1 ) ;
}
2009-11-19 23:08:50 +00:00
}
req - > remote_port = evcon - > port ;
2006-07-17 00:33:57 +00:00
req - > evcon = evcon ; /* the request ends up owning the connection */
req - > flags | = EVHTTP_REQ_OWN_CONNECTION ;
2009-01-27 21:10:31 +00:00
2017-12-01 01:29:32 +00:00
/* We did not present the request to the user yet, so treat it
* as if the user was done with the request . This allows us
* to free the request on a persistent connection if the
* client drops it without sending a request .
2010-04-23 18:59:22 -07:00
*/
req - > userdone = 1 ;
2006-01-22 05:08:50 +00:00
req - > kind = EVHTTP_REQUEST ;
2009-01-27 21:10:31 +00:00
2017-12-01 01:29:32 +00:00
if ( http - > newreqcb & & http - > newreqcb ( req , http - > newreqcbarg ) = = - 1 ) {
evhttp_request_free ( req ) ;
return ( - 1 ) ;
}
TAILQ_INSERT_TAIL ( & evcon - > requests , req , next ) ;
2006-01-22 05:08:50 +00:00
2012-02-29 15:07:33 -05:00
evhttp_start_read_ ( evcon ) ;
2009-01-27 21:10:31 +00:00
2006-11-18 03:05:26 +00:00
return ( 0 ) ;
}
2010-08-09 13:25:50 -04:00
static void
2007-11-25 21:53:06 +00:00
evhttp_get_request ( struct evhttp * http , evutil_socket_t fd ,
2022-08-01 10:16:18 +02:00
struct sockaddr * sa , ev_socklen_t salen ,
struct bufferevent * bev )
2006-11-18 03:05:26 +00:00
{
struct evhttp_connection * evcon ;
2022-08-01 10:16:18 +02:00
evcon = evhttp_get_request_connection ( http , fd , sa , salen , bev ) ;
2008-06-24 22:38:37 +00:00
if ( evcon = = NULL ) {
2012-11-01 17:38:34 -04:00
event_sock_warn ( fd , " %s: cannot get connection on " EV_SOCK_FMT ,
__func__ , EV_SOCK_ARG ( fd ) ) ;
2010-04-14 15:42:57 -04:00
evutil_closesocket ( fd ) ;
2006-11-18 03:05:26 +00:00
return ;
2008-06-24 22:38:37 +00:00
}
2006-11-18 03:05:26 +00:00
2006-11-23 06:32:20 +00:00
/* the timeout can be used by the server to close idle connections */
2019-03-04 06:53:42 +03:00
if ( evutil_timerisset ( & http - > timeout_read ) )
evhttp_connection_set_read_timeout_tv ( evcon , & http - > timeout_read ) ;
if ( evutil_timerisset ( & http - > timeout_write ) )
evhttp_connection_set_write_timeout_tv ( evcon , & http - > timeout_write ) ;
2006-11-23 06:32:20 +00:00
2009-01-27 21:10:31 +00:00
/*
2006-11-18 03:05:26 +00:00
* if we want to accept more than one request on a connection ,
* we need to know which http server it belongs to .
*/
evcon - > http_server = http ;
2016-01-08 13:36:20 -08:00
evcon - > ext_method_cmp = http - > ext_method_cmp ;
2006-11-18 03:05:26 +00:00
TAILQ_INSERT_TAIL ( & http - > connections , evcon , next ) ;
2018-01-30 15:39:41 -08:00
http - > connection_cnt + + ;
/* send "service unavailable" if we've reached the connection limit */
if ( http - > connection_max & & http - > connection_max < http - > connection_cnt ) {
struct evhttp_request * req ;
if ( ( req = evhttp_request_new ( evhttp_handle_request , http ) ) = = NULL ) {
evhttp_connection_free ( evcon ) ;
return ;
}
req - > evcon = evcon ; /* the request owns the connection */
req - > flags | = EVHTTP_REQ_OWN_CONNECTION ;
req - > kind = EVHTTP_REQUEST ;
/* note, req->remote_host not needed since we don't read */
TAILQ_INSERT_TAIL ( & evcon - > requests , req , next ) ;
/* send error to client */
evcon - > state = EVCON_WRITING ;
bufferevent_enable ( evcon - > bufev , EV_READ ) ; /* enable close events */
evhttp_send_error ( req , HTTP_SERVUNAVAIL , NULL ) ;
2009-01-27 21:10:31 +00:00
2018-01-30 15:39:41 -08:00
} else if ( evhttp_associate_new_request_with_connection ( evcon ) = = - 1 )
2006-11-18 03:05:26 +00:00
evhttp_connection_free ( evcon ) ;
2006-01-22 05:08:50 +00:00
}
/*
* Network helper functions that we do not want to export to the rest of
* the world .
*/
static void
2009-07-30 17:01:21 +00:00
name_from_addr ( struct sockaddr * sa , ev_socklen_t salen ,
2006-01-22 05:08:50 +00:00
char * * phost , char * * pport )
{
2008-04-17 15:50:28 +00:00
char ntop [ NI_MAXHOST ] ;
char strport [ NI_MAXSERV ] ;
2007-01-18 06:28:42 +00:00
int ni_result ;
2006-01-22 05:08:50 +00:00
2012-02-29 15:07:31 -05:00
# ifdef EVENT__HAVE_GETNAMEINFO
2008-06-24 22:38:37 +00:00
ni_result = getnameinfo ( sa , salen ,
2006-01-22 05:08:50 +00:00
ntop , sizeof ( ntop ) , strport , sizeof ( strport ) ,
2008-06-24 22:43:19 +00:00
NI_NUMERICHOST | NI_NUMERICSERV ) ;
2009-01-27 21:10:31 +00:00
2008-06-24 22:38:37 +00:00
if ( ni_result ! = 0 ) {
2009-11-16 22:25:46 +00:00
# ifdef EAI_SYSTEM
/* Windows doesn't have an EAI_SYSTEM. */
2007-01-18 06:28:42 +00:00
if ( ni_result = = EAI_SYSTEM )
event_err ( 1 , " getnameinfo failed " ) ;
else
2009-11-16 22:25:46 +00:00
# endif
2007-01-18 06:28:42 +00:00
event_errx ( 1 , " getnameinfo failed: %s " , gai_strerror ( ni_result ) ) ;
2008-04-17 15:50:28 +00:00
return ;
2007-01-18 06:28:42 +00:00
}
2008-06-24 22:38:37 +00:00
# else
ni_result = fake_getnameinfo ( sa , salen ,
ntop , sizeof ( ntop ) , strport , sizeof ( strport ) ,
NI_NUMERICHOST | NI_NUMERICSERV ) ;
if ( ni_result ! = 0 )
return ;
# endif
2008-04-17 15:50:28 +00:00
2008-04-25 01:18:08 +00:00
* phost = mm_strdup ( ntop ) ;
* pport = mm_strdup ( strport ) ;
2006-01-22 05:08:50 +00:00
}
2008-11-15 05:27:23 +00:00
/* Create a non-blocking socket and bind it */
2007-11-25 21:53:06 +00:00
static evutil_socket_t
2019-09-18 23:12:59 +03:00
create_bind_socket_nonblock ( struct evutil_addrinfo * ai , int reuse )
2006-01-22 05:08:50 +00:00
{
2010-02-18 17:41:15 -05:00
evutil_socket_t fd ;
2007-11-25 21:53:06 +00:00
2024-01-06 00:52:43 +08:00
int r ;
2006-01-22 05:08:50 +00:00
int serrno ;
2010-02-18 17:41:15 -05:00
/* Create listen socket */
2012-02-29 15:07:33 -05:00
fd = evutil_socket_ ( ai ? ai - > ai_family : AF_INET ,
2012-02-10 16:32:32 -05:00
SOCK_STREAM | EVUTIL_SOCK_NONBLOCK | EVUTIL_SOCK_CLOEXEC , 0 ) ;
2010-02-18 17:41:15 -05:00
if ( fd = = - 1 ) {
2008-09-05 16:29:56 +00:00
event_sock_warn ( - 1 , " socket " ) ;
return ( - 1 ) ;
2010-02-18 17:41:15 -05:00
}
2006-01-22 05:08:50 +00:00
2024-01-06 00:52:43 +08:00
/* TODO(panjf2000): make this TCP keep-alive value configurable */
if ( evutil_set_tcp_keepalive ( fd , 1 , 300 ) < 0 )
2012-07-26 10:39:05 -04:00
goto out ;
if ( reuse ) {
if ( evutil_make_listen_socket_reuseable ( fd ) < 0 )
goto out ;
}
2006-01-22 05:08:50 +00:00
2008-11-15 05:27:23 +00:00
if ( ai ! = NULL ) {
2010-11-01 13:59:04 -04:00
r = bind ( fd , ai - > ai_addr , ( ev_socklen_t ) ai - > ai_addrlen ) ;
2008-11-15 05:27:23 +00:00
if ( r = = - 1 )
goto out ;
}
2006-01-22 05:08:50 +00:00
return ( fd ) ;
out :
2007-09-24 16:26:11 +00:00
serrno = EVUTIL_SOCKET_ERROR ( ) ;
2010-04-14 15:42:57 -04:00
evutil_closesocket ( fd ) ;
2007-09-24 16:26:11 +00:00
EVUTIL_SET_SOCKET_ERROR ( serrno ) ;
2006-01-22 05:08:50 +00:00
return ( - 1 ) ;
}
2009-11-16 22:25:46 +00:00
static struct evutil_addrinfo *
2008-05-12 00:40:04 +00:00
make_addrinfo ( const char * address , ev_uint16_t port )
2006-01-22 05:08:50 +00:00
{
2010-02-18 17:41:15 -05:00
struct evutil_addrinfo * ai = NULL ;
2007-09-07 02:49:46 +00:00
2010-02-18 17:41:15 -05:00
struct evutil_addrinfo hints ;
char strport [ NI_MAXSERV ] ;
int ai_result ;
2007-01-18 06:28:42 +00:00
2010-02-18 17:41:15 -05:00
memset ( & hints , 0 , sizeof ( hints ) ) ;
hints . ai_family = AF_UNSPEC ;
hints . ai_socktype = SOCK_STREAM ;
2009-11-16 22:25:46 +00:00
/* turn NULL hostname into INADDR_ANY, and skip looking up any address
* types we don ' t have an interface to connect to . */
2010-02-18 17:41:15 -05:00
hints . ai_flags = EVUTIL_AI_PASSIVE | EVUTIL_AI_ADDRCONFIG ;
evutil_snprintf ( strport , sizeof ( strport ) , " %d " , port ) ;
if ( ( ai_result = evutil_getaddrinfo ( address , strport , & hints , & ai ) )
2009-11-16 22:25:46 +00:00
! = 0 ) {
2010-02-18 17:41:15 -05:00
if ( ai_result = = EVUTIL_EAI_SYSTEM )
event_warn ( " getaddrinfo " ) ;
else
event_warnx ( " getaddrinfo: %s " ,
2009-11-16 22:25:46 +00:00
evutil_gai_strerror ( ai_result ) ) ;
2007-09-07 02:49:46 +00:00
return ( NULL ) ;
2010-02-18 17:41:15 -05:00
}
2006-01-22 05:08:50 +00:00
2009-11-16 22:25:46 +00:00
return ( ai ) ;
2007-09-07 02:49:46 +00:00
}
2007-11-25 21:53:06 +00:00
static evutil_socket_t
2008-06-20 06:52:13 +00:00
bind_socket ( const char * address , ev_uint16_t port , int reuse )
2007-09-07 02:49:46 +00:00
{
2007-11-25 21:53:06 +00:00
evutil_socket_t fd ;
2009-11-16 22:25:46 +00:00
struct evutil_addrinfo * aitop = NULL ;
2008-11-15 05:27:23 +00:00
/* just create an unbound socket */
if ( address = = NULL & & port = = 0 )
2019-09-18 23:12:59 +03:00
return create_bind_socket_nonblock ( NULL , 0 ) ;
2009-01-27 21:10:31 +00:00
2008-11-15 05:27:23 +00:00
aitop = make_addrinfo ( address , port ) ;
2007-09-07 02:49:46 +00:00
if ( aitop = = NULL )
return ( - 1 ) ;
2019-09-18 23:12:59 +03:00
fd = create_bind_socket_nonblock ( aitop , reuse ) ;
2006-11-22 01:21:10 +00:00
2009-11-16 22:25:46 +00:00
evutil_freeaddrinfo ( aitop ) ;
2006-01-22 05:08:50 +00:00
return ( fd ) ;
}
2007-09-07 02:49:46 +00:00
2010-10-21 14:41:12 -04:00
struct evhttp_uri {
2011-02-13 00:41:22 -05:00
unsigned flags ;
2010-10-21 14:41:12 -04:00
char * scheme ; /* scheme; e.g http, ftp etc */
char * userinfo ; /* userinfo (typically username:pass), or NULL */
char * host ; /* hostname, IP address, or NULL */
int port ; /* port, or zero */
2016-01-31 11:31:00 +00:00
# ifndef _WIN32
char * unixsocket ; /* unix domain socket or NULL */
# endif
2010-10-21 14:41:12 -04:00
char * path ; /* path, or "". */
char * query ; /* query, or NULL */
char * fragment ; /* fragment or NULL */
} ;
struct evhttp_uri *
evhttp_uri_new ( void )
{
2023-11-26 20:52:32 +00:00
struct evhttp_uri * uri = mm_calloc ( 1 , sizeof ( struct evhttp_uri ) ) ;
2010-10-21 14:41:12 -04:00
if ( uri )
uri - > port = - 1 ;
return uri ;
}
2011-02-13 00:41:22 -05:00
void
evhttp_uri_set_flags ( struct evhttp_uri * uri , unsigned flags )
{
uri - > flags = flags ;
}
/* Return true if the string starting at s and ending immediately before eos
2010-10-19 11:26:59 -04:00
* is a valid URI scheme according to RFC3986
*/
static int
scheme_ok ( const char * s , const char * eos )
{
/* scheme = ALPHA *( ALPHA / DIGIT / "+" / "-" / "." ) */
EVUTIL_ASSERT ( eos > = s ) ;
if ( s = = eos )
return 0 ;
2012-02-29 15:07:33 -05:00
if ( ! EVUTIL_ISALPHA_ ( * s ) )
2010-10-19 11:26:59 -04:00
return 0 ;
while ( + + s < eos ) {
2012-02-29 15:07:33 -05:00
if ( ! EVUTIL_ISALNUM_ ( * s ) & &
2010-10-19 11:26:59 -04:00
* s ! = ' + ' & & * s ! = ' - ' & & * s ! = ' . ' )
return 0 ;
}
return 1 ;
}
# define SUBDELIMS "!$&'()*+,;="
/* Return true iff [s..eos) is a valid userinfo */
static int
userinfo_ok ( const char * s , const char * eos )
{
while ( s < eos ) {
if ( CHAR_IS_UNRESERVED ( * s ) | |
strchr ( SUBDELIMS , * s ) | |
* s = = ' : ' )
+ + s ;
else if ( * s = = ' % ' & & s + 2 < eos & &
2012-02-29 15:07:33 -05:00
EVUTIL_ISXDIGIT_ ( s [ 1 ] ) & &
EVUTIL_ISXDIGIT_ ( s [ 2 ] ) )
2010-10-19 11:26:59 -04:00
s + = 3 ;
else
return 0 ;
}
return 1 ;
}
static int
regname_ok ( const char * s , const char * eos )
{
while ( s & & s < eos ) {
if ( CHAR_IS_UNRESERVED ( * s ) | |
strchr ( SUBDELIMS , * s ) )
+ + s ;
else if ( * s = = ' % ' & &
2012-02-29 15:07:33 -05:00
EVUTIL_ISXDIGIT_ ( s [ 1 ] ) & &
EVUTIL_ISXDIGIT_ ( s [ 2 ] ) )
2010-10-19 11:26:59 -04:00
s + = 3 ;
else
return 0 ;
}
return 1 ;
}
static int
parse_port ( const char * s , const char * eos )
{
int portnum = 0 ;
while ( s < eos ) {
2012-02-29 15:07:33 -05:00
if ( ! EVUTIL_ISDIGIT_ ( * s ) )
2010-10-19 11:26:59 -04:00
return - 1 ;
portnum = ( portnum * 10 ) + ( * s - ' 0 ' ) ;
2010-10-19 12:35:50 -04:00
if ( portnum < 0 )
return - 1 ;
2014-03-18 11:39:23 -04:00
if ( portnum > 65535 )
return - 1 ;
2010-10-19 11:26:59 -04:00
+ + s ;
}
return portnum ;
}
/* returns 0 for bad, 1 for ipv6, 2 for IPvFuture */
static int
bracket_addr_ok ( const char * s , const char * eos )
{
if ( s + 3 > eos | | * s ! = ' [ ' | | * ( eos - 1 ) ! = ' ] ' )
return 0 ;
if ( s [ 1 ] = = ' v ' ) {
/* IPvFuture, or junk.
" v " 1 * HEXDIG " . " 1 * ( unreserved / sub - delims / " : " )
*/
s + = 2 ; /* skip [v */
- - eos ;
2012-02-29 15:07:33 -05:00
if ( ! EVUTIL_ISXDIGIT_ ( * s ) ) /*require at least one*/
2010-10-19 11:26:59 -04:00
return 0 ;
while ( s < eos & & * s ! = ' . ' ) {
2012-02-29 15:07:33 -05:00
if ( EVUTIL_ISXDIGIT_ ( * s ) )
2010-10-19 11:26:59 -04:00
+ + s ;
else
return 0 ;
}
if ( * s ! = ' . ' )
return 0 ;
+ + s ;
while ( s < eos ) {
if ( CHAR_IS_UNRESERVED ( * s ) | |
strchr ( SUBDELIMS , * s ) | |
* s = = ' : ' )
+ + s ;
else
return 0 ;
}
return 2 ;
} else {
/* IPv6, or junk */
char buf [ 64 ] ;
2010-11-01 13:59:04 -04:00
ev_ssize_t n_chars = eos - s - 2 ;
2010-10-19 11:26:59 -04:00
struct in6_addr in6 ;
if ( n_chars > = 64 ) /* way too long */
return 0 ;
memcpy ( buf , s + 1 , n_chars ) ;
buf [ n_chars ] = ' \0 ' ;
return ( evutil_inet_pton ( AF_INET6 , buf , & in6 ) = = 1 ) ? 1 : 0 ;
}
}
static int
2020-10-27 01:40:34 +03:00
parse_authority ( struct evhttp_uri * uri , char * s , char * eos , unsigned * flags )
2010-10-19 11:26:59 -04:00
{
2020-10-27 01:40:34 +03:00
size_t len ;
2010-10-19 11:26:59 -04:00
char * cp , * port ;
2020-10-27 01:40:34 +03:00
2010-10-19 11:26:59 -04:00
EVUTIL_ASSERT ( eos ) ;
if ( eos = = s ) {
uri - > host = mm_strdup ( " " ) ;
2010-12-18 01:07:27 -02:00
if ( uri - > host = = NULL ) {
event_warn ( " %s: strdup " , __func__ ) ;
return - 1 ;
}
2010-10-19 11:26:59 -04:00
return 0 ;
}
/* Optionally, we start with "userinfo@" */
cp = strchr ( s , ' @ ' ) ;
if ( cp & & cp < eos ) {
if ( ! userinfo_ok ( s , cp ) )
return - 1 ;
* cp + + = ' \0 ' ;
uri - > userinfo = mm_strdup ( s ) ;
2010-12-18 01:07:27 -02:00
if ( uri - > userinfo = = NULL ) {
event_warn ( " %s: strdup " , __func__ ) ;
return - 1 ;
}
2010-10-19 11:26:59 -04:00
} else {
cp = s ;
}
2016-01-31 11:31:00 +00:00
# ifndef _WIN32
if ( * flags & EVHTTP_URI_UNIX_SOCKET & & ! strncmp ( cp , " unix: " , 5 ) ) {
char * e = strchr ( cp + 5 , ' : ' ) ;
if ( e ) {
* e = ' \0 ' ;
uri - > unixsocket = mm_strdup ( cp + 5 ) ;
return 0 ;
} else {
return - 1 ;
}
}
# endif
2010-10-19 11:26:59 -04:00
/* Optionally, we end with ":port" */
2012-02-29 15:07:33 -05:00
for ( port = eos - 1 ; port > = cp & & EVUTIL_ISDIGIT_ ( * port ) ; - - port )
2010-10-19 11:26:59 -04:00
;
if ( port > = cp & & * port = = ' : ' ) {
2010-10-19 12:35:50 -04:00
if ( port + 1 = = eos ) /* Leave port unspecified; the RFC allows a
* nil port */
uri - > port = - 1 ;
else if ( ( uri - > port = parse_port ( port + 1 , eos ) ) < 0 )
2010-10-19 11:26:59 -04:00
return - 1 ;
eos = port ;
}
/* Now, cp..eos holds the "host" port, which can be an IPv4Address,
* an IP - Literal , or a reg - name */
EVUTIL_ASSERT ( eos > = cp ) ;
2020-10-27 01:40:34 +03:00
len = eos - cp ;
2010-10-19 11:26:59 -04:00
if ( * cp = = ' [ ' & & eos > = cp + 2 & & * ( eos - 1 ) = = ' ] ' ) {
/* IPv6address, IP-Literal, or junk. */
if ( ! bracket_addr_ok ( cp , eos ) )
return - 1 ;
2020-10-27 01:40:34 +03:00
if ( * flags & EVHTTP_URI_HOST_STRIP_BRACKETS )
len = eos - cp - 2 ;
2010-10-19 11:26:59 -04:00
} else {
/* Make sure the host part is ok. */
if ( ! regname_ok ( cp , eos ) ) /* Match IPv4Address or reg-name */
return - 1 ;
}
2020-10-27 01:40:34 +03:00
uri - > host = mm_malloc ( len + 1 ) ;
2010-12-18 01:07:27 -02:00
if ( uri - > host = = NULL ) {
event_warn ( " %s: malloc " , __func__ ) ;
return - 1 ;
}
2020-10-27 01:40:34 +03:00
if ( * cp = = ' [ ' & & * flags & EVHTTP_URI_HOST_STRIP_BRACKETS ) {
memcpy ( uri - > host , cp + 1 , len ) ;
* flags | = _EVHTTP_URI_HOST_HAS_BRACKETS ;
} else {
memcpy ( uri - > host , cp , len ) ;
}
uri - > host [ len ] = ' \0 ' ;
2010-10-19 11:26:59 -04:00
return 0 ;
}
2010-10-19 12:35:50 -04:00
static char *
end_of_authority ( char * cp )
{
while ( * cp ) {
if ( * cp = = ' ? ' | | * cp = = ' # ' | | * cp = = ' / ' )
return cp ;
+ + cp ;
}
return cp ;
}
2011-02-13 00:41:22 -05:00
enum uri_part {
PART_PATH ,
PART_QUERY ,
PART_FRAGMENT
} ;
2010-10-19 11:26:59 -04:00
/* Return the character after the longest prefix of 'cp' that matches...
* * pchar / " / " if allow_qchars is false , or
2011-02-13 00:41:22 -05:00
* * ( pchar / " / " / " ? " ) if allow_qchars is true .
2010-10-19 11:26:59 -04:00
*/
static char *
2011-02-13 00:41:22 -05:00
end_of_path ( char * cp , enum uri_part part , unsigned flags )
2010-10-19 11:26:59 -04:00
{
2011-02-13 00:41:22 -05:00
if ( flags & EVHTTP_URI_NONCONFORMANT ) {
/* If NONCONFORMANT:
* Path is everything up to a # or ? or nul .
* Query is everything up a # or nul
* Fragment is everything up to a nul .
*/
switch ( part ) {
case PART_PATH :
while ( * cp & & * cp ! = ' # ' & & * cp ! = ' ? ' )
+ + cp ;
break ;
case PART_QUERY :
while ( * cp & & * cp ! = ' # ' )
+ + cp ;
break ;
case PART_FRAGMENT :
cp + = strlen ( cp ) ;
break ;
} ;
return cp ;
}
2010-10-19 11:26:59 -04:00
while ( * cp ) {
if ( CHAR_IS_UNRESERVED ( * cp ) | |
strchr ( SUBDELIMS , * cp ) | |
* cp = = ' : ' | | * cp = = ' @ ' | | * cp = = ' / ' )
+ + cp ;
2012-02-29 15:07:33 -05:00
else if ( * cp = = ' % ' & & EVUTIL_ISXDIGIT_ ( cp [ 1 ] ) & &
EVUTIL_ISXDIGIT_ ( cp [ 2 ] ) )
2010-10-19 11:26:59 -04:00
cp + = 3 ;
2011-02-13 00:41:22 -05:00
else if ( * cp = = ' ? ' & & part ! = PART_PATH )
2010-10-19 11:26:59 -04:00
+ + cp ;
else
return cp ;
}
return cp ;
}
2010-10-19 12:35:50 -04:00
static int
path_matches_noscheme ( const char * cp )
{
while ( * cp ) {
if ( * cp = = ' : ' )
return 0 ;
else if ( * cp = = ' / ' )
return 1 ;
+ + cp ;
}
return 1 ;
}
2010-10-18 14:34:20 -04:00
struct evhttp_uri *
evhttp_uri_parse ( const char * source_uri )
2011-02-13 00:41:22 -05:00
{
return evhttp_uri_parse_with_flags ( source_uri , 0 ) ;
}
struct evhttp_uri *
evhttp_uri_parse_with_flags ( const char * source_uri , unsigned flags )
2010-08-08 16:46:39 +04:00
{
2010-10-19 11:26:59 -04:00
char * readbuf = NULL , * readp = NULL , * token = NULL , * query = NULL ;
char * path = NULL , * fragment = NULL ;
int got_authority = 0 ;
2010-08-08 16:46:39 +04:00
2010-10-18 14:34:20 -04:00
struct evhttp_uri * uri = mm_calloc ( 1 , sizeof ( struct evhttp_uri ) ) ;
2010-08-08 16:46:39 +04:00
if ( uri = = NULL ) {
2010-12-18 02:40:22 -02:00
event_warn ( " %s: calloc " , __func__ ) ;
2010-10-18 14:43:54 -04:00
goto err ;
2010-08-08 16:46:39 +04:00
}
2010-10-19 11:26:59 -04:00
uri - > port = - 1 ;
2011-02-13 00:41:22 -05:00
uri - > flags = flags ;
2010-08-08 16:46:39 +04:00
2010-10-18 14:34:20 -04:00
readbuf = mm_strdup ( source_uri ) ;
2010-08-08 16:46:39 +04:00
if ( readbuf = = NULL ) {
2010-12-18 02:40:22 -02:00
event_warn ( " %s: strdup " , __func__ ) ;
2010-10-18 14:43:54 -04:00
goto err ;
2010-08-08 16:46:39 +04:00
}
readp = readbuf ;
token = NULL ;
2010-10-19 11:26:59 -04:00
/* We try to follow RFC3986 here as much as we can, and match
the productions
2010-08-08 16:46:39 +04:00
2010-10-19 11:26:59 -04:00
URI = scheme " : " hier - part [ " ? " query ] [ " # " fragment ]
2010-08-08 16:46:39 +04:00
2010-12-09 11:47:54 -05:00
relative - ref = relative - part [ " ? " query ] [ " # " fragment ]
2010-10-19 11:26:59 -04:00
*/
2010-08-08 16:46:39 +04:00
2010-10-19 11:26:59 -04:00
/* 1. scheme: */
token = strchr ( readp , ' : ' ) ;
if ( token & & scheme_ok ( readp , token ) ) {
* token = ' \0 ' ;
uri - > scheme = mm_strdup ( readp ) ;
2010-12-18 01:07:27 -02:00
if ( uri - > scheme = = NULL ) {
2010-12-18 02:40:22 -02:00
event_warn ( " %s: strdup " , __func__ ) ;
2010-12-18 01:07:27 -02:00
goto err ;
}
2010-10-19 11:26:59 -04:00
readp = token + 1 ; /* eat : */
}
2010-08-08 16:46:39 +04:00
2010-10-19 11:26:59 -04:00
/* 2. Optionally, "//" then an 'authority' part. */
if ( readp [ 0 ] = = ' / ' & & readp [ 1 ] = = ' / ' ) {
char * authority ;
readp + = 2 ;
authority = readp ;
2010-10-19 12:35:50 -04:00
path = end_of_authority ( readp ) ;
2020-10-27 01:40:34 +03:00
if ( parse_authority ( uri , authority , path , & uri - > flags ) < 0 )
2010-10-19 11:26:59 -04:00
goto err ;
readp = path ;
got_authority = 1 ;
2010-08-08 16:46:39 +04:00
}
2010-10-19 11:26:59 -04:00
/* 3. Query: path-abempty, path-absolute, path-rootless, or path-empty
*/
path = readp ;
2011-02-13 00:41:22 -05:00
readp = end_of_path ( path , PART_PATH , flags ) ;
2010-10-19 11:26:59 -04:00
/* Query */
if ( * readp = = ' ? ' ) {
* readp = ' \0 ' ;
+ + readp ;
query = readp ;
2011-02-13 00:41:22 -05:00
readp = end_of_path ( readp , PART_QUERY , flags ) ;
2010-10-19 11:26:59 -04:00
}
/* fragment */
if ( * readp = = ' # ' ) {
* readp = ' \0 ' ;
+ + readp ;
fragment = readp ;
2011-02-13 00:41:22 -05:00
readp = end_of_path ( readp , PART_FRAGMENT , flags ) ;
2010-10-19 11:26:59 -04:00
}
if ( * readp ! = ' \0 ' ) {
goto err ;
2010-08-08 16:46:39 +04:00
}
2010-10-19 12:35:50 -04:00
/* These next two cases may be unreachable; I'm leaving them
* in to be defensive . */
2010-10-19 11:26:59 -04:00
/* If you didn't get an authority, the path can't begin with "//" */
if ( ! got_authority & & path [ 0 ] = = ' / ' & & path [ 1 ] = = ' / ' )
goto err ;
/* If you did get an authority, the path must begin with "/" or be
* empty . */
if ( got_authority & & path [ 0 ] ! = ' / ' & & path [ 0 ] ! = ' \0 ' )
goto err ;
2010-10-19 12:35:50 -04:00
/* (End of maybe-unreachable cases) */
2010-10-19 11:26:59 -04:00
2010-10-19 12:35:50 -04:00
/* If there was no scheme, the first part of the path (if any) must
* have no colon in it . */
if ( ! uri - > scheme & & ! path_matches_noscheme ( path ) )
goto err ;
EVUTIL_ASSERT ( path ) ;
uri - > path = mm_strdup ( path ) ;
2010-12-18 01:07:27 -02:00
if ( uri - > path = = NULL ) {
2010-12-18 02:40:22 -02:00
event_warn ( " %s: strdup " , __func__ ) ;
2010-12-18 01:07:27 -02:00
goto err ;
}
2010-10-19 11:26:59 -04:00
2010-12-18 01:07:27 -02:00
if ( query ) {
2010-10-19 11:26:59 -04:00
uri - > query = mm_strdup ( query ) ;
2010-12-18 01:07:27 -02:00
if ( uri - > query = = NULL ) {
2010-12-18 02:40:22 -02:00
event_warn ( " %s: strdup " , __func__ ) ;
2010-12-18 01:07:27 -02:00
goto err ;
}
}
if ( fragment ) {
2010-10-19 11:26:59 -04:00
uri - > fragment = mm_strdup ( fragment ) ;
2010-12-18 01:07:27 -02:00
if ( uri - > fragment = = NULL ) {
2010-12-18 02:40:22 -02:00
event_warn ( " %s: strdup " , __func__ ) ;
2010-12-18 01:07:27 -02:00
goto err ;
}
}
2010-08-08 16:46:39 +04:00
2010-10-18 14:34:20 -04:00
mm_free ( readbuf ) ;
2010-08-08 16:46:39 +04:00
return uri ;
2010-10-18 14:43:54 -04:00
err :
if ( uri )
evhttp_uri_free ( uri ) ;
if ( readbuf )
mm_free ( readbuf ) ;
return NULL ;
2010-08-08 16:46:39 +04:00
}
2017-12-02 12:53:57 -08:00
static struct evhttp_uri *
2020-10-27 01:40:34 +03:00
evhttp_uri_parse_authority ( char * source_uri , unsigned flags )
2017-12-02 12:53:57 -08:00
{
struct evhttp_uri * uri = mm_calloc ( 1 , sizeof ( struct evhttp_uri ) ) ;
2018-07-05 12:46:51 -04:00
char * end ;
2017-12-02 12:53:57 -08:00
if ( uri = = NULL ) {
event_warn ( " %s: calloc " , __func__ ) ;
goto err ;
}
uri - > port = - 1 ;
2020-10-27 01:40:34 +03:00
uri - > flags = flags ;
2017-12-02 12:53:57 -08:00
2018-07-05 12:46:51 -04:00
end = end_of_authority ( source_uri ) ;
2020-10-27 01:40:34 +03:00
if ( parse_authority ( uri , source_uri , end , & uri - > flags ) < 0 )
2017-12-02 12:53:57 -08:00
goto err ;
uri - > path = mm_strdup ( " " ) ;
if ( uri - > path = = NULL ) {
event_warn ( " %s: strdup " , __func__ ) ;
goto err ;
}
return uri ;
err :
if ( uri )
evhttp_uri_free ( uri ) ;
return NULL ;
}
2010-10-18 14:43:54 -04:00
void
evhttp_uri_free ( struct evhttp_uri * uri )
2010-08-08 16:46:39 +04:00
{
2012-02-29 15:07:32 -05:00
# define URI_FREE_STR_(f) \
2010-08-08 16:46:39 +04:00
if ( uri - > f ) { \
2010-10-18 14:34:20 -04:00
mm_free ( uri - > f ) ; \
2010-08-08 16:46:39 +04:00
}
2012-02-29 15:07:32 -05:00
URI_FREE_STR_ ( scheme ) ;
URI_FREE_STR_ ( userinfo ) ;
URI_FREE_STR_ ( host ) ;
2016-01-31 11:31:00 +00:00
# ifndef _WIN32
URI_FREE_STR_ ( unixsocket ) ;
# endif
2012-02-29 15:07:32 -05:00
URI_FREE_STR_ ( path ) ;
URI_FREE_STR_ ( query ) ;
URI_FREE_STR_ ( fragment ) ;
2010-08-08 16:46:39 +04:00
2010-10-18 14:34:20 -04:00
mm_free ( uri ) ;
2012-02-29 15:07:32 -05:00
# undef URI_FREE_STR_
2010-08-08 16:46:39 +04:00
}
2010-10-18 14:34:20 -04:00
char *
2017-05-15 00:36:47 -07:00
evhttp_uri_join ( const struct evhttp_uri * uri , char * buf , size_t limit )
2010-08-08 16:46:39 +04:00
{
struct evbuffer * tmp = 0 ;
size_t joined_size = 0 ;
2010-10-19 11:26:59 -04:00
char * output = NULL ;
2010-08-08 16:46:39 +04:00
2012-02-29 15:07:32 -05:00
# define URI_ADD_(f) evbuffer_add(tmp, uri->f, strlen(uri->f))
2010-10-19 11:26:59 -04:00
if ( ! uri | | ! buf | | ! limit )
2010-08-08 16:46:39 +04:00
return NULL ;
tmp = evbuffer_new ( ) ;
if ( ! tmp )
return NULL ;
2010-10-19 11:26:59 -04:00
if ( uri - > scheme ) {
2012-02-29 15:07:32 -05:00
URI_ADD_ ( scheme ) ;
2010-10-19 11:26:59 -04:00
evbuffer_add ( tmp , " : " , 1 ) ;
}
2016-01-31 11:31:00 +00:00
# ifndef _WIN32
if ( uri - > unixsocket ) {
evbuffer_add ( tmp , " // " , 2 ) ;
if ( uri - > userinfo )
evbuffer_add_printf ( tmp , " %s@ " , uri - > userinfo ) ;
evbuffer_add_printf ( tmp , " unix:%s: " , uri - > unixsocket ) ;
}
else
# endif
2010-10-19 11:26:59 -04:00
if ( uri - > host ) {
evbuffer_add ( tmp , " // " , 2 ) ;
if ( uri - > userinfo )
evbuffer_add_printf ( tmp , " %s@ " , uri - > userinfo ) ;
2020-10-27 01:40:34 +03:00
if ( uri - > flags & _EVHTTP_URI_HOST_HAS_BRACKETS ) {
evbuffer_add ( tmp , " [ " , 1 ) ;
URI_ADD_ ( host ) ;
evbuffer_add ( tmp , " ] " , 1 ) ;
} else {
URI_ADD_ ( host ) ;
}
2010-10-19 11:26:59 -04:00
if ( uri - > port > = 0 )
evbuffer_add_printf ( tmp , " :%d " , uri - > port ) ;
2010-08-08 16:46:39 +04:00
2010-10-19 11:26:59 -04:00
if ( uri - > path & & uri - > path [ 0 ] ! = ' / ' & & uri - > path [ 0 ] ! = ' \0 ' )
goto err ;
2010-08-08 16:46:39 +04:00
}
2010-10-19 11:26:59 -04:00
if ( uri - > path )
2012-02-29 15:07:32 -05:00
URI_ADD_ ( path ) ;
2010-08-08 16:46:39 +04:00
2010-10-19 11:26:59 -04:00
if ( uri - > query ) {
evbuffer_add ( tmp , " ? " , 1 ) ;
2012-02-29 15:07:32 -05:00
URI_ADD_ ( query ) ;
2010-10-19 11:26:59 -04:00
}
2010-08-08 16:46:39 +04:00
2010-10-19 11:26:59 -04:00
if ( uri - > fragment ) {
2010-08-08 16:46:39 +04:00
evbuffer_add ( tmp , " # " , 1 ) ;
2012-02-29 15:07:32 -05:00
URI_ADD_ ( fragment ) ;
2010-08-08 16:46:39 +04:00
}
evbuffer_add ( tmp , " \0 " , 1 ) ; /* NUL */
joined_size = evbuffer_get_length ( tmp ) ;
2010-10-18 14:38:48 -04:00
if ( joined_size > limit ) {
/* It doesn't fit. */
evbuffer_free ( tmp ) ;
return NULL ;
2010-08-08 16:46:39 +04:00
}
2020-10-27 01:40:34 +03:00
evbuffer_remove ( tmp , buf , joined_size ) ;
2010-10-18 14:38:48 -04:00
2010-10-19 11:26:59 -04:00
output = buf ;
err :
2010-08-08 16:46:39 +04:00
evbuffer_free ( tmp ) ;
2010-10-19 11:26:59 -04:00
return output ;
2012-02-29 15:07:32 -05:00
# undef URI_ADD_
2010-08-08 16:46:39 +04:00
}
2010-10-21 14:41:12 -04:00
const char *
evhttp_uri_get_scheme ( const struct evhttp_uri * uri )
{
return uri - > scheme ;
}
const char *
evhttp_uri_get_userinfo ( const struct evhttp_uri * uri )
{
return uri - > userinfo ;
}
const char *
evhttp_uri_get_host ( const struct evhttp_uri * uri )
{
return uri - > host ;
}
2016-01-31 11:31:00 +00:00
# ifndef _WIN32
const char *
evhttp_uri_get_unixsocket ( const struct evhttp_uri * uri )
{
return uri - > unixsocket ;
}
# endif
2010-10-21 14:41:12 -04:00
int
evhttp_uri_get_port ( const struct evhttp_uri * uri )
{
return uri - > port ;
}
const char *
evhttp_uri_get_path ( const struct evhttp_uri * uri )
{
return uri - > path ;
}
const char *
evhttp_uri_get_query ( const struct evhttp_uri * uri )
{
return uri - > query ;
}
const char *
evhttp_uri_get_fragment ( const struct evhttp_uri * uri )
{
return uri - > fragment ;
}
2012-02-29 15:07:32 -05:00
# define URI_SET_STR_(f) do { \
2010-10-21 14:41:12 -04:00
if ( uri - > f ) \
mm_free ( uri - > f ) ; \
if ( f ) { \
if ( ( uri - > f = mm_strdup ( f ) ) = = NULL ) { \
event_warn ( " %s: strdup() " , __func__ ) ; \
return - 1 ; \
} \
} else { \
uri - > f = NULL ; \
} \
} while ( 0 )
int
evhttp_uri_set_scheme ( struct evhttp_uri * uri , const char * scheme )
{
if ( scheme & & ! scheme_ok ( scheme , scheme + strlen ( scheme ) ) )
return - 1 ;
2012-02-29 15:07:32 -05:00
URI_SET_STR_ ( scheme ) ;
2010-10-21 14:41:12 -04:00
return 0 ;
}
int
evhttp_uri_set_userinfo ( struct evhttp_uri * uri , const char * userinfo )
{
if ( userinfo & & ! userinfo_ok ( userinfo , userinfo + strlen ( userinfo ) ) )
return - 1 ;
2012-02-29 15:07:32 -05:00
URI_SET_STR_ ( userinfo ) ;
2010-10-21 14:41:12 -04:00
return 0 ;
}
int
evhttp_uri_set_host ( struct evhttp_uri * uri , const char * host )
{
2020-10-27 01:40:34 +03:00
size_t len = 0 ;
2010-10-21 14:41:12 -04:00
if ( host ) {
2020-10-27 01:40:34 +03:00
len = strlen ( host ) ;
2010-10-21 14:41:12 -04:00
if ( host [ 0 ] = = ' [ ' ) {
2020-10-27 01:40:34 +03:00
if ( ! bracket_addr_ok ( host , host + len ) )
2010-10-21 14:41:12 -04:00
return - 1 ;
} else {
2020-10-27 01:40:34 +03:00
if ( ! regname_ok ( host , host + len ) )
2010-10-21 14:41:12 -04:00
return - 1 ;
}
}
2020-10-27 01:40:34 +03:00
if ( host & & host [ 0 ] = = ' [ ' & & uri - > flags & EVHTTP_URI_HOST_STRIP_BRACKETS ) {
char * new_host ;
len - = 2 ;
new_host = mm_realloc ( uri - > host , len + 1 ) ;
if ( ! new_host ) {
free ( uri - > host ) ;
uri - > host = NULL ;
} else {
memcpy ( new_host , host + 1 , len ) ;
new_host [ len ] = ' \0 ' ;
uri - > host = new_host ;
}
uri - > flags | = _EVHTTP_URI_HOST_HAS_BRACKETS ;
} else {
URI_SET_STR_ ( host ) ;
uri - > flags & = ~ _EVHTTP_URI_HOST_HAS_BRACKETS ;
}
2010-10-21 14:41:12 -04:00
return 0 ;
}
2016-01-31 11:31:00 +00:00
# ifndef _WIN32
int
evhttp_uri_set_unixsocket ( struct evhttp_uri * uri , const char * unixsocket )
{
URI_SET_STR_ ( unixsocket ) ;
return 0 ;
}
# endif
2010-10-21 14:41:12 -04:00
int
evhttp_uri_set_port ( struct evhttp_uri * uri , int port )
{
if ( port < - 1 )
return - 1 ;
uri - > port = port ;
return 0 ;
}
2011-02-13 00:41:22 -05:00
# define end_of_cpath(cp,p,f) \
( ( const char * ) ( end_of_path ( ( ( char * ) ( cp ) ) , ( p ) , ( f ) ) ) )
2010-10-21 14:41:12 -04:00
int
evhttp_uri_set_path ( struct evhttp_uri * uri , const char * path )
{
2011-02-13 00:41:22 -05:00
if ( path & & end_of_cpath ( path , PART_PATH , uri - > flags ) ! = path + strlen ( path ) )
2010-10-21 14:41:12 -04:00
return - 1 ;
2012-02-29 15:07:32 -05:00
URI_SET_STR_ ( path ) ;
2010-10-21 14:41:12 -04:00
return 0 ;
}
int
evhttp_uri_set_query ( struct evhttp_uri * uri , const char * query )
{
2011-02-13 00:41:22 -05:00
if ( query & & end_of_cpath ( query , PART_QUERY , uri - > flags ) ! = query + strlen ( query ) )
2010-10-21 14:41:12 -04:00
return - 1 ;
2012-02-29 15:07:32 -05:00
URI_SET_STR_ ( query ) ;
2010-10-21 14:41:12 -04:00
return 0 ;
}
int
evhttp_uri_set_fragment ( struct evhttp_uri * uri , const char * fragment )
{
2011-02-13 00:41:22 -05:00
if ( fragment & & end_of_cpath ( fragment , PART_FRAGMENT , uri - > flags ) ! = fragment + strlen ( fragment ) )
2010-10-21 14:41:12 -04:00
return - 1 ;
2012-02-29 15:07:32 -05:00
URI_SET_STR_ ( fragment ) ;
2010-10-21 14:41:12 -04:00
return 0 ;
}