Blame tests/test-is-public-all.c

Packit Service dcb6c2
/*
Packit Service dcb6c2
 * Copyright(c) 2014-2018 Tim Ruehsen
Packit Service dcb6c2
 *
Packit Service dcb6c2
 * Permission is hereby granted, free of charge, to any person obtaining a
Packit Service dcb6c2
 * copy of this software and associated documentation files (the "Software"),
Packit Service dcb6c2
 * to deal in the Software without restriction, including without limitation
Packit Service dcb6c2
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
Packit Service dcb6c2
 * and/or sell copies of the Software, and to permit persons to whom the
Packit Service dcb6c2
 * Software is furnished to do so, subject to the following conditions:
Packit Service dcb6c2
 *
Packit Service dcb6c2
 * The above copyright notice and this permission notice shall be included in
Packit Service dcb6c2
 * all copies or substantial portions of the Software.
Packit Service dcb6c2
 *
Packit Service dcb6c2
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit Service dcb6c2
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit Service dcb6c2
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
Packit Service dcb6c2
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
Packit Service dcb6c2
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
Packit Service dcb6c2
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
Packit Service dcb6c2
 * DEALINGS IN THE SOFTWARE.
Packit Service dcb6c2
 *
Packit Service dcb6c2
 * This file is part of the test suite of libpsl.
Packit Service dcb6c2
 *
Packit Service dcb6c2
 * Test psl_is_public_suffix() for all entries in public_suffix_list.dat
Packit Service dcb6c2
 *
Packit Service dcb6c2
 * Changelog
Packit Service dcb6c2
 * 19.03.2014  Tim Ruehsen  created
Packit Service dcb6c2
 *
Packit Service dcb6c2
 */
Packit Service dcb6c2
Packit Service dcb6c2
#if HAVE_CONFIG_H
Packit Service dcb6c2
# include <config.h>
Packit Service dcb6c2
#endif
Packit Service dcb6c2
Packit Service dcb6c2
#include <stdio.h>
Packit Service dcb6c2
#include <stdlib.h>
Packit Service dcb6c2
#include <string.h>
Packit Service dcb6c2
#include <ctype.h>
Packit Service dcb6c2
#ifdef HAVE_ALLOCA_H
Packit Service dcb6c2
#	include <alloca.h>
Packit Service dcb6c2
#endif
Packit Service dcb6c2
Packit Service dcb6c2
#include <libpsl.h>
Packit Service dcb6c2
Packit Service dcb6c2
static int
Packit Service dcb6c2
	ok,
Packit Service dcb6c2
	failed;
Packit Service dcb6c2
#ifdef HAVE_CLOCK_GETTIME
Packit Service dcb6c2
	static struct timespec ts1, ts2;
