Skip to content

Commit

Permalink
utils: default endpoint prefixes
Browse files Browse the repository at this point in the history
* NEW Introduces new cached property on extension state called
 `default_endpoint_prefixes` that maps `pid_type` to an
 `endpoint-prefix` used to build endpoints. (closes inveniosoftware#101)

Reviewed-by: Jiri Kuncar <[email protected]>
Signed-off-by: Samuele Kaplun <[email protected]>
  • Loading branch information
kaplun authored and jirikuncar committed Oct 7, 2016
1 parent 3fad849 commit c79ce2e
Show file tree
Hide file tree
Showing 7 changed files with 236 additions and 11 deletions.
13 changes: 9 additions & 4 deletions invenio_records_rest/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -88,8 +88,9 @@ def can(self):
RECORDS_REST_ENDPOINTS = {
'record-pid-type': {
'endpoint-prefix': {
'create_permission_factory_imp': permission_check_factory(),
'default_endpoint_prefix': True,
'default_media_type': 'application/json',
'delete_permission_factory_imp': permission_check_factory(),
'item_route': ''/recods/<pid(record-pid-type):pid_value>'',
Expand Down Expand Up @@ -132,6 +133,10 @@ def can(self):
:param create_permission_factory_imp: Import path to factory that create
permission object for a given record.
:param default_endpoint_prefix: declare the current endpoint as the default
when building endpoints for the defined ``pid_type``. By default the
default prefix is defined to be the value of ``pid_type``.
:param default_media_type: Default media type for both records and search.
:param delete_permission_factory_imp: Import path to factory that creates a
Expand All @@ -146,10 +151,10 @@ def can(self):
:param max_result_window: Maximum total number of records retrieved from a
query.
:param pid_type: It specifies the record pid type. It's used also to build the
endpoint name. Required.
:param pid_type: It specifies the record pid type. Required.
You can generate an URL to list all records of the given ``pid_type`` by
calling ``url_for('invenio_records_rest.{0}_list'.format(pid_type))``.
calling ``url_for('invenio_records_rest.{0}_list'.format(
current_records_rest.default_endpoint_prefixes[pid_type]))``.
:param pid_fetcher: It identifies the registered fetcher name. Required.
Expand Down
7 changes: 6 additions & 1 deletion invenio_records_rest/ext.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@
from werkzeug.utils import cached_property

from . import config
from .utils import load_or_import_from_config
from .utils import build_default_endpoint_prefixes, load_or_import_from_config
from .views import create_blueprint


Expand Down Expand Up @@ -75,6 +75,11 @@ def delete_permission_factory(self):
'RECORDS_REST_DEFAULT_DELETE_PERMISSION_FACTORY', app=self.app
)

@cached_property
def default_endpoint_prefixes(self):
"""Map between pid_type and endpoint_prefix."""
return build_default_endpoint_prefixes()

def reset_permission_factories(self):
"""Remove cached permission factories."""
for key in ('read', 'create', 'update', 'delete'):
Expand Down
5 changes: 4 additions & 1 deletion invenio_records_rest/links.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,14 +26,17 @@

from flask import url_for

from .proxies import current_records_rest


def default_links_factory(pid):
"""Factory for record links generation.
:param pid: A Persistent Identifier instance.
:returns: Dictionary containing a list of useful links for the record.
"""
endpoint = '.{0}_item'.format(pid.pid_type)
endpoint = '.{0}_item'.format(
current_records_rest.default_endpoint_prefixes[pid.pid_type])
links = dict(self=url_for(endpoint, pid_value=pid.pid_value,
_external=True))
return links
45 changes: 43 additions & 2 deletions invenio_records_rest/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,45 @@
from .errors import PIDDeletedRESTError, PIDDoesNotExistRESTError, \
PIDMissingObjectRESTError, PIDRedirectedRESTError, \
PIDUnregisteredRESTError
from .proxies import current_records_rest


def build_default_endpoint_prefixes():
"""Build the default_endpoint_prefixes map."""
ret = {}
record_rest_endpoints = current_app.config['RECORDS_REST_ENDPOINTS']
for endpoint in record_rest_endpoints.values():
pid_type = endpoint['pid_type']
ret[pid_type] = get_default_endpoint_for(pid_type,
record_rest_endpoints)

return ret


def get_default_endpoint_for(pid_type, _record_rest_endpoints=None):
"""Get default endpoint for the given pid_type."""
if _record_rest_endpoints is None:
_record_rest_endpoints = current_app.config['RECORDS_REST_ENDPOINTS']

endpoint_prefix = None

for key, value in _record_rest_endpoints.items():
if (value['pid_type'] == pid_type and
value.get('default_endpoint_prefix')):
if endpoint_prefix is None:
endpoint_prefix = key
else:
raise ValueError('More than one endpoint-prefix has been '
'defined as default for '
'pid_type="{0}"'.format(pid_type))

if endpoint_prefix:
return endpoint_prefix
if pid_type in _record_rest_endpoints:
return pid_type

raise ValueError('No endpoint-prefix corresponds to pid_type="{0}"'.format(
pid_type))


def obj_or_import_string(value, default=None):
Expand Down Expand Up @@ -129,7 +168,9 @@ def data(self):
except PIDRedirectedError as e:
try:
location = url_for(
'.{0}_item'.format(e.destination_pid.pid_type),
'.{0}_item'.format(
current_records_rest.default_endpoint_prefixes[
e.destination_pid.pid_type]),
pid_value=e.destination_pid.pid_value)
data = dict(
status=301,
Expand All @@ -139,7 +180,7 @@ def data(self):
response = make_response(jsonify(data), data['status'])
response.headers['Location'] = location
abort(response)
except BuildError:
except (BuildError, KeyError):
current_app.logger.exception(
'Invalid redirect - pid_type "{0}" '
'endpoint missing.'.format(
Expand Down
9 changes: 6 additions & 3 deletions invenio_records_rest/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -130,7 +130,7 @@ def create_url_rules(endpoint, list_route=None, item_route=None,
default_media_type=None,
max_result_window=None, use_options_view=True,
search_factory_imp=None, links_factory_imp=None,
suggesters=None):
suggesters=None, default_endpoint_prefix=None):
"""Create Werkzeug URL rules.
:param endpoint: Name of endpoint.
Expand All @@ -148,6 +148,7 @@ def create_url_rules(endpoint, list_route=None, item_route=None,
update permission object for a given record.
:param delete_permission_factory_imp: Import path to factory that creates a
delete permission object for a given record.
:param default_endpoint_prefix: ignored.
:param record_class: A record API class or importable string.
:param record_serializers: Serializers used for records.
:param record_loaders: It contains the list of record deserializers for
Expand Down Expand Up @@ -452,7 +453,8 @@ def get(self, **kwargs):
size=size,
_external=True,
)
endpoint = '.{0}_list'.format(self.pid_type)
endpoint = '.{0}_list'.format(
current_records_rest.default_endpoint_prefixes[self.pid_type])
links = dict(self=url_for(endpoint, page=page, **urlkwargs))
if page > 1:
links['prev'] = url_for(endpoint, page=page - 1, **urlkwargs)
Expand Down Expand Up @@ -515,7 +517,8 @@ def post(self, **kwargs):
pid, record, 201, links_factory=self.item_links_factory)

# Add location headers
endpoint = '.{0}_item'.format(pid.pid_type)
endpoint = '.{0}_item'.format(
current_records_rest.default_endpoint_prefixes[pid.pid_type])
location = url_for(endpoint, pid_value=pid.pid_value, _external=True)
response.headers.extend(dict(location=location))
return response
Expand Down
24 changes: 24 additions & 0 deletions tests/conftest.py
Original file line number Diff line number Diff line change
Expand Up @@ -112,6 +112,21 @@ def test_mytest(app, db, es):
This will parameterize the default 'recid' endpoint in
RECORDS_REST_ENDPOINTS.
Alternatively:
.. code-block:: python
@pytest.mark.parametrize('app', [dict(
records_rest_endpoints=dict(
recid=dict(
search_class='conftest:TestSearch',
)
)
def test_mytest(app, db, es):
# ...
This will fully parameterize RECORDS_REST_ENDPOINTS.
"""
instance_path = tempfile.mkdtemp()
app = Flask('testapp', instance_path=instance_path)
Expand Down Expand Up @@ -156,6 +171,15 @@ def test_mytest(app, db, es):
if 'endpoint' in request.param:
app.config['RECORDS_REST_ENDPOINTS']['recid'].update(
request.param['endpoint'])
if 'records_rest_endpoints' in request.param:
original_endpoint = app.config['RECORDS_REST_ENDPOINTS']['recid']
del app.config['RECORDS_REST_ENDPOINTS']['recid']
for new_endpoint_prefix, new_endpoint_value in \
request.param['records_rest_endpoints'].items():
new_endpoint = dict(original_endpoint)
new_endpoint.update(new_endpoint_value)
app.config['RECORDS_REST_ENDPOINTS'][new_endpoint_prefix] = \
new_endpoint

app.url_map.converters['pid'] = PIDConverter

Expand Down
144 changes: 144 additions & 0 deletions tests/test_utils.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,144 @@
# -*- coding: utf-8 -*-
#
# This file is part of Invenio.
# Copyright (C) 2016 CERN.
#
# Invenio is free software; you can redistribute it
# and/or modify it under the terms of the GNU General Public License as
# published by the Free Software Foundation; either version 2 of the
# License, or (at your option) any later version.
#
# Invenio is distributed in the hope that it will be
# useful, but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
# General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with Invenio; if not, write to the
# Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston,
# MA 02111-1307, USA.
#
# In applying this license, CERN does not
# waive the privileges and immunities granted to it by virtue of its status
# as an Intergovernmental Organization or submit itself to any jurisdiction.


"""Utils tests."""

from __future__ import absolute_import, print_function

import pytest

from invenio_records_rest.proxies import current_records_rest
from invenio_records_rest.utils import get_default_endpoint_for


@pytest.mark.parametrize('app', [dict(
records_rest_endpoints=dict(
recid=dict(
pid_type='recid',
default_endpoint_prefix=False,
)
))], indirect=['app'])
def test_build_default_endpoint_prefixes_simple(app):
with app.test_client():
assert current_records_rest.default_endpoint_prefixes['recid'] == \
'recid'


@pytest.mark.parametrize('app', [dict(
records_rest_endpoints=dict(
recid=dict(
pid_type='recid',
default_endpoint_prefix=True,
)
))], indirect=['app'])
def test_build_default_endpoint_prefixes_simple_with_default(app):
with app.test_client():
assert current_records_rest.default_endpoint_prefixes['recid'] == \
'recid'


@pytest.mark.parametrize('app', [dict(
records_rest_endpoints=dict(
recid=dict(
pid_type='recid',
default_endpoint_prefix=False,
),
recid2=dict(
pid_type='recid',
default_endpoint_prefix=False,
)
))], indirect=['app'])
def test_build_default_endpoint_prefixes_two_simple_endpoints(app):
with app.test_client():
assert current_records_rest.default_endpoint_prefixes['recid'] == \
'recid'


@pytest.mark.parametrize('app', [dict(
records_rest_endpoints=dict(
recid=dict(
pid_type='recid',
default_endpoint_prefix=True,
),
recid2=dict(
pid_type='recid',
default_endpoint_prefix=False,
)
))], indirect=['app'])
def test_build_default_endpoint_prefixes_redundant_default(app):
with app.test_client():
assert current_records_rest.default_endpoint_prefixes['recid'] == \
'recid'


@pytest.mark.parametrize('app', [dict(
records_rest_endpoints=dict(
recid=dict(
pid_type='recid',
default_endpoint_prefix=False,
),
recid2=dict(
pid_type='recid',
default_endpoint_prefix=True,
)
))], indirect=['app'])
def test_build_default_endpoint_prefixes_two_endpoints_with_default(app):
with app.test_client():
assert current_records_rest.default_endpoint_prefixes['recid'] == \
'recid2'


def test_get_default_endpoint_for_inconsistent(app):
with pytest.raises(ValueError) as excinfo:
get_default_endpoint_for('recid', {
'recid1': {
'pid_type': 'recid',
'default_endpoint_prefix': True,
},
'recid2': {
'pid_type': 'recid',
'default_endpoint_prefix': True,
}})
assert 'More than one' in str(excinfo.value)

with pytest.raises(ValueError) as excinfo:
get_default_endpoint_for('recid', {
'recid1': {
'pid_type': 'recid',
},
'recid2': {
'pid_type': 'recid',
}})
assert 'No endpoint-prefix' in str(excinfo.value)

with pytest.raises(ValueError) as excinfo:
get_default_endpoint_for('foo', {
'recid1': {
'pid_type': 'recid',
},
'recid2': {
'pid_type': 'recid',
}})
assert 'No endpoint-prefix' in str(excinfo.value)

0 comments on commit c79ce2e

Please sign in to comment.