// Copyright 2011-2023 David Robillard // SPDX-License-Identifier: ISC #undef NDEBUG #include "serd/buffer.h" #include "serd/env.h" #include "serd/event.h" #include "serd/memory.h" #include "serd/node.h" #include "serd/sink.h" #include "serd/statement.h" #include "serd/status.h" #include "serd/string_view.h" #include "serd/syntax.h" #include "serd/world.h" #include "serd/writer.h" #include #include #include #include static void test_write_bad_event(void) { SerdWorld* world = serd_world_new(); SerdEnv* env = serd_env_new(serd_empty_string()); SerdBuffer buffer = {NULL, 0}; SerdWriter* writer = serd_writer_new(world, SERD_TURTLE, 0U, env, serd_buffer_sink, &buffer); assert(writer); const SerdEvent event = {(SerdEventType)42}; assert(serd_sink_write_event(serd_writer_sink(writer), &event) == SERD_BAD_ARG); char* const out = serd_buffer_sink_finish(&buffer); assert(!strcmp(out, "")); serd_free(out); serd_writer_free(writer); serd_env_free(env); serd_world_free(world); } static void test_write_bad_prefix(void) { SerdWorld* world = serd_world_new(); SerdEnv* env = serd_env_new(serd_empty_string()); SerdBuffer buffer = {NULL, 0}; SerdWriter* writer = serd_writer_new(world, SERD_TURTLE, 0U, env, serd_buffer_sink, &buffer); assert(writer); SerdNode* name = serd_new_string(serd_string("eg")); SerdNode* uri = serd_new_uri(serd_string("rel")); assert(serd_sink_write_prefix(serd_writer_sink(writer), name, uri) == SERD_BAD_ARG); char* const out = serd_buffer_sink_finish(&buffer); assert(!strcmp(out, "")); serd_free(out); serd_node_free(uri); serd_node_free(name); serd_writer_free(writer); serd_env_free(env); serd_world_free(world); } static void test_write_long_literal(void) { SerdWorld* world = serd_world_new(); SerdEnv* env = serd_env_new(serd_empty_string()); SerdBuffer buffer = {NULL, 0}; SerdWriter* writer = serd_writer_new(world, SERD_TURTLE, 0U, env, serd_buffer_sink, &buffer); assert(writer); SerdNode* s = serd_new_uri(serd_string("http://example.org/s")); SerdNode* p = serd_new_uri(serd_string("http://example.org/p")); SerdNode* o = serd_new_string(serd_string("hello \"\"\"world\"\"\"!")); assert(!serd_sink_write(serd_writer_sink(writer), 0, s, p, o, NULL)); serd_node_free(o); serd_node_free(p); serd_node_free(s); serd_writer_free(writer); serd_env_free(env); char* out = serd_buffer_sink_finish(&buffer); static const char* const expected = "\n" "\t \"\"\"hello \"\"\\\"world\"\"\\\"!\"\"\" .\n"; assert(!strcmp((char*)out, expected)); serd_free(out); serd_world_free(world); } static size_t null_sink(const void* const buf, const size_t size, const size_t nmemb, void* const stream) { (void)buf; (void)stream; return size * nmemb; } static void test_writer_cleanup(void) { SerdStatus st = SERD_SUCCESS; SerdWorld* world = serd_world_new(); SerdEnv* env = serd_env_new(serd_empty_string()); SerdWriter* writer = serd_writer_new(world, SERD_TURTLE, 0U, env, null_sink, NULL); const SerdSink* sink = serd_writer_sink(writer); SerdNode* s = serd_new_uri(serd_string("http://example.org/s")); SerdNode* p = serd_new_uri(serd_string("http://example.org/p")); SerdNode* o = serd_new_blank(serd_string("start")); st = serd_sink_write(sink, SERD_ANON_O, s, p, o, NULL); assert(!st); // Write the start of several nested anonymous objects for (unsigned i = 0U; !st && i < 8U; ++i) { char buf[12] = {0}; snprintf(buf, sizeof(buf), "b%u", i); SerdNode* next_o = serd_new_blank(serd_string(buf)); st = serd_sink_write(sink, SERD_ANON_O, o, p, next_o, NULL); serd_node_free(o); o = next_o; } // Finish writing without terminating nodes assert(!(st = serd_writer_finish(writer))); // Set the base to an empty URI SerdNode* empty_uri = serd_new_uri(serd_string("")); assert(!(st = serd_sink_write_base(sink, empty_uri))); serd_node_free(empty_uri); // Free (which could leak if the writer doesn't clean up the stack properly) serd_node_free(o); serd_node_free(p); serd_node_free(s); serd_writer_free(writer); serd_env_free(env); serd_world_free(world); } static void test_strict_write(void) { const char* const path = "serd_strict_write_test.ttl"; FILE* const fd = fopen(path, "wb"); assert(fd); SerdWorld* world = serd_world_new(); SerdEnv* const env = serd_env_new(serd_empty_string()); SerdWriter* const writer = serd_writer_new( world, SERD_TURTLE, (SerdWriterFlags)SERD_WRITE_STRICT, env, null_sink, fd); assert(writer); const SerdSink* const sink = serd_writer_sink(writer); const uint8_t bad_str[] = {0xFF, 0x90, 'h', 'i', 0}; SerdNode* s = serd_new_uri(serd_string("http://example.org/s")); SerdNode* p = serd_new_uri(serd_string("http://example.org/p")); SerdNode* bad_lit = serd_new_string(serd_string((const char*)bad_str)); SerdNode* bad_uri = serd_new_uri(serd_string((const char*)bad_str)); assert(serd_sink_write(sink, 0, s, p, bad_lit, NULL) == SERD_BAD_TEXT); assert(serd_sink_write(sink, 0, s, p, bad_uri, NULL) == SERD_BAD_TEXT); serd_node_free(bad_uri); serd_node_free(bad_lit); serd_node_free(p); serd_node_free(s); serd_writer_free(writer); serd_env_free(env); serd_world_free(world); fclose(fd); remove(path); } // Produce a write error without setting errno static size_t error_sink(const void* const buf, const size_t size, const size_t len, void* const stream) { (void)buf; (void)size; (void)len; (void)stream; return 0U; } static void test_write_error(void) { SerdWorld* const world = serd_world_new(); SerdEnv* const env = serd_env_new(serd_empty_string()); SerdWriter* writer = NULL; SerdStatus st = SERD_SUCCESS; SerdNode* u = serd_new_uri(serd_string("http://example.com/u")); writer = serd_writer_new( world, SERD_TURTLE, (SerdWriterFlags)0, env, error_sink, NULL); assert(writer); const SerdSink* const sink = serd_writer_sink(writer); st = serd_sink_write(sink, 0U, u, u, u, NULL); assert(st == SERD_BAD_WRITE); serd_writer_free(writer); serd_node_free(u); serd_env_free(env); serd_world_free(world); } int main(void) { test_write_bad_event(); test_write_bad_prefix(); test_write_long_literal(); test_writer_cleanup(); test_strict_write(); test_write_error(); return 0; }