Packit Service dcb6c2
#endif
Packit Service dcb6c2
Packit Service dcb6c2
static int _isspace_ascii(const char c)
Packit Service dcb6c2
{
Packit Service dcb6c2
	return c == ' ' || c == '\t' || c == '\r' || c == '\n';
Packit Service dcb6c2
}
Packit Service dcb6c2
Packit Service dcb6c2
static const char *_type_string(int type)
Packit Service dcb6c2
{
Packit Service dcb6c2
	switch (type) {
Packit Service dcb6c2
	case PSL_TYPE_ANY: return "PSL_TYPE_ANY";
Packit Service dcb6c2
	case PSL_TYPE_PRIVATE: return "PSL_TYPE_PRIVATE";
Packit Service dcb6c2
	case PSL_TYPE_ICANN: return "PSL_TYPE_ICANN";
Packit Service dcb6c2
	case PSL_TYPE_ANY|PSL_TYPE_NO_STAR_RULE: return "PSL_TYPE_ANY|PSL_TYPE_NO_STAR_RULE";
Packit Service dcb6c2
	case PSL_TYPE_PRIVATE|PSL_TYPE_NO_STAR_RULE: return "PSL_TYPE_PRIVATE|PSL_TYPE_NO_STAR_RULE";
Packit Service dcb6c2
	case PSL_TYPE_ICANN|PSL_TYPE_NO_STAR_RULE: return "PSL_TYPE_ICANN|PSL_TYPE_NO_STAR_RULE";
Packit Service dcb6c2
	default: return "Unsupported type";
Packit Service dcb6c2
	}
Packit Service dcb6c2
}
Packit Service dcb6c2
Packit Service dcb6c2
static void test_ps(const psl_ctx_t *psl, const char *domain, int type, int expected)
Packit Service dcb6c2
{
Packit Service dcb6c2
	int result;
Packit Service dcb6c2
Packit Service dcb6c2
	if ((result = psl_is_public_suffix2(psl, domain, type)) != expected) {
Packit Service dcb6c2
		failed++;
Packit Service dcb6c2
		printf("psl_is_public_suffix2(%s, %s)=%d (expected %d)\n", domain, _type_string(type), result, expected);
Packit Service dcb6c2
	} else ok++;
Packit Service dcb6c2
}
Packit Service dcb6c2
Packit Service dcb6c2
/* section: either PSL_TYPE_PRIVATE or PSL_TYPE_ICANN */
Packit Service dcb6c2
static void test_type_any(const psl_ctx_t *psl, const char *domain, int type, int expected)
Packit Service dcb6c2
{
Packit Service dcb6c2
	int wildcard = (*domain == '.');
Packit Service dcb6c2
	int tld = !(strchr(domain + wildcard, '.'));
Packit Service dcb6c2
Packit Service dcb6c2
	test_ps(psl, domain, type, expected);
Packit Service dcb6c2
	test_ps(psl, domain, type|PSL_TYPE_NO_STAR_RULE, expected);
Packit Service dcb6c2
	test_ps(psl, domain, PSL_TYPE_ANY, expected);
Packit Service dcb6c2
	test_ps(psl, domain, PSL_TYPE_ANY|PSL_TYPE_NO_STAR_RULE, expected);
Packit Service dcb6c2
Packit Service dcb6c2
	if (type == PSL_TYPE_PRIVATE) {
Packit Service dcb6c2
		if (tld) {
Packit Service dcb6c2
			test_ps(psl, domain, PSL_TYPE_ICANN, 1);
Packit Service dcb6c2
			test_ps(psl, domain, PSL_TYPE_ICANN|PSL_TYPE_NO_STAR_RULE, 0);
Packit Service dcb6c2
		} else {
Packit Service dcb6c2
			test_ps(psl, domain, PSL_TYPE_ICANN, 0);
Packit Service dcb6c2
			test_ps(psl, domain, PSL_TYPE_ICANN|PSL_TYPE_NO_STAR_RULE, 0);
Packit Service dcb6c2
		}
Packit Service dcb6c2
	} else if (type == PSL_TYPE_ICANN) {
Packit Service dcb6c2
		if (tld) {
Packit Service dcb6c2
			test_ps(psl, domain, PSL_TYPE_PRIVATE, 1);
Packit Service dcb6c2
			test_ps(psl, domain, PSL_TYPE_PRIVATE|PSL_TYPE_NO_STAR_RULE, 0);
Packit Service dcb6c2
		} else {
Packit Service dcb6c2
			test_ps(psl, domain, PSL_TYPE_PRIVATE, 0);
Packit Service dcb6c2
			test_ps(psl, domain, PSL_TYPE_PRIVATE|PSL_TYPE_NO_STAR_RULE, 0);
Packit Service dcb6c2
		}
Packit Service dcb6c2
	}
Packit Service dcb6c2
}
Packit Service dcb6c2
Packit Service dcb6c2
static void test_psl_entry(const psl_ctx_t *psl, const char *domain, int type)
Packit Service dcb6c2
{
Packit Service dcb6c2
	if (*domain == '!') { /* an exception to a wildcard, e.g. !www.ck (wildcard is *.ck) */
Packit Service dcb6c2
		test_type_any(psl, domain + 1, type, 0); /* the exception itself is not a PS */
Packit Service dcb6c2
Packit Service dcb6c2
		if ((domain = strchr(domain, '.')))
Packit Service dcb6c2
			test_type_any(psl, domain, type, 1); /* the related wildcard domain is a PS */
Packit Service dcb6c2
Packit Service dcb6c2
	} else if (*domain == '*') { /* a wildcard, e.g. *.ck or *.platform.sh */
Packit Service dcb6c2
		/* '*.platform.sh' -> 'y.x.platform.sh' */
Packit Service dcb6c2
		size_t len = strlen(domain);
Packit Service dcb6c2
		char *xdomain = alloca(len + 3);
Packit Service dcb6c2
Packit Service dcb6c2
		memcpy(xdomain, "y.x", 3);
Packit Service dcb6c2
		memcpy(xdomain + 3, domain + 1, len);
Packit Service dcb6c2
Packit Service dcb6c2
		test_type_any(psl, domain + 1, type, 1); /* the domain without wildcard is a PS */
Packit Service dcb6c2
		test_type_any(psl, xdomain + 2, type, 1); /* random wildcard-matching domain is a PS... */
Packit Service dcb6c2
		test_type_any(psl, xdomain, type, 0); /* ... but sub domain is not */
Packit Service dcb6c2
Packit Service dcb6c2
	} else {
Packit Service dcb6c2
		test_type_any(psl, domain, type, 1); /* Any normal PSL entry */
Packit Service dcb6c2
	}
Packit Service dcb6c2
}
Packit Service dcb6c2
Packit Service dcb6c2
static void test_psl(void)
Packit Service dcb6c2
{
Packit Service dcb6c2
	FILE *fp;
Packit Service dcb6c2
	psl_ctx_t *psl, *psl3, *psl4, *psl5;
Packit Service dcb6c2
	const psl_ctx_t *psl2;
Packit Service dcb6c2
	int type = 0;
Packit Service dcb6c2
	char buf[256], *linep, *p;
Packit Service dcb6c2
Packit Service dcb6c2
	psl = psl_load_file(PSL_FILE); /* PSL_FILE can be set by ./configure --with-psl-file=[PATH] */
Packit Service dcb6c2
	printf("loaded %d suffixes and %d exceptions\n", psl_suffix_count(psl), psl_suffix_exception_count(psl));
Packit Service dcb6c2
Packit Service dcb6c2
	psl2 = psl_builtin();
Packit Service dcb6c2
	printf("builtin PSL has %d suffixes and %d exceptions\n", psl_suffix_count(psl2), psl_suffix_exception_count(psl2));
Packit Service dcb6c2
Packit Service dcb6c2
	if (!(psl3 = psl_load_file("psl.dafsa"))) {
Packit Service dcb6c2
		fprintf(stderr, "Failed to load 'psl.dafsa'\n");
Packit Service dcb6c2
		failed++;
Packit Service dcb6c2
	}
Packit Service dcb6c2
Packit Service dcb6c2
	if (!(psl4 = psl_load_file("psl_ascii.dafsa"))) {
Packit Service dcb6c2
		fprintf(stderr, "Failed to load 'psl_ascii.dafsa'\n");
Packit Service dcb6c2
		failed++;
Packit Service dcb6c2
	}
Packit Service dcb6c2
Packit Service dcb6c2
	psl5 = psl_latest("psl.dafsa");
Packit Service dcb6c2
Packit Service dcb6c2
	if ((fp = fopen(PSL_FILE, "r"))) {
Packit Service dcb6c2
#ifdef HAVE_CLOCK_GETTIME
Packit Service dcb6c2
		clock_gettime(CLOCK_REALTIME, &ts1;;
Packit Service dcb6c2
#endif
Packit Service dcb6c2
Packit Service dcb6c2
		while ((linep = fgets(buf, sizeof(buf), fp))) {
Packit Service dcb6c2
			while (_isspace_ascii(*linep)) linep++; /* ignore leading whitespace */
Packit Service dcb6c2
			if (!*linep) continue; /* skip empty lines */
Packit Service dcb6c2
Packit Service dcb6c2
			if (*linep == '/' && linep[1] == '/') {
Packit Service dcb6c2
				if (!type) {
Packit Service dcb6c2
					if (strstr(linep + 2, "===BEGIN ICANN DOMAINS==="))
Packit Service dcb6c2
						type = PSL_TYPE_ICANN;
Packit Service dcb6c2
					else if (!type && strstr(linep + 2, "===BEGIN PRIVATE DOMAINS==="))
Packit Service dcb6c2
						type = PSL_TYPE_PRIVATE;
Packit Service dcb6c2
				}
Packit Service dcb6c2
				else if (type == PSL_TYPE_ICANN && strstr(linep + 2, "===END ICANN DOMAINS==="))
Packit Service dcb6c2
					type = 0;
Packit Service dcb6c2
				else if (type == PSL_TYPE_PRIVATE && strstr(linep + 2, "===END PRIVATE DOMAINS==="))
Packit Service dcb6c2
					type = 0;
Packit Service dcb6c2
Packit Service dcb6c2
				continue; /* skip comments */
Packit Service dcb6c2
			}
Packit Service dcb6c2
Packit Service dcb6c2
			/* parse suffix rule */
Packit Service dcb6c2
			for (p = linep; *linep && !_isspace_ascii(*linep);) linep++;
Packit Service dcb6c2
			*linep = 0;
Packit Service dcb6c2
Packit Service dcb6c2
			test_psl_entry(psl, p, type);
Packit Service dcb6c2
Packit Service dcb6c2
			if (psl2)
Packit Service dcb6c2
				test_psl_entry(psl2, p, type);
Packit Service dcb6c2
Packit Service dcb6c2
			if (psl3)
Packit Service dcb6c2
				test_psl_entry(psl3, p, type);
Packit Service dcb6c2
Packit Service dcb6c2
			if (psl4)
Packit Service dcb6c2
				test_psl_entry(psl4, p, type);
Packit Service dcb6c2
Packit Service dcb6c2
			if (psl5)
Packit Service dcb6c2
				test_psl_entry(psl5, p, type);
Packit Service dcb6c2
		}
Packit Service dcb6c2
Packit Service dcb6c2
#ifdef HAVE_CLOCK_GETTIME
Packit Service dcb6c2
		clock_gettime(CLOCK_REALTIME, &ts2;;
Packit Service dcb6c2
#endif
Packit Service dcb6c2
		fclose(fp);
Packit Service dcb6c2
	} else {
Packit Service dcb6c2
		printf("Failed to open %s\n", PSL_FILE);
Packit Service dcb6c2
		failed++;
Packit Service dcb6c2
	}
Packit Service dcb6c2
Packit Service dcb6c2
	psl_free(psl5);
Packit Service dcb6c2
	psl_free(psl4);
Packit Service dcb6c2
	psl_free(psl3);
Packit Service dcb6c2
	psl_free((psl_ctx_t *)psl2);
Packit Service dcb6c2
	psl_free(psl);
Packit Service dcb6c2
}
Packit Service dcb6c2
Packit Service dcb6c2
int main(int argc, const char * const *argv)
Packit Service dcb6c2
{
Packit Service dcb6c2
#ifdef HAVE_CLOCK_GETTIME
Packit Service dcb6c2
	long ns;
Packit Service dcb6c2
#endif
Packit Service dcb6c2
Packit Service dcb6c2
	/* if VALGRIND testing is enabled, we have to call ourselves with valgrind checking */
Packit Service dcb6c2
	if (argc == 1) {
Packit Service dcb6c2
		const char *valgrind = getenv("TESTS_VALGRIND");
Packit Service dcb6c2
Packit Service dcb6c2
		if (valgrind && *valgrind) {
Packit Service dcb6c2
			size_t cmdsize = strlen(valgrind) + strlen(argv[0]) + 32;
Packit Service dcb6c2
			char *cmd = alloca(cmdsize);
Packit Service dcb6c2
Packit Service dcb6c2
			snprintf(cmd, cmdsize, "TESTS_VALGRIND="" %s %s", valgrind, argv[0]);
Packit Service dcb6c2
			return system(cmd) != 0;
Packit Service dcb6c2
		}
Packit Service dcb6c2
	}
Packit Service dcb6c2
Packit Service dcb6c2
	test_psl();
Packit Service dcb6c2
Packit Service dcb6c2
	if (failed) {
Packit Service dcb6c2
		printf("Summary: %d out of %d tests failed\n", failed, ok + failed);
Packit Service dcb6c2
		return 1;
Packit Service dcb6c2
	}
Packit Service dcb6c2
Packit Service dcb6c2
#ifdef HAVE_CLOCK_GETTIME
Packit Service dcb6c2
	if (ts1.tv_sec == ts2.tv_sec)
Packit Service dcb6c2
		ns = ts2.tv_nsec - ts1.tv_nsec;
Packit Service dcb6c2
	else if (ts1.tv_sec == ts2.tv_sec - 1)
Packit Service dcb6c2
		ns = 1000000000L - (ts2.tv_nsec - ts1.tv_nsec);
Packit Service dcb6c2
	else
Packit Service dcb6c2
		ns = 0; /* let's assume something is wrong and skip outputting measured time */
Packit Service dcb6c2
Packit Service dcb6c2
	if (ns)
Packit Service dcb6c2
		printf("Summary: All %d tests passed in %ld.%06ld ms\n", ok, ns / 1000000, ns % 1000000000);
Packit Service dcb6c2
	else
Packit Service dcb6c2
		printf("Summary: All %d tests passed\n", ok);
Packit Service dcb6c2
#else
Packit Service dcb6c2
	printf("Summary: All %d tests passed\n", ok);
Packit Service dcb6c2
#endif
Packit Service dcb6c2
Packit Service dcb6c2
	return 0;
Packit Service dcb6c2
}