/* Copyright 2011-2017 David Robillard Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies. THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include "serd/serd.h" #include "test_utils.h" #ifndef INFINITY # define INFINITY (DBL_MAX + DBL_MAX) #endif #ifndef NAN # define NAN (INFINITY - INFINITY) #endif static int test_strtod(double dbl, double max_delta) { char buf[1024]; snprintf(buf, sizeof(buf), "%f", dbl); size_t end = 0; const double out = serd_strtod(buf, &end); const double diff = fabs(out - dbl); if (diff > max_delta) { FAILF("Parsed %lf != %lf (delta %lf)\n", dbl, out, diff); } if (end != strlen(buf)) { FAILF("Parsed %lf length %zu != %zu\n", end, strlen(buf)); } return 0; } static SerdStatus count_prefixes(void* handle, const SerdNode* name, const SerdNode* uri) { ++*(int*)handle; return SERD_SUCCESS; } typedef struct { int n_statements; const SerdNode* graph; } ReaderTest; static SerdStatus test_sink(void* handle, SerdStatementFlags flags, const SerdNode* graph, const SerdNode* subject, const SerdNode* predicate, const SerdNode* object, const SerdNode* object_datatype, const SerdNode* object_lang) { ReaderTest* rt = (ReaderTest*)handle; ++rt->n_statements; rt->graph = graph; return SERD_SUCCESS; } static int check_file_uri(const char* hostname, const char* path, bool escape, const char* expected_uri, const char* expected_path) { if (!expected_path) { expected_path = path; } SerdNode* node = serd_node_new_file_uri(path, hostname, 0, escape); const char* node_str = serd_node_get_string(node); char* out_hostname = NULL; char* out_path = serd_file_uri_parse(node_str, &out_hostname); int ret = 0; if (strcmp(node_str, expected_uri)) { ret = FAILF("Bad URI %s != %s\n", node_str, expected_uri); } else if (!hostname && out_hostname) { ret = FAILF("hostname `%s' shouldn't exist\n", out_hostname); } else if (hostname && !out_hostname) { ret = FAILF("expected hostname `%s'\n", hostname); } else if (strcmp((const char*)out_path, (const char*)expected_path)) { ret = FAILF("path=>URI=>path failure %s => %s => %s\n", path, node_str, out_path); } free(out_path); free(out_hostname); serd_node_free(node); return ret; } int main(void) { #define MAX 1000000 #define NUM_TESTS 1000 for (int i = 0; i < NUM_TESTS; ++i) { double dbl = rand() % MAX; dbl += (rand() % MAX) / (double)MAX; if (test_strtod(dbl, 1 / (double)MAX)) { return 1; } } const double expt_test_nums[] = { 2.0E18, -5e19, +8e20, 2e+24, -5e-5, 8e0, 9e-0, 2e+0 }; const char* expt_test_strs[] = { "02e18", "-5e019", "+8e20", "2E+24", "-5E-5", "8E0", "9e-0", " 2e+0" }; for (unsigned i = 0; i < sizeof(expt_test_nums) / sizeof(double); ++i) { const double num = serd_strtod(expt_test_strs[i], NULL); const double delta = fabs(num - expt_test_nums[i]); if (delta > DBL_EPSILON) { FAILF("Parsed `%s' %lf != %lf (delta %lf)\n", expt_test_strs[i], num, expt_test_nums[i], delta); } } // Test serd_node_new_decimal const double dbl_test_nums[] = { 0.0, 9.0, 10.0, .01, 2.05, -16.00001, 5.000000005, 0.0000000001, NAN, INFINITY }; const char* dbl_test_strs[] = { "0.0", "9.0", "10.0", "0.01", "2.05", "-16.00001", "5.00000001", "0.0", NULL, NULL }; for (unsigned i = 0; i < sizeof(dbl_test_nums) / sizeof(double); ++i) { SerdNode* node = serd_node_new_decimal(dbl_test_nums[i], 8); const char* node_str = serd_node_get_string(node); const bool pass = (node_str && dbl_test_strs[i]) ? !strcmp(node_str, dbl_test_strs[i]) : (node_str == dbl_test_strs[i]); if (!pass) { FAILF("Serialised `%s' != %s\n", node_str, dbl_test_strs[i]); } const size_t len = node_str ? strlen(node_str) : 0; if (serd_node_get_length(node) != len) { FAILF("Length %zu != %zu\n", serd_node_get_length(node), len); } serd_node_free(node); } // Test serd_node_new_integer const long int_test_nums[] = { 0, -0, -23, 23, -12340, 1000, -1000 }; const char* int_test_strs[] = { "0", "0", "-23", "23", "-12340", "1000", "-1000" }; for (unsigned i = 0; i < sizeof(int_test_nums) / sizeof(double); ++i) { SerdNode* node = serd_node_new_integer(int_test_nums[i]); const char* node_str = serd_node_get_string(node); if (strcmp(node_str, (const char*)int_test_strs[i])) { FAILF("Serialised `%s' != %s\n", node_str, int_test_strs[i]); } const size_t len = strlen(node_str); if (serd_node_get_length(node) != len) { FAILF("Length %zu,%zu != %zu\n", serd_node_get_length(node), len); } serd_node_free(node); } // Test serd_node_new_blob for (size_t size = 0; size < 256; ++size) { uint8_t* data = (uint8_t*)malloc(size); for (size_t i = 0; i < size; ++i) { data[i] = (uint8_t)(rand() % 256); } size_t out_size; SerdNode* blob = serd_node_new_blob(data, size, size % 5); const char* blob_str = serd_node_get_string(blob); uint8_t* out = (uint8_t*)serd_base64_decode( blob_str, serd_node_get_length(blob), &out_size); if (out_size != size) { FAILF("Blob size %zu != %zu\n", out_size, size); } for (size_t i = 0; i < size; ++i) { if (out[i] != data[i]) { FAILF("Corrupt blob at byte %zu\n", i); } } serd_node_free(blob); free(out); free(data); } // Test serd_strlen const uint8_t str[] = { '"', '5', 0xE2, 0x82, 0xAC, '"', '\n', 0 }; SerdNodeFlags flags; size_t n_bytes = serd_strlen((const char*)str, &flags); if (n_bytes != 7 || flags != (SERD_HAS_QUOTE|SERD_HAS_NEWLINE)) { FAILF("Bad serd_strlen(%s) n_bytes=%zu flags=%u\n", str, n_bytes, flags); } // Test serd_strerror const char* msg = NULL; if (strcmp((msg = serd_strerror(SERD_SUCCESS)), "Success")) { FAILF("Bad message `%s' for SERD_SUCCESS\n", msg); } for (int i = SERD_FAILURE; i <= SERD_ERR_INTERNAL; ++i) { msg = serd_strerror((SerdStatus)i); if (!strcmp((const char*)msg, "Success")) { FAILF("Bad message `%s' for (SerdStatus)%d\n", msg, i); } } msg = serd_strerror((SerdStatus)-1); // Test file URI escaping and parsing if (check_file_uri(NULL, "C:/My 100%", true, "file:///C:/My%20100%%", NULL) || check_file_uri("ahost", "C:\\Pointless Space", true, "file://ahost/C:/Pointless%20Space", "C:/Pointless Space") || check_file_uri(NULL, "/foo/bar", true, "file:///foo/bar", NULL) || check_file_uri("bhost", "/foo/bar", true, "file://bhost/foo/bar", NULL) || check_file_uri(NULL, "a/relative path", false, "a/relative path", NULL) || check_file_uri(NULL, "a/relative ", true, "a/relative%20%3Cpath%3E", NULL)) { return 1; } // Test tolerance of parsing junk URI escapes char* out_path = serd_file_uri_parse("file:///foo/%0Xbar", NULL); if (strcmp(out_path, "/foo/bar")) { FAILF("bad tolerance of junk escape: `%s'\n", out_path); } free(out_path); // Test serd_node_equals const uint8_t replacement_char_str[] = { 0xEF, 0xBF, 0xBD, 0 }; SerdNode* lhs = serd_node_new_string(SERD_LITERAL, (const char*)replacement_char_str); SerdNode* rhs = serd_node_new_string(SERD_LITERAL, "123"); if (serd_node_equals(lhs, rhs)) { FAILF("%s == %s\n", serd_node_get_string(lhs), serd_node_get_string(rhs)); } SerdNode* qnode = serd_node_new_string(SERD_CURIE, "foo:bar"); if (serd_node_equals(lhs, qnode)) { FAILF("%s == %s\n", serd_node_get_string(lhs), serd_node_get_string(qnode)); } serd_node_free(qnode); if (!serd_node_equals(lhs, lhs)) { FAILF("%s != %s\n", serd_node_get_string(lhs), serd_node_get_string(lhs)); } if (serd_node_copy(NULL)) { FAIL("made non-null copy of null node\n"); } serd_node_free(lhs); serd_node_free(rhs); // Test serd_node_new_string SerdNode* hello = serd_node_new_string(SERD_LITERAL, "hello\""); if (serd_node_get_length(hello) != 6 || serd_node_get_flags(hello) != SERD_HAS_QUOTE || strcmp(serd_node_get_string(hello), "hello\"")) { FAILF("Bad node %s\n", serd_node_get_string(hello)); } if (serd_node_new_string(SERD_URI, NULL)) { FAIL("Successfully created node from NULL string\n"); } // Test serd_node_new_substring if (serd_node_new_substring(SERD_LITERAL, NULL, 32)) { FAIL("Successfully created node from NULL substring\n"); } SerdNode* a_b = serd_node_new_substring(SERD_LITERAL, "a\"bc", 3); if (serd_node_get_length(a_b) != 3 || serd_node_get_flags(a_b) != SERD_HAS_QUOTE || strncmp(serd_node_get_string(a_b), "a\"b", 3)) { FAILF("Bad node %s\n", serd_node_get_string(a_b)); } serd_node_free(a_b); a_b = serd_node_new_substring(SERD_LITERAL, "a\"bc", 10); if (serd_node_get_length(a_b) != 4 || serd_node_get_flags(a_b) != SERD_HAS_QUOTE || strncmp(serd_node_get_string(a_b), "a\"bc", 4)) { FAILF("Bad node %s\n", serd_node_get_string(a_b)); } serd_node_free(a_b); // Test serd_node_new_uri_from_string if (serd_node_new_uri_from_string(NULL, NULL, NULL)) { FAIL("Successfully created NULL URI\n"); } SerdURI base_uri; SerdNode* base = serd_node_new_uri_from_string("http://example.org/", NULL, &base_uri); SerdNode* nil = serd_node_new_uri_from_string(NULL, &base_uri, NULL); SerdNode* nil2 = serd_node_new_uri_from_string("", &base_uri, NULL); if (serd_node_get_type(nil) != SERD_URI || strcmp(serd_node_get_string(nil), serd_node_get_string(base)) || serd_node_get_type(nil2) != SERD_URI || strcmp(serd_node_get_string(nil2), serd_node_get_string(base))) { FAILF("URI %s != base %s\n", serd_node_get_string(nil), serd_node_get_string(base)); } // Test serd_node_new_relative_uri SerdNode* abs = serd_node_new_string(SERD_URI, "http://example.org/foo/bar"); SerdURI abs_uri; serd_uri_parse(serd_node_get_string(abs), &abs_uri); SerdURI rel_uri; SerdNode* rel = serd_node_new_relative_uri(&abs_uri, &base_uri, NULL, &rel_uri); if (strcmp(serd_node_get_string(rel), "/foo/bar")) { FAILF("Bad relative URI %s (expected '/foo/bar')\n", serd_node_get_string(rel)); } serd_node_free(abs); serd_node_free(nil); serd_node_free(nil2); serd_node_free(rel); serd_node_free(base); // Test SerdEnv SerdNode* u = serd_node_new_string(SERD_URI, "http://example.org/foo"); SerdNode* b = serd_node_new_string(SERD_CURIE, "invalid"); SerdNode* c = serd_node_new_string(SERD_CURIE, "eg.2:b"); SerdEnv* env = serd_env_new(NULL); serd_env_set_prefix_from_strings(env, "eg.2", "http://example.org/"); if (serd_env_get_base_uri(env, NULL)) { FAIL("Unexpected initial base URI\n"); } if (!serd_env_set_base_uri(env, NULL)) { FAIL("Successfully set NULL base URI\n"); } if (!serd_env_set_base_uri(env, hello)) { FAILF("Set base URI to %s\n", serd_node_get_string(hello)); } if (serd_env_get_base_uri(env, NULL)) { FAIL("Unexpected base URI\n"); } SerdSlice prefix, suffix; if (!serd_env_expand(env, b, &prefix, &suffix)) { FAILF("Expanded invalid curie %s\n", serd_node_get_string(b)); } SerdNode* xnode = serd_env_expand_node(env, hello); if (xnode) { FAILF("Expanded %s\n", serd_node_get_string(c)); } serd_node_free(hello); SerdNode* xu = serd_env_expand_node(env, u); if (strcmp(serd_node_get_string(xu), "http://example.org/foo")) { FAILF("Expanded %s to %s\n", serd_node_get_string(c), serd_node_get_string(xu)); } serd_node_free(xu); SerdNode* badpre = serd_node_new_string(SERD_CURIE, "hm:what"); SerdNode* xbadpre = serd_env_expand_node(env, badpre); if (xbadpre) { FAILF("Expanded invalid curie %s\n", serd_node_get_string(badpre)); } serd_node_free(badpre); SerdNode* xc = serd_env_expand_node(env, c); if (strcmp(serd_node_get_string(xc), "http://example.org/b")) { FAILF("Expanded %s to %s\n", serd_node_get_string(c), serd_node_get_string(xc)); } serd_node_free(xc); if (!serd_env_set_prefix(env, NULL, NULL)) { FAIL("Set NULL prefix\n"); } SerdNode* lit = serd_node_new_string(SERD_LITERAL, "hello"); if (!serd_env_set_prefix(env, b, lit)) { FAIL("Set prefix to literal\n"); } int n_prefixes = 0; serd_env_set_prefix_from_strings(env, "eg.2", "http://example.org/"); serd_env_foreach(env, count_prefixes, &n_prefixes); if (n_prefixes != 1) { FAILF("Bad prefix count %d\n", n_prefixes); } SerdNode* shorter_uri = serd_node_new_string(SERD_URI, "urn:foo"); const SerdNode* prefix_name; if (serd_env_qualify(env, shorter_uri, &prefix_name, &suffix)) { FAILF("Qualified %s\n", serd_node_get_string(shorter_uri)); } serd_node_free(shorter_uri); serd_node_free(u); serd_node_free(b); serd_node_free(c); // Test SerdReader and SerdWriter const char* path = "serd_test.ttl"; FILE* fd = fopen(path, "wb"); if (!fd) { FAILF("Failed to open file %s\n", path); } SerdWriter* writer = serd_writer_new( SERD_TURTLE, (SerdStyle)0, env, NULL, serd_file_sink, fd); if (!writer) { FAIL("Failed to create writer\n"); } serd_writer_chop_blank_prefix(writer, "tmp"); serd_writer_chop_blank_prefix(writer, NULL); if (!serd_writer_set_base_uri(writer, lit)) { FAILF("Set base URI to %s\n", serd_node_get_string(lit)); } else if (!serd_writer_set_prefix(writer, lit, lit)) { FAILF("Set prefix %s to %s\n", serd_node_get_string(lit), serd_node_get_string(lit)); } else if (!serd_writer_end_anon(writer, NULL)) { FAIL("Ended non-existent anonymous node\n"); } else if (serd_writer_get_env(writer) != env) { FAIL("Writer has incorrect env\n"); } uint8_t buf[] = { 0xEF, 0xBF, 0xBD, 0 }; SerdNode* s = serd_node_new_string(SERD_URI, ""); SerdNode* p = serd_node_new_string(SERD_URI, "http://example.org/pred"); SerdNode* o = serd_node_new_string(SERD_LITERAL, (char*)buf); // Write 3 invalid statements (should write nothing) const SerdNode* junk[][5] = { { s, p, NULL, NULL, NULL }, { s, NULL, o, NULL, NULL }, { NULL, p, o, NULL, NULL }, { s, p, NULL, NULL, NULL }, { s, NULL, o, NULL, NULL }, { NULL, p, o, NULL, NULL }, { s, o, o, NULL, NULL }, { o, p, o, NULL, NULL }, { s, p, NULL, NULL, NULL }, { NULL, NULL, NULL, NULL, NULL } }; for (unsigned i = 0; i < sizeof(junk) / (sizeof(SerdNode*) * 5); ++i) { if (!serd_writer_write_statement( writer, 0, NULL, junk[i][0], junk[i][1], junk[i][2], junk[i][3], junk[i][4])) { FAILF("Successfully wrote junk statement %d\n", i); } } SerdNode* t = serd_node_new_string(SERD_URI, "urn:Type"); SerdNode* l = serd_node_new_string(SERD_LITERAL, "en"); const SerdNode* good[][5] = { { s, p, o, NULL, NULL }, { s, p, o, NULL, NULL }, { s, p, o, t, NULL }, { s, p, o, NULL, l }, { s, p, o, t, l }, { s, p, o, t, NULL }, { s, p, o, NULL, l }, { s, p, o, NULL, NULL }, { s, p, o, NULL, NULL }, { s, p, o, NULL, NULL } }; for (unsigned i = 0; i < sizeof(good) / (sizeof(SerdNode*) * 5); ++i) { if (serd_writer_write_statement( writer, 0, NULL, good[i][0], good[i][1], good[i][2], good[i][3], good[i][4])) { FAILF("Failed to write good statement %d\n", i); } } // Write statements with bad UTF-8 (should be replaced) const char bad_str[] = { (char)0xFF, (char)0x90, 'h', 'i', 0 }; SerdNode* bad_lit = serd_node_new_string(SERD_LITERAL, bad_str); SerdNode* bad_uri = serd_node_new_string(SERD_URI, bad_str); if (serd_writer_write_statement(writer, 0, NULL, s, p, bad_lit, NULL, NULL)) { FAIL("Failed to write junk UTF-8 literal\n"); } else if (serd_writer_write_statement(writer, 0, NULL, s, p, bad_uri, NULL, NULL)) { FAIL("Failed to write junk UTF-8 URI\n"); } serd_node_free(bad_uri); serd_node_free(bad_lit); // Write 1 valid statement serd_node_free(o); o = serd_node_new_string(SERD_LITERAL, "hello"); if (serd_writer_write_statement(writer, 0, NULL, s, p, o, NULL, NULL)) { FAIL("Failed to write valid statement\n"); } serd_writer_free(writer); serd_node_free(lit); serd_node_free(s); serd_node_free(p); serd_node_free(o); serd_node_free(t); serd_node_free(l); // Test buffer sink SerdBuffer buffer = { NULL, 0 }; writer = serd_writer_new( SERD_TURTLE, (SerdStyle)0, env, NULL, serd_buffer_sink, &buffer); o = serd_node_new_string(SERD_URI, "http://example.org/base"); if (serd_writer_set_base_uri(writer, o)) { FAIL("Failed to write to chunk sink\n"); } serd_node_free(o); serd_writer_free(writer); char* out = serd_buffer_sink_finish(&buffer); if (strcmp((const char*)out, "@base .\n")) { FAILF("Incorrect buffer output:\n%s\n", buffer.buf); } free(out); // Rewind and test reader fseek(fd, 0, SEEK_SET); ReaderTest* rt = (ReaderTest*)calloc(1, sizeof(ReaderTest)); SerdReader* reader = serd_reader_new( SERD_TURTLE, rt, free, NULL, NULL, test_sink, NULL); if (!reader) { FAIL("Failed to create reader\n"); } if (serd_reader_get_handle(reader) != rt) { FAIL("Corrupt reader handle\n"); } SerdNode* g = serd_node_new_string(SERD_URI, "http://example.org/"); serd_reader_set_default_graph(reader, g); serd_reader_add_blank_prefix(reader, "tmp"); serd_reader_add_blank_prefix(reader, NULL); serd_node_free(g); if (!serd_reader_read_file(reader, "http://notafile")) { FAIL("Apparently read an http URI\n"); } else if (!serd_reader_read_file(reader, "file:///better/not/exist")) { FAIL("Apparently read a non-existent file\n"); } else if (!serd_reader_read_file(reader, "file://")) { FAIL("Apparently read a file with no path\n"); } const SerdStatus st = serd_reader_read_file(reader, path); if (st) { FAILF("Error reading file (%s)\n", serd_strerror(st)); } else if (rt->n_statements != 13) { FAILF("Bad statement count %d\n", rt->n_statements); } else if (!rt->graph || !serd_node_get_string(rt->graph) || strcmp(serd_node_get_string(rt->graph), "http://example.org/")) { FAILF("Bad graph %p\n", rt->graph); } if (!serd_reader_read_string(reader, "This isn't Turtle at all.")) { FAIL("Parsed invalid string successfully.\n"); } serd_reader_free(reader); fclose(fd); serd_env_free(env); printf("Success\n"); return 0; }