test_lmdb_store.py 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. import pytest
  2. from shutil import rmtree
  3. from rdflib import Namespace, URIRef
  4. from lakesuperior.store_layouts.ldp_rs.lmdb_store import LmdbStore, TxnManager
  5. @pytest.fixture(scope='class')
  6. def store():
  7. store = LmdbStore('/tmp/test_lmdbstore')
  8. yield store
  9. store.close()
  10. rmtree('/tmp/test_lmdbstore')
  11. @pytest.mark.usefixtures('store')
  12. class TestStoreInit:
  13. '''
  14. Tests for intializing and shutting down store and transactions.
  15. '''
  16. def test_open_close(self, store):
  17. '''
  18. Test opening and closing a store.
  19. '''
  20. tmpstore = LmdbStore('/tmp/test_lmdbstore_init')
  21. assert tmpstore.is_open
  22. tmpstore.close()
  23. assert not tmpstore.is_open
  24. def test_txn(self, store):
  25. '''
  26. Test opening and closing the main transaction.
  27. '''
  28. store.begin(True)
  29. assert store.is_txn_open
  30. store.commit()
  31. assert not store.is_txn_open
  32. store.begin(True)
  33. store.rollback()
  34. assert not store.is_txn_open
  35. def test_ctx_mgr(self, store):
  36. '''
  37. Test enclosing a transaction in a context.
  38. '''
  39. with TxnManager(store) as txn:
  40. pass
  41. assert not store.is_txn_open
  42. with TxnManager(store) as txn:
  43. raise RuntimeError
  44. assert not store.is_txn_open
  45. def test_rollback(self, store):
  46. '''
  47. Test rolling back a transaction.
  48. '''
  49. with TxnManager(store, True) as txn:
  50. store.add((
  51. URIRef('urn:nogo:s'), URIRef('urn:nogo:p'),
  52. URIRef('urn:nogo:o')))
  53. raise RuntimeError # This should roll back the transaction.
  54. with TxnManager(store) as txn:
  55. res = set(store.triples((None, None, None)))
  56. assert len(res) == 0
  57. @pytest.mark.usefixtures('store')
  58. class TestBasicOps:
  59. '''
  60. High-level tests for basic operations.
  61. '''
  62. def test_create_triple(self, store):
  63. '''
  64. Test creation of a single triple.
  65. '''
  66. trp = (
  67. URIRef('urn:test:s'), URIRef('urn:test:p'), URIRef('urn:test:o'))
  68. with TxnManager(store, True) as txn:
  69. store.add(trp)
  70. #import pdb; pdb.set_trace()
  71. res1 = set(store.triples((None, None, None)))
  72. res2 = set(store.triples(trp))
  73. assert len(res1) == 1
  74. assert len(res2) == 1
  75. assert trp in res1 & res2
  76. def test_triple_match_1bound(self, store):
  77. '''
  78. Test triple patterns matching one bound term.
  79. '''
  80. with TxnManager(store) as txn:
  81. res1 = set(store.triples((URIRef('urn:test:s'), None, None)))
  82. res2 = set(store.triples((None, URIRef('urn:test:p'), None)))
  83. res3 = set(store.triples((None, None, URIRef('urn:test:o'))))
  84. assert res1 == {(
  85. URIRef('urn:test:s'), URIRef('urn:test:p'),
  86. URIRef('urn:test:o'))}
  87. assert res2 == res1
  88. assert res3 == res2
  89. def test_triple_match_2bound(self, store):
  90. '''
  91. Test triple patterns matching two bound terms.
  92. '''
  93. with TxnManager(store) as txn:
  94. res1 = set(store.triples(
  95. (URIRef('urn:test:s'), URIRef('urn:test:p'), None)))
  96. res2 = set(store.triples(
  97. (URIRef('urn:test:s'), None, URIRef('urn:test:o'))))
  98. res3 = set(store.triples(
  99. (None, URIRef('urn:test:p'), URIRef('urn:test:o'))))
  100. assert res1 == {(
  101. URIRef('urn:test:s'), URIRef('urn:test:p'), URIRef('urn:test:o'))}
  102. assert res2 == res1
  103. assert res3 == res2
  104. def test_triple_no_match(self, store):
  105. '''
  106. Test various mismatches.
  107. '''
  108. with TxnManager(store, True) as txn:
  109. store.add((
  110. URIRef('urn:test:s'),
  111. URIRef('urn:test:p2'), URIRef('urn:test:o2')))
  112. store.add((
  113. URIRef('urn:test:s3'),
  114. URIRef('urn:test:p3'), URIRef('urn:test:o3')))
  115. res1 = set(store.triples((None, None, None)))
  116. assert len(res1) == 3
  117. res1 = set(store.triples(
  118. (URIRef('urn:test:s2'), URIRef('urn:test:p'), None)))
  119. res2 = set(store.triples(
  120. (URIRef('urn:test:s3'), None, URIRef('urn:test:o'))))
  121. res3 = set(store.triples(
  122. (None, URIRef('urn:test:p3'), URIRef('urn:test:o2'))))
  123. assert len(res1) == len(res2) == len(res3) == 0
  124. def test_remove(self, store):
  125. '''
  126. Test removing one or more triples.
  127. '''
  128. with TxnManager(store, True) as txn:
  129. store.remove((URIRef('urn:test:s3'),
  130. URIRef('urn:test:p3'), URIRef('urn:test:o3')))
  131. res1 = set(store.triples((None, None, None)))
  132. assert len(res1) == 2
  133. store.remove((URIRef('urn:test:s'), None, None))
  134. res2 = set(store.triples((None, None, None)))
  135. assert len(res2) == 0
  136. @pytest.mark.usefixtures('store')
  137. class TestBindings:
  138. '''
  139. Tests for namespace bindings.
  140. '''
  141. @pytest.fixture
  142. def bindings(self):
  143. return (
  144. ('ns1', Namespace('http://test.org/ns#')),
  145. ('ns2', Namespace('http://my_org.net/ns#')),
  146. ('ns3', Namespace('urn:test:')),
  147. ('ns4', Namespace('info:myinst/graph#')),
  148. )
  149. def test_ns(self, store, bindings):
  150. '''
  151. Test namespace bindings.
  152. '''
  153. with TxnManager(store, True) as txn:
  154. for b in bindings:
  155. store.bind(*b)
  156. nslist = list(store.namespaces())
  157. assert len(nslist) == len(bindings)
  158. for i in range(len(bindings)):
  159. assert nslist[i] == bindings[i]
  160. def test_ns2pfx(self, store, bindings):
  161. '''
  162. Test namespace to prefix conversion.
  163. '''
  164. with TxnManager(store, True) as txn:
  165. for b in bindings:
  166. pfx, ns = b
  167. assert store.namespace(pfx) == ns
  168. def test_pfx2ns(self, store, bindings):
  169. '''
  170. Test namespace to prefix conversion.
  171. '''
  172. with TxnManager(store, True) as txn:
  173. for b in bindings:
  174. pfx, ns = b
  175. assert store.prefix(ns) == pfx
  176. @pytest.mark.usefixtures('store')
  177. class TestContext:
  178. '''
  179. Tests for context handling.
  180. '''
  181. def test_add_graph(self, store):
  182. '''
  183. Test creating an empty and a non-empty graph.
  184. '''
  185. gr_uri = URIRef('urn:bogus:graph#a')
  186. with TxnManager(store, True) as txn:
  187. store.add_graph(gr_uri)
  188. assert gr_uri in store.contexts()
  189. def test_empty_context(self, store):
  190. '''
  191. Test creating and deleting empty contexts.
  192. '''
  193. gr_uri = URIRef('urn:bogus:empty#a')
  194. with TxnManager(store, True) as txn:
  195. store.add_graph(gr_uri)
  196. assert gr_uri in store.contexts()
  197. store.ermove_graph(gr_uri)
  198. assert gr_uri not in store.contexts()
  199. def test_add_trp_to_ctx(self, store):
  200. '''
  201. Test adding triples to a graph.
  202. '''
  203. gr_uri = URIRef('urn:bogus:graph#a') # From previous test
  204. gr2_uri = URIRef('urn:bogus:graph#b') # Never created before
  205. trp1 = (URIRef('urn:s:1'), URIRef('urn:p:1'), URIRef('urn:o:1'))
  206. trp2 = (URIRef('urn:s:2'), URIRef('urn:p:2'), URIRef('urn:o:2'))
  207. trp3 = (URIRef('urn:s:3'), URIRef('urn:p:3'), URIRef('urn:o:3'))
  208. with TxnManager(store, True) as txn:
  209. store.add(trp1, gr_uri)
  210. store.add(trp2, gr_uri)
  211. store.add(trp2, store.DEFAULT_GRAPH_URI)
  212. store.add(trp3, gr2_uri)
  213. store.add(trp3)
  214. assert len(set(store.triples((None, None, None)))) == 2
  215. assert len(set(store.triples((None, None, None), gr_uri))) == 2
  216. assert len(set(store.triples((None, None, None), gr2_uri))) == 1
  217. assert gr2_uri in store.contexts()
  218. assert trp1 not in store.triples((None, None, None))
  219. assert trp1 not in store.triples((None, None, None),
  220. store.DEFAULT_GRAPH_URI)
  221. assert trp2 in store.triples((None, None, None), gr_uri)
  222. assert trp2 in store.triples((None, None, None))
  223. assert trp3 in store.triples((None, None, None), gr2_uri)
  224. assert trp3 in store.triples((None, None, None),
  225. store.DEFAULT_GRAPH_URI)
  226. def test_delete_from_ctx(self, store):
  227. '''
  228. Delete triples from a named graph and from the default graph.
  229. '''
  230. gr_uri = URIRef('urn:bogus:graph#a')
  231. gr2_uri = URIRef('urn:bogus:graph#b')
  232. with TxnManager(store, True) as txn:
  233. store.remove((None, None, None))
  234. store.remove((None, None, None), gr2_uri)
  235. assert len(set(store.triples((None, None, None)))) == 0
  236. assert len(set(store.triples((None, None, None), gr_uri))) == 2
  237. assert len(set(store.triples((None, None, None), gr2_uri))) == 0