This is the plugin_template
repository to help plugin writers get started and write their own
plugin for Pulp Project 3.0+.
If you are planning on writing a new Pulp plugin, but have no idea what you're doing you've come to the right place. The purpose of this guide is to walk you through, step by step, the Pulp plugin creation process.
This guide specifically details how you write a new content plugin.
Why would you want to write a plugin?
What exactly is this Pulp thing?
It's recommend that you develop on a system that already has Pulp installed. This allows you to test your plugin at every step.
It's also recommended that you go through the planning guide before starting to develop your plugin.
The first step is to create a template_config.yml
for your new plugin. This file contains
settings used by the ./plugin-template
command when generating new plugins and for future updates.
The first time this file is generated. bootstrap this template. This will create a functional but useless plugin, with minimal code, docs, tests, and default Travis CI configuration. Later on we'll discuss exactly what each part of this template does and what to change to create a 'real' plugin.
-
Clone this repository
$ git clone https://github.com/pulp/plugin_template.git
$ cd plugin_template
-
Run the provided
./plugin-template --generate-config
$ ./plugin-template --generate-config --plugin-app-label <label> PLUGIN_NAME
NOTE : The
plugin-app-label
should identify the content type which you would like to support, e.g.rubygem
ormaven
. ThePLUGIN_NAME
is usuallypulp_
orpulp-
prepended to the--plugin-app-label
, e.g.pulp_maven
.
The first time this command is run, a new directory by the name of PLUGIN_NAME is created inside
the parent directory of the plugin_template' directory. The
template_config.yml` is written to
the root of this new directory. Subsequent uses of the command simply update that file.
The following settings are stored in template_config.yml
.
additional_plugins A list with additional plugins to be installed on Travis.
Each item in the list is a dict with the following fields:
name: the name of the plugin
branch: the git branch of the plugin. applies to non-tagged Travis jobs, such as PRs & cron jobs.
pip_version_specifier: A pip version specifier for when the gets installed from PyPI.
Applies to TAGGED (release) jobs.
See pulpcore_pip_version_specifier, but defaults to undefined, the latest.
pulpcore_branch The branch of pulpcore to check out and install on Travis.
This only applies to non-tagged Travis jobs, such as PRs & cron jobs.
"Required PR" in a commit message will override this.
Your requirements in "setup.py" may inadvertently override this as well.
pulpcore_pip_version_specifier
A pip version specifier for when pulpcore gets installed from PyPI.
This is only for Travis, and only applies to TAGGED (release) jobs.
An example is "~=3.0.0", which installs the latest 3.0.z,
and is equivalent to `pip install pulpcore~=3.0.0`.
Defaults to null, which installs the latest release from PyPI.
Your requirements in "setup.py" may inadvertently override this as well.
pypi_username The username that should be used when uploading packages to PyPI. It
is required unless deploy_client_to_pypi and deploy_daily_client_to_pypi
and deploy_to_pypi are specified.
docs_test Include a Travis build for testing the 'make html' command for sphinx docs.
deploy_to_pypi Include a Travis stage that publishes builds to PyPI
This stage only executes when a tag is associated with the commit being
built. When enabling this stage, the user is expected to provide a
secure environment variable called PYPI_PASSWORD. The variable can
be added in the travis-ci.com settings page for the project[0]. The PYPI
username is specified using --pypi-username option.
test_bindings Include a Travis stage that runs a script to test generated client
library.
This stage requires the plugin author to include a 'test_bindings.py'
script in the .travis directory of the plugin repository. This script
is supposed to exercise the generated client library.
test_s3 Include s3 job for running tests using [minio](https://github.com/minio/minio)
to emulate S3.
test_performance Include a Travis stage that runs a script to test performance. If using a
list, a separate stage will run a specific performance test file for each
entry in the list. Otherwise, all performance tests will be run together.
docker_fixtures In Travis, use the pulp-fixtures docker container to serve up fixtures
instead of using fedorapeople.org.
deploy_client_to_pypi Include a Travis stage that publishes a client library to PyPI.
This stage only executes when a tag is associated with the commit being
built. When enabling this stage, the user is expected to provide a
secure environment variable called PYPI_PASSWORD. The variable can
be added in the travis-ci.com settings page for the project[0]. The PYPI
username is specified using --pypi-username option.
This stage uses the OpenAPI schema for the plugin to generate a Python
client library using openapi-generator-cli.
deploy_client_to_rubygems
Include a Travis stage that publishes a client library to RubyGems.org.
This stage only executes when a tag is associated with the commit being
built. When enabling this stage, the user is expected to provide a
secure environment variable called RUBYGEMS_API_KEY. The variable can
be added in the travis-ci.com settings page for the project.
deploy_daily_client_to_pypi
Include a Travis stage that publishes a client library to PyPI.
This stage only executes when a tag is associated with the commit being
built. When enabling this stage, the user is expected to provide a
secure environment variable called PYPI_PASSWORD. The variable can
be added in the travis-ci.com settings page for the project[0]. The PYPI
username is specified using --pypi-username option.
This stage uses the OpenAPI schema for the plugin to generate a Python
client library using openapi-generator-cli.
[0] https://docs.travis-ci.com/user/environment-variables/
#defining-variables-in-repository-settings
deploy_daily_client_to_rubygems
Include a Travis stage that publishes a client library to RubyGems.org
with each CRON build.
This stage only executes on builds trigerred by CRON. When enabling
this stage, the user is expected to provide a secure environment
variable called RUBYGEMS_API_KEY. The variable can be added in the
travis-ci.com settings page for the project.
check_commit_message Include inspection of commit message for a reference to an issue in
pulp.plan.io.
coverage Include collection of coverage and reporting to coveralls.io
travis_addtl_services A list of additional services to add under .travis.yml . See
https://docs.travis-ci.com/user/database-setup/#starting-services for more info.
travis_notifications A yaml block that contains configuration for Travis build notifications. See
https://docs.travis-ci.com/user/notifications/ for configuration options.
cherry_pick_automation
A boolean that will enable a cron job to automatically attempt to cherry
pick changes to a stable release branch. This job will cherry pick any
PRs that are merged and have the label "Needs Cherry Pick". Requires the
template config setting `stable_branch` to be set. Also, requires a github
user account (e.g. pulpbot) to open the cherry pick PR. `GITHUB_USER` and
`GITHUB_TOKEN` must be defined as env variables in Travis. New tokens can be
generated at
[https://github.com/settings/tokens](https://github.com/settings/tokens).
stable_branch A string that points to the latest stable branch (e.g. "3.0"). This is used
for features like the cherry pick automation.
The next step is to bootstrap the plugin. This will create a functional but useless plugin, with minimal code and tests.
-
Run the
plugin-template --bootstrap
command. This will create a skeleton for your plugin. It will contain asetup.py
, expected plugin layout and stubs for necessary classes, methods, and tests.$ ./plugin-template --bootstrap PLUGIN_NAME
In addition to the basic plugin boilerplate, this template also provides a basic set of functional tests using the pulp_smash framework.
In order to use these tests, you will need to address the "FIXME" messages left in places where plugin-writer intervention is required.
The next step is to add a Travis configuration file and scripts for continuous integration. These are highly recommended, as they will make continuous verification of your plugin's functionality much easier.
-
Run the
./plugin-template --travis
command to generate the Travis CI config based on the settings intemplate_config.yml
.$ ./plugin-template --travis PLUGIN_NAME
Running the command again will update the plugin with the latest Travis CI configuration provided by the plugin-template.
The next step is to add documentation that can be hosted on Read the Docs.
-
Run the
./plugin-template --docs
command to generate the docs.$ ./plugin-template --docs PLUGIN_NAME
After bootstrapping, your plugin should be installable and discoverable by Pulp.
-
Install your bootstrapped plugin
pip install -e your_plugin_name
-
Start/restart the Pulp Server
django-admin runserver 24817
-
Check that everything worked and you have a remote endpoint
$ http GET http://localhost:24817/pulp/api/v3/remotes/{{ plugin_app_label }}/{{ plugin_dash_short }}/
The plugin specific /pulp/api/v3/content/{{ plugin_dash_short }}/
endpoints
should now also be available, and you can validate this by checking the hosted docs
http://localhost:24817/pulp/api/v3/docs
Your plugin is discoverable by Pulp because it is [a Django application that subclasses pulpcore.plugin.PulpPluginAppConfig]({{ plugin_snake }}/app/init.py)
For more information about plugin discoverability, including how it works and plugin entrypoints see the discoverability documentation
First, look at the overview of Pulp Models to understand how Pulp fits these pieces together.
Bootstrapping created three new endpoints (remote, and content). Additional information should be added to these to tell Pulp how to handle your content.
For each of these three endpoints, the bootstrap has created a model
, a serializer
and a viewset
.
The model is how the data is stored in the database.
The serializer converts complex data to easily parsable types (XML, JSON).
The viewset provides the handlers to serve/receive the serialized data.
Always subclass the relevant model, serializer, and viewset from the pulpcore.plugin
namespace. Pulp provides custom behavior for these, and although implementation details
are located in pulpcore.app
, plugins should always use pulpcore.plugin
instead,
since pulpcore.plugin
gurantees the plugin API semantic versioning
Models:
- model(s) for the specific content type(s) used in plugin, should be subclassed from pulpcore.plugin.models.Content model
- model(s) for the plugin specific repository(ies), should be subclassed from pulpcore.plugin.models.Repository model
- model(s) for the plugin specific remote(s), should be subclassed from pulpcore.plugin.models.Remote model
Serializers:
- serializer(s) for plugin specific content type(s), should be subclassed from pulpcore.plugin.serializers.ContentSerializer
- serializer(s) for plugin specific remote(s), should be subclassed from pulpcore.plugin.serializers.RemoteSerializer
- serializer(s) for plugin specific repository(ies), should be subclassed from pulpcore.plugin.serializers.RepositorySerializer
Viewsets:
- viewset(s) for plugin specific content type(s), should be subclassed from pulpcore.plugin.viewsets.ContentViewSet
- viewset(s) for plugin specific repository(ies), should be subclassed from pulpcore.plugin.viewsets.RepositoryViewset
- viewset(s) for plugin specific remote(s), should be subclassed from pulpcore.plugin.viewsets.RemoteViewset
Keep namespacing in mind when writing your viewsets.
First model your content type. This file is located at [{{ plugin_snake }}/app/models.py]({{ plugin_snake }}/app/models.py). Add any fields that correspond to the metadata of your content, the could be the project name, the author name, or any other type of metadata.
The TYPE
class attribute is used for filtering purposes.
If a uniqueness constraint is needed, add a Meta
class to the model like so:
class {{ plugin_camel_short }}Content(Content):
TYPE = '{{ plugin_dash_short }}'
filename = models.TextField(unique=True, db_index=True, blank=False)
class Meta:
unique_together = ('filename',)
After adding the model, you can run the migration with
pulp-manager makemigrations {{ plugin_app_label }}
And make sure all your fields are on the {{ plugin_app_label }} database table.
Next, add a corresponding serializer field on the in [{{ plugin_snake }}/app/serializers.py]({{ plugin_snake }}/app/serializers.py). See the DRF documentation on serializer fields to see what's available
Last, add any additional routes to your [{{ plugin_snake }}/app/viewsets.py]({{ plugin_snake }}/app/viewsets.py). The content viewset usually doesn't require any additional routes, so you can leave this alone for now.
Remotes provide metadata about how content should be downloaded into Pulp, such as the URL of the remote source, the download policy, and some authentication settings. The base Remote
class provided by Pulp Platform provides support for concurrent downloading of remote content.
First model your remote. This file is located at [{{ plugin_snake }}/app/models.py]({{ plugin_snake }}/app/models.py). Add any fields that correspond to the remote source.
Remember to define the TYPE
class attribute which is used for filtering purposes.
Next, add a corresponding serializer field on the in [{{ plugin_snake }}/app/serializers.py]({{ plugin_snake }}/app/serializers.py).
Last, add any additional routes to your [{{ plugin_snake }}/app/viewsets.py]({{ plugin_snake }}/app/viewsets.py). The remote viewset usually doesn't require any additinal routes, so you can leave this alone for now.
A Repository knows the specifics of which Content it supports and defines how to create new RepositoryVersions. It is also responsible for validating that those RepositoryVersions are valid.
First model your repository. This file is located at [{{ plugin_snake }}/app/models.py]({{ plugin_snake }}/app/models.py). Add any fields as necessary for your specific content type.
Remember to define the TYPE
class attribute which is used for filtering purposes, and CONTENT_TYPES
which
defines which types of content are supported by the Repository. This is a list of classes such as
{{ plugin_camel_short }}Content representing the various content types your plugin supports (that you want
this repository type to support, if there is more than one repository type in your plugin).
Also, if you want to provide validation that the whole collection of the content in your RepositoryVersion makes sense
together, you do that by defining finalize_new_version
on your repository model.
Next, add a corresponding serializer field on the in [{{ plugin_snake }}/app/serializers.py]({{ plugin_snake }}/app/serializers.py).
Last, add any additional routes to your [{{ plugin_snake }}/app/viewsets.py]({{ plugin_snake }}/app/viewsets.py). Note the sync route is predefined for you. This route kicks off a task [{{ plugin_snake }}.app.tasks.synchronizing.py]({{ plugin_snake }}.app.tasks.synchronizing.py).
If you have more than one Repository type in your plugin, or you change the name of your existing one, you will also
need to have a RepositoryVersionViewSet defined for it (just a viewset, no other objects needed). This hasfield, parent_viewset
, which should be set to the accompanying Repository class defined in your plugin.
TODO
Tasks such as sync and publish are needed to tell Pulp how to perform certain actions.
TODO
Your bootstrap template comes with a set of prepopulated docs. You can host these on readthedocs when you are ready.
Pulp also comes with a set of auto API docs. When your plugin is installed endpoints in the live api docs will be automatically populate.
When you run 'make html' command to build the docs, you must have the pulp-api running on
localhost. The 'make html' command first downloads the OpenAPI schema for the plugin and saves it
in docs/_static/api.json
. You should add this file to git. This file will then provide data
needed to display the restapi.html page in the root of the built docs.
This repository also provides a script for generating a Travis configuration. The script should be run with the following command.
$ ./plugin-template --travis plugin_name plugin_app_label
The default behavior enables two build stages that generate client libraries using the OpenAPI
schema. One publishes to PyPI using pypi-username
setting and the secret environment
variable called $PYPI_PASSWORD. The other stage publishes the client to rubygems.org and requires
the $RUBYGEMS_API_KEY environment variable to be set. Both environment variables can be created on
the travis-ci.com settings page for the plugin[0]. The stage that publishes tagged builds to PyPI
uses the same configs as the client publishing stage. The default pipeline can be created using the
following commands:
$ git clone git@github.com/pulp/{{ plugin_app_label }}
$ cd {{ plugin_app_label }}
$ # copying the requirements file is only needed if plugin was created before this file was added
$ # to the plugin template
$ cp doc_requirements.txt ../pulp_<plugin_name>/
$ touch ../pulp_<plugin_name>/.travis/test_bindings.py
$ ./plugin-template --travis --pypi-username your_pypi_username plugin_name
The before_install.sh, install.sh, before_script.sh, and script.sh can be augmented by plugin writers by creating specially named scripts in their .travis directories. The scripts are executed in the following order, with optional plugin provided scripts in bold:
- pre_before_install.sh
- before_install.sh
- post_before_install.sh
- install.sh
- pre_before_script.sh
- before_script.sh
- post_before_script.sh
- script.sh
- post_docs_test.sh
- post_script.sh
The pipeline can be modified, see the help text for available options.
$ ./plugin-template --help
usage: plugin-template [-h] [--generate-config] [--bootstrap] [--test]
[--travis] [--docs] [--all] [--verbose]
plugin_name plugin_app_label
Generate a .travis.yml and .travis directoryfor a specified plugin
positional arguments:
plugin_name Create or update this plugin.
plugin_app_label the Django app label for the plugin - usually the part after pulp_ or
pulp-
optional arguments:
-h, --help show this help message and exit
--bootstrap Create a new plugin and template boilerplate code.
--test Generate or update functional and unit tests.
--travis Generate or update travis configuration files.
--docs Generate or update plugin documentation.
--all Create a new plugin and template all non-excluded files.
--verbose Include more output.
-
Plugin django app is defined using PulpAppConfig as a parent
-
Plugin entry point is defined
-
Necessary models/serializers/viewsets are defined. At a minimum:
- models for plugin content type, repository, remote
- serializers for plugin content type, repository, remote
- viewset for plugin content type, repository, remote
-
Errors are handled according to Pulp conventions
-
Docs for plugin are available (any location and format preferred and provided by plugin writer)