test_ldp.py 30 KB

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