123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103 |
- import logging
- from flask import Blueprint, current_app, request, render_template, send_file
- from rdflib import URIRef
- from rdflib.plugin import PluginException
- from lakesuperior import env
- from lakesuperior.api import query as query_api
- from lakesuperior.dictionaries.namespaces import ns_collection as nsc
- from lakesuperior.dictionaries.namespaces import ns_mgr as nsm
- from lakesuperior.toolbox import Toolbox
- # Query endpoint. raw SPARQL queries exposing the underlying layout can be made
- # available. Also convenience methods that allow simple lookups based on simple
- # binary comparisons should be added. Binary lookups—maybe?
- # N.B All data sources are read-only for this endpoint.
- logger = logging.getLogger(__name__)
- rdfly = env.app_globals.rdfly
- query = Blueprint('query', __name__)
- @query.route('/term_search', methods=['GET', 'POST'])
- def term_search():
- """
- Search by entering a search term and optional property and comparison term.
- """
- operands = (
- ('_id', 'Has Type'),
- ('_id', 'Matches Term'),
- ('=', 'Is Equal To'),
- ('!=', 'Is Not Equal To'),
- ('<', 'Is Less Than'),
- ('>', 'Is Greater Than'),
- ('<=', 'Is Less Than Or Equal To'),
- ('>=', 'Is Greater Than Or Equal To'),
- )
- qres = term_list = []
- if request.method == 'POST':
- # Some magic needed to associate pseudo-array field notation with
- # an actual dict. Flask does not fully support this syntax as Rails
- # or other frameworks do: https://stackoverflow.com/q/24808660
- fnames = ('pred_ns', 'pred', 'op', 'val')
- term_list = [
- request.form.getlist('{}[]'.format(tn))
- for tn in fnames]
- # Transpose matrix.
- txm = list(zip(*term_list))
- logger.info('transposed matrix: {}'.format(txm))
- terms = []
- for row in txm:
- fmt_row = list(row)
- ns = fmt_row.pop(0)
- fmt_row[0] = nsc[ns][fmt_row[0]] if ns else URIRef(fmt_row[0])
- terms.append(fmt_row)
- logger.info('Terms: {}'.format(terms))
- or_logic = request.form.get('logic') == 'or'
- qres = query_api.term_query(terms, or_logic)
- def gl(uri):
- return uri.replace(nsc['fcres'], '/ldp')
- return render_template('term_search_results.html', qres=qres, gl=gl)
- else:
- return render_template(
- 'term_search.html', operands=operands, qres=qres, nsm=nsm)
- @query.route('/sparql', methods=['GET', 'POST'])
- def sparql():
- """
- Perform a direct SPARQL query on the underlying triplestore.
- :param str qry: SPARQL query string.
- """
- accept_mimetypes = {
- 'text/csv': 'csv',
- 'application/sparql-results+json': 'json',
- 'application/sparql-results+xml': 'xml',
- }
- if request.method == 'GET':
- return render_template('sparql_query.html', nsm=nsm)
- else:
- if request.mimetype == 'application/sparql-query':
- qstr = request.stream.read()
- else:
- qstr = request.form['query']
- logger.debug('Query: {}'.format(qstr))
- match = request.accept_mimetypes.best_match(accept_mimetypes.keys())
- fmt = (
- accept_mimetypes[match] if match
- else request.accept_mimetypes.best)
- try:
- out_stream = query_api.sparql_query(qstr, fmt)
- except PluginException:
- return (
- 'Unable to serialize results into format {}'.format(fmt), 406)
- return send_file(out_stream, mimetype=fmt), 200
|