.. _plugins: Plugins ======= GHC can be extended for Resource-specific healthchecks via Plugins. GHC already comes with a set of standard plugins that may suffice most installations. However, there is no limit to detailed healthchecks one may want to perform. Hence developers can extend or even replace the GHC standard Plugins with custom implementations. Two Plugin types exist that can be extended: the `Probe` and `Check` class. In v0.7.0 also plugins for Resource Authentication, `ResourceAuth`, were added and in v0.9.0 the geocoder plugin was introduced. Concepts -------- GHC versions after May 1, 2017 perform healthchecks exclusively via Plugins (see :ref:`upgrade` how to upgrade from older versions). The basic concept is simple: each `Resource` (typically an OWS endpoint) has one or more `Probes`. During a GHC run (via `cron` or manually), GHC sequentually invokes the `Probes` for each `Resource` to determine the health (QoS) of the `Resource`. A `Probe` typically implements a single request like a `WMS GetMap`. A `Probe` contains and applies one or more `Checks` (the other Plugin class). A `Check` implements typically a single check on the HTTP Response object of its parent `Probe`, for example if the HTTP response has no errors or if a `WMS GetMap` actually returns an image (content-type check). Each `Check` will supply a `CheckResult` to its parent `Probe`. The list of `CheckResults` will then ultimately determine the `ProbeResult`. The `Probe` will in turn supply the `ProbeResult` to its parent `ResourceResult`. The GHC healthchecker will then determine the final outcome of the `Run` (fail/success) for the `Resource`, adding the list of Probe/CheckResults to the historic Run-data in the DB. This data can later be used for reporting and determining which `Check(s)` were failing. So in summary: a `Resource` has one or more `Probes`, each `Probe` one or more `Checks`. On a GHC run these together provide a `Result`. Probes and Checks available to the GHC instance are configured in `config_site.py`, the GHC instance config file. Also configured there is the default `Probe` class to assign to a Resource-type when it is added. Assignment and configuration/parameterization of `Probes` and `Checks` is via de UI on the Resource-edit page and stored in the database (tables: `probe_vars` and `check_vars`). That way the GHC healthcheck runner can read (from the DB) the list of Probes/Checks and their config for each Resource. Implementation -------------- `Probes` and `Checks` plugins are implemented as Python classes derived from :class:`GeoHealthCheck.probe.Probe` and :class:`GeoHealthCheck.check.Check` respectively. These classes inherit from the GHC abstract base class :class:`GeoHealthCheck.plugin.Plugin`. This class mainly provides default attributes (in capitals) and introspection methods needed for UI configuration. *Class-attributes* (in capitals) are the most important concept of GHC Plugins in general. These provide metadata for various GHC functions (internal, UI etc). General class-attributes that Plugin authors should provide for derived `Probes` or `Checks` are: * `AUTHOR`: Plugin author or team. * `NAME`: Short name of Plugin. * `DESCRIPTION`: Longer description of Plugin. * `PARAM_DEFS`: Plugin Parameter definitions (see next) `PARAM_DEFS`, a Python `dict` defines the parameter definitions for the `Probe` or `Check` that a user can configure via the UI. Each parameter (name) is itself a `dict` entry key that with the following key/value pairs: * `type`: the parameter type, value: 'string', 'stringlist' (comma-separated strings) or 'bbox' (lowerX, lowerY, upperX, upperY), * `description`: description of the parameter, * `default`: parameter default value, * `required`: is parameter required?, * `range`: range of possible parameter values (array of strings), results in UI dropdown selector A `Probe` should supply these additional class-attributes: * `RESOURCE_TYPE` : GHC Resource type this Probe applies to, e.g. `OGC:WMS`, `*:*` (any Resource Type), see `enums.py` for range * `REQUEST_METHOD` : HTTP request method capitalized, 'GET' (default) or 'POST'. * `REQUEST_HEADERS` : `dict` of optional HTTP request headers * `REQUEST_TEMPLATE`: template in standard Python `str.format(*args)` to be substituted with actual parameters from `PARAM_DEFS` * `CHECKS_AVAIL` : available Check (classes) for this Probe. Note: `CHECKS_AVAIL` denotes all possible `Checks` that can be assigned, by default or via UI, to an instance of this `Probe`. A `Check` has no additional class-attributes. In many cases writing a `Probe` is a matter of just defining the above class-attributes. The GHC healthchecker :meth:`GeoHealthCheck.healthcheck.run_test_resource` will call lifecycle methods of the :class:`GeoHealthCheck.probe.Probe` base class, using the class-attributes and actualized parameters (stored in `probe_vars` table) as defined in `PARAM_DEFS` plus a list of the actual and parameterized Checks (stored in `check_vars` table) for its Probe instance. More advanced `Probes` can override base-class methods of `Probe` in particular :meth:`GeoHealthCheck.probe.Probe.perform_request`. In that case the Probe-author should add one or more :class:`GeoHealthCheck.result.Result` objects to `self.result` via `self.result.add_result(result)` Writing a `Check` class requires providing the Plugin class-attributes (see above) including optional `PARAM_DEFS`. The actual check is implemented by overriding the `Check` base class method :meth:`GeoHealthCheck.check.Check.perform`, setting the check-result via :meth:`GeoHealthCheck.check.Check.set_result`. Finally your Probes and Checks need to be made available to your GHC instance via `config_site.py` and need to be found on the Python-PATH of your app. The above may seem daunting at first. Examples below will hopefully make things clear as writing new `Probes` and `Checks` may sometimes be a matter of minutes! *TODO: may need VERSION variable class-attr to support upgrades* Examples -------- GHC includes Probes and Checks that on first setup are made available in `config_site.py`. By studying the the GHC standard Probes and Checks under the subdir `GeoHealthCheck/plugins`, Plugin-authors may get a feel how implementation can be effected. There are broadly two ways to write a `Probe`: * using a `REQUEST_*` class-attributes, i.e. letting GHC do the Probe's HTTP requests and checks * overriding :meth:`GeoHealthCheck.probe.Probe.perform_request`: making your own requests An example for each is provided, including the `Checks` used. The simplest Probe is one that does: * an HTTP GET on a `Resource` URL * checks if the HTTP Response is not errored, i.e. a 404 or 500 status * optionally checks if the HTTP Response (not) contains expected strings Below is the implementation of the class :class:`GeoHealthCheck.plugins.probe.http.HttpGet`: .. literalinclude:: ../GeoHealthCheck/plugins/probe/http.py :language: python :lines: 1-23 :linenos: Yes, this is the entire implementation of :class:`GeoHealthCheck.plugins.probe.http.HttpGet`! Only class-attributes are needed: * standard Plugin attributes: `AUTHOR` ('GHC Team' by default) `NAME`, `DESCRIPTION` * `RESOURCE_TYPE = '*:*'` denotes that any Resource may use this Probe (UI lists this Probe under "Probes Available" for Resource) * `REQUEST_METHOD = 'GET'` : GHC should use the HTTP GET request method * `CHECKS_AVAIL` : all Check classes that can be applied to this Probe (UI lists these under "Checks Available" for Probe) By setting: :: 'GeoHealthCheck.plugins.check.checks.HttpStatusNoError': { 'default': True }, that Check is automatically assigned to this Probe when created. The other Checks may be added and configured via the UI. Next look at the Checks, the class :class:`GeoHealthCheck.plugins.check.checks.HttpStatusNoError`: .. literalinclude:: ../GeoHealthCheck/plugins/check/checks.py :language: python :lines: 1-34 :linenos: Also this class is quite simple: providing class-attributes `NAME`, `DESCRIPTION` and implementing the base-class method :meth:`GeoHealthCheck.check.Check.perform`. Via `self.probe` a Check always has a reference to its parent Probe instance and the HTTP Response object via `self.probe.response`. The check itself is a test if the HTTP status code is in the 400 or 500-range. The `CheckResult` is implicitly created by setting: `self.set_result(False, 'HTTP Error status=%d' % status)` in case of errors. `self.set_result()` only needs to be called when a Check fails. By default the Result is succes (`True`). According to this pattern more advanced Probes are implemented for `OWS GetCapabilities`, the most basic test for OWS-es like WMS and WFS. Below the implementation of the class :class:`GeoHealthCheck.plugins.probe.owsgetcaps.OwsGetCaps` and its derived classes for specific OWS-es: .. literalinclude:: ../GeoHealthCheck/plugins/probe/owsgetcaps.py :language: python :lines: 1-108 :linenos: More elaborate but still only class-attributes are used! Compared to :class:`GeoHealthCheck.plugins.probe.http.HttpGet`, two additional class-attributes are used in :class:`GeoHealthCheck.plugins.probe.owsgetcaps.OwsGetCaps` : * `REQUEST_TEMPLATE ='?SERVICE={service}&VERSION={version}&REQUEST=GetCapabilities'` * `PARAM_DEFS` for the `REQUEST_TEMPLATE` GHC will recognize a `REQUEST_TEMPLATE` (for GET or POST) and use `PARAM_DEFS` to substitute configured or default values, here defined in subclasses. This string is then appended to the Resource URL. Three `Checks` are available, all included by default. Also see the construct: :: 'GeoHealthCheck.plugins.check.checks.ContainsStrings': { 'set_params': { 'strings': { 'name': 'Contains Title Element', 'value': ['Title>'] } }, 'default': True }, This not only assigns this Check automatically on creation, but also provides it with parameters, in this case a `Capabilities` response document should always contain a `