2020-03-30 22:45:41 +02:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* @defgroup sys_uri_parser A minimal, non-destructive URI parser
|
|
|
|
* @ingroup sys
|
2020-04-03 14:30:38 +02:00
|
|
|
* @brief A minimal, non-destructive parser for URI references
|
|
|
|
*
|
|
|
|
* This module acts on URI references, and thus process both relative
|
|
|
|
* references and URIs.
|
2020-03-30 22:45:41 +02:00
|
|
|
*
|
2020-04-03 14:32:55 +02:00
|
|
|
* Note that fragment identifiers are not handled by the implementation.
|
|
|
|
*
|
2020-03-30 22:45:41 +02:00
|
|
|
* @see https://tools.ietf.org/html/rfc3986
|
|
|
|
*
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @brief Handler functions for uri_parser
|
|
|
|
* @author Cenk Gündoğan <cenk.guendogan@haw-hamburg.de>
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef URI_PARSER_H
|
|
|
|
#define URI_PARSER_H
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief container that holds all results
|
|
|
|
*/
|
|
|
|
typedef struct {
|
2021-08-04 14:01:28 +02:00
|
|
|
const char *scheme; /**< scheme */
|
|
|
|
const char *userinfo; /**< userinfo */
|
2020-10-29 14:24:07 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief host part
|
|
|
|
*
|
|
|
|
* @note for IPv6 addresses, @ref host also includes the brackets
|
|
|
|
* '[' and ']' as well as the zoneid (with leading '%'), if
|
|
|
|
* present.
|
|
|
|
*/
|
2021-08-04 14:01:28 +02:00
|
|
|
const char *host;
|
2020-10-29 14:24:07 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Pointer to the start of the address, if @ref host is an
|
|
|
|
* IPv6 address and NULL otherwise
|
|
|
|
*
|
|
|
|
* @note @ref ipv6addr does not include the brackets '[' and ']'
|
|
|
|
* and the zoneid part.
|
|
|
|
*/
|
2021-08-04 14:01:28 +02:00
|
|
|
const char *ipv6addr;
|
2020-10-29 14:24:07 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief zoneid if @ref host is IPv6 address, NULL otherwise
|
|
|
|
*
|
|
|
|
* @see https://tools.ietf.org/html/rfc6874
|
|
|
|
*/
|
2021-08-04 14:01:28 +02:00
|
|
|
const char *zoneid;
|
2020-10-29 14:24:07 +01:00
|
|
|
|
2022-05-11 15:00:16 +02:00
|
|
|
const char *port_str; /**< port as str */
|
2021-08-04 14:01:28 +02:00
|
|
|
const char *path; /**< path */
|
|
|
|
const char *query; /**< query */
|
2022-05-11 15:00:16 +02:00
|
|
|
uint16_t port; /**< port as uint16_t */
|
2020-10-29 14:24:07 +01:00
|
|
|
uint16_t scheme_len; /**< length of @ref scheme */
|
|
|
|
uint16_t userinfo_len; /**< length of @ref userinfo */
|
|
|
|
uint16_t host_len; /**< length of @ref host */
|
|
|
|
uint16_t ipv6addr_len; /**< length of @ref ipv6addr */
|
|
|
|
uint16_t zoneid_len; /**< length of @ref zoneid */
|
2022-05-11 15:00:16 +02:00
|
|
|
uint16_t port_str_len; /**< length of @ref port_str */
|
2020-10-29 14:24:07 +01:00
|
|
|
uint16_t path_len; /**< length of @ref path */
|
|
|
|
uint16_t query_len; /**< length of @ref query */
|
2020-03-30 22:45:41 +02:00
|
|
|
} uri_parser_result_t;
|
|
|
|
|
2021-07-30 14:42:05 +02:00
|
|
|
/**
|
|
|
|
* @brief Container to represent a query parameter
|
|
|
|
*/
|
|
|
|
typedef struct {
|
2021-08-04 14:01:28 +02:00
|
|
|
const char *name; /**< name of the query parameter */
|
|
|
|
const char *value; /**< value of the query parameter */
|
2021-07-30 14:42:05 +02:00
|
|
|
uint16_t name_len; /**< length of @ref name */
|
|
|
|
uint16_t value_len; /**< length of @ref value */
|
|
|
|
} uri_parser_query_param_t;
|
|
|
|
|
2020-03-30 22:45:41 +02:00
|
|
|
/**
|
2020-04-03 14:30:38 +02:00
|
|
|
* @brief Checks whether @p uri is in absolute form
|
2020-03-30 22:45:41 +02:00
|
|
|
*
|
2020-04-03 14:30:38 +02:00
|
|
|
* @param[in] uri URI reference to check. Must not be `NULL`
|
2020-03-30 22:45:41 +02:00
|
|
|
* @param[in] uri_len Length of @p uri
|
|
|
|
*
|
|
|
|
* @pre `uri != NULL`
|
|
|
|
*
|
2020-04-03 14:30:38 +02:00
|
|
|
* @return true if @p uri is a URI
|
|
|
|
* @return false if @p uri is a relative reference
|
2020-03-30 22:45:41 +02:00
|
|
|
*/
|
|
|
|
bool uri_parser_is_absolute(const char *uri, size_t uri_len);
|
|
|
|
|
|
|
|
/**
|
2020-04-03 14:30:38 +02:00
|
|
|
* @brief Checks whether @p uri is in absolute form
|
2020-03-30 22:45:41 +02:00
|
|
|
*
|
2020-04-03 14:30:38 +02:00
|
|
|
* @param[in] uri Zero-terminated URI reference to check. Must not be `Null`
|
2020-03-30 22:45:41 +02:00
|
|
|
*
|
|
|
|
* @pre `uri != NULL`
|
|
|
|
*
|
2020-04-03 14:30:38 +02:00
|
|
|
* @return true if @p uri is a URI
|
|
|
|
* @return false if @p uri is a relative reference
|
2020-03-30 22:45:41 +02:00
|
|
|
*/
|
|
|
|
bool uri_parser_is_absolute_string(const char *uri);
|
|
|
|
|
|
|
|
/**
|
2020-04-03 14:30:38 +02:00
|
|
|
* @brief Parse a URI reference
|
2020-03-30 22:45:41 +02:00
|
|
|
*
|
|
|
|
* @param[out] result pointer to a container that will hold the result
|
|
|
|
* @param[in] uri URI to parse. Must not be `NULL`
|
|
|
|
* @param[in] uri_len Length of @p uri
|
|
|
|
*
|
|
|
|
* @pre `uri != NULL`
|
|
|
|
*
|
|
|
|
* @return 0 on success
|
|
|
|
* @return -1 on parsing error
|
|
|
|
*/
|
|
|
|
int uri_parser_process(uri_parser_result_t *result, const char *uri,
|
|
|
|
size_t uri_len);
|
|
|
|
|
|
|
|
/**
|
2020-04-03 14:30:38 +02:00
|
|
|
* @brief Parse a URI reference
|
2020-03-30 22:45:41 +02:00
|
|
|
*
|
|
|
|
* @param[out] result pointer to a container that will hold the result
|
|
|
|
* @param[in] uri Zero-terminated URI to parse. Must not be `NULL`
|
|
|
|
*
|
|
|
|
* @pre `uri != NULL`
|
|
|
|
*
|
|
|
|
* @return 0 on success
|
|
|
|
* @return -1 on parsing error
|
|
|
|
*/
|
|
|
|
int uri_parser_process_string(uri_parser_result_t *result, const char *uri);
|
|
|
|
|
2021-07-30 14:42:05 +02:00
|
|
|
/**
|
|
|
|
* @brief Provides a list of URI query parameters from a given URI parser
|
|
|
|
* result.
|
|
|
|
*
|
|
|
|
* @note The function **DOES NOT** check for duplicate query parameters.
|
|
|
|
*
|
|
|
|
* @pre `uri_parsed != NULL`
|
|
|
|
* @pre `params != NULL` and all its elements are set to zero.
|
|
|
|
*
|
|
|
|
* @param[in] uri_parsed A parsed URI result. Must not be NULL.
|
|
|
|
* @param[out] params An array of @ref uri_parser_query_param_t.
|
|
|
|
* Must not be NULL and all zero-valued on call. Will be
|
|
|
|
* filled with the name-value-pairs in
|
|
|
|
* uri_parser_result_t::query of @p uri_parsed. If the
|
|
|
|
* number of query parameters in @p uri_parsed exceeds
|
|
|
|
* @p params_len, the list will be truncated and the
|
|
|
|
* function returns -2.
|
|
|
|
* @param[in] params_len The length of @p params
|
|
|
|
*
|
|
|
|
* @return number of filled entries in @p params on success. Might be 0 if
|
|
|
|
* uri_parser_result_t::query is NULL.
|
|
|
|
* @return -1 on parsing error.
|
|
|
|
* @return -2 when the number of query parameters exceeds @p params_len.
|
|
|
|
* In that case, the array is filled with the first @p params_len
|
|
|
|
* name-value-pairs in uri_parser_result_t::query of @p uri_parsed.
|
|
|
|
*/
|
|
|
|
int uri_parser_split_query(const uri_parser_result_t *uri_parsed,
|
|
|
|
uri_parser_query_param_t *params,
|
|
|
|
size_t params_len);
|
|
|
|
|
2020-03-30 22:45:41 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* URI_PARSER_H */
|
|
|
|
/** @} */
|