title: Versioning summary: Add versioning to your database content through the Versioned extension. # Versioning Database content in SilverStripe can be "staged" before its publication, as well as track all changes through the lifetime of a database record. It is most commonly applied to pages in the CMS (the `SiteTree` class). Draft content edited in the CMS can be different from published content shown to your website visitors. Versioning in SilverStripe is handled through the [api:Versioned] class. As a [api:DataExtension] it is possible to be applied to any [api:DataObject] subclass. The extension class will automatically update read and write operations done via the ORM via the `augmentSQL` database hook. Adding Versioned to your `DataObject` subclass works the same as any other extension. It has one of two behaviours, which can be applied via the constructor argument. By default, adding the `Versioned extension will create a "Stage" and "Live" stage on your model, and will also track versioned history. :::php class MyStagedModel extends DataObject { private staic $extensions = [ "Versioned" ]; } Alternatively, staging can be disabled, so that only versioned changes are tracked for your model. This can be specified by setting the constructor argument to "Versioned" :::php class VersionedModel extends DataObject { private staic $extensions = [ "Versioned('Versioned')" ]; }
The extension is automatically applied to `SiteTree` class. For more information on extensions see [Extending](../extending) and the [Configuration](../configuration) documentation.
Versioning only works if you are adding the extension to the base class. That is, the first subclass of `DataObject`. Adding this extension to children of the base class will have unpredictable behaviour.
## Database Structure Depending on whether staging is enabled, one or more new tables will be created for your records. `_versions` is always created to track historic versions for your model. If staging is enabled this will also create a new `_Live` table once you've rebuilt the database.
Note that the "Stage" naming has a special meaning here, it will leave the original table name unchanged, rather than adding a suffix.
* `MyRecord` table: Contains staged data * `MyRecord_Live` table: Contains live data * `MyRecord_versions` table: Contains a version history (new record created on each save) Similarly, any subclass you create on top of a versioned base will trigger the creation of additional tables, which are automatically joined as required: * `MyRecordSubclass` table: Contains only staged data for subclass columns * `MyRecordSubclass_Live` table: Contains only live data for subclass columns * `MyRecordSubclass_versions` table: Contains only version history for subclass columns ## Usage ### Reading Versions By default, all records are retrieved from the "Draft" stage (so the `MyRecord` table in our example). You can explicitly request a certain stage through various getters on the `Versioned` class. :::php // Fetching multiple records $stageRecords = Versioned::get_by_stage('MyRecord', 'Stage'); $liveRecords = Versioned::get_by_stage('MyRecord', 'Live'); // Fetching a single record $stageRecord = Versioned::get_by_stage('MyRecord', 'Stage')->byID(99); $liveRecord = Versioned::get_by_stage('MyRecord', 'Live')->byID(99); ### Historical Versions The above commands will just retrieve the latest version of its respective stage for you, but not older versions stored in the `_versions` tables. :::php $historicalRecord = Versioned::get_version('MyRecord', , );
The record is retrieved as a `DataObject`, but saving back modifications via `write()` will create a new version, rather than modifying the existing one.
In order to get a list of all versions for a specific record, we need to generate specialized [api:Versioned_Version] objects, which expose the same database information as a `DataObject`, but also include information about when and how a record was published. :::php $record = MyRecord::get()->byID(99); // stage doesn't matter here $versions = $record->allVersions(); echo $versions->First()->Version; // instance of Versioned_Version ### Writing Versions and Changing Stages The usual call to `DataObject->write()` will write to whatever stage is currently active, as defined by the `Versioned::current_stage()` global setting. Each call will automatically create a new version in the `_versions` table. To avoid this, use [api:Versioned::writeWithoutVersion()] instead. To move a saved version from one stage to another, call [writeToStage()](api:Versioned->writeToStage()) on the object. The process of moving a version to a different stage is also called "publishing", so we've created a shortcut for this: `publish(, )`. :::php $record = Versioned::get_by_stage('MyRecord', 'Stage')->byID(99); $record->MyField = 'changed'; // will update `MyRecord` table (assuming Versioned::current_stage() == 'Stage'), // and write a row to `MyRecord_versions`. $record->write(); // will copy the saved record information to the `MyRecord_Live` table $record->publish('Stage', 'Live'); Similarly, an "unpublish" operation does the reverse, and removes a record from a specific stage. :::php $record = MyRecord::get()->byID(99); // stage doesn't matter here // will remove the row from the `MyRecord_Live` table $record->deleteFromStage('Live'); ### Forcing the Current Stage The current stage is stored as global state on the object. It is usually modified by controllers, e.g. when a preview is initialized. But it can also be set and reset temporarily to force a specific operation to run on a certain stage. :::php $origMode = Versioned::get_reading_mode(); // save current mode $obj = MyRecord::getComplexObjectRetrieval(); // returns 'Live' records Versioned::set_reading_mode('Stage'); // temporarily overwrite mode $obj = MyRecord::getComplexObjectRetrieval(); // returns 'Stage' records Versioned::set_reading_mode($origMode); // reset current mode ### File ownership Typically when publishing versioned dataobjects, it is necessary to ensure that some linked components are published along with it. Unless this is done, site front-end content can appear incorrectly published. For instance, a page which has a list of rotating banners will require that those banners are published whenever that page is. The solution to this problem is the ownership API, which declares a two-way relationship between objects along database relations. This relationship is similar to many_many/belongs_many_many and has_one/has_many, however it relies on a pre-existing relationship to function. For instance, in order to specify this dependency, you must apply `owns` and `owned_by` config on a relationship. When pages of type `MyPage` are published, any owned images and banners will be automatically published, without requiring any custom code. :::php class MyPage extends Page { private static $has_many = array( 'Banners' => 'Banner' ); private static $owns = array( 'Banners' ); } class Banner extends Page { private static $extensions = array( 'Versioned' ); private static $has_one = array( 'Parent' => 'MyPage', 'Image' => 'Image', ); private static $owned_by = array( 'Parent' ); private static $owns = array( 'Image' ); } class BannerImageExtension extends DataExtension { private static $has_many = array( 'Banners' => 'Banner' ); private static $owned_by = array( 'Banners' ); } With the config: :::yaml Image: extensions: - BannerImageExtension Note that it's important to define both `owns` and `owned_by` components of the relationship, similar to how you would apply `has_one` and `has_many`, or `many_many` and `belongs_many_many`. If you are using `[api:HTMLText]` or `[api:HTMLVarchar]` fields in your `DataObject::$db` definitions, it's likely that your authors can insert images into those fields via the CMS interface. These images are usually considered to be owned by the `DataObject`, and should be published alongside it. The ownership relationship is tracked through an `[image]` [shortcode](/developer-guides/extending/shortcodes), which is automatically transformed into an `` tag at render time. In addition to storing the image path, the shortcode references the database identifier of the `Image` object. ### Custom SQL We generally discourage writing `Versioned` queries from scratch, due to the complexities involved through joining multiple tables across an inherited table scheme (see [api:Versioned::augmentSQL()]). If possible, try to stick to smaller modifications of the generated `DataList` objects. Example: Get the first 10 live records, filtered by creation date: :::php $records = Versioned::get_by_stage('MyRecord', 'Live')->limit(10)->sort('Created', 'ASC'); ### Permissions By default, `Versioned` will come out of the box with security extensions which restrict the visibility of objects in Draft (stage) or Archive viewing mode.
As is standard practice, user code should always invoke `canView()` on any object before rendering it. DataLists do not filter on `canView()` automatically, so this must be done via user code. This be be achieved either by wrapping `<% if $canView %>` in your template, or by implementing your visibility check in PHP.
Versioned object visibility can be customised in one of the following ways by editing your user code: * Override the `canViewVersioned` method in your code. Make sure that this returns true or false if the user is not allowed to view this object in the current viewing mode. * Override the `canView` method to override the method visibility completely. E.g. :::php class MyObject extends DataObject { private static $extensions = array( 'Versioned' ); public function canViewVersioned($member = null) { // Check if site is live $mode = $this->getSourceQueryParam("Versioned.mode"); $stage = $this->getSourceQueryParam("Versioned.stage"); if ($mode === 'Stage' && $stage === 'Live') { return true; } // Only admins can view non-live objects return Permission::checkMember($member, 'ADMIN'); } } If you want to control permissions of an object in an extension, you can also use one of the below extension points in your `DataExtension` subclass: * `canView` to update the visibility of the object's `canView` * `canViewNonLive` to update the visibility of this object only in non-live mode. Note that unlike canViewVersioned, the canViewNonLive method will only be invoked if the object is in a non-published state. E.g. :::php class MyObjectExtension extends DataExtension { public function canViewNonLive($member = null) { return Permission::check($member, 'DRAFT_STATUS'); } } If none of the above checks are overridden, visibility will be determined by the permissions in the `TargetObject.non_live_permissions` config. E.g. :::php class MyObject extends DataObject { private static $extensions = array( 'Versioned' ); private static $non_live_permissions = array('ADMIN'); } Versioned applies no additional permissions to `canEdit` or `canCreate`, and such these permissions should be implemented as per standard unversioned DataObjects. ### Page Specific Operations Since the `Versioned` extension is primarily used for page objects, the underlying `SiteTree` class has some additional helpers. ### Templates Variables In templates, you don't need to worry about this distinction. The `$Content` variable contain the published content by default, and only preview draft content if explicitly requested (e.g. by the "preview" feature in the CMS, or by adding ?stage=Stage to the URL). If you want to force a specific stage, we recommend the `Controller->init()` method for this purpose, for example: **mysite/code/MyController.php** :::php public function init() { parent::init(); Versioned::set_reading_mode('Stage.Stage'); } ### Controllers The current stage for each request is determined by `VersionedRequestFilter` before any controllers initialize, through `Versioned::choose_site_stage()`. It checks for a `Stage` GET parameter, so you can force a draft stage by appending `?stage=Stage` to your request. The setting is "sticky" in the PHP session, so any subsequent requests will also be in draft stage.
The `choose_site_stage()` call only deals with setting the default stage, and doesn't check if the user is authenticated to view it. As with any other controller logic, please use `DataObject->canView()` to determine permissions, and avoid exposing unpublished content to your users.
## API Documentation * [api:Versioned]