translator.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. import logging
  2. from collections import defaultdict
  3. from flask import request
  4. from rdflib.term import URIRef
  5. from lakesuperior.dictionaries.namespaces import ns_collection as nsc
  6. from lakesuperior.store_layouts.rdf.base_rdf_layout import BaseRdfLayout
  7. class Translator:
  8. '''
  9. Utility class to perform translations of strings and their wrappers.
  10. All static methods.
  11. '''
  12. _logger = logging.getLogger(__name__)
  13. @staticmethod
  14. def base_url():
  15. return request.host_url + request.path.split('/')[1]
  16. @staticmethod
  17. def camelcase(word):
  18. '''
  19. Convert a string with underscores with a camel-cased one.
  20. Ripped from https://stackoverflow.com/a/6425628
  21. '''
  22. return ''.join(x.capitalize() or '_' for x in word.split('_'))
  23. @staticmethod
  24. def uuid_to_uri(uuid):
  25. '''Convert a UUID to a URI.
  26. @return URIRef
  27. '''
  28. return URIRef('{}/{}'.format(Translator.base_url(), uuid))
  29. @staticmethod
  30. def uri_to_uuid(uri):
  31. '''Convert an absolute URI (internal or external) to a UUID.
  32. @return string
  33. '''
  34. if uri.startswith(nsc['fcres']):
  35. return str(uri).replace(nsc['fcres'], '')
  36. else:
  37. return str(uri).replace(Translator.base_url(), '')
  38. @staticmethod
  39. def localize_string(s):
  40. '''Convert URIs into URNs in a string using the application base URI.
  41. @param string s Input string.
  42. @return string
  43. '''
  44. return s.replace(Translator.base_url()+'/', str(nsc['fcres']))\
  45. .replace(Translator.base_url(), str(nsc['fcres']))
  46. @staticmethod
  47. def localize_term(uri):
  48. '''
  49. Convert an URI into an URN.
  50. @param rdflib.term.URIRef urn Input URI.
  51. @return rdflib.term.URIRef
  52. '''
  53. Translator._logger.debug('Input URI: {}'.format(uri))
  54. if uri.strip('/') == Translator.base_url():
  55. return BaseRdfLayout.ROOT_NODE_URN
  56. return URIRef(Translator.localize_string(str(uri)))
  57. @staticmethod
  58. def globalize_string(s):
  59. '''Convert URNs into URIs in a string using the application base URI.
  60. @param string s Input string.
  61. @return string
  62. '''
  63. return s.replace(str(nsc['fcres']), Translator.base_url() + '/')
  64. @staticmethod
  65. def globalize_term(urn):
  66. '''
  67. Convert an URN into an URI using the application base URI.
  68. @param rdflib.term.URIRef urn Input URN.
  69. @return rdflib.term.URIRef
  70. '''
  71. if urn == BaseRdfLayout.ROOT_NODE_URN:
  72. urn = nsc['fcres']
  73. return URIRef(Translator.globalize_string(str(urn)))
  74. @staticmethod
  75. def globalize_graph(g):
  76. '''
  77. Globalize a graph.
  78. '''
  79. from lakesuperior.model.ldpr import Ldpr
  80. q = '''
  81. CONSTRUCT {{ ?s ?p ?o . }} WHERE {{
  82. {{
  83. ?s ?p ?o .
  84. FILTER (
  85. STRSTARTS(str(?s), "{0}")
  86. ||
  87. STRSTARTS(str(?o), "{0}")
  88. ||
  89. STRSTARTS(str(?s), "{1}")
  90. ||
  91. STRSTARTS(str(?o), "{1}")
  92. ) .
  93. }}
  94. }}'''.format(nsc['fcres'], BaseRdfLayout.ROOT_NODE_URN)
  95. flt_g = g.query(q)
  96. for t in flt_g:
  97. global_s = Translator.globalize_term(t[0])
  98. global_o = Translator.globalize_term(t[2]) \
  99. if isinstance(t[2], URIRef) \
  100. else t[2]
  101. g.remove(t)
  102. g.add((global_s, t[1], global_o))
  103. return g
  104. @staticmethod
  105. def globalize_rsrc(rsrc):
  106. '''
  107. Globalize a resource.
  108. '''
  109. g = rsrc.graph
  110. urn = rsrc.identifier
  111. global_g = Translator.globalize_graph(g)
  112. global_uri = Translator.globalize_term(urn)
  113. return global_g.resource(global_uri)
  114. @staticmethod
  115. def parse_rfc7240(h_str):
  116. '''
  117. Parse `Prefer` header as per https://tools.ietf.org/html/rfc7240
  118. The `cgi.parse_header` standard method does not work with all possible
  119. use cases for this header.
  120. @param h_str (string) The header(s) as a comma-separated list of Prefer
  121. statements, excluding the `Prefer: ` token.
  122. '''
  123. parsed_hdr = defaultdict(dict)
  124. # Split up headers by comma
  125. hdr_list = [ x.strip() for x in h_str.split(',') ]
  126. for hdr in hdr_list:
  127. parsed_pref = defaultdict(dict)
  128. # Split up tokens by semicolon
  129. token_list = [ token.strip() for token in hdr.split(';') ]
  130. prefer_token = token_list.pop(0).split('=')
  131. prefer_name = prefer_token[0]
  132. # If preference has a '=', it has a value, else none.
  133. if len(prefer_token)>1:
  134. parsed_pref['value'] = prefer_token[1].strip('"')
  135. for param_token in token_list:
  136. # If the token list had a ';' the preference has a parameter.
  137. print('Param token: {}'.format(param_token))
  138. param_parts = [ prm.strip().strip('"') \
  139. for prm in param_token.split('=') ]
  140. param_value = param_parts[1] if len(param_parts) > 1 else None
  141. parsed_pref['parameters'][param_parts[0]] = param_value
  142. parsed_hdr[prefer_name] = parsed_pref
  143. return parsed_hdr