1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/tests/unittests/tests-uri_parser/tests-uri_parser.c

724 lines
16 KiB
C
Raw Normal View History

/*
* Copyright (C) 2020 HAW Hamburg
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @{
*
* @file
*/
#include <stdio.h>
#include <stdbool.h>
#include "embUnit.h"
#include "uri_parser.h"
#include "unittests-constants.h"
#include "tests-uri_parser.h"
#define VEC(u, f, s, us, h, v6a, z, po_str, po, pa, q, e) \
{ .uri = u, .full_uri = f, .scheme = s, .userinfo = us, .host = h, \
.ipv6addr = v6a, .zoneid = z, .port_str = po_str, .port = po, \
.path = pa, .query = q, .expected = e }
#define VEC_CHECK_INT(comp, i, vec_msg) \
do { \
vec_msg[0] = '\0'; \
stdimpl_strcat(vec_msg, "Unexpected " # comp " member "); \
stdimpl_lltoa(validate_uris[i].comp, &vec_msg[strlen(vec_msg)], 10);\
stdimpl_strcat(vec_msg, " for \""); \
stdimpl_strcat(vec_msg, validate_uris[i].uri); \
stdimpl_strcat(vec_msg, "\""); \
TEST_ASSERT_MESSAGE(validate_uris[i].comp == ures.comp, vec_msg); \
} while (0)
#define VEC_CHECK_STR(comp, i, vec_msg) \
do { \
if (ures.comp == NULL) { \
TEST_ASSERT(validate_uris[i].comp[0] == '\0'); \
} \
else { \
TEST_ASSERT_EQUAL_INT(strlen(validate_uris[i].comp), \
ures.comp##_len); \
vec_msg[0] = '\0'; \
stdimpl_strcat(vec_msg, "Unexpected " # comp " member \""); \
stdimpl_strcat(vec_msg, validate_uris[i].comp); \
stdimpl_strcat(vec_msg, "\" for \""); \
stdimpl_strcat(vec_msg, validate_uris[i].uri); \
stdimpl_strcat(vec_msg, "\""); \
TEST_ASSERT_MESSAGE(0 == \
strncmp(ures.comp, \
validate_uris[i].comp, \
strlen(validate_uris[i].comp)), \
vec_msg); \
} \
} while (0)
#define VEC_MSG_LEN (256)
typedef struct {
char uri[64];
bool full_uri;
char scheme[8];
char userinfo[16];
char host[24];
char ipv6addr[16];
char zoneid[8];
char port_str[32];
uint16_t port;
char path[48];
char query[32];
int expected;
} validate_t;
/*
VEC(uri_to_parse,
scheme, userinfo, host, port_str, port,
path, query, expected return value)
*/
static const validate_t validate_uris[] = {
/* uri to parse */
VEC("coap://RIOT:test@[fe80:db8::1%tap0]:5683/.well-known/core?v=1",
/* is URI */
true,
/* parsed scheme */
"coap",
/* parsed userinfo */
"RIOT:test",
/* parsed host */
"[fe80:db8::1%tap0]",
/* parsed host without zoneid */
"fe80:db8::1",
/* parsed zoneid */
"tap0",
/* parsed port_str */
"5683",
/* parsed port */
5683,
/* parsed path */
"/.well-known/core",
/* parsed query */
"v=1",
/* expected return value */
0),
VEC("coap://RIOT:test@[fe80:db8::1%]:5683/.well-known/core?v=1",
true,
"coap",
"RIOT:test",
"[fe80:db8::1%]",
"fe80:db8::1",
"",
"5683",
5683,
"/.well-known/core",
"v=1",
-1),
VEC("coap://[fe80::1]/foo%20bar",
true,
"coap",
"",
"[fe80::1]",
"fe80::1",
"",
"",
0,
"/foo%20bar",
"",
0),
VEC("/.well-known/core?v=1",
false,
"",
"",
"",
"",
"",
"",
0,
"/.well-known/core",
"v=1",
0),
VEC("coap://R@[2001:db8::1]:5/v=1",
true,
"coap",
"R",
"[2001:db8::1]",
"2001:db8::1",
"",
"5",
5,
"/v=1",
"",
0),
VEC("coap://R@[2001:db8::1]:5/:v=1",
true,
"coap",
"R",
"[2001:db8::1]",
"2001:db8::1",
"",
"5",
5,
"/:v=1",
"",
0),
VEC("cap://R@[2001:db8::1]:5/?v=1",
true,
"cap",
"R",
"[2001:db8::1]",
"2001:db8::1",
"",
"5",
5,
"/",
"v=1",
0),
VEC("oap://Y2001:db8::1]:5/av=1",
true,
"oap",
"",
"Y2001:db8::1]",
"",
"",
"5",
5,
"/av=1",
"",
0),
VEC("coap://R@[2001:db8::1]:5own/v=1",
true,
"coap",
"R",
"[2001:db8::1]",
"2001:db8::1",
"",
"5own",
5,
"/v=1",
"",
-1),
VEC("coap://R@[2001:db8::1]:5own/:v=1",
true,
"coap",
"R",
"[2001:db8::1]",
"2001:db8::1",
"",
"5own",
5,
"/:v=1",
"",
-1),
VEC("cap://R@[2001:db8::1]:5own/?v=1",
true,
"cap",
"R",
"[2001:db8::1]",
"2001:db8::1",
"",
"5own",
5,
"/",
"v=1",
-1),
VEC("oap://Y2001:db8::1]:5own/av=1",
true,
"oap",
"",
"Y2001:db8::1]",
"",
"",
"5own",
5,
"/av=1",
"",
-1),
VEC("//Rb[ʰ00J:d/5v=0",
false,
"",
"",
"",
"",
"",
"",
0,
"//Rb[ʰ00J:d/5v=0",
"",
0),
VEC("coap://oap://P@[2001:b",
true,
"",
"",
"",
"",
"",
"",
0,
"",
"",
-1),
VEC("coap:///R@[2008::1]:5own//R@[2008::1]:5own/?v=1",
true,
"coap",
"",
"",
"",
"",
"",
0,
"/R@[2008::1]:5own//R@[2008::1]:5own/",
"v=1",
0),
VEC("coaP://R/RZ[2001[8:01[8::1]:5o:1]:5oTMv=1",
true,
"coaP",
"",
"R",
"",
"",
"",
0,
"/RZ[2001[8:01[8::1]:5o:1]:5oTMv=1",
"",
0),
VEC("coap://R@////////////////7///v=1",
true,
"coap",
"R",
"",
"",
"",
"",
0,
"////////////////7///v=1",
"",
0),
VEC("coa[:////[2001:db5ow:5own/Ov=1",
false,
"",
"",
"",
"",
"",
"",
0,
"coa[:////[2001:db5ow:5own/Ov=1",
"",
0),
VEC("tel:+1-816-555-1212",
true,
"tel",
"",
"",
"",
"",
"",
0,
"+1-816-555-1212",
"",
0),
VEC("sms:+15105550101,+15105550102?body=hello%20there",
true,
"sms",
"",
"",
"",
"",
"",
0,
"+15105550101,+15105550102",
"body=hello%20there",
0),
VEC("a",
false,
"",
"",
"",
"",
"",
"",
0,
"a",
"",
0),
VEC("mailto:test@example.com",
true,
"mailto",
"",
"",
"",
"",
"",
0,
"test@example.com",
"",
0),
VEC("ftp://ftp.is.co.za/rfc/rfc1808.txt",
true,
"ftp",
"",
"ftp.is.co.za",
"",
"",
"",
0,
"/rfc/rfc1808.txt",
"",
0),
VEC("http://www.ietf.org/rfc/rfc2396.txt",
true,
"http",
"",
"www.ietf.org",
"",
"",
"",
0,
"/rfc/rfc2396.txt",
"",
0),
VEC("ldap://[2001:db8::7]/c=GB?objectClass?one",
true,
"ldap",
"",
"[2001:db8::7]",
"2001:db8::7",
"",
"",
0,
"/c=GB",
"objectClass?one",
0),
VEC("mailto:John.Doe@example.com",
true,
"mailto",
"",
"",
"",
"",
"",
0,
"John.Doe@example.com",
"",
0),
VEC("news:comp.infosystems.www.servers.unix",
true,
"news",
"",
"",
"",
"",
"",
0,
"comp.infosystems.www.servers.unix",
"",
0),
VEC("tel:+1-816-555-1212",
true,
"tel",
"",
"",
"",
"",
"",
0,
"+1-816-555-1212",
"",
0),
VEC("telnet://192.0.2.16:80/",
true,
"telnet",
"",
"192.0.2.16",
"",
"",
"80",
80,
"/",
"",
0),
VEC("urn:oasis:names:specification:docbook:dtd:xml:4.1.2",
true,
"urn",
"",
"",
"",
"",
"",
0,
"oasis:names:specification:docbook:dtd:xml:4.1.2",
"",
0),
VEC("",
false,
"",
"",
"",
"",
"",
"",
0,
"",
"",
-1),
VEC("/",
false,
"",
"",
"",
"",
"",
"",
0,
"/",
"",
0),
VEC("./this:that",
false,
"",
"",
"",
"",
"",
"",
0,
"./this:that",
"",
0),
VEC("pP://",
true,
"pP",
"",
"",
"",
"",
"",
0,
"",
"",
0),
VEC("A://@",
true,
"A",
"",
"",
"",
"",
"",
0,
"",
"",
0),
VEC("coap://example.com:1234568910",
/* is URI */
true,
/* parsed scheme */
"coap",
/* parsed userinfo */
"",
/* parsed host */
"example.com",
/* parsed host without zoneid */
"",
/* parsed zoneid */
"",
/* parsed port_str */
"",
/* parsed port */
0,
/* parsed path */
"",
/* parsed query */
"",
/* expected return value */
-1),
VEC("coap://example.com: 12/foo",
true,
"",
"",
"example.com",
"",
"",
"",
0,
"",
"",
-1),
VEC("coap://example.com:0x12/foo",
true,
"",
"",
"example.com",
"",
"",
"",
0,
"",
"",
-1),
};
static char _failure_msg[VEC_MSG_LEN];
static void test_uri_parser__validate(void)
{
uri_parser_result_t ures;
for (unsigned i = 0; i < ARRAY_SIZE(validate_uris); i++) {
int res = uri_parser_process_string(&ures, validate_uris[i].uri);
TEST_ASSERT_EQUAL_INT(validate_uris[i].full_uri,
uri_parser_is_absolute_string(validate_uris[i].uri));
TEST_ASSERT_EQUAL_INT(validate_uris[i].expected, res);
if (res == 0) {
VEC_CHECK_STR(scheme, i, _failure_msg);
VEC_CHECK_STR(userinfo, i, _failure_msg);
VEC_CHECK_STR(host, i, _failure_msg);
VEC_CHECK_STR(ipv6addr, i, _failure_msg);
VEC_CHECK_STR(zoneid, i, _failure_msg);
VEC_CHECK_STR(port_str, i, _failure_msg);
VEC_CHECK_INT(port, i, _failure_msg);
VEC_CHECK_STR(path, i, _failure_msg);
VEC_CHECK_STR(query, i, _failure_msg);
}
}
}
static void test_uri_parser__unterminated_string(void)
{
uri_parser_result_t ures;
char uri[64];
/* initialize with a non-null character */
memset(uri, 'Z', sizeof(uri));
memcpy(uri, validate_uris[0].uri, strlen(validate_uris[0].uri));
int res = uri_parser_process(&ures, uri, strlen(validate_uris[0].uri));
TEST_ASSERT_EQUAL_INT(0, res);
VEC_CHECK_STR(scheme, 0, _failure_msg);
VEC_CHECK_STR(userinfo, 0, _failure_msg);
VEC_CHECK_STR(host, 0, _failure_msg);
VEC_CHECK_STR(ipv6addr, 0, _failure_msg);
VEC_CHECK_STR(zoneid, 0, _failure_msg);
VEC_CHECK_STR(port_str, 0, _failure_msg);
VEC_CHECK_INT(port, 0, _failure_msg);
VEC_CHECK_STR(path, 0, _failure_msg);
VEC_CHECK_STR(query, 0, _failure_msg);
}
Test *tests_uri_parser_tests(void)
{
EMB_UNIT_TESTFIXTURES(fixtures) {
new_TestFixture(test_uri_parser__validate),
new_TestFixture(test_uri_parser__unterminated_string),
};
EMB_UNIT_TESTCALLER(uri_parser_tests, NULL, NULL, fixtures);
return (Test *)&uri_parser_tests;
}
#define PARAMS_NUMOF (4U)
#define INIT_URI_RESULTS(q) \
_uri_results.query = (q); \
_uri_results.query_len = sizeof(q) - 1
#define TEST_ASSERT_PARAM(exp, idx, comp) \
TEST_ASSERT_EQUAL_INT(sizeof(exp) - 1, _params[idx].comp ## _len); \
TEST_ASSERT_MESSAGE(!strncmp(exp, _params[idx].comp, _params[idx].comp ## _len), \
#comp " was not " exp);
static uri_parser_query_param_t _params[4U];
static uri_parser_result_t _uri_results;
static void _setup_query(void)
{
memset(_params, 0, sizeof(_params));
memset(&_uri_results, 0, sizeof(_uri_results));
}
static void test_split_query__broken_input(void)
{
int res;
INIT_URI_RESULTS("&");
res = uri_parser_split_query(&_uri_results, _params, ARRAY_SIZE(_params));
TEST_ASSERT_EQUAL_INT(-1, res);
INIT_URI_RESULTS("=&");
res = uri_parser_split_query(&_uri_results, _params, ARRAY_SIZE(_params));
TEST_ASSERT_EQUAL_INT(-1, res);
INIT_URI_RESULTS("=&&");
res = uri_parser_split_query(&_uri_results, _params, ARRAY_SIZE(_params));
TEST_ASSERT_EQUAL_INT(-1, res);
INIT_URI_RESULTS("==");
res = uri_parser_split_query(&_uri_results, _params, ARRAY_SIZE(_params));
TEST_ASSERT_EQUAL_INT(-1, res);
INIT_URI_RESULTS("key=value&name=value=another");
res = uri_parser_split_query(&_uri_results, _params, ARRAY_SIZE(_params));
TEST_ASSERT_EQUAL_INT(-1, res);
}
void test_split_query__truncated(void)
{
int res;
INIT_URI_RESULTS("this=0&is=1&a=very&long=3&query=foo");
TEST_ASSERT_EQUAL_INT(4, ARRAY_SIZE(_params));
res = uri_parser_split_query(&_uri_results, _params, ARRAY_SIZE(_params));
TEST_ASSERT_EQUAL_INT(-2, res);
TEST_ASSERT_PARAM("this", 0, name);
TEST_ASSERT_PARAM("0", 0, value);
TEST_ASSERT_PARAM("is", 1, name);
TEST_ASSERT_PARAM("1", 1, value);
TEST_ASSERT_PARAM("a", 2, name);
TEST_ASSERT_PARAM("very", 2, value);
TEST_ASSERT_PARAM("long", 3, name);
TEST_ASSERT_PARAM("3", 3, value);
}
void test_split_query__success(void)
{
int res;
INIT_URI_RESULTS("foo=&=&bar=1");
res = uri_parser_split_query(&_uri_results, _params, ARRAY_SIZE(_params));
TEST_ASSERT_EQUAL_INT(3, res);
TEST_ASSERT_PARAM("foo", 0, name);
TEST_ASSERT_PARAM("", 0, value);
TEST_ASSERT_PARAM("", 1, name);
TEST_ASSERT_PARAM("", 1, value);
TEST_ASSERT_PARAM("bar", 2, name);
TEST_ASSERT_PARAM("1", 2, value);
TEST_ASSERT_EQUAL_INT(0, _params[3].name_len);
TEST_ASSERT_NULL(_params[3].name);
TEST_ASSERT_EQUAL_INT(0, _params[3].value_len);
TEST_ASSERT_NULL(_params[3].value);
}
Test *tests_query_split_tests(void)
{
EMB_UNIT_TESTFIXTURES(fixtures) {
new_TestFixture(test_split_query__broken_input),
new_TestFixture(test_split_query__truncated),
new_TestFixture(test_split_query__success),
};
EMB_UNIT_TESTCALLER(query_split_tests, _setup_query, NULL, fixtures);
return (Test *)&query_split_tests;
}
void tests_uri_parser(void)
{
TESTS_RUN(tests_uri_parser_tests());
TESTS_RUN(tests_query_split_tests());
}
/** @} */