import sys import unittest from io import BytesIO, StringIO, UnsupportedOperation from tempfile import NamedTemporaryFile, TemporaryFile from pyoxigraph import ( Literal, NamedNode, Quad, QueryBoolean, QuerySolutions, parse, parse_query_results, serialize, ) EXAMPLE_TRIPLE = Quad( NamedNode("http://example.com/foo"), NamedNode("http://example.com/p"), Literal("éù"), ) EXAMPLE_QUAD = Quad( NamedNode("http://example.com/foo"), NamedNode("http://example.com/p"), Literal("1"), NamedNode("http://example.com/g"), ) class TestParse(unittest.TestCase): def test_parse_file(self) -> None: with NamedTemporaryFile(suffix=".ttl") as fp: fp.write('

"éù" .'.encode()) fp.flush() self.assertEqual( list(parse(path=fp.name, base_iri="http://example.com/")), [EXAMPLE_TRIPLE], ) def test_parse_not_existing_file(self) -> None: with self.assertRaises(IOError) as _: parse(path="/tmp/not-existing-oxigraph-file.ttl", format="text/turtle") def test_parse_str(self) -> None: self.assertEqual( list( parse( '

"éù" .', "text/turtle", base_iri="http://example.com/", ) ), [EXAMPLE_TRIPLE], ) def test_parse_bytes(self) -> None: self.assertEqual( list( parse( '

"éù" .'.encode(), "text/turtle", base_iri="http://example.com/", ) ), [EXAMPLE_TRIPLE], ) def test_parse_str_io(self) -> None: self.assertEqual( list( parse( StringIO('

"éù" .'), "text/turtle", base_iri="http://example.com/", ) ), [EXAMPLE_TRIPLE], ) def test_parse_long_str_io(self) -> None: self.assertEqual( list( parse( StringIO('

"éù" .\n' * 1024), "text/turtle", base_iri="http://example.com/", ) ), [EXAMPLE_TRIPLE] * 1024, ) def test_parse_bytes_io(self) -> None: self.assertEqual( list( parse( BytesIO('

"éù" .'.encode()), "text/turtle", base_iri="http://example.com/", ) ), [EXAMPLE_TRIPLE], ) def test_parse_io_error(self) -> None: with self.assertRaises(UnsupportedOperation) as _, TemporaryFile("wb") as fp: list(parse(fp, "nt")) def test_parse_quad(self) -> None: self.assertEqual( list( parse( ' {

"1" }', "application/trig", base_iri="http://example.com/", ) ), [EXAMPLE_QUAD], ) def test_parse_syntax_error(self) -> None: with NamedTemporaryFile() as fp: fp.write(b"@base .\n") fp.write(b' "p" "1"') fp.flush() with self.assertRaises(SyntaxError) as ctx: list(parse(path=fp.name, format="text/turtle")) self.assertEqual(ctx.exception.filename, fp.name) self.assertEqual(ctx.exception.lineno, 2) self.assertEqual(ctx.exception.offset, 7) if sys.version_info >= (3, 10): self.assertEqual(ctx.exception.end_lineno, 2) self.assertEqual(ctx.exception.end_offset, 10) def test_parse_without_named_graphs(self) -> None: with self.assertRaises(SyntaxError) as _: list( parse( ' {

"1" }', "application/trig", base_iri="http://example.com/", without_named_graphs=True, ) ) def test_parse_rename_blank_nodes(self) -> None: self.assertNotEqual( list( parse( '_:s "o" .', "application/n-triples", rename_blank_nodes=True, ) ), list( parse( '_:s "o" .', "application/n-triples", rename_blank_nodes=True, ) ), ) class TestSerialize(unittest.TestCase): def test_serialize_to_bytes(self) -> None: self.assertEqual( (serialize([EXAMPLE_TRIPLE.triple], None, "text/turtle") or b"").decode(), ' "éù" .\n', ) def test_serialize_to_bytes_io(self) -> None: output = BytesIO() serialize([EXAMPLE_TRIPLE.triple], output, "text/turtle") self.assertEqual( output.getvalue().decode(), ' "éù" .\n', ) def test_serialize_to_file(self) -> None: with NamedTemporaryFile(suffix=".ttl") as fp: serialize([EXAMPLE_TRIPLE], fp.name) self.assertEqual( fp.read().decode(), ' "éù" .\n', ) def test_serialize_io_error(self) -> None: with self.assertRaises(UnsupportedOperation) as _, TemporaryFile("rb") as fp: serialize([EXAMPLE_TRIPLE], fp, "text/turtle") def test_serialize_quad(self) -> None: output = BytesIO() serialize([EXAMPLE_QUAD], output, "application/trig") self.assertEqual( output.getvalue(), b' {\n\t "1" .\n}\n', ) class TestParseQuerySolutions(unittest.TestCase): def test_parse_file(self) -> None: with NamedTemporaryFile(suffix=".tsv") as fp: fp.write(b'?s\t?p\t?o\n\t\t"1"\n') fp.flush() r = parse_query_results(path=fp.name) self.assertIsInstance(r, QuerySolutions) results = list(r) # type: ignore[arg-type] self.assertEqual(results[0]["s"], NamedNode("http://example.com/s")) self.assertEqual(results[0][2], Literal("1")) def test_parse_not_existing_file(self) -> None: with self.assertRaises(IOError) as _: parse_query_results(path="/tmp/not-existing-oxigraph-file.ttl", format="application/json") def test_parse_str(self) -> None: result = parse_query_results("true", "tsv") self.assertIsInstance(result, QueryBoolean) self.assertTrue(result) def test_parse_bytes(self) -> None: result = parse_query_results(b"false", "tsv") self.assertIsInstance(result, QueryBoolean) self.assertFalse(result) def test_parse_str_io(self) -> None: result = parse_query_results("true", "tsv") self.assertIsInstance(result, QueryBoolean) self.assertTrue(result) def test_parse_bytes_io(self) -> None: result = parse_query_results(BytesIO(b"false"), "tsv") self.assertIsInstance(result, QueryBoolean) self.assertFalse(result) def test_parse_io_error(self) -> None: with self.assertRaises(UnsupportedOperation) as _, TemporaryFile("wb") as fp: parse_query_results(fp, "srx") def test_parse_syntax_error_json(self) -> None: with NamedTemporaryFile() as fp: fp.write(b"{]") fp.flush() with self.assertRaises(SyntaxError) as ctx: list(parse_query_results(path=fp.name, format="srj")) # type: ignore[arg-type] self.assertEqual(ctx.exception.filename, fp.name) self.assertEqual(ctx.exception.lineno, 1) self.assertEqual(ctx.exception.offset, 2) if sys.version_info >= (3, 10): self.assertEqual(ctx.exception.end_lineno, 1) self.assertEqual(ctx.exception.end_offset, 3) def test_parse_syntax_error_tsv(self) -> None: with NamedTemporaryFile() as fp: fp.write(b"?a\t?test\n") fp.write(b"1\t\n") fp.flush() with self.assertRaises(SyntaxError) as ctx: list(parse_query_results(path=fp.name, format="tsv")) # type: ignore[arg-type] self.assertEqual(ctx.exception.filename, fp.name) self.assertEqual(ctx.exception.lineno, 2) self.assertEqual(ctx.exception.offset, 3) if sys.version_info >= (3, 10): self.assertEqual(ctx.exception.end_lineno, 2) self.assertEqual(ctx.exception.end_offset, 9)