test_ldp.py 29 KB

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