comparison ext/sord/src/sord_test.c @ 226:c5cdc9e6a4bf

Add these external library files
author Chris Cannam <cannam@all-day-breakfast.com>
date Fri, 09 Jun 2017 16:41:31 +0100
parents
children
comparison
equal deleted inserted replaced
225:025b3e2f7c17 226:c5cdc9e6a4bf
1 /*
2 Copyright 2011-2016 David Robillard <http://drobilla.net>
3
4 Permission to use, copy, modify, and/or distribute this software for any
5 purpose with or without fee is hereby granted, provided that the above
6 copyright notice and this permission notice appear in all copies.
7
8 THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 */
16
17 #include <stdarg.h>
18 #include <stdio.h>
19 #include <stdlib.h>
20 #include <string.h>
21
22 #include "sord/sord.h"
23
24 static const int DIGITS = 3;
25 static const unsigned n_objects_per = 2;
26
27 static int n_expected_errors = 0;
28
29 typedef struct {
30 SordQuad query;
31 int expected_num_results;
32 } QueryTest;
33
34 #define USTR(s) ((const uint8_t*)(s))
35
36 static SordNode*
37 uri(SordWorld* world, int num)
38 {
39 if (num == 0)
40 return 0;
41
42 char str[] = "eg:000";
43 char* uri_num = str + 3; // First `0'
44 snprintf(uri_num, DIGITS + 1, "%0*d", DIGITS, num);
45 return sord_new_uri(world, (const uint8_t*)str);
46 }
47
48 static int
49 test_fail(const char* fmt, ...)
50 {
51 va_list args;
52 va_start(args, fmt);
53 fprintf(stderr, "error: ");
54 vfprintf(stderr, fmt, args);
55 va_end(args);
56 return 1;
57 }
58
59 static int
60 generate(SordWorld* world,
61 SordModel* sord,
62 size_t n_quads,
63 SordNode* graph)
64 {
65 fprintf(stderr, "Generating %zu (S P *) quads with %u objects each\n",
66 n_quads, n_objects_per);
67
68 for (size_t i = 0; i < n_quads; ++i) {
69 int num = (i * n_objects_per) + 1;
70
71 SordNode* ids[2 + n_objects_per];
72 for (unsigned j = 0; j < 2 + n_objects_per; ++j) {
73 ids[j] = uri(world, num++);
74 }
75
76 for (unsigned j = 0; j < n_objects_per; ++j) {
77 SordQuad tup = { ids[0], ids[1], ids[2 + j], graph };
78 if (!sord_add(sord, tup)) {
79 return test_fail("Fail: Failed to add quad\n");
80 }
81 }
82
83 for (unsigned j = 0; j < 2 + n_objects_per; ++j) {
84 sord_node_free(world, ids[j]);
85 }
86 }
87
88 // Add some literals
89
90 // (98 4 "hello") and (98 4 "hello"^^<5>)
91 SordQuad tup = { 0, 0, 0, 0 };
92 tup[0] = uri(world, 98);
93 tup[1] = uri(world, 4);
94 tup[2] = sord_new_literal(world, 0, USTR("hello"), NULL);
95 tup[3] = graph;
96 sord_add(sord, tup);
97 sord_node_free(world, (SordNode*)tup[2]);
98 tup[2] = sord_new_literal(world, uri(world, 5), USTR("hello"), NULL);
99 if (!sord_add(sord, tup)) {
100 return test_fail("Failed to add typed literal\n");
101 }
102
103 // (96 4 "hello"^^<4>) and (96 4 "hello"^^<5>)
104 tup[0] = uri(world, 96);
105 tup[1] = uri(world, 4);
106 tup[2] = sord_new_literal(world, uri(world, 4), USTR("hello"), NULL);
107 tup[3] = graph;
108 sord_add(sord, tup);
109 sord_node_free(world, (SordNode*)tup[2]);
110 tup[2] = sord_new_literal(world, uri(world, 5), USTR("hello"), NULL);
111 if (!sord_add(sord, tup)) {
112 return test_fail("Failed to add typed literal\n");
113 }
114
115 // (94 5 "hello") and (94 5 "hello"@en-gb)
116 tup[0] = uri(world, 94);
117 tup[1] = uri(world, 5);
118 tup[2] = sord_new_literal(world, 0, USTR("hello"), NULL);
119 tup[3] = graph;
120 sord_add(sord, tup);
121 sord_node_free(world, (SordNode*)tup[2]);
122 tup[2] = sord_new_literal(world, NULL, USTR("hello"), "en-gb");
123 if (!sord_add(sord, tup)) {
124 return test_fail("Failed to add literal with language\n");
125 }
126
127 // (92 6 "hello"@en-us) and (92 5 "hello"@en-gb)
128 tup[0] = uri(world, 92);
129 tup[1] = uri(world, 6);
130 tup[2] = sord_new_literal(world, 0, USTR("hello"), "en-us");
131 tup[3] = graph;
132 sord_add(sord, tup);
133 sord_node_free(world, (SordNode*)tup[2]);
134 tup[2] = sord_new_literal(world, NULL, USTR("hello"), "en-gb");
135 if (!sord_add(sord, tup)) {
136 return test_fail("Failed to add literal with language\n");
137 }
138
139 sord_node_free(world, (SordNode*)tup[0]);
140 sord_node_free(world, (SordNode*)tup[2]);
141 tup[0] = uri(world, 14);
142 tup[2] = sord_new_literal(world, 0, USTR("bonjour"), "fr");
143 sord_add(sord, tup);
144 sord_node_free(world, (SordNode*)tup[2]);
145 tup[2] = sord_new_literal(world, 0, USTR("salut"), "fr");
146 sord_add(sord, tup);
147
148 // Attempt to add some duplicates
149 if (sord_add(sord, tup)) {
150 return test_fail("Fail: Successfully added duplicate quad\n");
151 }
152 if (sord_add(sord, tup)) {
153 return test_fail("Fail: Successfully added duplicate quad\n");
154 }
155
156 // Add a blank node subject
157 sord_node_free(world, (SordNode*)tup[0]);
158 tup[0] = sord_new_blank(world, USTR("ablank"));
159 sord_add(sord, tup);
160
161 sord_node_free(world, (SordNode*)tup[1]);
162 sord_node_free(world, (SordNode*)tup[2]);
163 tup[1] = uri(world, 6);
164 tup[2] = uri(world, 7);
165 sord_add(sord, tup);
166 sord_node_free(world, (SordNode*)tup[0]);
167 sord_node_free(world, (SordNode*)tup[1]);
168 sord_node_free(world, (SordNode*)tup[2]);
169
170 return EXIT_SUCCESS;
171 }
172
173 #define TUP_FMT "(%6s %6s %6s)"
174 #define TUP_FMT_ARGS(t) \
175 ((t)[0] ? sord_node_get_string((t)[0]) : USTR("*")), \
176 ((t)[1] ? sord_node_get_string((t)[1]) : USTR("*")), \
177 ((t)[2] ? sord_node_get_string((t)[2]) : USTR("*"))
178
179 static int
180 test_read(SordWorld* world, SordModel* sord, SordNode* g,
181 const size_t n_quads)
182 {
183 int ret = EXIT_SUCCESS;
184
185 SordQuad id;
186
187 SordIter* iter = sord_begin(sord);
188 if (sord_iter_get_model(iter) != sord) {
189 return test_fail("Fail: Iterator has incorrect sord pointer\n");
190 }
191
192 for (; !sord_iter_end(iter); sord_iter_next(iter))
193 sord_iter_get(iter, id);
194
195 // Attempt to increment past end
196 if (!sord_iter_next(iter)) {
197 return test_fail("Fail: Successfully incremented past end\n");
198 }
199
200 sord_iter_free(iter);
201
202 const uint8_t* s = USTR("hello");
203 SordNode* plain_hello = sord_new_literal(world, 0, s, NULL);
204 SordNode* type4_hello = sord_new_literal(world, uri(world, 4), s, NULL);
205 SordNode* type5_hello = sord_new_literal(world, uri(world, 5), s, NULL);
206 SordNode* gb_hello = sord_new_literal(world, NULL, s, "en-gb");
207 SordNode* us_hello = sord_new_literal(world, NULL, s, "en-us");
208
209 #define NUM_PATTERNS 18
210
211 QueryTest patterns[NUM_PATTERNS] = {
212 { { 0, 0, 0 }, (int)(n_quads * n_objects_per) + 12 },
213 { { uri(world, 1), 0, 0 }, 2 },
214 { { uri(world, 9), uri(world, 9), uri(world, 9) }, 0 },
215 { { uri(world, 1), uri(world, 2), uri(world, 4) }, 1 },
216 { { uri(world, 3), uri(world, 4), uri(world, 0) }, 2 },
217 { { uri(world, 0), uri(world, 2), uri(world, 4) }, 1 },
218 { { uri(world, 0), uri(world, 0), uri(world, 4) }, 1 },
219 { { uri(world, 1), uri(world, 0), uri(world, 0) }, 2 },
220 { { uri(world, 1), uri(world, 0), uri(world, 4) }, 1 },
221 { { uri(world, 0), uri(world, 2), uri(world, 0) }, 2 },
222 { { uri(world, 98), uri(world, 4), plain_hello }, 1 },
223 { { uri(world, 98), uri(world, 4), type5_hello }, 1 },
224 { { uri(world, 96), uri(world, 4), type4_hello }, 1 },
225 { { uri(world, 96), uri(world, 4), type5_hello }, 1 },
226 { { uri(world, 94), uri(world, 5), plain_hello }, 1 },
227 { { uri(world, 94), uri(world, 5), gb_hello }, 1 },
228 { { uri(world, 92), uri(world, 6), gb_hello }, 1 },
229 { { uri(world, 92), uri(world, 6), us_hello }, 1 } };
230
231 SordQuad match = { uri(world, 1), uri(world, 2), uri(world, 4), g };
232 if (!sord_contains(sord, match)) {
233 return test_fail("Fail: No match for " TUP_FMT "\n",
234 TUP_FMT_ARGS(match));
235 }
236
237 SordQuad nomatch = { uri(world, 1), uri(world, 2), uri(world, 9), g };
238 if (sord_contains(sord, nomatch)) {
239 return test_fail("Fail: False match for " TUP_FMT "\n",
240 TUP_FMT_ARGS(nomatch));
241 }
242
243 if (sord_get(sord, NULL, NULL, uri(world, 3), g)) {
244 return test_fail("Fail: Get *,*,3 succeeded\n");
245 } else if (!sord_node_equals(
246 sord_get(sord, uri(world, 1), uri(world, 2), NULL, g),
247 uri(world, 3))) {
248 return test_fail("Fail: Get 1,2,* != 3\n");
249 } else if (!sord_node_equals(
250 sord_get(sord, uri(world, 1), NULL, uri(world, 3), g),
251 uri(world, 2))) {
252 return test_fail("Fail: Get 1,*,3 != 2\n");
253 } else if (!sord_node_equals(
254 sord_get(sord, NULL, uri(world, 2), uri(world, 3), g),
255 uri(world, 1))) {
256 return test_fail("Fail: Get *,2,3 != 1\n");
257 }
258
259 for (unsigned i = 0; i < NUM_PATTERNS; ++i) {
260 QueryTest test = patterns[i];
261 SordQuad pat = { test.query[0], test.query[1], test.query[2], g };
262 fprintf(stderr, "Query " TUP_FMT "... ", TUP_FMT_ARGS(pat));
263
264 iter = sord_find(sord, pat);
265 int num_results = 0;
266 for (; !sord_iter_end(iter); sord_iter_next(iter)) {
267 sord_iter_get(iter, id);
268 ++num_results;
269 if (!sord_quad_match(pat, id)) {
270 sord_iter_free(iter);
271 return test_fail(
272 "Fail: Query result " TUP_FMT " does not match pattern\n",
273 TUP_FMT_ARGS(id));
274 }
275 }
276 sord_iter_free(iter);
277 if (num_results != test.expected_num_results) {
278 return test_fail("Fail: Expected %d results, got %d\n",
279 test.expected_num_results, num_results);
280 }
281 fprintf(stderr, "OK (%u matches)\n", test.expected_num_results);
282 }
283
284 // Query blank node subject
285 SordQuad pat = { sord_new_blank(world, USTR("ablank")), 0, 0 };
286 if (!pat[0]) {
287 return test_fail("Blank node subject lost\n");
288 }
289 fprintf(stderr, "Query " TUP_FMT "... ", TUP_FMT_ARGS(pat));
290 iter = sord_find(sord, pat);
291 int num_results = 0;
292 for (; !sord_iter_end(iter); sord_iter_next(iter)) {
293 sord_iter_get(iter, id);
294 ++num_results;
295 if (!sord_quad_match(pat, id)) {
296 sord_iter_free(iter);
297 return test_fail(
298 "Fail: Query result " TUP_FMT " does not match pattern\n",
299 TUP_FMT_ARGS(id));
300 }
301 }
302 fprintf(stderr, "OK\n");
303 sord_node_free(world, (SordNode*)pat[0]);
304 sord_iter_free(iter);
305 if (num_results != 2) {
306 return test_fail("Blank node subject query failed\n");
307 }
308
309 // Test nested queries
310 fprintf(stderr, "Nested Queries... ");
311 const SordNode* last_subject = 0;
312 iter = sord_search(sord, NULL, NULL, NULL, NULL);
313 for (; !sord_iter_end(iter); sord_iter_next(iter)) {
314 sord_iter_get(iter, id);
315 if (id[0] == last_subject)
316 continue;
317
318 SordQuad subpat = { id[0], 0, 0 };
319 SordIter* subiter = sord_find(sord, subpat);
320 uint64_t num_sub_results = 0;
321 if (sord_iter_get_node(subiter, SORD_SUBJECT) != id[0]) {
322 return test_fail("Fail: Incorrect initial submatch\n");
323 }
324 for (; !sord_iter_end(subiter); sord_iter_next(subiter)) {
325 SordQuad subid;
326 sord_iter_get(subiter, subid);
327 if (!sord_quad_match(subpat, subid)) {
328 sord_iter_free(iter);
329 sord_iter_free(subiter);
330 return test_fail(
331 "Fail: Nested query result does not match pattern\n");
332 }
333 ++num_sub_results;
334 }
335 sord_iter_free(subiter);
336 if (num_sub_results != n_objects_per) {
337 return test_fail(
338 "Fail: Nested query " TUP_FMT " failed"
339 " (%d results, expected %d)\n",
340 TUP_FMT_ARGS(subpat), num_sub_results, n_objects_per);
341 }
342
343 uint64_t count = sord_count(sord, id[0], 0, 0, 0);
344 if (count != num_sub_results) {
345 return test_fail("Fail: Query " TUP_FMT " sord_count() %d"
346 "does not match result count %d\n",
347 TUP_FMT_ARGS(subpat), count, num_sub_results);
348 }
349
350 last_subject = id[0];
351 }
352 fprintf(stderr, "OK\n\n");
353 sord_iter_free(iter);
354
355 return ret;
356 }
357
358 static SerdStatus
359 unexpected_error(void* handle, const SerdError* error)
360 {
361 fprintf(stderr, "unexpected error: ");
362 vfprintf(stderr, error->fmt, *error->args);
363 return SERD_SUCCESS;
364 }
365
366 static SerdStatus
367 expected_error(void* handle, const SerdError* error)
368 {
369 fprintf(stderr, "expected error: ");
370 vfprintf(stderr, error->fmt, *error->args);
371 ++n_expected_errors;
372 return SERD_SUCCESS;
373 }
374
375 static int
376 finished(SordWorld* world, SordModel* sord, int status)
377 {
378 sord_free(sord);
379 sord_world_free(world);
380 return status;
381 }
382
383 int
384 main(int argc, char** argv)
385 {
386 static const size_t n_quads = 300;
387
388 sord_free(NULL); // Shouldn't crash
389
390 SordWorld* world = sord_world_new();
391
392
393 // Attempt to create invalid URI
394 fprintf(stderr, "expected ");
395 SordNode* bad_uri = sord_new_uri(world, USTR("noscheme"));
396 if (bad_uri) {
397 return test_fail("Successfully created invalid URI \"noscheme\"\n");
398 }
399 sord_node_free(world, bad_uri);
400
401 sord_world_set_error_sink(world, expected_error, NULL);
402
403 // Attempt to create invalid CURIE
404 SerdNode base = serd_node_from_string(SERD_URI, USTR("http://example.org/"));
405 SerdEnv* env = serd_env_new(&base);
406 SerdNode sbadns = serd_node_from_string(SERD_CURIE, USTR("badns:"));
407 SordNode* badns = sord_node_from_serd_node(world, env, &sbadns, NULL, NULL);
408 if (badns) {
409 return test_fail("Successfully created CURIE with bad namespace\n");
410 }
411 sord_node_free(world, badns);
412 serd_env_free(env);
413
414 // Attempt to create node from garbage
415 SerdNode junk = SERD_NODE_NULL;
416 junk.type = (SerdType)1234;
417 if (sord_node_from_serd_node(world, env, &junk, NULL, NULL)) {
418 return test_fail("Successfully created node from garbage serd node\n");
419 }
420
421 // Attempt to create NULL node
422 SordNode* nil_node = sord_node_from_serd_node(
423 world, NULL, &SERD_NODE_NULL, NULL, NULL);
424 if (nil_node) {
425 return test_fail("Successfully created NULL node\n");
426 }
427 sord_node_free(world, nil_node);
428
429 // Attempt to double-free a node
430 SordNode* garbage = sord_new_uri(world, USTR("urn:garbage"));
431 sord_node_free(world, garbage);
432 sord_world_set_error_sink(world, expected_error, NULL);
433 sord_node_free(world, garbage);
434 sord_world_set_error_sink(world, unexpected_error, NULL);
435 if (n_expected_errors != 2) {
436 return test_fail("Successfully freed node twice\n");
437 }
438
439 sord_world_set_error_sink(world, unexpected_error, NULL);
440
441 // Check node flags are set properly
442 SordNode* with_newline = sord_new_literal(world, NULL, USTR("a\nb"), NULL);
443 if (!(sord_node_get_flags(with_newline) & SERD_HAS_NEWLINE)) {
444 return test_fail("Newline flag not set\n");
445 }
446 SordNode* with_quote = sord_new_literal(world, NULL, USTR("a\"b"), NULL);
447 if (!(sord_node_get_flags(with_quote) & SERD_HAS_QUOTE)) {
448 return test_fail("Quote flag not set\n");
449 }
450
451 // Create with minimal indexing
452 SordModel* sord = sord_new(world, SORD_SPO, false);
453 generate(world, sord, n_quads, NULL);
454
455 if (test_read(world, sord, NULL, n_quads)) {
456 sord_free(sord);
457 sord_world_free(world);
458 return EXIT_FAILURE;
459 }
460
461 // Check adding tuples with NULL fields fails
462 sord_world_set_error_sink(world, expected_error, NULL);
463 const size_t initial_num_quads = sord_num_quads(sord);
464 SordQuad tup = { 0, 0, 0, 0};
465 if (sord_add(sord, tup)) {
466 return test_fail("Added NULL tuple\n");
467 }
468 tup[0] = uri(world, 1);
469 if (sord_add(sord, tup)) {
470 return test_fail("Added tuple with NULL P and O\n");
471 }
472 tup[1] = uri(world, 2);
473 if (sord_add(sord, tup)) {
474 return test_fail("Added tuple with NULL O\n");
475 }
476
477 if (sord_num_quads(sord) != initial_num_quads) {
478 return test_fail("Num quads %zu != %zu\n",
479 sord_num_quads(sord), initial_num_quads);
480 }
481
482 // Check adding tuples with an active iterator fails
483 SordIter* iter = sord_begin(sord);
484 tup[2] = uri(world, 3);
485 if (sord_add(sord, tup)) {
486 return test_fail("Added tuple with active iterator\n");
487 }
488
489 // Check removing tuples with several active iterator fails
490 SordIter* iter2 = sord_begin(sord);
491 if (!sord_erase(sord, iter)) {
492 return test_fail("Erased tuple with several active iterators\n");
493 }
494 n_expected_errors = 0;
495 sord_remove(sord, tup);
496 if (n_expected_errors != 1) {
497 return test_fail("Removed tuple with several active iterators\n");
498 }
499 sord_iter_free(iter);
500 sord_iter_free(iter2);
501
502 sord_world_set_error_sink(world, unexpected_error, NULL);
503
504 // Check interning merges equivalent values
505 SordNode* uri_id = sord_new_uri(world, USTR("http://example.org"));
506 SordNode* blank_id = sord_new_blank(world, USTR("testblank"));
507 SordNode* lit_id = sord_new_literal(world, uri_id, USTR("hello"), NULL);
508 if (sord_node_get_type(uri_id) != SORD_URI) {
509 return test_fail("URI node has incorrect type\n");
510 } else if (sord_node_get_type(blank_id) != SORD_BLANK) {
511 return test_fail("Blank node has incorrect type\n");
512 } else if (sord_node_get_type(lit_id) != SORD_LITERAL) {
513 return test_fail("Literal node has incorrect type\n");
514 }
515
516 const size_t initial_num_nodes = sord_num_nodes(world);
517
518 SordNode* uri_id2 = sord_new_uri(world, USTR("http://example.org"));
519 SordNode* blank_id2 = sord_new_blank(world, USTR("testblank"));
520 SordNode* lit_id2 = sord_new_literal(world, uri_id, USTR("hello"), NULL);
521 if (uri_id2 != uri_id || !sord_node_equals(uri_id2, uri_id)) {
522 fprintf(stderr, "Fail: URI interning failed (duplicates)\n");
523 return finished(world, sord, EXIT_FAILURE);
524 } else if (blank_id2 != blank_id
525 || !sord_node_equals(blank_id2, blank_id)) {
526 fprintf(stderr, "Fail: Blank node interning failed (duplicates)\n");
527 return finished(world, sord, EXIT_FAILURE);
528 } else if (lit_id2 != lit_id || !sord_node_equals(lit_id2, lit_id)) {
529 fprintf(stderr, "Fail: Literal interning failed (duplicates)\n");
530 return finished(world, sord, EXIT_FAILURE);
531 }
532
533 if (sord_num_nodes(world) != initial_num_nodes) {
534 return test_fail("Num nodes %zu != %zu\n",
535 sord_num_nodes(world), initial_num_nodes);
536 }
537
538 const uint8_t ni_hao[] = { 0xE4, 0xBD, 0xA0, 0xE5, 0xA5, 0xBD };
539 SordNode* chello = sord_new_literal(world, NULL, ni_hao, "cmn");
540
541 // Test literal length
542 size_t n_bytes;
543 size_t n_chars;
544 const uint8_t* str = sord_node_get_string_counted(lit_id2, &n_bytes);
545 if (strcmp((const char*)str, "hello")) {
546 return test_fail("Literal node corrupt\n");
547 } else if (n_bytes != strlen("hello")) {
548 return test_fail("ASCII literal byte count incorrect\n");
549 }
550
551 str = sord_node_get_string_measured(lit_id2, &n_bytes, &n_chars);
552 if (n_bytes != strlen("hello") || n_chars != strlen("hello")) {
553 return test_fail("ASCII literal measured length incorrect\n");
554 }
555
556 str = sord_node_get_string_measured(chello, &n_bytes, &n_chars);
557 if (n_bytes != 6) {
558 return test_fail("Multi-byte literal byte count incorrect\n");
559 } else if (n_chars != 2) {
560 return test_fail("Multi-byte literal character count incorrect\n");
561 }
562
563 // Check interning doesn't clash non-equivalent values
564 SordNode* uri_id3 = sord_new_uri(world, USTR("http://example.orgX"));
565 SordNode* blank_id3 = sord_new_blank(world, USTR("testblankX"));
566 SordNode* lit_id3 = sord_new_literal(world, uri_id, USTR("helloX"), NULL);
567 if (uri_id3 == uri_id || sord_node_equals(uri_id3, uri_id)) {
568 fprintf(stderr, "Fail: URI interning failed (clash)\n");
569 return finished(world, sord, EXIT_FAILURE);
570 } else if (blank_id3 == blank_id || sord_node_equals(blank_id3, blank_id)) {
571 fprintf(stderr, "Fail: Blank node interning failed (clash)\n");
572 return finished(world, sord, EXIT_FAILURE);
573 } else if (lit_id3 == lit_id || sord_node_equals(lit_id3, lit_id)) {
574 fprintf(stderr, "Fail: Literal interning failed (clash)\n");
575 return finished(world, sord, EXIT_FAILURE);
576 }
577
578 // Check literal interning
579 SordNode* lit4 = sord_new_literal(world, NULL, USTR("hello"), NULL);
580 SordNode* lit5 = sord_new_literal(world, uri_id2, USTR("hello"), NULL);
581 SordNode* lit6 = sord_new_literal(world, NULL, USTR("hello"), "en-ca");
582 if (lit4 == lit5 || sord_node_equals(lit4, lit5)
583 || lit4 == lit6 || sord_node_equals(lit4, lit6)
584 || lit5 == lit6 || sord_node_equals(lit5, lit6)) {
585 fprintf(stderr, "Fail: Literal interning failed (type/lang clash)\n");
586 return finished(world, sord, EXIT_FAILURE);
587 }
588
589 // Check relative URI construction
590 SordNode* reluri = sord_new_relative_uri(
591 world, USTR("a/b"), USTR("http://example.org/"));
592 if (strcmp((const char*)sord_node_get_string(reluri),
593 "http://example.org/a/b")) {
594 fprintf(stderr, "Fail: Bad relative URI constructed: <%s>\n",
595 sord_node_get_string(reluri));
596 return finished(world, sord, EXIT_FAILURE);
597 }
598 SordNode* reluri2 = sord_new_relative_uri(
599 world, USTR("http://drobilla.net/"), USTR("http://example.org/"));
600 if (strcmp((const char*)sord_node_get_string(reluri2),
601 "http://drobilla.net/")) {
602 fprintf(stderr, "Fail: Bad relative URI constructed: <%s>\n",
603 sord_node_get_string(reluri));
604 return finished(world, sord, EXIT_FAILURE);
605 }
606
607 // Check comparison with NULL
608 sord_node_free(world, uri_id);
609 sord_node_free(world, blank_id);
610 sord_node_free(world, lit_id);
611 sord_node_free(world, uri_id2);
612 sord_node_free(world, blank_id2);
613 sord_node_free(world, lit_id2);
614 sord_node_free(world, uri_id3);
615 sord_node_free(world, blank_id3);
616 sord_node_free(world, lit_id3);
617 sord_free(sord);
618
619 static const char* const index_names[6] = {
620 "spo", "sop", "ops", "osp", "pso", "pos"
621 };
622
623 for (int i = 0; i < 6; ++i) {
624 sord = sord_new(world, (1 << i), false);
625 printf("Testing Index `%s'\n", index_names[i]);
626 generate(world, sord, n_quads, 0);
627 if (test_read(world, sord, 0, n_quads))
628 return finished(world, sord, EXIT_FAILURE);
629 sord_free(sord);
630 }
631
632 static const char* const graph_index_names[6] = {
633 "gspo", "gsop", "gops", "gosp", "gpso", "gpos"
634 };
635
636 for (int i = 0; i < 6; ++i) {
637 sord = sord_new(world, (1 << i), true);
638 printf("Testing Index `%s'\n", graph_index_names[i]);
639 SordNode* graph = uri(world, 42);
640 generate(world, sord, n_quads, graph);
641 if (test_read(world, sord, graph, n_quads))
642 return finished(world, sord, EXIT_FAILURE);
643 sord_free(sord);
644 }
645
646 // Test removing
647 sord = sord_new(world, SORD_SPO, true);
648 tup[0] = uri(world, 1);
649 tup[1] = uri(world, 2);
650 tup[2] = sord_new_literal(world, 0, USTR("hello"), NULL);
651 tup[3] = 0;
652 sord_add(sord, tup);
653 if (!sord_ask(sord, tup[0], tup[1], tup[2], tup[3])) {
654 fprintf(stderr, "Failed to add tuple\n");
655 return finished(world, sord, EXIT_FAILURE);
656 }
657 sord_node_free(world, (SordNode*)tup[2]);
658 tup[2] = sord_new_literal(world, 0, USTR("hi"), NULL);
659 sord_add(sord, tup);
660 sord_remove(sord, tup);
661 if (sord_num_quads(sord) != 1) {
662 fprintf(stderr, "Remove failed (%zu quads, expected 1)\n",
663 sord_num_quads(sord));
664 return finished(world, sord, EXIT_FAILURE);
665 }
666
667 iter = sord_find(sord, tup);
668 if (!sord_iter_end(iter)) {
669 fprintf(stderr, "Found removed tuple\n");
670 return finished(world, sord, EXIT_FAILURE);
671 }
672 sord_iter_free(iter);
673
674 // Test double remove (silent success)
675 sord_remove(sord, tup);
676
677 // Load a couple graphs
678 SordNode* graph42 = uri(world, 42);
679 SordNode* graph43 = uri(world, 43);
680 generate(world, sord, 1, graph42);
681 generate(world, sord, 1, graph43);
682
683 // Remove one graph via iterator
684 SerdStatus st;
685 iter = sord_search(sord, NULL, NULL, NULL, graph43);
686 while (!sord_iter_end(iter)) {
687 if ((st = sord_erase(sord, iter))) {
688 fprintf(stderr, "Remove by iterator failed (%s)\n",
689 serd_strerror(st));
690 return finished(world, sord, EXIT_FAILURE);
691 }
692 }
693 sord_iter_free(iter);
694
695 // Erase the first tuple (an element in the default graph)
696 iter = sord_begin(sord);
697 if (sord_erase(sord, iter)) {
698 return test_fail("Failed to erase begin iterator on non-empty model\n");
699 }
700 sord_iter_free(iter);
701
702 // Ensure only the other graph is left
703 SordQuad quad;
704 SordQuad pat = { 0, 0, 0, graph42 };
705 for (iter = sord_begin(sord); !sord_iter_end(iter); sord_iter_next(iter)) {
706 sord_iter_get(iter, quad);
707 if (!sord_quad_match(quad, pat)) {
708 fprintf(stderr, "Graph removal via iteration failed\n");
709 return finished(world, sord, EXIT_FAILURE);
710 }
711 }
712 sord_iter_free(iter);
713
714 // Load file into two separate graphs
715 sord_free(sord);
716 sord = sord_new(world, SORD_SPO, true);
717 env = serd_env_new(&base);
718 SordNode* graph1 = sord_new_uri(world, USTR("http://example.org/graph1"));
719 SordNode* graph2 = sord_new_uri(world, USTR("http://example.org/graph2"));
720 SerdReader* reader = sord_new_reader(sord, env, SERD_TURTLE, graph1);
721 if ((st = serd_reader_read_string(reader, USTR("<s> <p> <o> .")))) {
722 fprintf(stderr, "Failed to read string (%s)\n", serd_strerror(st));
723 return finished(world, sord, EXIT_FAILURE);
724 }
725 serd_reader_free(reader);
726 reader = sord_new_reader(sord, env, SERD_TURTLE, graph2);
727 if ((st = serd_reader_read_string(reader, USTR("<s> <p> <o> .")))) {
728 fprintf(stderr, "Failed to re-read string (%s)\n", serd_strerror(st));
729 return finished(world, sord, EXIT_FAILURE);
730 }
731 serd_reader_free(reader);
732 serd_env_free(env);
733
734 // Ensure we only see triple once
735 size_t n_triples = 0;
736 for (iter = sord_begin(sord); !sord_iter_end(iter); sord_iter_next(iter)) {
737 fprintf(stderr, "%s %s %s %s\n",
738 sord_node_get_string(sord_iter_get_node(iter, SORD_SUBJECT)),
739 sord_node_get_string(sord_iter_get_node(iter, SORD_PREDICATE)),
740 sord_node_get_string(sord_iter_get_node(iter, SORD_OBJECT)),
741 sord_node_get_string(sord_iter_get_node(iter, SORD_GRAPH)));
742
743 ++n_triples;
744 }
745 sord_iter_free(iter);
746 if (n_triples != 1) {
747 fprintf(stderr, "Found duplicate triple\n");
748 return finished(world, sord, EXIT_FAILURE);
749 }
750
751 // Test SPO iteration on an SOP indexed store
752 sord_free(sord);
753 sord = sord_new(world, SORD_SOP, false);
754 generate(world, sord, 1, graph42);
755 for (iter = sord_begin(sord); !sord_iter_end(iter); sord_iter_next(iter)) {
756 ++n_triples;
757 }
758 sord_iter_free(iter);
759
760 return finished(world, sord, EXIT_SUCCESS);
761 }