` for more information about the Swagger page.
In addition, the ``autoDescribeRoute`` decorator handles a lot of the validation
and type coercion for you, with the benefit of ensuring that the documentation of
the endpoint inputs matches their actual behavior. Documented parameters will be
sent to the method as kwargs (so the order you declare them in the header doesn't matter).
Any additional parameters that were passed but not listed in the ``Description`` object
will be contained in the ``params`` kwarg as a dictionary, if that parameter is present. The
validation of required parameters, coercion to the correct data type, and setting default
values is all handled automatically for you based on the parameter descriptions in the
``Description`` object passed. Two special methods of the ``Description`` object can be used for
additional behavior control: :py:func:`girder.api.describe.Description.modelParam` and
:py:func:`girder.api.describe.Description.jsonParam`.
The ``modelParam`` method is used to convert parameters passed in as IDs to the model document
corresponding to those IDs, and also can perform access checks to ensure that the user calling the
endpoint has the requisite access level on the resource. For example, we can convert the above
handler to use it:
.. code-block:: python
@access.public
@autoDescribeRoute(
Description('Retrieve the cat for a given item.')
.modelParam('id', 'The item ID', model='item', level=AccessType.READ)
.param('cat', 'The cat value.', required=False)
.errorResponse())
def myHandler(item, cat, params):
return {
'item': item,
'cat': cat
}
The ``jsonParam`` method can be used to indicate that a parameter should be parsed as
a JSON string into the corresponding python value and passed as such.
If you are creating routes that you explicitly do not wish to be exposed in the
Swagger documentation for whatever reason, you can pass ``hide=True`` to the
``autoDescribeRoute`` decorator, and no warning will appear.
.. code-block:: python
@autoDescribeRoute(Description(...), hide=True)
Adding a new resource type to the web API
*****************************************
Perhaps for our use case we determine that ``cat`` should be its own resource
type rather than being referenced via the ``item`` resource. If we wish to add
a new resource type entirely, it will look much like one of the core resource
classes, and we can add it to the API in the ``load()`` method.
.. code-block:: python
from girder.api.rest import Resource
class Cat(Resource):
def __init__(self):
super().__init__()
self.resourceName = 'cat'
self.route('GET', (), self.findCat)
self.route('GET', (':id',), self.getCat)
self.route('POST', (), self.createCat)
self.route('PUT', (':id',), self.updateCat)
self.route('DELETE', (':id',), self.deleteCat)
def getCat(self, id, params):
...
As done when extending an existing resource, this should be mounted into Girder's
API inside your plugin's load method:
.. code-block:: python
from girder.plugin import GirderPlugin
class CatsPlugin(GirderPlugin)
def load(self, info):
info['apiRoot'].cat = Cat()
Adding a prefix to an API
*************************
It is possible to provide a prefix to your API, allowing associated endpoints to
be grouped together. This is done by creating a prefix when mounting the resource.
Note that ``resourceName`` is **not** provided as the resource name is also derived
from the mount location.
.. code-block:: python
from girder.api.rest import Resource, Prefix
from girder.plugin import GirderPlugin
class Cat(Resource):
def __init__(self):
super().__init__()
self.route('GET', (), self.findCat)
self.route('GET', (':id',), self.getCat)
self.route('POST', (), self.createCat)
self.route('PUT', (':id',), self.updateCat)
self.route('DELETE', (':id',), self.deleteCat)
def getCat(self, id, params):
...
class CatsPlugin(GirderPlugin):
def load(self, info):
info['apiRoot'].meow = Prefix()
info['apiRoot'].meow.cat = Cat()
The endpoints are now mounted at meow/cat/
Adding a new model type in your plugin
**************************************
Most of the time, if you add a new resource type in your plugin, you'll have a
``Model`` class backing it. These model classes work just like the core model
classes as described in the :ref:`models` section. If you need to use the
:py:class:`~girder.utility.model_importer.ModelImporter` class with your model type,
you will need to explicitly register the model type to a string, e.g.
.. code-block:: python
from girder.plugin import GirderPlugin
from girder.utility.model_importer import ModelImporter
from .models.cat import Cat
class CatsPlugin(GirderPlugin):
def load(self, info):
ModelImporter.registerModel('cat', Cat, plugin='cats')
Adding custom access flags
**************************
Girder core provides a way to assign a permission level (read, write, and own) to data in the
hierarchy to individual users or groups. In addition to this level, users and groups can also
be granted special access flags on resources in the hierarchy. If you want to expose a new
access flag on data, have your plugin globally register the flag in the system:
.. code-block:: python
from girder.constants import registerAccessFlag
registerAccessFlag(key='cats.feed', name='Feed cats', description='Allows users to feed cats')
When your plugin is installed, a new checkbox will automatically appear in the access control
dialog allowing resource owners to specify what users and groups are allowed to feed
cats (assuming cats are represented by data in the hierarchy). Additionally, if your resource is
public, you will also be able to configure which access flags are available to the public.
If your plugin exposes another endpoint, say ``POST cat/{id}/food``, inside that route handler, you
can call ``requireAccessFlags``, e.g.:
.. code-block:: python
from girder_cat import Cat
@access.user
@autoDescribeRoute(
Description('Feed a cat')
.modelParam('id', 'ID of the cat', model=Cat, level=AccessType.WRITE)
)
def feedCats(self, cat, params):
Cat().requireAccessFlags(item, user=getCurrentUser(), flags='cats.feed')
# Feed the cats ...
That will throw an ``AccessException`` if the user does not possess the specified access
flag(s) on the given resource. You can equivalently use the ``Description.modelParam``
method using ``autoDescribeRoute``, passing a ``requiredFlags`` parameter, e.g.:
.. code-block:: python
@access.user
@autoDescribeRoute(
Description('Feed a cat')
.modelParam('id', 'ID of the cat', model=Cat, level=AccessType.WRITE,
requiredFlags='cats.feed')
)
def feedCats(self, cat, params):
# Feed the cats ...
Normally, anyone with ownership access on the resource will be allowed to enable the flag on
their resources. If instead you want to make it so that only site administrators can enable your
custom access flag, pass ``admin=True`` when registering the flag, e.g.
.. code-block:: python
registerAccessFlag(key='cats.feed', name='Feed cats', admin=True)
We cannot prescribe exactly how access flags should be used; Girder core does not
expose any on its own, and the sorts of policies that they will enforce will be entirely
defined by the logic of your plugin.
The events system
*****************
In addition to being able to augment the core API as described above, the core
system fires a known set of events that plugins can bind to and handle as
they wish.
In the most general sense, the events framework is simply a way of binding
arbitrary events with handlers. The events are identified by a unique string
that can be used to bind handlers to them. For example, if the following logic
is executed by your plugin at startup time,
.. code-block:: python
from girder import events
def handler(event):
print event.info
events.bind('some_event', 'my_handler', handler)
And then during runtime the following code executes:
.. code-block:: python
events.trigger('some_event', info='hello')
Then ``hello`` would be printed to the console at that time. More information
can be found in the API documentation for :ref:`events`.
There are a specific set of known events that are fired from the core system.
Plugins should bind to these events at ``load`` time. The semantics of these
events are enumerated below.
* **Before REST call**
Whenever a REST API route is called, just before executing its default handler,
plugins will have an opportunity to execute code or conditionally override the
default behavior using ``preventDefault`` and ``addResponse``. The identifiers
for these events are of the form ``rest.get.item/:id.before``. They
receive the same kwargs as the default route handler in the event's info.
Since handlers of this event run prior to the normal access level check of the
underlying route handler, they are bound by the same access level rules as route
handlers; they must be decorated by one of the functions in ``girder.api.access``.
If you do not decorate them with one, they will default to requiring administrator
access. This is to prevent accidental reduction of security by plugin developers.
You may change the access level of the route in your handler, but you will
need to do so explicitly by declaring a different decorator than the underlying
route handler.
* **Exception during REST call**
This event is fired if an exception is raised while the default handler is executing.
Like the before REST call event, this event receives the same kwargs as the default
route handler. The caught exception is raised again after this event is handled.
The identifier for this event is, e.g., ``rest.get.item/:id.failed``.
* **After REST call**
Just like the before REST call event, but this is fired after the default
handler has already executed and returned its value. That return value is
also passed in the event.info for possible alteration by the receiving handler.
The identifier for this event is, e.g., ``rest.get.item/:id.after``.
You may alter the existing return value, for example adding an additional property ::
event.info['returnVal']['myProperty'] = 'myPropertyValue'
or override it completely using ``preventDefault`` and ``addResponse`` on the event ::
event.addResponse(myReplacementResponse)
event.preventDefault()
* **Before model save**
You can receive an event each time a document of a specific resource type is
saved. For example, you can bind to ``model.folder.save`` if you wish to
perform logic each time a folder is saved to the database. You can use
``preventDefault`` on the passed event if you wish for the normal saving logic
not to be performed.
* **After model creation**
You can receive an event ``after`` a resource of a specific type is created and
saved to the database. This is sent immediately before the after-save event,
but only occurs upon creation of a new document. You cannot prevent any default
actions with this hook. The format of the event name is, e.g.
``model.folder.save.created``.
* **After model save**
You can also receive an event ``after`` a resource of a specific type is saved
to the database. This is useful if your handler needs to know the ``_id`` field
of the document. You cannot prevent any default actions with this hook. The
format of the event name is, e.g. ``model.folder.save.after``.
* **Before model deletion**
Triggered each time a model is about to be deleted. You can bind to this via
e.g., ``model.folder.remove`` and optionally ``preventDefault`` on the event.
* **During model copy**
Some models have a custom copy method (folder uses copyFolder, item uses
copyItem). When a model is copied, after the initial record is created, but
before associated models are copied, a copy.prepare event is sent, e.g.
``model.folder.copy.prepare``. The event handler is passed a tuple of
``((original model document), (copied model document))``. If the copied model
is altered, the handler should save it without triggering events.
When the copy is fully complete, and copy.after event is sent, e.g.
``model.folder.copy.after``.
* **Override model validation**
You can also override or augment the default ``validate`` methods for a core
model type. Like the normal validation, you should raise a
``ValidationException`` for failure cases, and you can also ``preventDefault``
if you wish for the normal validation procedure not to be executed. The
identifier for these events is, e.g., ``model.user.validate``.
* **Override user authentication**
If you want to override or augment the normal user authentication process in
your plugin, bind to the ``auth.user.get`` event. If your plugin can
successfully authenticate the user, it should perform the logic it needs and
then ``preventDefault`` on the event and ``addResponse`` containing the
authenticated user document.
* **Before file upload**
This event is triggered as an upload is being initialized. The event
``model.upload.assetstore`` is sent before the ``model.upload.save`` event.
The event information is a dictionary containing ``model`` and ``resource``
with the resource model type and resource document of the upload parent. For
new uploads, the model type will be either ``item`` or ``folder``. When the
contents of a file are being replaced, this will be a ``file``. To change from
the current assetstore, add an ``assetstore`` key to the event information
dictionary that contains an assetstore model document.
* **Just before a file upload completes**
The event ``model.upload.finalize`` after the upload is completed but before
the new file is saved. This can be used if the file needs to be altered or the
upload should be cancelled at the last moment.
* **On file upload**
This event is always triggered asynchronously and is fired after a file has
been uploaded. The file document that was created is passed in the event info.
You can bind to this event using the identifier ``data.process``.
* **Before file move**
The event ``model.upload.movefile`` is triggered when a file is about to be
moved from one assetstore to another. The event information is a dictionary
containing ``file`` and ``assetstore`` with the current file document and the
target assetstore document. If ``preventDefault`` is called, the move will be
cancelled.
.. note:: If you anticipate your plugin being used as a dependency by other
plugins, and want to potentially alert them of your own events, it can
be worthwhile to trigger your own events from within the plugin. If you do
that, the identifiers for those events should begin with the name of your
plugin, e.g., ``events.trigger('cats.something_happened', info='foo')``
* **User login**
The event ``model.user.authenticate`` is fired when a user is attempting to
login via a username and password. This allows alternative authentication
modes to be used instead of core, or prior to attempting core authentication.
The event info contains two keys, "login" and "password".
Customizing the Swagger page
****************************
To customize text on the Swagger page, create a
`Mako template `_ file that inherits from the
base template and overrides one or more blocks. For example,
``plugins/cats/server/custom_api_docs.mako``:
.. code-block:: html+mako
<%inherit file="${context.get('baseTemplateFilename')}"/>
<%block name="docsHeader">
Cat programming interface
%block>
<%block name="docsBody">
Manage your cats using the resources below.
%block>
Install the custom template in the plugin's ``load`` function:
.. code-block:: python
import os
from girder.plugin import GirderPlugin
PLUGIN_PATH = os.path.dirname(__file__)
class CustomTemplatePlugin(GirderPlugin):
def load(self, info):
# Initially, the value of info['apiRoot'].templateFilename is
# 'api_docs.mako'. Because custom_api_docs.mako inherits from this
# base template, pass 'api_docs.mako' in the variable that the
# <%inherit> directive references.
baseTemplateFilename = info['apiRoot'].templateFilename
info['apiRoot'].updateHtmlVars({
'baseTemplateFilename': baseTemplateFilename
})
# Set the path to the custom template
templatePath = os.path.join(PLUGIN_PATH, 'custom_api_docs.mako')
info['apiRoot'].setTemplatePath(templatePath)
.. _client-side-plugins:
Extending the Client-Side Application
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
The web client may be extended independently of the server side. Plugins may
import Pug templates, Stylus files, and JavaScript files into the application.
All of your plugin's extensions to the web client must live in a directory inside
of your python package. By convention, this is in a directory called **web_client**. ::
cd girder_cats ; mkdir web_client
When present, this directory must contain a valid npm package, which includes a ``package.json``
file. (See the `npm documentation `_ for details.)
What follows is a typical npm package file for a Girder client side extension:
.. code-block:: json
{
"name": "@girder/cats",
"version": "1.0.0",
"peerDependencies": {
"@girder/core": "*",
"@girder/jobs": "*"
},
"dependencies": {
"othermodule": "^1.2.4"
},
"girderPlugin": {
"name": "cats",
"main": "./main.js",
"dependencies": ["jobs"],
"webpack": "webpack.helper"
}
}
In addition to the standard ``package.json`` properties, Girder plugins
**must** also define a ``girderPlugin`` object to register themselves with
Girder's client build system. The important keys in the object are as follows:
``name``
This must be **exactly** the entrypoint name registered in your ``setup.py`` file.
``main``
This is the entrypoint into your plugin on the client. All runtime initialization
should occur from here.
``dependencies``
This is an array of entrypoint names that your plugin depends on. Specifying this
explicitly here is what allows Girder's client build system to build the plugin
assets in the correct order.
``webpack``
This is an optional property whose value is a node module that exports a
function that can make arbitrary modification the webpack config used to
build the plugin bundle.
By default, Girder includes loaders for pug, stylus, css, fonts, and images
in all paths. For javascript inside the plugin, the code is transpiled
through babel using ``babel-preset-env``; however, this is not done for
dependencies resolved inside ``node_modules``. This option makes it
easy to include additional transpilation rules. For an example of this in
use, see the built in ``dicom_viewer`` plugin.
Core Girder code can be imported relative to the path **@girder/core**, for example
``import View from '@girder/core/views/View';``. The entry point defined in your
"main" file will be loaded into the browser after Girder's core library, but
before the application is initialized.
JavaScript extension capabilities
*********************************
Plugins may bind to any of the normal events triggered by core via a global
events object that can be imported like so:
.. code-block:: javascript
import events from '@girder/core/events';
...
this.listenTo(events, 'g:event_name', () => { do.something(); });
This will accommodate certain events, such as before
and after the application is initially loaded, and when a user logs in or out,
but most of the time plugins will augment the core system using the power of
JavaScript rather than the explicit events framework. One of the most common
use cases for plugins is to execute some code either before or after one of the
core model or view functions is executed. In an object-oriented language, this
would be a simple matter of extending the core class and making a call to the
parent method. The prototypal nature of JavaScript makes that pattern impossible;
instead, we'll use a slightly less straightforward but equally powerful
mechanism. This is best demonstrated by example. Let's say we want to execute
some code any time the core ``HierarchyWidget`` is rendered, for instance to
inject some additional elements into the view. We use Girder's ``wrap`` utility
function to ``wrap`` the method of the core prototype with our own function.
.. code-block:: javascript
import HierarchyWidget from '@girder/core/views/widgets/HierarchyWidget';
import { wrap } from '@girder/core/utilities/PluginUtils';
// Import our template file from our plugin using a relative path
import myTemplate from './templates/hierachyWidgetExtension.pug';
// CSS files pertaining to this view should be imported as a side-effect
import './stylesheets/hierarchyWidgetExtension.styl';
wrap(HierarchyWidget, 'render', function (render) {
// Call the underlying render function that we are wrapping
render.call(this);
// Add a link just below the widget using our custom template
this.$('.g-hierarchy-widget').after(myTemplate());
});
Notice that instead of simply calling ``render()``, we call ``render.call(this)``.
That is important, as otherwise the value of ``this`` will not be set properly
in the wrapped function.
Now that we have added the link to the core view, we can bind an event handler to
it to make it functional:
.. code-block:: javascript
HierarchyWidget.prototype.events['click a.cat-link'] = () => {
alert('meow!');
};
This demonstrates one simple use case for client plugins, but using these same
techniques, you should be able to do almost anything to change the core
application as you need.
JavaScript events
*****************
The JavaScript client handles notifications from the server and Backbone events
in client-specific code. The server notifications originate on the server and
can be monitored by both the server's Python code and the client's JavaScript
code. The client Backbone events are solely within the web client, and do not
get transmitted to the server.
If the connection to the server is interrupted, the client will not receive
server events. Periodically, the client will attempt to reconnect to the
server to resume handling events. Similarly, if client's browser tab is placed
in the background for a long enough period of time, the connection that listens
for server events will be stopped to prevent excessive resource use. When the
browser's tab regains focus, the client will once again receive server events.
When the connection to the server's event stream is interrupted, a
``g:eventStream.stop`` Backbone event is triggered on the ``EventStream``
object. When the server is once more sending events, it first sends a
``g:eventStream.start`` event. Clients can listen to these events and refresh
necessary components to ensure that data is current.
Setting an empty layout for a route
***********************************
If you have a route in your plugin that you would like to have an empty layout,
meaning that the Girder header, nav bar, and footer are hidden and the Girder body is
evenly padded and displayed, you can specify an empty layout in the ``navigateTo``
event trigger.
As an example, say your plugin wanted a ``frontPage`` route for a Collection which
would display the Collection with only the Girder body shown, you could add the following
route to your plugin.
.. code-block:: javascript
import events from '@girder/core/events';
import router from '@girder/core/router';
import { Layout } from '@girder/core/constants';
import CollectionModel from '@girder/core/models/CollectionModel';
import CollectionView from '@girder/core/views/body/CollectionView';
router.route('collection/:id/frontPage', 'collectionFrontPage', function (collectionId, params) {
var collection = new CollectionModel();
collection.set({
_id: collectionId
}).on('g:fetched', function () {
events.trigger('g:navigateTo', CollectionView, _.extend({
collection: collection
}, params || {}), {layout: Layout.EMPTY});
}, this).on('g:error', function () {
router.navigate('/collections', {trigger: true});
}, this).fetch();
});
Using another plugin inside a plugin
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
Girder plugins can use and extend other plugins as well. To do this, you need
to add and load the other plugin explicitly so that it installs and initializes
automatically. There are a number of places that the dependency relationship
needs to be specified.
* Python package
If you directly rely on another plugin for any reason, you should always add
the dependency to your plugin's ``setup.py`` file. This is done in the same
way all python dependencies are specified and will ensure that all the required
packages are installed when you plugin is "pip installed".
.. code-block:: python
# setup.py depending on girder-jobs and girder-homepage
setup(
name='girder-example-plugin',
# ...
install_requires=['girder-jobs', 'girder-homepage']
)
* Plugin loading
By default, Girder does not load its installed plugins in a deterministic order.
If your plugin depends on other Girder being loaded prior to itself, your plugin
must explicitly load the other dependent plugins during your plugin's own loading.
Girder will guarantee that a given plugin is actually loaded only once, so multiple
calls to load another plugin are safe and have no effect. Finally, it is possible
to check for the existence of another plugin before loading it or performing other
configuration, to support optional dependencies.
.. code-block:: python
from girder.plugin import getPlugin, GirderPlugin
# An example of loading dependent plugins
class ExamplePlugin(GirderPlugin)
def load(self, info):
getPlugin('jobs').load(info)
homepagePlugin = getPlugin('homepage')
if homepagePlugin:
# Optional dependency
homepagePlugin.load(info)
# ...
* Javascript client
If your plugin contains a javascript client and it imports code from another plugin, then
you need to add this dependency relationship to your web client ``package.json`` file. If
you depend on another plugin, but do not directly import code from the other package in you
javascript code, then this is not necessary.
.. code-block:: javascript
// package.json depending on "girder-jobs"
{
"name": "@girder/example",
"peerDependencies": {
"@girder/core": "*",
// This ensures that `import '@girder/jobs'` can be resolved.
"@girder/jobs": "*"
},
"girderPlugin": {
"name": "example",
"main": "./main.js",
// This ensures that "girder-jobs" is built before this plugin.
"dependencies": ["jobs"]
}
}
Automated testing for plugins
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
We recommend using `pytest `_ to create automated tests for your plugin code.
The core Girder development team maintains the `pytest-girder `_ package,
which contains several useful fixtures and other utilities that make testing Girder plugins easier.
Example
*******
This example shows the use of the ``server`` fixture, which spins up the Girder server and allows requests to be
made against its REST API.
.. code-block:: python
from girder_cats.models import Cat
import pytest
from pytest_girder.assertions import assertStatusOk
@pytest.mark.plugin('cats') # Makes sure the cats plugin will load for this test
def testCatCreation(server):
resp = server.request('/cat', method='POST', params={
'name': 'Helga',
'age': 4
})
assertStatusOk(resp)
records = Cat().find()
assert records.count() == 1
assert records[0]['name'] == 'Helga'