import pytest import uuid from hashlib import sha1 from flask import g from rdflib import Graph from rdflib.compare import isomorphic from rdflib.namespace import RDF from rdflib.term import Literal, URIRef from lakesuperior.dictionaries.namespaces import ns_collection as nsc from lakesuperior.model.ldpr import Ldpr @pytest.fixture(scope='module') def random_uuid(): return str(uuid.uuid4()) @pytest.mark.usefixtures('client_class') @pytest.mark.usefixtures('db') class TestLdp: ''' Test HTTP interaction with LDP endpoint. ''' def test_get_root_node(self): ''' Get the root node from two different endpoints. The test triplestore must be initialized, hence the `db` fixture. ''' ldp_resp = self.client.get('/ldp') rest_resp = self.client.get('/rest') assert ldp_resp.status_code == 200 assert rest_resp.status_code == 200 def test_put_empty_resource(self, random_uuid): ''' Check response headers for a PUT operation with empty payload. ''' resp = self.client.put('/ldp/new_resource') assert resp.status_code == 201 assert resp.data == bytes( '{}/new_resource'.format(g.webroot), 'utf-8') def test_put_existing_resource(self, random_uuid): ''' Trying to PUT an existing resource should return a 204 if the payload is empty. ''' path = '/ldp/nonidempotent01' put1_resp = self.client.put(path) assert put1_resp.status_code == 201 assert self.client.get(path).status_code == 200 put2_resp = self.client.put(path) assert put2_resp.status_code == 204 assert put2_resp.data == b'' def test_put_tree(self, client): ''' PUT a resource with several path segments. The test should create intermediate path segments that are not accessible to PUT or POST. ''' path = '/ldp/test_tree/a/b/c/d/e/f/g' self.client.put(path) assert self.client.get(path).status_code == 200 assert self.client.put('/ldp/test_tree/a').status_code == 409 assert self.client.post('/ldp/test_tree/a').status_code == 409 def test_put_nested_tree(self, client): ''' Verify that containment is set correctly in nested hierarchies. First put a new hierarchy and verify that the root node is its container; then put another hierarchy under it and verify that the first hierarchy is the container of the second one. ''' uuid1 = 'test_nested_tree/a/b/c/d' uuid2 = uuid1 + '/e/f/g' path1 = '/ldp/' + uuid1 path2 = '/ldp/' + uuid2 self.client.put(path1) cont1_data = self.client.get('/ldp').data gr1 = Graph().parse(data=cont1_data, format='turtle') assert gr1[ URIRef(g.webroot + '/') : nsc['ldp'].contains : \ URIRef(g.webroot + '/' + uuid1) ] self.client.put(path2) cont2_data = self.client.get(path1).data gr2 = Graph().parse(data=cont2_data, format='turtle') assert gr2[ URIRef(g.webroot + '/' + uuid1) : \ nsc['ldp'].contains : \ URIRef(g.webroot + '/' + uuid2) ] def test_put_ldp_rs(self, client): ''' PUT a resource with RDF payload and verify. ''' with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f: self.client.put('/ldp/ldprs01', data=f, content_type='text/turtle') resp = self.client.get('/ldp/ldprs01', headers={'accept' : 'text/turtle'}) assert resp.status_code == 200 gr = Graph().parse(data=resp.data, format='text/turtle') assert URIRef('http://vocab.getty.edu/ontology#Subject') in \ gr.objects(None, RDF.type) def test_put_ldp_nr(self, rnd_img): ''' PUT a resource with binary payload and verify checksums. ''' rnd_img['content'].seek(0) resp = self.client.put('/ldp/ldpnr01', data=rnd_img['content'], headers={ 'Content-Disposition' : 'attachment; filename={}'.format( rnd_img['filename'])}) assert resp.status_code == 201 resp = self.client.get('/ldp/ldpnr01', headers={'accept' : 'image/png'}) assert resp.status_code == 200 assert sha1(resp.data).hexdigest() == rnd_img['hash'] def test_put_mismatched_ldp_rs(self, rnd_img): ''' Verify MIME type / LDP mismatch. PUT a LDP-RS, then PUT a LDP-NR on the same location and verify it fails. ''' path = '/ldp/' + str(uuid.uuid4()) rnd_img['content'].seek(0) ldp_nr_resp = self.client.put(path, data=rnd_img['content'], headers={ 'Content-Disposition' : 'attachment; filename={}'.format( rnd_img['filename'])}) assert ldp_nr_resp.status_code == 201 with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f: ldp_rs_resp = self.client.put(path, data=f, content_type='text/turtle') assert ldp_rs_resp.status_code == 415 def test_put_mismatched_ldp_nr(self, rnd_img): ''' Verify MIME type / LDP mismatch. PUT a LDP-NR, then PUT a LDP-RS on the same location and verify it fails. ''' path = '/ldp/' + str(uuid.uuid4()) with open('tests/data/marcel_duchamp_single_subject.ttl', 'rb') as f: ldp_rs_resp = self.client.put(path, data=f, content_type='text/turtle') assert ldp_rs_resp.status_code == 201 rnd_img['content'].seek(0) ldp_nr_resp = self.client.put(path, data=rnd_img['content'], headers={ 'Content-Disposition' : 'attachment; filename={}'.format( rnd_img['filename'])}) assert ldp_nr_resp.status_code == 415 def test_post_resource(self, client): ''' Check response headers for a POST operation with empty payload. ''' res = self.client.post('/ldp/') assert res.status_code == 201 assert 'Location' in res.headers def test_post_slug(self): ''' Verify that a POST with slug results in the expected URI only if the resource does not exist already. ''' slug01_resp = self.client.post('/ldp', headers={'slug' : 'slug01'}) assert slug01_resp.status_code == 201 assert slug01_resp.headers['location'] == \ g.webroot + '/slug01' slug02_resp = self.client.post('/ldp', headers={'slug' : 'slug01'}) assert slug02_resp.status_code == 201 assert slug02_resp.headers['location'] != \ g.webroot + '/slug01' def test_post_404(self): ''' Verify that a POST to a non-existing parent results in a 404. ''' assert self.client.post('/ldp/{}'.format(uuid.uuid4()))\ .status_code == 404 def test_post_409(self, rnd_img): ''' Verify that you cannot POST to a binary resource. ''' rnd_img['content'].seek(0) self.client.put('/ldp/post_409', data=rnd_img['content'], headers={ 'Content-Disposition' : 'attachment; filename={}'.format( rnd_img['filename'])}) assert self.client.post('/ldp/post_409').status_code == 409 def test_patch(self): ''' Test patching a resource. ''' path = '/ldp/test_patch01' self.client.put(path) uri = g.webroot + '/test_patch01' with open('tests/data/sparql_update/simple_insert.sparql') as data: resp = self.client.patch(path, data=data, headers={'content-type' : 'application/sparql-update'}) assert resp.status_code == 204 resp = self.client.get(path) gr = Graph().parse(data=resp.data, format='text/turtle') assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Hello') ] self.client.patch(path, data=open('tests/data/sparql_update/delete+insert+where.sparql'), headers={'content-type' : 'application/sparql-update'}) resp = self.client.get(path) gr = Graph().parse(data=resp.data, format='text/turtle') assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Ciao') ] def test_patch_ldp_nr_metadata(self): ''' Test patching a LDP-NR metadata resource, both from the fcr:metadata and the resource URIs. ''' path = '/ldp/ldpnr01' with open('tests/data/sparql_update/simple_insert.sparql') as data: self.client.patch(path + '/fcr:metadata', data=data, headers={'content-type' : 'application/sparql-update'}) resp = self.client.get(path + '/fcr:metadata') assert resp.status_code == 200 uri = g.webroot + '/ldpnr01' gr = Graph().parse(data=resp.data, format='text/turtle') assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Hello') ] with open( 'tests/data/sparql_update/delete+insert+where.sparql') as data: patch_resp = self.client.patch(path, data=data, headers={'content-type' : 'application/sparql-update'}) assert patch_resp.status_code == 204 resp = self.client.get(path + '/fcr:metadata') assert resp.status_code == 200 gr = Graph().parse(data=resp.data, format='text/turtle') assert gr[ URIRef(uri) : nsc['dc'].title : Literal('Ciao') ] def test_patch_ldp_nr(self, rnd_img): ''' Verify that a PATCH using anything other than an `application/sparql-update` MIME type results in an error. ''' rnd_img['content'].seek(0) resp = self.client.patch('/ldp/ldpnr01/fcr:metadata', data=rnd_img, headers={'content-type' : 'image/jpeg'}) assert resp.status_code == 415 def test_delete(self): ''' Test delete response codes. ''' create_resp = self.client.put('/ldp/test_delete01') delete_resp = self.client.delete('/ldp/test_delete01') assert delete_resp.status_code == 204 bogus_delete_resp = self.client.delete('/ldp/test_delete101') assert bogus_delete_resp.status_code == 404 def test_tombstone(self): ''' Test tombstone behaviors. ''' tstone_resp = self.client.get('/ldp/test_delete01') assert tstone_resp.status_code == 410 assert tstone_resp.headers['Link'] == \ '<{}/test_delete01/fcr:tombstone>; rel="hasTombstone"'\ .format(g.webroot) tstone_path = '/ldp/test_delete01/fcr:tombstone' assert self.client.get(tstone_path).status_code == 405 assert self.client.put(tstone_path).status_code == 405 assert self.client.post(tstone_path).status_code == 405 assert self.client.delete(tstone_path).status_code == 204 assert self.client.get('/ldp/test_delete01').status_code == 404 def test_delete_recursive(self): ''' Test response codes for resources deleted recursively and their tombstones. ''' self.client.put('/ldp/test_delete_recursive01') self.client.put('/ldp/test_delete_recursive01/a') self.client.delete('/ldp/test_delete_recursive01') tstone_resp = self.client.get('/ldp/test_delete_recursive01') assert tstone_resp.status_code == 410 assert tstone_resp.headers['Link'] == \ '<{}/test_delete_recursive01/fcr:tombstone>; rel="hasTombstone"'\ .format(g.webroot) child_tstone_resp = self.client.get('/ldp/test_delete_recursive01/a') assert child_tstone_resp.status_code == tstone_resp.status_code assert 'Link' not in child_tstone_resp.headers.keys() @pytest.mark.usefixtures('client_class') @pytest.mark.usefixtures('db') class TestPrefHeader: ''' Test various combinations of `Prefer` header. ''' @pytest.fixture(scope='class') def cont_structure(self): ''' Create a container structure to be used for subsequent requests. ''' parent_path = '/ldp/test_parent' self.client.put(parent_path) self.client.put(parent_path + '/child1') self.client.put(parent_path + '/child2') self.client.put(parent_path + '/child3') return { 'path' : parent_path, 'response' : self.client.get(parent_path), 'subject' : URIRef(g.webroot + '/test_parent') } def test_put_prefer_handling(self, random_uuid): ''' Trying to PUT an existing resource should: - Return a 204 if the payload is empty - Return a 204 if the payload is RDF, server-managed triples are included and the 'Prefer' header is set to 'handling=lenient' - Return a 412 (ServerManagedTermError) if the payload is RDF, server-managed triples are included and handling is set to 'strict' ''' path = '/ldp/put_pref_header01' assert self.client.put(path).status_code == 201 assert self.client.get(path).status_code == 200 assert self.client.put(path).status_code == 204 with open('tests/data/rdf_payload_w_srv_mgd_trp.ttl', 'rb') as f: rsp_len = self.client.put( path, headers={ 'Prefer' : 'handling=lenient', 'Content-Type' : 'text/turtle', }, data=f ) assert rsp_len.status_code == 204 with open('tests/data/rdf_payload_w_srv_mgd_trp.ttl', 'rb') as f: rsp_strict = self.client.put( path, headers={ 'Prefer' : 'handling=strict', 'Content-Type' : 'text/turtle', }, data=f ) assert rsp_strict.status_code == 412 def test_embed_children(self, cont_structure): ''' verify the "embed children" prefer header. ''' parent_path = cont_structure['path'] cont_resp = cont_structure['response'] cont_subject = cont_structure['subject'] minimal_resp = self.client.get(parent_path, headers={ 'Prefer' : 'return=minimal', }) incl_embed_children_resp = self.client.get(parent_path, headers={ 'Prefer' : 'return=representation; include={}'\ .format(Ldpr.EMBED_CHILD_RES_URI), }) omit_embed_children_resp = self.client.get(parent_path, headers={ 'Prefer' : 'return=representation; omit={}'\ .format(Ldpr.EMBED_CHILD_RES_URI), }) assert omit_embed_children_resp.data == cont_resp.data incl_gr = Graph().parse( data=incl_embed_children_resp.data, format='turtle') omit_gr = Graph().parse( data=omit_embed_children_resp.data, format='turtle') children = set(incl_gr[cont_subject : nsc['ldp'].contains]) assert len(children) == 3 children = set(incl_gr[cont_subject : nsc['ldp'].contains]) for child_uri in children: assert set(incl_gr[ child_uri : : ]) assert not set(omit_gr[ child_uri : : ]) def test_return_children(self, cont_structure): ''' verify the "return children" prefer header. ''' parent_path = cont_structure['path'] cont_resp = cont_structure['response'] cont_subject = cont_structure['subject'] incl_children_resp = self.client.get(parent_path, headers={ 'Prefer' : 'return=representation; include={}'\ .format(Ldpr.RETURN_CHILD_RES_URI), }) omit_children_resp = self.client.get(parent_path, headers={ 'Prefer' : 'return=representation; omit={}'\ .format(Ldpr.RETURN_CHILD_RES_URI), }) default_gr = Graph().parse(data=cont_resp.data, format='turtle') incl_gr = Graph().parse(data=incl_children_resp.data, format='turtle') omit_gr = Graph().parse(data=omit_children_resp.data, format='turtle') assert isomorphic(incl_gr, default_gr) children = incl_gr[cont_subject : nsc['ldp'].contains] for child_uri in children: assert not omit_gr[ cont_subject : nsc['ldp'].contains : child_uri ] def test_inbound_rel(self, cont_structure): ''' verify the "inboud relationships" prefer header. ''' parent_path = cont_structure['path'] cont_resp = cont_structure['response'] cont_subject = cont_structure['subject'] incl_inbound_resp = self.client.get(parent_path, headers={ 'Prefer' : 'return=representation; include={}'\ .format(Ldpr.RETURN_INBOUND_REF_URI), }) omit_inbound_resp = self.client.get(parent_path, headers={ 'Prefer' : 'return=representation; omit={}'\ .format(Ldpr.RETURN_INBOUND_REF_URI), }) default_gr = Graph().parse(data=cont_resp.data, format='turtle') incl_gr = Graph().parse(data=incl_inbound_resp.data, format='turtle') omit_gr = Graph().parse(data=omit_inbound_resp.data, format='turtle') assert isomorphic(omit_gr, default_gr) assert set(incl_gr[ : : cont_subject ]) assert not set(omit_gr[ : : cont_subject ]) def test_srv_mgd_triples(self, cont_structure): ''' verify the "server managed triples" prefer header. ''' parent_path = cont_structure['path'] cont_resp = cont_structure['response'] cont_subject = cont_structure['subject'] incl_srv_mgd_resp = self.client.get(parent_path, headers={ 'Prefer' : 'return=representation; include={}'\ .format(Ldpr.RETURN_SRV_MGD_RES_URI), }) omit_srv_mgd_resp = self.client.get(parent_path, headers={ 'Prefer' : 'return=representation; omit={}'\ .format(Ldpr.RETURN_SRV_MGD_RES_URI), }) assert incl_srv_mgd_resp.data == cont_resp.data incl_gr = Graph().parse(data=incl_srv_mgd_resp.data, format='turtle') omit_gr = Graph().parse(data=omit_srv_mgd_resp.data, format='turtle') for pred in { nsc['fcrepo'].created, nsc['fcrepo'].createdBy, nsc['fcrepo'].lastModified, nsc['fcrepo'].lastModifiedBy, nsc['ldp'].contains, }: assert set(incl_gr[ cont_subject : pred : ]) assert not set(omit_gr[ cont_subject : pred : ]) for type in { nsc['fcrepo'].Resource, nsc['ldp'].Container, nsc['ldp'].Resource, }: assert incl_gr[ cont_subject : RDF.type : type ] assert not omit_gr[ cont_subject : RDF.type : type ] def test_delete_no_tstone(self): ''' Test the `no-tombstone` Prefer option. ''' self.client.put('/ldp/test_delete_no_tstone01') self.client.put('/ldp/test_delete_no_tstone01/a') self.client.delete('/ldp/test_delete_no_tstone01', headers={ 'prefer' : 'no-tombstone'}) resp = self.client.get('/ldp/test_delete_no_tstone01') assert resp.status_code == 404 child_resp = self.client.get('/ldp/test_delete_no_tstone01/a') assert child_resp.status_code == 404 @pytest.mark.usefixtures('client_class') @pytest.mark.usefixtures('db') class TestVersion: ''' Test version creation, retrieval and deletion. ''' def test_create_versions(self): ''' Test that POSTing multiple times to fcr:versions creates the 'hasVersions' triple and yields multiple version snapshots. ''' self.client.put('/ldp/test_version') create_rsp = self.client.post('/ldp/test_version/fcr:versions') assert create_rsp.status_code == 201 rsrc_rsp = self.client.get('/ldp/test_version') rsrc_gr = Graph().parse(data=rsrc_rsp.data, format='turtle') assert len(set(rsrc_gr[: nsc['fcrepo'].hasVersions :])) == 1 info_rsp = self.client.get('/ldp/test_version/fcr:versions') assert info_rsp.status_code == 200 info_gr = Graph().parse(data=info_rsp.data, format='turtle') assert len(set(info_gr[: nsc['fcrepo'].hasVersion :])) == 1 self.client.post('/ldp/test_version/fcr:versions') info2_rsp = self.client.get('/ldp/test_version/fcr:versions') info2_gr = Graph().parse(data=info2_rsp.data, format='turtle') assert len(set(info2_gr[: nsc['fcrepo'].hasVersion :])) == 2 def test_version_with_slug(self): ''' Test a version with a slug. ''' self.client.put('/ldp/test_version_slug') create_rsp = self.client.post('/ldp/test_version_slug/fcr:versions', headers={'slug' : 'v1'}) new_ver_uri = create_rsp.headers['Location'] assert new_ver_uri == g.webroot + '/test_version_slug/fcr:versions/v1' info_rsp = self.client.get('/ldp/test_version_slug/fcr:versions') info_gr = Graph().parse(data=info_rsp.data, format='turtle') assert info_gr[ URIRef(new_ver_uri) : nsc['fcrepo'].hasVersionLabel : Literal('v1')]