mirror of
https://github.com/ipxe/ipxe
synced 2025-12-15 00:12:19 +03:00
[uri] Refactor URI parsing and formatting
Add support for parsing of URIs containing literal IPv6 addresses (e.g. "http://[fe80::69ff:fe50:5845%25net0]/boot.ipxe"). Duplicate URIs by directly copying the relevant fields, rather than by formatting and reparsing a URI string. This relaxes the requirements on the URI formatting code and allows it to focus on generating human-readable URIs (e.g. by not escaping ':' characters within literal IPv6 addresses). As a side-effect, this allows relative URIs containing parameter lists (e.g. "../boot.php##params") to function as expected. Add validity check for FTP paths to ensure that only printable characters are accepted (since FTP is a human-readable line-based protocol with no support for character escaping). Construct TFTP next-server+filename URIs directly, rather than parsing a constructed "tftp://..." string, Add self-tests for URI functions. Signed-off-by: Michael Brown <mcb30@ipxe.org>
This commit is contained in:
@@ -53,3 +53,4 @@ REQUIRE_OBJECT ( pnm_test );
|
||||
REQUIRE_OBJECT ( deflate_test );
|
||||
REQUIRE_OBJECT ( png_test );
|
||||
REQUIRE_OBJECT ( dns_test );
|
||||
REQUIRE_OBJECT ( uri_test );
|
||||
|
||||
@@ -1,146 +1,858 @@
|
||||
#include <stdint.h>
|
||||
#include <stddef.h>
|
||||
#include <stdio.h>
|
||||
/*
|
||||
* Copyright (C) 2014 Michael Brown <mbrown@fensystems.co.uk>.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU General Public License as
|
||||
* published by the Free Software Foundation; either version 2 of the
|
||||
* License, or any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||
* 02110-1301, USA.
|
||||
*/
|
||||
|
||||
FILE_LICENCE ( GPL2_OR_LATER );
|
||||
|
||||
/** @file
|
||||
*
|
||||
* URI self-tests
|
||||
*
|
||||
*/
|
||||
|
||||
/* Forcibly enable assertions */
|
||||
#undef NDEBUG
|
||||
|
||||
#include <string.h>
|
||||
#include <errno.h>
|
||||
#include <byteswap.h>
|
||||
#include <ipxe/uri.h>
|
||||
#include <ipxe/params.h>
|
||||
#include <ipxe/test.h>
|
||||
|
||||
#define URI_MAX_LEN 1024
|
||||
|
||||
/** A URI parsing/formatting test */
|
||||
struct uri_test {
|
||||
const char *base_uri_string;
|
||||
const char *relative_uri_string;
|
||||
const char *resolved_uri_string;
|
||||
/** URI string */
|
||||
const char *string;
|
||||
/** URI */
|
||||
struct uri uri;
|
||||
};
|
||||
|
||||
static struct uri_test uri_tests[] = {
|
||||
{ "http://www.fensystems.co.uk", "",
|
||||
"http://www.fensystems.co.uk/" },
|
||||
{ "http://ipxe.org/wiki/page1", "page2",
|
||||
"http://ipxe.org/wiki/page2" },
|
||||
{ "http://ipxe.org/wiki/page1", "../page3",
|
||||
"http://ipxe.org/page3" },
|
||||
{ "tftp://192.168.0.1/", "/tftpboot/vmlinuz",
|
||||
"tftp://192.168.0.1/tftpboot/vmlinuz" },
|
||||
{ "ftp://the%41nswer%3d:%34ty%32wo@ether%62oot.org:8080/p%41th/foo",
|
||||
"to?%41=b#%43d",
|
||||
"ftp://theAnswer%3d:4ty2wo@ipxe.org:8080/path/to?a=b#cd" },
|
||||
#if 0
|
||||
"http://www.ipxe.org/wiki",
|
||||
"mailto:bob@nowhere.com",
|
||||
"ftp://joe:secret@insecure.org:8081/hidden/path/to?what=is#this",
|
||||
#endif
|
||||
/** A URI port number test */
|
||||
struct uri_port_test {
|
||||
/** URI string */
|
||||
const char *string;
|
||||
/** Default port number */
|
||||
unsigned int default_port;
|
||||
/** Expected port number */
|
||||
unsigned int port;
|
||||
};
|
||||
|
||||
static int test_parse_unparse ( const char *uri_string ) {
|
||||
char buf[URI_MAX_LEN];
|
||||
struct uri *uri = NULL;
|
||||
int rc;
|
||||
/** A URI or path resolution test */
|
||||
struct uri_resolve_test {
|
||||
/** Base path or URI */
|
||||
const char *base;
|
||||
/** Relative path or URI */
|
||||
const char *relative;
|
||||
/** Expected resolved path or URI */
|
||||
const char *resolved;
|
||||
};
|
||||
|
||||
/* Parse and unparse URI */
|
||||
uri = parse_uri ( uri_string );
|
||||
if ( ! uri ) {
|
||||
rc = -ENOMEM;
|
||||
goto done;
|
||||
/** A TFTP URI test */
|
||||
struct uri_tftp_test {
|
||||
/** Next-server address */
|
||||
struct in_addr next_server;
|
||||
/** Filename */
|
||||
const char *filename;
|
||||
/** URI */
|
||||
struct uri uri;
|
||||
/** URI string (for display only; cannot be reparsed) */
|
||||
const char *string;
|
||||
};
|
||||
|
||||
/** A current working URI test */
|
||||
struct uri_churi_test {
|
||||
/** Relative URI */
|
||||
const char *relative;
|
||||
/** Expected new working URI */
|
||||
const char *expected;
|
||||
};
|
||||
|
||||
/** A form parameter URI test list */
|
||||
struct uri_params_test_list {
|
||||
/** Key */
|
||||
const char *key;
|
||||
/** Value */
|
||||
const char *value;
|
||||
};
|
||||
|
||||
/** A form parameter URI test */
|
||||
struct uri_params_test {
|
||||
/** URI string */
|
||||
const char *string;
|
||||
/** URI */
|
||||
struct uri uri;
|
||||
/** Parameter list name */
|
||||
const char *name;
|
||||
/** Parameter list */
|
||||
struct uri_params_test_list *list;
|
||||
};
|
||||
|
||||
/**
|
||||
* Compare two URI component strings
|
||||
*
|
||||
* @v first First string, or NULL
|
||||
* @v second Second string, or NULL
|
||||
* @v difference Difference
|
||||
*/
|
||||
static int uristrcmp ( const char *first, const char *second ) {
|
||||
|
||||
/* Compare strings, allowing for either to be NULL */
|
||||
if ( first == second ) {
|
||||
return 0;
|
||||
} else if ( ( first == NULL ) || ( second == NULL ) ) {
|
||||
return -1;
|
||||
} else {
|
||||
return strcmp ( first, second );
|
||||
}
|
||||
unparse_uri ( buf, sizeof ( buf ), uri, URI_ALL );
|
||||
|
||||
/* Compare result */
|
||||
if ( strcmp ( buf, uri_string ) != 0 ) {
|
||||
printf ( "Unparse of \"%s\" produced \"%s\"\n",
|
||||
uri_string, buf );
|
||||
rc = -EINVAL;
|
||||
goto done;
|
||||
}
|
||||
|
||||
rc = 0;
|
||||
|
||||
done:
|
||||
uri_put ( uri );
|
||||
if ( rc ) {
|
||||
printf ( "URI parse-unparse of \"%s\" failed: %s\n",
|
||||
uri_string, strerror ( rc ) );
|
||||
}
|
||||
return rc;
|
||||
}
|
||||
|
||||
static int test_resolve ( const char *base_uri_string,
|
||||
const char *relative_uri_string,
|
||||
const char *resolved_uri_string ) {
|
||||
struct uri *base_uri = NULL;
|
||||
struct uri *relative_uri = NULL;
|
||||
struct uri *resolved_uri = NULL;
|
||||
char buf[URI_MAX_LEN];
|
||||
int rc;
|
||||
/**
|
||||
* Report URI equality test result
|
||||
*
|
||||
* @v uri URI
|
||||
* @v expected Expected URI
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_okx ( struct uri *uri, struct uri *expected, const char *file,
|
||||
unsigned int line ) {
|
||||
|
||||
okx ( uristrcmp ( uri->scheme, expected->scheme ) == 0, file, line );
|
||||
okx ( uristrcmp ( uri->opaque, expected->opaque ) == 0, file, line );
|
||||
okx ( uristrcmp ( uri->user, expected->user ) == 0, file, line );
|
||||
okx ( uristrcmp ( uri->password, expected->password ) == 0, file, line);
|
||||
okx ( uristrcmp ( uri->host, expected->host ) == 0, file, line );
|
||||
okx ( uristrcmp ( uri->port, expected->port ) == 0, file, line );
|
||||
okx ( uristrcmp ( uri->path, expected->path ) == 0, file, line );
|
||||
okx ( uristrcmp ( uri->query, expected->query ) == 0, file, line );
|
||||
okx ( uristrcmp ( uri->fragment, expected->fragment ) == 0, file, line);
|
||||
okx ( uri->params == expected->params, file, line );
|
||||
}
|
||||
#define uri_ok( uri, expected ) uri_okx ( uri, expected, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report URI parsing test result
|
||||
*
|
||||
* @v test URI test
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_parse_okx ( struct uri_test *test, const char *file,
|
||||
unsigned int line ) {
|
||||
struct uri *uri;
|
||||
|
||||
/* Parse URI */
|
||||
uri = parse_uri ( test->string );
|
||||
okx ( uri != NULL, file, line );
|
||||
if ( uri )
|
||||
uri_okx ( uri, &test->uri, file, line );
|
||||
uri_put ( uri );
|
||||
}
|
||||
#define uri_parse_ok( test ) uri_parse_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report URI formatting test result
|
||||
*
|
||||
* @v test URI test
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_format_okx ( struct uri_test *test, const char *file,
|
||||
unsigned int line ) {
|
||||
char buf[ strlen ( test->string ) + 1 /* NUL */ ];
|
||||
char *tmp;
|
||||
size_t len;
|
||||
|
||||
/* Format into fixed-size buffer */
|
||||
len = format_uri ( &test->uri, buf, sizeof ( buf ) );
|
||||
okx ( len == ( sizeof ( buf ) - 1 /* NUL */ ), file, line );
|
||||
okx ( strcmp ( buf, test->string ) == 0, file, line );
|
||||
|
||||
/* Format into temporarily allocated buffer */
|
||||
tmp = format_uri_alloc ( &test->uri );
|
||||
okx ( tmp != NULL, file, line );
|
||||
if ( tmp )
|
||||
okx ( strcmp ( tmp, test->string ) == 0, file, line );
|
||||
free ( tmp );
|
||||
}
|
||||
#define uri_format_ok( test ) uri_format_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report URI duplication test result
|
||||
*
|
||||
* @v test URI
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_dup_okx ( struct uri *uri, const char *file,
|
||||
unsigned int line ) {
|
||||
struct uri *dup;
|
||||
|
||||
dup = uri_dup ( uri );
|
||||
okx ( dup != NULL, file, line );
|
||||
if ( dup )
|
||||
uri_okx ( dup, uri, file, line );
|
||||
uri_put ( dup );
|
||||
}
|
||||
#define uri_dup_ok( test ) uri_dup_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report URI combined parsing and formatting test result
|
||||
*
|
||||
* @v test URI test
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_parse_format_dup_okx ( struct uri_test *test, const char *file,
|
||||
unsigned int line ) {
|
||||
|
||||
uri_parse_okx ( test, file, line );
|
||||
uri_format_okx ( test, file, line );
|
||||
uri_dup_okx ( &test->uri, file, line );
|
||||
}
|
||||
#define uri_parse_format_dup_ok( test ) \
|
||||
uri_parse_format_dup_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report URI port number test result
|
||||
*
|
||||
* @v test URI port number test
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_port_okx ( struct uri_port_test *test, const char *file,
|
||||
unsigned int line ) {
|
||||
struct uri *uri;
|
||||
unsigned int port;
|
||||
|
||||
/* Parse URI */
|
||||
uri = parse_uri ( test->string );
|
||||
okx ( uri != NULL, file, line );
|
||||
if ( uri ) {
|
||||
port = uri_port ( uri, test->default_port );
|
||||
okx ( port == test->port, file, line );
|
||||
}
|
||||
uri_put ( uri );
|
||||
}
|
||||
#define uri_port_ok( test ) uri_port_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report URI resolution test result
|
||||
*
|
||||
* @v test Path resolution test
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_resolve_okx ( struct uri_resolve_test *test,
|
||||
const char *file, unsigned int line ) {
|
||||
struct uri *base;
|
||||
struct uri *relative;
|
||||
struct uri *resolved = NULL;
|
||||
char *formatted;
|
||||
|
||||
/* Parse URIs */
|
||||
base_uri = parse_uri ( base_uri_string );
|
||||
if ( ! base_uri ) {
|
||||
rc = -ENOMEM;
|
||||
goto done;
|
||||
}
|
||||
relative_uri = parse_uri ( relative_uri_string );
|
||||
if ( ! relative_uri ) {
|
||||
rc = -ENOMEM;
|
||||
goto done;
|
||||
base = parse_uri ( test->base );
|
||||
okx ( base != NULL, file, line );
|
||||
relative = parse_uri ( test->relative );
|
||||
okx ( relative != NULL, file, line );
|
||||
|
||||
/* Resolve URI */
|
||||
if ( base && relative ) {
|
||||
resolved = resolve_uri ( base, relative );
|
||||
okx ( resolved != NULL, file, line );
|
||||
}
|
||||
|
||||
/* Resolve URI */
|
||||
resolved_uri = resolve_uri ( base_uri, relative_uri );
|
||||
if ( ! resolved_uri ) {
|
||||
rc = -ENOMEM;
|
||||
goto done;
|
||||
/* Format resolved URI */
|
||||
formatted = format_uri_alloc ( resolved );
|
||||
okx ( formatted != NULL, file, line );
|
||||
|
||||
/* Check resolved URI */
|
||||
if ( formatted )
|
||||
okx ( strcmp ( formatted, test->resolved ) == 0, file, line );
|
||||
|
||||
free ( formatted );
|
||||
uri_put ( resolved );
|
||||
uri_put ( relative );
|
||||
uri_put ( base );
|
||||
}
|
||||
#define uri_resolve_ok( test ) uri_resolve_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report path resolution test result
|
||||
*
|
||||
* @v test Path resolution test
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_resolve_path_okx ( struct uri_resolve_test *test,
|
||||
const char *file, unsigned int line ) {
|
||||
char *resolved;
|
||||
|
||||
/* Resolve paths using resolve_path() directly */
|
||||
resolved = resolve_path ( test->base, test->relative );
|
||||
okx ( resolved != NULL, file, line );
|
||||
if ( resolved )
|
||||
okx ( strcmp ( resolved, test->resolved ) == 0, file, line );
|
||||
free ( resolved );
|
||||
|
||||
/* Resolve paths as URIs (since all paths are valid URIs) */
|
||||
uri_resolve_okx ( test, file, line );
|
||||
}
|
||||
#define uri_resolve_path_ok( test ) \
|
||||
uri_resolve_path_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report URI TFTP test result
|
||||
*
|
||||
* @v test URI TFTP test
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_tftp_okx ( struct uri_tftp_test *test, const char *file,
|
||||
unsigned int line ) {
|
||||
char buf[ strlen ( test->string ) + 1 /* NUL */ ];
|
||||
struct uri *uri;
|
||||
size_t len;
|
||||
|
||||
/* Construct URI */
|
||||
uri = tftp_uri ( test->next_server, test->filename );
|
||||
okx ( uri != NULL, file, line );
|
||||
if ( uri ) {
|
||||
uri_okx ( uri, &test->uri, file, line );
|
||||
len = format_uri ( uri, buf, sizeof ( buf ) );
|
||||
okx ( len == ( sizeof ( buf ) - 1 /* NUL */ ), file, line );
|
||||
okx ( strcmp ( buf, test->string ) == 0, file, line );
|
||||
}
|
||||
uri_put ( uri );
|
||||
}
|
||||
#define uri_tftp_ok( test ) uri_tftp_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report current working URI test result
|
||||
*
|
||||
* @v tests List of current working URI tests
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_churi_okx ( struct uri_churi_test *test, const char *file,
|
||||
unsigned int line ) {
|
||||
struct uri *old_cwuri;
|
||||
struct uri *uri;
|
||||
char *formatted;
|
||||
|
||||
/* Preserve original current working URI */
|
||||
old_cwuri = uri_get ( cwuri );
|
||||
|
||||
/* Perform sequence of current working URI changes */
|
||||
do {
|
||||
/* Parse relative URI */
|
||||
uri = parse_uri ( test->relative );
|
||||
okx ( uri != NULL, file, line );
|
||||
|
||||
/* Move to this URI */
|
||||
churi ( uri );
|
||||
|
||||
/* Format new current working URI */
|
||||
formatted = format_uri_alloc ( cwuri );
|
||||
okx ( formatted != NULL, file, line );
|
||||
if ( formatted ) {
|
||||
okx ( strcmp ( formatted, test->expected ) == 0,
|
||||
file, line );
|
||||
}
|
||||
|
||||
/* Free temporary storage */
|
||||
free ( formatted );
|
||||
uri_put ( uri );
|
||||
|
||||
/* Move to next current working URI test */
|
||||
test++;
|
||||
|
||||
} while ( test->relative != NULL );
|
||||
|
||||
/* Restore original current working URI */
|
||||
churi ( old_cwuri );
|
||||
uri_put ( old_cwuri );
|
||||
}
|
||||
#define uri_churi_ok( test ) uri_churi_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report form parameter URI test list result
|
||||
*
|
||||
* @v test Form parameter URI test
|
||||
* @v uri URI
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_params_list_okx ( struct uri_params_test *test,
|
||||
struct uri *uri, const char *file,
|
||||
unsigned int line ) {
|
||||
struct uri_params_test_list *list;
|
||||
struct parameter *param;
|
||||
|
||||
/* Check URI */
|
||||
uri_okx ( uri, &test->uri, file, line );
|
||||
|
||||
/* Check URI parameters */
|
||||
okx ( uri->params != NULL, file, line );
|
||||
if ( uri->params ) {
|
||||
list = test->list;
|
||||
for_each_param ( param, uri->params ) {
|
||||
okx ( strcmp ( param->key, list->key ) == 0,
|
||||
file, line );
|
||||
okx ( strcmp ( param->value, list->value ) == 0,
|
||||
file, line );
|
||||
list++;
|
||||
}
|
||||
okx ( list->key == NULL, file, line );
|
||||
}
|
||||
}
|
||||
#define uri_params_list_ok( test ) \
|
||||
uri_params_list_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/**
|
||||
* Report form parameter URI test result
|
||||
*
|
||||
* @v test Form parameter URI test
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
static void uri_params_okx ( struct uri_params_test *test, const char *file,
|
||||
unsigned int line ) {
|
||||
struct uri_params_test_list *list;
|
||||
struct parameters *params;
|
||||
struct parameter *param;
|
||||
struct uri *uri;
|
||||
struct uri *dup;
|
||||
|
||||
/* Create parameter list */
|
||||
params = create_parameters ( test->name );
|
||||
okx ( params != NULL, file, line );
|
||||
if ( params ) {
|
||||
for ( list = test->list ; list->key ; list++ ) {
|
||||
param = add_parameter ( params, list->key, list->value);
|
||||
okx ( param != NULL, file, line );
|
||||
}
|
||||
}
|
||||
|
||||
/* Compare result */
|
||||
unparse_uri ( buf, sizeof ( buf ), resolved_uri, URI_ALL );
|
||||
if ( strcmp ( buf, resolved_uri_string ) != 0 ) {
|
||||
printf ( "Resolution of \"%s\"+\"%s\" produced \"%s\"\n",
|
||||
base_uri_string, relative_uri_string, buf );
|
||||
rc = -EINVAL;
|
||||
goto done;
|
||||
}
|
||||
/* Record parameter list as part of expected URI */
|
||||
test->uri.params = params;
|
||||
|
||||
rc = 0;
|
||||
/* Parse URI */
|
||||
uri = parse_uri ( test->string );
|
||||
okx ( uri != NULL, file, line );
|
||||
if ( uri )
|
||||
uri_params_list_okx ( test, uri, file, line );
|
||||
|
||||
done:
|
||||
uri_put ( base_uri );
|
||||
uri_put ( relative_uri );
|
||||
uri_put ( resolved_uri );
|
||||
if ( rc ) {
|
||||
printf ( "URI resolution of \"%s\"+\"%s\" failed: %s\n",
|
||||
base_uri_string, relative_uri_string,
|
||||
strerror ( rc ) );
|
||||
/* Duplicate URI */
|
||||
dup = uri_dup ( uri );
|
||||
okx ( dup != NULL, file, line );
|
||||
if ( dup )
|
||||
uri_params_list_okx ( test, dup, file, line );
|
||||
|
||||
/* Clear parameter list in expected URI */
|
||||
test->uri.params = NULL;
|
||||
|
||||
uri_put ( uri );
|
||||
uri_put ( dup );
|
||||
}
|
||||
#define uri_params_ok( test ) uri_params_okx ( test, __FILE__, __LINE__ )
|
||||
|
||||
/** Empty URI */
|
||||
static struct uri_test uri_empty = {
|
||||
.string = "",
|
||||
};
|
||||
|
||||
/** Basic HTTP URI */
|
||||
static struct uri_test uri_boot_ipxe_org = {
|
||||
"http://boot.ipxe.org/demo/boot.php",
|
||||
{ .scheme = "http", .host = "boot.ipxe.org", .path = "/demo/boot.php" }
|
||||
};
|
||||
|
||||
/** Basic opaque URI */
|
||||
static struct uri_test uri_mailto = {
|
||||
"mailto:ipxe-devel@lists.ipxe.org",
|
||||
{ .scheme = "mailto", .opaque = "ipxe-devel@lists.ipxe.org" }
|
||||
};
|
||||
|
||||
/** HTTP URI with all the trimmings */
|
||||
static struct uri_test uri_http_all = {
|
||||
"http://anon:password@example.com:3001/~foo/cgi-bin/foo.pl?a=b&c=d#bit",
|
||||
{
|
||||
.scheme = "http",
|
||||
.user = "anon",
|
||||
.password = "password",
|
||||
.host = "example.com",
|
||||
.port = "3001",
|
||||
.path = "/~foo/cgi-bin/foo.pl",
|
||||
.query = "a=b&c=d",
|
||||
.fragment = "bit",
|
||||
},
|
||||
};
|
||||
|
||||
/** HTTP URI with escaped characters */
|
||||
static struct uri_test uri_http_escaped = {
|
||||
"https://test.ipxe.org/wtf%3F%0A?kind%23of/uri%20is#this%3F",
|
||||
{
|
||||
.scheme = "https",
|
||||
.host = "test.ipxe.org",
|
||||
.path = "/wtf?\n",
|
||||
.query = "kind#of/uri is",
|
||||
.fragment = "this?",
|
||||
},
|
||||
};
|
||||
|
||||
/** HTTP URI with improperly escaped characters */
|
||||
static struct uri_test uri_http_escaped_improper = {
|
||||
/* We accept for parsing improperly escaped characters.
|
||||
* (Formatting the parsed URI would produce the properly
|
||||
* encoded form, and so would not exactly match the original
|
||||
* URI string.)
|
||||
*/
|
||||
"https://test%2eipxe.org/wt%66%3f\n?kind%23of/uri is#this?",
|
||||
{
|
||||
.scheme = "https",
|
||||
.host = "test.ipxe.org",
|
||||
.path = "/wtf?\n",
|
||||
.query = "kind#of/uri is",
|
||||
.fragment = "this?",
|
||||
},
|
||||
};
|
||||
|
||||
/** IPv6 URI */
|
||||
static struct uri_test uri_ipv6 = {
|
||||
"http://[2001:ba8:0:1d4::6950:5845]/",
|
||||
{
|
||||
.scheme = "http",
|
||||
.host = "[2001:ba8:0:1d4::6950:5845]",
|
||||
.path = "/",
|
||||
},
|
||||
};
|
||||
|
||||
/** IPv6 URI with port */
|
||||
static struct uri_test uri_ipv6_port = {
|
||||
"http://[2001:ba8:0:1d4::6950:5845]:8001/boot",
|
||||
{
|
||||
.scheme = "http",
|
||||
.host = "[2001:ba8:0:1d4::6950:5845]",
|
||||
.port = "8001",
|
||||
.path = "/boot",
|
||||
},
|
||||
};
|
||||
|
||||
/** IPv6 URI with link-local address */
|
||||
static struct uri_test uri_ipv6_local = {
|
||||
"http://[fe80::69ff:fe50:5845%25net0]/ipxe",
|
||||
{
|
||||
.scheme = "http",
|
||||
.host = "[fe80::69ff:fe50:5845%net0]",
|
||||
.path = "/ipxe",
|
||||
},
|
||||
};
|
||||
|
||||
/** IPv6 URI with link-local address not conforming to RFC 6874 */
|
||||
static struct uri_test uri_ipv6_local_non_conforming = {
|
||||
/* We accept for parsing a single "%" in "%net0" (rather than
|
||||
* the properly encoded form "%25net0"). (Formatting the
|
||||
* parsed URI would produce the properly encoded form, and so
|
||||
* would not exactly match the original URI string.)
|
||||
*/
|
||||
"http://[fe80::69ff:fe50:5845%net0]/ipxe",
|
||||
{
|
||||
.scheme = "http",
|
||||
.host = "[fe80::69ff:fe50:5845%net0]",
|
||||
.path = "/ipxe",
|
||||
},
|
||||
};
|
||||
|
||||
/** iSCSI URI */
|
||||
static struct uri_test uri_iscsi = {
|
||||
"iscsi:10.253.253.1::::iqn.2010-04.org.ipxe:rabbit",
|
||||
{
|
||||
.scheme = "iscsi",
|
||||
.opaque = "10.253.253.1::::iqn.2010-04.org.ipxe:rabbit",
|
||||
},
|
||||
};
|
||||
|
||||
/** URI with port number */
|
||||
static struct uri_port_test uri_explicit_port = {
|
||||
"http://192.168.0.1:8080/boot.php",
|
||||
80,
|
||||
8080,
|
||||
};
|
||||
|
||||
/** URI without port number */
|
||||
static struct uri_port_test uri_default_port = {
|
||||
"http://192.168.0.1/boot.php",
|
||||
80,
|
||||
80,
|
||||
};
|
||||
|
||||
/** Simple path resolution test */
|
||||
static struct uri_resolve_test uri_simple_path = {
|
||||
"/etc/passwd",
|
||||
"group",
|
||||
"/etc/group",
|
||||
};
|
||||
|
||||
/** Path resolution test with "." and ".." elements */
|
||||
static struct uri_resolve_test uri_relative_path = {
|
||||
"/var/lib/tftpboot/pxe/pxelinux.0",
|
||||
"./../ipxe/undionly.kpxe",
|
||||
"/var/lib/tftpboot/ipxe/undionly.kpxe",
|
||||
};
|
||||
|
||||
/** Path resolution test terminating with directory */
|
||||
static struct uri_resolve_test uri_directory_path = {
|
||||
"/test/cgi-bin.pl/boot.ipxe",
|
||||
"..",
|
||||
"/test/",
|
||||
};
|
||||
|
||||
/** Path resolution test with excessive ".." elements */
|
||||
static struct uri_resolve_test uri_excessive_path = {
|
||||
"/var/lib/tftpboot/ipxe.pxe",
|
||||
"../../../../../../../foo",
|
||||
"/foo",
|
||||
};
|
||||
|
||||
/** Path resolution test with absolute path */
|
||||
static struct uri_resolve_test uri_absolute_path = {
|
||||
"/var/lib/tftpboot",
|
||||
"/etc/hostname",
|
||||
"/etc/hostname",
|
||||
};
|
||||
|
||||
/** Relative URI resolution test */
|
||||
static struct uri_resolve_test uri_relative = {
|
||||
"http://boot.ipxe.org/demo/boot.php?vendor=10ec&device=8139",
|
||||
"initrd.img",
|
||||
"http://boot.ipxe.org/demo/initrd.img",
|
||||
};
|
||||
|
||||
/** Absolute URI resolution test */
|
||||
static struct uri_resolve_test uri_absolute = {
|
||||
"http://boot.ipxe.org/demo/boot.php",
|
||||
"ftp://192.168.0.1/boot.ipxe",
|
||||
"ftp://192.168.0.1/boot.ipxe",
|
||||
};
|
||||
|
||||
/** Absolute path URI resolution test */
|
||||
static struct uri_resolve_test uri_absolute_uri_path = {
|
||||
"http://boot.ipxe.org/demo/boot.php#test",
|
||||
"/demo/vmlinuz",
|
||||
"http://boot.ipxe.org/demo/vmlinuz",
|
||||
};
|
||||
|
||||
/** Query URI resolution test */
|
||||
static struct uri_resolve_test uri_query = {
|
||||
"http://10.253.253.1/test.pl?mac=02-00-69-50-58-45",
|
||||
"?mac=00-1f-16-bc-fe-2f",
|
||||
"http://10.253.253.1/test.pl?mac=00-1f-16-bc-fe-2f",
|
||||
};
|
||||
|
||||
/** Fragment URI resolution test */
|
||||
static struct uri_resolve_test uri_fragment = {
|
||||
"http://192.168.0.254/test#foo",
|
||||
"#bar",
|
||||
"http://192.168.0.254/test#bar",
|
||||
};
|
||||
|
||||
/** TFTP URI with absolute path */
|
||||
static struct uri_tftp_test uri_tftp_absolute = {
|
||||
{ .s_addr = htonl ( 0xc0a80002 ) /* 192.168.0.2 */ },
|
||||
"/absolute/path",
|
||||
{
|
||||
.scheme = "tftp",
|
||||
.host = "192.168.0.2",
|
||||
.path = "/absolute/path",
|
||||
},
|
||||
"tftp://192.168.0.2/absolute/path",
|
||||
};
|
||||
|
||||
/** TFTP URI with relative path */
|
||||
static struct uri_tftp_test uri_tftp_relative = {
|
||||
{ .s_addr = htonl ( 0xc0a80003 ) /* 192.168.0.3 */ },
|
||||
"relative/path",
|
||||
{
|
||||
.scheme = "tftp",
|
||||
.host = "192.168.0.3",
|
||||
.path = "relative/path",
|
||||
},
|
||||
"tftp://192.168.0.3/relative/path",
|
||||
};
|
||||
|
||||
/** TFTP URI with path containing special characters */
|
||||
static struct uri_tftp_test uri_tftp_icky = {
|
||||
{ .s_addr = htonl ( 0x0a000006 ) /* 10.0.0.6 */ },
|
||||
"C:\\tftpboot\\icky#path",
|
||||
{
|
||||
.scheme = "tftp",
|
||||
.host = "10.0.0.6",
|
||||
.path = "C:\\tftpboot\\icky#path",
|
||||
},
|
||||
"tftp://10.0.0.6/C%3A\\tftpboot\\icky%23path",
|
||||
};
|
||||
|
||||
/** Current working URI test */
|
||||
static struct uri_churi_test uri_churi[] = {
|
||||
{
|
||||
"http://boot.ipxe.org/demo/boot.php",
|
||||
"http://boot.ipxe.org/demo/boot.php",
|
||||
},
|
||||
{
|
||||
"?vendor=10ec&device=8139",
|
||||
"http://boot.ipxe.org/demo/boot.php?vendor=10ec&device=8139",
|
||||
},
|
||||
{
|
||||
"fedora/fedora.ipxe",
|
||||
"http://boot.ipxe.org/demo/fedora/fedora.ipxe",
|
||||
},
|
||||
{
|
||||
"vmlinuz",
|
||||
"http://boot.ipxe.org/demo/fedora/vmlinuz",
|
||||
},
|
||||
{
|
||||
"http://local/boot/initrd.img",
|
||||
"http://local/boot/initrd.img",
|
||||
},
|
||||
{
|
||||
"modules/8139too.ko",
|
||||
"http://local/boot/modules/8139too.ko",
|
||||
},
|
||||
{
|
||||
NULL,
|
||||
NULL,
|
||||
}
|
||||
return rc;
|
||||
};
|
||||
|
||||
/** Form parameter URI test list */
|
||||
static struct uri_params_test_list uri_params_list[] = {
|
||||
{
|
||||
"vendor",
|
||||
"10ec",
|
||||
},
|
||||
{
|
||||
"device",
|
||||
"8139",
|
||||
},
|
||||
{
|
||||
"uuid",
|
||||
"f59fac00-758f-498f-9fe5-87d790045d94",
|
||||
},
|
||||
{
|
||||
NULL,
|
||||
NULL,
|
||||
}
|
||||
};
|
||||
|
||||
/** Form parameter URI test */
|
||||
static struct uri_params_test uri_params = {
|
||||
"http://boot.ipxe.org/demo/boot.php##params",
|
||||
{
|
||||
.scheme = "http",
|
||||
.host = "boot.ipxe.org",
|
||||
.path = "/demo/boot.php",
|
||||
},
|
||||
NULL,
|
||||
uri_params_list,
|
||||
};
|
||||
|
||||
/** Named form parameter URI test list */
|
||||
static struct uri_params_test_list uri_named_params_list[] = {
|
||||
{
|
||||
"mac",
|
||||
"00:1e:65:80:d3:b6",
|
||||
},
|
||||
{
|
||||
"serial",
|
||||
"LXTQ20Z1139322762F2000",
|
||||
},
|
||||
{
|
||||
NULL,
|
||||
NULL,
|
||||
}
|
||||
};
|
||||
|
||||
/** Named form parameter URI test */
|
||||
static struct uri_params_test uri_named_params = {
|
||||
"http://192.168.100.4:3001/register##params=foo",
|
||||
{
|
||||
.scheme = "http",
|
||||
.host = "192.168.100.4",
|
||||
.port = "3001",
|
||||
.path = "/register",
|
||||
},
|
||||
"foo",
|
||||
uri_named_params_list,
|
||||
};
|
||||
|
||||
/**
|
||||
* Perform URI self-test
|
||||
*
|
||||
*/
|
||||
static void uri_test_exec ( void ) {
|
||||
|
||||
/* URI parsing, formatting, and duplication tests */
|
||||
uri_parse_format_dup_ok ( &uri_empty );
|
||||
uri_parse_format_dup_ok ( &uri_boot_ipxe_org );
|
||||
uri_parse_format_dup_ok ( &uri_mailto );
|
||||
uri_parse_format_dup_ok ( &uri_http_all );
|
||||
uri_parse_format_dup_ok ( &uri_http_escaped );
|
||||
uri_parse_ok ( &uri_http_escaped_improper ); /* Parse only */
|
||||
uri_parse_format_dup_ok ( &uri_ipv6 );
|
||||
uri_parse_format_dup_ok ( &uri_ipv6_port );
|
||||
uri_parse_format_dup_ok ( &uri_ipv6_local );
|
||||
uri_parse_ok ( &uri_ipv6_local_non_conforming ); /* Parse only */
|
||||
uri_parse_format_dup_ok ( &uri_iscsi );
|
||||
|
||||
/** URI port number tests */
|
||||
uri_port_ok ( &uri_explicit_port );
|
||||
uri_port_ok ( &uri_default_port );
|
||||
|
||||
/** Path resolution tests */
|
||||
uri_resolve_path_ok ( &uri_simple_path );
|
||||
uri_resolve_path_ok ( &uri_relative_path );
|
||||
uri_resolve_path_ok ( &uri_directory_path );
|
||||
uri_resolve_path_ok ( &uri_excessive_path );
|
||||
uri_resolve_path_ok ( &uri_absolute_path );
|
||||
|
||||
/** URI resolution tests */
|
||||
uri_resolve_ok ( &uri_relative );
|
||||
uri_resolve_ok ( &uri_absolute );
|
||||
uri_resolve_ok ( &uri_absolute_uri_path );
|
||||
uri_resolve_ok ( &uri_query );
|
||||
uri_resolve_ok ( &uri_fragment );
|
||||
|
||||
/* TFTP URI construction tests */
|
||||
uri_tftp_ok ( &uri_tftp_absolute );
|
||||
uri_tftp_ok ( &uri_tftp_relative );
|
||||
uri_tftp_ok ( &uri_tftp_icky );
|
||||
|
||||
/* Current working URI tests */
|
||||
uri_churi_ok ( uri_churi );
|
||||
|
||||
/* Form parameter URI tests */
|
||||
uri_params_ok ( &uri_params );
|
||||
uri_params_ok ( &uri_named_params );
|
||||
}
|
||||
|
||||
int uri_test ( void ) {
|
||||
unsigned int i;
|
||||
struct uri_test *uri_test;
|
||||
int rc;
|
||||
int overall_rc = 0;
|
||||
|
||||
for ( i = 0 ; i < ( sizeof ( uri_tests ) /
|
||||
sizeof ( uri_tests[0] ) ) ; i++ ) {
|
||||
uri_test = &uri_tests[i];
|
||||
rc = test_parse_unparse ( uri_test->base_uri_string );
|
||||
if ( rc != 0 )
|
||||
overall_rc = rc;
|
||||
rc = test_parse_unparse ( uri_test->relative_uri_string );
|
||||
if ( rc != 0 )
|
||||
overall_rc = rc;
|
||||
rc = test_parse_unparse ( uri_test->resolved_uri_string );
|
||||
if ( rc != 0 )
|
||||
overall_rc = rc;
|
||||
rc = test_resolve ( uri_test->base_uri_string,
|
||||
uri_test->relative_uri_string,
|
||||
uri_test->resolved_uri_string );
|
||||
if ( rc != 0 )
|
||||
overall_rc = rc;
|
||||
}
|
||||
|
||||
if ( overall_rc )
|
||||
printf ( "URI tests failed: %s\n", strerror ( overall_rc ) );
|
||||
return overall_rc;
|
||||
}
|
||||
/** URI self-test */
|
||||
struct self_test uri_test __self_test = {
|
||||
.name = "uri",
|
||||
.exec = uri_test_exec,
|
||||
};
|
||||
|
||||
Reference in New Issue
Block a user