blob: 67f95deb52dc28b2cb795b2c3e9b269ba23090b4 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <platforms/axutil_platform_auto_sense.h>
#include <axis2_http_svr_thread.h>
#include <axis2_http_server.h>
#include <axis2_http_transport.h>
#include <axutil_string.h>
#include <axutil_network_handler.h>
#include <axis2_http_simple_request.h>
#include <axis2_simple_http_svr_conn.h>
#include <axutil_url.h>
#include <axutil_error_default.h>
#include <axiom_xml_reader.h>
#include <signal.h>
AXIS2_EXPORT int axis2_http_socket_read_timeout = AXIS2_HTTP_DEFAULT_SO_TIMEOUT;
struct axis2_http_svr_thread
{
int listen_socket;
axis2_bool_t stopped;
axis2_http_worker_t *worker;
int port;
};
typedef struct axis2_http_svr_thd_args
{
axutil_env_t *env;
axis2_socket_t socket;
axis2_http_worker_t *worker;
axutil_thread_t *thread;
} axis2_http_svr_thd_args_t;
static void *AXIS2_THREAD_FUNC
axis2_svr_thread_worker_func(
axutil_thread_t * thd,
void *data);
axis2_http_svr_thread_t *AXIS2_CALL
axis2_http_svr_thread_create(
const axutil_env_t * env,
int port)
{
axis2_http_svr_thread_t *svr_thread = NULL;
svr_thread = (axis2_http_svr_thread_t *)AXIS2_MALLOC(env->allocator,
sizeof(axis2_http_svr_thread_t));
if(!svr_thread)
{
AXIS2_HANDLE_ERROR(env, AXIS2_ERROR_NO_MEMORY, AXIS2_FAILURE);
return NULL;
}
memset((void *)svr_thread, 0, sizeof(axis2_http_svr_thread_t));
svr_thread->port = port;
svr_thread->listen_socket = (int)axutil_network_handler_create_server_socket(env,
svr_thread->port);
if(-1 == svr_thread->listen_socket)
{
AXIS2_LOG_ERROR(env->log, AXIS2_LOG_SI, "Http server thread socket creation failed.");
axis2_http_svr_thread_free((axis2_http_svr_thread_t *)svr_thread, env);
return NULL;
}
return svr_thread;
}
void AXIS2_CALL
axis2_http_svr_thread_free(
axis2_http_svr_thread_t * svr_thread,
const axutil_env_t * env)
{
if(!svr_thread)
{
return;
}
if(svr_thread->worker)
{
axis2_http_worker_free(svr_thread->worker, env);
svr_thread->worker = NULL;
}
if(-1 != svr_thread->listen_socket)
{
axutil_network_handler_close_socket(env, svr_thread->listen_socket);
svr_thread->listen_socket = -1;
}
svr_thread->stopped = AXIS2_TRUE;
AXIS2_FREE(env->allocator, svr_thread);
}
axis2_status_t AXIS2_CALL
axis2_http_svr_thread_run(
axis2_http_svr_thread_t * svr_thread,
const axutil_env_t * env)
{
while(AXIS2_FALSE == svr_thread->stopped)
{
int socket = -1;
axis2_http_svr_thd_args_t *arg_list = NULL;
#ifdef AXIS2_SVR_MULTI_THREADED
axutil_thread_t *worker_thread = NULL;
#endif
socket = (int)axutil_network_handler_svr_socket_accept(env, svr_thread-> listen_socket);
if(!svr_thread->worker)
{
AXIS2_LOG_WARNING(env->log, AXIS2_LOG_SI,
"Worker not ready yet. Cannot serve the request");
axutil_network_handler_close_socket(env, socket);
continue;
}
arg_list = AXIS2_MALLOC(env->allocator, sizeof(axis2_http_svr_thd_args_t));
if(!arg_list)
{
AXIS2_LOG_ERROR(env->log, AXIS2_LOG_SI,
"Memory allocation error in the svr thread loop");
continue;
}
arg_list->env = (axutil_env_t *)env;
arg_list->socket = socket;
arg_list->worker = svr_thread->worker;
#ifdef AXIS2_SVR_MULTI_THREADED
worker_thread = axutil_thread_pool_get_thread(env->thread_pool,
axis2_svr_thread_worker_func, (void *) arg_list);
if (!worker_thread)
{
AXIS2_LOG_ERROR(env->log, AXIS2_LOG_SI, "Thread creation failed server thread loop");
continue;
}
axutil_thread_pool_thread_detach(env->thread_pool, worker_thread);
#else
axis2_svr_thread_worker_func(NULL, (void *)arg_list);
#endif
}
return AXIS2_SUCCESS;
}
AXIS2_EXTERN axis2_status_t AXIS2_CALL
axis2_http_svr_thread_destroy(
axis2_http_svr_thread_t * svr_thread,
const axutil_env_t * env)
{
if(AXIS2_TRUE == svr_thread->stopped)
{
return AXIS2_SUCCESS;
}
svr_thread->stopped = AXIS2_TRUE;
AXIS2_LOG_DEBUG(env->log, AXIS2_LOG_SI, "Terminating HTTP server thread.");
if(svr_thread->listen_socket)
{
axutil_network_handler_close_socket(env, svr_thread->listen_socket);
svr_thread->listen_socket = -1;
}
return AXIS2_SUCCESS;
}
AXIS2_EXTERN int AXIS2_CALL
axis2_http_svr_thread_get_local_port(
const axis2_http_svr_thread_t * svr_thread,
const axutil_env_t * env)
{
return svr_thread->port;
}
AXIS2_EXTERN axis2_bool_t AXIS2_CALL
axis2_http_svr_thread_is_running(
axis2_http_svr_thread_t * svr_thread,
const axutil_env_t * env)
{
return !(svr_thread->stopped);
}
AXIS2_EXTERN axis2_status_t AXIS2_CALL
axis2_http_svr_thread_set_worker(
axis2_http_svr_thread_t * svr_thread,
const axutil_env_t * env,
axis2_http_worker_t * worker)
{
AXIS2_PARAM_CHECK(env->error, worker, AXIS2_FAILURE);
svr_thread->worker = worker;
return AXIS2_SUCCESS;
}
/**
* Thread worker function.
*/
static void *AXIS2_THREAD_FUNC
axis2_svr_thread_worker_func(
axutil_thread_t * thd,
void *data)
{
struct AXIS2_PLATFORM_TIMEB t1, t2;
axis2_simple_http_svr_conn_t *svr_conn = NULL;
axis2_http_simple_request_t *request = NULL;
int millisecs = 0;
double secs = 0;
axis2_http_worker_t *tmp = NULL;
axis2_status_t status = AXIS2_FAILURE;
axutil_env_t *env = NULL;
axis2_socket_t socket;
axutil_env_t *thread_env = NULL;
axis2_http_svr_thd_args_t *arg_list = NULL;
#ifndef WIN32
#ifdef AXIS2_SVR_MULTI_THREADED
signal(SIGPIPE, SIG_IGN);
#endif
#endif
if(!data)
{
return NULL;
}
arg_list = (axis2_http_svr_thd_args_t *)data;
env = arg_list->env;
thread_env = axutil_init_thread_env(env);
IF_AXIS2_LOG_DEBUG_ENABLED(env->log)
{
AXIS2_PLATFORM_GET_TIME_IN_MILLIS(&t1);
}
socket = arg_list->socket;
svr_conn = axis2_simple_http_svr_conn_create(thread_env, (int)socket);
if(!svr_conn)
{
AXIS2_LOG_ERROR(env->log, AXIS2_LOG_SI, "creating simple_http_svr_connection failed");
axutil_free_thread_env(thread_env);
thread_env = NULL;
#ifdef AXIS2_SVR_MULTI_THREADED
axutil_thread_pool_exit_thread(env->thread_pool, thd);
#endif
return NULL;
}
axis2_simple_http_svr_conn_set_rcv_timeout(svr_conn, thread_env, axis2_http_socket_read_timeout);
/* read HTTPMethod, URL, HTTP Version and http headers. Leave the remaining in the stream */
request = axis2_simple_http_svr_conn_read_request(svr_conn, thread_env);
if(!request)
{
AXIS2_LOG_ERROR(env->log, AXIS2_LOG_SI, "Could not create request");
return NULL;
}
tmp = arg_list->worker;
status = axis2_http_worker_process_request(tmp, thread_env, svr_conn, request);
axis2_simple_http_svr_conn_free(svr_conn, thread_env);
axis2_http_simple_request_free(request, thread_env);
IF_AXIS2_LOG_DEBUG_ENABLED(env->log)
{
AXIS2_PLATFORM_GET_TIME_IN_MILLIS(&t2);
millisecs = t2.millitm - t1.millitm;
secs = difftime(t2.time, t1.time);
if(millisecs < 0)
{
millisecs += 1000;
secs--;
}
secs += millisecs / 1000.0;
#if defined(WIN32)
AXIS2_LOG_DEBUG(thread_env->log, AXIS2_LOG_SI, "Request processed...");
#else
AXIS2_LOG_DEBUG(thread_env->log, AXIS2_LOG_SI, "Request processed in %.3f seconds", secs);
#endif
}
if(status == AXIS2_SUCCESS)
{
AXIS2_LOG_DEBUG(thread_env->log, AXIS2_LOG_SI, "Request served successfully");
}
else
{
AXIS2_LOG_WARNING(thread_env->log, AXIS2_LOG_SI, "Error occurred in processing request ");
}
AXIS2_FREE(thread_env->allocator, arg_list);
axutil_free_thread_env(thread_env);
thread_env = NULL;
#ifdef AXIS2_SVR_MULTI_THREADED
axutil_thread_pool_exit_thread(env->thread_pool, thd);
#endif
return NULL;
}
AXIS2_EXTERN int AXIS2_CALL
axis2_http_svr_thread_get_listen_socket(
axis2_http_svr_thread_t *svr_thread,
const axutil_env_t *env)
{
return svr_thread->listen_socket;
}