ldpr.py 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. from abc import ABCMeta
  2. from importlib import import_module
  3. import arrow
  4. from rdflib import Graph
  5. from rdflib.namespace import XSD
  6. from rdflib.term import Literal
  7. from lakesuperior.config_parser import config
  8. from lakesuperior.connectors.filesystem_connector import FilesystemConnector
  9. from lakesuperior.core.namespaces import ns_collection as nsc
  10. class Ldpr(metaclass=ABCMeta):
  11. '''LDPR (LDP Resource).
  12. Definition: https://www.w3.org/TR/ldp/#ldpr-resource
  13. This class and related subclasses contain the implementation pieces of
  14. the vanilla LDP specifications. This is extended by the
  15. `lakesuperior.fcrepo.Resource` class.
  16. Inheritance graph: https://www.w3.org/TR/ldp/#fig-ldpc-types
  17. Note: Even though LdpNr (which is a subclass of Ldpr) handles binary files,
  18. it still has an RDF representation in the triplestore. Hence, some of the
  19. RDF-related methods are defined in this class rather than in the LdpRs
  20. class.
  21. Convention notes:
  22. All the methods in this class handle internal UUIDs (URN). Public-facing
  23. URIs are converted from URNs and passed by these methods to the methods
  24. handling HTTP negotiation.
  25. The data passed to the store strategy for processing should be in a graph.
  26. All conversion from request payload strings is done here.
  27. '''
  28. LDP_NR_TYPE = nsc['ldp'].NonRDFSource
  29. LDP_RS_TYPE = nsc['ldp'].RDFSource
  30. store_strategy = config['application']['store']['ldp_rs']['strategy']
  31. ## MAGIC METHODS ##
  32. def __init__(self, uuid):
  33. '''Instantiate an in-memory LDP resource that can be loaded from and
  34. persisted to storage.
  35. @param uuid (string) UUID of the resource.
  36. '''
  37. # Dynamically load the store strategy indicated in the configuration.
  38. store_mod = import_module(
  39. 'lakesuperior.store_strategies.rdf.{}'.format(
  40. self.store_strategy))
  41. store_cls = getattr(store_mod, self._camelcase(self.store_strategy))
  42. self.gs = store_cls()
  43. # Same thing coud be done for the filesystem store strategy, but we
  44. # will keep it simple for now.
  45. self.fs = FilesystemConnector()
  46. self.uuid = uuid
  47. @property
  48. def urn(self):
  49. return nsc['fcres'][self.uuid]
  50. @property
  51. def uri(self):
  52. return self.gs.uuid_to_uri(self.uuid)
  53. @property
  54. def types(self):
  55. '''The LDP types.
  56. @return tuple(rdflib.term.URIRef)
  57. '''
  58. return self.gs.list_types(self.uuid)
  59. ## LDP METHODS ##
  60. def get(self):
  61. '''
  62. https://www.w3.org/TR/ldp/#ldpr-HTTP_GET
  63. '''
  64. ret = self.gs.get_rsrc(self.uuid)
  65. return ret
  66. def post(self, data, format='text/turtle'):
  67. '''
  68. https://www.w3.org/TR/ldp/#ldpr-HTTP_POST
  69. '''
  70. # @TODO Use gunicorn to get request timestamp.
  71. ts = Literal(arrow.utcnow(), datatype=XSD.dateTime)
  72. g = Graph()
  73. g.parse(data=data, format=format, publicID=self.urn)
  74. data.add((self.urn, nsc['fedora'].lastUpdated, ts))
  75. data.add((self.urn, nsc['fedora'].lastUpdatedBy,
  76. Literal('BypassAdmin')))
  77. self.gs.create_rsrc(self.urn, g, ts)
  78. self.gs.conn.store.commit()
  79. def put(self, data, format='text/turtle'):
  80. '''
  81. https://www.w3.org/TR/ldp/#ldpr-HTTP_PUT
  82. '''
  83. # @TODO Use gunicorn to get request timestamp.
  84. ts = Literal(arrow.utcnow(), datatype=XSD.dateTime)
  85. g = Graph()
  86. g.parse(data=data, format=format, publicID=self.urn)
  87. g.add((self.urn, nsc['fedora'].lastUpdated, ts))
  88. g.add((self.urn, nsc['fedora'].lastUpdatedBy,
  89. Literal('BypassAdmin')))
  90. self.gs.create_or_replace_rsrc(self.urn, g, ts)
  91. self.gs.conn.store.commit()
  92. def delete(self):
  93. '''
  94. https://www.w3.org/TR/ldp/#ldpr-HTTP_DELETE
  95. '''
  96. self.gs.delete_rsrc(self.urn, commit=True)
  97. ## PROTECTED METHODS ##
  98. def _create_containment_rel(self):
  99. '''Find the closest parent in the path indicated by the UUID and
  100. establish a containment triple.
  101. E.g. If ONLY urn:res:a exist:
  102. - If a/b is being created, a becomes container of a/b.
  103. - If a/b/c/d is being created, a becomes container of a/b/c/d.
  104. - If e is being created, the root node becomes container of e.
  105. (verify if this is useful or necessary in any way).
  106. - If only a and a/b/c/d exist, and therefore a contains a/b/c/d, and
  107. a/b is created:
  108. - a ceases to be the container of a/b/c/d
  109. - a becomes container of a/b
  110. - a/b becomes container of a/b/c/d.
  111. '''
  112. if self.gs.list_containment_statements(self.urn):
  113. pass
  114. def _camelcase(self, word):
  115. '''
  116. Convert a string with underscores with a camel-cased one.
  117. Ripped from https://stackoverflow.com/a/6425628
  118. '''
  119. return ''.join(x.capitalize() or '_' for x in word.split('_'))
  120. class LdpRs(Ldpr):
  121. '''LDP-RS (LDP RDF source).
  122. Definition: https://www.w3.org/TR/ldp/#ldprs
  123. '''
  124. base_types = {
  125. nsc['ldp'].RDFSource
  126. }
  127. def patch(self, data):
  128. '''
  129. https://www.w3.org/TR/ldp/#ldpr-HTTP_PATCH
  130. '''
  131. ts = Literal(arrow.utcnow(), datatype=XSD.dateTime)
  132. self.gs.patch_rsrc(self.urn, data, ts)
  133. self.gs.ds.add((self.urn, nsc['fedora'].lastUpdated, ts))
  134. self.gs.ds.add((self.urn, nsc['fedora'].lastUpdatedBy,
  135. Literal('BypassAdmin')))
  136. self.gs.conn.store.commit()
  137. class LdpNr(LdpRs):
  138. '''LDP-NR (Non-RDF Source).
  139. Definition: https://www.w3.org/TR/ldp/#ldpnr
  140. '''
  141. pass
  142. class Ldpc(LdpRs):
  143. '''LDPC (LDP Container).'''
  144. def __init__(self, uuid):
  145. super().__init__(uuid)
  146. self.base_types.update({
  147. nsc['ldp'].Container,
  148. })
  149. class LdpBc(Ldpc):
  150. '''LDP-BC (LDP Basic Container).'''
  151. pass
  152. class LdpDc(Ldpc):
  153. '''LDP-DC (LDP Direct Container).'''
  154. def __init__(self, uuid):
  155. super().__init__(uuid)
  156. self.base_types.update({
  157. nsc['ldp'].DirectContainer,
  158. })
  159. class LdpIc(Ldpc):
  160. '''LDP-IC (LDP Indirect Container).'''
  161. def __init__(self, uuid):
  162. super().__init__(uuid)
  163. self.base_types.update({
  164. nsc['ldp'].IndirectContainer,
  165. })