2006-12-21 06:30:28 +00:00
|
|
|
/*
|
|
|
|
* This file is part of the Sofia-SIP package
|
|
|
|
*
|
|
|
|
* Copyright (C) 2005 Nokia Corporation.
|
|
|
|
*
|
|
|
|
* Contact: Pekka Pessi <pekka.pessi@nokia.com>
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2.1 of
|
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
|
|
* 02110-1301 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
Sync to current darcs tree:
Mon Sep 17 14:50:04 EDT 2007 Pekka.Pessi@nokia.com
* sofia-sip/sip_util.h: updated documentation
Mon Sep 17 14:50:18 EDT 2007 Pekka.Pessi@nokia.com
* sofia-sip/tport_tag.h: updated documentation
Mon Sep 17 14:50:28 EDT 2007 Pekka.Pessi@nokia.com
* soa_tag.c: updated documentation
Wed Sep 19 12:50:01 EDT 2007 Pekka.Pessi@nokia.com
* msg: updated documentation
Wed Sep 19 13:29:50 EDT 2007 Pekka.Pessi@nokia.com
* url: updated documentation
Wed Sep 19 13:32:14 EDT 2007 Pekka.Pessi@nokia.com
* nth: updated documentation
Wed Sep 19 13:32:27 EDT 2007 Pekka.Pessi@nokia.com
* nea: updated documentation
Wed Sep 19 13:33:36 EDT 2007 Pekka.Pessi@nokia.com
* http: updated documentation
Wed Sep 19 13:36:58 EDT 2007 Pekka.Pessi@nokia.com
* bnf: updated documentation
Wed Sep 19 13:38:58 EDT 2007 Pekka.Pessi@nokia.com
* nua: updated nua_stack_init_handle() prototype
Wed Sep 19 18:45:56 EDT 2007 Pekka.Pessi@nokia.com
* sip: added sip_name_addr_xtra(), sip_name_addr_dup()
Wed Sep 19 19:00:19 EDT 2007 Pekka.Pessi@nokia.com
* sip_basic.c: cleaned old crud
Thu Sep 20 13:34:04 EDT 2007 Pekka.Pessi@nokia.com
* iptsec: updated documentation
Thu Sep 20 13:36:22 EDT 2007 Pekka.Pessi@nokia.com
* tport: updated documentation
Thu Sep 20 13:36:56 EDT 2007 Pekka.Pessi@nokia.com
* su: updated documentation
Removed internal files from doxygen-generated documentation.
Thu Sep 20 13:38:29 EDT 2007 Pekka.Pessi@nokia.com
* soa: fixed documentation
Thu Sep 20 13:39:56 EDT 2007 Pekka.Pessi@nokia.com
* sdp: updated documentation
Thu Sep 20 13:40:16 EDT 2007 Pekka.Pessi@nokia.com
* ipt: updated documentation
Thu Sep 20 14:24:20 EDT 2007 Pekka.Pessi@nokia.com
* nta: updated documentation
Thu Sep 20 14:41:04 EDT 2007 Pekka.Pessi@nokia.com
* nua: updated documentation
Updated tag documentation.
Moved doxygen doc entries from sofia-sip/nua_tag.h to nua_tag.c.
Removed internal datatypes and files from the generated documents.
Wed Sep 19 13:34:20 EDT 2007 Pekka.Pessi@nokia.com
* docs: updated the generation of documentation. Updated links to header files.
Thu Sep 20 08:45:32 EDT 2007 Pekka.Pessi@nokia.com
* sip/Makefile.am: added tags to <sofia-sip/sip_extra.h>
Added check for extra tags in torture_sip.c.
Thu Sep 20 14:45:22 EDT 2007 Pekka.Pessi@nokia.com
* stun: updated documentation
Wed Jul 4 18:55:20 EDT 2007 Pekka.Pessi@nokia.com
* torture_heap.c: added tests for ##sort() and su_smoothsort()
Wed Jul 4 18:56:59 EDT 2007 Pekka.Pessi@nokia.com
* Makefile.am: added smoothsort.c
Fri Jul 13 12:38:44 EDT 2007 Pekka.Pessi@nokia.com
* sofia-sip/heap.h: heap_remove() now set()s index to 0 on removed item
Mon Jul 23 11:14:22 EDT 2007 Pekka.Pessi@nokia.com
* sofia-sip/heap.h: fixed bug in heap##remove()
If left kid was in heap but right was not, left kid was ignored.
Wed Jul 4 18:51:08 EDT 2007 Pekka.Pessi@nokia.com
* smoothsort.c: added
Wed Jul 4 18:51:34 EDT 2007 Pekka.Pessi@nokia.com
* heap.h: using su_smoothsort()
Fri Jul 6 10:20:27 EDT 2007 Pekka.Pessi@nokia.com
* smoothsort.c: added
Wed Sep 19 17:40:30 EDT 2007 Pekka.Pessi@nokia.com
* msg_parser.awk: generate two parser tables, default and extended
Wed Sep 19 18:39:45 EDT 2007 Pekka.Pessi@nokia.com
* msg_parser.awk: just generate list of extra headers
Allocate extended parser dynamically.
Wed Sep 19 18:59:59 EDT 2007 Pekka.Pessi@nokia.com
* sip: added Remote-Party-ID, P-Asserted-Identity, P-Preferred-Identity
Added functions sip_update_default_mclass() and sip_extend_mclass()
for handling the extended parser. Note that Reply-To and Alert-Info are only
available with the extended parser.
Wed Sep 19 19:05:44 EDT 2007 Pekka.Pessi@nokia.com
* RELEASE: updated
Thu Sep 20 13:38:59 EDT 2007 Pekka.Pessi@nokia.com
* sip: updated documentation
Thu Sep 20 14:17:28 EDT 2007 Pekka.Pessi@nokia.com
* docs/conformance.docs: updated
Mon Oct 1 10:11:14 EDT 2007 Pekka.Pessi@nokia.com
* tport_tag.c: re-enabled tptag_trusted
Thu Oct 4 09:21:07 EDT 2007 Pekka.Pessi@nokia.com
* su_osx_runloop.c: moved virtual function table after struct definition
Preparing for su_port_vtable_t refactoring.
Thu Oct 4 10:22:03 EDT 2007 Pekka.Pessi@nokia.com
* su_source.c: refactored initialization/deinitialization
Fri Oct 5 04:58:18 EDT 2007 Pekka Pessi <Pekka.Pessi@nokia.com>
* sip_extra.c: fixed prototypes with isize_t
Fri Oct 5 04:58:45 EDT 2007 Pekka Pessi <Pekka.Pessi@nokia.com>
* test_nta_api.c: removed warnings about signedness
Fri Oct 5 04:59:02 EDT 2007 Pekka Pessi <Pekka.Pessi@nokia.com>
* test_nua_params.c: removed warnings about constness
Fri Oct 5 07:20:26 EDT 2007 Pekka Pessi <first.lastname@nokia.com>
* su_port.h, su_root.c: cleaned argument checking
The su_root_*() and su_port_*() functions now check their arguments once
and do not assert() with NULL arguments. The sur_task->sut_port should
always be valid while su_root_t is alive.
Fri Oct 5 07:22:09 EDT 2007 Pekka Pessi <first.lastname@nokia.com>
* su: added su_root_obtain(), su_root_release() and su_root_has_thread()
When root is created with su_root_create() or cloned with su_clone_start(),
the resulting root is obtained by the calling or created thread,
respectively.
The root can be released with su_root_release() and another thread can
obtain it.
The function su_root_has_thread() can be used to check if a thread has
obtained or released the root.
Implementation upgraded the su_port_own_thread() method as su_port_thread().
Fri Oct 5 07:28:10 EDT 2007 Pekka Pessi <first.lastname@nokia.com>
* su_port.h: removed su_port_threadsafe() and su_port_yield() methods
su_port_wait_events() replaces su_port_yield().
Fri Oct 5 13:26:04 EDT 2007 Pekka Pessi <Pekka.Pessi@nokia.com>
* msg_parser.awk: not extending header structure unless needed.
Removed gawk-ish /* comments */.
Fri Oct 5 14:32:25 EDT 2007 Pekka Pessi <Pekka.Pessi@nokia.com>
* run_test_su: removed GNUisms
Fri Oct 5 14:32:47 EDT 2007 Pekka Pessi <Pekka.Pessi@nokia.com>
* Makefile.am: removed implicit check target test_urlmap
Fri Oct 5 14:22:32 EDT 2007 Pekka Pessi <first.lastname@nokia.com>
* torture_sresolv.c: use CLOCK_REALTIME if no CLOCK_PROCESS_CPUTIME_ID available
Casting timespec tv_sec to unsigned long.
Fri Oct * nua_s added handling nua_prack()
Thanks to Fabio Margarido for the patch.
Mon Oct 8 10:24:35 EDT 2007 Pekka.Pessi@nokia.com
* test_nua: added test for sf.net bug #1803686
Mon Oct 8 08:15:23 EDT 2007 Pekka.Pessi@nokia.com
* RELEASE: updated.
Mon Oct 8 09:30:36 EDT 2007 Pekka.Pessi@nokia.com
* nua_stack: added handling nua_prack()
Thanks to Fabio Margarido for the patch.
Mon Oct 8 10:24:35 EDT 2007 Pekka.Pessi@nokia.com
* test_nua: added test for sf.net bug #1803686
Mon Oct 8 10:26:31 EDT 2007 Pekka.Pessi@nokia.com
* nua: added test for nua_prack() (sf.net bug #1804248)
Avoid sending nua_i_state after nua_prack() if no SDP O/A is happening, too.
Mon Oct 8 10:32:04 EDT 2007 Mikhail Zabaluev <mikhail.zabaluev@nokia.com>
* su_source.c: don t leak the wait arrays
Mon Oct 8 10:37:11 EDT 2007 Pekka.Pessi@nokia.com
* RELEASE: updated
Wed Oct 10 11:55:21 EDT 2007 Pekka.Pessi@nokia.com
* sip_parser.c: silenced warning about extra const in sip_extend_mclass()
Wed Oct 10 11:57:08 EDT 2007 Pekka.Pessi@nokia.com
* nta_tag.c: updated tag documentation
Wed Oct 10 13:16:40 EDT 2007 Pekka.Pessi@nokia.com
* nua: fix logging crash if outbound used with application contact
Silenced warnings.
Wed Oct 10 13:30:45 EDT 2007 Pekka.Pessi@nokia.com
* msg_parser.awk: removed extra "const"
Wed Oct 10 13:31:45 EDT 2007 Pekka.Pessi@nokia.com
* Makefile.am's: fixed distclean of documentation
git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@5840 d0543943-73ff-0310-b7d9-9358b9ac24b2
2007-10-11 14:16:59 +00:00
|
|
|
/**@internal @file nth_server.c
|
2006-12-21 06:30:28 +00:00
|
|
|
* @brief HTTP server.
|
2008-12-16 18:05:22 +00:00
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
* @author Pekka Pessi <Pekka.Pessi@nokia.com>
|
2008-12-16 18:05:22 +00:00
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
* @date Created: Sat Oct 19 01:37:36 2002 ppessi
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2009-02-11 16:48:40 +00:00
|
|
|
#include <sofia-sip/su_string.h>
|
2007-05-08 19:14:10 +00:00
|
|
|
#include <sofia-sip/su.h>
|
2006-12-21 06:30:28 +00:00
|
|
|
|
|
|
|
typedef struct server_s server_t;
|
|
|
|
|
|
|
|
/** @internal SU timer argument pointer type */
|
|
|
|
#define SU_TIMER_ARG_T server_t
|
|
|
|
|
|
|
|
#include <sofia-sip/http_header.h>
|
|
|
|
#include <sofia-sip/http_status.h>
|
|
|
|
#include <sofia-sip/http_tag.h>
|
|
|
|
|
|
|
|
#include "sofia-sip/nth.h"
|
|
|
|
|
|
|
|
#include <sofia-sip/msg_date.h>
|
|
|
|
#include <sofia-sip/msg_addr.h>
|
|
|
|
#include <sofia-sip/su_tagarg.h>
|
|
|
|
|
|
|
|
#include <sofia-sip/hostdomain.h>
|
|
|
|
|
|
|
|
/* We are customer of tport_t */
|
|
|
|
#define TP_STACK_T server_t
|
|
|
|
#define TP_MAGIC_T void
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
#include <sofia-sip/tport.h>
|
|
|
|
#include <sofia-sip/htable.h>
|
|
|
|
|
|
|
|
#include <sofia-sip/auth_module.h>
|
|
|
|
|
2007-05-08 19:14:10 +00:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
#ifndef UINT32_MAX
|
|
|
|
#define UINT32_MAX (0xffffffffU)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
enum { SERVER_TICK = 1000 };
|
|
|
|
|
|
|
|
#define SERVER_VERSION "nth/" NTH_VERSION
|
|
|
|
|
|
|
|
HTABLE_DECLARE(hc_htable, hct, nth_client_t);
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
struct server_s
|
2006-12-21 06:30:28 +00:00
|
|
|
{
|
|
|
|
su_home_t srv_home[1];
|
|
|
|
su_root_t *srv_root;
|
|
|
|
|
|
|
|
su_timer_t *srv_timer;
|
|
|
|
unsigned srv_now;
|
|
|
|
|
|
|
|
msg_mclass_t const*srv_mclass;
|
|
|
|
int srv_mflags; /**< Message flags */
|
|
|
|
|
|
|
|
tport_t *srv_tports;
|
|
|
|
unsigned srv_queuesize; /**< Maximum number of queued responses */
|
|
|
|
|
|
|
|
size_t srv_max_bodylen; /**< Maximum accepted length */
|
|
|
|
|
|
|
|
unsigned srv_persistent:1; /**< Allow persistent connections */
|
|
|
|
|
|
|
|
/** Sites */
|
|
|
|
nth_site_t *srv_sites;
|
|
|
|
|
|
|
|
/* Statistics */
|
|
|
|
struct {
|
|
|
|
uint32_t st_requests; /**< Received requests */
|
|
|
|
uint32_t st_responses; /**< Sent responses */
|
|
|
|
uint32_t st_bad; /**< Bad requests */
|
|
|
|
} srv_stats[1];
|
|
|
|
|
|
|
|
http_server_t *srv_server; /**< Server header */
|
|
|
|
};
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
struct nth_site_s
|
2006-12-21 06:30:28 +00:00
|
|
|
{
|
|
|
|
nth_site_t *site_next, **site_prev;
|
|
|
|
|
|
|
|
nth_site_t *site_kids;
|
|
|
|
|
|
|
|
server_t *site_server;
|
|
|
|
auth_mod_t *site_auth;
|
|
|
|
|
|
|
|
url_t *site_url;
|
|
|
|
char const *site_path;
|
|
|
|
size_t site_path_len;
|
|
|
|
|
|
|
|
nth_request_f *site_callback;
|
|
|
|
nth_site_magic_t *site_magic;
|
|
|
|
|
|
|
|
su_time_t site_access; /**< Last request served */
|
|
|
|
|
|
|
|
/** Host header must match with server name */
|
|
|
|
unsigned site_strict : 1;
|
|
|
|
/** Site can have kids */
|
|
|
|
unsigned site_isdir : 1;
|
|
|
|
/** Site does not have domain name */
|
|
|
|
unsigned site_wildcard : 1;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct nth_request_s
|
|
|
|
{
|
|
|
|
server_t *req_server;
|
|
|
|
|
|
|
|
http_method_t req_method;
|
|
|
|
char const *req_method_name;
|
|
|
|
url_t const *req_url; /**< RequestURI */
|
|
|
|
char const *req_version;
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
tport_t *req_tport;
|
|
|
|
msg_t *req_request;
|
|
|
|
msg_t *req_response;
|
|
|
|
|
|
|
|
auth_status_t *req_as;
|
|
|
|
|
|
|
|
unsigned short req_status;
|
|
|
|
unsigned req_close : 1; /**< Client asked for close */
|
|
|
|
unsigned req_in_callback : 1;
|
|
|
|
unsigned req_destroyed : 1;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* ====================================================================== */
|
|
|
|
/* Debug log settings */
|
|
|
|
|
|
|
|
#define SU_LOG nth_server_log
|
|
|
|
|
|
|
|
#ifdef SU_DEBUG_H
|
|
|
|
#error <su_debug.h> included directly.
|
|
|
|
#endif
|
|
|
|
#include <sofia-sip/su_debug.h>
|
|
|
|
|
|
|
|
/**Environment variable determining the debug log level for @b nth
|
|
|
|
* module.
|
|
|
|
*
|
|
|
|
* The NTH_DEBUG environment variable is used to determine the debug
|
|
|
|
* logging level for @b nth module. The default level is 1.
|
2008-12-16 18:05:22 +00:00
|
|
|
*
|
2007-08-06 19:24:10 +00:00
|
|
|
* @sa <sofia-sip/su_debug.h>, nth_server_log, SOFIA_DEBUG
|
2006-12-21 06:30:28 +00:00
|
|
|
*/
|
|
|
|
extern char const NTH_DEBUG[];
|
|
|
|
|
|
|
|
#ifndef SU_DEBUG
|
|
|
|
#define SU_DEBUG 1
|
|
|
|
#endif
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
/**Debug log for @b nth module.
|
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
* The nth_server_log is the log object used by @b nth module. The level of
|
|
|
|
* #nth_server_log is set using #NTH_DEBUG environment variable.
|
|
|
|
*/
|
|
|
|
su_log_t nth_server_log[] = { SU_LOG_INIT("nth", "NTH_DEBUG", SU_DEBUG) };
|
|
|
|
|
|
|
|
#if HAVE_FUNC
|
|
|
|
#elif HAVE_FUNCTION
|
|
|
|
#define __func__ __FUNCTION__
|
|
|
|
#else
|
|
|
|
static char const __func__[] = "nth";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* ====================================================================== */
|
|
|
|
/** Server side
|
|
|
|
*/
|
|
|
|
static server_t *server_create(url_t const *url,
|
|
|
|
tag_type_t tag, tag_value_t value, ...);
|
|
|
|
void server_destroy(server_t *srv);
|
2007-05-08 19:14:10 +00:00
|
|
|
su_inline int server_timer_init(server_t *srv);
|
2006-12-21 06:30:28 +00:00
|
|
|
static void server_timer(su_root_magic_t *rm, su_timer_t *timer, server_t *srv);
|
2007-05-08 19:14:10 +00:00
|
|
|
su_inline uint32_t server_now(server_t const *srv);
|
2006-12-21 06:30:28 +00:00
|
|
|
static void server_request(server_t *srv, tport_t *tport, msg_t *msg,
|
|
|
|
void *arg, su_time_t now);
|
|
|
|
static nth_site_t **site_get_host(nth_site_t **, char const *host, char const *port);
|
|
|
|
static nth_site_t **site_get_rslot(nth_site_t *parent, char *path,
|
|
|
|
char **return_rest);
|
|
|
|
static nth_site_t *site_get_subdir(nth_site_t *parent, char const *path, char const **res);
|
|
|
|
static void server_tport_error(server_t *srv, tport_t *tport,
|
|
|
|
int errcode, char const *remote);
|
2008-12-16 18:05:22 +00:00
|
|
|
static msg_t *server_msg_create(server_t *srv, int flags,
|
2006-12-21 06:30:28 +00:00
|
|
|
char const data[], usize_t dlen,
|
|
|
|
tport_t const *tp, tp_client_t *tpc);
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
static void server_reply(server_t *srv, tport_t *tport,
|
2006-12-21 06:30:28 +00:00
|
|
|
msg_t *request, msg_t *response,
|
|
|
|
int status, char const *phrase);
|
|
|
|
|
|
|
|
static
|
|
|
|
void nth_site_request(server_t *srv,
|
|
|
|
nth_site_t *site,
|
|
|
|
tport_t *tport,
|
|
|
|
msg_t *request,
|
|
|
|
http_t *http,
|
|
|
|
char const *path,
|
|
|
|
msg_t *response);
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------------
|
|
|
|
* 5) Site functions
|
|
|
|
*/
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
/** Create a http site object.
|
2006-12-21 06:30:28 +00:00
|
|
|
*
|
|
|
|
* The function nth_site_create() allocates and initializes a web site
|
2008-12-16 18:05:22 +00:00
|
|
|
* object. A web site object can be either
|
2006-12-21 06:30:28 +00:00
|
|
|
* - a primary http server (@a parent is NULL),
|
2008-12-16 18:05:22 +00:00
|
|
|
* - a virtual http server (@a address contains hostpart), or
|
|
|
|
* - a site within a server
|
2006-12-21 06:30:28 +00:00
|
|
|
* (@a address does not have hostpart, only path part).
|
|
|
|
*
|
|
|
|
* @param parent pointer to parent site
|
|
|
|
* (NULL when creating a primary server object)
|
|
|
|
* @param callback pointer to callback function called when
|
|
|
|
* a request is received
|
|
|
|
* @param magic application context included in callback parameters
|
|
|
|
* @param address absolute or relative URI specifying the address of
|
|
|
|
* site
|
|
|
|
* @param tag, value, ... list of tagged parameters
|
|
|
|
*
|
|
|
|
* @TAGS
|
|
|
|
* If the @a parent is NULL, the list of tagged parameters must contain
|
|
|
|
* NTHTAG_ROOT() used to create the server engine. Tags supported when @a
|
|
|
|
* parent is NULL are NTHTAG_ROOT(), NTHTAG_MCLASS(), TPTAG_REUSE(),
|
|
|
|
* HTTPTAG_SERVER(), and HTTPTAG_SERVER_STR(). All the tags are passed to
|
|
|
|
* tport_tcreate() and tport_tbind(), too.
|
|
|
|
*
|
|
|
|
* @since Support for multiple sites was added to @VERSION_1_12_4
|
|
|
|
*/
|
2008-12-16 18:05:22 +00:00
|
|
|
nth_site_t *nth_site_create(nth_site_t *parent,
|
2006-12-21 06:30:28 +00:00
|
|
|
nth_request_f *callback,
|
|
|
|
nth_site_magic_t *magic,
|
|
|
|
url_string_t const *address,
|
|
|
|
tag_type_t tag, tag_value_t value,
|
|
|
|
...)
|
|
|
|
{
|
|
|
|
nth_site_t *site = NULL, **prev = NULL;
|
|
|
|
su_home_t home[SU_HOME_AUTO_SIZE(256)];
|
|
|
|
url_t *url, url0[1];
|
|
|
|
server_t *srv = NULL;
|
|
|
|
ta_list ta;
|
|
|
|
char *path = NULL;
|
|
|
|
size_t usize;
|
|
|
|
int is_host, is_path, wildcard = 0;
|
|
|
|
|
|
|
|
su_home_auto(home, sizeof home);
|
|
|
|
|
|
|
|
if (parent && url_is_string(address)) {
|
|
|
|
char const *s = (char const *)address;
|
|
|
|
size_t sep = strcspn(s, "/:");
|
|
|
|
|
|
|
|
if (parent->site_path) {
|
|
|
|
/* subpath */
|
2009-07-10 00:54:04 +00:00
|
|
|
url_init(url0, (enum url_type_e)parent->site_url->url_type);
|
2006-12-21 06:30:28 +00:00
|
|
|
url0->url_path = s;
|
|
|
|
address = (url_string_t*)url0;
|
|
|
|
}
|
|
|
|
else if (s[sep] == ':')
|
|
|
|
/* absolute URL with scheme */;
|
|
|
|
else if (s[sep] == '\0' && strchr(s, '.') && host_is_valid(s)) {
|
|
|
|
/* looks like a domain name */;
|
2009-07-10 00:54:04 +00:00
|
|
|
url_init(url0, (enum url_type_e)parent->site_url->url_type);
|
2006-12-21 06:30:28 +00:00
|
|
|
url0->url_host = s;
|
|
|
|
address = (url_string_t*)url0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* looks like a path */
|
2009-07-10 00:54:04 +00:00
|
|
|
url_init(url0, (enum url_type_e)parent->site_url->url_type);
|
2006-12-21 06:30:28 +00:00
|
|
|
url0->url_path = s;
|
|
|
|
address = (url_string_t*)url0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
url = url_hdup(home, address->us_url);
|
|
|
|
|
|
|
|
if (!url || !callback)
|
|
|
|
return NULL;
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
is_host = url->url_host != NULL;
|
|
|
|
is_path = url->url_path != NULL;
|
|
|
|
|
|
|
|
if (is_host && is_path) {
|
2012-10-26 19:15:14 +00:00
|
|
|
SU_DEBUG_3(("nth_site_create(): virtual host and path simultanously\n" VA_NONE));
|
2006-12-21 06:30:28 +00:00
|
|
|
errno = EINVAL;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!parent && !is_host) {
|
2012-10-26 19:15:14 +00:00
|
|
|
SU_DEBUG_3(("nth_site_create(): host is required\n" VA_NONE));
|
2006-12-21 06:30:28 +00:00
|
|
|
errno = EINVAL;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parent) {
|
|
|
|
if (!parent->site_isdir) {
|
2012-10-26 19:15:14 +00:00
|
|
|
SU_DEBUG_3(("nth_site_create(): invalid parent resource \n" VA_NONE));
|
2006-12-21 06:30:28 +00:00
|
|
|
errno = EINVAL;
|
|
|
|
goto error;
|
|
|
|
}
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
srv = parent->site_server; assert(srv);
|
|
|
|
if (is_host) {
|
|
|
|
prev = site_get_host(&srv->srv_sites, url->url_host, url->url_port);
|
|
|
|
|
|
|
|
if (prev == NULL) {
|
|
|
|
SU_DEBUG_3(("nth_site_create(): host %s:%s already exists\n",
|
|
|
|
url->url_host, url->url_port ? url->url_port : ""));
|
|
|
|
errno = EEXIST;
|
|
|
|
goto error;
|
|
|
|
}
|
2008-12-16 18:05:22 +00:00
|
|
|
}
|
2006-12-21 06:30:28 +00:00
|
|
|
else {
|
|
|
|
size_t i, j;
|
|
|
|
|
|
|
|
path = (char *)url->url_path;
|
|
|
|
while (path[0] == '/')
|
|
|
|
path++;
|
|
|
|
|
|
|
|
/* Remove duplicate // */
|
|
|
|
for (i = j = 0; path[i];) {
|
|
|
|
while (path[i] == '/' && path[i + 1] == '/')
|
|
|
|
i++;
|
|
|
|
path[j++] = path[i++];
|
|
|
|
}
|
|
|
|
path[j] = path[i];
|
|
|
|
|
|
|
|
url = url0, *url = *parent->site_url;
|
|
|
|
|
|
|
|
if (url->url_path) {
|
|
|
|
url->url_path = su_strcat(home, url->url_path, path);
|
|
|
|
if (!url->url_path)
|
|
|
|
goto error;
|
|
|
|
path = (char *)url->url_path + strlen(parent->site_url->url_path);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
url->url_path = path;
|
|
|
|
|
|
|
|
prev = site_get_rslot(parent, path, &path);
|
|
|
|
|
|
|
|
if (!prev || path[0] == '\0') {
|
2008-12-16 18:05:22 +00:00
|
|
|
SU_DEBUG_3(("nth_site_create(): directory \"%s\" already exists\n",
|
2006-12-21 06:30:28 +00:00
|
|
|
url->url_path));
|
|
|
|
errno = EEXIST;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!parent) {
|
|
|
|
if (strcmp(url->url_host, "*") == 0 ||
|
|
|
|
host_cmp(url->url_host, "0.0.0.0") == 0 ||
|
|
|
|
host_cmp(url->url_host, "::") == 0)
|
|
|
|
wildcard = 1, url->url_host = "*";
|
|
|
|
}
|
|
|
|
|
|
|
|
usize = sizeof(*url) + url_xtra(url);
|
|
|
|
|
|
|
|
ta_start(ta, tag, value);
|
|
|
|
|
|
|
|
if (!parent) {
|
|
|
|
srv = server_create(url, ta_tags(ta));
|
|
|
|
prev = &srv->srv_sites;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (srv && (site = su_zalloc(srv->srv_home, (sizeof *site) + usize))) {
|
|
|
|
site->site_url = (url_t *)(site + 1);
|
2008-12-16 18:05:22 +00:00
|
|
|
url_dup((void *)(site->site_url + 1), usize - sizeof(*url),
|
2006-12-21 06:30:28 +00:00
|
|
|
site->site_url, url);
|
|
|
|
|
|
|
|
assert(prev);
|
|
|
|
if ((site->site_next = *prev))
|
|
|
|
site->site_next->site_prev = &site->site_next;
|
|
|
|
*prev = site, site->site_prev = prev;
|
|
|
|
site->site_server = srv;
|
|
|
|
|
|
|
|
if (path) {
|
|
|
|
size_t path_len;
|
|
|
|
|
|
|
|
site->site_path = site->site_url->url_path + (path - url->url_path);
|
|
|
|
path_len = strlen(site->site_path); assert(path_len > 0);
|
|
|
|
if (path_len > 0 && site->site_path[path_len - 1] == '/')
|
|
|
|
path_len--, site->site_isdir = 1;
|
|
|
|
site->site_path_len = path_len;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
site->site_isdir = is_host;
|
|
|
|
site->site_path = "";
|
|
|
|
site->site_path_len = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
site->site_wildcard = wildcard;
|
|
|
|
site->site_callback = callback;
|
|
|
|
site->site_magic = magic;
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
if (parent)
|
|
|
|
site->site_auth = parent->site_auth;
|
|
|
|
|
|
|
|
nth_site_set_params(site, ta_tags(ta));
|
|
|
|
}
|
|
|
|
|
|
|
|
ta_end(ta);
|
|
|
|
|
|
|
|
error:
|
|
|
|
su_home_deinit(home);
|
|
|
|
return site;
|
|
|
|
}
|
|
|
|
|
|
|
|
void nth_site_destroy(nth_site_t *site)
|
|
|
|
{
|
|
|
|
if (site == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (site->site_auth)
|
|
|
|
auth_mod_unref(site->site_auth), site->site_auth = NULL;
|
|
|
|
|
|
|
|
if (site->site_server->srv_sites == site) {
|
|
|
|
server_destroy(site->site_server);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
nth_site_magic_t *nth_site_magic(nth_site_t const *site)
|
|
|
|
{
|
|
|
|
return site ? site->site_magic : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
void nth_site_bind(nth_site_t *site,
|
|
|
|
nth_request_f *callback,
|
2006-12-21 06:30:28 +00:00
|
|
|
nth_site_magic_t *magic)
|
|
|
|
{
|
|
|
|
if (site) {
|
|
|
|
site->site_callback = callback;
|
|
|
|
site->site_magic = magic;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** Get the site URL. @NEW_1_12_4. */
|
|
|
|
url_t const *nth_site_url(nth_site_t const *site)
|
|
|
|
{
|
|
|
|
return site ? site->site_url : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Return server name and version */
|
|
|
|
char const *nth_site_server_version(void)
|
|
|
|
{
|
|
|
|
return "nth/" NTH_VERSION;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Get the time last time served. @NEW_1_12_4. */
|
|
|
|
su_time_t nth_site_access_time(nth_site_t const *site)
|
|
|
|
{
|
|
|
|
su_time_t const never = { 0, 0 };
|
|
|
|
|
|
|
|
return site ? site->site_access : never;
|
|
|
|
}
|
|
|
|
|
|
|
|
int nth_site_set_params(nth_site_t *site,
|
|
|
|
tag_type_t tag, tag_value_t value, ...)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
ta_list ta;
|
|
|
|
|
|
|
|
server_t *server;
|
|
|
|
int master;
|
|
|
|
msg_mclass_t const *mclass;
|
|
|
|
int mflags;
|
|
|
|
auth_mod_t *am;
|
|
|
|
|
|
|
|
if (site == NULL)
|
|
|
|
return (errno = EINVAL), -1;
|
|
|
|
|
|
|
|
server = site->site_server;
|
|
|
|
master = site == server->srv_sites;
|
|
|
|
am = site->site_auth;
|
|
|
|
|
|
|
|
mclass = server->srv_mclass;
|
|
|
|
mflags = server->srv_mflags;
|
|
|
|
|
|
|
|
ta_start(ta, tag, value);
|
|
|
|
|
|
|
|
n = tl_gets(ta_args(ta),
|
|
|
|
TAG_IF(master, NTHTAG_MCLASS_REF(mclass)),
|
|
|
|
TAG_IF(master, NTHTAG_MFLAGS_REF(mflags)),
|
|
|
|
NTHTAG_AUTH_MODULE_REF(am),
|
|
|
|
TAG_END());
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
if (n > 0) {
|
|
|
|
if (mclass)
|
|
|
|
server->srv_mclass = mclass;
|
|
|
|
else
|
|
|
|
server->srv_mclass = http_default_mclass();
|
|
|
|
server->srv_mflags = mflags;
|
|
|
|
auth_mod_ref(am), auth_mod_unref(site->site_auth), site->site_auth = am;
|
|
|
|
}
|
|
|
|
|
|
|
|
ta_end(ta);
|
|
|
|
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
int nth_site_get_params(nth_site_t const *site,
|
|
|
|
tag_type_t tag, tag_value_t value, ...)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
ta_list ta;
|
|
|
|
server_t *server;
|
|
|
|
int master;
|
|
|
|
msg_mclass_t const *mclass;
|
|
|
|
|
|
|
|
if (site == NULL)
|
|
|
|
return (errno = EINVAL), -1;
|
|
|
|
|
|
|
|
server = site->site_server;
|
|
|
|
master = site == server->srv_sites;
|
|
|
|
|
|
|
|
if (master && server->srv_mclass != http_default_mclass())
|
|
|
|
mclass = server->srv_mclass;
|
|
|
|
else
|
|
|
|
mclass = NULL;
|
|
|
|
|
|
|
|
ta_start(ta, tag, value);
|
|
|
|
|
|
|
|
n = tl_tgets(ta_args(ta),
|
|
|
|
TAG_IF(master, NTHTAG_MCLASS(mclass)),
|
|
|
|
TAG_IF(master, NTHTAG_MFLAGS(server->srv_mflags)),
|
|
|
|
TAG_END());
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
ta_end(ta);
|
|
|
|
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
int nth_site_get_stats(nth_site_t const *site,
|
2006-12-21 06:30:28 +00:00
|
|
|
tag_type_t tag, tag_value_t value, ...)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
ta_list ta;
|
|
|
|
|
|
|
|
if (site == NULL)
|
|
|
|
return (errno = EINVAL), -1;
|
|
|
|
|
|
|
|
ta_start(ta, tag, value);
|
|
|
|
|
|
|
|
n = tl_tgets(ta_args(ta),
|
|
|
|
TAG_END());
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
ta_end(ta);
|
|
|
|
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
nth_site_t **site_get_host(nth_site_t **list, char const *host, char const *port)
|
|
|
|
{
|
|
|
|
nth_site_t *site;
|
|
|
|
|
|
|
|
assert(host);
|
|
|
|
|
|
|
|
for (; (site = *list); list = &site->site_next) {
|
|
|
|
if (host_cmp(host, site->site_url->url_host) == 0 &&
|
2009-02-11 16:48:40 +00:00
|
|
|
su_strcmp(port, site->site_url->url_port) == 0) {
|
2006-12-21 06:30:28 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Find a place to insert site from the hierarchy.
|
|
|
|
*
|
|
|
|
* A resource can be either a 'dir' (name ends with '/') or 'file'.
|
|
|
|
* When a resource
|
|
|
|
*/
|
|
|
|
static
|
2008-12-16 18:05:22 +00:00
|
|
|
nth_site_t **site_get_rslot(nth_site_t *parent, char *path,
|
2006-12-21 06:30:28 +00:00
|
|
|
char **return_rest)
|
|
|
|
{
|
|
|
|
nth_site_t *site, **prev;
|
|
|
|
size_t len;
|
|
|
|
int cmp;
|
|
|
|
|
|
|
|
assert(path);
|
|
|
|
|
|
|
|
if (path[0] == '\0')
|
|
|
|
return errno = EEXIST, NULL;
|
|
|
|
|
|
|
|
for (prev = &parent->site_kids; (site = *prev); prev = &site->site_next) {
|
|
|
|
cmp = strncmp(path, site->site_path, len = site->site_path_len);
|
|
|
|
if (cmp > 0)
|
|
|
|
break;
|
|
|
|
if (cmp < 0)
|
|
|
|
continue;
|
|
|
|
if (path[len] == '\0') {
|
|
|
|
if (site->site_isdir)
|
2008-12-16 18:05:22 +00:00
|
|
|
return *return_rest = path, prev;
|
2006-12-21 06:30:28 +00:00
|
|
|
return errno = EEXIST, NULL;
|
|
|
|
}
|
|
|
|
if (path[len] != '/' || site->site_path[len] != '/')
|
|
|
|
continue;
|
|
|
|
|
|
|
|
while (path[++len] == '/')
|
|
|
|
;
|
|
|
|
|
|
|
|
return site_get_rslot(site, path + len, return_rest);
|
|
|
|
}
|
|
|
|
|
|
|
|
*return_rest = path;
|
|
|
|
|
|
|
|
return prev;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char const site_nodir_match[] = "";
|
|
|
|
|
|
|
|
/** Find a subdir from site hierarchy */
|
|
|
|
static
|
|
|
|
nth_site_t *site_get_subdir(nth_site_t *parent,
|
|
|
|
char const *path,
|
|
|
|
char const **return_rest)
|
|
|
|
{
|
|
|
|
nth_site_t *site;
|
|
|
|
size_t len;
|
|
|
|
int cmp;
|
|
|
|
|
|
|
|
assert(path);
|
|
|
|
|
|
|
|
while (path[0] == '/') /* Skip multiple slashes */
|
|
|
|
path++;
|
|
|
|
|
|
|
|
if (path[0] == '\0')
|
|
|
|
return *return_rest = path, parent;
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
for (site = parent->site_kids; site; site = site->site_next) {
|
|
|
|
cmp = strncmp(path, site->site_path, len = site->site_path_len);
|
|
|
|
if (cmp > 0)
|
|
|
|
break;
|
|
|
|
if (cmp < 0)
|
|
|
|
continue;
|
|
|
|
if (path[len] == '\0')
|
|
|
|
return *return_rest = site_nodir_match, site;
|
|
|
|
if (site->site_path[len] == '/' && path[len] == '/')
|
|
|
|
return site_get_subdir(site, path + len + 1, return_rest);
|
|
|
|
}
|
|
|
|
|
|
|
|
return *return_rest = path, parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------------
|
|
|
|
* Server functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
static char const * const http_tports[] =
|
|
|
|
{
|
|
|
|
"tcp", "tls", NULL
|
|
|
|
};
|
|
|
|
|
2008-05-27 00:18:31 +00:00
|
|
|
static char const * const http_no_tls_tports[] = { "tcp", NULL };
|
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
static tp_stack_class_t nth_server_class[1] =
|
|
|
|
{{
|
|
|
|
sizeof(nth_server_class),
|
|
|
|
server_request,
|
|
|
|
server_tport_error,
|
|
|
|
server_msg_create
|
|
|
|
}};
|
|
|
|
|
|
|
|
server_t *server_create(url_t const *url,
|
|
|
|
tag_type_t tag, tag_value_t value, ...)
|
|
|
|
{
|
|
|
|
server_t *srv;
|
2007-04-15 02:03:41 +00:00
|
|
|
msg_mclass_t const *mclass = NULL;
|
2006-12-21 06:30:28 +00:00
|
|
|
tp_name_t tpn[1] = {{ NULL }};
|
|
|
|
su_root_t *root = NULL;
|
|
|
|
http_server_t const *server = NULL;
|
|
|
|
int persistent = 0;
|
|
|
|
char const *server_str = SERVER_VERSION;
|
|
|
|
ta_list ta;
|
|
|
|
|
|
|
|
ta_start(ta, tag, value);
|
2008-12-16 18:05:22 +00:00
|
|
|
tl_gets(ta_args(ta),
|
2006-12-21 06:30:28 +00:00
|
|
|
NTHTAG_ROOT_REF(root),
|
|
|
|
NTHTAG_MCLASS_REF(mclass),
|
|
|
|
TPTAG_REUSE_REF(persistent),
|
|
|
|
HTTPTAG_SERVER_REF(server),
|
|
|
|
HTTPTAG_SERVER_STR_REF(server_str),
|
|
|
|
TAG_END());
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
if (!root || !url ||
|
2006-12-21 06:30:28 +00:00
|
|
|
(url->url_type != url_http && url->url_type != url_https)
|
|
|
|
|| !(srv = su_home_new(sizeof(*srv)))) {
|
|
|
|
ta_end(ta);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2009-07-10 00:54:04 +00:00
|
|
|
tpn->tpn_proto = url_tport_default((enum url_type_e)url->url_type);
|
2006-12-21 06:30:28 +00:00
|
|
|
tpn->tpn_canon = url->url_host;
|
|
|
|
tpn->tpn_host = url->url_host;
|
|
|
|
tpn->tpn_port = url_port(url);
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
srv->srv_tports = tport_tcreate(srv, nth_server_class, root,
|
2006-12-21 06:30:28 +00:00
|
|
|
TPTAG_IDLE(600000),
|
|
|
|
TPTAG_TIMEOUT(300000),
|
|
|
|
ta_tags(ta));
|
|
|
|
|
|
|
|
srv->srv_persistent = persistent;
|
|
|
|
srv->srv_max_bodylen = 1 << 30; /* 1 GB */
|
|
|
|
|
|
|
|
if (tport_tbind(srv->srv_tports, tpn, http_tports,
|
2008-05-27 00:18:31 +00:00
|
|
|
TAG_END()) >= 0 ||
|
|
|
|
tport_tbind(srv->srv_tports, tpn, http_no_tls_tports,
|
2006-12-21 06:30:28 +00:00
|
|
|
TAG_END()) >= 0) {
|
|
|
|
srv->srv_root = root;
|
|
|
|
srv->srv_mclass = mclass ? mclass : http_default_mclass();
|
|
|
|
srv->srv_mflags = MSG_DO_CANONIC;
|
|
|
|
|
|
|
|
if (server)
|
|
|
|
srv->srv_server = http_server_dup(srv->srv_home, server);
|
|
|
|
else
|
|
|
|
srv->srv_server = http_server_make(srv->srv_home, server_str);
|
|
|
|
|
|
|
|
tport_get_params(srv->srv_tports,
|
|
|
|
TPTAG_QUEUESIZE_REF(srv->srv_queuesize),
|
|
|
|
TAG_END());
|
|
|
|
}
|
|
|
|
else {
|
2008-12-16 18:05:22 +00:00
|
|
|
SU_DEBUG_1(("nth_server_create: cannot bind transports "
|
2006-12-21 06:30:28 +00:00
|
|
|
URL_FORMAT_STRING "\n",
|
2008-12-16 18:05:22 +00:00
|
|
|
URL_PRINT_ARGS(url)));
|
2006-12-21 06:30:28 +00:00
|
|
|
server_destroy(srv), srv = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ta_end(ta);
|
|
|
|
|
|
|
|
return srv;
|
|
|
|
}
|
|
|
|
|
|
|
|
void server_destroy(server_t *srv)
|
|
|
|
{
|
|
|
|
tport_destroy(srv->srv_tports);
|
|
|
|
su_timer_destroy(srv->srv_timer);
|
|
|
|
su_home_unref(srv->srv_home);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Initialize server timer. */
|
2007-05-08 19:14:10 +00:00
|
|
|
su_inline
|
2006-12-21 06:30:28 +00:00
|
|
|
int server_timer_init(server_t *srv)
|
|
|
|
{
|
|
|
|
if (0) {
|
|
|
|
srv->srv_timer = su_timer_create(su_root_task(srv->srv_root), SERVER_TICK);
|
|
|
|
return su_timer_set(srv->srv_timer, server_timer, srv);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Server timer routine.
|
|
|
|
*/
|
|
|
|
static
|
|
|
|
void server_timer(su_root_magic_t *rm, su_timer_t *timer, server_t *srv)
|
|
|
|
{
|
|
|
|
uint32_t now;
|
|
|
|
|
|
|
|
su_timer_set(timer, server_timer, srv);
|
|
|
|
|
|
|
|
now = su_time_ms(su_now()); now += now == 0; srv->srv_now = now;
|
|
|
|
|
|
|
|
/* Xyzzy */
|
|
|
|
|
|
|
|
srv->srv_now = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Get current timestamp in milliseconds */
|
2007-05-08 19:14:10 +00:00
|
|
|
su_inline
|
2006-12-21 06:30:28 +00:00
|
|
|
uint32_t server_now(server_t const *srv)
|
|
|
|
{
|
|
|
|
if (srv->srv_now)
|
|
|
|
return srv->srv_now;
|
|
|
|
else
|
|
|
|
return su_time_ms(su_now());
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Process incoming request message */
|
|
|
|
static
|
|
|
|
void server_request(server_t *srv,
|
|
|
|
tport_t *tport,
|
|
|
|
msg_t *request,
|
|
|
|
void *arg,
|
|
|
|
su_time_t now)
|
|
|
|
{
|
|
|
|
nth_site_t *site = NULL, *subsite = NULL;
|
|
|
|
msg_t *response;
|
|
|
|
http_t *http = http_object(request);
|
|
|
|
http_host_t *h;
|
|
|
|
char const *host, *port, *path, *subpath = NULL;
|
|
|
|
|
|
|
|
/* Disable streaming */
|
|
|
|
if (msg_is_streaming(request)) {
|
2009-07-10 00:54:04 +00:00
|
|
|
msg_set_streaming(request, (enum msg_streaming_status)0);
|
2006-12-21 06:30:28 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Create a response message */
|
|
|
|
response = server_msg_create(srv, 0, NULL, 0, NULL, NULL);
|
|
|
|
tport_tqueue(tport, response, TAG_END());
|
|
|
|
|
|
|
|
if (http && http->http_flags & MSG_FLG_TIMEOUT) {
|
|
|
|
server_reply(srv, tport, request, response, 400, "Request timeout");
|
|
|
|
return;
|
|
|
|
} else if (http && http->http_flags & MSG_FLG_TOOLARGE) {
|
|
|
|
server_reply(srv, tport, request, response, HTTP_413_ENTITY_TOO_LARGE);
|
|
|
|
return;
|
2008-12-16 18:05:22 +00:00
|
|
|
} else if (!http || !http->http_request ||
|
2006-12-21 06:30:28 +00:00
|
|
|
(http->http_flags & MSG_FLG_ERROR)) {
|
|
|
|
server_reply(srv, tport, request, response, HTTP_400_BAD_REQUEST);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (http->http_request->rq_version != http_version_1_0 &&
|
|
|
|
http->http_request->rq_version != http_version_1_1) {
|
|
|
|
server_reply(srv, tport, request, response, HTTP_505_HTTP_VERSION);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
h = http->http_host;
|
|
|
|
|
|
|
|
if (h) {
|
|
|
|
host = h->h_host, port = h->h_port;
|
|
|
|
}
|
|
|
|
else if (http->http_request->rq_url->url_host) {
|
|
|
|
host = http->http_request->rq_url->url_host;
|
|
|
|
port = http->http_request->rq_url->url_port;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
host = NULL, port = NULL;
|
|
|
|
|
|
|
|
path = http->http_request->rq_url->url_path;
|
|
|
|
|
|
|
|
if (host)
|
|
|
|
site = *site_get_host(&srv->srv_sites, host, port);
|
|
|
|
|
|
|
|
if (site == NULL && !srv->srv_sites->site_strict)
|
|
|
|
site = srv->srv_sites;
|
|
|
|
|
|
|
|
if (path == NULL)
|
|
|
|
path = "";
|
|
|
|
|
|
|
|
if (path[0])
|
|
|
|
subsite = site_get_subdir(site, path, &subpath);
|
|
|
|
|
|
|
|
if (subsite)
|
|
|
|
subsite->site_access = now;
|
|
|
|
else
|
|
|
|
site->site_access = now;
|
|
|
|
|
|
|
|
if (subsite && subsite->site_isdir && subpath == site_nodir_match) {
|
|
|
|
/* Answer with 301 */
|
|
|
|
http_location_t loc[1];
|
|
|
|
http_location_init(loc);
|
|
|
|
|
|
|
|
*loc->loc_url = *site->site_url;
|
|
|
|
|
|
|
|
if (site->site_wildcard) {
|
|
|
|
if (http->http_host) {
|
|
|
|
loc->loc_url->url_host = http->http_host->h_host;
|
|
|
|
loc->loc_url->url_port = http->http_host->h_port;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
tp_name_t const *tpn = tport_name(tport); assert(tpn);
|
|
|
|
loc->loc_url->url_host = tpn->tpn_canon;
|
2009-07-10 00:54:04 +00:00
|
|
|
if (strcmp(url_port_default((enum url_type_e)loc->loc_url->url_type), tpn->tpn_port))
|
2006-12-21 06:30:28 +00:00
|
|
|
loc->loc_url->url_port = tpn->tpn_port;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
loc->loc_url->url_root = 1;
|
|
|
|
loc->loc_url->url_path = subsite->site_url->url_path;
|
|
|
|
|
|
|
|
msg_header_add_dup(response, NULL, (msg_header_t *)loc);
|
|
|
|
|
|
|
|
server_reply(srv, tport, request, response, HTTP_301_MOVED_PERMANENTLY);
|
|
|
|
}
|
|
|
|
else if (subsite)
|
|
|
|
nth_site_request(srv, subsite, tport, request, http, subpath, response);
|
|
|
|
else if (site)
|
|
|
|
nth_site_request(srv, site, tport, request, http, path, response);
|
|
|
|
else
|
|
|
|
/* Answer with 404 */
|
|
|
|
server_reply(srv, tport, request, response, HTTP_404_NOT_FOUND);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void server_tport_error(server_t *srv,
|
|
|
|
tport_t *tport,
|
|
|
|
int errcode,
|
|
|
|
char const *remote)
|
|
|
|
{
|
|
|
|
su_log("\nth: tport: %s%s%s\n",
|
|
|
|
remote ? remote : "", remote ? ": " : "",
|
|
|
|
su_strerror(errcode));
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Respond without creating a request structure */
|
2008-12-16 18:05:22 +00:00
|
|
|
static void server_reply(server_t *srv, tport_t *tport,
|
2006-12-21 06:30:28 +00:00
|
|
|
msg_t *request, msg_t *response,
|
|
|
|
int status, char const *phrase)
|
|
|
|
{
|
|
|
|
http_t *http;
|
|
|
|
http_payload_t *pl;
|
|
|
|
int close;
|
|
|
|
http_status_t st[1];
|
|
|
|
char const *req_version = NULL;
|
|
|
|
|
|
|
|
if (status < 200 || status >= 600)
|
|
|
|
status = 500, phrase = http_500_internal_server;
|
|
|
|
|
|
|
|
http = http_object(request);
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
if (http && http->http_request)
|
|
|
|
req_version = http->http_request->rq_version;
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
close = status >= 200 &&
|
2006-12-21 06:30:28 +00:00
|
|
|
(!srv->srv_persistent
|
|
|
|
|| status == 400
|
2008-12-16 18:05:22 +00:00
|
|
|
|| (http && http->http_request &&
|
2006-12-21 06:30:28 +00:00
|
|
|
http->http_request->rq_version != http_version_1_1)
|
2008-12-16 18:05:22 +00:00
|
|
|
|| (http && http->http_connection &&
|
2006-12-21 06:30:28 +00:00
|
|
|
msg_params_find(http->http_connection->k_items, "close")));
|
|
|
|
|
|
|
|
msg_destroy(request);
|
|
|
|
|
|
|
|
http = http_object(response);
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
pl = http_payload_format(msg_home(response),
|
2006-12-21 06:30:28 +00:00
|
|
|
"<html>\n"
|
|
|
|
"<head><title>%u %s</title></head>\n"
|
|
|
|
"<body><h2>%u %s</h2></body>\n"
|
|
|
|
"</html>\n", status, phrase, status, phrase);
|
|
|
|
|
|
|
|
msg_header_insert(response, (msg_pub_t *)http, (msg_header_t *)pl);
|
|
|
|
|
|
|
|
if (req_version != http_version_0_9) {
|
|
|
|
http_status_init(st);
|
|
|
|
st->st_version = http_version_1_1;
|
|
|
|
st->st_status = status;
|
|
|
|
st->st_phrase = phrase;
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
http_add_tl(response, http,
|
|
|
|
HTTPTAG_STATUS(st),
|
|
|
|
HTTPTAG_SERVER(srv->srv_server),
|
|
|
|
HTTPTAG_CONTENT_TYPE_STR("text/html"),
|
|
|
|
HTTPTAG_SEPARATOR_STR("\r\n"),
|
|
|
|
TAG_IF(close, HTTPTAG_CONNECTION_STR("close")),
|
|
|
|
TAG_END());
|
|
|
|
|
|
|
|
msg_serialize(response, (msg_pub_t *)http);
|
|
|
|
} else {
|
|
|
|
/* Just send the response */
|
|
|
|
*msg_chain_head(response) = (msg_header_t *)pl;
|
|
|
|
close = 1;
|
|
|
|
}
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
if (tport_tqsend(tport, response, NULL,
|
|
|
|
TPTAG_CLOSE_AFTER(close),
|
2006-12-21 06:30:28 +00:00
|
|
|
TAG_END()) == -1) {
|
2012-10-26 19:15:14 +00:00
|
|
|
SU_DEBUG_3(("server_reply(): cannot queue response\n" VA_NONE));
|
2006-12-21 06:30:28 +00:00
|
|
|
tport_shutdown(tport, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
msg_destroy(response);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Create a new message for transport */
|
|
|
|
static
|
2008-12-16 18:05:22 +00:00
|
|
|
msg_t *server_msg_create(server_t *srv, int flags,
|
2006-12-21 06:30:28 +00:00
|
|
|
char const data[], usize_t dlen,
|
|
|
|
tport_t const *tp, tp_client_t *tpc)
|
|
|
|
{
|
|
|
|
msg_t *msg = msg_create(srv->srv_mclass, srv->srv_mflags | flags);
|
|
|
|
|
|
|
|
return msg;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------------
|
2008-12-16 18:05:22 +00:00
|
|
|
* 6) Server transactions
|
2006-12-21 06:30:28 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
struct auth_info
|
|
|
|
{
|
|
|
|
nth_site_t *site;
|
|
|
|
nth_request_t *req;
|
|
|
|
http_t const *http;
|
|
|
|
char const *path;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void nth_authentication_result(void *ai0, auth_status_t *as);
|
|
|
|
|
|
|
|
static
|
|
|
|
void nth_site_request(server_t *srv,
|
|
|
|
nth_site_t *site,
|
|
|
|
tport_t *tport,
|
|
|
|
msg_t *request,
|
|
|
|
http_t *http,
|
|
|
|
char const *path,
|
|
|
|
msg_t *response)
|
|
|
|
{
|
|
|
|
auth_mod_t *am = site->site_auth;
|
|
|
|
nth_request_t *req;
|
|
|
|
auth_status_t *as;
|
|
|
|
struct auth_info *ai;
|
|
|
|
size_t size = (am ? (sizeof *as) + (sizeof *ai) : 0) + (sizeof *req);
|
|
|
|
int status;
|
|
|
|
|
|
|
|
req = su_zalloc(srv->srv_home, size);
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
if (req == NULL) {
|
|
|
|
server_reply(srv, tport, request, response, HTTP_500_INTERNAL_SERVER);
|
|
|
|
return;
|
|
|
|
}
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
if (am)
|
|
|
|
as = auth_status_init(req + 1, sizeof *as), ai = (void *)(as + 1);
|
|
|
|
else
|
|
|
|
as = NULL, ai = NULL;
|
|
|
|
|
|
|
|
req->req_server = srv;
|
|
|
|
req->req_method = http->http_request->rq_method;
|
|
|
|
req->req_method_name = http->http_request->rq_method_name;
|
|
|
|
req->req_url = http->http_request->rq_url;
|
|
|
|
req->req_version = http->http_request->rq_version;
|
|
|
|
|
|
|
|
req->req_tport = tport_incref(tport);
|
|
|
|
req->req_request = request;
|
|
|
|
req->req_response = response;
|
|
|
|
|
|
|
|
req->req_status = 100;
|
2008-12-16 18:05:22 +00:00
|
|
|
req->req_close =
|
2006-12-21 06:30:28 +00:00
|
|
|
!srv->srv_persistent
|
|
|
|
|| http->http_request->rq_version != http_version_1_1
|
2008-12-16 18:05:22 +00:00
|
|
|
|| (http->http_connection &&
|
2006-12-21 06:30:28 +00:00
|
|
|
msg_params_find(http->http_connection->k_items, "close"));
|
|
|
|
|
|
|
|
if (am) {
|
2008-12-16 18:05:22 +00:00
|
|
|
static auth_challenger_t const http_server_challenger[] =
|
2006-12-21 06:30:28 +00:00
|
|
|
{{ HTTP_401_UNAUTHORIZED, http_www_authenticate_class }};
|
|
|
|
|
|
|
|
req->req_as = as;
|
|
|
|
|
|
|
|
as->as_method = http->http_request->rq_method_name;
|
|
|
|
as->as_uri = path;
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
if (http->http_payload) {
|
|
|
|
as->as_body = http->http_payload->pl_data;
|
|
|
|
as->as_bodylen = http->http_payload->pl_len;
|
|
|
|
}
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
auth_mod_check_client(am, as,
|
2006-12-21 06:30:28 +00:00
|
|
|
http->http_authorization,
|
|
|
|
http_server_challenger);
|
|
|
|
|
|
|
|
if (as->as_status == 100) {
|
|
|
|
/* Stall transport - do not read more requests */
|
|
|
|
if (tport_queuelen(tport) * 2 >= srv->srv_queuesize)
|
|
|
|
tport_stall(tport);
|
|
|
|
|
|
|
|
as->as_callback = nth_authentication_result;
|
|
|
|
as->as_magic = ai;
|
|
|
|
ai->site = site;
|
|
|
|
ai->req = req;
|
|
|
|
ai->http = http;
|
|
|
|
ai->path = path;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else if (as->as_status) {
|
|
|
|
assert(as->as_status >= 200);
|
|
|
|
nth_request_treply(req, as->as_status, as->as_phrase,
|
|
|
|
HTTPTAG_HEADER((http_header_t *)as->as_response),
|
|
|
|
HTTPTAG_HEADER((http_header_t *)as->as_info),
|
|
|
|
TAG_END());
|
|
|
|
nth_request_destroy(req);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
req->req_in_callback = 1;
|
|
|
|
status = site->site_callback(site->site_magic, site, req, http, path);
|
|
|
|
req->req_in_callback = 0;
|
|
|
|
|
|
|
|
if (status != 0 && (status < 100 || status >= 600))
|
|
|
|
status = 500;
|
|
|
|
|
|
|
|
if (status != 0 && req->req_status < 200) {
|
|
|
|
nth_request_treply(req, status, NULL, TAG_END());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (req->req_status < 100) {
|
|
|
|
/* Stall transport - do not read more requests */
|
|
|
|
if (tport_queuelen(tport) * 2 >= srv->srv_queuesize)
|
|
|
|
tport_stall(tport);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status >= 200 || req->req_destroyed)
|
|
|
|
nth_request_destroy(req);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void nth_authentication_result(void *ai0, auth_status_t *as)
|
|
|
|
{
|
|
|
|
struct auth_info *ai = ai0;
|
|
|
|
nth_request_t *req = ai->req;
|
|
|
|
int status;
|
|
|
|
|
|
|
|
if (as->as_status != 0) {
|
|
|
|
assert(as->as_status >= 300);
|
2008-12-16 18:05:22 +00:00
|
|
|
nth_request_treply(req, status = as->as_status, as->as_phrase,
|
|
|
|
HTTPTAG_HEADER((http_header_t *)as->as_response),
|
2006-12-21 06:30:28 +00:00
|
|
|
TAG_END());
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
req->req_in_callback = 1;
|
2008-12-16 18:05:22 +00:00
|
|
|
status = ai->site->site_callback(ai->site->site_magic,
|
2006-12-21 06:30:28 +00:00
|
|
|
ai->site,
|
|
|
|
ai->req,
|
|
|
|
ai->http,
|
|
|
|
ai->path);
|
|
|
|
req->req_in_callback = 0;
|
|
|
|
|
|
|
|
if (status != 0 && (status < 100 || status >= 600))
|
|
|
|
status = 500;
|
|
|
|
|
|
|
|
if (status != 0 && req->req_status < 200) {
|
|
|
|
nth_request_treply(req, status, NULL, TAG_END());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status >= 200 || req->req_destroyed)
|
|
|
|
nth_request_destroy(req);
|
|
|
|
}
|
|
|
|
|
|
|
|
void nth_request_destroy(nth_request_t *req)
|
|
|
|
{
|
|
|
|
if (req == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (req->req_status < 200)
|
|
|
|
nth_request_treply(req, HTTP_500_INTERNAL_SERVER, TAG_END());
|
|
|
|
|
|
|
|
req->req_destroyed = 1;
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
if (req->req_in_callback)
|
2006-12-21 06:30:28 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (req->req_as)
|
|
|
|
su_home_deinit(req->req_as->as_home);
|
|
|
|
|
|
|
|
tport_decref(&req->req_tport), req->req_tport = NULL;
|
|
|
|
msg_destroy(req->req_request), req->req_request = NULL;
|
|
|
|
msg_destroy(req->req_response), req->req_response = NULL;
|
|
|
|
su_free(req->req_server->srv_home, req);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Return request authentication status.
|
|
|
|
*
|
|
|
|
* @param req pointer to HTTP request object
|
2008-12-16 18:05:22 +00:00
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
* @retval Status code
|
|
|
|
*
|
|
|
|
* @since New in @VERSION_1_12_4
|
|
|
|
*/
|
|
|
|
int nth_request_status(nth_request_t const *req)
|
|
|
|
{
|
|
|
|
return req ? req->req_status : 400;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Return request authentication status.
|
|
|
|
*
|
|
|
|
* @param req pointer to HTTP request object
|
2008-12-16 18:05:22 +00:00
|
|
|
*
|
2006-12-21 06:30:28 +00:00
|
|
|
* @retval Pointer to authentication status struct
|
|
|
|
*
|
|
|
|
* @note The authentication status struct is freed when the #nth_request_t
|
|
|
|
* object is destroyed.
|
|
|
|
*
|
|
|
|
* @since New in @VERSION_1_12_4
|
|
|
|
*
|
|
|
|
* @sa AUTH
|
|
|
|
*/
|
|
|
|
auth_status_t *nth_request_auth(nth_request_t const *req)
|
|
|
|
{
|
|
|
|
return req ? req->req_as : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
http_method_t nth_request_method(nth_request_t const *req)
|
|
|
|
{
|
|
|
|
return req ? req->req_method : http_method_invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
msg_t *nth_request_message(nth_request_t *req)
|
|
|
|
{
|
|
|
|
msg_t *retval = NULL;
|
|
|
|
|
|
|
|
if (req)
|
|
|
|
retval = msg_ref_create(req->req_request);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
int nth_request_treply(nth_request_t *req,
|
|
|
|
int status, char const *phrase,
|
2006-12-21 06:30:28 +00:00
|
|
|
tag_type_t tag, tag_value_t value, ...)
|
|
|
|
{
|
|
|
|
msg_t *response, *next = NULL;
|
|
|
|
http_t *http;
|
|
|
|
int retval = -1;
|
|
|
|
int req_close, close;
|
|
|
|
ta_list ta;
|
|
|
|
http_header_t const *as_info = NULL;
|
|
|
|
|
|
|
|
if (req == NULL || status < 100 || status >= 600) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
response = req->req_response;
|
|
|
|
http = http_object(response);
|
|
|
|
|
|
|
|
if (status >= 200 && req->req_as)
|
|
|
|
as_info = (http_header_t const *)req->req_as->as_info;
|
|
|
|
|
|
|
|
ta_start(ta, tag, value);
|
|
|
|
|
|
|
|
http_add_tl(response, http,
|
|
|
|
HTTPTAG_SERVER(req->req_server->srv_server),
|
2008-12-16 18:05:22 +00:00
|
|
|
HTTPTAG_HEADER(as_info),
|
2006-12-21 06:30:28 +00:00
|
|
|
ta_tags(ta));
|
|
|
|
|
|
|
|
if (http->http_payload && !http->http_content_length) {
|
|
|
|
http_content_length_t *l;
|
|
|
|
http_payload_t *pl;
|
|
|
|
size_t len = 0;
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
for (pl = http->http_payload; pl; pl = pl->pl_next)
|
|
|
|
len += pl->pl_len;
|
|
|
|
|
|
|
|
if (len > UINT32_MAX)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
l = http_content_length_create(msg_home(response), (uint32_t)len);
|
|
|
|
|
|
|
|
msg_header_insert(response, (msg_pub_t *)http, (msg_header_t *)l);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (req->req_method == http_method_head && http->http_payload) {
|
|
|
|
http_payload_t *pl;
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
for (pl = http->http_payload; pl; pl = pl->pl_next)
|
|
|
|
msg_header_remove(response, (msg_pub_t *)http, (msg_header_t *)pl);
|
|
|
|
}
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
http_complete_response(response, status, phrase,
|
2006-12-21 06:30:28 +00:00
|
|
|
http_object(req->req_request));
|
|
|
|
|
|
|
|
if (!http->http_date) {
|
|
|
|
http_date_t date[1];
|
|
|
|
http_date_init(date)->d_time = msg_now();
|
|
|
|
msg_header_add_dup(response, (msg_pub_t *)http, (msg_header_t*)date);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status < 200) {
|
|
|
|
close = 0;
|
|
|
|
next = server_msg_create(req->req_server, 0, NULL, 0, NULL, NULL);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
req_close = req->req_close;
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
close = (http->http_connection &&
|
2006-12-21 06:30:28 +00:00
|
|
|
msg_params_find(http->http_connection->k_items, "close"));
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
if (req_close && !close && status >= 200) {
|
|
|
|
close = 1;
|
|
|
|
http_add_tl(response, http, HTTPTAG_CONNECTION_STR("close"), TAG_END());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
msg_serialize(response, (msg_pub_t *)http);
|
|
|
|
|
2008-12-16 18:05:22 +00:00
|
|
|
retval = tport_tqsend(req->req_tport, response, next,
|
2006-12-21 06:30:28 +00:00
|
|
|
TAG_IF(close, TPTAG_CLOSE_AFTER(1)),
|
|
|
|
ta_tags(ta));
|
|
|
|
|
|
|
|
fail:
|
|
|
|
ta_end(ta);
|
2008-12-16 18:05:22 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
if (retval == 0)
|
|
|
|
req->req_status = status;
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|