Mercurial > dovecot > core-2.2
view src/lib-http/test-http-url.c @ 22656:1789bf2a1e01
director: Make sure HOST-RESET-USERS isn't used with max_moving_users=0
The reset command would just hang in that case. doveadm would never have
sent this, so this is just an extra sanity check.
author | Timo Sirainen <timo.sirainen@dovecot.fi> |
---|---|
date | Sun, 05 Nov 2017 23:51:56 +0200 |
parents | 2e2563132d5f |
children | cb108f786fb4 |
line wrap: on
line source
/* Copyright (c) 2009-2017 Dovecot authors, see the included COPYING file */ #include "lib.h" #include "net.h" #include "http-url.h" #include "test-common.h" struct valid_http_url_test { const char *url; enum http_url_parse_flags flags; struct http_url url_base; struct http_url url_parsed; }; /* Valid HTTP URL tests */ static struct valid_http_url_test valid_url_tests[] = { /* Generic tests */ { .url = "http://localhost", .url_parsed = { .host_name = "localhost" } },{ .url = "http://www.%65%78%61%6d%70%6c%65.com", .url_parsed = { .host_name = "www.example.com" } },{ .url = "http://www.dovecot.org:8080", .url_parsed = { .host_name = "www.dovecot.org", .port = 8080, .have_port = TRUE } },{ .url = "http://127.0.0.1", .url_parsed = { .host_name = "127.0.0.1", .have_host_ip = TRUE } #ifdef HAVE_IPV6 },{ .url = "http://[::1]", .url_parsed = { .host_name = "[::1]", .have_host_ip = TRUE } },{ .url = "http://[::1]:8080", .url_parsed = { .host_name = "[::1]", .have_host_ip = TRUE, .port = 8080, .have_port = TRUE } #endif },{ .url = "http://user@api.dovecot.org", .flags = HTTP_URL_ALLOW_USERINFO_PART, .url_parsed = { .host_name = "api.dovecot.org", .user = "user" } },{ .url = "http://userid:secret@api.dovecot.org", .flags = HTTP_URL_ALLOW_USERINFO_PART, .url_parsed = { .host_name = "api.dovecot.org", .user = "userid", .password = "secret" } },{ .url = "http://su%3auserid:secret@api.dovecot.org", .flags = HTTP_URL_ALLOW_USERINFO_PART, .url_parsed = { .host_name = "api.dovecot.org", .user = "su:userid", .password = "secret" } },{ .url = "http://www.example.com/" "?question=What%20are%20you%20doing%3f&answer=Nothing.", .url_parsed = { .path = "/", .host_name = "www.example.com", .enc_query = "question=What%20are%20you%20doing%3f&answer=Nothing." } },{ /* These next 2 URLs don't follow the recommendations in http://tools.ietf.org/html/rfc1034#section-3.5 and http://tools.ietf.org/html/rfc3696 However they satisfy the grammar in http://tools.ietf.org/html/rfc1123#section-2 and http://tools.ietf.org/html/rfc952 so we should parse them. */ .url = "http://256.0.0.1/that/reverts/to/DNS", .url_parsed = { .path = "/that/reverts/to/DNS", .host_name = "256.0.0.1" } },{ .url = "http://127.0.0.284/this/also/reverts/to/DNS", .url_parsed = { .path = "/this/also/reverts/to/DNS", .host_name = "127.0.0.284" } },{ .url = "http://www.example.com/#Status%20of%20development", .flags = HTTP_URL_ALLOW_FRAGMENT_PART, .url_parsed = { .path = "/", .host_name = "www.example.com", .enc_fragment = "Status%20of%20development" } /* RFC 3986, Section 5.4. Reference Resolution Examples * * Within a representation with a well defined base URI of * * http://a/b/c/d;p?q * * a relative reference is transformed to its target URI as follows. * * 5.4.1. Normal Examples */ },{ // "g" = "http://a/b/c/g" .url = "g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g" } },{ // "./g" = "http://a/b/c/g" .url = "./g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g" } },{ // "g/" = "http://a/b/c/g/" .url = "g/", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g/" } },{ // "/g" = "http://a/g" .url = "/g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/g" } },{ // "//g" = "http://g" .url = "//g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "g" } },{ // "?y" = "http://a/b/c/d;p?y" .url = "?y", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "y" } },{ // "g?y" = "http://a/b/c/g?y" .url = "g?y", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g", .enc_query = "y" } },{ // "#s" = "http://a/b/c/d;p?q#s" .url = "#s", .flags = HTTP_URL_ALLOW_FRAGMENT_PART, .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q", .enc_fragment = "s" } },{ // "g#s" = "http://a/b/c/g#s" .url = "g#s", .flags = HTTP_URL_ALLOW_FRAGMENT_PART, .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g", .enc_fragment = "s" } },{ // "g?y#s" = "http://a/b/c/g?y#s" .url = "g?y#s", .flags = HTTP_URL_ALLOW_FRAGMENT_PART, .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g", .enc_query = "y", .enc_fragment = "s" } },{ // ";x" = "http://a/b/c/;x" .url = ";x", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/;x" } },{ // "g;x" = "http://a/b/c/g;x" .url = "g;x", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g;x" } },{ // "g;x?y#s" = "http://a/b/c/g;x?y#s" .url = "g;x?y#s", .flags = HTTP_URL_ALLOW_FRAGMENT_PART, .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g;x", .enc_query = "y", .enc_fragment = "s" } },{ // "" = "http://a/b/c/d;p?q" .url = "", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" } },{ // "." = "http://a/b/c/" .url = ".", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/" } },{ // "./" = "http://a/b/c/" .url = "./", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/" } },{ // ".." = "http://a/b/" .url = "..", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/" } },{ // "../" = "http://a/b/" .url = "../", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/" } },{ // "../g" = "http://a/b/g" .url = "../g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/g" } },{ // "../.." = "http://a/" .url = "../..", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/" } },{ // "../../" = "http://a/" .url = "../../", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/" } },{ // "../../g" = "http://a/g" .url = "../../g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/g" } /* 5.4.2. Abnormal Examples */ },{ // "../../../g" = "http://a/g" .url = "../../../g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/g" } },{ // "../../../../g" = "http://a/g" .url = "../../../../g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/g" } },{ // "/./g" = "http://a/g" .url = "/./g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/g" } },{ // "/../g" = "http://a/g" .url = "/../g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/g" } },{ // "g." = "http://a/b/c/g." .url = "g.", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g." } },{ // ".g" = "http://a/b/c/.g" .url = ".g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/.g" } },{ // "g.." = "http://a/b/c/g.." .url = "g..", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g.." } },{ // "..g" = "http://a/b/c/..g" .url = "..g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/..g" } },{ // "./../g" = "http://a/b/g" .url = "./../g", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/g" } },{ // "./g/." = "http://a/b/c/g/" .url = "./g/.", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g/" } },{ // "g/./h" = "http://a/b/c/g/h" .url = "g/./h", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g/h" } },{ // "g/../h" = "http://a/b/c/h" .url = "g/../h", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/h" } },{ // "g;x=1/./y" = "http://a/b/c/g;x=1/y" .url = "g;x=1/./y", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g;x=1/y" } },{ // "g;x=1/../y" = "http://a/b/c/y" .url = "g;x=1/../y", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/y" } },{ // "g?y/./x" = "http://a/b/c/g?y/./x" .url = "g?y/./x", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g", .enc_query = "y/./x" } },{ // "g?y/../x" = "http://a/b/c/g?y/../x" .url = "g?y/../x", .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g", .enc_query = "y/../x" } },{ // "g#s/./x" = "http://a/b/c/g#s/./x" .url = "g#s/./x", .flags = HTTP_URL_ALLOW_FRAGMENT_PART, .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g", .enc_fragment = "s/./x" } },{ // "g#s/../x" = "http://a/b/c/g#s/../x" .url = "g#s/../x", .flags = HTTP_URL_ALLOW_FRAGMENT_PART, .url_base = { .host_name = "a", .path = "/b/c/d;p", .enc_query = "q" }, .url_parsed = { .host_name = "a", .path = "/b/c/g", .enc_fragment = "s/../x" } } }; static unsigned int valid_url_test_count = N_ELEMENTS(valid_url_tests); static void test_http_url_valid(void) { unsigned int i; for (i = 0; i < valid_url_test_count; i++) T_BEGIN { const char *url = valid_url_tests[i].url; enum http_url_parse_flags flags = valid_url_tests[i].flags; struct http_url *urlt = &valid_url_tests[i].url_parsed; struct http_url *urlb = &valid_url_tests[i].url_base; struct http_url *urlp; const char *error = NULL; test_begin(t_strdup_printf("http url valid [%d]", i)); if (urlb->host_name == NULL) urlb = NULL; if (http_url_parse(url, urlb, flags, pool_datastack_create(), &urlp, &error) < 0) urlp = NULL; test_out_reason(t_strdup_printf("http_url_parse(%s)", valid_url_tests[i].url), urlp != NULL, error); if (urlp != NULL) { if (urlp->host_name == NULL || urlt->host_name == NULL) { test_assert(urlp->host_name == urlt->host_name); } else { test_assert(strcmp(urlp->host_name, urlt->host_name) == 0); } if (!urlp->have_port) { test_assert(urlp->have_port == urlt->have_port); } else { test_assert(urlp->have_port == urlt->have_port && urlp->port == urlt->port); } if (!urlp->have_host_ip) { test_assert(urlp->have_host_ip == urlt->have_host_ip); } else { test_assert(urlp->have_host_ip == urlt->have_host_ip); } if (urlp->user == NULL || urlt->user == NULL) { test_assert(urlp->user == urlt->user); } else { test_assert(strcmp(urlp->user, urlt->user) == 0); } if (urlp->password == NULL || urlt->password == NULL) { test_assert(urlp->password == urlt->password); } else { test_assert(strcmp(urlp->password, urlt->password) == 0); } if (urlp->path == NULL || urlt->path == NULL) { test_assert(urlp->path == urlt->path); } else { test_assert(strcmp(urlp->path, urlt->path) == 0); } if (urlp->enc_query == NULL || urlt->enc_query == NULL) { test_assert(urlp->enc_query == urlt->enc_query); } else { test_assert(strcmp(urlp->enc_query, urlt->enc_query) == 0); } if (urlp->enc_fragment == NULL || urlt->enc_fragment == NULL) { test_assert(urlp->enc_fragment == urlt->enc_fragment); } else { test_assert(strcmp(urlp->enc_fragment, urlt->enc_fragment) == 0); } } test_end(); } T_END; } struct invalid_http_url_test { const char *url; enum http_url_parse_flags flags; struct http_url url_base; }; static struct invalid_http_url_test invalid_url_tests[] = { { .url = "imap://example.com/INBOX" },{ .url = "http:/www.example.com" },{ .url = "" },{ .url = "/index.html" },{ .url = "http://www.example.com/index.html\"" },{ .url = "http:///dovecot.org" },{ .url = "http://[]/index.html" },{ .url = "http://[v08.234:232:234:234:2221]/index.html" #ifdef HAVE_IPV6 },{ .url = "http://[1::34a:34:234::6]/index.html" #endif },{ .url = "http://example%a.com/index.html" },{ .url = "http://example.com%/index.html" },{ .url = "http://example%00.com/index.html" },{ .url = "http://example.com:65536/index.html" },{ .url = "http://example.com:72817/index.html" },{ .url = "http://example.com/settings/%00/" },{ .url = "http://example.com/settings/%0r/" },{ .url = "http://example.com/settings/misc/%/" },{ .url = "http://example.com/?%00" },{ .url = "http://www.example.com/network.html#IMAP_Server" },{ .url = "http://example.com/#%00", .flags = HTTP_URL_ALLOW_FRAGMENT_PART } }; static unsigned int invalid_url_test_count = N_ELEMENTS(invalid_url_tests); static void test_http_url_invalid(void) { unsigned int i; for (i = 0; i < invalid_url_test_count; i++) T_BEGIN { const char *url = invalid_url_tests[i].url; enum http_url_parse_flags flags = invalid_url_tests[i].flags; struct http_url *urlb = &invalid_url_tests[i].url_base; struct http_url *urlp; const char *error = NULL; if (urlb->host_name == NULL) urlb = NULL; test_begin(t_strdup_printf("http url invalid [%d]", i)); if (http_url_parse(url, urlb, flags, pool_datastack_create(), &urlp, &error) < 0) urlp = NULL; test_out_reason(t_strdup_printf("parse %s", url), urlp == NULL, error); test_end(); } T_END; } static const char *parse_create_url_tests[] = { "http://www.example.com/", "http://10.0.0.1/", #ifdef HAVE_IPV6 "http://[::1]/", #endif "http://www.example.com:993/", "http://www.example.com/index.html", "http://www.example.com/settings/index.html", "http://ww.%23example.com/", "http://www.example.com/%23shared/news", "http://www.example.com/query.php?name=Hendrik%20Visser", "http://www.example.com/network.html#IMAP%20Server", }; static unsigned int parse_create_url_test_count = N_ELEMENTS(parse_create_url_tests); static void test_http_url_parse_create(void) { unsigned int i; for (i = 0; i < parse_create_url_test_count; i++) T_BEGIN { const char *url = parse_create_url_tests[i]; struct http_url *urlp; const char *error = NULL; test_begin(t_strdup_printf("http url parse/create [%d]", i)); if (http_url_parse (url, NULL, HTTP_URL_ALLOW_FRAGMENT_PART, pool_datastack_create(), &urlp, &error) < 0) urlp = NULL; test_out_reason(t_strdup_printf("parse %s", url), urlp != NULL, error); if (urlp != NULL) { const char *urlnew = http_url_create(urlp); test_out(t_strdup_printf ("create %s", urlnew), strcmp(url, urlnew) == 0); } test_end(); } T_END; } int main(void) { static void (*test_functions[])(void) = { test_http_url_valid, test_http_url_invalid, test_http_url_parse_create, NULL }; return test_run(test_functions); }