Skip to content

Commit

Permalink
add enhanced support for limits (RFC5) (#1856)
Browse files Browse the repository at this point in the history
  • Loading branch information
tomkralidis committed Nov 20, 2024
1 parent 3188db9 commit af080e1
Show file tree
Hide file tree
Showing 24 changed files with 188 additions and 92 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/containers.yml
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ jobs:
contents: read
steps:
- name: Check out the repo
uses: actions/checkout@v3
uses: actions/checkout@master

- name: Set up QEMU
uses: docker/[email protected]
Expand Down
4 changes: 2 additions & 2 deletions .github/workflows/docs.yml
Original file line number Diff line number Diff line change
Expand Up @@ -23,8 +23,8 @@ jobs:
include:
- python-version: '3.10'
steps:
- uses: actions/checkout@v2
- uses: actions/setup-python@v2
- uses: actions/checkout@master
- uses: actions/setup-python@v5
name: Setup Python ${{ matrix.python-version }}
with:
python-version: ${{ matrix.python-version }}
Expand Down
4 changes: 2 additions & 2 deletions .github/workflows/flake8.yml
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,8 @@ jobs:
flake8_py3:
runs-on: ubuntu-22.04
steps:
- uses: actions/checkout@v3
- uses: actions/setup-python@v3
- uses: actions/checkout@master
- uses: actions/setup-python@v5
name: setup Python
with:
python-version: '3.10'
Expand Down
8 changes: 4 additions & 4 deletions .github/workflows/main.yml
Original file line number Diff line number Diff line change
Expand Up @@ -37,8 +37,8 @@ jobs:
- name: Chown user
run: |
sudo chown -R $USER:$USER $GITHUB_WORKSPACE
- uses: actions/checkout@v2
- uses: actions/setup-python@v2
- uses: actions/checkout@master
- uses: actions/setup-python@v5
name: Setup Python ${{ matrix.python-version }}
with:
python-version: ${{ matrix.python-version }}
Expand Down Expand Up @@ -156,8 +156,8 @@ jobs:
PYGEOAPI_CONFIG: "tests/pygeoapi-test-config-admin.yml"
PYGEOAPI_OPENAPI: "tests/pygeoapi-test-openapi-admin.yml"
steps:
- uses: actions/checkout@v2
- uses: actions/setup-python@v2
- uses: actions/checkout@master
- uses: actions/setup-python@v5
name: Setup Python ${{ matrix.python-version }}
with:
python-version: ${{ matrix.python-version }}
Expand Down
4 changes: 3 additions & 1 deletion docker/default.config.yml
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,9 @@ server:
language: en-US
cors: true
pretty_print: true
limit: 10
limits:
defaultitems: 10
maxitems: 50
# templates: /path/to/templates
map:
url: https://tile.openstreetmap.org/{z}/{x}/{y}.png
Expand Down
7 changes: 6 additions & 1 deletion docs/source/configuration.rst
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,12 @@ For more information related to API design rules (the ``api_rules`` property in
gzip: false # default server config to gzip/compress responses to requests with gzip in the Accept-Encoding header
cors: true # boolean on whether server should support CORS
pretty_print: true # whether JSON responses should be pretty-printed
limit: 10 # server limit on number of items to return
limits: # server limits on number of items to return. This property can also be defined at the resource level to override global server settings
defaultitems: 10
maxitems: 100
maxdistance: [25, 25]
admin: false # whether to enable the Admin API
templates: # optional configuration to specify a different set of templates for HTML pages. Recommend using absolute paths. Omit this to use the default provided templates
Expand Down
40 changes: 40 additions & 0 deletions pygeoapi/api/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -1721,3 +1721,43 @@ def validate_subset(value: str) -> dict:
subsets[subset_name] = list(map(get_typed_value, values))

return subsets


def evaluate_limit(requested: Union[None, int], server_limits: dict,
collection_limits: dict) -> int:
"""
Helper function to evaluate limit parameter
:param requested: the limit requested by the client
:param server_limits: `dict` of server limits
:param collection_limits: `dict` of collection limits
:returns: `int` of evaluated limit
"""

if collection_limits:
LOGGER.debug('Using collection defined limit')
max_ = collection_limits.get('maxitems', 10)
default = collection_limits.get('defaultitems', 10)
else:
LOGGER.debug('Using server defined limit')
max_ = server_limits.get('maxitems', 10)
default = server_limits.get('defaultitems', 10)

LOGGER.debug(f'Requested limit: {requested}')
LOGGER.debug(f'Default limit: {default}')
LOGGER.debug(f'Maximum limit: {max_}')

if requested is None:
LOGGER.debug('no limit requested; returning default')
return default

requested2 = get_typed_value(requested)
if not isinstance(requested2, int):
raise ValueError('limit value should be an integer')

if requested2 <= 0:
raise ValueError('limit value should be strictly positive')
else:
LOGGER.debug('limit requested')
return min(requested2, max_)
15 changes: 13 additions & 2 deletions pygeoapi/api/environmental_data_retrieval.py
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@
from shapely.wkt import loads as shapely_loads

from pygeoapi import l10n
from pygeoapi.api import evaluate_limit
from pygeoapi.plugin import load_plugin, PLUGINS
from pygeoapi.provider.base import ProviderGenericError
from pygeoapi.util import (
Expand Down Expand Up @@ -175,6 +176,16 @@ def get_collection_edr_query(api: API, request: APIRequest,
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)

LOGGER.debug('Processing limit parameter')
try:
limit = evaluate_limit(request.params.get('limit'),
api.config['server'].get('limits', {}),
collections[dataset].get('limits', {}))
except ValueError as err:
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', str(err))

query_args = dict(
query_type=query_type,
instance=instance,
Expand All @@ -186,8 +197,8 @@ def get_collection_edr_query(api: API, request: APIRequest,
bbox=bbox,
within=within,
within_units=within_units,
limit=int(api.config['server']['limit']),
location_id=location_id,
limit=limit,
location_id=location_id
)

try:
Expand Down
45 changes: 14 additions & 31 deletions pygeoapi/api/itemtypes.py
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,7 @@
from pyproj.exceptions import CRSError

from pygeoapi import l10n
from pygeoapi.api import evaluate_limit
from pygeoapi.formatter.base import FormatterSerializationError
from pygeoapi.linked_data import geojson2jsonld
from pygeoapi.plugin import load_plugin, PLUGINS
Expand Down Expand Up @@ -238,33 +239,24 @@ def get_collection_items(
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)
except TypeError as err:
LOGGER.warning(err)
offset = 0
except ValueError:
msg = 'offset value should be an integer'
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)
except TypeError as err:
LOGGER.warning(err)
offset = 0

LOGGER.debug('Processing limit parameter')
try:
limit = int(request.params.get('limit'))
# TODO: We should do more validation, against the min and max
# allowed by the server configuration
if limit <= 0:
msg = 'limit value should be strictly positive'
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)
except TypeError as err:
LOGGER.warning(err)
limit = int(api.config['server']['limit'])
except ValueError:
msg = 'limit value should be an integer'
limit = evaluate_limit(request.params.get('limit'),
api.config['server'].get('limits', {}),
collections[dataset].get('limits', {}))
except ValueError as err:
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)
'InvalidParameterValue', str(err))

