ldp_factory.py 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. import logging
  2. from pprint import pformat
  3. import rdflib
  4. from flask import current_app, g
  5. from rdflib import Graph
  6. from rdflib.resource import Resource
  7. from rdflib.namespace import RDF
  8. from lakesuperior import model
  9. from lakesuperior.model.generic_resource import PathSegment
  10. from lakesuperior.dictionaries.namespaces import ns_collection as nsc
  11. from lakesuperior.exceptions import (IncompatibleLdpTypeError,
  12. InvalidResourceError, ResourceNotExistsError)
  13. class LdpFactory:
  14. '''
  15. Generate LDP instances.
  16. The instance classes are based on provided client data or on stored data.
  17. '''
  18. LDP_NR_TYPE = nsc['ldp'].NonRDFSource
  19. LDP_RS_TYPE = nsc['ldp'].RDFSource
  20. _logger = logging.getLogger(__name__)
  21. @staticmethod
  22. def from_stored(uid, repr_opts={}, **kwargs):
  23. '''
  24. Create an instance for retrieval purposes.
  25. This factory method creates and returns an instance of an LDPR subclass
  26. based on information that needs to be queried from the underlying
  27. graph store.
  28. N.B. The resource must exist.
  29. @param uid UID of the instance.
  30. '''
  31. #__class__._logger.info('Retrieving stored resource: {}'.format(uid))
  32. imr_urn = nsc['fcres'][uid]
  33. rsrc_meta = current_app.rdfly.get_metadata(uid)
  34. __class__._logger.debug('Extracted metadata: {}'.format(
  35. pformat(set(rsrc_meta.graph))))
  36. rdf_types = set(rsrc_meta.graph[imr_urn : RDF.type])
  37. if __class__.LDP_NR_TYPE in rdf_types:
  38. __class__._logger.info('Resource is a LDP-NR.')
  39. rsrc = model.ldp_nr.LdpNr(uid, repr_opts, **kwargs)
  40. elif __class__.LDP_RS_TYPE in rdf_types:
  41. __class__._logger.info('Resource is a LDP-RS.')
  42. rsrc = model.ldp_rs.LdpRs(uid, repr_opts, **kwargs)
  43. elif nsc['fcsystem']['PathSegment'] in rdf_types:
  44. return PathSegment(uid)
  45. else:
  46. raise ResourceNotExistsError(uid)
  47. # Sneak in the already extracted metadata to save a query.
  48. rsrc._metadata = rsrc_meta
  49. return rsrc
  50. @staticmethod
  51. def from_provided(uid, content_length, mimetype, stream, **kwargs):
  52. '''
  53. Determine LDP type from request content.
  54. @param uid (string) UID of the resource to be created or updated.
  55. @param content_length (int) The provided content length.
  56. @param mimetype (string) The provided content MIME type.
  57. @param stream (IOStream) The provided data stream. This can be RDF or
  58. non-RDF content.
  59. '''
  60. urn = nsc['fcres'][uid]
  61. logger = __class__._logger
  62. if not content_length:
  63. # Create empty LDPC.
  64. logger.info('No data received in request. '
  65. 'Creating empty container.')
  66. inst = model.ldp_rs.Ldpc(
  67. uid, provided_imr=Resource(Graph(), urn), **kwargs)
  68. elif __class__.is_rdf_parsable(mimetype):
  69. # Create container and populate it with provided RDF data.
  70. input_rdf = stream.read()
  71. provided_gr = Graph().parse(data=input_rdf,
  72. format=mimetype, publicID=urn)
  73. #logger.debug('Provided graph: {}'.format(
  74. # pformat(set(provided_gr))))
  75. local_gr = g.tbox.localize_graph(provided_gr)
  76. #logger.debug('Parsed local graph: {}'.format(
  77. # pformat(set(local_gr))))
  78. provided_imr = Resource(local_gr, urn)
  79. # Determine whether it is a basic, direct or indirect container.
  80. Ldpr = model.ldpr.Ldpr
  81. if Ldpr.MBR_RSRC_URI in local_gr.predicates() and \
  82. Ldpr.MBR_REL_URI in local_gr.predicates():
  83. if Ldpr.INS_CNT_REL_URI in local_gr.predicates():
  84. cls = model.ldp_rs.LdpIc
  85. else:
  86. cls = model.ldp_rs.LdpDc
  87. else:
  88. cls = model.ldp_rs.Ldpc
  89. inst = cls(uid, provided_imr=provided_imr, **kwargs)
  90. # Make sure we are not updating an LDP-RS with an LDP-NR.
  91. if inst.is_stored and __class__.LDP_NR_TYPE in inst.ldp_types:
  92. raise IncompatibleLdpTypeError(uid, mimetype)
  93. if kwargs.get('handling', 'strict') != 'none':
  94. inst._check_mgd_terms(inst.provided_imr.graph)
  95. else:
  96. # Create a LDP-NR and equip it with the binary file provided.
  97. provided_imr = Resource(Graph(), urn)
  98. inst = model.ldp_nr.LdpNr(uid, stream=stream, mimetype=mimetype,
  99. provided_imr=provided_imr, **kwargs)
  100. # Make sure we are not updating an LDP-NR with an LDP-RS.
  101. if inst.is_stored and __class__.LDP_RS_TYPE in inst.ldp_types:
  102. raise IncompatibleLdpTypeError(uid, mimetype)
  103. logger.info('Creating resource of type: {}'.format(
  104. inst.__class__.__name__))
  105. try:
  106. types = inst.types
  107. except:
  108. types = set()
  109. if nsc['fcrepo'].Pairtree in types:
  110. raise InvalidResourceError(inst.uid, 'Resource {} is a Pairtree.')
  111. return inst
  112. @staticmethod
  113. def is_rdf_parsable(mimetype):
  114. '''
  115. Checks whether a MIME type support RDF parsing by a RDFLib plugin.
  116. @param mimetype (string) MIME type to check.
  117. '''
  118. try:
  119. rdflib.plugin.get(mimetype, rdflib.parser.Parser)
  120. except rdflib.plugin.PluginException:
  121. return False
  122. else:
  123. return True
  124. @staticmethod
  125. def is_rdf_serializable(mimetype):
  126. '''
  127. Checks whether a MIME type support RDF serialization by a RDFLib plugin
  128. @param mimetype (string) MIME type to check.
  129. '''
  130. try:
  131. rdflib.plugin.get(mimetype, rdflib.serializer.Serializer)
  132. except rdflib.plugin.PluginException:
  133. return False
  134. else:
  135. return True