123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220 |
- from copy import deepcopy
- from rdflib import Graph
- from rdflib.namespace import RDF, XSD
- from rdflib.plugins.sparql.parser import parseUpdate
- from rdflib.term import URIRef, Literal, Variable
- from lakesuperior.dictionaries.namespaces import ns_collection as nsc
- from lakesuperior.dictionaries.srv_mgd_terms import srv_mgd_subjects, \
- srv_mgd_predicates, srv_mgd_types
- from lakesuperior.model.ldpr import Ldpr, transactional, must_exist, \
- ResourceNotExistsError, ServerManagedTermError
- from lakesuperior.util.translator import Translator
- class LdpRs(Ldpr):
- '''LDP-RS (LDP RDF source).
- Definition: https://www.w3.org/TR/ldp/#ldprs
- '''
- base_types = {
- nsc['ldp'].RDFSource
- }
- std_headers = {
- 'Accept-Post' : {
- 'text/turtle',
- 'text/rdf+n3',
- 'text/n3',
- 'application/rdf+xml',
- 'application/n-triples',
- 'application/ld+json',
- 'multipart/form-data',
- 'application/sparql-update',
- },
- 'Accept-Patch' : {
- 'application/sparql-update',
- },
- }
- def head(self):
- '''
- Return values for the headers.
- '''
- headers = self.rdfly.headers
- for t in self.ldp_types:
- headers['Link'].append('{};rel="type"'.format(t.identifier.n3()))
- return headers
- def get(self, inbound=False, children=True, srv_mgd=True):
- '''
- https://www.w3.org/TR/ldp/#ldpr-HTTP_GET
- '''
- im_rsrc = self.rdfly.out_rsrc(inbound)
- if not len(im_rsrc.graph):
- raise ResourceNotExistsError()
- return Translator.globalize_rsrc(im_rsrc)
- @transactional
- def post(self, data, format='text/turtle'):
- '''
- https://www.w3.org/TR/ldp/#ldpr-HTTP_POST
- Perform a POST action after a valid resource URI has been found.
- '''
- g = Graph()
- g.parse(data=data, format=format, publicID=self.urn)
- self._check_mgd_terms_rdf(g)
- for t in self.base_types:
- g.add((self.urn, RDF.type, t))
- self.rdfly.create_rsrc(g)
- self._set_containment_rel()
- @transactional
- def put(self, data, format='text/turtle'):
- '''
- https://www.w3.org/TR/ldp/#ldpr-HTTP_PUT
- '''
- g = Graph()
- g.parse(data=data, format=format, publicID=self.urn)
- self._check_mgd_terms_rdf(g)
- for t in self.base_types:
- g.add((self.urn, RDF.type, t))
- self.rdfly.create_or_replace_rsrc(g)
- self._set_containment_rel()
- @transactional
- @must_exist
- def patch(self, data):
- '''
- https://www.w3.org/TR/ldp/#ldpr-HTTP_PATCH
- '''
- self._check_mgd_terms_sparql(data)
- self.rdfly.patch_rsrc(data)
- ## PROTECTED METHODS ##
- def _check_mgd_terms_rdf(self, g):
- '''
- Check whether server-managed terms are in a RDF payload.
- '''
- offending_subjects = set(g.subjects()) & srv_mgd_subjects
- if offending_subjects:
- raise ServerManagedTermError('Some subjects in RDF payload '
- 'are server managed and cannot be modified: {}'
- .format(' , '.join(offending_subjects)))
- offending_predicates = set(g.predicates()) & srv_mgd_predicates
- if offending_predicates:
- raise ServerManagedTermError('Some predicates in RDF payload '
- 'are server managed and cannot be modified: {}'
- .format(' , '.join(offending_predicates)))
- offending_types = set(g.objects(predicate=RDF.type)) & srv_mgd_types
- if offending_types:
- raise ServerManagedTermError('Some RDF types in RDF payload '
- 'are server managed and cannot be modified: {}'
- .format(' , '.join(offending_types)))
- def _check_mgd_terms_sparql(self, q):
- '''
- Parse tokens in update query and verify that none of the terms being
- modified is server-managed.
- The only reasonable way to do this is to perform the query on a copy
- and verify if any of the server managed terms is in the delta. If it
- is, it means that some server-managed term is being modified and
- an error should be raised.
- NOTE: This only checks if a server-managed term is effectively being
- modified. If a server-managed term is present in the query but does not
- cause any change in the updated resource, no error is raised.
- '''
- before_test = self.rdfly.extract_imr().graph
- after_test = deepcopy(before_test)
- after_test.update(q)
- delta = before_test ^ after_test
- self._logger.info('Delta: {}'.format(delta.serialize(format='turtle')
- .decode('utf8')))
- for s,p,o in delta:
- if s in srv_mgd_subjects:
- raise ServerManagedTermError(
- 'Subject {} is server managed and cannot be modified.'
- .format(s))
- if p in srv_mgd_predicates:
- raise ServerManagedTermError(
- 'Predicate {} is server managed and cannot be modified.'
- .format(p))
- if p == RDF.type and o in srv_mgd_types:
- raise ServerManagedTermError(
- 'RDF type {} is server managed and cannot be modified.'
- .format(o))
- class Ldpc(LdpRs):
- '''LDPC (LDP Container).'''
- def __init__(self, uuid):
- super().__init__(uuid)
- self.base_types.update({
- nsc['ldp'].Container,
- })
- class LdpBc(Ldpc):
- '''LDP-BC (LDP Basic Container).'''
- def __init__(self, uuid):
- super().__init__(uuid)
- self.base_types.update({
- nsc['ldp'].BasicContainer,
- })
- class LdpDc(Ldpc):
- '''LDP-DC (LDP Direct Container).'''
- def __init__(self, uuid):
- super().__init__(uuid)
- self.base_types.update({
- nsc['ldp'].DirectContainer,
- })
- class LdpIc(Ldpc):
- '''LDP-IC (LDP Indirect Container).'''
- def __init__(self, uuid):
- super().__init__(uuid)
- self.base_types.update({
- nsc['ldp'].IndirectContainer,
- })
|