test_ldp.py 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806
  1. import pytest
  2. import uuid
  3. from hashlib import sha1
  4. from flask import g
  5. from rdflib import Graph
  6. from rdflib.compare import isomorphic
  7. from rdflib.namespace import RDF
  8. from rdflib.term import Literal, URIRef
  9. from lakesuperior.dictionaries.namespaces import ns_collection as nsc
  10. from lakesuperior.model.ldpr import Ldpr
  11. @pytest.fixture(scope='module')
  12. def random_uuid():
  13. return str(uuid.uuid4())
  14. @pytest.mark.usefixtures('client_class')
  15. @pytest.mark.usefixtures('db')
  16. class TestLdp:
  17. '''
  18. Test HTTP interaction with LDP endpoint.
  19. '''
  20. def test_get_root_node(self):
  21. '''
  22. Get the root node from two different endpoints.
  23. The test triplestore must be initialized, hence the `db` fixture.
  24. '''
  25. ldp_resp = self.client.get('/ldp')
  26. rest_resp = self.client.get('/rest')
  27. assert ldp_resp.status_code == 200
  28. assert rest_resp.status_code == 200
  29. def test_put_empty_resource(self, random_uuid):
  30. '''
  31. Check response headers for a PUT operation with empty payload.
  32. '''
  33. resp = self.client.put('/ldp/new_resource')
  34. assert resp.status_code == 201
  35. assert resp.data == bytes(
  36. '{}/new_resource'.format(g.webroot), 'utf-8')
  37. def test_put_existing_resource(self, random_uuid):
  38. '''
  39. Trying to PUT an existing resource should return a 204 if the payload
  40. is empty.
  41. '''
  42. path = '/ldp/nonidempotent01'
  43. put1_resp = self.client.put(path)
  44. assert put1_resp.status_code == 201
  45. assert self.client.get(path).status_code == 200
  46. put2_resp = self.client.put(path)
  47. with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f:
  48. put2_resp = self.client.put(
  49. path, data=f, content_type='text/turtle')
  50. assert put2_resp.status_code == 204
  51. put2_resp = self.client.put(path)
  52. assert put2_resp.status_code == 409
  53. def test_put_tree(self, client):
  54. '''
  55. PUT a resource with several path segments.
  56. The test should create intermediate path segments that are not
  57. accessible to PUT or POST.
  58. '''
  59. path = '/ldp/test_tree/a/b/c/d/e/f/g'
  60. self.client.put(path)
  61. assert self.client.get(path).status_code == 200
  62. assert self.client.put('/ldp/test_tree/a').status_code == 409
  63. assert self.client.post('/ldp/test_tree/a').status_code == 409
  64. def test_put_nested_tree(self, client):
  65. '''
  66. Verify that containment is set correctly in nested hierarchies.
  67. First put a new hierarchy and verify that the root node is its
  68. container; then put another hierarchy under it and verify that the
  69. first hierarchy is the container of the second one.
  70. '''
  71. uuid1 = 'test_nested_tree/a/b/c/d'
  72. uuid2 = uuid1 + '/e/f/g'
  73. path1 = '/ldp/' + uuid1
  74. path2 = '/ldp/' + uuid2
  75. self.client.put(path1)
  76. cont1_data = self.client.get('/ldp').data
  77. gr1 = Graph().parse(data=cont1_data, format='turtle')
  78. assert gr1[ URIRef(g.webroot + '/') : nsc['ldp'].contains : \
  79. URIRef(g.webroot + '/' + uuid1) ]
  80. self.client.put(path2)
  81. cont2_data = self.client.get(path1).data
  82. gr2 = Graph().parse(data=cont2_data, format='turtle')
  83. assert gr2[ URIRef(g.webroot + '/' + uuid1) : \
  84. nsc['ldp'].contains : \
  85. URIRef(g.webroot + '/' + uuid2) ]
  86. def test_put_ldp_rs(self, client):
  87. '''
  88. PUT a resource with RDF payload and verify.
  89. '''
  90. with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f:
  91. self.client.put('/ldp/ldprs01', data=f, content_type='text/turtle')
  92. resp = self.client.get('/ldp/ldprs01',
  93. headers={'accept' : 'text/turtle'})
  94. assert resp.status_code == 200
  95. gr = Graph().parse(data=resp.data, format='text/turtle')
  96. assert URIRef('http://vocab.getty.edu/ontology#Subject') in \
  97. gr.objects(None, RDF.type)
  98. def test_put_ldp_nr(self, rnd_img):
  99. '''
  100. PUT a resource with binary payload and verify checksums.
  101. '''
  102. rnd_img['content'].seek(0)
  103. resp = self.client.put('/ldp/ldpnr01', data=rnd_img['content'],
  104. headers={
  105. 'Content-Disposition' : 'attachment; filename={}'.format(
  106. rnd_img['filename'])})
  107. assert resp.status_code == 201
  108. resp = self.client.get('/ldp/ldpnr01', headers={'accept' : 'image/png'})
  109. assert resp.status_code == 200
  110. assert sha1(resp.data).hexdigest() == rnd_img['hash']
  111. def test_put_mismatched_ldp_rs(self, rnd_img):
  112. '''
  113. Verify MIME type / LDP mismatch.
  114. PUT a LDP-RS, then PUT a LDP-NR on the same location and verify it
  115. fails.
  116. '''
  117. path = '/ldp/' + str(uuid.uuid4())
  118. rnd_img['content'].seek(0)
  119. ldp_nr_resp = self.client.put(path, data=rnd_img['content'],
  120. headers={
  121. 'Content-Disposition' : 'attachment; filename={}'.format(
  122. rnd_img['filename'])})
  123. assert ldp_nr_resp.status_code == 201
  124. with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f:
  125. ldp_rs_resp = self.client.put(path, data=f,
  126. content_type='text/turtle')
  127. assert ldp_rs_resp.status_code == 415
  128. def test_put_mismatched_ldp_nr(self, rnd_img):
  129. '''
  130. Verify MIME type / LDP mismatch.
  131. PUT a LDP-NR, then PUT a LDP-RS on the same location and verify it
  132. fails.
  133. '''
  134. path = '/ldp/' + str(uuid.uuid4())
  135. with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f:
  136. ldp_rs_resp = self.client.put(path, data=f,
  137. content_type='text/turtle')
  138. assert ldp_rs_resp.status_code == 201
  139. rnd_img['content'].seek(0)
  140. ldp_nr_resp = self.client.put(path, data=rnd_img['content'],
  141. headers={
  142. 'Content-Disposition' : 'attachment; filename={}'.format(
  143. rnd_img['filename'])})
  144. assert ldp_nr_resp.status_code == 415
  145. def test_post_resource(self, client):
  146. '''
  147. Check response headers for a POST operation with empty payload.
  148. '''
  149. res = self.client.post('/ldp/')
  150. assert res.status_code == 201
  151. assert 'Location' in res.headers
  152. def test_post_slug(self):
  153. '''
  154. Verify that a POST with slug results in the expected URI only if the
  155. resource does not exist already.
  156. '''
  157. slug01_resp = self.client.post('/ldp', headers={'slug' : 'slug01'})
  158. assert slug01_resp.status_code == 201
  159. assert slug01_resp.headers['location'] == \
  160. g.webroot + '/slug01'
  161. slug02_resp = self.client.post('/ldp', headers={'slug' : 'slug01'})
  162. assert slug02_resp.status_code == 201
  163. assert slug02_resp.headers['location'] != \
  164. g.webroot + '/slug01'
  165. def test_post_404(self):
  166. '''
  167. Verify that a POST to a non-existing parent results in a 404.
  168. '''
  169. assert self.client.post('/ldp/{}'.format(uuid.uuid4()))\
  170. .status_code == 404
  171. def test_post_409(self, rnd_img):
  172. '''
  173. Verify that you cannot POST to a binary resource.
  174. '''
  175. rnd_img['content'].seek(0)
  176. self.client.put('/ldp/post_409', data=rnd_img['content'], headers={
  177. 'Content-Disposition' : 'attachment; filename={}'.format(
  178. rnd_img['filename'])})
  179. assert self.client.post('/ldp/post_409').status_code == 409
  180. def test_patch(self):
  181. '''
  182. Test patching a resource.
  183. '''
  184. path = '/ldp/test_patch01'
  185. self.client.put(path)
  186. uri = g.webroot + '/test_patch01'
  187. with open('tests/data/sparql_update/simple_insert.sparql') as data:
  188. resp = self.client.patch(path,
  189. data=data,
  190. headers={'content-type' : 'application/sparql-update'})
  191. assert resp.status_code == 204
  192. resp = self.client.get(path)
  193. gr = Graph().parse(data=resp.data, format='text/turtle')
  194. assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Hello') ]
  195. self.client.patch(path,
  196. data=open('tests/data/sparql_update/delete+insert+where.sparql'),
  197. headers={'content-type' : 'application/sparql-update'})
  198. resp = self.client.get(path)
  199. gr = Graph().parse(data=resp.data, format='text/turtle')
  200. assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Ciao') ]
  201. def test_patch_ssr(self):
  202. '''
  203. Test patching a resource violating the single-subject rule.
  204. '''
  205. path = '/ldp/test_patch_ssr'
  206. self.client.put(path)
  207. uri = g.webroot + '/test_patch_ssr'
  208. nossr_qry = 'INSERT { <http://bogus.org> a <urn:ns:A> . } WHERE {}'
  209. abs_qry = 'INSERT {{ <{}> a <urn:ns:A> . }} WHERE {{}}'.format(uri)
  210. frag_qry = 'INSERT {{ <{}#frag> a <urn:ns:A> . }} WHERE {{}}'\
  211. .format(uri)
  212. assert self.client.patch(
  213. path, data=nossr_qry,
  214. headers={'content-type' : 'application/sparql-update'}
  215. ).status_code == 412
  216. assert self.client.patch(
  217. path, data=abs_qry,
  218. headers={'content-type' : 'application/sparql-update'}
  219. ).status_code == 204
  220. assert self.client.patch(
  221. path, data=frag_qry,
  222. headers={'content-type' : 'application/sparql-update'}
  223. ).status_code == 204
  224. def test_patch_ldp_nr_metadata(self):
  225. '''
  226. Test patching a LDP-NR metadata resource, both from the fcr:metadata
  227. and the resource URIs.
  228. '''
  229. path = '/ldp/ldpnr01'
  230. with open('tests/data/sparql_update/simple_insert.sparql') as data:
  231. self.client.patch(path + '/fcr:metadata',
  232. data=data,
  233. headers={'content-type' : 'application/sparql-update'})
  234. resp = self.client.get(path + '/fcr:metadata')
  235. assert resp.status_code == 200
  236. uri = g.webroot + '/ldpnr01'
  237. gr = Graph().parse(data=resp.data, format='text/turtle')
  238. assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Hello') ]
  239. with open(
  240. 'tests/data/sparql_update/delete+insert+where.sparql') as data:
  241. patch_resp = self.client.patch(path,
  242. data=data,
  243. headers={'content-type' : 'application/sparql-update'})
  244. assert patch_resp.status_code == 204
  245. resp = self.client.get(path + '/fcr:metadata')
  246. assert resp.status_code == 200
  247. gr = Graph().parse(data=resp.data, format='text/turtle')
  248. assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Ciao') ]
  249. def test_patch_ldp_nr(self, rnd_img):
  250. '''
  251. Verify that a PATCH using anything other than an
  252. `application/sparql-update` MIME type results in an error.
  253. '''
  254. rnd_img['content'].seek(0)
  255. resp = self.client.patch('/ldp/ldpnr01/fcr:metadata',
  256. data=rnd_img,
  257. headers={'content-type' : 'image/jpeg'})
  258. assert resp.status_code == 415
  259. def test_delete(self):
  260. '''
  261. Test delete response codes.
  262. '''
  263. create_resp = self.client.put('/ldp/test_delete01')
  264. delete_resp = self.client.delete('/ldp/test_delete01')
  265. assert delete_resp.status_code == 204
  266. bogus_delete_resp = self.client.delete('/ldp/test_delete101')
  267. assert bogus_delete_resp.status_code == 404
  268. def test_tombstone(self):
  269. '''
  270. Test tombstone behaviors.
  271. For POST on a tombstone, check `test_resurrection`.
  272. '''
  273. tstone_resp = self.client.get('/ldp/test_delete01')
  274. assert tstone_resp.status_code == 410
  275. assert tstone_resp.headers['Link'] == \
  276. '<{}/test_delete01/fcr:tombstone>; rel="hasTombstone"'\
  277. .format(g.webroot)
  278. tstone_path = '/ldp/test_delete01/fcr:tombstone'
  279. assert self.client.get(tstone_path).status_code == 405
  280. assert self.client.put(tstone_path).status_code == 405
  281. assert self.client.delete(tstone_path).status_code == 204
  282. assert self.client.get('/ldp/test_delete01').status_code == 404
  283. def test_delete_recursive(self):
  284. '''
  285. Test response codes for resources deleted recursively and their
  286. tombstones.
  287. '''
  288. self.client.put('/ldp/test_delete_recursive01')
  289. self.client.put('/ldp/test_delete_recursive01/a')
  290. self.client.delete('/ldp/test_delete_recursive01')
  291. tstone_resp = self.client.get('/ldp/test_delete_recursive01')
  292. assert tstone_resp.status_code == 410
  293. assert tstone_resp.headers['Link'] == \
  294. '<{}/test_delete_recursive01/fcr:tombstone>; rel="hasTombstone"'\
  295. .format(g.webroot)
  296. child_tstone_resp = self.client.get('/ldp/test_delete_recursive01/a')
  297. assert child_tstone_resp.status_code == tstone_resp.status_code
  298. assert 'Link' not in child_tstone_resp.headers.keys()
  299. def test_put_fragments(self):
  300. '''
  301. Test the correct handling of fragment URIs on PUT and GET.
  302. '''
  303. with open('tests/data/fragments.ttl', 'rb') as f:
  304. self.client.put(
  305. '/ldp/test_fragment01',
  306. headers={
  307. 'Content-Type' : 'text/turtle',
  308. },
  309. data=f
  310. )
  311. rsp = self.client.get('/ldp/test_fragment01')
  312. gr = Graph().parse(data=rsp.data, format='text/turtle')
  313. assert gr[
  314. URIRef(g.webroot + '/test_fragment01#hash1')
  315. : URIRef('http://ex.org/p2') : URIRef('http://ex.org/o2')]
  316. def test_patch_fragments(self):
  317. '''
  318. Test the correct handling of fragment URIs on PATCH.
  319. '''
  320. self.client.put('/ldp/test_fragment_patch')
  321. with open('tests/data/fragments_insert.sparql', 'rb') as f:
  322. self.client.patch(
  323. '/ldp/test_fragment_patch',
  324. headers={
  325. 'Content-Type' : 'application/sparql-update',
  326. },
  327. data=f
  328. )
  329. ins_rsp = self.client.get('/ldp/test_fragment_patch')
  330. ins_gr = Graph().parse(data=ins_rsp.data, format='text/turtle')
  331. assert ins_gr[
  332. URIRef(g.webroot + '/test_fragment_patch#hash1234')
  333. : URIRef('http://ex.org/p3') : URIRef('http://ex.org/o3')]
  334. with open('tests/data/fragments_delete.sparql', 'rb') as f:
  335. self.client.patch(
  336. '/ldp/test_fragment_patch',
  337. headers={
  338. 'Content-Type' : 'application/sparql-update',
  339. },
  340. data=f
  341. )
  342. del_rsp = self.client.get('/ldp/test_fragment_patch')
  343. del_gr = Graph().parse(data=del_rsp.data, format='text/turtle')
  344. assert not del_gr[
  345. URIRef(g.webroot + '/test_fragment_patch#hash1234')
  346. : URIRef('http://ex.org/p3') : URIRef('http://ex.org/o3')]
  347. @pytest.mark.usefixtures('client_class')
  348. @pytest.mark.usefixtures('db')
  349. class TestPrefHeader:
  350. '''
  351. Test various combinations of `Prefer` header.
  352. '''
  353. @pytest.fixture(scope='class')
  354. def cont_structure(self):
  355. '''
  356. Create a container structure to be used for subsequent requests.
  357. '''
  358. parent_path = '/ldp/test_parent'
  359. self.client.put(parent_path)
  360. self.client.put(parent_path + '/child1')
  361. self.client.put(parent_path + '/child2')
  362. self.client.put(parent_path + '/child3')
  363. return {
  364. 'path' : parent_path,
  365. 'response' : self.client.get(parent_path),
  366. 'subject' : URIRef(g.webroot + '/test_parent')
  367. }
  368. def test_put_prefer_handling(self, random_uuid):
  369. '''
  370. Trying to PUT an existing resource should:
  371. - Return a 204 if the payload is empty
  372. - Return a 204 if the payload is RDF, server-managed triples are
  373. included and the 'Prefer' header is set to 'handling=lenient'
  374. - Return a 412 (ServerManagedTermError) if the payload is RDF,
  375. server-managed triples are included and handling is set to 'strict'
  376. '''
  377. path = '/ldp/put_pref_header01'
  378. assert self.client.put(path).status_code == 201
  379. assert self.client.get(path).status_code == 200
  380. assert self.client.put(path).status_code == 409
  381. with open('tests/data/rdf_payload_w_srv_mgd_trp.ttl', 'rb') as f:
  382. rsp_len = self.client.put(
  383. path,
  384. headers={
  385. 'Prefer' : 'handling=lenient',
  386. 'Content-Type' : 'text/turtle',
  387. },
  388. data=f
  389. )
  390. assert rsp_len.status_code == 204
  391. with open('tests/data/rdf_payload_w_srv_mgd_trp.ttl', 'rb') as f:
  392. rsp_strict = self.client.put(
  393. path,
  394. headers={
  395. 'Prefer' : 'handling=strict',
  396. 'Content-Type' : 'text/turtle',
  397. },
  398. data=f
  399. )
  400. assert rsp_strict.status_code == 412
  401. def test_embed_children(self, cont_structure):
  402. '''
  403. verify the "embed children" prefer header.
  404. '''
  405. parent_path = cont_structure['path']
  406. cont_resp = cont_structure['response']
  407. cont_subject = cont_structure['subject']
  408. minimal_resp = self.client.get(parent_path, headers={
  409. 'Prefer' : 'return=minimal',
  410. })
  411. incl_embed_children_resp = self.client.get(parent_path, headers={
  412. 'Prefer' : 'return=representation; include={}'\
  413. .format(Ldpr.EMBED_CHILD_RES_URI),
  414. })
  415. omit_embed_children_resp = self.client.get(parent_path, headers={
  416. 'Prefer' : 'return=representation; omit={}'\
  417. .format(Ldpr.EMBED_CHILD_RES_URI),
  418. })
  419. default_gr = Graph().parse(data=cont_resp.data, format='turtle')
  420. incl_gr = Graph().parse(
  421. data=incl_embed_children_resp.data, format='turtle')
  422. omit_gr = Graph().parse(
  423. data=omit_embed_children_resp.data, format='turtle')
  424. assert isomorphic(omit_gr, default_gr)
  425. children = set(incl_gr[cont_subject : nsc['ldp'].contains])
  426. assert len(children) == 3
  427. children = set(incl_gr[cont_subject : nsc['ldp'].contains])
  428. for child_uri in children:
  429. assert set(incl_gr[ child_uri : : ])
  430. assert not set(omit_gr[ child_uri : : ])
  431. def test_return_children(self, cont_structure):
  432. '''
  433. verify the "return children" prefer header.
  434. '''
  435. parent_path = cont_structure['path']
  436. cont_resp = cont_structure['response']
  437. cont_subject = cont_structure['subject']
  438. incl_children_resp = self.client.get(parent_path, headers={
  439. 'Prefer' : 'return=representation; include={}'\
  440. .format(Ldpr.RETURN_CHILD_RES_URI),
  441. })
  442. omit_children_resp = self.client.get(parent_path, headers={
  443. 'Prefer' : 'return=representation; omit={}'\
  444. .format(Ldpr.RETURN_CHILD_RES_URI),
  445. })
  446. default_gr = Graph().parse(data=cont_resp.data, format='turtle')
  447. incl_gr = Graph().parse(data=incl_children_resp.data, format='turtle')
  448. omit_gr = Graph().parse(data=omit_children_resp.data, format='turtle')
  449. assert isomorphic(incl_gr, default_gr)
  450. children = incl_gr[cont_subject : nsc['ldp'].contains]
  451. for child_uri in children:
  452. assert not omit_gr[ cont_subject : nsc['ldp'].contains : child_uri ]
  453. def test_inbound_rel(self, cont_structure):
  454. '''
  455. verify the "inboud relationships" prefer header.
  456. '''
  457. parent_path = cont_structure['path']
  458. cont_resp = cont_structure['response']
  459. cont_subject = cont_structure['subject']
  460. incl_inbound_resp = self.client.get(parent_path, headers={
  461. 'Prefer' : 'return=representation; include={}'\
  462. .format(Ldpr.RETURN_INBOUND_REF_URI),
  463. })
  464. omit_inbound_resp = self.client.get(parent_path, headers={
  465. 'Prefer' : 'return=representation; omit={}'\
  466. .format(Ldpr.RETURN_INBOUND_REF_URI),
  467. })
  468. default_gr = Graph().parse(data=cont_resp.data, format='turtle')
  469. incl_gr = Graph().parse(data=incl_inbound_resp.data, format='turtle')
  470. omit_gr = Graph().parse(data=omit_inbound_resp.data, format='turtle')
  471. assert isomorphic(omit_gr, default_gr)
  472. assert set(incl_gr[ : : cont_subject ])
  473. assert not set(omit_gr[ : : cont_subject ])
  474. def test_srv_mgd_triples(self, cont_structure):
  475. '''
  476. verify the "server managed triples" prefer header.
  477. '''
  478. parent_path = cont_structure['path']
  479. cont_resp = cont_structure['response']
  480. cont_subject = cont_structure['subject']
  481. incl_srv_mgd_resp = self.client.get(parent_path, headers={
  482. 'Prefer' : 'return=representation; include={}'\
  483. .format(Ldpr.RETURN_SRV_MGD_RES_URI),
  484. })
  485. omit_srv_mgd_resp = self.client.get(parent_path, headers={
  486. 'Prefer' : 'return=representation; omit={}'\
  487. .format(Ldpr.RETURN_SRV_MGD_RES_URI),
  488. })
  489. default_gr = Graph().parse(data=cont_resp.data, format='turtle')
  490. incl_gr = Graph().parse(data=incl_srv_mgd_resp.data, format='turtle')
  491. omit_gr = Graph().parse(data=omit_srv_mgd_resp.data, format='turtle')
  492. assert isomorphic(incl_gr, default_gr)
  493. for pred in {
  494. nsc['fcrepo'].created,
  495. nsc['fcrepo'].createdBy,
  496. nsc['fcrepo'].lastModified,
  497. nsc['fcrepo'].lastModifiedBy,
  498. nsc['ldp'].contains,
  499. }:
  500. assert set(incl_gr[ cont_subject : pred : ])
  501. assert not set(omit_gr[ cont_subject : pred : ])
  502. for type in {
  503. nsc['fcrepo'].Resource,
  504. nsc['ldp'].Container,
  505. nsc['ldp'].Resource,
  506. }:
  507. assert incl_gr[ cont_subject : RDF.type : type ]
  508. assert not omit_gr[ cont_subject : RDF.type : type ]
  509. def test_delete_no_tstone(self):
  510. '''
  511. Test the `no-tombstone` Prefer option.
  512. '''
  513. self.client.put('/ldp/test_delete_no_tstone01')
  514. self.client.put('/ldp/test_delete_no_tstone01/a')
  515. self.client.delete('/ldp/test_delete_no_tstone01', headers={
  516. 'prefer' : 'no-tombstone'})
  517. resp = self.client.get('/ldp/test_delete_no_tstone01')
  518. assert resp.status_code == 404
  519. child_resp = self.client.get('/ldp/test_delete_no_tstone01/a')
  520. assert child_resp.status_code == 404
  521. @pytest.mark.usefixtures('client_class')
  522. @pytest.mark.usefixtures('db')
  523. class TestVersion:
  524. '''
  525. Test version creation, retrieval and deletion.
  526. '''
  527. def test_create_versions(self):
  528. '''
  529. Test that POSTing multiple times to fcr:versions creates the
  530. 'hasVersions' triple and yields multiple version snapshots.
  531. '''
  532. self.client.put('/ldp/test_version')
  533. create_rsp = self.client.post('/ldp/test_version/fcr:versions')
  534. assert create_rsp.status_code == 201
  535. rsrc_rsp = self.client.get('/ldp/test_version')
  536. rsrc_gr = Graph().parse(data=rsrc_rsp.data, format='turtle')
  537. assert len(set(rsrc_gr[: nsc['fcrepo'].hasVersions :])) == 1
  538. info_rsp = self.client.get('/ldp/test_version/fcr:versions')
  539. assert info_rsp.status_code == 200
  540. info_gr = Graph().parse(data=info_rsp.data, format='turtle')
  541. assert len(set(info_gr[: nsc['fcrepo'].hasVersion :])) == 1
  542. self.client.post('/ldp/test_version/fcr:versions')
  543. info2_rsp = self.client.get('/ldp/test_version/fcr:versions')
  544. info2_gr = Graph().parse(data=info2_rsp.data, format='turtle')
  545. assert len(set(info2_gr[: nsc['fcrepo'].hasVersion :])) == 2
  546. def test_version_with_slug(self):
  547. '''
  548. Test a version with a slug.
  549. '''
  550. self.client.put('/ldp/test_version_slug')
  551. create_rsp = self.client.post('/ldp/test_version_slug/fcr:versions',
  552. headers={'slug' : 'v1'})
  553. new_ver_uri = create_rsp.headers['Location']
  554. assert new_ver_uri == g.webroot + '/test_version_slug/fcr:versions/v1'
  555. info_rsp = self.client.get('/ldp/test_version_slug/fcr:versions')
  556. info_gr = Graph().parse(data=info_rsp.data, format='turtle')
  557. assert info_gr[
  558. URIRef(new_ver_uri) :
  559. nsc['fcrepo'].hasVersionLabel :
  560. Literal('v1')]
  561. def test_dupl_version(self):
  562. '''
  563. Make sure that two POSTs with the same slug result in two different
  564. versions.
  565. '''
  566. path = '/ldp/test_duplicate_slug'
  567. self.client.put(path)
  568. v1_rsp = self.client.post(path + '/fcr:versions',
  569. headers={'slug' : 'v1'})
  570. v1_uri = v1_rsp.headers['Location']
  571. dup_rsp = self.client.post(path + '/fcr:versions',
  572. headers={'slug' : 'v1'})
  573. dup_uri = dup_rsp.headers['Location']
  574. assert v1_uri != dup_uri
  575. def test_revert_version(self):
  576. '''
  577. Take a version snapshot, update a resource, and then revert to the
  578. previous vresion.
  579. '''
  580. rsrc_path = '/ldp/test_revert_version'
  581. payload1 = '<> <urn:demo:p1> <urn:demo:o1> .'
  582. payload2 = '<> <urn:demo:p1> <urn:demo:o2> .'
  583. self.client.put(rsrc_path, headers={
  584. 'content-type': 'text/turtle'}, data=payload1)
  585. self.client.post(
  586. rsrc_path + '/fcr:versions', headers={'slug': 'v1'})
  587. v1_rsp = self.client.get(rsrc_path)
  588. v1_gr = Graph().parse(data=v1_rsp.data, format='turtle')
  589. assert v1_gr[
  590. URIRef(g.webroot + '/test_revert_version')
  591. : URIRef('urn:demo:p1')
  592. : URIRef('urn:demo:o1')
  593. ]
  594. self.client.put(rsrc_path, headers={
  595. 'content-type': 'text/turtle'}, data=payload2)
  596. v2_rsp = self.client.get(rsrc_path)
  597. v2_gr = Graph().parse(data=v2_rsp.data, format='turtle')
  598. assert v2_gr[
  599. URIRef(g.webroot + '/test_revert_version')
  600. : URIRef('urn:demo:p1')
  601. : URIRef('urn:demo:o2')
  602. ]
  603. self.client.patch(rsrc_path + '/fcr:versions/v1')
  604. revert_rsp = self.client.get(rsrc_path)
  605. revert_gr = Graph().parse(data=revert_rsp.data, format='turtle')
  606. assert revert_gr[
  607. URIRef(g.webroot + '/test_revert_version')
  608. : URIRef('urn:demo:p1')
  609. : URIRef('urn:demo:o1')
  610. ]
  611. def test_resurrection(self):
  612. '''
  613. Delete and then resurrect a resource.
  614. Make sure that the resource is resurrected to the latest version.
  615. '''
  616. path = '/ldp/test_lazarus'
  617. self.client.put(path)
  618. self.client.post(path + '/fcr:versions')
  619. self.client.put(
  620. path, headers={'content-type': 'text/turtle'},
  621. data=b'<> <urn:demo:p1> <urn:demo:o1> .')
  622. self.client.post(path + '/fcr:versions')
  623. self.client.put(
  624. path, headers={'content-type': 'text/turtle'},
  625. data=b'<> <urn:demo:p1> <urn:demo:o2> .')
  626. self.client.delete(path)
  627. assert self.client.get(path).status_code == 410
  628. self.client.post(path + '/fcr:tombstone')
  629. laz_data = self.client.get(path).data
  630. laz_gr = Graph().parse(data=laz_data, format='turtle')
  631. assert laz_gr[
  632. URIRef(g.webroot + '/test_lazarus')
  633. : URIRef('urn:demo:p1')
  634. : URIRef('urn:demo:o2')
  635. ]