blob: e1960a07cf2457f28c8dc9233df171096458f4ea [file] [log] [blame]
/*
* Copyright 2004,2005 The Apache Software Foundation.
*
* Licensed 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_om_children_qname_iterator.h>
#include <axis2_om_element.h>
/************** function prototypes *************************************/
axis2_status_t AXIS2_CALL
axis2_om_children_qname_iterator_free
(axis2_om_children_qname_iterator_t *iterator,
axis2_env_t **env);
axis2_status_t AXIS2_CALL
axis2_om_children_qname_iterator_remove
(axis2_om_children_qname_iterator_t *iterator,
axis2_env_t **env);
axis2_bool_t AXIS2_CALL
axis2_om_children_qname_iterator_has_next
(axis2_om_children_qname_iterator_t *iterator,
axis2_env_t **env);
axis2_om_node_t* AXIS2_CALL
axis2_om_children_qname_iterator_next
(axis2_om_children_qname_iterator_t *iterator,
axis2_env_t **env);
static axis2_bool_t AXIS2_CALL
axis2_om_children_qname_iterator_qname_matches
(axis2_env_t **env,
axis2_qname_t *element_qname,
axis2_qname_t *qname_to_match);
/****************** impl struct ***********************************************/
typedef struct axis2_om_children_qname_iterator_impl_t
{
axis2_om_children_qname_iterator_t iterator;
axis2_om_node_t *current_child;
axis2_om_node_t *last_child;
axis2_bool_t next_called;
axis2_bool_t remove_called;
axis2_qname_t *given_qname;
axis2_bool_t need_to_move_forward;
axis2_bool_t matching_node_found;
}axis2_om_children_qname_iterator_impl_t;
/*************************** Macro *********************************************/
#define AXIS2_INTF_TO_IMPL(iterator) \
((axis2_om_children_qname_iterator_impl_t*)iterator)
/*******************************************************************************/
AXIS2_DECLARE(axis2_om_children_qname_iterator_t *)
axis2_om_children_qname_iterator_create(axis2_env_t **env,
axis2_om_node_t *current_child,
axis2_qname_t *given_qname)
{
axis2_om_children_qname_iterator_impl_t *iterator_impl = NULL;
AXIS2_ENV_CHECK(env, NULL);
AXIS2_PARAM_CHECK((*env)->error, current_child, NULL);
iterator_impl = (axis2_om_children_qname_iterator_impl_t*)AXIS2_MALLOC(
(*env)->allocator,
sizeof(axis2_om_children_qname_iterator_impl_t));
if(!iterator_impl)
{
AXIS2_ERROR_SET((*env)->error, AXIS2_ERROR_NO_MEMORY, AXIS2_FAILURE);
return NULL;
}
iterator_impl->iterator.ops = NULL;
iterator_impl->current_child = NULL;
iterator_impl->last_child = NULL;
iterator_impl->need_to_move_forward = AXIS2_TRUE;
iterator_impl->matching_node_found = AXIS2_FALSE;
iterator_impl->next_called = AXIS2_FALSE;
iterator_impl->remove_called = AXIS2_FALSE;
iterator_impl->given_qname = NULL;
iterator_impl->iterator.ops =
(axis2_om_children_qname_iterator_ops_t*)AXIS2_MALLOC((*env)->allocator,
sizeof(axis2_om_children_qname_iterator_ops_t));
if(!(iterator_impl->iterator.ops))
{
AXIS2_OM_CHILDREN_QNAME_ITERATOR_FREE(&(iterator_impl->iterator), env);
AXIS2_ERROR_SET((*env)->error, AXIS2_ERROR_NO_MEMORY, AXIS2_FAILURE);
return NULL;
}
iterator_impl->current_child = current_child;
if(given_qname)
{
iterator_impl->given_qname = AXIS2_QNAME_CLONE(given_qname, env);
/* iterator_impl->given_qname = given_qname; */
}
iterator_impl->iterator.ops->free_fn =
axis2_om_children_qname_iterator_free;
iterator_impl->iterator.ops->remove =
axis2_om_children_qname_iterator_remove;
iterator_impl->iterator.ops->has_next =
axis2_om_children_qname_iterator_has_next;
iterator_impl->iterator.ops->next =
axis2_om_children_qname_iterator_next;
return &(iterator_impl->iterator);
}
axis2_status_t AXIS2_CALL
axis2_om_children_qname_iterator_free(axis2_om_children_qname_iterator_t *iterator,
axis2_env_t **env)
{
axis2_om_children_qname_iterator_impl_t *iterator_impl = NULL;
AXIS2_ENV_CHECK( env, AXIS2_FAILURE);
iterator_impl = AXIS2_INTF_TO_IMPL(iterator);
if(iterator->ops)
AXIS2_FREE((*env)->allocator, iterator->ops);
if(iterator_impl->given_qname)
{
AXIS2_QNAME_FREE(iterator_impl->given_qname, env);
iterator_impl->given_qname = NULL;
}
AXIS2_FREE((*env)->allocator, iterator_impl);
return AXIS2_SUCCESS;
}
axis2_status_t AXIS2_CALL
axis2_om_children_qname_iterator_remove(axis2_om_children_qname_iterator_t *iterator,
axis2_env_t **env)
{
axis2_om_children_qname_iterator_impl_t *iterator_impl = NULL;
AXIS2_ENV_CHECK( env, AXIS2_FAILURE);
iterator_impl = AXIS2_INTF_TO_IMPL(iterator);
if(!(iterator_impl->next_called))
return AXIS2_FAILURE;
if(iterator_impl->remove_called)
return AXIS2_FAILURE;
iterator_impl->remove_called = AXIS2_TRUE;
if(!(iterator_impl->last_child))
return AXIS2_FAILURE;
AXIS2_OM_NODE_DETACH(iterator_impl->last_child, env);
return AXIS2_SUCCESS;
}
axis2_bool_t AXIS2_CALL
axis2_om_children_qname_iterator_has_next
(axis2_om_children_qname_iterator_t *iterator,
axis2_env_t **env)
{
axis2_om_children_qname_iterator_impl_t *iterator_impl = NULL;
AXIS2_ENV_CHECK( env, AXIS2_FALSE);
iterator_impl = AXIS2_INTF_TO_IMPL(iterator);
while(iterator_impl->need_to_move_forward)
{
if(iterator_impl->current_child)
{
axis2_om_element_t *om_element = NULL;
if(AXIS2_OM_NODE_GET_NODE_TYPE(iterator_impl->current_child, env) == AXIS2_OM_ELEMENT)
{
om_element = (axis2_om_element_t *)AXIS2_OM_NODE_GET_DATA_ELEMENT(
iterator_impl->current_child, env);
}
if(om_element && axis2_om_children_qname_iterator_qname_matches(env,
AXIS2_OM_ELEMENT_GET_QNAME(om_element, env), iterator_impl->given_qname))
{
iterator_impl->matching_node_found = AXIS2_TRUE;
iterator_impl->need_to_move_forward = AXIS2_FALSE;
}
else
{
iterator_impl->current_child = AXIS2_OM_NODE_GET_NEXT_SIBLING(iterator_impl->current_child, env);
if(iterator_impl->current_child)
{
iterator_impl->need_to_move_forward = AXIS2_TRUE;
iterator_impl->matching_node_found = AXIS2_TRUE;
}
else
{
iterator_impl->need_to_move_forward = AXIS2_FALSE;
iterator_impl->matching_node_found = AXIS2_FALSE;
}
}
}
else
{
iterator_impl->need_to_move_forward = AXIS2_FALSE;
}
}
return iterator_impl->matching_node_found;
}
axis2_om_node_t* AXIS2_CALL
axis2_om_children_qname_iterator_next(axis2_om_children_qname_iterator_t *iterator,
axis2_env_t **env)
{
axis2_om_children_qname_iterator_impl_t *iterator_impl = NULL;
AXIS2_ENV_CHECK( env, NULL);
iterator_impl = AXIS2_INTF_TO_IMPL(iterator);
iterator_impl->need_to_move_forward = AXIS2_TRUE;
iterator_impl->matching_node_found = AXIS2_FALSE;
iterator_impl->next_called = AXIS2_TRUE;
iterator_impl->remove_called = AXIS2_FALSE;
iterator_impl->last_child = iterator_impl->current_child;
iterator_impl->current_child = AXIS2_OM_NODE_GET_NEXT_SIBLING(iterator_impl->current_child, env);
return iterator_impl->last_child;
}
static axis2_bool_t AXIS2_CALL
axis2_om_children_qname_iterator_qname_matches(axis2_env_t **env,
axis2_qname_t *element_qname,
axis2_qname_t *qname_to_match)
{
int lparts_match = 0;
int uris_match = 0;
axis2_char_t *ele_lpart = NULL;
axis2_char_t *match_lpart = NULL;
axis2_char_t *ele_nsuri = NULL;
axis2_char_t *match_nsuri = NULL;
if(!(qname_to_match))
return AXIS2_TRUE;
if(qname_to_match)
{
match_lpart = AXIS2_QNAME_GET_LOCALPART(qname_to_match, env);
match_nsuri = AXIS2_QNAME_GET_URI(qname_to_match, env);
}
if(element_qname)
{
ele_lpart = AXIS2_QNAME_GET_LOCALPART(element_qname, env);
ele_nsuri = AXIS2_QNAME_GET_URI(element_qname, env);
}
lparts_match = ( !match_lpart ||
(AXIS2_STRCMP(match_lpart,"") == 0) ||
(element_qname && (AXIS2_STRCMP(ele_lpart, match_lpart) == 0)));
uris_match = ( !match_nsuri || (AXIS2_STRCMP(match_nsuri,"") == 0) ||
(element_qname && (AXIS2_STRCMP(ele_nsuri, match_nsuri) == 0)));
return lparts_match && uris_match;
}