blob: 95eafa1f4b82049825dcd3cf60d883745dc4eb5a [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 "axis2_callback_recv.h"
#include <axis2_const.h>
#include <axutil_hash.h>
struct axis2_callback_recv
{
/** base context struct */
axis2_msg_recv_t *base;
axis2_bool_t base_deep_copy;
/** callback map */
axutil_hash_t *callback_map;
axutil_thread_mutex_t *mutex;
};
static axis2_status_t AXIS2_CALL
axis2_callback_recv_receive(
axis2_msg_recv_t *msg_recv,
const axutil_env_t *env,
axis2_msg_ctx_t *msg_ctx,
void *callback_recv_param);
AXIS2_EXTERN axis2_callback_recv_t *AXIS2_CALL
axis2_callback_recv_create(
const axutil_env_t *env)
{
axis2_callback_recv_t *callback_recv = NULL;
AXIS2_ENV_CHECK(env, NULL);
callback_recv =
AXIS2_MALLOC(env->allocator, sizeof(axis2_callback_recv_t));
if (!callback_recv)
{
AXIS2_ERROR_SET(env->error, AXIS2_ERROR_NO_MEMORY, AXIS2_FAILURE);
return NULL;
}
callback_recv->base = NULL;
callback_recv->base_deep_copy = AXIS2_TRUE;
callback_recv->callback_map = NULL;
callback_recv->mutex = NULL;
callback_recv->base = axis2_msg_recv_create(env);
if (!(callback_recv->base))
{
axis2_callback_recv_free(callback_recv, env);
return NULL;
}
axis2_msg_recv_set_derived(callback_recv->base, env, callback_recv);
axis2_msg_recv_set_receive(callback_recv->base, env, axis2_callback_recv_receive);
callback_recv->callback_map = axutil_hash_make(env);
if (!(callback_recv->callback_map))
{
axis2_callback_recv_free(callback_recv, env);
return NULL;
}
callback_recv->mutex = axutil_thread_mutex_create(env->allocator,
AXIS2_THREAD_MUTEX_DEFAULT);
return callback_recv;
}
AXIS2_EXTERN axis2_msg_recv_t *AXIS2_CALL
axis2_callback_recv_get_base(
axis2_callback_recv_t *callback_recv,
const axutil_env_t *env)
{
callback_recv->base_deep_copy = AXIS2_FALSE;
return callback_recv->base;
}
AXIS2_EXTERN void AXIS2_CALL
axis2_callback_recv_free(
axis2_callback_recv_t *callback_recv,
const axutil_env_t *env)
{
AXIS2_ENV_CHECK(env, void);
if(callback_recv->mutex)
{
axutil_thread_mutex_destroy(callback_recv->mutex);
}
if (callback_recv->callback_map)
{
axutil_hash_index_t *hi = NULL;
const void *key = NULL;
void *val = NULL;
for (hi = axutil_hash_first(callback_recv->callback_map, env); hi;
hi = axutil_hash_next(env, hi))
{
axutil_hash_this(hi, &key, NULL, &val);
if (key)
AXIS2_FREE(env->allocator, (char*)key);
}
axutil_hash_free(callback_recv->callback_map, env);
}
if (callback_recv->base && callback_recv->base_deep_copy)
{
axis2_msg_recv_free(callback_recv->base, env);
}
if (callback_recv)
{
AXIS2_FREE(env->allocator, callback_recv);
}
return;
}
AXIS2_EXTERN axis2_status_t AXIS2_CALL
axis2_callback_recv_add_callback(
axis2_callback_recv_t *callback_recv,
const axutil_env_t *env,
const axis2_char_t *msg_id,
axis2_callback_t *callback)
{
AXIS2_ENV_CHECK(env, AXIS2_FAILURE);
if (msg_id)
{
axis2_char_t *mid = axutil_strdup(env, msg_id);
axutil_hash_set(callback_recv->callback_map,
mid, AXIS2_HASH_KEY_STRING, callback);
}
return AXIS2_SUCCESS;
}
static axis2_status_t AXIS2_CALL
axis2_callback_recv_receive(
axis2_msg_recv_t *msg_recv,
const axutil_env_t *env,
axis2_msg_ctx_t *msg_ctx,
void *callback_recv_param)
{
axis2_callback_recv_t *callback_recv = NULL;
axis2_relates_to_t *relates_to = NULL;
axis2_msg_info_headers_t *msg_info_headers = NULL;
AXIS2_ENV_CHECK(env, AXIS2_FAILURE);
callback_recv = axis2_msg_recv_get_derived(msg_recv, env);
msg_info_headers = axis2_msg_ctx_get_msg_info_headers(msg_ctx, env);
if (msg_info_headers)
{
relates_to = axis2_msg_info_headers_get_relates_to(msg_info_headers, env);
if (relates_to)
{
const axis2_char_t *msg_id = axis2_relates_to_get_value(relates_to, env);
if (msg_id)
{
axis2_async_result_t *result = NULL;
axis2_callback_t *callback = (axis2_callback_t*)
axutil_hash_get(callback_recv->callback_map, msg_id, AXIS2_HASH_KEY_STRING);
result = axis2_async_result_create(env, msg_ctx);
if (callback && result)
{
axis2_callback_invoke_on_complete(callback, env, result);
axis2_callback_set_complete(callback, env, AXIS2_TRUE);
axis2_msg_ctx_set_soap_envelope(msg_ctx, env, NULL);
}
axis2_async_result_free(result, env);
if (callback && result)
return AXIS2_SUCCESS;
}
}
}
return AXIS2_FAILURE;
}