Skip to content

Commit

Permalink
Code for official PR to add hot-reload logic mechanisms
Browse files Browse the repository at this point in the history
Comments
  • Loading branch information
Alex-NRCan committed Jan 9, 2025
1 parent 921fb20 commit e5808a5
Show file tree
Hide file tree
Showing 4 changed files with 103 additions and 4 deletions.
94 changes: 93 additions & 1 deletion pygeoapi/api/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@

from collections import OrderedDict
from copy import deepcopy
from datetime import datetime
from datetime import (datetime, timezone)
from functools import partial
from gzip import compress
from http import HTTPStatus
Expand Down Expand Up @@ -165,6 +165,32 @@ def apply_gzip(headers: dict, content: Union[str, bytes]) -> Union[str, bytes]:
return content


def pre_load_colls(func):
"""
Decorator function that makes sure the loaded collections are updated.
This is used when the resources are loaded dynamically, not strictly
from the yaml file.
:param func: decorated function
:returns: `func`
"""

def inner(*args, **kwargs):
cls = args[0]

# Validation on the method name for the provided class instance on this
# decoration function
if hasattr(cls, 'reload_resources_if_necessary'):
# Validate the resources are up to date
cls.reload_resources_if_necessary()

# Continue
return func(*args, **kwargs)

return inner


class APIRequest:
"""
Transforms an incoming server-specific Request into an object
Expand Down Expand Up @@ -564,9 +590,74 @@ def __init__(self, config, openapi):
self.tpl_config = deepcopy(self.config)
self.tpl_config['server']['url'] = self.base_url

# Now that the basic configuration is read, call the load_resources function. # noqa
# This call enables the api engine to load resources dynamically.
# This pattern allows for loading resources coming from another
# source (e.g. a database) rather than from the yaml file.
# This, along with the @pre_load_colls decorative function, enables
# resources management on multiple distributed pygeoapi instances.
self.load_resources()

self.manager = get_manager(self.config)
LOGGER.info('Process manager plugin loaded')

def on_load_resources(self, resources: dict) -> dict:
"""
Overridable function to load the available resources dynamically.
By default, this function simply returns the provided resources
as-is. This is the native behavior of the API; expecting
resources to be configured in the yaml config file.
:param resources: the resources as currently configured
(self.config['resources'])
:returns: the resources dictionary that's available in the API.
"""

# By default, return the same resources object, unchanged.
return resources

def on_load_resources_check(self, last_loaded_resources: datetime) -> bool: # noqa
"""
Overridable function to check if the resources should be reloaded.
Return True in your API implementation when resources should be
reloaded. This implementation depends on your environment and
messaging broker.
Natively, the resources used by the pygeoapi instance are strictly
the ones from the yaml configuration file. It doesn't support
resources changing on-the-fly. Therefore, False is returned here
and they are never reloaded.
"""

# By default, return False to not reload the resources.
return False

def load_resources(self) -> None:
"""
Calls on_load_resources and reassigns the resources configuration.
"""

# Call on_load_resources sending the current resources configuration.
self.config['resources'] = self.on_load_resources(self.config['resources']) # noqa

# Copy over for the template config also
# TODO: Check relevancy of this line
self.tpl_config['resources'] = deepcopy(self.config['resources'])

# Keep track of UTC date of last time resources were loaded
self.last_loaded_resources = datetime.now(timezone.utc)

def reload_resources_if_necessary(self) -> None:
"""
Checks if the resources should be reloaded by calling overridable
function 'on_load_resources_check' and then, when necessary, calls
'load_resources'.
"""

# If the resources should be reloaded
if self.on_load_resources_check(self.last_loaded_resources):
# Reload the resources
self.load_resources()

def get_exception(self, status, headers, format_, code,
description) -> Tuple[dict, int, str]:
"""
Expand Down Expand Up @@ -916,6 +1007,7 @@ def conformance(api, request: APIRequest) -> Tuple[dict, int, str]:


@jsonldify
@pre_load_colls
def describe_collections(api: API, request: APIRequest,
dataset=None) -> Tuple[dict, int, str]:
"""
Expand Down
3 changes: 2 additions & 1 deletion pygeoapi/api/coverages.py
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@

from . import (
APIRequest, API, F_JSON, SYSTEM_LOCALE, validate_bbox, validate_datetime,
validate_subset
validate_subset, pre_load_colls
)

LOGGER = logging.getLogger(__name__)
Expand All @@ -68,6 +68,7 @@
]


@pre_load_colls
def get_collection_coverage(
api: API, request: APIRequest, dataset) -> Tuple[dict, int, str]:
"""
Expand Down
7 changes: 6 additions & 1 deletion pygeoapi/api/itemtypes.py
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,7 @@

from . import (
APIRequest, API, SYSTEM_LOCALE, F_JSON, FORMAT_TYPES, F_HTML, F_JSONLD,
validate_bbox, validate_datetime
validate_bbox, validate_datetime, pre_load_colls
)

LOGGER = logging.getLogger(__name__)
Expand Down Expand Up @@ -100,6 +100,7 @@
]


@pre_load_colls
def get_collection_queryables(api: API, request: Union[APIRequest, Any],
dataset=None) -> Tuple[dict, int, str]:
"""
Expand Down Expand Up @@ -196,6 +197,7 @@ def get_collection_queryables(api: API, request: Union[APIRequest, Any],
return headers, HTTPStatus.OK, to_json(queryables, api.pretty_print)


@pre_load_colls
def get_collection_items(
api: API, request: Union[APIRequest, Any],
dataset) -> Tuple[dict, int, str]:
Expand Down Expand Up @@ -637,6 +639,7 @@ def get_collection_items(
return headers, HTTPStatus.OK, to_json(content, api.pretty_print)


@pre_load_colls
def post_collection_items(
api: API, request: APIRequest, dataset) -> Tuple[dict, int, str]:
"""
Expand Down Expand Up @@ -922,6 +925,7 @@ def post_collection_items(
return headers, HTTPStatus.OK, to_json(content, api.pretty_print)


@pre_load_colls
def manage_collection_item(
api: API, request: APIRequest,
action, dataset, identifier=None) -> Tuple[dict, int, str]:
Expand Down Expand Up @@ -1033,6 +1037,7 @@ def manage_collection_item(
return headers, HTTPStatus.OK, ''


@pre_load_colls
def get_collection_item(api: API, request: APIRequest,
dataset, identifier) -> Tuple[dict, int, str]:
"""
Expand Down
3 changes: 2 additions & 1 deletion pygeoapi/api/maps.py
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@
filter_dict_by_key_value
)

from . import APIRequest, API, validate_datetime
from . import APIRequest, API, validate_datetime, pre_load_colls

LOGGER = logging.getLogger(__name__)

Expand All @@ -60,6 +60,7 @@
]


@pre_load_colls
def get_collection_map(api: API, request: APIRequest,
dataset, style=None) -> Tuple[dict, int, str]:
"""
Expand Down

0 comments on commit e5808a5

Please sign in to comment.