resulttype = request.params.get('resulttype') or 'results'

Expand Down Expand Up @@ -687,22 +679,13 @@ def post_collection_items(

LOGGER.debug('Processing limit parameter')
try:
limit = int(request.params.get('limit'))
# TODO: We should do more validation, against the min and max
# allowed by the server configuration
if limit <= 0:
msg = 'limit value should be strictly positive'
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)
except TypeError as err:
LOGGER.warning(err)
limit = int(api.config['server']['limit'])
except ValueError:
msg = 'limit value should be an integer'
limit = evaluate_limit(request.params.get('limit'),
api.config['server'].get('limits', {}),
collections[dataset].get('limits', {}))
except ValueError as err:
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)
'InvalidParameterValue', str(err))

resulttype = request.params.get('resulttype') or 'results'

Expand Down
38 changes: 11 additions & 27 deletions pygeoapi/api/processes.py
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@
import urllib.parse

from pygeoapi import l10n
from pygeoapi.api import evaluate_limit
from pygeoapi.util import (
json_serial, render_j2_template, JobStatus, RequestedProcessExecutionMode,
to_json, DATETIME_FORMAT)
Expand Down Expand Up @@ -101,23 +102,14 @@ def describe_processes(api: API, request: APIRequest,
else:
LOGGER.debug('Processing limit parameter')
try:
limit = int(request.params.get('limit'))

if limit <= 0:
msg = 'limit value should be strictly positive'
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)

