query.py 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. import logging
  2. from flask import Blueprint, current_app, request, render_template, send_file
  3. from rdflib import URIRef
  4. from rdflib.plugin import PluginException
  5. from lakesuperior import env
  6. from lakesuperior.api import query as query_api
  7. from lakesuperior.dictionaries.namespaces import ns_collection as nsc
  8. from lakesuperior.dictionaries.namespaces import ns_mgr as nsm
  9. from lakesuperior.toolbox import Toolbox
  10. # Query endpoint. raw SPARQL queries exposing the underlying layout can be made
  11. # available. Also convenience methods that allow simple lookups based on simple
  12. # binary comparisons should be added. Binary lookups—maybe?
  13. # N.B All data sources are read-only for this endpoint.
  14. logger = logging.getLogger(__name__)
  15. rdfly = env.app_globals.rdfly
  16. query = Blueprint('query', __name__)
  17. @query.route('/term_search', methods=['GET', 'POST'])
  18. def term_search():
  19. """
  20. Search by entering a search term and optional property and comparison term.
  21. """
  22. operands = (
  23. ('_id', 'Has Type'),
  24. ('_id', 'Matches Term'),
  25. ('=', 'Is Equal To'),
  26. ('!=', 'Is Not Equal To'),
  27. ('<', 'Is Less Than'),
  28. ('>', 'Is Greater Than'),
  29. ('<=', 'Is Less Than Or Equal To'),
  30. ('>=', 'Is Greater Than Or Equal To'),
  31. )
  32. qres = term_list = []
  33. if request.method == 'POST':
  34. # Some magic needed to associate pseudo-array field notation with
  35. # an actual dict. Flask does not fully support this syntax as Rails
  36. # or other frameworks do: https://stackoverflow.com/q/24808660
  37. fnames = ('pred_ns', 'pred', 'op', 'val')
  38. term_list = [
  39. request.form.getlist('{}[]'.format(tn))
  40. for tn in fnames]
  41. # Transpose matrix.
  42. txm = list(zip(*term_list))
  43. logger.info('transposed matrix: {}'.format(txm))
  44. terms = []
  45. for row in txm:
  46. fmt_row = list(row)
  47. ns = fmt_row.pop(0)
  48. fmt_row[0] = nsc[ns][fmt_row[0]] if ns else URIRef(fmt_row[0])
  49. terms.append(fmt_row)
  50. logger.info('Terms: {}'.format(terms))
  51. or_logic = request.form.get('logic') == 'or'
  52. qres = query_api.term_query(terms, or_logic)
  53. def gl(uri):
  54. return uri.replace(nsc['fcres'], '/ldp')
  55. return render_template('term_search_results.html', qres=qres, gl=gl)
  56. else:
  57. return render_template(
  58. 'term_search.html', operands=operands, qres=qres, nsm=nsm)
  59. @query.route('/sparql', methods=['GET', 'POST'])
  60. def sparql():
  61. """
  62. Perform a direct SPARQL query on the underlying triplestore.
  63. :param str qry: SPARQL query string.
  64. """
  65. accept_mimetypes = {
  66. 'text/csv': 'csv',
  67. 'application/sparql-results+json': 'json',
  68. 'application/sparql-results+xml': 'xml',
  69. }
  70. if request.method == 'GET':
  71. return render_template('sparql_query.html', nsm=nsm)
  72. else:
  73. if request.mimetype == 'application/sparql-query':
  74. qstr = request.stream.read()
  75. else:
  76. qstr = request.form['query']
  77. logger.debug('Query: {}'.format(qstr))
  78. match = request.accept_mimetypes.best_match(accept_mimetypes.keys())
  79. fmt = (
  80. accept_mimetypes[match] if match
  81. else request.accept_mimetypes.best)
  82. try:
  83. out_stream = query_api.sparql_query(qstr, fmt)
  84. except PluginException:
  85. return (
  86. 'Unable to serialize results into format {}'.format(fmt), 406)
  87. return send_file(out_stream, mimetype=fmt), 200