test_ldp.py 39 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130
  1. import pdb
  2. import pytest
  3. import uuid
  4. from base64 import b64encode
  5. from hashlib import sha1
  6. from flask import g
  7. from rdflib import Graph
  8. from rdflib.compare import isomorphic
  9. from rdflib.namespace import RDF
  10. from rdflib.term import Literal, URIRef
  11. from lakesuperior.dictionaries.namespaces import ns_collection as nsc
  12. from lakesuperior.model.ldpr import Ldpr
  13. @pytest.fixture(scope='module')
  14. def random_uuid():
  15. return str(uuid.uuid4())
  16. @pytest.mark.usefixtures('client_class')
  17. @pytest.mark.usefixtures('db')
  18. class TestLdp:
  19. """
  20. Test HTTP interaction with LDP endpoint.
  21. """
  22. def test_get_root_node(self):
  23. """
  24. Get the root node from two different endpoints.
  25. The test triplestore must be initialized, hence the `db` fixture.
  26. """
  27. ldp_resp = self.client.get('/ldp')
  28. rest_resp = self.client.get('/rest')
  29. assert ldp_resp.status_code == 200
  30. assert rest_resp.status_code == 200
  31. def test_put_empty_resource(self, random_uuid):
  32. """
  33. Check response headers for a PUT operation with empty payload.
  34. """
  35. resp = self.client.put('/ldp/new_resource')
  36. assert resp.status_code == 201
  37. assert resp.data == bytes(
  38. '{}/new_resource'.format(g.webroot), 'utf-8')
  39. def test_put_existing_resource(self, random_uuid):
  40. """
  41. Trying to PUT an existing resource should return a 204 if the payload
  42. is empty.
  43. """
  44. path = '/ldp/nonidempotent01'
  45. put1_resp = self.client.put(path)
  46. assert put1_resp.status_code == 201
  47. assert self.client.get(path).status_code == 200
  48. put2_resp = self.client.put(path)
  49. with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f:
  50. put2_resp = self.client.put(
  51. path, data=f, content_type='text/turtle')
  52. assert put2_resp.status_code == 204
  53. put2_resp = self.client.put(path)
  54. assert put2_resp.status_code == 204
  55. def test_put_tree(self, client):
  56. """
  57. PUT a resource with several path segments.
  58. The test should create intermediate path segments that are LDPCs,
  59. accessible to PUT or POST.
  60. """
  61. path = '/ldp/test_tree/a/b/c/d/e/f/g'
  62. self.client.put(path)
  63. assert self.client.get(path).status_code == 200
  64. assert self.client.get('/ldp/test_tree/a/b/c').status_code == 200
  65. assert self.client.post('/ldp/test_tree/a/b').status_code == 201
  66. with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f:
  67. put_int_resp = self.client.put(
  68. 'ldp/test_tree/a', data=f, content_type='text/turtle')
  69. assert put_int_resp.status_code == 204
  70. # @TODO More thorough testing of contents
  71. def test_put_nested_tree(self, client):
  72. """
  73. Verify that containment is set correctly in nested hierarchies.
  74. First put a new hierarchy and verify that the root node is its
  75. container; then put another hierarchy under it and verify that the
  76. first hierarchy is the container of the second one.
  77. """
  78. uuid1 = 'test_nested_tree/a/b/c/d'
  79. uuid2 = uuid1 + '/e/f/g'
  80. path1 = '/ldp/' + uuid1
  81. path2 = '/ldp/' + uuid2
  82. self.client.put(path1)
  83. cont1_data = self.client.get('/ldp').data
  84. gr1 = Graph().parse(data=cont1_data, format='turtle')
  85. assert gr1[ URIRef(g.webroot + '/') : nsc['ldp'].contains : \
  86. URIRef(g.webroot + '/test_nested_tree') ]
  87. self.client.put(path2)
  88. cont2_data = self.client.get(path1).data
  89. gr2 = Graph().parse(data=cont2_data, format='turtle')
  90. assert gr2[ URIRef(g.webroot + '/' + uuid1) : \
  91. nsc['ldp'].contains : \
  92. URIRef(g.webroot + '/' + uuid1 + '/e') ]
  93. def test_put_ldp_rs(self, client):
  94. """
  95. PUT a resource with RDF payload and verify.
  96. """
  97. with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f:
  98. self.client.put('/ldp/ldprs01', data=f, content_type='text/turtle')
  99. resp = self.client.get('/ldp/ldprs01',
  100. headers={'accept' : 'text/turtle'})
  101. assert resp.status_code == 200
  102. gr = Graph().parse(data=resp.data, format='text/turtle')
  103. assert URIRef('http://vocab.getty.edu/ontology#Subject') in \
  104. gr.objects(None, RDF.type)
  105. def test_put_ldp_nr(self, rnd_img):
  106. """
  107. PUT a resource with binary payload and verify checksums.
  108. """
  109. rnd_img['content'].seek(0)
  110. resp = self.client.put('/ldp/ldpnr01', data=rnd_img['content'],
  111. headers={
  112. 'Content-Type': 'image/png',
  113. 'Content-Disposition' : 'attachment; filename={}'.format(
  114. rnd_img['filename'])})
  115. assert resp.status_code == 201
  116. resp = self.client.get(
  117. '/ldp/ldpnr01', headers={'accept' : 'image/png'})
  118. assert resp.status_code == 200
  119. assert sha1(resp.data).hexdigest() == rnd_img['hash']
  120. def test_put_ldp_nr_multipart(self, rnd_img):
  121. """
  122. PUT a resource with a multipart/form-data payload.
  123. """
  124. rnd_img['content'].seek(0)
  125. resp = self.client.put(
  126. '/ldp/ldpnr02',
  127. data={
  128. 'file': (
  129. rnd_img['content'], rnd_img['filename'],
  130. 'image/png',
  131. )
  132. }
  133. )
  134. assert resp.status_code == 201
  135. resp = self.client.get(
  136. '/ldp/ldpnr02', headers={'accept' : 'image/png'})
  137. assert resp.status_code == 200
  138. assert sha1(resp.data).hexdigest() == rnd_img['hash']
  139. def test_get_ldp_nr(self, rnd_img):
  140. """
  141. PUT a resource with binary payload and test various retieval methods.
  142. """
  143. uid = '/ldpnr03'
  144. path = '/ldp' + uid
  145. content = b'This is some exciting content.'
  146. resp = self.client.put(path, data=content,
  147. headers={
  148. 'Content-Type': 'text/plain',
  149. 'Content-Disposition' : 'attachment; filename=exciting.txt'})
  150. assert resp.status_code == 201
  151. uri = g.webroot + uid
  152. # Content retrieval methods.
  153. resp_bin1 = self.client.get(path)
  154. assert resp_bin1.status_code == 200
  155. assert resp_bin1.data == content
  156. resp_bin2 = self.client.get(path, headers={'accept' : 'text/plain'})
  157. assert resp_bin2.status_code == 200
  158. assert resp_bin2.data == content
  159. resp_bin3 = self.client.get(path + '/fcr:content')
  160. assert resp_bin3.status_code == 200
  161. assert resp_bin3.data == content
  162. # Metadata retrieval methods.
  163. resp_md1 = self.client.get(path, headers={'accept' : 'text/turtle'})
  164. assert resp_md1.status_code == 200
  165. gr1 = Graph().parse(data=resp_md1.data, format='text/turtle')
  166. assert gr1[ URIRef(uri) : nsc['rdf'].type : nsc['ldp'].Resource]
  167. resp_md2 = self.client.get(path + '/fcr:metadata')
  168. assert resp_md2.status_code == 200
  169. gr2 = Graph().parse(data=resp_md2.data, format='text/turtle')
  170. assert isomorphic(gr1, gr2)
  171. def test_put_mismatched_ldp_rs(self, rnd_img):
  172. """
  173. Verify MIME type / LDP mismatch.
  174. PUT a LDP-RS, then PUT a LDP-NR on the same location and verify it
  175. fails.
  176. """
  177. path = '/ldp/' + str(uuid.uuid4())
  178. rnd_img['content'].seek(0)
  179. ldp_nr_resp = self.client.put(path, data=rnd_img['content'],
  180. headers={
  181. 'Content-Disposition' : 'attachment; filename={}'.format(
  182. rnd_img['filename'])})
  183. assert ldp_nr_resp.status_code == 201
  184. with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f:
  185. ldp_rs_resp = self.client.put(path, data=f,
  186. content_type='text/turtle')
  187. assert ldp_rs_resp.status_code == 415
  188. def test_put_mismatched_ldp_nr(self, rnd_img):
  189. """
  190. Verify MIME type / LDP mismatch.
  191. PUT a LDP-NR, then PUT a LDP-RS on the same location and verify it
  192. fails.
  193. """
  194. path = '/ldp/' + str(uuid.uuid4())
  195. with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f:
  196. ldp_rs_resp = self.client.put(path, data=f,
  197. content_type='text/turtle')
  198. assert ldp_rs_resp.status_code == 201
  199. rnd_img['content'].seek(0)
  200. ldp_nr_resp = self.client.put(path, data=rnd_img['content'],
  201. headers={
  202. 'Content-Disposition' : 'attachment; filename={}'.format(
  203. rnd_img['filename'])})
  204. assert ldp_nr_resp.status_code == 415
  205. def test_missing_reference(self, client):
  206. """
  207. PUT a resource with RDF payload referencing a non-existing in-repo
  208. resource.
  209. """
  210. self.client.get('/ldp')
  211. data = '''
  212. PREFIX ns: <http://example.org#>
  213. PREFIX res: <http://example-source.org/res/>
  214. <> ns:p1 res:bogus ;
  215. ns:p2 <{0}> ;
  216. ns:p3 <{0}/> ;
  217. ns:p4 <{0}/nonexistent> .
  218. '''.format(g.webroot)
  219. put_rsp = self.client.put('/ldp/test_missing_ref', data=data, headers={
  220. 'content-type': 'text/turtle'})
  221. assert put_rsp.status_code == 201
  222. resp = self.client.get('/ldp/test_missing_ref',
  223. headers={'accept' : 'text/turtle'})
  224. assert resp.status_code == 200
  225. gr = Graph().parse(data=resp.data, format='text/turtle')
  226. assert URIRef('http://example-source.org/res/bogus') in \
  227. gr.objects(None, URIRef('http://example.org#p1'))
  228. assert URIRef(g.webroot + '/') in (
  229. gr.objects(None, URIRef('http://example.org#p2')))
  230. assert URIRef(g.webroot + '/') in (
  231. gr.objects(None, URIRef('http://example.org#p3')))
  232. assert URIRef(g.webroot + '/nonexistent') not in (
  233. gr.objects(None, URIRef('http://example.org#p4')))
  234. def test_post_resource(self, client):
  235. """
  236. Check response headers for a POST operation with empty payload.
  237. """
  238. res = self.client.post('/ldp/')
  239. assert res.status_code == 201
  240. assert 'Location' in res.headers
  241. def test_post_ldp_nr(self, rnd_img):
  242. """
  243. POST a resource with binary payload and verify checksums.
  244. """
  245. import pdb; pdb.set_trace()
  246. rnd_img['content'].seek(0)
  247. resp = self.client.post('/ldp/', data=rnd_img['content'],
  248. headers={
  249. 'slug': 'ldpnr04',
  250. 'Content-Type': 'image/png',
  251. 'Content-Disposition' : 'attachment; filename={}'.format(
  252. rnd_img['filename'])})
  253. assert resp.status_code == 201
  254. resp = self.client.get(
  255. '/ldp/ldpnr04', headers={'accept' : 'image/png'})
  256. assert resp.status_code == 200
  257. assert sha1(resp.data).hexdigest() == rnd_img['hash']
  258. def test_post_slug(self):
  259. """
  260. Verify that a POST with slug results in the expected URI only if the
  261. resource does not exist already.
  262. """
  263. slug01_resp = self.client.post('/ldp', headers={'slug' : 'slug01'})
  264. assert slug01_resp.status_code == 201
  265. assert slug01_resp.headers['location'] == \
  266. g.webroot + '/slug01'
  267. slug02_resp = self.client.post('/ldp', headers={'slug' : 'slug01'})
  268. assert slug02_resp.status_code == 201
  269. assert slug02_resp.headers['location'] != \
  270. g.webroot + '/slug01'
  271. def test_post_404(self):
  272. """
  273. Verify that a POST to a non-existing parent results in a 404.
  274. """
  275. assert self.client.post('/ldp/{}'.format(uuid.uuid4()))\
  276. .status_code == 404
  277. def test_post_409(self, rnd_img):
  278. """
  279. Verify that you cannot POST to a binary resource.
  280. """
  281. rnd_img['content'].seek(0)
  282. self.client.put('/ldp/post_409', data=rnd_img['content'], headers={
  283. 'Content-Disposition' : 'attachment; filename={}'.format(
  284. rnd_img['filename'])})
  285. assert self.client.post('/ldp/post_409').status_code == 409
  286. def test_patch_root(self):
  287. """
  288. Test patching root node.
  289. """
  290. path = '/ldp/'
  291. self.client.get(path)
  292. uri = g.webroot + '/'
  293. with open('tests/data/sparql_update/simple_insert.sparql') as data:
  294. resp = self.client.patch(path,
  295. data=data,
  296. headers={'content-type' : 'application/sparql-update'})
  297. assert resp.status_code == 204
  298. resp = self.client.get(path)
  299. gr = Graph().parse(data=resp.data, format='text/turtle')
  300. assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Hello') ]
  301. def test_patch(self):
  302. """
  303. Test patching a resource.
  304. """
  305. path = '/ldp/test_patch01'
  306. self.client.put(path)
  307. uri = g.webroot + '/test_patch01'
  308. with open('tests/data/sparql_update/simple_insert.sparql') as data:
  309. resp = self.client.patch(path,
  310. data=data,
  311. headers={'content-type' : 'application/sparql-update'})
  312. assert resp.status_code == 204
  313. resp = self.client.get(path)
  314. gr = Graph().parse(data=resp.data, format='text/turtle')
  315. assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Hello') ]
  316. self.client.patch(path,
  317. data=open('tests/data/sparql_update/delete+insert+where.sparql'),
  318. headers={'content-type' : 'application/sparql-update'})
  319. resp = self.client.get(path)
  320. gr = Graph().parse(data=resp.data, format='text/turtle')
  321. assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Ciao') ]
  322. def test_patch_ssr(self):
  323. """
  324. Test patching a resource violating the single-subject rule.
  325. """
  326. path = '/ldp/test_patch_ssr'
  327. self.client.put(path)
  328. uri = g.webroot + '/test_patch_ssr'
  329. nossr_qry = 'INSERT { <http://bogus.org> a <urn:ns:A> . } WHERE {}'
  330. abs_qry = 'INSERT {{ <{}> a <urn:ns:A> . }} WHERE {{}}'.format(uri)
  331. frag_qry = 'INSERT {{ <{}#frag> a <urn:ns:A> . }} WHERE {{}}'\
  332. .format(uri)
  333. # @TODO Leave commented until a decision is made about SSR.
  334. assert self.client.patch(
  335. path, data=nossr_qry,
  336. headers={'content-type': 'application/sparql-update'}
  337. ).status_code == 204
  338. assert self.client.patch(
  339. path, data=abs_qry,
  340. headers={'content-type': 'application/sparql-update'}
  341. ).status_code == 204
  342. assert self.client.patch(
  343. path, data=frag_qry,
  344. headers={'content-type': 'application/sparql-update'}
  345. ).status_code == 204
  346. def test_patch_ldp_nr_metadata(self):
  347. """
  348. Test patching a LDP-NR metadata resource from the fcr:metadata URI.
  349. """
  350. path = '/ldp/ldpnr01'
  351. with open('tests/data/sparql_update/simple_insert.sparql') as data:
  352. self.client.patch(path + '/fcr:metadata',
  353. data=data,
  354. headers={'content-type' : 'application/sparql-update'})
  355. resp = self.client.get(path + '/fcr:metadata')
  356. assert resp.status_code == 200
  357. uri = g.webroot + '/ldpnr01'
  358. gr = Graph().parse(data=resp.data, format='text/turtle')
  359. assert gr[URIRef(uri) : nsc['dc'].title : Literal('Hello')]
  360. with open(
  361. 'tests/data/sparql_update/delete+insert+where.sparql') as data:
  362. patch_resp = self.client.patch(path + '/fcr:metadata',
  363. data=data,
  364. headers={'content-type' : 'application/sparql-update'})
  365. assert patch_resp.status_code == 204
  366. resp = self.client.get(path + '/fcr:metadata')
  367. assert resp.status_code == 200
  368. gr = Graph().parse(data=resp.data, format='text/turtle')
  369. assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Ciao') ]
  370. def test_patch_ldpnr(self):
  371. """
  372. Verify that a direct PATCH to a LDP-NR results in a 415.
  373. """
  374. with open(
  375. 'tests/data/sparql_update/delete+insert+where.sparql') as data:
  376. patch_resp = self.client.patch('/ldp/ldpnr01',
  377. data=data,
  378. headers={'content-type': 'application/sparql-update'})
  379. assert patch_resp.status_code == 415
  380. def test_patch_invalid_mimetype(self, rnd_img):
  381. """
  382. Verify that a PATCH using anything other than an
  383. `application/sparql-update` MIME type results in an error.
  384. """
  385. self.client.put('/ldp/test_patch_invalid_mimetype')
  386. rnd_img['content'].seek(0)
  387. ldpnr_resp = self.client.patch('/ldp/ldpnr01/fcr:metadata',
  388. data=rnd_img,
  389. headers={'content-type' : 'image/jpeg'})
  390. ldprs_resp = self.client.patch('/ldp/test_patch_invalid_mimetype',
  391. data=b'Hello, I\'m not a SPARQL update.',
  392. headers={'content-type' : 'text/plain'})
  393. assert ldprs_resp.status_code == ldpnr_resp.status_code == 415
  394. def test_delete(self):
  395. """
  396. Test delete response codes.
  397. """
  398. self.client.put('/ldp/test_delete01')
  399. delete_resp = self.client.delete('/ldp/test_delete01')
  400. assert delete_resp.status_code == 204
  401. bogus_delete_resp = self.client.delete('/ldp/test_delete101')
  402. assert bogus_delete_resp.status_code == 404
  403. def test_tombstone(self):
  404. """
  405. Test tombstone behaviors.
  406. For POST on a tombstone, check `test_resurrection`.
  407. """
  408. tstone_resp = self.client.get('/ldp/test_delete01')
  409. assert tstone_resp.status_code == 410
  410. assert tstone_resp.headers['Link'] == \
  411. '<{}/test_delete01/fcr:tombstone>; rel="hasTombstone"'\
  412. .format(g.webroot)
  413. tstone_path = '/ldp/test_delete01/fcr:tombstone'
  414. assert self.client.get(tstone_path).status_code == 405
  415. assert self.client.put(tstone_path).status_code == 405
  416. assert self.client.delete(tstone_path).status_code == 204
  417. assert self.client.get('/ldp/test_delete01').status_code == 404
  418. def test_delete_recursive(self):
  419. """
  420. Test response codes for resources deleted recursively and their
  421. tombstones.
  422. """
  423. child_suffixes = ('a', 'a/b', 'a/b/c', 'a1', 'a1/b1')
  424. self.client.put('/ldp/test_delete_recursive01')
  425. for cs in child_suffixes:
  426. self.client.put('/ldp/test_delete_recursive01/{}'.format(cs))
  427. assert self.client.delete(
  428. '/ldp/test_delete_recursive01').status_code == 204
  429. tstone_resp = self.client.get('/ldp/test_delete_recursive01')
  430. assert tstone_resp.status_code == 410
  431. assert tstone_resp.headers['Link'] == \
  432. '<{}/test_delete_recursive01/fcr:tombstone>; rel="hasTombstone"'\
  433. .format(g.webroot)
  434. for cs in child_suffixes:
  435. child_tstone_resp = self.client.get(
  436. '/ldp/test_delete_recursive01/{}'.format(cs))
  437. assert child_tstone_resp.status_code == tstone_resp.status_code
  438. assert 'Link' not in child_tstone_resp.headers.keys()
  439. def test_put_fragments(self):
  440. """
  441. Test the correct handling of fragment URIs on PUT and GET.
  442. """
  443. with open('tests/data/fragments.ttl', 'rb') as f:
  444. self.client.put(
  445. '/ldp/test_fragment01',
  446. headers={
  447. 'Content-Type' : 'text/turtle',
  448. },
  449. data=f
  450. )
  451. rsp = self.client.get('/ldp/test_fragment01')
  452. gr = Graph().parse(data=rsp.data, format='text/turtle')
  453. assert gr[
  454. URIRef(g.webroot + '/test_fragment01#hash1')
  455. : URIRef('http://ex.org/p2') : URIRef('http://ex.org/o2')]
  456. def test_patch_fragments(self):
  457. """
  458. Test the correct handling of fragment URIs on PATCH.
  459. """
  460. self.client.put('/ldp/test_fragment_patch')
  461. with open('tests/data/fragments_insert.sparql', 'rb') as f:
  462. self.client.patch(
  463. '/ldp/test_fragment_patch',
  464. headers={
  465. 'Content-Type' : 'application/sparql-update',
  466. },
  467. data=f
  468. )
  469. ins_rsp = self.client.get('/ldp/test_fragment_patch')
  470. ins_gr = Graph().parse(data=ins_rsp.data, format='text/turtle')
  471. assert ins_gr[
  472. URIRef(g.webroot + '/test_fragment_patch#hash1234')
  473. : URIRef('http://ex.org/p3') : URIRef('http://ex.org/o3')]
  474. with open('tests/data/fragments_delete.sparql', 'rb') as f:
  475. self.client.patch(
  476. '/ldp/test_fragment_patch',
  477. headers={
  478. 'Content-Type' : 'application/sparql-update',
  479. },
  480. data=f
  481. )
  482. del_rsp = self.client.get('/ldp/test_fragment_patch')
  483. del_gr = Graph().parse(data=del_rsp.data, format='text/turtle')
  484. assert not del_gr[
  485. URIRef(g.webroot + '/test_fragment_patch#hash1234')
  486. : URIRef('http://ex.org/p3') : URIRef('http://ex.org/o3')]
  487. @pytest.mark.usefixtures('client_class')
  488. @pytest.mark.usefixtures('db')
  489. class TestMimeType:
  490. """
  491. Test ``Accept`` headers and input & output formats.
  492. """
  493. def test_accept(self):
  494. """
  495. Verify the default serialization method.
  496. """
  497. accept_list = {
  498. ('', 'text/turtle'),
  499. ('text/turtle', 'text/turtle'),
  500. ('application/rdf+xml', 'application/rdf+xml'),
  501. ('application/n-triples', 'application/n-triples'),
  502. ('application/bogus', 'text/turtle'),
  503. (
  504. 'application/rdf+xml;q=0.5,application/n-triples;q=0.7',
  505. 'application/n-triples'),
  506. (
  507. 'application/rdf+xml;q=0.5,application/bogus;q=0.7',
  508. 'application/rdf+xml'),
  509. ('application/rdf+xml;q=0.5,text/n3;q=0.7', 'text/n3'),
  510. (
  511. 'application/rdf+xml;q=0.5,application/ld+json;q=0.7',
  512. 'application/ld+json'),
  513. }
  514. for mimetype, fmt in accept_list:
  515. rsp = self.client.get('/ldp', headers={'Accept': mimetype})
  516. assert rsp.mimetype == fmt
  517. gr = Graph(identifier=g.webroot + '/').parse(
  518. data=rsp.data, format=fmt)
  519. assert nsc['fcrepo'].RepositoryRoot in set(gr.objects())
  520. def test_provided_rdf(self):
  521. """
  522. Test several input RDF serialiation formats.
  523. """
  524. self.client.get('/ldp')
  525. gr = Graph()
  526. gr.add((
  527. URIRef(g.webroot + '/test_mimetype'),
  528. nsc['dcterms'].title, Literal('Test MIME type.')))
  529. test_list = {
  530. 'application/n-triples',
  531. 'application/rdf+xml',
  532. 'text/n3',
  533. 'text/turtle',
  534. 'application/ld+json',
  535. }
  536. for mimetype in test_list:
  537. rdf_data = gr.serialize(format=mimetype)
  538. self.client.put('/ldp/test_mimetype', data=rdf_data, headers={
  539. 'content-type': mimetype})
  540. rsp = self.client.get('/ldp/test_mimetype')
  541. rsp_gr = Graph(identifier=g.webroot + '/test_mimetype').parse(
  542. data=rsp.data, format='text/turtle')
  543. assert (
  544. URIRef(g.webroot + '/test_mimetype'),
  545. nsc['dcterms'].title, Literal('Test MIME type.')) in rsp_gr
  546. @pytest.mark.usefixtures('client_class')
  547. @pytest.mark.usefixtures('db')
  548. class TestPrefHeader:
  549. """
  550. Test various combinations of `Prefer` header.
  551. """
  552. @pytest.fixture(scope='class')
  553. def cont_structure(self):
  554. """
  555. Create a container structure to be used for subsequent requests.
  556. """
  557. parent_path = '/ldp/test_parent'
  558. self.client.put(parent_path)
  559. self.client.put(parent_path + '/child1')
  560. self.client.put(parent_path + '/child2')
  561. self.client.put(parent_path + '/child3')
  562. return {
  563. 'path' : parent_path,
  564. 'response' : self.client.get(parent_path),
  565. }
  566. def test_put_prefer_handling(self, random_uuid):
  567. """
  568. Trying to PUT an existing resource should:
  569. - Return a 204 if the payload is empty
  570. - Return a 204 if the payload is RDF, server-managed triples are
  571. included and the 'Prefer' header is set to 'handling=lenient'
  572. - Return a 412 (ServerManagedTermError) if the payload is RDF,
  573. server-managed triples are included and handling is set to 'strict',
  574. or not set.
  575. """
  576. path = '/ldp/put_pref_header01'
  577. assert self.client.put(path).status_code == 201
  578. assert self.client.get(path).status_code == 200
  579. assert self.client.put(path).status_code == 204
  580. # Default handling is strict.
  581. with open('tests/data/rdf_payload_w_srv_mgd_trp.ttl', 'rb') as f:
  582. rsp_default = self.client.put(
  583. path,
  584. headers={
  585. 'Content-Type' : 'text/turtle',
  586. },
  587. data=f
  588. )
  589. assert rsp_default.status_code == 412
  590. with open('tests/data/rdf_payload_w_srv_mgd_trp.ttl', 'rb') as f:
  591. rsp_len = self.client.put(
  592. path,
  593. headers={
  594. 'Prefer' : 'handling=lenient',
  595. 'Content-Type' : 'text/turtle',
  596. },
  597. data=f
  598. )
  599. assert rsp_len.status_code == 204
  600. with open('tests/data/rdf_payload_w_srv_mgd_trp.ttl', 'rb') as f:
  601. rsp_strict = self.client.put(
  602. path,
  603. headers={
  604. 'Prefer' : 'handling=strict',
  605. 'Content-Type' : 'text/turtle',
  606. },
  607. data=f
  608. )
  609. assert rsp_strict.status_code == 412
  610. # @HOLD Embed children is debated.
  611. def _disabled_test_embed_children(self, cont_structure):
  612. """
  613. verify the "embed children" prefer header.
  614. """
  615. self.client.get('/ldp')
  616. parent_path = cont_structure['path']
  617. cont_resp = cont_structure['response']
  618. cont_subject = URIRef(g.webroot + '/test_parent')
  619. #minimal_resp = self.client.get(parent_path, headers={
  620. # 'Prefer' : 'return=minimal',
  621. #})
  622. incl_embed_children_resp = self.client.get(parent_path, headers={
  623. 'Prefer' : 'return=representation; include={}'\
  624. .format(Ldpr.EMBED_CHILD_RES_URI),
  625. })
  626. omit_embed_children_resp = self.client.get(parent_path, headers={
  627. 'Prefer' : 'return=representation; omit={}'\
  628. .format(Ldpr.EMBED_CHILD_RES_URI),
  629. })
  630. default_gr = Graph().parse(data=cont_resp.data, format='turtle')
  631. incl_gr = Graph().parse(
  632. data=incl_embed_children_resp.data, format='turtle')
  633. omit_gr = Graph().parse(
  634. data=omit_embed_children_resp.data, format='turtle')
  635. assert isomorphic(omit_gr, default_gr)
  636. children = set(incl_gr[cont_subject : nsc['ldp'].contains])
  637. assert len(children) == 3
  638. children = set(incl_gr[cont_subject : nsc['ldp'].contains])
  639. for child_uri in children:
  640. assert set(incl_gr[ child_uri : : ])
  641. assert not set(omit_gr[ child_uri : : ])
  642. def test_return_children(self, cont_structure):
  643. """
  644. verify the "return children" prefer header.
  645. """
  646. self.client.get('/ldp')
  647. parent_path = cont_structure['path']
  648. cont_resp = cont_structure['response']
  649. cont_subject = URIRef(g.webroot + '/test_parent')
  650. incl_children_resp = self.client.get(parent_path, headers={
  651. 'Prefer' : 'return=representation; include={}'\
  652. .format(Ldpr.RETURN_CHILD_RES_URI),
  653. })
  654. omit_children_resp = self.client.get(parent_path, headers={
  655. 'Prefer' : 'return=representation; omit={}'\
  656. .format(Ldpr.RETURN_CHILD_RES_URI),
  657. })
  658. default_gr = Graph().parse(data=cont_resp.data, format='turtle')
  659. incl_gr = Graph().parse(data=incl_children_resp.data, format='turtle')
  660. omit_gr = Graph().parse(data=omit_children_resp.data, format='turtle')
  661. assert isomorphic(incl_gr, default_gr)
  662. children = incl_gr[cont_subject : nsc['ldp'].contains]
  663. for child_uri in children:
  664. assert not omit_gr[cont_subject : nsc['ldp'].contains : child_uri]
  665. def test_inbound_rel(self, cont_structure):
  666. """
  667. verify the "inbound relationships" prefer header.
  668. """
  669. self.client.put('/ldp/test_target')
  670. data = '<> <http://ex.org/ns#shoots> <{}> .'.format(
  671. g.webroot + '/test_target')
  672. self.client.put('/ldp/test_shooter', data=data,
  673. headers={'Content-Type': 'text/turtle'})
  674. cont_resp = self.client.get('/ldp/test_target')
  675. incl_inbound_resp = self.client.get('/ldp/test_target', headers={
  676. 'Prefer' : 'return=representation; include="{}"'\
  677. .format(Ldpr.RETURN_INBOUND_REF_URI),
  678. })
  679. omit_inbound_resp = self.client.get('/ldp/test_target', headers={
  680. 'Prefer' : 'return=representation; omit="{}"'\
  681. .format(Ldpr.RETURN_INBOUND_REF_URI),
  682. })
  683. default_gr = Graph().parse(data=cont_resp.data, format='turtle')
  684. incl_gr = Graph().parse(data=incl_inbound_resp.data, format='turtle')
  685. omit_gr = Graph().parse(data=omit_inbound_resp.data, format='turtle')
  686. subject = URIRef(g.webroot + '/test_target')
  687. inbd_subject = URIRef(g.webroot + '/test_shooter')
  688. assert isomorphic(omit_gr, default_gr)
  689. assert len(set(incl_gr[inbd_subject : : ])) == 1
  690. assert incl_gr[
  691. inbd_subject : URIRef('http://ex.org/ns#shoots') : subject]
  692. assert not len(set(omit_gr[inbd_subject : :]))
  693. def test_srv_mgd_triples(self, cont_structure):
  694. """
  695. verify the "server managed triples" prefer header.
  696. """
  697. self.client.get('/ldp')
  698. parent_path = cont_structure['path']
  699. cont_resp = cont_structure['response']
  700. cont_subject = URIRef(g.webroot + '/test_parent')
  701. incl_srv_mgd_resp = self.client.get(parent_path, headers={
  702. 'Prefer' : 'return=representation; include={}'\
  703. .format(Ldpr.RETURN_SRV_MGD_RES_URI),
  704. })
  705. omit_srv_mgd_resp = self.client.get(parent_path, headers={
  706. 'Prefer' : 'return=representation; omit={}'\
  707. .format(Ldpr.RETURN_SRV_MGD_RES_URI),
  708. })
  709. default_gr = Graph().parse(data=cont_resp.data, format='turtle')
  710. incl_gr = Graph().parse(data=incl_srv_mgd_resp.data, format='turtle')
  711. omit_gr = Graph().parse(data=omit_srv_mgd_resp.data, format='turtle')
  712. assert isomorphic(incl_gr, default_gr)
  713. for pred in {
  714. nsc['fcrepo'].created,
  715. nsc['fcrepo'].createdBy,
  716. nsc['fcrepo'].lastModified,
  717. nsc['fcrepo'].lastModifiedBy,
  718. nsc['ldp'].contains,
  719. }:
  720. assert set(incl_gr[ cont_subject : pred : ])
  721. assert not set(omit_gr[ cont_subject : pred : ])
  722. for type in {
  723. nsc['fcrepo'].Resource,
  724. nsc['ldp'].Container,
  725. nsc['ldp'].Resource,
  726. }:
  727. assert incl_gr[ cont_subject : RDF.type : type ]
  728. assert not omit_gr[ cont_subject : RDF.type : type ]
  729. def test_delete_no_tstone(self):
  730. """
  731. Test the `no-tombstone` Prefer option.
  732. """
  733. self.client.put('/ldp/test_delete_no_tstone01')
  734. self.client.put('/ldp/test_delete_no_tstone01/a')
  735. self.client.delete('/ldp/test_delete_no_tstone01', headers={
  736. 'prefer' : 'no-tombstone'})
  737. resp = self.client.get('/ldp/test_delete_no_tstone01')
  738. assert resp.status_code == 404
  739. child_resp = self.client.get('/ldp/test_delete_no_tstone01/a')
  740. assert child_resp.status_code == 404
  741. #@pytest.mark.usefixtures('client_class')
  742. #@pytest.mark.usefixtures('db')
  743. #class TestDigest:
  744. # """
  745. # Test digest and ETag handling.
  746. # """
  747. # @pytest.mark.skip(reason='TODO Need to implement async digest queue')
  748. # def test_digest_post(self):
  749. # """
  750. # Test ``Digest`` and ``ETag`` headers on resource POST.
  751. # """
  752. # resp = self.client.post('/ldp/')
  753. # assert 'Digest' in resp.headers
  754. # assert 'ETag' in resp.headers
  755. # assert (
  756. # b64encode(bytes.fromhex(
  757. # resp.headers['ETag'].replace('W/', '')
  758. # )).decode('ascii') ==
  759. # resp.headers['Digest'].replace('SHA256=', ''))
  760. #
  761. #
  762. # @pytest.mark.skip(reason='TODO Need to implement async digest queue')
  763. # def test_digest_put(self):
  764. # """
  765. # Test ``Digest`` and ``ETag`` headers on resource PUT.
  766. # """
  767. # resp_put = self.client.put('/ldp/test_digest_put')
  768. # assert 'Digest' in resp_put.headers
  769. # assert 'ETag' in resp_put.headers
  770. # assert (
  771. # b64encode(bytes.fromhex(
  772. # resp_put.headers['ETag'].replace('W/', '')
  773. # )).decode('ascii') ==
  774. # resp_put.headers['Digest'].replace('SHA256=', ''))
  775. #
  776. # resp_get = self.client.get('/ldp/test_digest_put')
  777. # assert 'Digest' in resp_get.headers
  778. # assert 'ETag' in resp_get.headers
  779. # assert (
  780. # b64encode(bytes.fromhex(
  781. # resp_get.headers['ETag'].replace('W/', '')
  782. # )).decode('ascii') ==
  783. # resp_get.headers['Digest'].replace('SHA256=', ''))
  784. #
  785. #
  786. # @pytest.mark.skip(reason='TODO Need to implement async digest queue')
  787. # def test_digest_patch(self):
  788. # """
  789. # Verify that the digest and ETag change on resource change.
  790. # """
  791. # path = '/ldp/test_digest_patch'
  792. # self.client.put(path)
  793. # rsp1 = self.client.get(path)
  794. #
  795. # self.client.patch(
  796. # path, data=b'DELETE {} INSERT {<> a <http://ex.org/Test> .} '
  797. # b'WHERE {}',
  798. # headers={'Content-Type': 'application/sparql-update'})
  799. # rsp2 = self.client.get(path)
  800. #
  801. # assert rsp1.headers['ETag'] != rsp2.headers['ETag']
  802. # assert rsp1.headers['Digest'] != rsp2.headers['Digest']
  803. @pytest.mark.usefixtures('client_class')
  804. @pytest.mark.usefixtures('db')
  805. class TestVersion:
  806. """
  807. Test version creation, retrieval and deletion.
  808. """
  809. def test_create_versions(self):
  810. """
  811. Test that POSTing multiple times to fcr:versions creates the
  812. 'hasVersions' triple and yields multiple version snapshots.
  813. """
  814. self.client.put('/ldp/test_version')
  815. create_rsp = self.client.post('/ldp/test_version/fcr:versions')
  816. assert create_rsp.status_code == 201
  817. rsrc_rsp = self.client.get('/ldp/test_version')
  818. rsrc_gr = Graph().parse(data=rsrc_rsp.data, format='turtle')
  819. assert len(set(rsrc_gr[: nsc['fcrepo'].hasVersions :])) == 1
  820. info_rsp = self.client.get('/ldp/test_version/fcr:versions')
  821. assert info_rsp.status_code == 200
  822. info_gr = Graph().parse(data=info_rsp.data, format='turtle')
  823. assert len(set(info_gr[: nsc['fcrepo'].hasVersion :])) == 1
  824. self.client.post('/ldp/test_version/fcr:versions')
  825. info2_rsp = self.client.get('/ldp/test_version/fcr:versions')
  826. info2_gr = Graph().parse(data=info2_rsp.data, format='turtle')
  827. assert len(set(info2_gr[: nsc['fcrepo'].hasVersion :])) == 2
  828. def test_version_with_slug(self):
  829. """
  830. Test a version with a slug.
  831. """
  832. self.client.put('/ldp/test_version_slug')
  833. create_rsp = self.client.post('/ldp/test_version_slug/fcr:versions',
  834. headers={'slug' : 'v1'})
  835. new_ver_uri = create_rsp.headers['Location']
  836. assert new_ver_uri == g.webroot + '/test_version_slug/fcr:versions/v1'
  837. info_rsp = self.client.get('/ldp/test_version_slug/fcr:versions')
  838. info_gr = Graph().parse(data=info_rsp.data, format='turtle')
  839. assert info_gr[
  840. URIRef(new_ver_uri) :
  841. nsc['fcrepo'].hasVersionLabel :
  842. Literal('v1')]
  843. def test_dupl_version(self):
  844. """
  845. Make sure that two POSTs with the same slug result in two different
  846. versions.
  847. """
  848. path = '/ldp/test_duplicate_slug'
  849. self.client.put(path)
  850. v1_rsp = self.client.post(path + '/fcr:versions',
  851. headers={'slug' : 'v1'})
  852. v1_uri = v1_rsp.headers['Location']
  853. dup_rsp = self.client.post(path + '/fcr:versions',
  854. headers={'slug' : 'v1'})
  855. dup_uri = dup_rsp.headers['Location']
  856. assert v1_uri != dup_uri
  857. @pytest.mark.skip(
  858. reason='TODO Reverting from version and resurrecting is not fully '
  859. 'functional.')
  860. def test_revert_version(self):
  861. """
  862. Take a version snapshot, update a resource, and then revert to the
  863. previous vresion.
  864. """
  865. rsrc_path = '/ldp/test_revert_version'
  866. payload1 = '<> <urn:demo:p1> <urn:demo:o1> .'
  867. payload2 = '<> <urn:demo:p1> <urn:demo:o2> .'
  868. self.client.put(rsrc_path, headers={
  869. 'content-type': 'text/turtle'}, data=payload1)
  870. self.client.post(
  871. rsrc_path + '/fcr:versions', headers={'slug': 'v1'})
  872. v1_rsp = self.client.get(rsrc_path)
  873. v1_gr = Graph().parse(data=v1_rsp.data, format='turtle')
  874. assert v1_gr[
  875. URIRef(g.webroot + '/test_revert_version')
  876. : URIRef('urn:demo:p1')
  877. : URIRef('urn:demo:o1')
  878. ]
  879. self.client.put(rsrc_path, headers={
  880. 'content-type': 'text/turtle'}, data=payload2)
  881. v2_rsp = self.client.get(rsrc_path)
  882. v2_gr = Graph().parse(data=v2_rsp.data, format='turtle')
  883. assert v2_gr[
  884. URIRef(g.webroot + '/test_revert_version')
  885. : URIRef('urn:demo:p1')
  886. : URIRef('urn:demo:o2')
  887. ]
  888. self.client.patch(rsrc_path + '/fcr:versions/v1')
  889. revert_rsp = self.client.get(rsrc_path)
  890. revert_gr = Graph().parse(data=revert_rsp.data, format='turtle')
  891. assert revert_gr[
  892. URIRef(g.webroot + '/test_revert_version')
  893. : URIRef('urn:demo:p1')
  894. : URIRef('urn:demo:o1')
  895. ]
  896. def test_resurrection(self):
  897. """
  898. Delete and then resurrect a resource.
  899. Make sure that the resource is resurrected to the latest version.
  900. """
  901. path = '/ldp/test_lazarus'
  902. self.client.put(path)
  903. self.client.post(path + '/fcr:versions', headers={'slug': 'v1'})
  904. self.client.put(
  905. path, headers={'content-type': 'text/turtle'},
  906. data=b'<> <urn:demo:p1> <urn:demo:o1> .')
  907. self.client.post(path + '/fcr:versions', headers={'slug': 'v2'})
  908. self.client.put(
  909. path, headers={'content-type': 'text/turtle'},
  910. data=b'<> <urn:demo:p1> <urn:demo:o2> .')
  911. self.client.delete(path)
  912. assert self.client.get(path).status_code == 410
  913. self.client.post(path + '/fcr:tombstone')
  914. laz_data = self.client.get(path).data
  915. laz_gr = Graph().parse(data=laz_data, format='turtle')
  916. assert laz_gr[
  917. URIRef(g.webroot + '/test_lazarus')
  918. : URIRef('urn:demo:p1')
  919. : URIRef('urn:demo:o2')
  920. ]