Create a Plone distribution#
This section explains how a developer can create a custom Plone distribution. A Plone distribution is a pre-packaged version of Plone that includes specific features, themes, modules, and configurations.
See also
For a conceptual guide, see Plone distributions.
Create a backend add-on#
These instructions assume that you already have created a Plone backend add-on package, and now you want to add a distribution to it.
A Plone distribution exists inside a Python package that can be installed by pip
.
Update setup.py
#
Your package should follow conventions that make it discoverable by other developers.
In your setup.py
file, always add the correct Python Trove classifiers.
"Framework :: Plone",
"Framework :: Plone :: 6.1",
"Framework :: Plone :: Distribution",
Add plone.distribution
to your install_requires
stanza in your setup.py
file.
install_requires=[
"Products.CMFPlone",
"setuptools",
"plone.distribution",
],
Update configure.zcml
#
In your main configure.zcml
file, add the plone
XML namespace with the following declaration.
<configure
xmlns="http://namespaces.zope.org/zope"
xmlns:plone="http://namespaces.plone.org/plone"
>
Register plone.distribution
as a package to include with the following <include>
directive.
<include package="plone.distribution" />
Then declare the distributions you want to include in your package.
<plone:distribution
name="blog"
title="Personal Blog"
description="A Plone site already configured to host a personal blog."
directory="distributions/blog"
/>
The above example registers a distribution that will configure a personal blog with some default content.
Add distribution handlers#
When registering a distribution, you can provide a pre_handler
, a handler
, and a post_handler
, each of which must be a function with their respective signature, as shown in the following example.
def pre_handler(answers: dict) -> dict:
return answers
def handler(distribution: Distribution, site, answers: dict):
return site
def post_handler(distribution: Distribution, site, answers: dict):
return site
Each of those handlers will be called as follows.
pre_handler
Processes the answers to prepare the distribution before creating the site.
handler
Runs after creating the bare Plone site instead of the default handler. It installs the required GenericSetup profiles and creates the content.
post_handler
Runs after the site is set up.
To add extra configuration to your Plone site, and assuming you added extra inputs to the Plone site creation form, then you can add your own handler, registering it as shown in the following example.
<plone:distribution
name="blog"
title="Personal Blog"
description="A Plone site already configured to host a personal Blog."
directory="distributions/blog"
post_handler=".handlers.blog.post_handler"
/>
Add a distribution folder#
To organize your distribution configuration, you can follow the convention to use the distributions/<distribution_name>
folder in the root of your package.
In that folder, you need to provide the items described in the following sections.
image.png
#
A 1080 pixels wide by 768 pixels tall image in PNG format representing your distribution. It could be the default page of a new site, your logo, or any other way of representing your distribution.
profiles.json
#
A file profiles.json
containing the GenericSetup profiles that your distribution uses during installation.
This file needs to contain two keys.
base
List of profiles to install in every new site using this distribution.
content
List of profiles to install when the user decides to create a site with example content.
As an example, the configuration for a new Plone site with Volto as its frontend would be the following.
{
"base": [
"plone.app.contenttypes:default",
"plone.app.caching:default",
"plonetheme.barceloneta:default",
"plone.volto:default"
],
"content": [
"plone.volto:default-homepage"
]
}
schema.json
#
If you require additional input from the user during site creation, you can customize the form using the schema.json
file.
The file should contain two keys.
schema
A JSON Schema definition.
uischema
A
react-jsonschema-form
configuration to modify the display of the form.
The schema
should have at least the following keys.
site_id
title
description
default_language
portal_timezone
setup_content
The following code example is the content of the schema.json
file for creating the site.
{
"schema": {
"title": "Create a Plone site",
"description": "Adds a new Plone content management system site to the underlying application server.",
"type": "object",
"required": [
"site_id",
"title"
],
"properties": {
"site_id": {
"type": "string",
"title": "Path Identifier",
"default": "Plone",
"description": "The ID of the site. No special characters or spaces are allowed. This ends up as part of the URL unless hidden by an upstream web server."
},
"title": {
"type": "string",
"title": "Title",
"default": "Site",
"description": "A short title for the site. This will be shown as part of the title of the browser window on each page."
},
"description": {
"type": "string",
"title": "Site Description",
"default": "A Plone Site"
},
"default_language": {"$ref": "#/definitions/languages"},
"portal_timezone": {"$ref": "#/definitions/timezones"},
"setup_content": {
"type": "boolean",
"title": "Create Content",
"description": "Should example content be added during site creation?",
"default": false
}
}
},
"uischema": {
}
}
Note
You may have noticed the entries for both default_language
and portal_timezone
.
"default_language": {"$ref": "#/definitions/languages"},
"portal_timezone": {"$ref": "#/definitions/timezones"},
plone.distribution
adds both definitions at runtime, providing a list of languages and timezones available on the installation.
If you want to hide fields in the form, you can use the uischema
.
This is especially useful for fields that must be in the schema, but for which you always want the default value.
The next lines hide three fields:
{
"schema": {
"…": "…"
},
"uischema": {
"description": {
"ui:widget": "hidden"
},
"default_language": {
"ui:widget": "hidden"
},
"portal_timezone": {
"ui:widget": "hidden"
}
}
}
Add a dependency on an add-on#
If you want to add a Plone backend add-on to your Plone distribution, then you must perform the following steps.
Add your add-on, such as collective.person
, to your setup.py
file's install_requires
stanza.
install_requires=[
"setuptools",
"Plone",
"plone.distribution>=1.0.0b2",
"plone.api",
"collective.person",
],
Then add it to your dependencies.zcml
file.
<!-- List all packages your distribution depends on here -->
<include package="plone.volto" />
<include package="plone.restapi" />
<include package="collective.person" />
<include package="plone.distribution" />
</configure>
Finally, add it to your profiles.json
file.
"base": [
"plone.app.contenttypes:default",
"plone.app.caching:default",
"plone.restapi:default",
"plone.volto:default",
"collective.person:default",
"plonetheme.barceloneta:default"
],
Add example content#
The distribution loads its content from JSON data in the content
folder.
To export content from a site into this folder, use the bin/export-distribution
script.
bin/export-distribution path/to/zope.conf Plone
In the example above, Plone
is the ID of the Plone site to export.
Limit available distributions#
By default, Plone 6.1 ships with two ready-to-use distributions.
default
Create a Plone site with the Volto frontend.
classic
Create a Plone site with the Classic UI frontend.
If you want to limit the choice of distributions when creating a new site, you can set the environment variable ALLOWED_DISTRIBUTIONS
to a comma-separated sting of only those distributions' names.
ALLOWED_DISTRIBUTIONS=default