| /* |
| * 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; |
| |
| } |