mirror of
https://github.com/asterisk/asterisk.git
synced 2025-09-05 04:11:08 +00:00
If using the custom URI parsing code (not external uriparser lib) and there was no query parameters the resulting pointer would be NULL and then an attempt was made to subtract from it. The pointer is now set to a valid value if there is no query parameter(s). Also, in the 'ast_uri_make_host_with_port' function when setting the terminator on the resulting string it was writing it one past the end of allocated memory. It now writes the string terminator appropriately. git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@415358 65c4cc65-6c06-0410-ace0-fbb531ad65f3
324 lines
8.1 KiB
C
324 lines
8.1 KiB
C
/*
|
|
* Asterisk -- An open source telephony toolkit.
|
|
*
|
|
* Copyright (C) 2014, Digium, Inc.
|
|
*
|
|
* Kevin Harwell <kharwell@digium.com>
|
|
*
|
|
* See http://www.asterisk.org for more information about
|
|
* the Asterisk project. Please do not directly contact
|
|
* any of the maintainers of this project for assistance;
|
|
* the project provides a web site, mailing lists and IRC
|
|
* channels for your use.
|
|
*
|
|
* This program is free software, distributed under the terms of
|
|
* the GNU General Public License Version 2. See the LICENSE file
|
|
* at the top of the source tree.
|
|
*/
|
|
|
|
#include "asterisk.h"
|
|
|
|
#include "asterisk/astobj2.h"
|
|
#include "asterisk/strings.h"
|
|
#include "asterisk/uri.h"
|
|
|
|
#ifdef HAVE_URIPARSER
|
|
#include <uriparser/Uri.h>
|
|
#endif
|
|
|
|
/*! \brief Stores parsed uri information */
|
|
struct ast_uri {
|
|
/*! scheme (e.g. http, https, ws, wss, etc...) */
|
|
char *scheme;
|
|
/*! username:password */
|
|
char *user_info;
|
|
/*! host name or address */
|
|
char *host;
|
|
/*! associated port */
|
|
char *port;
|
|
/*! path info following host[:port] */
|
|
char *path;
|
|
/*! query information */
|
|
char *query;
|
|
/*! storage for uri string */
|
|
char uri[0];
|
|
};
|
|
|
|
/*!
|
|
* \brief Construct a uri object with the given values.
|
|
*
|
|
* \note The size parameters [should] include room for the string terminator
|
|
* (strlen(<param>) + 1). For instance, if a scheme of 'http' is given
|
|
* then the 'scheme_size' should be equal to 5.
|
|
*/
|
|
static struct ast_uri *ast_uri_create_(
|
|
const char *scheme, unsigned int scheme_size,
|
|
const char *user_info, unsigned int user_info_size,
|
|
const char *host, unsigned int host_size,
|
|
const char *port, unsigned int port_size,
|
|
const char *path, unsigned int path_size,
|
|
const char *query, unsigned int query_size)
|
|
{
|
|
#define SET_VALUE(param, field, size) \
|
|
do { if (param) { \
|
|
ast_copy_string(p, param, size); \
|
|
field = p; \
|
|
p += size; } } while (0)
|
|
|
|
char *p;
|
|
struct ast_uri *res = ao2_alloc(
|
|
sizeof(*res) + scheme_size + user_info_size + host_size +
|
|
port_size + path_size + query_size, NULL);
|
|
|
|
if (!res) {
|
|
ast_log(LOG_ERROR, "Unable to create URI object\n");
|
|
return NULL;
|
|
}
|
|
|
|
p = res->uri;
|
|
SET_VALUE(scheme, res->scheme, scheme_size);
|
|
SET_VALUE(user_info, res->user_info, user_info_size);
|
|
SET_VALUE(host, res->host, host_size);
|
|
SET_VALUE(port, res->port, port_size);
|
|
SET_VALUE(path, res->path, path_size);
|
|
SET_VALUE(query, res->query, query_size);
|
|
return res;
|
|
}
|
|
|
|
struct ast_uri *ast_uri_create(const char *scheme, const char *user_info,
|
|
const char *host, const char *port,
|
|
const char *path, const char *query)
|
|
{
|
|
return ast_uri_create_(
|
|
scheme, scheme ? strlen(scheme) + 1 : 0,
|
|
user_info, user_info ? strlen(user_info) + 1 : 0,
|
|
host, host ? strlen(host) + 1 : 0,
|
|
port, port ? strlen(port) + 1 : 0,
|
|
path, path ? strlen(path) + 1 : 0,
|
|
query, query ? strlen(query) + 1 : 0);
|
|
}
|
|
|
|
struct ast_uri *ast_uri_copy_replace(const struct ast_uri *uri, const char *scheme,
|
|
const char *user_info, const char *host,
|
|
const char *port, const char *path,
|
|
const char *query)
|
|
{
|
|
return ast_uri_create(
|
|
scheme ? scheme : uri->scheme,
|
|
user_info ? user_info : uri->user_info,
|
|
host ? host : uri->host,
|
|
port ? port : uri->port,
|
|
path ? path : uri->path,
|
|
query ? query : uri->query);
|
|
}
|
|
|
|
const char *ast_uri_scheme(const struct ast_uri *uri)
|
|
{
|
|
return uri->scheme;
|
|
}
|
|
|
|
const char *ast_uri_user_info(const struct ast_uri *uri)
|
|
{
|
|
return uri->user_info;
|
|
}
|
|
|
|
const char *ast_uri_host(const struct ast_uri *uri)
|
|
{
|
|
return uri->host;
|
|
}
|
|
|
|
const char *ast_uri_port(const struct ast_uri *uri)
|
|
{
|
|
return uri->port;
|
|
}
|
|
|
|
const char *ast_uri_path(const struct ast_uri *uri)
|
|
{
|
|
return uri->path;
|
|
}
|
|
|
|
const char *ast_uri_query(const struct ast_uri *uri)
|
|
{
|
|
return uri->query;
|
|
}
|
|
|
|
const int ast_uri_is_secure(const struct ast_uri *uri)
|
|
{
|
|
return ast_strlen_zero(uri->scheme) ? 0 :
|
|
*(uri->scheme + strlen(uri->scheme) - 1) == 's';
|
|
}
|
|
|
|
#ifdef HAVE_URIPARSER
|
|
struct ast_uri *ast_uri_parse(const char *uri)
|
|
{
|
|
UriParserStateA state;
|
|
UriUriA uria;
|
|
struct ast_uri *res;
|
|
unsigned int scheme_size, user_info_size, host_size;
|
|
unsigned int port_size, path_size, query_size;
|
|
const char *path_start, *path_end;
|
|
|
|
state.uri = &uria;
|
|
if (uriParseUriA(&state, uri) != URI_SUCCESS) {
|
|
ast_log(LOG_ERROR, "Unable to parse URI %s\n", uri);
|
|
uriFreeUriMembersA(&uria);
|
|
return NULL;
|
|
}
|
|
|
|
scheme_size = uria.scheme.first ?
|
|
uria.scheme.afterLast - uria.scheme.first + 1 : 0;
|
|
user_info_size = uria.userInfo.first ?
|
|
uria.userInfo.afterLast - uria.userInfo.first + 1 : 0;
|
|
host_size = uria.hostText.first ?
|
|
uria.hostText.afterLast - uria.hostText.first + 1 : 0;
|
|
port_size = uria.portText.first ?
|
|
uria.portText.afterLast - uria.portText.first + 1 : 0;
|
|
|
|
path_start = uria.pathHead && uria.pathHead->text.first ?
|
|
uria.pathHead->text.first : NULL;
|
|
path_end = path_start ? uria.pathTail->text.afterLast : NULL;
|
|
path_size = path_end ? path_end - path_start + 1 : 0;
|
|
|
|
query_size = uria.query.first ?
|
|
uria.query.afterLast - uria.query.first + 1 : 0;
|
|
|
|
res = ast_uri_create_(uria.scheme.first, scheme_size,
|
|
uria.userInfo.first, user_info_size,
|
|
uria.hostText.first, host_size,
|
|
uria.portText.first, port_size,
|
|
path_start, path_size,
|
|
uria.query.first, query_size);
|
|
uriFreeUriMembersA(&uria);
|
|
return res;
|
|
}
|
|
#else
|
|
struct ast_uri *ast_uri_parse(const char *uri)
|
|
{
|
|
#define SET_VALUES(value) \
|
|
value = uri; \
|
|
size_##value = p - uri + 1; \
|
|
uri = p + 1;
|
|
|
|
const char *p, *scheme = NULL, *user_info = NULL, *host = NULL;
|
|
const char *port = NULL, *path = NULL, *query = NULL;
|
|
unsigned int size_scheme = 0, size_user_info = 0, size_host = 0;
|
|
unsigned int size_port = 0, size_path = 0, size_query = 0;
|
|
|
|
if ((p = strstr(uri, "://"))) {
|
|
scheme = uri;
|
|
size_scheme = p - uri + 1;
|
|
uri = p + 3;
|
|
}
|
|
|
|
if ((p = strchr(uri, '@'))) {
|
|
SET_VALUES(user_info);
|
|
}
|
|
|
|
if ((p = strchr(uri, ':'))) {
|
|
SET_VALUES(host);
|
|
}
|
|
|
|
if ((p = strchr(uri, '/'))) {
|
|
if (!host) {
|
|
SET_VALUES(host);
|
|
} else {
|
|
SET_VALUES(port);
|
|
}
|
|
}
|
|
|
|
if ((p = strchr(uri, '?'))) {
|
|
query = p + 1;
|
|
size_query = strlen(query) + 1;
|
|
} else {
|
|
p = uri + strlen(uri);
|
|
}
|
|
|
|
if (!host) {
|
|
SET_VALUES(host);
|
|
} else if (*(uri - 1) == ':') {
|
|
SET_VALUES(port);
|
|
} else if (*(uri - 1) == '/') {
|
|
SET_VALUES(path);
|
|
}
|
|
|
|
return ast_uri_create_(scheme, size_scheme,
|
|
user_info, size_user_info,
|
|
host, size_host,
|
|
port, size_port,
|
|
path, size_path,
|
|
query, size_query);
|
|
}
|
|
#endif
|
|
|
|
static struct ast_uri *uri_parse_and_default(const char *uri, const char *scheme,
|
|
const char *port, const char *secure_port)
|
|
{
|
|
struct ast_uri *res;
|
|
int len = strlen(scheme);
|
|
|
|
if (!strncmp(uri, scheme, len)) {
|
|
res = ast_uri_parse(uri);
|
|
} else {
|
|
/* make room for <scheme>:// */
|
|
char *with_scheme = ast_malloc(len + strlen(uri) + 4);
|
|
if (!with_scheme) {
|
|
ast_log(LOG_ERROR, "Unable to allocate uri '%s' with "
|
|
"scheme '%s'", uri, scheme);
|
|
return NULL;
|
|
}
|
|
|
|
/* safe - 'with_scheme' created with size equal to len of
|
|
scheme plus length of uri plus space for extra characters
|
|
'://' and terminator */
|
|
sprintf(with_scheme, "%s://%s", scheme, uri);
|
|
res = ast_uri_parse(with_scheme);
|
|
ast_free(with_scheme);
|
|
}
|
|
|
|
if (res && ast_strlen_zero(ast_uri_port(res))) {
|
|
/* default the port if not given */
|
|
struct ast_uri *tmp = ast_uri_copy_replace(
|
|
res, NULL, NULL, NULL,
|
|
ast_uri_is_secure(res) ? secure_port : port,
|
|
NULL, NULL);
|
|
ao2_ref(res, -1);
|
|
res = tmp;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
struct ast_uri *ast_uri_parse_http(const char *uri)
|
|
{
|
|
return uri_parse_and_default(uri, "http", "80", "443");
|
|
}
|
|
|
|
struct ast_uri *ast_uri_parse_websocket(const char *uri)
|
|
{
|
|
return uri_parse_and_default(uri, "ws", "80", "443");
|
|
}
|
|
|
|
char *ast_uri_make_host_with_port(const struct ast_uri *uri)
|
|
{
|
|
int host_size = ast_uri_host(uri) ?
|
|
strlen(ast_uri_host(uri)) : 0;
|
|
/* if there is a port +1 for the colon */
|
|
int port_size = ast_uri_port(uri) ?
|
|
strlen(ast_uri_port(uri)) + 1 : 0;
|
|
char *res = ast_malloc(host_size + port_size + 1);
|
|
|
|
if (!res) {
|
|
return NULL;
|
|
}
|
|
|
|
memcpy(res, ast_uri_host(uri), host_size);
|
|
|
|
if (ast_uri_port(uri)) {
|
|
res[host_size] = ':';
|
|
memcpy(res + host_size + 1,
|
|
ast_uri_port(uri), port_size - 1);
|
|
}
|
|
|
|
res[host_size + port_size] = '\0';
|
|
return res;
|
|
}
|