limit = evaluate_limit(request.params.get('limit'),
api.config['server'].get('limits', {}),
{})
relevant_processes = list(api.manager.processes)[:limit]
except TypeError:
LOGGER.debug('returning all processes')
relevant_processes = api.manager.processes.keys()
except ValueError:
msg = 'limit value should be an integer'
except ValueError as err:
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)
'InvalidParameterValue', str(err))

for key in relevant_processes:
p = api.manager.get_processor(key)
Expand Down Expand Up @@ -243,21 +235,13 @@ def get_jobs(api: API, request: APIRequest,
**api.api_headers)
LOGGER.debug('Processing limit parameter')
try:
limit = int(request.params.get('limit'))

if limit <= 0:
msg = 'limit value should be strictly positive'
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)
except TypeError:
limit = int(api.config['server']['limit'])
LOGGER.debug('returning all jobs')
except ValueError:
msg = 'limit value should be an integer'
limit = evaluate_limit(request.params.get('limit'),
api.config['server'].get('limits', {}),
{})
except ValueError as err:
return api.get_exception(
HTTPStatus.BAD_REQUEST, headers, request.format,
'InvalidParameterValue', msg)
'InvalidParameterValue', str(err))

LOGGER.debug('Processing offset parameter')
try:
Expand Down
26 changes: 25 additions & 1 deletion pygeoapi/schemas/config/pygeoapi-config-0.x.yml
Original file line number Diff line number Diff line change
Expand Up @@ -60,8 +60,29 @@ properties:
default: false
limit:
type: integer
description: server limit on number of items to return
default: 10
description: "limit of items to return. DEPRECATED: use limits instead"
limits: &x-limits
type: object
description: server level limits on number of items to return
properties:
maxitems:
type: integer
minimum: 1
default: 10
description: maximum limit of items to return for feature and record providers
defaultitems:
type: integer
minimum: 1
default: 10
description: default limit of items to return for feature and record providers
maxdistance:
type: array
minItems: 2
maxItems: 2
items:
type: number
description: maximum distance in x and y for all data providers
templates:
type: object
description: optional configuration to specify a different set of templates for HTML pages. Recommend using absolute paths. Omit this to use the default provided templates
Expand Down Expand Up @@ -417,6 +438,9 @@ properties:
default: 'http://www.opengis.net/def/uom/ISO-8601/0/Gregorian'
required:
- spatial
limits:
<<: *x-limits
description: collection level limits on number of items to return
providers:
type: array
description: required connection information
Expand Down
29 changes: 28 additions & 1 deletion tests/api/test_api.py
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@

from pygeoapi.api import (
API, APIRequest, FORMAT_TYPES, F_HTML, F_JSON, F_JSONLD, F_GZIP,
__version__, validate_bbox, validate_datetime,
__version__, evaluate_limit, validate_bbox, validate_datetime,
validate_subset
)
from pygeoapi.util import yaml_load, get_api_rules, get_base_url
Expand Down Expand Up @@ -875,3 +875,30 @@ def test_get_exception(config, api_):
assert content['description'] == 'oops'

d = api_.get_exception(500, {}, 'html', 'NoApplicableCode', 'oops')


def test_evaluate_limit():
collection = {}
server = {}

with pytest.raises(ValueError):
assert evaluate_limit('1.1', server, collection) == 10

with pytest.raises(ValueError):
assert evaluate_limit('-12', server, collection) == 10

assert evaluate_limit('1', server, collection) == 1

collection = {}
server = {'defaultitems': 2, 'maxitems': 3}

assert evaluate_limit(None, server, collection) == 2
assert evaluate_limit('1', server, collection) == 1
assert evaluate_limit('4', server, collection) == 3

collection = {'defaultitems': 10, 'maxitems': 50}
server = {'defaultitems': 100, 'maxitems': 1000}

assert evaluate_limit(None, server, collection) == 10
assert evaluate_limit('40', server, collection) == 40
assert evaluate_limit('60', server, collection) == 50
Loading

0 comments on commit af080e1

Please sign in to comment.