Skip to main content

Low-level container for draft content

Project description

Introduction

plone.app.drafts implements services for managing auto-saved content drafts in Plone. This addresses two problems:

  • If the browser is accidentally closed or crashes whilst a page is being edited, all changes are lost.

  • Some data may need to be managed asynchronously, e.g. via a pop-up dialogue box in the visual editor. This data should not be saved until the form is saved (and in the case of an add form, it is impossible to do so).

The former problem pertains to any content add or edit form. The latter applies in particular to the “tiles” model as implemented by plone.app.tiles and its dependencies.

Installation

You can install plone.app.drafts as normal, by depending on it in your setup.py or adding it to the eggs list in your buildout. The package will self-configure for Plone; there is no need to add a ZCML slug.

You will also need to install the product from the portal_quickinstaller tool or Plone’s Add-ons control panel.

Draft storage

After installation, you should notice a new tool called portal_drafts in the ZMI. When drafts have been created, you can browse them here, and purge them if you believe they are stale.

To access the draft storage in code, look it up as a (local) utility:

>>> from zope.component import getUtility
>>> draftStorage = getUtility(IDraftStorage)

The draft storage contains methods for creating, finding and discarding drafts. This is mostly useful for integration logic.

A draft is accessed by using a hierarchy of keys, all strings:

  1. The user id of the user owning the draft

  2. A unique “target object” key that represents the object being drafted

  3. A unique draft name

The target object key is by convention a string representation of a unique integer id (via zope.intid) for drafts that represent existing content objects being edited, or a string like:

'<container-intid>:<portal_type>'

(e.g. '123456:Document') for drafts representing objects being added to a particular container.

The draft name is unique and assigned when the draft is created.

See the IDraftStorage interface for details on how to access drafts based on these keys.

The draft object itself is a minimal persistent object providing the IDraft interface. Importantly, it is not a full-blown content object. It has no intrinsic security, no workflow, and no standard fields. It is however annotatable, i.e. it may be adapted to IAnnotations.

A draft has a few basic attributes (__name__, _draftUserId, and _draftTargetKey), but is otherwise a blank canvas. Draft data may be stored as attributes, or in annotations. The attributes that are used depends on how the draft is integrated. The two primary patterns are:

Autosave

An explicit or timed background request submits the edit form to a handler, which extracts the form data and saves it on the draft object, e.g. in a form dictionary. The draft is updated periodically. On a successful save or cancel the draft is simply discarded. If the user returns to the edit screen after a browser crash or abandoned session, however, the request may be restored by copying the draft data to the real request.form dictionary prior to rendering the edit form. Provided the edit form is well-behaved, it should then show the last auto-saved values. These values can then be edited, before they are saved as normal.

Asynchronous updates

An AJAX dialogue box can be used to configure an object asynchronously. For example, a content type that supports attachments may use such a dialogue box to upload attached files. These must be stored temporarily, but should not be persisted with the real content object until the underlying edit form is saved (and should be discarded if it is cancelled). The file upload handler can save the data to the drafts storage, and then copy it to its final location on save.

A helper function called syncDraft is provided for this purpose in the plone.app.drafts.utils module. It looks up any number of named IDraftSyncer multi-adapters (on the draft object and the target content object) and calls them in turn.

Current draft management

To access the current draft from code, use the getCurrentDraft() helper function, passing it the current request:

>>> from plone.app.drafts.utils import getCurrentDraft
>>> currentDraft = getCurrentDraft(request)

This may return None if there is no current draft. It is possible that the necessary information for creating a draft (user id and target key) are known, but that no draft has been created yet. In this case, you can request that a new draft is created on demand, by passing create=True.

The current draft user id, target key and (once the draft has been created) draft name are looked up from the request, by adapting it to the interface ICurrentDraftManagement. You should not normally need to use this yourself, unless you are integrating the draft storage with an external framework.

The default ICurrentDraftManagement adapter allows the user id, target key and draft name to be set explicitly. If they are not set, they are read from the request. This means that they may come in request parameters, or in cookies. The request keys are plone.app.drafts.targetKey and plone.app.drafts.draftName. The user id always defaults to the currently logged in user’s id.

The ICurrentDraftManagement adapter also exposes lifecycle functions that can save or discard the current draft information. The default implementation does this using cookies that are set for a path corresponding to the edit page. It is the responsibility of the add/edit form integration code to ensure that this cookie is set for a path that is specific enough not to “leak” to other edit pages, but still allows AJAX dialogue boxes and other asynchronous requests to obtain the draft information if required.

Integration

Archetypes integration is provided in the archetypes module, which is configured if Archetypes is installed. The integration works as follows:

  • An IEditBegunEvent is fired by Archetypes when the user enters an add/edit form. An event handler for this event will calculate a target key for the context, taking “add forms” based on the portal_factory tool into account. Provided a key can be calculated, it is saved via an ICurrentDraftManagement adapter as explained below. A draft is not created immediately, but if a single draft is discovered in the storage for this user id and key, that draft name is saved so that it will be returned when getCurrentDraft() is called. The cookie path is calculated as well and saved. This ensures that if the draft is created in an asynchronous request with a “deeper” URL, the cookie path will be the same.

  • An event handler is installed for IObjectInitializedEvent and IObjectEditedEvent, which are fired when the user clicks Save on a valid add or edit form, respectively. This handler will get the current draft if it has been created during the editing cycle, and uses the syncDraft() method to synchronise it as necessary. The draft is then discarded, as is the current draft information, causing the cookies to expire.

  • An event handler is also installed for IEditCancelledEvent, which is fired when the user clicks Cancel. This simply discards the draft and current draft information without synchronisation.

