rcl/rcl_yaml_param_parser/src/namespace.c
brawner 2c513aa0ed
[foxy backport] Refactor parser.c for better testability (#754) (#784)
* Refactor parser.c for better testability (#754)

* Refactor rcl_yaml_param_parser for better testability

Signed-off-by: Stephen Brawner <brawner@gmail.com>

* Reorder parser.c to match parser.h

Signed-off-by: Stephen Brawner <brawner@gmail.com>

squash! Reorder parser.c to match parser.h

* Refactor yaml_variant.c for deduplication

Signed-off-by: Stephen Brawner <brawner@gmail.com>

* ADD_VALUE_TO_SIMPLE_ARRAY for deduplication

Signed-off-by: Stephen Brawner <brawner@gmail.com>

* Remove fprintf

Signed-off-by: Stephen Brawner <brawner@gmail.com>

* PR Fixup

Signed-off-by: Stephen Brawner <brawner@gmail.com>

* Move headers to src directory

Signed-off-by: Stephen Brawner <brawner@gmail.com>

* PR Fixup

Signed-off-by: Stephen Brawner <brawner@gmail.com>

* Rebase #780

Signed-off-by: Stephen Brawner <brawner@gmail.com>

* Missing includes

Signed-off-by: Stephen Brawner <brawner@gmail.com>
2020-10-05 18:01:07 -07:00

193 lines
5.4 KiB
C

// Copyright 2018 Apex.AI, Inc.
//
// 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 "./impl/namespace.h"
#include "rcutils/error_handling.h"
#include "rcutils/strdup.h"
rcutils_ret_t add_name_to_ns(
namespace_tracker_t * ns_tracker,
const char * name,
const namespace_type_t namespace_type,
rcutils_allocator_t allocator)
{
char * cur_ns;
uint32_t * cur_count;
char * sep_str;
size_t name_len;
size_t ns_len;
size_t sep_len;
size_t tot_len;
switch (namespace_type) {
case NS_TYPE_NODE:
cur_ns = ns_tracker->node_ns;
cur_count = &(ns_tracker->num_node_ns);
sep_str = NODE_NS_SEPERATOR;
break;
case NS_TYPE_PARAM:
cur_ns = ns_tracker->parameter_ns;
cur_count = &(ns_tracker->num_parameter_ns);
sep_str = PARAMETER_NS_SEPERATOR;
break;
default:
return RCUTILS_RET_ERROR;
}
/// Add a name to ns
if (NULL == name) {
return RCUTILS_RET_INVALID_ARGUMENT;
}
if (0U == *cur_count) {
cur_ns = rcutils_strdup(name, allocator);
if (NULL == cur_ns) {
return RCUTILS_RET_BAD_ALLOC;
}
} else {
ns_len = strlen(cur_ns);
name_len = strlen(name);
sep_len = strlen(sep_str);
// Check the last sep_len characters of the current NS against the separator string.
if (strcmp(cur_ns + ns_len - sep_len, sep_str) == 0) {
// Current NS already ends with the separator: don't put another separator in.
sep_len = 0;
sep_str = "";
}
tot_len = ns_len + sep_len + name_len + 1U;
char * tmp_ns_ptr = allocator.reallocate(cur_ns, tot_len, allocator.state);
if (NULL == tmp_ns_ptr) {
return RCUTILS_RET_BAD_ALLOC;
}
cur_ns = tmp_ns_ptr;
memcpy((cur_ns + ns_len), sep_str, sep_len);
memcpy((cur_ns + ns_len + sep_len), name, name_len);
cur_ns[tot_len - 1U] = '\0';
}
*cur_count = (*cur_count + 1U);
if (NS_TYPE_NODE == namespace_type) {
ns_tracker->node_ns = cur_ns;
} else {
ns_tracker->parameter_ns = cur_ns;
}
return RCUTILS_RET_OK;
}
rcutils_ret_t rem_name_from_ns(
namespace_tracker_t * ns_tracker,
const namespace_type_t namespace_type,
rcutils_allocator_t allocator)
{
char * cur_ns;
uint32_t * cur_count;
char * sep_str;
size_t ns_len;
size_t tot_len;
switch (namespace_type) {
case NS_TYPE_NODE:
cur_ns = ns_tracker->node_ns;
cur_count = &(ns_tracker->num_node_ns);
sep_str = NODE_NS_SEPERATOR;
break;
case NS_TYPE_PARAM:
cur_ns = ns_tracker->parameter_ns;
cur_count = &(ns_tracker->num_parameter_ns);
sep_str = PARAMETER_NS_SEPERATOR;
break;
default:
return RCUTILS_RET_ERROR;
}
/// Remove last name from ns
if (*cur_count > 0U) {
if (1U == *cur_count) {
allocator.deallocate(cur_ns, allocator.state);
cur_ns = NULL;
} else {
ns_len = strlen(cur_ns);
char * last_idx = NULL;
char * next_str = NULL;
const char * end_ptr = (cur_ns + ns_len);
next_str = strstr(cur_ns, sep_str);
while (NULL != next_str) {
if (next_str > end_ptr) {
RCUTILS_SET_ERROR_MSG("Internal error. Crossing array boundary");
return RCUTILS_RET_ERROR;
}
last_idx = next_str;
next_str = (next_str + strlen(sep_str));
next_str = strstr(next_str, sep_str);
}
if (NULL != last_idx) {
tot_len = ((size_t)(last_idx - cur_ns) + 1U);
char * tmp_ns_ptr = allocator.reallocate(cur_ns, tot_len, allocator.state);
if (NULL == tmp_ns_ptr) {
return RCUTILS_RET_BAD_ALLOC;
}
cur_ns = tmp_ns_ptr;
cur_ns[tot_len - 1U] = '\0';
}
}
*cur_count = (*cur_count - 1U);
}
if (NS_TYPE_NODE == namespace_type) {
ns_tracker->node_ns = cur_ns;
} else {
ns_tracker->parameter_ns = cur_ns;
}
return RCUTILS_RET_OK;
}
rcutils_ret_t replace_ns(
namespace_tracker_t * ns_tracker,
char * const new_ns,
const uint32_t new_ns_count,
const namespace_type_t namespace_type,
rcutils_allocator_t allocator)
{
rcutils_ret_t res = RCUTILS_RET_OK;
/// Remove the old namespace and point to the new namespace
switch (namespace_type) {
case NS_TYPE_NODE:
if (NULL != ns_tracker->node_ns) {
allocator.deallocate(ns_tracker->node_ns, allocator.state);
}
ns_tracker->node_ns = rcutils_strdup(new_ns, allocator);
if (NULL == ns_tracker->node_ns) {
return RCUTILS_RET_BAD_ALLOC;
}
ns_tracker->num_node_ns = new_ns_count;
break;
case NS_TYPE_PARAM:
if (NULL != ns_tracker->parameter_ns) {
allocator.deallocate(ns_tracker->parameter_ns, allocator.state);
}
ns_tracker->parameter_ns = rcutils_strdup(new_ns, allocator);
if (NULL == ns_tracker->parameter_ns) {
return RCUTILS_RET_BAD_ALLOC;
}
ns_tracker->num_parameter_ns = new_ns_count;
break;
default:
res = RCUTILS_RET_ERROR;
break;
}
return res;
}