The draft proxy

Simple drafting integration will tend to just store data on the draft object directly. However, it may sometimes be useful to have an object that behaves as a facade onto a “real” object, so that:

  • If an attribute or annotation value that has never been set on the draft is read, the value from the underlying target object is used.

  • If an attribute or annotation value is set, it is written to the draft. If it is subsequently read, it is read from the draft.

  • If an attribute or annotation value is deleted, it is deleted from the draft, and the fact that it was deleted is recorded so that this may be later synchronised with the underlying object when the draft is “saved” (e.g. via an IDraftSyncer adapter).

To get these semantics, create a DraftProxy object like so:

>>> from plone.app.drafts.proxy import DraftProxy
>>> proxy = DraftProxy(draft, target)

Here, draft is an IDraft object and target is the object it is a draft of. If attributes are deleted, they will be stored in one of two sets:

>>> deletedAttributes = getattr(draft, '_proxyDeleted', set())
>>> deletedAnnotations = getattr(draft, '_proxyAnnotationsDeleted', set())

Note that these attributes may not be present if nothing has ever been deleted, so we need to fetch them defensively.

Changelog

3.0.0 (2025-12-29)

Breaking changes:

  • Replace pkg_resources namespace with PEP 420 native namespace. Support only Plone 6.2 and Python 3.10+. (#3928)

Bug fixes:

  • Remove unneeded backwards compatibility code for old Zope and Plone versions. [maurits]

2.0.0 (2022-07-22)

  • remove Archetypes. This version targets Plone 5.2+ with Python 3.6+ only.

  • Fix class security warnings [petschki]

1.1.3 (2019-02-10)

  • Python 3 compatible [vangheem, petschki]

1.1.2 (2017-07-03)

  • Fix issue where draft sync failed because draft might have been without aq wrapper [datakurre]

1.1.1 (2016-09-09)

  • Remove forgotten debug print [datakurre]

1.1.0 (2016-09-09)

  • Add support for drafted content preview for Dexterity content when request is marked with IDisplayFormDrafting [datakurre]

  • Fix not set cookie values twice [vangheem]

  • Fix to always sync drafts before object modified event subscribers (especially indexing) are called [datakurre]

  • Behavior shortname plone.draftable added. [jensens, datakurre]

  • Update testing infrastructure and fix code analysis errors. [gforcada]

1.0 (2016-03-28)

  • Make sure draft is available before initializing the draft proxy object [vangheem]

1.0b3 (2015-06-10)

  • Fix issue where drafting caused ‘AttributeError: This object has no id’ [datakurre]

  • Fix issue where add forms with different content type but the context had conflicting draft [datakurre]

1.0b2 (2015-06-02)

  • Fix rare issue where Dexterity draft had wrong portal_type [datakurre]

1.0b1 (2015-05-26)

  • Add support for drafting on Dexterity add and edit forms when the drafting behavior is enabled for the content type [datakurre]

  • Add autosave (using AJAX validation calls) support for Dexterity add and edit forms when drafting behavior is enabled for the content type [datakurre]

  • Change to use UUIDs instead of intids [datakurre]

  • Change Archetypes-support to be disabled by default [datakurre]

    Archetypes-support can included in zcml with:

    <include package="plone.app.drafts" file="archetypes.zcml" />

1.0a2 (2011-10-11)

  • Added MANIFEST.in to fix our previous release. It was missing the history file.

1.0a1 (2011-10-10)

  • Initial release.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

plone_app_drafts-3.0.0.tar.gz (38.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

plone_app_drafts-3.0.0-py3-none-any.whl (30.7 kB view details)

Uploaded Python 3

File details

Details for the file plone_app_drafts-3.0.0.tar.gz.

File metadata

  • Download URL: plone_app_drafts-3.0.0.tar.gz
  • Upload date:
  • Size: 38.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.11

File hashes

Hashes for plone_app_drafts-3.0.0.tar.gz
Algorithm Hash digest
SHA256 0297d708cc35b843749681a4794c32d0b519de524f391e23ae2e34a39a4a6f31
MD5 65ed6fde4151e6e2e554c5d506265ce9
BLAKE2b-256 0d4cf57d2a9466d194508473c06af952379aae8db8ec2f2bcc496c9ba47befa4

See more details on using hashes here.

File details

Details for the file plone_app_drafts-3.0.0-py3-none-any.whl.

File metadata

File hashes

Hashes for plone_app_drafts-3.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 6644573584b742315405d84af493c63701ee3a626ded4948b2e97b04eca94b04
MD5 bf7216c60776922ff3e5a601df5dd051
BLAKE2b-256 161ac95e813940c8622d00cb12e112ac8bd4e70114f90b8de4e0fb66e2d33037

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page