mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
MINOR Moved SiteTree, SiteTreeDecorator, SiteConfig, ErrorPage, RedirectorPage, VirtualPage (and related tests) to "cms" module
This commit is contained in:
parent
5bf5b744d8
commit
204fd2c4ce
@ -1,275 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* ErrorPage holds the content for the page of an error response.
|
|
||||||
* Renders the page on each publish action into a static HTML file
|
|
||||||
* within the assets directory, after the naming convention
|
|
||||||
* /assets/error-<statuscode>.html.
|
|
||||||
* This enables us to show errors even if PHP experiences a recoverable error.
|
|
||||||
* ErrorPages
|
|
||||||
*
|
|
||||||
* @see Debug::friendlyError()
|
|
||||||
*
|
|
||||||
* @package cms
|
|
||||||
*/
|
|
||||||
class ErrorPage extends Page {
|
|
||||||
|
|
||||||
static $db = array(
|
|
||||||
"ErrorCode" => "Int",
|
|
||||||
);
|
|
||||||
|
|
||||||
static $defaults = array(
|
|
||||||
"ShowInMenus" => 0,
|
|
||||||
"ShowInSearch" => 0
|
|
||||||
);
|
|
||||||
|
|
||||||
static $icon = array("sapphire/javascript/tree/images/page", "file");
|
|
||||||
|
|
||||||
protected static $static_filepath = ASSETS_PATH;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get a {@link SS_HTTPResponse} to response to a HTTP error code if an {@link ErrorPage} for that code is present.
|
|
||||||
*
|
|
||||||
* @param int $statusCode
|
|
||||||
* @return SS_HTTPResponse
|
|
||||||
*/
|
|
||||||
public static function response_for($statusCode) {
|
|
||||||
// first look for cached files
|
|
||||||
$cachedPath = self::get_filepath_for_errorcode($statusCode, Translatable::get_current_locale());
|
|
||||||
|
|
||||||
if(file_exists($cachedPath)) {
|
|
||||||
$response = new SS_HTTPResponse();
|
|
||||||
|
|
||||||
$response->setStatusCode($statusCode);
|
|
||||||
$response->setBody(file_get_contents($cachedPath));
|
|
||||||
|
|
||||||
return $response;
|
|
||||||
} else if($errorPage = DataObject::get_one('ErrorPage', "\"ErrorCode\" = $statusCode")) {
|
|
||||||
// then attempt to dynamically generate the error page
|
|
||||||
return ModelAsController::controller_for($errorPage)->handleRequest(new SS_HTTPRequest('GET', ''));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Ensures that there is always a 404 page
|
|
||||||
* by checking if there's an instance of
|
|
||||||
* ErrorPage with a 404 and 500 error code. If there
|
|
||||||
* is not, one is created when the DB is built.
|
|
||||||
*/
|
|
||||||
function requireDefaultRecords() {
|
|
||||||
parent::requireDefaultRecords();
|
|
||||||
|
|
||||||
// Ensure that an assets path exists before we do any error page creation
|
|
||||||
if(!file_exists(ASSETS_PATH)) {
|
|
||||||
mkdir(ASSETS_PATH);
|
|
||||||
}
|
|
||||||
|
|
||||||
$pageNotFoundErrorPage = DataObject::get_one('ErrorPage', "\"ErrorCode\" = '404'");
|
|
||||||
$pageNotFoundErrorPageExists = ($pageNotFoundErrorPage && $pageNotFoundErrorPage->exists()) ? true : false;
|
|
||||||
$pageNotFoundErrorPagePath = self::get_filepath_for_errorcode(404);
|
|
||||||
if(!($pageNotFoundErrorPageExists && file_exists($pageNotFoundErrorPagePath))) {
|
|
||||||
if(!$pageNotFoundErrorPageExists) {
|
|
||||||
$pageNotFoundErrorPage = new ErrorPage();
|
|
||||||
$pageNotFoundErrorPage->ErrorCode = 404;
|
|
||||||
$pageNotFoundErrorPage->Title = _t('ErrorPage.DEFAULTERRORPAGETITLE', 'Page not found');
|
|
||||||
$pageNotFoundErrorPage->Content = _t('ErrorPage.DEFAULTERRORPAGECONTENT', '<p>Sorry, it seems you were trying to access a page that doesn\'t exist.</p><p>Please check the spelling of the URL you were trying to access and try again.</p>');
|
|
||||||
$pageNotFoundErrorPage->write();
|
|
||||||
$pageNotFoundErrorPage->publish('Stage', 'Live');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure a static error page is created from latest error page content
|
|
||||||
$response = Director::test(Director::makeRelative($pageNotFoundErrorPage->Link()));
|
|
||||||
if($fh = fopen($pageNotFoundErrorPagePath, 'w')) {
|
|
||||||
$written = fwrite($fh, $response->getBody());
|
|
||||||
fclose($fh);
|
|
||||||
}
|
|
||||||
|
|
||||||
if($written) {
|
|
||||||
DB::alteration_message('404 error page created', 'created');
|
|
||||||
} else {
|
|
||||||
DB::alteration_message(sprintf('404 error page could not be created at %s. Please check permissions', $pageNotFoundErrorPagePath), 'error');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$serverErrorPage = DataObject::get_one('ErrorPage', "\"ErrorCode\" = '500'");
|
|
||||||
$serverErrorPageExists = ($serverErrorPage && $serverErrorPage->exists()) ? true : false;
|
|
||||||
$serverErrorPagePath = self::get_filepath_for_errorcode(500);
|
|
||||||
if(!($serverErrorPageExists && file_exists($serverErrorPagePath))) {
|
|
||||||
if(!$serverErrorPageExists) {
|
|
||||||
$serverErrorPage = new ErrorPage();
|
|
||||||
$serverErrorPage->ErrorCode = 500;
|
|
||||||
$serverErrorPage->Title = _t('ErrorPage.DEFAULTSERVERERRORPAGETITLE', 'Server error');
|
|
||||||
$serverErrorPage->Content = _t('ErrorPage.DEFAULTSERVERERRORPAGECONTENT', '<p>Sorry, there was a problem with handling your request.</p>');
|
|
||||||
$serverErrorPage->write();
|
|
||||||
$serverErrorPage->publish('Stage', 'Live');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure a static error page is created from latest error page content
|
|
||||||
$response = Director::test(Director::makeRelative($serverErrorPage->Link()));
|
|
||||||
if($fh = fopen($serverErrorPagePath, 'w')) {
|
|
||||||
$written = fwrite($fh, $response->getBody());
|
|
||||||
fclose($fh);
|
|
||||||
}
|
|
||||||
|
|
||||||
if($written) {
|
|
||||||
DB::alteration_message('500 error page created', 'created');
|
|
||||||
} else {
|
|
||||||
DB::alteration_message(sprintf('500 error page could not be created at %s. Please check permissions', $serverErrorPagePath), 'error');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function getCMSFields() {
|
|
||||||
$fields = parent::getCMSFields();
|
|
||||||
|
|
||||||
$fields->addFieldToTab(
|
|
||||||
"Root.Content.Main",
|
|
||||||
new DropdownField(
|
|
||||||
"ErrorCode",
|
|
||||||
$this->fieldLabel('ErrorCode'),
|
|
||||||
array(
|
|
||||||
400 => _t('ErrorPage.400', '400 - Bad Request'),
|
|
||||||
401 => _t('ErrorPage.401', '401 - Unauthorized'),
|
|
||||||
403 => _t('ErrorPage.403', '403 - Forbidden'),
|
|
||||||
404 => _t('ErrorPage.404', '404 - Not Found'),
|
|
||||||
405 => _t('ErrorPage.405', '405 - Method Not Allowed'),
|
|
||||||
406 => _t('ErrorPage.406', '406 - Not Acceptable'),
|
|
||||||
407 => _t('ErrorPage.407', '407 - Proxy Authentication Required'),
|
|
||||||
408 => _t('ErrorPage.408', '408 - Request Timeout'),
|
|
||||||
409 => _t('ErrorPage.409', '409 - Conflict'),
|
|
||||||
410 => _t('ErrorPage.410', '410 - Gone'),
|
|
||||||
411 => _t('ErrorPage.411', '411 - Length Required'),
|
|
||||||
412 => _t('ErrorPage.412', '412 - Precondition Failed'),
|
|
||||||
413 => _t('ErrorPage.413', '413 - Request Entity Too Large'),
|
|
||||||
414 => _t('ErrorPage.414', '414 - Request-URI Too Long'),
|
|
||||||
415 => _t('ErrorPage.415', '415 - Unsupported Media Type'),
|
|
||||||
416 => _t('ErrorPage.416', '416 - Request Range Not Satisfiable'),
|
|
||||||
417 => _t('ErrorPage.417', '417 - Expectation Failed'),
|
|
||||||
500 => _t('ErrorPage.500', '500 - Internal Server Error'),
|
|
||||||
501 => _t('ErrorPage.501', '501 - Not Implemented'),
|
|
||||||
502 => _t('ErrorPage.502', '502 - Bad Gateway'),
|
|
||||||
503 => _t('ErrorPage.503', '503 - Service Unavailable'),
|
|
||||||
504 => _t('ErrorPage.504', '504 - Gateway Timeout'),
|
|
||||||
505 => _t('ErrorPage.505', '505 - HTTP Version Not Supported'),
|
|
||||||
)
|
|
||||||
),
|
|
||||||
"Content"
|
|
||||||
);
|
|
||||||
|
|
||||||
return $fields;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* When an error page is published, create a static HTML page with its
|
|
||||||
* content, so the page can be shown even when SilverStripe is not
|
|
||||||
* functioning correctly before publishing this page normally.
|
|
||||||
* @param string|int $fromStage Place to copy from. Can be either a stage name or a version number.
|
|
||||||
* @param string $toStage Place to copy to. Must be a stage name.
|
|
||||||
* @param boolean $createNewVersion Set this to true to create a new version number. By default, the existing version number will be copied over.
|
|
||||||
*/
|
|
||||||
function doPublish() {
|
|
||||||
parent::doPublish();
|
|
||||||
|
|
||||||
// Run the page (reset the theme, it might've been disabled by LeftAndMain::init())
|
|
||||||
$oldTheme = SSViewer::current_theme();
|
|
||||||
SSViewer::set_theme(SSViewer::current_custom_theme());
|
|
||||||
$response = Director::test(Director::makeRelative($this->Link()));
|
|
||||||
SSViewer::set_theme($oldTheme);
|
|
||||||
|
|
||||||
$errorContent = $response->getBody();
|
|
||||||
|
|
||||||
// Make the base tag dynamic.
|
|
||||||
// $errorContent = preg_replace('/<base[^>]+href="' . str_replace('/','\\/', Director::absoluteBaseURL()) . '"[^>]*>/i', '<base href="$BaseURL" />', $errorContent);
|
|
||||||
|
|
||||||
// Check we have an assets base directory, creating if it we don't
|
|
||||||
if(!file_exists(ASSETS_PATH)) {
|
|
||||||
mkdir(ASSETS_PATH, 02775);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// if the page is published in a language other than default language,
|
|
||||||
// write a specific language version of the HTML page
|
|
||||||
$filePath = self::get_filepath_for_errorcode($this->ErrorCode, $this->Locale);
|
|
||||||
if($fh = fopen($filePath, "w")) {
|
|
||||||
fwrite($fh, $errorContent);
|
|
||||||
fclose($fh);
|
|
||||||
} else {
|
|
||||||
$fileErrorText = sprintf(
|
|
||||||
_t(
|
|
||||||
"ErrorPage.ERRORFILEPROBLEM",
|
|
||||||
"Error opening file \"%s\" for writing. Please check file permissions."
|
|
||||||
),
|
|
||||||
$errorFile
|
|
||||||
);
|
|
||||||
FormResponse::status_message($fileErrorText, 'bad');
|
|
||||||
FormResponse::respond();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @param boolean $includerelations a boolean value to indicate if the labels returned include relation fields
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
function fieldLabels($includerelations = true) {
|
|
||||||
$labels = parent::fieldLabels($includerelations);
|
|
||||||
$labels['ErrorCode'] = _t('ErrorPage.CODE', "Error code");
|
|
||||||
|
|
||||||
return $labels;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns an absolute filesystem path to a static error file
|
|
||||||
* which is generated through {@link publish()}.
|
|
||||||
*
|
|
||||||
* @param int $statusCode A HTTP Statuscode, mostly 404 or 500
|
|
||||||
* @param String $locale A locale, e.g. 'de_DE' (Optional)
|
|
||||||
* @return String
|
|
||||||
*/
|
|
||||||
static function get_filepath_for_errorcode($statusCode, $locale = null) {
|
|
||||||
if (singleton('ErrorPage')->hasMethod('alternateFilepathForErrorcode')) {
|
|
||||||
return singleton('ErrorPage')-> alternateFilepathForErrorcode($statusCode, $locale);
|
|
||||||
}
|
|
||||||
if(singleton('SiteTree')->hasExtension('Translatable') && $locale && $locale != Translatable::default_locale()) {
|
|
||||||
return self::$static_filepath . "/error-{$statusCode}-{$locale}.html";
|
|
||||||
} else {
|
|
||||||
return self::$static_filepath . "/error-{$statusCode}.html";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the path where static error files are saved through {@link publish()}.
|
|
||||||
* Defaults to /assets.
|
|
||||||
*
|
|
||||||
* @param string $path
|
|
||||||
*/
|
|
||||||
static function set_static_filepath($path) {
|
|
||||||
self::$static_filepath = $path;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
static function get_static_filepath() {
|
|
||||||
return self::$static_filepath;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Controller for ErrorPages.
|
|
||||||
* @package cms
|
|
||||||
*/
|
|
||||||
class ErrorPage_Controller extends Page_Controller {
|
|
||||||
function init() {
|
|
||||||
parent::init();
|
|
||||||
|
|
||||||
$action = $this->request->param('Action');
|
|
||||||
if(!$action || $action == 'index') {
|
|
||||||
Director::set_status_code($this->failover->ErrorCode ? $this->failover->ErrorCode : 404);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
?>
|
|
@ -1,181 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* A redirector page redirects when the page is visited.
|
|
||||||
*
|
|
||||||
* @package cms
|
|
||||||
* @subpackage content
|
|
||||||
*/
|
|
||||||
class RedirectorPage extends Page {
|
|
||||||
|
|
||||||
static $icon = array("cms/images/treeicons/page-shortcut","file");
|
|
||||||
|
|
||||||
static $db = array(
|
|
||||||
"RedirectionType" => "Enum('Internal,External','Internal')",
|
|
||||||
"ExternalURL" => "Varchar(2083)" // 2083 is the maximum length of a URL in Internet Explorer.
|
|
||||||
);
|
|
||||||
|
|
||||||
static $defaults = array(
|
|
||||||
"RedirectionType" => "Internal"
|
|
||||||
);
|
|
||||||
|
|
||||||
static $has_one = array(
|
|
||||||
"LinkTo" => "SiteTree",
|
|
||||||
);
|
|
||||||
|
|
||||||
static $many_many = array(
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns this page if the redirect is external, otherwise
|
|
||||||
* returns the target page.
|
|
||||||
* @return SiteTree
|
|
||||||
*/
|
|
||||||
function ContentSource() {
|
|
||||||
if($this->RedirectionType == 'Internal') {
|
|
||||||
return $this->LinkTo();
|
|
||||||
} else {
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the the link that should be used for this redirector page, in navigation, etc.
|
|
||||||
* If the redirectorpage has been appropriately configured, then it will return the redirection
|
|
||||||
* destination, to prevent unnecessary 30x redirections. However, if it's misconfigured, then
|
|
||||||
* it will return a link to itself, which will then display an error message.
|
|
||||||
*/
|
|
||||||
function Link() {
|
|
||||||
if($link = $this->redirectionLink()) return $link;
|
|
||||||
else return $this->regularLink();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the normal link directly to this page. Once you visit this link, a 30x redirection
|
|
||||||
* will take you to your final destination.
|
|
||||||
*/
|
|
||||||
function regularLink($action = null) {
|
|
||||||
return parent::Link($action);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the link that we should redirect to.
|
|
||||||
* Only return a value if there is a legal redirection destination.
|
|
||||||
*/
|
|
||||||
function redirectionLink() {
|
|
||||||
if($this->RedirectionType == 'External') {
|
|
||||||
if($this->ExternalURL) {
|
|
||||||
return $this->ExternalURL;
|
|
||||||
}
|
|
||||||
|
|
||||||
} else {
|
|
||||||
$linkTo = $this->LinkToID ? DataObject::get_by_id("SiteTree", $this->LinkToID) : null;
|
|
||||||
|
|
||||||
if($linkTo) {
|
|
||||||
// We shouldn't point to ourselves - that would create an infinite loop! Return null since we have a
|
|
||||||
// bad configuration
|
|
||||||
if($this->ID == $linkTo->ID) {
|
|
||||||
return null;
|
|
||||||
|
|
||||||
// If we're linking to another redirectorpage then just return the URLSegment, to prevent a cycle of redirector
|
|
||||||
// pages from causing an infinite loop. Instead, they will cause a 30x redirection loop in the browser, but
|
|
||||||
// this can be handled sufficiently gracefully by the browser.
|
|
||||||
} elseif($linkTo instanceof RedirectorPage) {
|
|
||||||
return $linkTo->regularLink();
|
|
||||||
|
|
||||||
// For all other pages, just return the link of the page.
|
|
||||||
} else {
|
|
||||||
return $linkTo->Link();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function syncLinkTracking() {
|
|
||||||
if ($this->RedirectionType == 'Internal') {
|
|
||||||
if($this->LinkToID) {
|
|
||||||
$this->HasBrokenLink = DataObject::get_by_id('SiteTree', $this->LinkToID) ? false : true;
|
|
||||||
} else {
|
|
||||||
// An incomplete redirector page definitely has a broken link
|
|
||||||
$this->HasBrokenLink = true;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// TODO implement checking of a remote site
|
|
||||||
$this->HasBrokenLink = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function onBeforeWrite() {
|
|
||||||
parent::onBeforeWrite();
|
|
||||||
|
|
||||||
// Prefix the URL with "http://" if no prefix is found
|
|
||||||
if($this->ExternalURL && (strpos($this->ExternalURL, '://') === false)) {
|
|
||||||
$this->ExternalURL = 'http://' . $this->ExternalURL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function getCMSFields() {
|
|
||||||
Requirements::javascript(SAPPHIRE_DIR . "/javascript/RedirectorPage.js");
|
|
||||||
|
|
||||||
$fields = parent::getCMSFields();
|
|
||||||
$fields->removeByName('Content', true);
|
|
||||||
|
|
||||||
// Remove all metadata fields, does not apply for redirector pages
|
|
||||||
$fields->removeByName('MetaTagsHeader');
|
|
||||||
$fields->removeByName('MetaTitle');
|
|
||||||
$fields->removeByName('MetaKeywords');
|
|
||||||
$fields->removeByName('MetaDescription');
|
|
||||||
$fields->removeByName('ExtraMeta');
|
|
||||||
|
|
||||||
$fields->addFieldsToTab('Root.Content.Main',
|
|
||||||
array(
|
|
||||||
new HeaderField('RedirectorDescHeader',_t('RedirectorPage.HEADER', "This page will redirect users to another page")),
|
|
||||||
new OptionsetField(
|
|
||||||
"RedirectionType",
|
|
||||||
_t('RedirectorPage.REDIRECTTO', "Redirect to"),
|
|
||||||
array(
|
|
||||||
"Internal" => _t('RedirectorPage.REDIRECTTOPAGE', "A page on your website"),
|
|
||||||
"External" => _t('RedirectorPage.REDIRECTTOEXTERNAL', "Another website"),
|
|
||||||
),
|
|
||||||
"Internal"
|
|
||||||
),
|
|
||||||
new TreeDropdownField(
|
|
||||||
"LinkToID",
|
|
||||||
_t('RedirectorPage.YOURPAGE', "Page on your website"),
|
|
||||||
"SiteTree"
|
|
||||||
),
|
|
||||||
new TextField("ExternalURL", _t('RedirectorPage.OTHERURL', "Other website URL"))
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
return $fields;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Don't cache RedirectorPages
|
|
||||||
function subPagesToCache() {
|
|
||||||
return array();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Controller for the {@link RedirectorPage}.
|
|
||||||
* @package cms
|
|
||||||
* @subpackage content
|
|
||||||
*/
|
|
||||||
class RedirectorPage_Controller extends Page_Controller {
|
|
||||||
function init() {
|
|
||||||
if($link = $this->redirectionLink()) {
|
|
||||||
Director::redirect($link, 301);
|
|
||||||
}
|
|
||||||
|
|
||||||
parent::init();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* If we ever get this far, it means that the redirection failed.
|
|
||||||
*/
|
|
||||||
function Content() {
|
|
||||||
return "<p class=\"message-setupWithoutRedirect\">" .
|
|
||||||
_t('RedirectorPage.HASBEENSETUP', 'A redirector page has been set up without anywhere to redirect to.') .
|
|
||||||
"</p>";
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,289 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* Sitewide configuration.
|
|
||||||
*
|
|
||||||
* h2. Translation
|
|
||||||
*
|
|
||||||
* To enable translation of configurations alongside the {@link Translatable} extension.
|
|
||||||
* This also allows assigning language-specific toplevel permissions for viewing and editing
|
|
||||||
* pages, in addition to the normal `TRANSLATE_*`/`TRANSLATE_ALL` permissions.
|
|
||||||
*
|
|
||||||
* Object::add_extension('SiteConfig', 'Translatable');
|
|
||||||
*
|
|
||||||
* @author Tom Rix
|
|
||||||
* @package cms
|
|
||||||
*/
|
|
||||||
class SiteConfig extends DataObject implements PermissionProvider {
|
|
||||||
static $db = array(
|
|
||||||
"Title" => "Varchar(255)",
|
|
||||||
"Tagline" => "Varchar(255)",
|
|
||||||
"Theme" => "Varchar(255)",
|
|
||||||
"CanViewType" => "Enum('Anyone, LoggedInUsers, OnlyTheseUsers', 'Anyone')",
|
|
||||||
"CanEditType" => "Enum('LoggedInUsers, OnlyTheseUsers', 'LoggedInUsers')",
|
|
||||||
"CanCreateTopLevelType" => "Enum('LoggedInUsers, OnlyTheseUsers', 'LoggedInUsers')",
|
|
||||||
);
|
|
||||||
|
|
||||||
static $many_many = array(
|
|
||||||
"ViewerGroups" => "Group",
|
|
||||||
"EditorGroups" => "Group",
|
|
||||||
"CreateTopLevelGroups" => "Group"
|
|
||||||
);
|
|
||||||
|
|
||||||
protected static $disabled_themes = array();
|
|
||||||
|
|
||||||
public static function disable_theme($theme) {
|
|
||||||
self::$disabled_themes[$theme] = $theme;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the fields that are sent to the CMS. In
|
|
||||||
* your decorators: updateCMSFields(&$fields)
|
|
||||||
*
|
|
||||||
* @return Fieldset
|
|
||||||
*/
|
|
||||||
function getCMSFields() {
|
|
||||||
Requirements::javascript(CMS_DIR . "/javascript/SitetreeAccess.js");
|
|
||||||
|
|
||||||
$fields = new FieldSet(
|
|
||||||
new TabSet("Root",
|
|
||||||
$tabMain = new Tab('Main',
|
|
||||||
$titleField = new TextField("Title", _t('SiteConfig.SITETITLE', "Site title")),
|
|
||||||
$taglineField = new TextField("Tagline", _t('SiteConfig.SITETAGLINE', "Site Tagline/Slogan")),
|
|
||||||
new DropdownField("Theme", _t('SiteConfig.THEME', 'Theme'), $this->getAvailableThemes(), '', null, _t('SiteConfig.DEFAULTTHEME', '(Use default theme)'))
|
|
||||||
),
|
|
||||||
$tabAccess = new Tab('Access',
|
|
||||||
new HeaderField('WhoCanViewHeader', _t('SiteConfig.VIEWHEADER', "Who can view pages on this site?"), 2),
|
|
||||||
$viewersOptionsField = new OptionsetField("CanViewType"),
|
|
||||||
$viewerGroupsField = new TreeMultiselectField("ViewerGroups", _t('SiteTree.VIEWERGROUPS', "Viewer Groups")),
|
|
||||||
new HeaderField('WhoCanEditHeader', _t('SiteConfig.EDITHEADER', "Who can edit pages on this site?"), 2),
|
|
||||||
$editorsOptionsField = new OptionsetField("CanEditType"),
|
|
||||||
$editorGroupsField = new TreeMultiselectField("EditorGroups", _t('SiteTree.EDITORGROUPS', "Editor Groups")),
|
|
||||||
new HeaderField('WhoCanCreateTopLevelHeader', _t('SiteConfig.TOPLEVELCREATE', "Who can create pages in the root of the site?"), 2),
|
|
||||||
$topLevelCreatorsOptionsField = new OptionsetField("CanCreateTopLevelType"),
|
|
||||||
$topLevelCreatorsGroupsField = new TreeMultiselectField("CreateTopLevelGroups", _t('SiteTree.TOPLEVELCREATORGROUPS', "Top level creators"))
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
$viewersOptionsSource = array();
|
|
||||||
$viewersOptionsSource["Anyone"] = _t('SiteTree.ACCESSANYONE', "Anyone");
|
|
||||||
$viewersOptionsSource["LoggedInUsers"] = _t('SiteTree.ACCESSLOGGEDIN', "Logged-in users");
|
|
||||||
$viewersOptionsSource["OnlyTheseUsers"] = _t('SiteTree.ACCESSONLYTHESE', "Only these people (choose from list)");
|
|
||||||
$viewersOptionsField->setSource($viewersOptionsSource);
|
|
||||||
|
|
||||||
$editorsOptionsSource = array();
|
|
||||||
$editorsOptionsSource["LoggedInUsers"] = _t('SiteTree.EDITANYONE', "Anyone who can log-in to the CMS");
|
|
||||||
$editorsOptionsSource["OnlyTheseUsers"] = _t('SiteTree.EDITONLYTHESE', "Only these people (choose from list)");
|
|
||||||
$editorsOptionsField->setSource($editorsOptionsSource);
|
|
||||||
|
|
||||||
$topLevelCreatorsOptionsField->setSource($editorsOptionsSource);
|
|
||||||
|
|
||||||
// Translatable doesn't handle updateCMSFields on DataObjects,
|
|
||||||
// so add it here to save the current Locale,
|
|
||||||
// because onBeforeWrite does not work.
|
|
||||||
if(Object::has_extension('SiteConfig',"Translatable")){
|
|
||||||
$fields->push(new HiddenField("Locale"));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!Permission::check('EDIT_SITECONFIG')) {
|
|
||||||
$fields->makeFieldReadonly($viewersOptionsField);
|
|
||||||
$fields->makeFieldReadonly($viewerGroupsField);
|
|
||||||
$fields->makeFieldReadonly($editorsOptionsField);
|
|
||||||
$fields->makeFieldReadonly($editorGroupsField);
|
|
||||||
$fields->makeFieldReadonly($topLevelCreatorsOptionsField);
|
|
||||||
$fields->makeFieldReadonly($topLevelCreatorsGroupsField);
|
|
||||||
$fields->makeFieldReadonly($taglineField);
|
|
||||||
$fields->makeFieldReadonly($titleField);
|
|
||||||
}
|
|
||||||
|
|
||||||
if(file_exists(BASE_PATH . '/install.php')) {
|
|
||||||
$fields->addFieldToTab("Root.Main", new LiteralField("InstallWarningHeader",
|
|
||||||
"<p class=\"message warning\">" . _t("SiteTree.REMOVE_INSTALL_WARNING",
|
|
||||||
"Warning: You should remove install.php from this SilverStripe install for security reasons.")
|
|
||||||
. "</p>"), "Title");
|
|
||||||
}
|
|
||||||
|
|
||||||
$tabMain->setTitle(_t('SiteConfig.TABMAIN', "Main"));
|
|
||||||
$tabAccess->setTitle(_t('SiteConfig.TABACCESS', "Access"));
|
|
||||||
$this->extend('updateCMSFields', $fields);
|
|
||||||
|
|
||||||
return $fields;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get all available themes that haven't been marked as disabled.
|
|
||||||
* @param string $baseDir Optional alternative theme base directory for testing
|
|
||||||
* @return array of theme directory names
|
|
||||||
*/
|
|
||||||
public function getAvailableThemes($baseDir = null) {
|
|
||||||
$themes = SSViewer::get_themes($baseDir);
|
|
||||||
foreach(self::$disabled_themes as $theme) {
|
|
||||||
if(isset($themes[$theme])) unset($themes[$theme]);
|
|
||||||
}
|
|
||||||
return $themes;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the actions that are sent to the CMS. In
|
|
||||||
* your decorators: updateEditFormActions(&$actions)
|
|
||||||
*
|
|
||||||
* @return Fieldset
|
|
||||||
*/
|
|
||||||
function getCMSActions() {
|
|
||||||
if (Permission::check('ADMIN') || Permission::check('EDIT_SITECONFIG')) {
|
|
||||||
$actions = new FieldSet(
|
|
||||||
new FormAction('save_siteconfig', _t('CMSMain.SAVE','Save'))
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
$actions = new FieldSet();
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->extend('updateCMSActions', $actions);
|
|
||||||
|
|
||||||
return $actions;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the current sites SiteConfig, and creates a new one
|
|
||||||
* through {@link make_site_config()} if none is found.
|
|
||||||
*
|
|
||||||
* @param string $locale
|
|
||||||
* @return SiteConfig
|
|
||||||
*/
|
|
||||||
static function current_site_config($locale = null) {
|
|
||||||
if(Object::has_extension('SiteConfig',"Translatable")){
|
|
||||||
$locale = isset($locale) ? $locale : Translatable::get_current_locale();
|
|
||||||
$siteConfig = Translatable::get_one_by_locale('SiteConfig', $locale);
|
|
||||||
} else {
|
|
||||||
$siteConfig = DataObject::get_one('SiteConfig');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!$siteConfig) $siteConfig = self::make_site_config($locale);
|
|
||||||
|
|
||||||
return $siteConfig;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Setup a default SiteConfig record if none exists
|
|
||||||
*/
|
|
||||||
function requireDefaultRecords() {
|
|
||||||
parent::requireDefaultRecords();
|
|
||||||
$siteConfig = DataObject::get_one('SiteConfig');
|
|
||||||
if(!$siteConfig) {
|
|
||||||
self::make_site_config();
|
|
||||||
DB::alteration_message("Added default site config","created");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create SiteConfig with defaults from language file.
|
|
||||||
* if Translatable is enabled on SiteConfig, see if one already exist
|
|
||||||
* and use those values for the translated defaults.
|
|
||||||
*
|
|
||||||
* @param string $locale
|
|
||||||
* @return SiteConfig
|
|
||||||
*/
|
|
||||||
static function make_site_config($locale = null) {
|
|
||||||
if(!$locale) $locale = Translatable::get_current_locale();
|
|
||||||
|
|
||||||
$siteConfig = new SiteConfig();
|
|
||||||
$siteConfig->Title = _t('SiteConfig.SITENAMEDEFAULT',"Your Site Name");
|
|
||||||
$siteConfig->Tagline = _t('SiteConfig.TAGLINEDEFAULT',"your tagline here");
|
|
||||||
|
|
||||||
if($siteConfig->hasExtension('Translatable')){
|
|
||||||
$defaultConfig = DataObject::get_one('SiteConfig');
|
|
||||||
if($defaultConfig){
|
|
||||||
$siteConfig->Title = $defaultConfig->Title;
|
|
||||||
$siteConfig->Tagline = $defaultConfig->Tagline;
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO Copy view/edit group settings
|
|
||||||
|
|
||||||
// set the correct Locale
|
|
||||||
$siteConfig->Locale = $locale;
|
|
||||||
}
|
|
||||||
|
|
||||||
$siteConfig->write();
|
|
||||||
|
|
||||||
return $siteConfig;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Can a user view pages on this site? This method is only
|
|
||||||
* called if a page is set to Inherit, but there is nothing
|
|
||||||
* to inherit from.
|
|
||||||
*
|
|
||||||
* @param mixed $member
|
|
||||||
* @return boolean
|
|
||||||
*/
|
|
||||||
public function canView($member = null) {
|
|
||||||
if(!$member) $member = Member::currentUserID();
|
|
||||||
if($member && is_numeric($member)) $member = DataObject::get_by_id('Member', $member);
|
|
||||||
|
|
||||||
if (!$this->CanViewType || $this->CanViewType == 'Anyone') return true;
|
|
||||||
|
|
||||||
// check for any logged-in users
|
|
||||||
if($this->CanViewType == 'LoggedInUsers' && $member) return true;
|
|
||||||
|
|
||||||
// check for specific groups
|
|
||||||
if($this->CanViewType == 'OnlyTheseUsers' && $member && $member->inGroups($this->ViewerGroups())) return true;
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Can a user edit pages on this site? This method is only
|
|
||||||
* called if a page is set to Inherit, but there is nothing
|
|
||||||
* to inherit from.
|
|
||||||
*
|
|
||||||
* @param mixed $member
|
|
||||||
* @return boolean
|
|
||||||
*/
|
|
||||||
public function canEdit($member = null) {
|
|
||||||
if(!$member) $member = Member::currentUserID();
|
|
||||||
if($member && is_numeric($member)) $member = DataObject::get_by_id('Member', $member);
|
|
||||||
|
|
||||||
// check for any logged-in users
|
|
||||||
if(!$this->CanEditType || $this->CanEditType == 'LoggedInUsers' && $member) return true;
|
|
||||||
|
|
||||||
// check for specific groups
|
|
||||||
if($this->CanEditType == 'OnlyTheseUsers' && $member && $member->inGroups($this->EditorGroups())) return true;
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
function providePermissions() {
|
|
||||||
return array(
|
|
||||||
'EDIT_SITECONFIG' => array(
|
|
||||||
'name' => _t('SiteConfig.EDIT_PERMISSION', 'Manage site configuration'),
|
|
||||||
'category' => _t('Permissions.PERMISSIONS_CATEGORY', 'Roles and access permissions'),
|
|
||||||
'help' => _t('SiteConfig.EDIT_PERMISSION_HELP', 'Ability to edit global access settings/top-level page permissions.'),
|
|
||||||
'sort' => 400
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Can a user create pages in the root of this site?
|
|
||||||
*
|
|
||||||
* @param mixed $member
|
|
||||||
* @return boolean
|
|
||||||
*/
|
|
||||||
public function canCreateTopLevel($member = null) {
|
|
||||||
if(!$member || !(is_a($member, 'Member')) || is_numeric($member)) {
|
|
||||||
$member = Member::currentUserID();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Permission::check('ADMIN')) return true;
|
|
||||||
|
|
||||||
// check for any logged-in users
|
|
||||||
if($this->CanCreateTopLevelType == 'LoggedInUsers' && $member) return true;
|
|
||||||
|
|
||||||
// check for specific groups
|
|
||||||
if($member && is_numeric($member)) $member = DataObject::get_by_id('Member', $member);
|
|
||||||
if($this->CanCreateTopLevelType == 'OnlyTheseUsers' && $member && $member->inGroups($this->CreateTopLevelGroups())) return true;
|
|
||||||
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,2595 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* Basic data-object representing all pages within the site tree.
|
|
||||||
* This data-object takes care of the heirachy. All page types that live within the heirachy
|
|
||||||
* should inherit from this.
|
|
||||||
*
|
|
||||||
* In addition, it contains a number of static methods for querying the site tree.
|
|
||||||
* @package cms
|
|
||||||
*/
|
|
||||||
class SiteTree extends DataObject implements PermissionProvider,i18nEntityProvider {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Indicates what kind of children this page type can have.
|
|
||||||
* This can be an array of allowed child classes, or the string "none" -
|
|
||||||
* indicating that this page type can't have children.
|
|
||||||
* If a classname is prefixed by "*", such as "*Page", then only that
|
|
||||||
* class is allowed - no subclasses. Otherwise, the class and all its
|
|
||||||
* subclasses are allowed.
|
|
||||||
*
|
|
||||||
* @var array
|
|
||||||
*/
|
|
||||||
static $allowed_children = array("SiteTree");
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The default child class for this page.
|
|
||||||
*
|
|
||||||
* @var string
|
|
||||||
*/
|
|
||||||
static $default_child = "Page";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The default parent class for this page.
|
|
||||||
*
|
|
||||||
* @var string
|
|
||||||
*/
|
|
||||||
static $default_parent = null;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Controls whether a page can be in the root of the site tree.
|
|
||||||
*
|
|
||||||
* @var bool
|
|
||||||
*/
|
|
||||||
static $can_be_root = true;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* List of permission codes a user can have to allow a user to create a
|
|
||||||
* page of this type.
|
|
||||||
*
|
|
||||||
* @var array
|
|
||||||
*/
|
|
||||||
static $need_permission = null;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* If you extend a class, and don't want to be able to select the old class
|
|
||||||
* in the cms, set this to the old class name. Eg, if you extended Product
|
|
||||||
* to make ImprovedProduct, then you would set $hide_ancestor to Product.
|
|
||||||
*
|
|
||||||
* @var string
|
|
||||||
*/
|
|
||||||
static $hide_ancestor = null;
|
|
||||||
|
|
||||||
static $db = array(
|
|
||||||
"URLSegment" => "Varchar(255)",
|
|
||||||
"Title" => "Varchar(255)",
|
|
||||||
"MenuTitle" => "Varchar(100)",
|
|
||||||
"Content" => "HTMLText",
|
|
||||||
"MetaTitle" => "Varchar(255)",
|
|
||||||
"MetaDescription" => "Text",
|
|
||||||
"MetaKeywords" => "Varchar(255)",
|
|
||||||
"ExtraMeta" => "HTMLText",
|
|
||||||
"ShowInMenus" => "Boolean",
|
|
||||||
"ShowInSearch" => "Boolean",
|
|
||||||
"HomepageForDomain" => "Varchar(100)",
|
|
||||||
"Sort" => "Int",
|
|
||||||
"HasBrokenFile" => "Boolean",
|
|
||||||
"HasBrokenLink" => "Boolean",
|
|
||||||
"ReportClass" => "Varchar",
|
|
||||||
"CanViewType" => "Enum('Anyone, LoggedInUsers, OnlyTheseUsers, Inherit', 'Inherit')",
|
|
||||||
"CanEditType" => "Enum('LoggedInUsers, OnlyTheseUsers, Inherit', 'Inherit')",
|
|
||||||
|
|
||||||
// Simple task tracking
|
|
||||||
"ToDo" => "Text",
|
|
||||||
);
|
|
||||||
|
|
||||||
static $indexes = array(
|
|
||||||
"URLSegment" => true,
|
|
||||||
);
|
|
||||||
|
|
||||||
static $many_many = array(
|
|
||||||
"LinkTracking" => "SiteTree",
|
|
||||||
"ImageTracking" => "File",
|
|
||||||
"ViewerGroups" => "Group",
|
|
||||||
"EditorGroups" => "Group",
|
|
||||||
);
|
|
||||||
|
|
||||||
static $belongs_many_many = array(
|
|
||||||
"BackLinkTracking" => "SiteTree"
|
|
||||||
);
|
|
||||||
|
|
||||||
static $many_many_extraFields = array(
|
|
||||||
"LinkTracking" => array("FieldName" => "Varchar"),
|
|
||||||
"ImageTracking" => array("FieldName" => "Varchar")
|
|
||||||
);
|
|
||||||
|
|
||||||
static $casting = array(
|
|
||||||
"Breadcrumbs" => "HTMLText",
|
|
||||||
"LastEdited" => "SS_Datetime",
|
|
||||||
"Created" => "SS_Datetime",
|
|
||||||
'Link' => 'Text',
|
|
||||||
'RelativeLink' => 'Text',
|
|
||||||
'AbsoluteLink' => 'Text',
|
|
||||||
);
|
|
||||||
|
|
||||||
static $defaults = array(
|
|
||||||
"ShowInMenus" => 1,
|
|
||||||
"ShowInSearch" => 1,
|
|
||||||
"CanViewType" => "Inherit",
|
|
||||||
"CanEditType" => "Inherit"
|
|
||||||
);
|
|
||||||
|
|
||||||
static $versioning = array(
|
|
||||||
"Stage", "Live"
|
|
||||||
);
|
|
||||||
|
|
||||||
static $default_sort = "\"Sort\"";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* If this is false, the class cannot be created in the CMS.
|
|
||||||
* @var boolean
|
|
||||||
*/
|
|
||||||
static $can_create = true;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Icon to use in the CMS
|
|
||||||
*
|
|
||||||
* This should be the base filename. The suffixes -file.gif,
|
|
||||||
* -openfolder.gif and -closedfolder.gif will be appended to the base name
|
|
||||||
* that you provide there.
|
|
||||||
* If you prefer, you can pass an array:
|
|
||||||
* array("sapphire\javascript\tree\images\page", $option).
|
|
||||||
* $option can be either "file" or "folder" to force the icon to always
|
|
||||||
* be a file or folder, regardless of whether the page has children or not
|
|
||||||
*
|
|
||||||
* @var string|array
|
|
||||||
*/
|
|
||||||
static $icon = array("sapphire/javascript/tree/images/page", "file");
|
|
||||||
|
|
||||||
|
|
||||||
static $extensions = array(
|
|
||||||
"Hierarchy",
|
|
||||||
"Versioned('Stage', 'Live')",
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Delimit breadcrumb-links generated by BreadCrumbs()
|
|
||||||
*
|
|
||||||
* @var string
|
|
||||||
*/
|
|
||||||
public static $breadcrumbs_delimiter = " » ";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Whether or not to write the homepage map for static publisher
|
|
||||||
*/
|
|
||||||
public static $write_homepage_map = true;
|
|
||||||
|
|
||||||
static $searchable_fields = array(
|
|
||||||
'Title',
|
|
||||||
'Content',
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see SiteTree::nested_urls()
|
|
||||||
*/
|
|
||||||
private static $nested_urls = false;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see SiteTree::set_create_defaultpages()
|
|
||||||
*/
|
|
||||||
private static $create_default_pages = true;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This controls whether of not extendCMSFields() is called by getCMSFields.
|
|
||||||
*/
|
|
||||||
private static $runCMSFieldsExtensions = true;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Cache for canView/Edit/Publish/Delete permissions
|
|
||||||
*/
|
|
||||||
public static $cache_permissions = array();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see SiteTree::enforce_strict_hierarchy()
|
|
||||||
*/
|
|
||||||
private static $enforce_strict_hierarchy = true;
|
|
||||||
|
|
||||||
public static function set_enforce_strict_hierarchy($to) {
|
|
||||||
self::$enforce_strict_hierarchy = $to;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function get_enforce_strict_hierarchy() {
|
|
||||||
return self::$enforce_strict_hierarchy;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns TRUE if nested URLs (e.g. page/sub-page/) are currently enabled on this site.
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public static function nested_urls() {
|
|
||||||
return self::$nested_urls;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function enable_nested_urls() {
|
|
||||||
self::$nested_urls = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function disable_nested_urls() {
|
|
||||||
self::$nested_urls = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the (re)creation of default pages on /dev/build
|
|
||||||
*
|
|
||||||
* @param bool $option
|
|
||||||
*/
|
|
||||||
public static function set_create_default_pages($option = true) {
|
|
||||||
self::$create_default_pages = $option;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Fetches the {@link SiteTree} object that maps to a link.
|
|
||||||
*
|
|
||||||
* If you have enabled {@link SiteTree::nested_urls()} on this site, then you can use a nested link such as
|
|
||||||
* "about-us/staff/", and this function will traverse down the URL chain and grab the appropriate link.
|
|
||||||
*
|
|
||||||
* Note that if no model can be found, this method will fall over to a decorated alternateGetByLink method provided
|
|
||||||
* by a decorator attached to {@link SiteTree}
|
|
||||||
*
|
|
||||||
* @param string $link
|
|
||||||
* @param bool $cache
|
|
||||||
* @return SiteTree
|
|
||||||
*/
|
|
||||||
public static function get_by_link($link, $cache = true) {
|
|
||||||
if(trim($link, '/')) {
|
|
||||||
$link = trim(Director::makeRelative($link), '/');
|
|
||||||
} else {
|
|
||||||
$link = RootURLController::get_homepage_link();
|
|
||||||
}
|
|
||||||
|
|
||||||
$parts = Convert::raw2sql(preg_split('|/+|', $link));
|
|
||||||
|
|
||||||
// Grab the initial root level page to traverse down from.
|
|
||||||
$URLSegment = array_shift($parts);
|
|
||||||
$sitetree = DataObject::get_one (
|
|
||||||
'SiteTree', "\"URLSegment\" = '$URLSegment'" . (self::nested_urls() ? ' AND "ParentID" = 0' : ''), $cache
|
|
||||||
);
|
|
||||||
|
|
||||||
/// Fall back on a unique URLSegment for b/c.
|
|
||||||
if(!$sitetree && self::nested_urls() && $pages = DataObject::get('SiteTree', "\"URLSegment\" = '$URLSegment'")) {
|
|
||||||
return ($pages->Count() == 1) ? $pages->First() : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Attempt to grab an alternative page from decorators.
|
|
||||||
if(!$sitetree) {
|
|
||||||
$parentID = self::nested_urls() ? 0 : null;
|
|
||||||
|
|
||||||
if($alternatives = singleton('SiteTree')->extend('alternateGetByLink', $URLSegment, $parentID)) {
|
|
||||||
foreach($alternatives as $alternative) if($alternative) $sitetree = $alternative;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(!$sitetree) return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if we have any more URL parts to parse.
|
|
||||||
if(!self::nested_urls() || !count($parts)) return $sitetree;
|
|
||||||
|
|
||||||
// Traverse down the remaining URL segments and grab the relevant SiteTree objects.
|
|
||||||
foreach($parts as $segment) {
|
|
||||||
$next = DataObject::get_one (
|
|
||||||
'SiteTree', "\"URLSegment\" = '$segment' AND \"ParentID\" = $sitetree->ID", $cache
|
|
||||||
);
|
|
||||||
|
|
||||||
if(!$next) {
|
|
||||||
$parentID = (int) $sitetree->ID;
|
|
||||||
|
|
||||||
if($alternatives = singleton('SiteTree')->extend('alternateGetByLink', $segment, $parentID)) {
|
|
||||||
foreach($alternatives as $alternative) if($alternative) $next = $alternative;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(!$next) return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
$sitetree->destroy();
|
|
||||||
$sitetree = $next;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $sitetree;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return a subclass map of SiteTree
|
|
||||||
* that shouldn't be hidden through
|
|
||||||
* {@link SiteTree::$hide_ancestor}
|
|
||||||
*
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
public static function page_type_classes() {
|
|
||||||
$classes = ClassInfo::getValidSubClasses();
|
|
||||||
|
|
||||||
$baseClassIndex = array_search('SiteTree', $classes);
|
|
||||||
if($baseClassIndex !== FALSE) unset($classes[$baseClassIndex]);
|
|
||||||
|
|
||||||
$kill_ancestors = array();
|
|
||||||
|
|
||||||
// figure out if there are any classes we don't want to appear
|
|
||||||
foreach($classes as $class) {
|
|
||||||
$instance = singleton($class);
|
|
||||||
|
|
||||||
// do any of the progeny want to hide an ancestor?
|
|
||||||
if($ancestor_to_hide = $instance->stat('hide_ancestor')) {
|
|
||||||
// note for killing later
|
|
||||||
$kill_ancestors[] = $ancestor_to_hide;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If any of the descendents don't want any of the elders to show up, cruelly render the elders surplus to requirements.
|
|
||||||
if($kill_ancestors) {
|
|
||||||
$kill_ancestors = array_unique($kill_ancestors);
|
|
||||||
foreach($kill_ancestors as $mark) {
|
|
||||||
// unset from $classes
|
|
||||||
$idx = array_search($mark, $classes);
|
|
||||||
unset($classes[$idx]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $classes;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Replace a "[sitetree_link id=n]" shortcode with a link to the page with the corresponding ID.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public static function link_shortcode_handler($arguments, $content = null, $parser = null) {
|
|
||||||
if(!isset($arguments['id']) || !is_numeric($arguments['id'])) return;
|
|
||||||
|
|
||||||
if (
|
|
||||||
!($page = DataObject::get_by_id('SiteTree', $arguments['id'])) // Get the current page by ID.
|
|
||||||
&& !($page = Versioned::get_latest_version('SiteTree', $arguments['id'])) // Attempt link to old version.
|
|
||||||
&& !($page = DataObject::get_one('ErrorPage', '"ErrorCode" = \'404\'')) // Link to 404 page directly.
|
|
||||||
) {
|
|
||||||
return; // There were no suitable matches at all.
|
|
||||||
}
|
|
||||||
|
|
||||||
if($content) {
|
|
||||||
return sprintf('<a href="%s">%s</a>', $page->Link(), $parser->parse($content));
|
|
||||||
} else {
|
|
||||||
return $page->Link();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the link for this {@link SiteTree} object, with the {@link Director::baseURL()} included.
|
|
||||||
*
|
|
||||||
* @param string $action Optional controller action (method).
|
|
||||||
* Note: URI encoding of this parameter is applied automatically through template casting,
|
|
||||||
* don't encode the passed parameter.
|
|
||||||
* Please use {@link Controller::join_links()} instead to append GET parameters.
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function Link($action = null) {
|
|
||||||
return Controller::join_links(Director::baseURL(), $this->RelativeLink($action));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the absolute URL for this page, including protocol and host.
|
|
||||||
*
|
|
||||||
* @param string $action See {@link Link()}
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function AbsoluteLink($action = null) {
|
|
||||||
if($this->hasMethod('alternateAbsoluteLink')) {
|
|
||||||
return $this->alternateAbsoluteLink($action);
|
|
||||||
} else {
|
|
||||||
return Director::absoluteURL($this->Link($action));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the link for this {@link SiteTree} object relative to the SilverStripe root.
|
|
||||||
*
|
|
||||||
* By default, it this page is the current home page, and there is no action specified then this will return a link
|
|
||||||
* to the root of the site. However, if you set the $action parameter to TRUE then the link will not be rewritten
|
|
||||||
* and returned in its full form.
|
|
||||||
*
|
|
||||||
* @uses RootURLController::get_homepage_link()
|
|
||||||
*
|
|
||||||
* @param string $action See {@link Link()}
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function RelativeLink($action = null) {
|
|
||||||
if($this->ParentID && self::nested_urls()) {
|
|
||||||
$base = $this->Parent()->RelativeLink($this->URLSegment);
|
|
||||||
} else {
|
|
||||||
$base = $this->URLSegment;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Unset base for homepage URLSegments in their default language.
|
|
||||||
// Homepages with action parameters or in different languages
|
|
||||||
// need to retain their URLSegment. We can only do this if the homepage
|
|
||||||
// is on the root level.
|
|
||||||
if(!$action && $base == RootURLController::get_homepage_link() && !$this->ParentID) {
|
|
||||||
$base = null;
|
|
||||||
if($this->hasExtension('Translatable') && $this->Locale != Translatable::default_locale()){
|
|
||||||
$base = $this->URLSegment;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Legacy support
|
|
||||||
if($action === true) $action = null;
|
|
||||||
|
|
||||||
return Controller::join_links($base, '/', $action);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the absolute URL for this page on the Live site.
|
|
||||||
*/
|
|
||||||
public function getAbsoluteLiveLink($includeStageEqualsLive = true) {
|
|
||||||
$live = Versioned::get_one_by_stage('SiteTree', 'Live', '"SiteTree"."ID" = ' . $this->ID);
|
|
||||||
|
|
||||||
if($live) {
|
|
||||||
$link = $live->AbsoluteLink();
|
|
||||||
|
|
||||||
if($includeStageEqualsLive) {
|
|
||||||
$link .= '?stage=Live';
|
|
||||||
}
|
|
||||||
|
|
||||||
return $link;
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return a CSS identifier generated from this page's link.
|
|
||||||
*
|
|
||||||
* @return string The URL segment
|
|
||||||
*/
|
|
||||||
public function ElementName() {
|
|
||||||
return str_replace('/', '-', trim($this->RelativeLink(true), '/'));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns TRUE if this is the currently active page that is being used to handle a request.
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function isCurrent() {
|
|
||||||
return $this->ID ? $this->ID == Director::get_current_page()->ID : $this === Director::get_current_page();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if this page is in the currently active section (e.g. it is either current or one of it's children is
|
|
||||||
* currently being viewed.
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function isSection() {
|
|
||||||
return $this->isCurrent() || (
|
|
||||||
Director::get_current_page() instanceof SiteTree && in_array($this->ID, Director::get_current_page()->getAncestors()->column())
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return "link" or "current" depending on if this is the {@link SiteTree::isCurrent()} current page.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function LinkOrCurrent() {
|
|
||||||
return $this->isCurrent() ? 'current' : 'link';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return "link" or "section" depending on if this is the {@link SiteTree::isSeciton()} current section.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function LinkOrSection() {
|
|
||||||
return $this->isSection() ? 'section' : 'link';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return "link", "current" or section depending on if this page is the current page, or not on the current page but
|
|
||||||
* in the current section.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function LinkingMode() {
|
|
||||||
if($this->isCurrent()) {
|
|
||||||
return 'current';
|
|
||||||
} elseif($this->isSection()) {
|
|
||||||
return 'section';
|
|
||||||
} else {
|
|
||||||
return 'link';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if this page is in the given current section.
|
|
||||||
*
|
|
||||||
* @param string $sectionName Name of the section to check.
|
|
||||||
* @return boolean True if we are in the given section.
|
|
||||||
*/
|
|
||||||
public function InSection($sectionName) {
|
|
||||||
$page = Director::get_current_page();
|
|
||||||
while($page) {
|
|
||||||
if($sectionName == $page->URLSegment)
|
|
||||||
return true;
|
|
||||||
$page = $page->Parent;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a duplicate of this node. Doesn't affect joined data - create a
|
|
||||||
* custom overloading of this if you need such behaviour.
|
|
||||||
*
|
|
||||||
* @return SiteTree The duplicated object.
|
|
||||||
*/
|
|
||||||
public function duplicate($doWrite = true) {
|
|
||||||
|
|
||||||
$page = parent::duplicate(false);
|
|
||||||
$page->Sort = 0;
|
|
||||||
$this->extend('onBeforeDuplicate', $page);
|
|
||||||
|
|
||||||
if($doWrite) {
|
|
||||||
$page->write();
|
|
||||||
|
|
||||||
$page = $this->duplicateManyManyRelations($this, $page);
|
|
||||||
}
|
|
||||||
$this->extend('onAfterDuplicate', $page);
|
|
||||||
|
|
||||||
return $page;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Duplicates each child of this node recursively and returns the
|
|
||||||
* duplicate node.
|
|
||||||
*
|
|
||||||
* @return SiteTree The duplicated object.
|
|
||||||
*/
|
|
||||||
public function duplicateWithChildren() {
|
|
||||||
$clone = $this->duplicate();
|
|
||||||
$children = $this->AllChildren();
|
|
||||||
|
|
||||||
if($children) {
|
|
||||||
foreach($children as $child) {
|
|
||||||
$childClone = method_exists($child, 'duplicateWithChildren')
|
|
||||||
? $child->duplicateWithChildren()
|
|
||||||
: $child->duplicate();
|
|
||||||
$childClone->ParentID = $clone->ID;
|
|
||||||
$childClone->write();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $clone;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Duplicate this node and its children as a child of the node with the
|
|
||||||
* given ID
|
|
||||||
*
|
|
||||||
* @param int $id ID of the new node's new parent
|
|
||||||
*/
|
|
||||||
public function duplicateAsChild($id) {
|
|
||||||
$newSiteTree = $this->duplicate();
|
|
||||||
$newSiteTree->ParentID = $id;
|
|
||||||
$newSiteTree->Sort = 0;
|
|
||||||
$newSiteTree->write();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return a breadcrumb trail to this page. Excludes "hidden" pages
|
|
||||||
* (with ShowInMenus=0).
|
|
||||||
*
|
|
||||||
* @param int $maxDepth The maximum depth to traverse.
|
|
||||||
* @param boolean $unlinked Do not make page names links
|
|
||||||
* @param string $stopAtPageType ClassName of a page to stop the upwards traversal.
|
|
||||||
* @param boolean $showHidden Include pages marked with the attribute ShowInMenus = 0
|
|
||||||
* @return string The breadcrumb trail.
|
|
||||||
*/
|
|
||||||
public function Breadcrumbs($maxDepth = 20, $unlinked = false, $stopAtPageType = false, $showHidden = false) {
|
|
||||||
$page = $this;
|
|
||||||
$parts = array();
|
|
||||||
$i = 0;
|
|
||||||
while(
|
|
||||||
$page
|
|
||||||
&& (!$maxDepth || sizeof($parts) < $maxDepth)
|
|
||||||
&& (!$stopAtPageType || $page->ClassName != $stopAtPageType)
|
|
||||||
) {
|
|
||||||
if($showHidden || $page->ShowInMenus || ($page->ID == $this->ID)) {
|
|
||||||
if($page->URLSegment == 'home') $hasHome = true;
|
|
||||||
if(($page->ID == $this->ID) || $unlinked) {
|
|
||||||
$parts[] = Convert::raw2xml($page->Title);
|
|
||||||
} else {
|
|
||||||
$parts[] = ("<a href=\"" . $page->Link() . "\">" . Convert::raw2xml($page->Title) . "</a>");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$page = $page->Parent;
|
|
||||||
}
|
|
||||||
|
|
||||||
return implode(self::$breadcrumbs_delimiter, array_reverse($parts));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Make this page a child of another page.
|
|
||||||
*
|
|
||||||
* If the parent page does not exist, resolve it to a valid ID
|
|
||||||
* before updating this page's reference.
|
|
||||||
*
|
|
||||||
* @param SiteTree|int $item Either the parent object, or the parent ID
|
|
||||||
*/
|
|
||||||
public function setParent($item) {
|
|
||||||
if(is_object($item)) {
|
|
||||||
if (!$item->exists()) $item->write();
|
|
||||||
$this->setField("ParentID", $item->ID);
|
|
||||||
} else {
|
|
||||||
$this->setField("ParentID", $item);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the parent of this page.
|
|
||||||
*
|
|
||||||
* @return SiteTree Parent of this page.
|
|
||||||
*/
|
|
||||||
public function getParent() {
|
|
||||||
if ($this->getField("ParentID")) {
|
|
||||||
return DataObject::get_one("SiteTree", "\"SiteTree\".\"ID\" = " . $this->getField("ParentID"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return a string of the form "parent - page" or
|
|
||||||
* "grandparent - parent - page".
|
|
||||||
*
|
|
||||||
* @param int $level The maximum amount of levels to traverse.
|
|
||||||
* @param string $seperator Seperating string
|
|
||||||
* @return string The resulting string
|
|
||||||
*/
|
|
||||||
function NestedTitle($level = 2, $separator = " - ") {
|
|
||||||
$item = $this;
|
|
||||||
while($item && $level > 0) {
|
|
||||||
$parts[] = $item->Title;
|
|
||||||
$item = $item->Parent;
|
|
||||||
$level--;
|
|
||||||
}
|
|
||||||
return implode($separator, array_reverse($parts));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This function should return true if the current user can add children
|
|
||||||
* to this page. It can be overloaded to customise the security model for an
|
|
||||||
* application.
|
|
||||||
*
|
|
||||||
* Returns true if the member is allowed to do the given action.
|
|
||||||
*
|
|
||||||
* @uses DataObjectDecorator->can()
|
|
||||||
*
|
|
||||||
* If a page is set to inherit, but has no parent, it inherits from
|
|
||||||
* {@link SiteConfig}
|
|
||||||
*
|
|
||||||
* @param string $perm The permission to be checked, such as 'View'.
|
|
||||||
* @param Member $member The member whose permissions need checking.
|
|
||||||
* Defaults to the currently logged in user.
|
|
||||||
*
|
|
||||||
* @return boolean True if the the member is allowed to do the given
|
|
||||||
* action.
|
|
||||||
*
|
|
||||||
* @todo Check we get a endless recursion if we use parent::can()
|
|
||||||
*/
|
|
||||||
function can($perm, $member = null) {
|
|
||||||
if(!$member || !(is_a($member, 'Member')) || is_numeric($member)) {
|
|
||||||
$member = Member::currentUserID();
|
|
||||||
}
|
|
||||||
|
|
||||||
if($member && Permission::checkMember($member, "ADMIN")) return true;
|
|
||||||
|
|
||||||
if(method_exists($this, 'can' . ucfirst($perm))) {
|
|
||||||
$method = 'can' . ucfirst($perm);
|
|
||||||
return $this->$method($member);
|
|
||||||
}
|
|
||||||
|
|
||||||
$results = $this->extend('can', $member);
|
|
||||||
if($results && is_array($results)) if(!min($results)) return false;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This function should return true if the current user can add children
|
|
||||||
* to this page. It can be overloaded to customise the security model for an
|
|
||||||
* application.
|
|
||||||
*
|
|
||||||
* Denies permission if any of the following conditions is TRUE:
|
|
||||||
* - alternateCanAddChildren() on a decorator returns FALSE
|
|
||||||
* - canEdit() is not granted
|
|
||||||
* - There are no classes defined in {@link $allowed_children}
|
|
||||||
*
|
|
||||||
* @uses SiteTreeDecorator->canAddChildren()
|
|
||||||
* @uses canEdit()
|
|
||||||
* @uses $allowed_children
|
|
||||||
*
|
|
||||||
* @return boolean True if the current user can add children.
|
|
||||||
*/
|
|
||||||
public function canAddChildren($member = null) {
|
|
||||||
if(!$member || !(is_a($member, 'Member')) || is_numeric($member)) {
|
|
||||||
$member = Member::currentUserID();
|
|
||||||
}
|
|
||||||
|
|
||||||
if($member && Permission::checkMember($member, "ADMIN")) return true;
|
|
||||||
|
|
||||||
// Standard mechanism for accepting permission changes from decorators
|
|
||||||
$extended = $this->extendedCan('canAddChildren', $member);
|
|
||||||
if($extended !== null) return $extended;
|
|
||||||
|
|
||||||
return $this->canEdit($member) && $this->stat('allowed_children') != 'none';
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This function should return true if the current user can view this
|
|
||||||
* page. It can be overloaded to customise the security model for an
|
|
||||||
* application.
|
|
||||||
*
|
|
||||||
* Denies permission if any of the following conditions is TRUE:
|
|
||||||
* - canView() on any decorator returns FALSE
|
|
||||||
* - "CanViewType" directive is set to "Inherit" and any parent page return false for canView()
|
|
||||||
* - "CanViewType" directive is set to "LoggedInUsers" and no user is logged in
|
|
||||||
* - "CanViewType" directive is set to "OnlyTheseUsers" and user is not in the given groups
|
|
||||||
*
|
|
||||||
* @uses DataObjectDecorator->canView()
|
|
||||||
* @uses ViewerGroups()
|
|
||||||
*
|
|
||||||
* @return boolean True if the current user can view this page.
|
|
||||||
*/
|
|
||||||
public function canView($member = null) {
|
|
||||||
if(!$member || !(is_a($member, 'Member')) || is_numeric($member)) {
|
|
||||||
$member = Member::currentUserID();
|
|
||||||
}
|
|
||||||
|
|
||||||
// admin override
|
|
||||||
if($member && Permission::checkMember($member, array("ADMIN", "SITETREE_VIEW_ALL"))) return true;
|
|
||||||
|
|
||||||
// Standard mechanism for accepting permission changes from decorators
|
|
||||||
$extended = $this->extendedCan('canView', $member);
|
|
||||||
if($extended !== null) return $extended;
|
|
||||||
|
|
||||||
// check for empty spec
|
|
||||||
if(!$this->CanViewType || $this->CanViewType == 'Anyone') return true;
|
|
||||||
|
|
||||||
// check for inherit
|
|
||||||
if($this->CanViewType == 'Inherit') {
|
|
||||||
if($this->ParentID) return $this->Parent()->canView($member);
|
|
||||||
else return $this->getSiteConfig()->canView($member);
|
|
||||||
}
|
|
||||||
|
|
||||||
// check for any logged-in users
|
|
||||||
if($this->CanViewType == 'LoggedInUsers' && $member) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// check for specific groups
|
|
||||||
if($member && is_numeric($member)) $member = DataObject::get_by_id('Member', $member);
|
|
||||||
if(
|
|
||||||
$this->CanViewType == 'OnlyTheseUsers'
|
|
||||||
&& $member
|
|
||||||
&& $member->inGroups($this->ViewerGroups())
|
|
||||||
) return true;
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determines permissions for a specific stage (see {@link Versioned}).
|
|
||||||
* Usually the stage is read from {@link Versioned::current_stage()}.
|
|
||||||
* Falls back to {@link canView}.
|
|
||||||
*
|
|
||||||
* @todo Implement in CMS UI.
|
|
||||||
*
|
|
||||||
* @param String $stage
|
|
||||||
* @param Member $member
|
|
||||||
* @return boolean
|
|
||||||
*/
|
|
||||||
function canViewStage($stage, $member = null) {
|
|
||||||
if(!$member) $member = Member::currentUser();
|
|
||||||
|
|
||||||
if(
|
|
||||||
strtolower($stage) == 'stage' &&
|
|
||||||
!(Permission::checkMember($member, 'CMS_ACCESS_CMSMain') || Permission::checkMember($member, 'VIEW_DRAFT_CONTENT'))
|
|
||||||
) return false;
|
|
||||||
|
|
||||||
return $this->canView($member);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This function should return true if the current user can delete this
|
|
||||||
* page. It can be overloaded to customise the security model for an
|
|
||||||
* application.
|
|
||||||
*
|
|
||||||
* Denies permission if any of the following conditions is TRUE:
|
|
||||||
* - canDelete() returns FALSE on any decorator
|
|
||||||
* - canEdit() returns FALSE
|
|
||||||
* - any descendant page returns FALSE for canDelete()
|
|
||||||
*
|
|
||||||
* @uses canDelete()
|
|
||||||
* @uses DataObjectDecorator->canDelete()
|
|
||||||
* @uses canEdit()
|
|
||||||
*
|
|
||||||
* @param Member $member
|
|
||||||
* @return boolean True if the current user can delete this page.
|
|
||||||
*/
|
|
||||||
public function canDelete($member = null) {
|
|
||||||
if($member instanceof Member) $memberID = $member->ID;
|
|
||||||
else if(is_numeric($member)) $memberID = $member;
|
|
||||||
else $memberID = Member::currentUserID();
|
|
||||||
|
|
||||||
if($memberID && Permission::checkMember($memberID, array("ADMIN", "SITETREE_EDIT_ALL"))) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Standard mechanism for accepting permission changes from decorators
|
|
||||||
$extended = $this->extendedCan('canDelete', $memberID);
|
|
||||||
if($extended !== null) return $extended;
|
|
||||||
|
|
||||||
// Check cache (the can_edit_multiple call below will also do this, but this is quicker)
|
|
||||||
if(isset(self::$cache_permissions['delete'][$this->ID])) {
|
|
||||||
return self::$cache_permissions['delete'][$this->ID];
|
|
||||||
}
|
|
||||||
|
|
||||||
// Regular canEdit logic is handled by can_edit_multiple
|
|
||||||
$results = self::can_delete_multiple(array($this->ID), $memberID);
|
|
||||||
|
|
||||||
// If this page no longer exists in stage/live results won't contain the page.
|
|
||||||
// Fail-over to false
|
|
||||||
return isset($results[$this->ID]) ? $results[$this->ID] : false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This function should return true if the current user can create new
|
|
||||||
* pages of this class. It can be overloaded to customise the security model for an
|
|
||||||
* application.
|
|
||||||
*
|
|
||||||
* Denies permission if any of the following conditions is TRUE:
|
|
||||||
* - canCreate() returns FALSE on any decorator
|
|
||||||
* - $can_create is set to FALSE and the site is not in "dev mode"
|
|
||||||
*
|
|
||||||
* Use {@link canAddChildren()} to control behaviour of creating children under this page.
|
|
||||||
*
|
|
||||||
* @uses $can_create
|
|
||||||
* @uses DataObjectDecorator->canCreate()
|
|
||||||
*
|
|
||||||
* @param Member $member
|
|
||||||
* @return boolean True if the current user can create pages on this class.
|
|
||||||
*/
|
|
||||||
public function canCreate($member = null) {
|
|
||||||
if(!$member || !(is_a($member, 'Member')) || is_numeric($member)) {
|
|
||||||
$member = Member::currentUserID();
|
|
||||||
}
|
|
||||||
|
|
||||||
if($member && Permission::checkMember($member, "ADMIN")) return true;
|
|
||||||
|
|
||||||
// Standard mechanism for accepting permission changes from decorators
|
|
||||||
$extended = $this->extendedCan('canCreate', $member);
|
|
||||||
if($extended !== null) return $extended;
|
|
||||||
|
|
||||||
return $this->stat('can_create') != false || Director::isDev();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This function should return true if the current user can edit this
|
|
||||||
* page. It can be overloaded to customise the security model for an
|
|
||||||
* application.
|
|
||||||
*
|
|
||||||
* Denies permission if any of the following conditions is TRUE:
|
|
||||||
* - canEdit() on any decorator returns FALSE
|
|
||||||
* - canView() return false
|
|
||||||
* - "CanEditType" directive is set to "Inherit" and any parent page return false for canEdit()
|
|
||||||
* - "CanEditType" directive is set to "LoggedInUsers" and no user is logged in or doesn't have the CMS_Access_CMSMAIN permission code
|
|
||||||
* - "CanEditType" directive is set to "OnlyTheseUsers" and user is not in the given groups
|
|
||||||
*
|
|
||||||
* @uses canView()
|
|
||||||
* @uses EditorGroups()
|
|
||||||
* @uses DataObjectDecorator->canEdit()
|
|
||||||
*
|
|
||||||
* @param Member $member Set to FALSE if you want to explicitly test permissions without a valid user (useful for unit tests)
|
|
||||||
* @return boolean True if the current user can edit this page.
|
|
||||||
*/
|
|
||||||
public function canEdit($member = null) {
|
|
||||||
if($member instanceof Member) $memberID = $member->ID;
|
|
||||||
else if(is_numeric($member)) $memberID = $member;
|
|
||||||
else $memberID = Member::currentUserID();
|
|
||||||
|
|
||||||
if($memberID && Permission::checkMember($memberID, array("ADMIN", "SITETREE_EDIT_ALL"))) return true;
|
|
||||||
|
|
||||||
// Standard mechanism for accepting permission changes from decorators
|
|
||||||
$extended = $this->extendedCan('canEdit', $memberID);
|
|
||||||
if($extended !== null) return $extended;
|
|
||||||
|
|
||||||
if($this->ID) {
|
|
||||||
// Check cache (the can_edit_multiple call below will also do this, but this is quicker)
|
|
||||||
if(isset(self::$cache_permissions['CanEditType'][$this->ID])) {
|
|
||||||
return self::$cache_permissions['CanEditType'][$this->ID];
|
|
||||||
}
|
|
||||||
|
|
||||||
// Regular canEdit logic is handled by can_edit_multiple
|
|
||||||
$results = self::can_edit_multiple(array($this->ID), $memberID);
|
|
||||||
|
|
||||||
// If this page no longer exists in stage/live results won't contain the page.
|
|
||||||
// Fail-over to false
|
|
||||||
return isset($results[$this->ID]) ? $results[$this->ID] : false;
|
|
||||||
|
|
||||||
// Default for unsaved pages
|
|
||||||
} else {
|
|
||||||
return $this->getSiteConfig()->canEdit($member);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This function should return true if the current user can publish this
|
|
||||||
* page. It can be overloaded to customise the security model for an
|
|
||||||
* application.
|
|
||||||
*
|
|
||||||
* Denies permission if any of the following conditions is TRUE:
|
|
||||||
* - canPublish() on any decorator returns FALSE
|
|
||||||
* - canEdit() returns FALSE
|
|
||||||
*
|
|
||||||
* @uses SiteTreeDecorator->canPublish()
|
|
||||||
*
|
|
||||||
* @param Member $member
|
|
||||||
* @return boolean True if the current user can publish this page.
|
|
||||||
*/
|
|
||||||
public function canPublish($member = null) {
|
|
||||||
if(!$member || !(is_a($member, 'Member')) || is_numeric($member)) $member = Member::currentUser();
|
|
||||||
|
|
||||||
if($member && Permission::checkMember($member, "ADMIN")) return true;
|
|
||||||
|
|
||||||
// Standard mechanism for accepting permission changes from decorators
|
|
||||||
$extended = $this->extendedCan('canPublish', $member);
|
|
||||||
if($extended !== null) return $extended;
|
|
||||||
|
|
||||||
// Normal case - fail over to canEdit()
|
|
||||||
return $this->canEdit($member);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function canDeleteFromLive($member = null) {
|
|
||||||
// Standard mechanism for accepting permission changes from decorators
|
|
||||||
$extended = $this->extendedCan('canDeleteFromLive', $member);
|
|
||||||
if($extended !==null) return $extended;
|
|
||||||
|
|
||||||
return $this->canPublish($member);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Stub method to get the site config, provided so it's easy to override
|
|
||||||
*/
|
|
||||||
function getSiteConfig() {
|
|
||||||
$altConfig = false;
|
|
||||||
if($this->hasMethod('alternateSiteConfig')) {
|
|
||||||
$altConfig = $this->alternateSiteConfig();
|
|
||||||
}
|
|
||||||
if($altConfig) {
|
|
||||||
return $altConfig;
|
|
||||||
} elseif($this->hasExtension('Translatable')) {
|
|
||||||
return SiteConfig::current_site_config($this->Locale);
|
|
||||||
} else {
|
|
||||||
return SiteConfig::current_site_config();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Pre-populate the cache of canEdit, canView, canDelete, canPublish permissions.
|
|
||||||
* This method will use the static can_(perm)_multiple method for efficiency.
|
|
||||||
*
|
|
||||||
* @param $permission String The permission: edit, view, publish, approve, etc.
|
|
||||||
* @param $ids array An array of page IDs
|
|
||||||
* @param $batchCallBack The function/static method to call to calculate permissions. Defaults
|
|
||||||
* to 'SiteTree::can_(permission)_multiple'
|
|
||||||
*/
|
|
||||||
static function prepopuplate_permission_cache($permission = 'CanEditType', $ids, $batchCallback = null) {
|
|
||||||
if(!$batchCallback) $batchCallback = "SiteTree::can_{$permission}_multiple";
|
|
||||||
|
|
||||||
//PHP 5.1 requires an array rather than a string for the call_user_func function
|
|
||||||
$batchCallback=explode('::', $batchCallback);
|
|
||||||
|
|
||||||
if(is_callable($batchCallback)) {
|
|
||||||
$permissionValues = call_user_func($batchCallback, $ids,
|
|
||||||
Member::currentUserID(), false);
|
|
||||||
|
|
||||||
if(!isset(self::$cache_permissions[$permission])) {
|
|
||||||
self::$cache_permissions[$permission] = array();
|
|
||||||
}
|
|
||||||
|
|
||||||
self::$cache_permissions[$permission] = $permissionValues
|
|
||||||
+ self::$cache_permissions[$permission];
|
|
||||||
|
|
||||||
} else {
|
|
||||||
user_error("SiteTree::prepopuplate_permission_cache can't calculate '$permission' "
|
|
||||||
. "with callback '$batchCallback'", E_USER_WARNING);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static function batch_permission_check($ids, $memberID, $typeField, $groupJoinTable, $siteConfigMethod, $globalPermission = 'CMS_ACCESS_CMSMain', $useCached = true) {
|
|
||||||
// Sanitise the IDs
|
|
||||||
$ids = array_filter($ids, 'is_numeric');
|
|
||||||
|
|
||||||
// This is the name used on the permission cache
|
|
||||||
// converts something like 'CanEditType' to 'edit'.
|
|
||||||
$cacheKey = strtolower(substr($typeField, 3, -4));
|
|
||||||
|
|
||||||
// Default result: nothing editable
|
|
||||||
$result = array_fill_keys($ids, false);
|
|
||||||
if($ids) {
|
|
||||||
|
|
||||||
// Look in the cache for values
|
|
||||||
if($useCached && isset(self::$cache_permissions[$cacheKey])) {
|
|
||||||
$cachedValues = array_intersect_key(self::$cache_permissions[$cacheKey], $result);
|
|
||||||
|
|
||||||
// If we can't find everything in the cache, then look up the remainder separately
|
|
||||||
$uncachedValues = array_diff_key($result, self::$cache_permissions[$cacheKey]);
|
|
||||||
if($uncachedValues) {
|
|
||||||
$cachedValues = self::batch_permission_check(array_keys($uncachedValues), $memberID, $typeField, $groupJoinTable, $siteConfigMethod, $globalPermission, false) + $cachedValues;
|
|
||||||
}
|
|
||||||
return $cachedValues;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If a member doesn't have CMS_ACCESS_CMSMain permission then they can't edit anything
|
|
||||||
if(!$memberID || ($globalPermission && !Permission::checkMember($memberID, $globalPermission))) {
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
$SQL_idList = implode($ids, ", ");
|
|
||||||
|
|
||||||
// if page can't be viewed, don't grant edit permissions
|
|
||||||
// to do - implement can_view_multiple(), so this can be enabled
|
|
||||||
//$ids = array_keys(array_filter(self::can_view_multiple($ids, $memberID)));
|
|
||||||
|
|
||||||
// Get the groups that the given member belongs to
|
|
||||||
$groupIDs = DataObject::get_by_id('Member', $memberID)->Groups()->column("ID");
|
|
||||||
$SQL_groupList = implode(", ", $groupIDs);
|
|
||||||
if (!$SQL_groupList) $SQL_groupList = '0';
|
|
||||||
|
|
||||||
$combinedStageResult = array();
|
|
||||||
|
|
||||||
foreach(array('Stage', 'Live') as $stage) {
|
|
||||||
// Start by filling the array with the pages that actually exist
|
|
||||||
$table = ($stage=='Stage') ? "SiteTree" : "SiteTree_$stage";
|
|
||||||
|
|
||||||
$result = array_fill_keys(DB::query("SELECT \"ID\" FROM \"$table\"
|
|
||||||
WHERE \"ID\" IN (".implode(", ", $ids).")")->column(), false);
|
|
||||||
|
|
||||||
// Get the uninherited permissions
|
|
||||||
$uninheritedPermissions = Versioned::get_by_stage("SiteTree", $stage, "(\"$typeField\" = 'LoggedInUsers' OR
|
|
||||||
(\"$typeField\" = 'OnlyTheseUsers' AND \"$groupJoinTable\".\"SiteTreeID\" IS NOT NULL))
|
|
||||||
AND \"SiteTree\".\"ID\" IN ($SQL_idList)",
|
|
||||||
"",
|
|
||||||
"LEFT JOIN \"$groupJoinTable\"
|
|
||||||
ON \"$groupJoinTable\".\"SiteTreeID\" = \"SiteTree\".\"ID\"
|
|
||||||
AND \"$groupJoinTable\".\"GroupID\" IN ($SQL_groupList)");
|
|
||||||
|
|
||||||
if($uninheritedPermissions) {
|
|
||||||
// Set all the relevant items in $result to true
|
|
||||||
$result = array_fill_keys($uninheritedPermissions->column('ID'), true) + $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get permissions that are inherited
|
|
||||||
$potentiallyInherited = Versioned::get_by_stage("SiteTree", $stage, "\"$typeField\" = 'Inherit'
|
|
||||||
AND \"SiteTree\".\"ID\" IN ($SQL_idList)");
|
|
||||||
|
|
||||||
if($potentiallyInherited) {
|
|
||||||
// Group $potentiallyInherited by ParentID; we'll look at the permission of all those
|
|
||||||
// parents and then see which ones the user has permission on
|
|
||||||
$siteConfigPermission = SiteConfig::current_site_config()->{$siteConfigMethod}($memberID);
|
|
||||||
$groupedByParent = array();
|
|
||||||
foreach($potentiallyInherited as $item) {
|
|
||||||
if($item->ParentID) {
|
|
||||||
if(!isset($groupedByParent[$item->ParentID])) $groupedByParent[$item->ParentID] = array();
|
|
||||||
$groupedByParent[$item->ParentID][] = $item->ID;
|
|
||||||
} else {
|
|
||||||
$result[$item->ID] = $siteConfigPermission;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if($groupedByParent) {
|
|
||||||
$actuallyInherited = self::batch_permission_check(array_keys($groupedByParent), $memberID, $typeField, $groupJoinTable, $siteConfigMethod);
|
|
||||||
if($actuallyInherited) {
|
|
||||||
$parentIDs = array_keys(array_filter($actuallyInherited));
|
|
||||||
foreach($parentIDs as $parentID) {
|
|
||||||
// Set all the relevant items in $result to true
|
|
||||||
$result = array_fill_keys($groupedByParent[$parentID], true) + $result;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$combinedStageResult = $combinedStageResult + $result;
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if(isset($combinedStageResult)) {
|
|
||||||
// Cache results
|
|
||||||
// TODO - Caching permissions is breaking unit tests. One possible issue
|
|
||||||
// is the cache needs to be flushed when permission on a page is changed,
|
|
||||||
// but this only solved some of the failing unit tests. Disabled for now.
|
|
||||||
/*foreach($combinedStageResult as $id => $val) {
|
|
||||||
self::$cache_permissions[$typeField][$id] = $val;
|
|
||||||
}*/
|
|
||||||
return $combinedStageResult;
|
|
||||||
} else {
|
|
||||||
return array();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the 'can edit' information for a number of SiteTree pages.
|
|
||||||
*
|
|
||||||
* @param An array of IDs of the SiteTree pages to look up.
|
|
||||||
* @param useCached Return values from the permission cache if they exist.
|
|
||||||
* @return A map where the IDs are keys and the values are booleans stating whether the given
|
|
||||||
* page can be edited.
|
|
||||||
*/
|
|
||||||
static function can_edit_multiple($ids, $memberID, $useCached = true) {
|
|
||||||
return self::batch_permission_check($ids, $memberID, 'CanEditType', 'SiteTree_EditorGroups', 'canEdit', 'CMS_ACCESS_CMSMain', $useCached);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the 'can edit' information for a number of SiteTree pages.
|
|
||||||
* @param An array of IDs of the SiteTree pages to look up.
|
|
||||||
* @param useCached Return values from the permission cache if they exist.
|
|
||||||
*/
|
|
||||||
static function can_delete_multiple($ids, $memberID, $useCached = true) {
|
|
||||||
$deletable = array();
|
|
||||||
|
|
||||||
$result = array_fill_keys($ids, false);
|
|
||||||
|
|
||||||
// Look in the cache for values
|
|
||||||
if($useCached && isset(self::$cache_permissions['delete'])) {
|
|
||||||
$cachedValues = array_intersect_key(self::$cache_permissions['delete'], $result);
|
|
||||||
|
|
||||||
// If we can't find everything in the cache, then look up the remainder separately
|
|
||||||
$uncachedValues = array_diff_key($result, self::$cache_permissions['delete']);
|
|
||||||
if($uncachedValues) {
|
|
||||||
$cachedValues = self::can_delete_multiple(array_keys($uncachedValues), $memberID, false)
|
|
||||||
+ $cachedValues;
|
|
||||||
}
|
|
||||||
return $cachedValues;
|
|
||||||
}
|
|
||||||
|
|
||||||
// You can only delete pages that you can edit
|
|
||||||
$editableIDs = array_keys(array_filter(self::can_edit_multiple($ids, $memberID)));
|
|
||||||
if($editableIDs) {
|
|
||||||
$idList = implode(",", $editableIDs);
|
|
||||||
|
|
||||||
// You can only delete pages whose children you can delete
|
|
||||||
$childRecords = DataObject::get("SiteTree", "\"ParentID\" IN ($idList)");
|
|
||||||
if($childRecords) {
|
|
||||||
$children = $childRecords->map("ID", "ParentID");
|
|
||||||
|
|
||||||
// Find out the children that can be deleted
|
|
||||||
$deletableChildren = self::can_delete_multiple(array_keys($children), $memberID);
|
|
||||||
|
|
||||||
// Get a list of all the parents that have no undeletable children
|
|
||||||
$deletableParents = array_fill_keys($editableIDs, true);
|
|
||||||
foreach($deletableChildren as $id => $canDelete) {
|
|
||||||
if(!$canDelete) unset($deletableParents[$children[$id]]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Use that to filter the list of deletable parents that have children
|
|
||||||
$deletableParents = array_keys($deletableParents);
|
|
||||||
|
|
||||||
// Also get the $ids that don't have children
|
|
||||||
$parents = array_unique($children);
|
|
||||||
$deletableLeafNodes = array_diff($editableIDs, $parents);
|
|
||||||
|
|
||||||
// Combine the two
|
|
||||||
$deletable = array_merge($deletableParents, $deletableLeafNodes);
|
|
||||||
|
|
||||||
} else {
|
|
||||||
$deletable = $editableIDs;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
$deletable = array();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Convert the array of deletable IDs into a map of the original IDs with true/false as the
|
|
||||||
// value
|
|
||||||
return array_fill_keys($deletable, true) + array_fill_keys($ids, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Collate selected descendants of this page.
|
|
||||||
*
|
|
||||||
* {@link $condition} will be evaluated on each descendant, and if it is
|
|
||||||
* succeeds, that item will be added to the $collator array.
|
|
||||||
*
|
|
||||||
* @param string $condition The PHP condition to be evaluated. The page
|
|
||||||
* will be called $item
|
|
||||||
* @param array $collator An array, passed by reference, to collect all
|
|
||||||
* of the matching descendants.
|
|
||||||
*/
|
|
||||||
public function collateDescendants($condition, &$collator) {
|
|
||||||
if($children = $this->Children()) {
|
|
||||||
foreach($children as $item) {
|
|
||||||
if(eval("return $condition;")) $collator[] = $item;
|
|
||||||
$item->collateDescendants($condition, $collator);
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the title, description, keywords and language metatags.
|
|
||||||
*
|
|
||||||
* @todo Move <title> tag in separate getter for easier customization and more obvious usage
|
|
||||||
*
|
|
||||||
* @param boolean|string $includeTitle Show default <title>-tag, set to false for custom templating
|
|
||||||
* @param boolean $includeTitle Show default <title>-tag, set to false for
|
|
||||||
* custom templating
|
|
||||||
* @return string The XHTML metatags
|
|
||||||
*/
|
|
||||||
public function MetaTags($includeTitle = true) {
|
|
||||||
$tags = "";
|
|
||||||
if($includeTitle === true || $includeTitle == 'true') {
|
|
||||||
$tags .= "<title>" . Convert::raw2xml(($this->MetaTitle)
|
|
||||||
? $this->MetaTitle
|
|
||||||
: $this->Title) . "</title>\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
$tags .= "<meta name=\"generator\" content=\"SilverStripe - http://silverstripe.org\" />\n";
|
|
||||||
|
|
||||||
$charset = ContentNegotiator::get_encoding();
|
|
||||||
$tags .= "<meta http-equiv=\"Content-type\" content=\"text/html; charset=$charset\" />\n";
|
|
||||||
if($this->MetaKeywords) {
|
|
||||||
$tags .= "<meta name=\"keywords\" content=\"" . Convert::raw2att($this->MetaKeywords) . "\" />\n";
|
|
||||||
}
|
|
||||||
if($this->MetaDescription) {
|
|
||||||
$tags .= "<meta name=\"description\" content=\"" . Convert::raw2att($this->MetaDescription) . "\" />\n";
|
|
||||||
}
|
|
||||||
if($this->ExtraMeta) {
|
|
||||||
$tags .= $this->ExtraMeta . "\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->extend('MetaTags', $tags);
|
|
||||||
|
|
||||||
return $tags;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the object that contains the content that a user would
|
|
||||||
* associate with this page.
|
|
||||||
*
|
|
||||||
* Ordinarily, this is just the page itself, but for example on
|
|
||||||
* RedirectorPages or VirtualPages ContentSource() will return the page
|
|
||||||
* that is linked to.
|
|
||||||
*
|
|
||||||
* @return SiteTree The content source.
|
|
||||||
*/
|
|
||||||
public function ContentSource() {
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add default records to database.
|
|
||||||
*
|
|
||||||
* This function is called whenever the database is built, after the
|
|
||||||
* database tables have all been created. Overload this to add default
|
|
||||||
* records when the database is built, but make sure you call
|
|
||||||
* parent::requireDefaultRecords().
|
|
||||||
*/
|
|
||||||
function requireDefaultRecords() {
|
|
||||||
parent::requireDefaultRecords();
|
|
||||||
|
|
||||||
// default pages
|
|
||||||
if($this->class == 'SiteTree' && self::$create_default_pages) {
|
|
||||||
if(!SiteTree::get_by_link('home')) {
|
|
||||||
$homepage = new Page();
|
|
||||||
$homepage->Title = _t('SiteTree.DEFAULTHOMETITLE', 'Home');
|
|
||||||
$homepage->Content = _t('SiteTree.DEFAULTHOMECONTENT', '<p>Welcome to SilverStripe! This is the default homepage. You can edit this page by opening <a href="admin/">the CMS</a>. You can now access the <a href="http://doc.silverstripe.org">developer documentation</a>, or begin <a href="http://doc.silverstripe.org/doku.php?id=tutorials">the tutorials.</a></p>');
|
|
||||||
$homepage->URLSegment = 'home';
|
|
||||||
$homepage->Sort = 1;
|
|
||||||
$homepage->write();
|
|
||||||
$homepage->publish('Stage', 'Live');
|
|
||||||
$homepage->flushCache();
|
|
||||||
DB::alteration_message('Home page created', 'created');
|
|
||||||
}
|
|
||||||
|
|
||||||
if(DB::query("SELECT COUNT(*) FROM \"SiteTree\"")->value() == 1) {
|
|
||||||
$aboutus = new Page();
|
|
||||||
$aboutus->Title = _t('SiteTree.DEFAULTABOUTTITLE', 'About Us');
|
|
||||||
$aboutus->Content = _t('SiteTree.DEFAULTABOUTCONTENT', '<p>You can fill this page out with your own content, or delete it and create your own pages.<br /></p>');
|
|
||||||
$aboutus->Sort = 2;
|
|
||||||
$aboutus->write();
|
|
||||||
$aboutus->publish('Stage', 'Live');
|
|
||||||
$aboutus->flushCache();
|
|
||||||
DB::alteration_message('About Us page created', 'created');
|
|
||||||
|
|
||||||
$contactus = new Page();
|
|
||||||
$contactus->Title = _t('SiteTree.DEFAULTCONTACTTITLE', 'Contact Us');
|
|
||||||
$contactus->Content = _t('SiteTree.DEFAULTCONTACTCONTENT', '<p>You can fill this page out with your own content, or delete it and create your own pages.<br /></p>');
|
|
||||||
$contactus->Sort = 3;
|
|
||||||
$contactus->write();
|
|
||||||
$contactus->publish('Stage', 'Live');
|
|
||||||
$contactus->flushCache();
|
|
||||||
DB::alteration_message('Contact Us page created', 'created');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// schema migration
|
|
||||||
// @todo Move to migration task once infrastructure is implemented
|
|
||||||
if($this->class == 'SiteTree') {
|
|
||||||
$conn = DB::getConn();
|
|
||||||
// only execute command if fields haven't been renamed to _obsolete_<fieldname> already by the task
|
|
||||||
if(array_key_exists('Viewers', $conn->fieldList('SiteTree'))) {
|
|
||||||
$task = new UpgradeSiteTreePermissionSchemaTask();
|
|
||||||
$task->run(new SS_HTTPRequest('GET','/'));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
//------------------------------------------------------------------------------------//
|
|
||||||
|
|
||||||
protected function onBeforeWrite() {
|
|
||||||
parent::onBeforeWrite();
|
|
||||||
|
|
||||||
// If Sort hasn't been set, make this page come after it's siblings
|
|
||||||
if(!$this->Sort) {
|
|
||||||
$parentID = ($this->ParentID) ? $this->ParentID : 0;
|
|
||||||
$this->Sort = DB::query("SELECT MAX(\"Sort\") + 1 FROM \"SiteTree\" WHERE \"ParentID\" = $parentID")->value();
|
|
||||||
}
|
|
||||||
|
|
||||||
// If there is no URLSegment set, generate one from Title
|
|
||||||
if((!$this->URLSegment || $this->URLSegment == 'new-page') && $this->Title) {
|
|
||||||
$this->URLSegment = $this->generateURLSegment($this->Title);
|
|
||||||
} else if($this->isChanged('URLSegment')) {
|
|
||||||
// Make sure the URLSegment is valid for use in a URL
|
|
||||||
$segment = ereg_replace('[^A-Za-z0-9]+','-',$this->URLSegment);
|
|
||||||
$segment = ereg_replace('-+','-',$segment);
|
|
||||||
|
|
||||||
// If after sanitising there is no URLSegment, give it a reasonable default
|
|
||||||
if(!$segment) {
|
|
||||||
$segment = "page-$this->ID";
|
|
||||||
}
|
|
||||||
$this->URLSegment = $segment;
|
|
||||||
}
|
|
||||||
|
|
||||||
DataObject::set_context_obj($this);
|
|
||||||
|
|
||||||
// Ensure that this object has a non-conflicting URLSegment value.
|
|
||||||
$count = 2;
|
|
||||||
while(!$this->validURLSegment()) {
|
|
||||||
$this->URLSegment = preg_replace('/-[0-9]+$/', null, $this->URLSegment) . '-' . $count;
|
|
||||||
$count++;
|
|
||||||
}
|
|
||||||
|
|
||||||
DataObject::set_context_obj(null);
|
|
||||||
|
|
||||||
$this->syncLinkTracking();
|
|
||||||
|
|
||||||
// Check to see if we've only altered fields that shouldn't affect versioning
|
|
||||||
$fieldsIgnoredByVersioning = array('HasBrokenLink', 'Status', 'HasBrokenFile', 'ToDo');
|
|
||||||
$changedFields = array_keys($this->getChangedFields(true, 2));
|
|
||||||
|
|
||||||
// This more rigorous check is inline with the test that write()
|
|
||||||
// does to dedcide whether or not to write to the DB. We use that
|
|
||||||
// to avoid cluttering the system with a migrateVersion() call
|
|
||||||
// that doesn't get used
|
|
||||||
$oneChangedFields = array_keys($this->getChangedFields(true, 1));
|
|
||||||
|
|
||||||
if($oneChangedFields && !array_diff($changedFields, $fieldsIgnoredByVersioning)) {
|
|
||||||
// This will have the affect of preserving the versioning
|
|
||||||
$this->migrateVersion($this->Version);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function syncLinkTracking() {
|
|
||||||
// Build a list of HTMLText fields
|
|
||||||
$allFields = $this->db();
|
|
||||||
$htmlFields = array();
|
|
||||||
foreach($allFields as $field => $fieldSpec) {
|
|
||||||
if(preg_match('/([^(]+)/', $fieldSpec, $matches)) {
|
|
||||||
$class = $matches[0];
|
|
||||||
if(class_exists($class)){
|
|
||||||
if($class == 'HTMLText' || is_subclass_of($class, 'HTMLText')) $htmlFields[] = $field;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$linkedPages = array();
|
|
||||||
$linkedFiles = array();
|
|
||||||
$this->HasBrokenLink = false;
|
|
||||||
$this->HasBrokenFile = false;
|
|
||||||
|
|
||||||
foreach($htmlFields as $field) {
|
|
||||||
$formField = new HTMLEditorField($field);
|
|
||||||
$formField->setValue($this->$field);
|
|
||||||
$formField->saveInto($this);
|
|
||||||
}
|
|
||||||
$this->extend('augmentSyncLinkTracking');
|
|
||||||
}
|
|
||||||
|
|
||||||
function onAfterWrite() {
|
|
||||||
// Need to flush cache to avoid outdated versionnumber references
|
|
||||||
$this->flushCache();
|
|
||||||
|
|
||||||
// Update any virtual pages that might need updating
|
|
||||||
$linkedPages = $this->VirtualPages();
|
|
||||||
if($linkedPages) foreach($linkedPages as $page) {
|
|
||||||
$page->copyFrom($page->CopyContentFrom());
|
|
||||||
$page->write();
|
|
||||||
}
|
|
||||||
|
|
||||||
parent::onAfterWrite();
|
|
||||||
}
|
|
||||||
|
|
||||||
function onBeforeDelete() {
|
|
||||||
parent::onBeforeDelete();
|
|
||||||
|
|
||||||
// If deleting this page, delete all its children.
|
|
||||||
if(SiteTree::get_enforce_strict_hierarchy() && $children = $this->Children()) {
|
|
||||||
foreach($children as $child) {
|
|
||||||
$child->delete();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
function onAfterDelete() {
|
|
||||||
// Need to flush cache to avoid outdated versionnumber references
|
|
||||||
$this->flushCache();
|
|
||||||
|
|
||||||
// Need to mark pages depending to this one as broken
|
|
||||||
$dependentPages = $this->DependentPages();
|
|
||||||
if($dependentPages) foreach($dependentPages as $page) {
|
|
||||||
// $page->write() calls syncLinkTracking, which does all the hard work for us.
|
|
||||||
$page->write();
|
|
||||||
}
|
|
||||||
|
|
||||||
parent::onAfterDelete();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns TRUE if this object has a URLSegment value that does not conflict with any other objects. This methods
|
|
||||||
* checks for:
|
|
||||||
* - A page with the same URLSegment that has a conflict.
|
|
||||||
* - Conflicts with actions on the parent page.
|
|
||||||
* - A conflict caused by a root page having the same URLSegment as a class name.
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function validURLSegment() {
|
|
||||||
if(self::nested_urls() && $parent = $this->Parent()) {
|
|
||||||
if($controller = ModelAsController::controller_for($parent)) {
|
|
||||||
if($controller instanceof Controller && $controller->hasAction($this->URLSegment)) return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if(!self::nested_urls() || !$this->ParentID) {
|
|
||||||
if(class_exists($this->URLSegment) && is_subclass_of($this->URLSegment, 'RequestHandler')) return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
$IDFilter = ($this->ID) ? "AND \"SiteTree\".\"ID\" <> $this->ID" : null;
|
|
||||||
$parentFilter = null;
|
|
||||||
|
|
||||||
if(self::nested_urls()) {
|
|
||||||
if($this->ParentID) {
|
|
||||||
$parentFilter = " AND \"SiteTree\".\"ParentID\" = $this->ParentID";
|
|
||||||
} else {
|
|
||||||
$parentFilter = ' AND "SiteTree"."ParentID" = 0';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$existingPage = DataObject::get_one(
|
|
||||||
'SiteTree',
|
|
||||||
"\"URLSegment\" = '$this->URLSegment' $IDFilter $parentFilter"
|
|
||||||
);
|
|
||||||
if ($existingPage) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
$values = $this->extend('augmentValidURLSegment');
|
|
||||||
if (count($values) && !min($values)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate a URL segment based on the title provided.
|
|
||||||
*
|
|
||||||
* If {@link Extension}s wish to alter URL segment generation, they can do so by defining
|
|
||||||
* updateURLSegment(&$url, $title). $url will be passed by reference and should be modified.
|
|
||||||
* $title will contain the title that was originally used as the source of this generated URL.
|
|
||||||
* This lets decorators either start from scratch, or incrementally modify the generated URL.
|
|
||||||
*
|
|
||||||
* @param string $title Page title.
|
|
||||||
* @return string Generated url segment
|
|
||||||
*/
|
|
||||||
function generateURLSegment($title){
|
|
||||||
$t = Convert::raw2url($title);
|
|
||||||
|
|
||||||
if(!$t || $t == '-' || $t == '-1') {
|
|
||||||
$t = "page-$this->ID";
|
|
||||||
}
|
|
||||||
|
|
||||||
// Hook for decorators
|
|
||||||
$this->extend('updateURLSegment', $t, $title);
|
|
||||||
|
|
||||||
return $t;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
function getStageURLSegment() {
|
|
||||||
$stageRecord = Versioned::get_one_by_stage(
|
|
||||||
'SiteTree',
|
|
||||||
'Stage',
|
|
||||||
"\"SiteTree\".\"ID\" = $this->ID"
|
|
||||||
);
|
|
||||||
return ($stageRecord) ? $stageRecord->URLSegment : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
function getLiveURLSegment() {
|
|
||||||
$liveRecord = Versioned::get_one_by_stage(
|
|
||||||
'SiteTree',
|
|
||||||
'Live',
|
|
||||||
"\"SiteTree\".\"ID\" = $this->ID"
|
|
||||||
);
|
|
||||||
return ($liveRecord) ? $liveRecord->URLSegment : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Rewrite a file URL on this page, after its been renamed.
|
|
||||||
* Triggers the onRenameLinkedAsset action on extensions.
|
|
||||||
*/
|
|
||||||
function rewriteFileURL($old, $new) {
|
|
||||||
$fields = $this->inheritedDatabaseFields();
|
|
||||||
// Update the content without actually creating a new version
|
|
||||||
foreach(array("SiteTree_Live", "SiteTree") as $table) {
|
|
||||||
// Published site
|
|
||||||
$published = DB::query("SELECT * FROM \"$table\" WHERE \"ID\" = $this->ID")->record();
|
|
||||||
$origPublished = $published;
|
|
||||||
|
|
||||||
foreach($fields as $fieldName => $fieldType) {
|
|
||||||
if ($fieldType != 'HTMLText') continue;
|
|
||||||
|
|
||||||
// TODO: This doesn't work for HTMLText fields on other tables.
|
|
||||||
if(isset($published[$fieldName])) {
|
|
||||||
$published[$fieldName] = str_replace($old, $new, $published[$fieldName], $numReplaced);
|
|
||||||
if($numReplaced) {
|
|
||||||
DB::query("UPDATE \"$table\" SET \"$fieldName\" = '"
|
|
||||||
. Convert::raw2sql($published[$fieldName]) . "' WHERE \"ID\" = $this->ID");
|
|
||||||
|
|
||||||
// Tell static caching to update itself
|
|
||||||
if($table == 'SiteTree_Live') {
|
|
||||||
$publishedClass = $origPublished['ClassName'];
|
|
||||||
$origPublishedObj = new $publishedClass($origPublished);
|
|
||||||
$this->extend('onRenameLinkedAsset', $origPublishedObj);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the pages that depend on this page.
|
|
||||||
* This includes virtual pages, pages that link to it, etc.
|
|
||||||
*
|
|
||||||
* @param $includeVirtuals Set to false to exlcude virtual pages.
|
|
||||||
*/
|
|
||||||
function DependentPages($includeVirtuals = true) {
|
|
||||||
if(is_callable('Subsite::disable_subsite_filter')) Subsite::disable_subsite_filter(true);
|
|
||||||
|
|
||||||
// Content links
|
|
||||||
$items = $this->BackLinkTracking();
|
|
||||||
if(!$items) $items = new DataObjectSet();
|
|
||||||
else foreach($items as $item) $item->DependentLinkType = 'Content link';
|
|
||||||
|
|
||||||
// Virtual pages
|
|
||||||
if($includeVirtuals) {
|
|
||||||
$virtuals = $this->VirtualPages();
|
|
||||||
if($virtuals) {
|
|
||||||
foreach($virtuals as $item) $item->DependentLinkType = 'Virtual page';
|
|
||||||
$items->merge($virtuals);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Redirector pages
|
|
||||||
$redirectors = DataObject::get("RedirectorPage", "\"RedirectorPage\".\"RedirectionType\" = 'Internal' AND \"LinkToID\" = $this->ID");
|
|
||||||
if($redirectors) {
|
|
||||||
foreach($redirectors as $item) $item->DependentLinkType = 'Redirector page';
|
|
||||||
$items->merge($redirectors);
|
|
||||||
}
|
|
||||||
|
|
||||||
if(is_callable('Subsite::disable_subsite_filter')) Subsite::disable_subsite_filter(false);
|
|
||||||
return $items;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the number of {@link DependentPages()}
|
|
||||||
*
|
|
||||||
* @param $includeVirtuals Set to false to exlcude virtual pages.
|
|
||||||
*/
|
|
||||||
function DependentPagesCount($includeVirtuals = true) {
|
|
||||||
$links = DB::query("SELECT COUNT(*) FROM \"SiteTree_LinkTracking\"
|
|
||||||
INNER JOIN \"SiteTree\" ON \"SiteTree\".\"ID\" = \"SiteTree_LinkTracking\".\"SiteTreeID\"
|
|
||||||
WHERE \"ChildID\" = $this->ID ")->value();
|
|
||||||
if($includeVirtuals) {
|
|
||||||
$virtuals = DB::query("SELECT COUNT(*) FROM \"VirtualPage\"
|
|
||||||
INNER JOIN \"SiteTree\" ON \"SiteTree\".\"ID\" = \"VirtualPage\".\"ID\"
|
|
||||||
WHERE \"CopyContentFromID\" = $this->ID")->value();
|
|
||||||
} else {
|
|
||||||
$virtuals = 0;
|
|
||||||
}
|
|
||||||
$redirectors = DB::query("SELECT COUNT(*) FROM \"RedirectorPage\"
|
|
||||||
INNER JOIN \"SiteTree\" ON \"SiteTree\".\"ID\" = \"RedirectorPage\".\"ID\"
|
|
||||||
WHERE \"RedirectionType\" = 'Internal' AND \"LinkToID\" = $this->ID")->value();
|
|
||||||
|
|
||||||
|
|
||||||
return 0 + $links + $virtuals + $redirectors;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return all virtual pages that link to this page
|
|
||||||
*/
|
|
||||||
function VirtualPages() {
|
|
||||||
if(!$this->ID) return null;
|
|
||||||
if(class_exists('Subsite')) {
|
|
||||||
return Subsite::get_from_all_subsites('VirtualPage', "\"CopyContentFromID\" = " . (int)$this->ID);
|
|
||||||
} else {
|
|
||||||
return DataObject::get('VirtualPage', "\"CopyContentFromID\" = " . (int)$this->ID);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns a FieldSet with which to create the CMS editing form.
|
|
||||||
*
|
|
||||||
* You can override this in your child classes to add extra fields - first
|
|
||||||
* get the parent fields using parent::getCMSFields(), then use
|
|
||||||
* addFieldToTab() on the FieldSet.
|
|
||||||
*
|
|
||||||
* @return FieldSet The fields to be displayed in the CMS.
|
|
||||||
*/
|
|
||||||
function getCMSFields() {
|
|
||||||
require_once("forms/Form.php");
|
|
||||||
|
|
||||||
// Status / message
|
|
||||||
// Create a status message for multiple parents
|
|
||||||
if($this->ID && is_numeric($this->ID)) {
|
|
||||||
$linkedPages = $this->VirtualPages();
|
|
||||||
}
|
|
||||||
|
|
||||||
$parentPageLinks = array();
|
|
||||||
|
|
||||||
if(isset($linkedPages)) {
|
|
||||||
foreach($linkedPages as $linkedPage) {
|
|
||||||
$parentPage = $linkedPage->Parent;
|
|
||||||
if($parentPage) {
|
|
||||||
if($parentPage->ID) {
|
|
||||||
$parentPageLinks[] = "<a class=\"cmsEditlink\" href=\"admin/show/$linkedPage->ID\">{$parentPage->Title}</a>";
|
|
||||||
} else {
|
|
||||||
$parentPageLinks[] = "<a class=\"cmsEditlink\" href=\"admin/show/$linkedPage->ID\">" .
|
|
||||||
_t('SiteTree.TOPLEVEL', 'Site Content (Top Level)') .
|
|
||||||
"</a>";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$lastParent = array_pop($parentPageLinks);
|
|
||||||
$parentList = "'$lastParent'";
|
|
||||||
|
|
||||||
if(count($parentPageLinks) > 0) {
|
|
||||||
$parentList = "'" . implode("', '", $parentPageLinks) . "' and "
|
|
||||||
. $parentList;
|
|
||||||
}
|
|
||||||
|
|
||||||
$statusMessage[] = sprintf(
|
|
||||||
_t('SiteTree.APPEARSVIRTUALPAGES', "This content also appears on the virtual pages in the %s sections."),
|
|
||||||
$parentList
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if($this->HasBrokenLink || $this->HasBrokenFile) {
|
|
||||||
$statusMessage[] = _t('SiteTree.HASBROKENLINKS', "This page has broken links.");
|
|
||||||
}
|
|
||||||
|
|
||||||
$dependentNote = '';
|
|
||||||
$dependentTable = new LiteralField('DependentNote', '<p></p>');
|
|
||||||
|
|
||||||
// Create a table for showing pages linked to this one
|
|
||||||
$dependentPagesCount = $this->DependentPagesCount();
|
|
||||||
if($dependentPagesCount) {
|
|
||||||
$dependentColumns = array(
|
|
||||||
'Title' => $this->fieldLabel('Title'),
|
|
||||||
'AbsoluteLink' => _t('SiteTree.DependtPageColumnURL', 'URL'),
|
|
||||||
'DependentLinkType' => _t('SiteTree.DependtPageColumnLinkType', 'Link type'),
|
|
||||||
);
|
|
||||||
if(class_exists('Subsite')) $dependentColumns['Subsite.Title'] = singleton('Subsite')->i18n_singular_name();
|
|
||||||
|
|
||||||
$dependentNote = new LiteralField('DependentNote', '<p>' . _t('SiteTree.DEPENDENT_NOTE', 'The following pages depend on this page. This includes virtual pages, redirector pages, and pages with content links.') . '</p>');
|
|
||||||
$dependentTable = new TableListField(
|
|
||||||
'DependentPages',
|
|
||||||
'SiteTree',
|
|
||||||
$dependentColumns
|
|
||||||
);
|
|
||||||
$dependentTable->setCustomSourceItems($this->DependentPages());
|
|
||||||
$dependentTable->setFieldFormatting(array(
|
|
||||||
'Title' => '<a href=\"admin/show/$ID\">$Title</a>',
|
|
||||||
'AbsoluteLink' => '<a href=\"$value\">$value</a>',
|
|
||||||
));
|
|
||||||
$dependentTable->setPermissions(array(
|
|
||||||
'show',
|
|
||||||
'export'
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Lay out the fields
|
|
||||||
$fields = new FieldSet(
|
|
||||||
$rootTab = new TabSet("Root",
|
|
||||||
$tabContent = new TabSet('Content',
|
|
||||||
$tabMain = new Tab('Main',
|
|
||||||
new TextField("Title", $this->fieldLabel('Title')),
|
|
||||||
new TextField("MenuTitle", $this->fieldLabel('MenuTitle')),
|
|
||||||
new HtmlEditorField("Content", _t('SiteTree.HTMLEDITORTITLE', "Content", PR_MEDIUM, 'HTML editor title'))
|
|
||||||
),
|
|
||||||
$tabMeta = new Tab('Metadata',
|
|
||||||
new FieldGroup(_t('SiteTree.URL', "URL"),
|
|
||||||
new LabelField('BaseUrlLabel',Controller::join_links (
|
|
||||||
Director::absoluteBaseURL(),
|
|
||||||
(self::nested_urls() && $this->ParentID ? $this->Parent()->RelativeLink(true) : null)
|
|
||||||
)),
|
|
||||||
new TextField("URLSegment","URLSegment"),
|
|
||||||
new LabelField('TrailingSlashLabel',"/")
|
|
||||||
),
|
|
||||||
new LiteralField('LinkChangeNote', self::nested_urls() && count($this->Children()) ?
|
|
||||||
'<p>' . $this->fieldLabel('LinkChangeNote'). '</p>' : null
|
|
||||||
),
|
|
||||||
new HeaderField('MetaTagsHeader',$this->fieldLabel('MetaTagsHeader')),
|
|
||||||
new TextField("MetaTitle", $this->fieldLabel('MetaTitle')),
|
|
||||||
new TextareaField("MetaKeywords", $this->fieldLabel('MetaKeywords'), 1),
|
|
||||||
new TextareaField("MetaDescription", $this->fieldLabel('MetaDescription')),
|
|
||||||
new TextareaField("ExtraMeta",$this->fieldLabel('ExtraMeta'))
|
|
||||||
)
|
|
||||||
),
|
|
||||||
$tabBehaviour = new Tab('Behaviour',
|
|
||||||
new DropdownField(
|
|
||||||
"ClassName",
|
|
||||||
$this->fieldLabel('ClassName'),
|
|
||||||
$this->getClassDropdown()
|
|
||||||
),
|
|
||||||
|
|
||||||
new OptionsetField("ParentType", _t("SiteTree.PAGELOCATION", "Page location"), array(
|
|
||||||
"root" => _t("SiteTree.PARENTTYPE_ROOT", "Top-level page"),
|
|
||||||
"subpage" => _t("SiteTree.PARENTTYPE_SUBPAGE", "Sub-page underneath a parent page (choose below)"),
|
|
||||||
)),
|
|
||||||
$parentIDField = new TreeDropdownField("ParentID", $this->fieldLabel('ParentID'), 'SiteTree', 'ID', 'MenuTitle'),
|
|
||||||
|
|
||||||
new CheckboxField("ShowInMenus", $this->fieldLabel('ShowInMenus')),
|
|
||||||
new CheckboxField("ShowInSearch", $this->fieldLabel('ShowInSearch')),
|
|
||||||
new LiteralField(
|
|
||||||
"HomepageForDomainInfo",
|
|
||||||
"<p>" .
|
|
||||||
_t('SiteTree.NOTEUSEASHOMEPAGE',
|
|
||||||
"Use this page as the 'home page' for the following domains:
|
|
||||||
(separate multiple domains with commas)") .
|
|
||||||
"</p>"
|
|
||||||
),
|
|
||||||
new TextField(
|
|
||||||
"HomepageForDomain",
|
|
||||||
_t('SiteTree.HOMEPAGEFORDOMAIN', "Domain(s)", PR_MEDIUM, 'Listing domains that should be used as homepage')
|
|
||||||
)
|
|
||||||
),
|
|
||||||
$tabToDo = new Tab(_t('SiteTree.TABTODO', 'To-do') . ($this->ToDo ? '**' : ''),
|
|
||||||
new LiteralField("ToDoHelp", _t('SiteTree.TODOHELP', "<p>You can use this to keep track of work that needs to be done to the content of your site. To see all your pages with to do information, open the 'Site Reports' window on the left and select 'To Do'</p>")),
|
|
||||||
new TextareaField("ToDo", "", 10)
|
|
||||||
),
|
|
||||||
$tabDependent = new Tab('Dependent',
|
|
||||||
$dependentNote,
|
|
||||||
$dependentTable
|
|
||||||
),
|
|
||||||
$tabAccess = new Tab('Access',
|
|
||||||
new HeaderField('WhoCanViewHeader',_t('SiteTree.ACCESSHEADER', "Who can view this page?"), 2),
|
|
||||||
$viewersOptionsField = new OptionsetField(
|
|
||||||
"CanViewType",
|
|
||||||
""
|
|
||||||
),
|
|
||||||
$viewerGroupsField = new TreeMultiselectField("ViewerGroups", $this->fieldLabel('ViewerGroups')),
|
|
||||||
new HeaderField('WhoCanEditHeader',_t('SiteTree.EDITHEADER', "Who can edit this page?"), 2),
|
|
||||||
$editorsOptionsField = new OptionsetField(
|
|
||||||
"CanEditType",
|
|
||||||
""
|
|
||||||
),
|
|
||||||
$editorGroupsField = new TreeMultiselectField("EditorGroups", $this->fieldLabel('EditorGroups'))
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This filter ensures that the ParentID dropdown selection does not show this node,
|
|
||||||
* or its descendents, as this causes vanishing bugs.
|
|
||||||
*/
|
|
||||||
$parentIDField->setFilterFunction(create_function('$node', "return \$node->ID != {$this->ID};"));
|
|
||||||
|
|
||||||
// Conditional dependent pages tab
|
|
||||||
if($dependentPagesCount) $tabDependent->setTitle(_t('SiteTree.TABDEPENDENT', "Dependent pages") . " ($dependentPagesCount)");
|
|
||||||
else $fields->removeFieldFromTab('Root', 'Dependent');
|
|
||||||
|
|
||||||
// Make page location fields read-only if the user doesn't have the appropriate permission
|
|
||||||
if(!Permission::check("SITETREE_REORGANISE")) {
|
|
||||||
$fields->makeFieldReadonly('ParentType');
|
|
||||||
if($this->ParentType == 'root') {
|
|
||||||
$fields->removeByName('ParentID');
|
|
||||||
} else {
|
|
||||||
$fields->makeFieldReadonly('ParentID');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$viewersOptionsSource = array();
|
|
||||||
$viewersOptionsSource["Inherit"] = _t('SiteTree.INHERIT', "Inherit from parent page");
|
|
||||||
$viewersOptionsSource["Anyone"] = _t('SiteTree.ACCESSANYONE', "Anyone");
|
|
||||||
$viewersOptionsSource["LoggedInUsers"] = _t('SiteTree.ACCESSLOGGEDIN', "Logged-in users");
|
|
||||||
$viewersOptionsSource["OnlyTheseUsers"] = _t('SiteTree.ACCESSONLYTHESE', "Only these people (choose from list)");
|
|
||||||
$viewersOptionsField->setSource($viewersOptionsSource);
|
|
||||||
|
|
||||||
$editorsOptionsSource = array();
|
|
||||||
$editorsOptionsSource["Inherit"] = _t('SiteTree.INHERIT', "Inherit from parent page");
|
|
||||||
$editorsOptionsSource["LoggedInUsers"] = _t('SiteTree.EDITANYONE', "Anyone who can log-in to the CMS");
|
|
||||||
$editorsOptionsSource["OnlyTheseUsers"] = _t('SiteTree.EDITONLYTHESE', "Only these people (choose from list)");
|
|
||||||
$editorsOptionsField->setSource($editorsOptionsSource);
|
|
||||||
|
|
||||||
if(!Permission::check('SITETREE_GRANT_ACCESS')) {
|
|
||||||
$fields->makeFieldReadonly($viewersOptionsField);
|
|
||||||
if($this->CanViewType == 'OnlyTheseUsers') {
|
|
||||||
$fields->makeFieldReadonly($viewerGroupsField);
|
|
||||||
} else {
|
|
||||||
$fields->removeByName('ViewerGroups');
|
|
||||||
}
|
|
||||||
|
|
||||||
$fields->makeFieldReadonly($editorsOptionsField);
|
|
||||||
if($this->CanEditType == 'OnlyTheseUsers') {
|
|
||||||
$fields->makeFieldReadonly($editorGroupsField);
|
|
||||||
} else {
|
|
||||||
$fields->removeByName('EditorGroups');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$tabContent->setTitle(_t('SiteTree.TABCONTENT', "Content"));
|
|
||||||
$tabMain->setTitle(_t('SiteTree.TABMAIN', "Main"));
|
|
||||||
$tabMeta->setTitle(_t('SiteTree.TABMETA', "Metadata"));
|
|
||||||
$tabBehaviour->setTitle(_t('SiteTree.TABBEHAVIOUR', "Behavior"));
|
|
||||||
$tabAccess->setTitle(_t('SiteTree.TABACCESS', "Access"));
|
|
||||||
|
|
||||||
if(file_exists(BASE_PATH . '/install.php')) {
|
|
||||||
$fields->addFieldToTab("Root.Content.Main", new LiteralField("InstallWarningHeader",
|
|
||||||
"<p class=\"message warning\">" . _t("SiteTree.REMOVE_INSTALL_WARNING",
|
|
||||||
"Warning: You should remove install.php from this SilverStripe install for security reasons.")
|
|
||||||
. "</p>"), "Title");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if(self::$runCMSFieldsExtensions) {
|
|
||||||
$this->extend('updateCMSFields', $fields);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $fields;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @param boolean $includerelations a boolean value to indicate if the labels returned include relation fields
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
function fieldLabels($includerelations = true) {
|
|
||||||
$labels = parent::fieldLabels($includerelations);
|
|
||||||
|
|
||||||
$labels['Title'] = _t('SiteTree.PAGETITLE', "Page name");
|
|
||||||
$labels['MenuTitle'] = _t('SiteTree.MENUTITLE', "Navigation label");
|
|
||||||
$labels['MetaTagsHeader'] = _t('SiteTree.METAHEADER', "Search Engine Meta-tags");
|
|
||||||
$labels['MetaTitle'] = _t('SiteTree.METATITLE', "Title");
|
|
||||||
$labels['MetaDescription'] = _t('SiteTree.METADESC', "Description");
|
|
||||||
$labels['MetaKeywords'] = _t('SiteTree.METAKEYWORDS', "Keywords");
|
|
||||||
$labels['ExtraMeta'] = _t('SiteTree.METAEXTRA', "Custom Meta Tags");
|
|
||||||
$labels['ClassName'] = _t('SiteTree.PAGETYPE', "Page type", PR_MEDIUM, 'Classname of a page object');
|
|
||||||
$labels['ParentType'] = _t('SiteTree.PARENTTYPE', "Page location", PR_MEDIUM);
|
|
||||||
$labels['ParentID'] = _t('SiteTree.PARENTID', "Parent page", PR_MEDIUM);
|
|
||||||
$labels['ShowInMenus'] =_t('SiteTree.SHOWINMENUS', "Show in menus?");
|
|
||||||
$labels['ShowInSearch'] = _t('SiteTree.SHOWINSEARCH', "Show in search?");
|
|
||||||
$labels['ProvideComments'] = _t('SiteTree.ALLOWCOMMENTS', "Allow comments on this page?");
|
|
||||||
$labels['ViewerGroups'] = _t('SiteTree.VIEWERGROUPS', "Viewer Groups");
|
|
||||||
$labels['EditorGroups'] = _t('SiteTree.EDITORGROUPS', "Editor Groups");
|
|
||||||
$labels['URLSegment'] = _t('SiteTree.URLSegment', 'URL Segment', PR_MEDIUM, 'URL for this page');
|
|
||||||
$labels['Content'] = _t('SiteTree.Content', 'Content', PR_MEDIUM, 'Main HTML Content for a page');
|
|
||||||
$labels['HomepageForDomain'] = _t('SiteTree.HomepageForDomain', 'Hompage for this domain');
|
|
||||||
$labels['CanViewType'] = _t('SiteTree.Viewers', 'Viewers Groups');
|
|
||||||
$labels['CanEditType'] = _t('SiteTree.Editors', 'Editors Groups');
|
|
||||||
$labels['ToDo'] = _t('SiteTree.ToDo', 'Todo Notes');
|
|
||||||
$labels['Comments'] = _t('SiteTree.Comments', 'Comments');
|
|
||||||
$labels['LinkChangeNote'] = _t (
|
|
||||||
'SiteTree.LINKCHANGENOTE', 'Changing this page\'s link will also affect the links of all child pages.'
|
|
||||||
);
|
|
||||||
|
|
||||||
if($includerelations){
|
|
||||||
$labels['Parent'] = _t('SiteTree.has_one_Parent', 'Parent Page', PR_MEDIUM, 'The parent page in the site hierarchy');
|
|
||||||
$labels['LinkTracking'] = _t('SiteTree.many_many_LinkTracking', 'Link Tracking');
|
|
||||||
$labels['ImageTracking'] = _t('SiteTree.many_many_ImageTracking', 'Image Tracking');
|
|
||||||
$labels['BackLinkTracking'] = _t('SiteTree.many_many_BackLinkTracking', 'Backlink Tracking');
|
|
||||||
}
|
|
||||||
|
|
||||||
return $labels;
|
|
||||||
}
|
|
||||||
|
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the actions available in the CMS for this page - eg Save, Publish.
|
|
||||||
* @return FieldSet The available actions for this page.
|
|
||||||
*/
|
|
||||||
function getCMSActions() {
|
|
||||||
$actions = new FieldSet();
|
|
||||||
|
|
||||||
// "readonly"/viewing version that isn't the current version of the record
|
|
||||||
$stageOrLiveRecord = Versioned::get_one_by_stage($this->class, Versioned::current_stage(), sprintf('"SiteTree"."ID" = %d', $this->ID));
|
|
||||||
if($stageOrLiveRecord && $stageOrLiveRecord->Version != $this->Version) {
|
|
||||||
$actions->push(new FormAction('email', _t('CMSMain.EMAIL', 'Email')));
|
|
||||||
$actions->push(new FormAction('rollback', _t('CMSMain.ROLLBACK', 'Roll back to this version')));
|
|
||||||
|
|
||||||
// getCMSActions() can be extended with updateCMSActions() on a decorator
|
|
||||||
$this->extend('updateCMSActions', $actions);
|
|
||||||
|
|
||||||
return $actions;
|
|
||||||
}
|
|
||||||
|
|
||||||
if($this->isPublished() && $this->canPublish() && !$this->IsDeletedFromStage) {
|
|
||||||
// "unpublish"
|
|
||||||
$unpublish = FormAction::create('unpublish', _t('SiteTree.BUTTONUNPUBLISH', 'Unpublish'), 'delete');
|
|
||||||
$unpublish->describe(_t('SiteTree.BUTTONUNPUBLISHDESC', 'Remove this page from the published site'));
|
|
||||||
$unpublish->addExtraClass('delete');
|
|
||||||
$actions->push($unpublish);
|
|
||||||
}
|
|
||||||
|
|
||||||
if($this->stagesDiffer('Stage', 'Live') && !$this->IsDeletedFromStage) {
|
|
||||||
if($this->isPublished() && $this->canEdit()) {
|
|
||||||
// "rollback"
|
|
||||||
$rollback = FormAction::create('rollback', _t('SiteTree.BUTTONCANCELDRAFT', 'Cancel draft changes'), 'delete');
|
|
||||||
$rollback->describe(_t('SiteTree.BUTTONCANCELDRAFTDESC', 'Delete your draft and revert to the currently published page'));
|
|
||||||
$rollback->addExtraClass('delete');
|
|
||||||
$actions->push($rollback);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if($this->canEdit()) {
|
|
||||||
if($this->IsDeletedFromStage) {
|
|
||||||
if($this->ExistsOnLive) {
|
|
||||||
// "restore"
|
|
||||||
$actions->push(new FormAction('revert',_t('CMSMain.RESTORE','Restore')));
|
|
||||||
if($this->canDelete() && $this->canDeleteFromLive()) {
|
|
||||||
// "delete from live"
|
|
||||||
$actions->push(new FormAction('deletefromlive',_t('CMSMain.DELETEFP','Delete from the published site')));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// "restore"
|
|
||||||
$actions->push(new FormAction('restore',_t('CMSMain.RESTORE','Restore')));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if($this->canDelete()) {
|
|
||||||
// "delete"
|
|
||||||
$actions->push($deleteAction = new FormAction('delete',_t('CMSMain.DELETE','Delete from the draft site')));
|
|
||||||
$deleteAction->addExtraClass('delete');
|
|
||||||
}
|
|
||||||
|
|
||||||
// "save"
|
|
||||||
$actions->push(new FormAction('save',_t('CMSMain.SAVE','Save')));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if($this->canPublish() && !$this->IsDeletedFromStage) {
|
|
||||||
// "publish"
|
|
||||||
$actions->push(new FormAction('publish', _t('SiteTree.BUTTONSAVEPUBLISH', 'Save and Publish')));
|
|
||||||
}
|
|
||||||
|
|
||||||
// getCMSActions() can be extended with updateCMSActions() on a decorator
|
|
||||||
$this->extend('updateCMSActions', $actions);
|
|
||||||
|
|
||||||
return $actions;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Publish this page.
|
|
||||||
*
|
|
||||||
* @uses SiteTreeDecorator->onBeforePublish()
|
|
||||||
* @uses SiteTreeDecorator->onAfterPublish()
|
|
||||||
*/
|
|
||||||
function doPublish() {
|
|
||||||
if (!$this->canPublish()) return false;
|
|
||||||
|
|
||||||
$original = Versioned::get_one_by_stage("SiteTree", "Live", "\"SiteTree\".\"ID\" = $this->ID");
|
|
||||||
if(!$original) $original = new SiteTree();
|
|
||||||
|
|
||||||
// Handle activities undertaken by decorators
|
|
||||||
$this->invokeWithExtensions('onBeforePublish', $original);
|
|
||||||
//$this->PublishedByID = Member::currentUser()->ID;
|
|
||||||
$this->write();
|
|
||||||
$this->publish("Stage", "Live");
|
|
||||||
|
|
||||||
DB::query("UPDATE \"SiteTree_Live\"
|
|
||||||
SET \"Sort\" = (SELECT \"SiteTree\".\"Sort\" FROM \"SiteTree\" WHERE \"SiteTree_Live\".\"ID\" = \"SiteTree\".\"ID\")
|
|
||||||
WHERE EXISTS (SELECT \"SiteTree\".\"Sort\" FROM \"SiteTree\" WHERE \"SiteTree_Live\".\"ID\" = \"SiteTree\".\"ID\") AND \"ParentID\" = " . sprintf('%d', $this->ParentID) );
|
|
||||||
|
|
||||||
// Publish any virtual pages that might need publishing
|
|
||||||
$linkedPages = $this->VirtualPages();
|
|
||||||
if($linkedPages) foreach($linkedPages as $page) {
|
|
||||||
$page->copyFrom($page->CopyContentFrom());
|
|
||||||
$page->write();
|
|
||||||
if($page->ExistsOnLive) $page->doPublish();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Need to update pages linking to this one as no longer broken, on the live site
|
|
||||||
$origMode = Versioned::get_reading_mode();
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
foreach($this->DependentPages(false) as $page) {
|
|
||||||
// $page->write() calls syncLinkTracking, which does all the hard work for us.
|
|
||||||
$page->write();
|
|
||||||
}
|
|
||||||
Versioned::set_reading_mode($origMode);
|
|
||||||
|
|
||||||
// Check to write CMS homepage map.
|
|
||||||
$usingStaticPublishing = false;
|
|
||||||
foreach(ClassInfo::subclassesFor('StaticPublisher') as $class) if ($this->hasExtension($class)) $usingStaticPublishing = true;
|
|
||||||
|
|
||||||
// NOTE: if you change the path here, you must also change it in sapphire/static-main.php
|
|
||||||
if (self::$write_homepage_map) {
|
|
||||||
if ($usingStaticPublishing && $map = SiteTree::generate_homepage_domain_map()) {
|
|
||||||
@file_put_contents(BASE_PATH.'/'.ASSETS_DIR.'/_homepage-map.php', "<?php\n\$homepageMap = ".var_export($map, true)."; ?>");
|
|
||||||
} else { if (file_exists(BASE_PATH.'/'.ASSETS_DIR.'/_homepage-map.php')) unlink(BASE_PATH.'/'.ASSETS_DIR.'/_homepage-map.php'); }
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle activities undertaken by decorators
|
|
||||||
$this->invokeWithExtensions('onAfterPublish', $original);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static function generate_homepage_domain_map() {
|
|
||||||
$domainSpecificHomepages = Versioned::get_by_stage('Page', 'Live', "\"HomepageForDomain\" != ''", "\"URLSegment\" ASC");
|
|
||||||
if (!$domainSpecificHomepages) return false;
|
|
||||||
|
|
||||||
$map = array();
|
|
||||||
foreach($domainSpecificHomepages->map('URLSegment', 'HomepageForDomain') as $url => $domains) {
|
|
||||||
foreach(explode(',', $domains) as $domain) $map[$domain] = $url;
|
|
||||||
}
|
|
||||||
return $map;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Unpublish this page - remove it from the live site
|
|
||||||
*
|
|
||||||
* @uses SiteTreeDecorator->onBeforeUnpublish()
|
|
||||||
* @uses SiteTreeDecorator->onAfterUnpublish()
|
|
||||||
*/
|
|
||||||
function doUnpublish() {
|
|
||||||
if(!$this->canDeleteFromLive()) return false;
|
|
||||||
if(!$this->ID) return false;
|
|
||||||
|
|
||||||
$this->extend('onBeforeUnpublish');
|
|
||||||
|
|
||||||
$origStage = Versioned::current_stage();
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
|
|
||||||
// We should only unpublish virtualpages that exist on live
|
|
||||||
$virtualPages = $this->VirtualPages();
|
|
||||||
|
|
||||||
// This way our ID won't be unset
|
|
||||||
$clone = clone $this;
|
|
||||||
$clone->delete();
|
|
||||||
|
|
||||||
// Rewrite backlinks
|
|
||||||
$dependentPages = $this->DependentPages(false);
|
|
||||||
if($dependentPages) foreach($dependentPages as $page) {
|
|
||||||
// $page->write() calls syncLinkTracking, which does all the hard work for us.
|
|
||||||
$page->write();
|
|
||||||
}
|
|
||||||
Versioned::reading_stage($origStage);
|
|
||||||
|
|
||||||
// Unpublish any published virtual pages
|
|
||||||
if ($virtualPages) foreach($virtualPages as $vp) $vp->doUnpublish();
|
|
||||||
|
|
||||||
// If we're on the draft site, then we can update the status.
|
|
||||||
// Otherwise, these lines will resurrect an inappropriate record
|
|
||||||
if(DB::query("SELECT \"ID\" FROM \"SiteTree\" WHERE \"ID\" = $this->ID")->value()
|
|
||||||
&& Versioned::current_stage() != 'Live') {
|
|
||||||
$this->write();
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->extend('onAfterUnpublish');
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Roll the draft version of this page to match the published page.
|
|
||||||
* Caution: Doesn't overwrite the object properties with the rolled back version.
|
|
||||||
*
|
|
||||||
* @param $version Either the string 'Live' or a version number
|
|
||||||
*/
|
|
||||||
function doRollbackTo($version) {
|
|
||||||
$this->publish($version, "Stage", true);
|
|
||||||
$this->writeWithoutVersion();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Revert the draft changes: replace the draft content with the content on live
|
|
||||||
*/
|
|
||||||
function doRevertToLive() {
|
|
||||||
$this->publish("Live", "Stage", false);
|
|
||||||
|
|
||||||
// Use a clone to get the updates made by $this->publish
|
|
||||||
$clone = DataObject::get_by_id("SiteTree", $this->ID);
|
|
||||||
$clone->writeWithoutVersion();
|
|
||||||
|
|
||||||
// Need to update pages linking to this one as no longer broken
|
|
||||||
foreach($this->DependentPages(false) as $page) {
|
|
||||||
// $page->write() calls syncLinkTracking, which does all the hard work for us.
|
|
||||||
$page->write();
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->extend('onAfterRevertToLive');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Restore the content in the active copy of this SiteTree page to the stage site.
|
|
||||||
* @return The SiteTree object.
|
|
||||||
*/
|
|
||||||
function doRestoreToStage() {
|
|
||||||
// if no record can be found on draft stage (meaning it has been "deleted from draft" before),
|
|
||||||
// create an empty record
|
|
||||||
if(!DB::query("SELECT \"ID\" FROM \"SiteTree\" WHERE \"ID\" = $this->ID")->value()) {
|
|
||||||
$conn = DB::getConn();
|
|
||||||
if(method_exists($conn, 'allowPrimaryKeyEditing')) $conn->allowPrimaryKeyEditing('SiteTree', true);
|
|
||||||
DB::query("INSERT INTO \"SiteTree\" (\"ID\") VALUES ($this->ID)");
|
|
||||||
if(method_exists($conn, 'allowPrimaryKeyEditing')) $conn->allowPrimaryKeyEditing('SiteTree', false);
|
|
||||||
}
|
|
||||||
|
|
||||||
$oldStage = Versioned::current_stage();
|
|
||||||
Versioned::reading_stage('Stage');
|
|
||||||
$this->forceChange();
|
|
||||||
$this->writeWithoutVersion();
|
|
||||||
|
|
||||||
$result = DataObject::get_by_id($this->class, $this->ID);
|
|
||||||
|
|
||||||
// Need to update pages linking to this one as no longer broken
|
|
||||||
foreach($result->DependentPages(false) as $page) {
|
|
||||||
// $page->write() calls syncLinkTracking, which does all the hard work for us.
|
|
||||||
$page->write();
|
|
||||||
}
|
|
||||||
|
|
||||||
Versioned::reading_stage($oldStage);
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Synonym of {@link doUnpublish}
|
|
||||||
*/
|
|
||||||
function doDeleteFromLive() {
|
|
||||||
return $this->doUnpublish();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if this page is new - that is, if it has yet to have been written
|
|
||||||
* to the database.
|
|
||||||
*
|
|
||||||
* @return boolean True if this page is new.
|
|
||||||
*/
|
|
||||||
function isNew() {
|
|
||||||
/**
|
|
||||||
* This check was a problem for a self-hosted site, and may indicate a
|
|
||||||
* bug in the interpreter on their server, or a bug here
|
|
||||||
* Changing the condition from empty($this->ID) to
|
|
||||||
* !$this->ID && !$this->record['ID'] fixed this.
|
|
||||||
*/
|
|
||||||
if(empty($this->ID)) return true;
|
|
||||||
|
|
||||||
if(is_numeric($this->ID)) return false;
|
|
||||||
|
|
||||||
return stripos($this->ID, 'new') === 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if this page has been published.
|
|
||||||
*
|
|
||||||
* @return boolean True if this page has been published.
|
|
||||||
*/
|
|
||||||
function isPublished() {
|
|
||||||
if($this->isNew())
|
|
||||||
return false;
|
|
||||||
|
|
||||||
return (DB::query("SELECT \"ID\" FROM \"SiteTree_Live\" WHERE \"ID\" = $this->ID")->value())
|
|
||||||
? true
|
|
||||||
: false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the class dropdown used in the CMS to change the class of a page.
|
|
||||||
* This returns the list of options in the drop as a Map from class name
|
|
||||||
* to text in dropdown.
|
|
||||||
*
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
protected function getClassDropdown() {
|
|
||||||
$classes = self::page_type_classes();
|
|
||||||
$currentClass = null;
|
|
||||||
$result = array();
|
|
||||||
|
|
||||||
$result = array();
|
|
||||||
foreach($classes as $class) {
|
|
||||||
$instance = singleton($class);
|
|
||||||
if((($instance instanceof HiddenClass) || !$instance->canCreate()) && ($class != $this->class)) continue;
|
|
||||||
|
|
||||||
$pageTypeName = $instance->i18n_singular_name();
|
|
||||||
|
|
||||||
if($class == $this->class) {
|
|
||||||
$currentClass = $class;
|
|
||||||
$result[$class] = $pageTypeName;
|
|
||||||
} else {
|
|
||||||
$translation = _t(
|
|
||||||
'SiteTree.CHANGETO',
|
|
||||||
'Change to "%s"',
|
|
||||||
PR_MEDIUM,
|
|
||||||
"Pagetype selection dropdown with class names"
|
|
||||||
);
|
|
||||||
|
|
||||||
// @todo legacy fix to avoid empty classname dropdowns when translation doesn't include %s
|
|
||||||
if(strpos($translation, '%s') !== FALSE) {
|
|
||||||
$result[$class] = sprintf(
|
|
||||||
$translation,
|
|
||||||
$pageTypeName
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
$result[$class] = "{$translation} \"{$pageTypeName}\"";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// if we're in translation mode, the link between the translated pagetype
|
|
||||||
// title and the actual classname might not be obvious, so we add it in parantheses
|
|
||||||
// Example: class "RedirectorPage" has the title "Weiterleitung" in German,
|
|
||||||
// so it shows up as "Weiterleitung (RedirectorPage)"
|
|
||||||
if(i18n::get_locale() != 'en_US') {
|
|
||||||
$result[$class] = $result[$class] . " ({$class})";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// sort alphabetically, and put current on top
|
|
||||||
asort($result);
|
|
||||||
if($currentClass) {
|
|
||||||
$currentPageTypeName = $result[$currentClass];
|
|
||||||
unset($result[$currentClass]);
|
|
||||||
$result = array_reverse($result);
|
|
||||||
$result[$currentClass] = $currentPageTypeName;
|
|
||||||
$result = array_reverse($result);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns an array of the class names of classes that are allowed
|
|
||||||
* to be children of this class.
|
|
||||||
*
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
function allowedChildren() {
|
|
||||||
$candidates = $this->stat('allowed_children');
|
|
||||||
if($candidates && $candidates != "none" && $candidates != "SiteTree_root") {
|
|
||||||
foreach($candidates as $candidate) {
|
|
||||||
if(substr($candidate,0,1) == '*') {
|
|
||||||
$allowedChildren[] = substr($candidate,1);
|
|
||||||
} else {
|
|
||||||
$subclasses = ClassInfo::subclassesFor($candidate);
|
|
||||||
foreach($subclasses as $subclass) {
|
|
||||||
if($subclass != "SiteTree_root") $allowedChildren[] = $subclass;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return $allowedChildren;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the class name of the default class for children of this page.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
function defaultChild() {
|
|
||||||
$default = $this->stat('default_child');
|
|
||||||
$allowed = $this->allowedChildren();
|
|
||||||
if($allowed) {
|
|
||||||
if(!$default || !in_array($default, $allowed))
|
|
||||||
$default = reset($allowed);
|
|
||||||
return $default;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the class name of the default class for the parent of this
|
|
||||||
* page.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
function defaultParent() {
|
|
||||||
return $this->stat('default_parent');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the title for use in menus for this page. If the MenuTitle
|
|
||||||
* field is set it returns that, else it returns the Title field.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
function getMenuTitle(){
|
|
||||||
if($value = $this->getField("MenuTitle")) {
|
|
||||||
return $value;
|
|
||||||
} else {
|
|
||||||
return $this->getField("Title");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the menu title for this page.
|
|
||||||
*
|
|
||||||
* @param string $value
|
|
||||||
*/
|
|
||||||
function setMenuTitle($value) {
|
|
||||||
if($value == $this->getField("Title")) {
|
|
||||||
$this->setField("MenuTitle", null);
|
|
||||||
} else {
|
|
||||||
$this->setField("MenuTitle", $value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @deprecated 3.0 Use getTreeTitle()
|
|
||||||
*/
|
|
||||||
function TreeTitle() {
|
|
||||||
return $this->getTreeTitle();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* TitleWithStatus will return the title in an <span class="ins">, <span class="del"> or
|
|
||||||
* <span class=\"modified\"> tag depending on its publication status.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
function getTreeTitle() {
|
|
||||||
if($this->IsDeletedFromStage) {
|
|
||||||
if($this->ExistsOnLive) {
|
|
||||||
$tag ="span class=\"del\" title=\"" . _t('SiteTree.REMOVEDFROMDRAFT', 'Removed from draft site') . "\"";
|
|
||||||
} else {
|
|
||||||
$tag ="span class=\"del\" class=\"deletedOnLive\" title=\"" . _t('SiteTree.DELETEDPAGE', 'Deleted page') . "\"";
|
|
||||||
}
|
|
||||||
} elseif($this->IsAddedToStage) {
|
|
||||||
$tag = "span class=\"ins\" title=\"" . _t('SiteTree.ADDEDTODRAFT', 'Added to draft site') . "\"";
|
|
||||||
} elseif($this->IsModifiedOnStage) {
|
|
||||||
$tag = "span title=\"" . _t('SiteTree.MODIFIEDONDRAFT', 'Modified on draft site') . "\" class=\"modified\"";
|
|
||||||
} else {
|
|
||||||
$tag = '';
|
|
||||||
}
|
|
||||||
|
|
||||||
$text = Convert::raw2xml(str_replace(array("\n","\r"),"",$this->MenuTitle));
|
|
||||||
return ($tag) ? "<$tag>" . $text . "</" . strtok($tag,' ') . ">" : $text;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the page in the current page stack of the given level.
|
|
||||||
* Level(1) will return the main menu item that we're currently inside, etc.
|
|
||||||
*/
|
|
||||||
public function Level($level) {
|
|
||||||
$parent = $this;
|
|
||||||
$stack = array($parent);
|
|
||||||
while($parent = $parent->Parent) {
|
|
||||||
array_unshift($stack, $parent);
|
|
||||||
}
|
|
||||||
|
|
||||||
return isset($stack[$level-1]) ? $stack[$level-1] : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the CSS classes to apply to this node in the CMS tree
|
|
||||||
*
|
|
||||||
* @param Controller $controller The controller object that the tree
|
|
||||||
* appears on
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
function CMSTreeClasses($controller) {
|
|
||||||
$classes = sprintf('class-%s', $this->class);
|
|
||||||
if($this->HasBrokenFile || $this->HasBrokenLink)
|
|
||||||
$classes .= " BrokenLink";
|
|
||||||
|
|
||||||
if(!$this->canAddChildren())
|
|
||||||
$classes .= " nochildren";
|
|
||||||
|
|
||||||
if($controller->isCurrentPage($this))
|
|
||||||
$classes .= " current";
|
|
||||||
|
|
||||||
if(!$this->canEdit() && !$this->canAddChildren())
|
|
||||||
$classes .= " disabled";
|
|
||||||
|
|
||||||
if(!$this->ShowInMenus)
|
|
||||||
$classes .= " notinmenu";
|
|
||||||
|
|
||||||
//TODO: Add integration
|
|
||||||
/*
|
|
||||||
if($this->hasExtension('Translatable') && $controller->Locale != Translatable::default_locale() && !$this->isTranslation())
|
|
||||||
$classes .= " untranslated ";
|
|
||||||
*/
|
|
||||||
$classes .= $this->markingClasses();
|
|
||||||
|
|
||||||
return $classes;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Compares current draft with live version,
|
|
||||||
* and returns TRUE if no draft version of this page exists,
|
|
||||||
* but the page is still published (after triggering "Delete from draft site" in the CMS).
|
|
||||||
*
|
|
||||||
* @return boolean
|
|
||||||
*/
|
|
||||||
function getIsDeletedFromStage() {
|
|
||||||
if(!$this->ID) return true;
|
|
||||||
if($this->isNew()) return false;
|
|
||||||
|
|
||||||
$stageVersion = Versioned::get_versionnumber_by_stage('SiteTree', 'Stage', $this->ID);
|
|
||||||
|
|
||||||
// Return true for both completely deleted pages and for pages just deleted from stage.
|
|
||||||
return !($stageVersion);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return true if this page exists on the live site
|
|
||||||
*/
|
|
||||||
function getExistsOnLive() {
|
|
||||||
return (bool)Versioned::get_versionnumber_by_stage('SiteTree', 'Live', $this->ID);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Compares current draft with live version,
|
|
||||||
* and returns TRUE if these versions differ,
|
|
||||||
* meaning there have been unpublished changes to the draft site.
|
|
||||||
*
|
|
||||||
* @return boolean
|
|
||||||
*/
|
|
||||||
public function getIsModifiedOnStage() {
|
|
||||||
// new unsaved pages could be never be published
|
|
||||||
if($this->isNew()) return false;
|
|
||||||
|
|
||||||
$stageVersion = Versioned::get_versionnumber_by_stage('SiteTree', 'Stage', $this->ID);
|
|
||||||
$liveVersion = Versioned::get_versionnumber_by_stage('SiteTree', 'Live', $this->ID);
|
|
||||||
|
|
||||||
return ($stageVersion && $stageVersion != $liveVersion);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Compares current draft with live version,
|
|
||||||
* and returns true if no live version exists,
|
|
||||||
* meaning the page was never published.
|
|
||||||
*
|
|
||||||
* @return boolean
|
|
||||||
*/
|
|
||||||
public function getIsAddedToStage() {
|
|
||||||
// new unsaved pages could be never be published
|
|
||||||
if($this->isNew()) return false;
|
|
||||||
|
|
||||||
$stageVersion = Versioned::get_versionnumber_by_stage('SiteTree', 'Stage', $this->ID);
|
|
||||||
$liveVersion = Versioned::get_versionnumber_by_stage('SiteTree', 'Live', $this->ID);
|
|
||||||
|
|
||||||
return ($stageVersion && !$liveVersion);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Stops extendCMSFields() being called on getCMSFields().
|
|
||||||
* This is useful when you need access to fields added by subclasses
|
|
||||||
* of SiteTree in a decorator. Call before calling parent::getCMSFields(),
|
|
||||||
* and reenable afterwards.
|
|
||||||
*/
|
|
||||||
public static function disableCMSFieldsExtensions() {
|
|
||||||
self::$runCMSFieldsExtensions = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Reenables extendCMSFields() being called on getCMSFields() after
|
|
||||||
* it has been disabled by disableCMSFieldsExtensions().
|
|
||||||
*/
|
|
||||||
public static function enableCMSFieldsExtensions() {
|
|
||||||
self::$runCMSFieldsExtensions = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
function providePermissions() {
|
|
||||||
return array(
|
|
||||||
'SITETREE_GRANT_ACCESS' => array(
|
|
||||||
'name' => _t('SiteTree.PERMISSION_GRANTACCESS_DESCRIPTION', 'Manage access rights for content'),
|
|
||||||
'help' => _t('SiteTree.PERMISSION_GRANTACCESS_HELP', 'Allow setting of page-specific access restrictions in the "Pages" section.'),
|
|
||||||
'category' => _t('Permissions.PERMISSIONS_CATEGORY', 'Roles and access permissions'),
|
|
||||||
'sort' => 100
|
|
||||||
),
|
|
||||||
'SITETREE_VIEW_ALL' => array(
|
|
||||||
'name' => _t('SiteTree.VIEW_ALL_DESCRIPTION', 'View any page'),
|
|
||||||
'category' => _t('Permissions.CONTENT_CATEGORY', 'Content permissions'),
|
|
||||||
'sort' => -100,
|
|
||||||
'help' => _t('SiteTree.VIEW_ALL_HELP', 'Ability to view any page on the site, regardless of the settings on the Access tab. Requires the "Access to Site Content" permission')
|
|
||||||
),
|
|
||||||
'SITETREE_EDIT_ALL' => array(
|
|
||||||
'name' => _t('SiteTree.EDIT_ALL_DESCRIPTION', 'Edit any page'),
|
|
||||||
'category' => _t('Permissions.CONTENT_CATEGORY', 'Content permissions'),
|
|
||||||
'sort' => -50,
|
|
||||||
'help' => _t('SiteTree.EDIT_ALL_HELP', 'Ability to edit any page on the site, regardless of the settings on the Access tab. Requires the "Access to Site Content" permission')
|
|
||||||
),
|
|
||||||
'SITETREE_REORGANISE' => array(
|
|
||||||
'name' => _t('SiteTree.REORGANISE_DESCRIPTION', 'Change site structure'),
|
|
||||||
'category' => _t('Permissions.CONTENT_CATEGORY', 'Content permissions'),
|
|
||||||
'help' => _t('SiteTree.REORGANISE_HELP', 'Rearrange pages in the site tree through drag&drop.'),
|
|
||||||
'sort' => 100
|
|
||||||
),
|
|
||||||
'VIEW_DRAFT_CONTENT' => array(
|
|
||||||
'name' => _t('SiteTree.VIEW_DRAFT_CONTENT', 'View draft content'),
|
|
||||||
'category' => _t('Permissions.CONTENT_CATEGORY', 'Content permissions'),
|
|
||||||
'help' => _t('SiteTree.VIEW_DRAFT_CONTENT_HELP', 'Applies to viewing pages outside of the CMS in draft mode. Useful for external collaborators without CMS access.'),
|
|
||||||
'sort' => 100
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the translated Singular name
|
|
||||||
*
|
|
||||||
* @return String
|
|
||||||
*/
|
|
||||||
function i18n_singular_name() {
|
|
||||||
return _t($this->class.'.SINGULARNAME', $this->singular_name());
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Overloaded to also provide entities for 'Page' class which is usually
|
|
||||||
* located in custom code, hence textcollector picks it up for the wrong folder.
|
|
||||||
*/
|
|
||||||
function provideI18nEntities() {
|
|
||||||
$entities = parent::provideI18nEntities();
|
|
||||||
|
|
||||||
if(isset($entities['Page.SINGULARNAME'])) $entities['Page.SINGULARNAME'][3] = 'sapphire';
|
|
||||||
if(isset($entities['Page.PLURALNAME'])) $entities['Page.PLURALNAME'][3] = 'sapphire';
|
|
||||||
|
|
||||||
return $entities;
|
|
||||||
}
|
|
||||||
|
|
||||||
function getParentType() {
|
|
||||||
return $this->ParentID == 0 ? 'root' : 'subpage';
|
|
||||||
}
|
|
||||||
|
|
||||||
static function reset() {
|
|
||||||
self::$cache_permissions = array();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
?>
|
|
@ -1,31 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* Plug-ins for additional functionality in your SiteTree classes.
|
|
||||||
*
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage model
|
|
||||||
*/
|
|
||||||
abstract class SiteTreeDecorator extends DataObjectDecorator {
|
|
||||||
|
|
||||||
function onBeforePublish(&$original) {
|
|
||||||
}
|
|
||||||
|
|
||||||
function onAfterPublish(&$original) {
|
|
||||||
}
|
|
||||||
|
|
||||||
function onBeforeUnpublish() {
|
|
||||||
}
|
|
||||||
|
|
||||||
function onAfterUnpublish() {
|
|
||||||
}
|
|
||||||
|
|
||||||
function canAddChildren($member) {
|
|
||||||
}
|
|
||||||
|
|
||||||
function canPublish($member) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
?>
|
|
@ -1,403 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* Virtual Page creates an instance of a page, with the same fields that the original page had, but readonly.
|
|
||||||
* This allows you can have a page in mulitple places in the site structure, with different children without duplicating the content
|
|
||||||
* Note: This Only duplicates $db fields and not the $has_one etc..
|
|
||||||
* @package cms
|
|
||||||
*/
|
|
||||||
class VirtualPage extends Page {
|
|
||||||
|
|
||||||
static $icon = array("cms/images/treeicons/page-shortcut-gold","file");
|
|
||||||
|
|
||||||
public static $virtualFields;
|
|
||||||
|
|
||||||
static $has_one = array(
|
|
||||||
"CopyContentFrom" => "SiteTree",
|
|
||||||
);
|
|
||||||
|
|
||||||
static $db = array(
|
|
||||||
"VersionID" => "Int",
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generates the array of fields required for the page type.
|
|
||||||
*/
|
|
||||||
function getVirtualFields() {
|
|
||||||
$nonVirtualFields = array(
|
|
||||||
"SecurityTypeID",
|
|
||||||
"OwnerID",
|
|
||||||
"URLSegment",
|
|
||||||
"Sort",
|
|
||||||
"Status",
|
|
||||||
'ShowInMenus',
|
|
||||||
// 'Locale'
|
|
||||||
'ShowInSearch',
|
|
||||||
'Version',
|
|
||||||
"Embargo",
|
|
||||||
"Expiry",
|
|
||||||
);
|
|
||||||
|
|
||||||
$allFields = $this->db();
|
|
||||||
if($hasOne = $this->has_one()) foreach($hasOne as $link) $allFields[$link . 'ID'] = "Int";
|
|
||||||
foreach($allFields as $field => $type) {
|
|
||||||
if(!in_array($field, $nonVirtualFields)) $virtualFields[] = $field;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $virtualFields;
|
|
||||||
}
|
|
||||||
|
|
||||||
function CopyContentFrom() {
|
|
||||||
if(empty($this->record['CopyContentFromID'])) return new SiteTree();
|
|
||||||
|
|
||||||
if(!isset($this->components['CopyContentFrom'])) {
|
|
||||||
$this->components['CopyContentFrom'] = DataObject::get_by_id("SiteTree",
|
|
||||||
$this->record['CopyContentFromID']);
|
|
||||||
|
|
||||||
// Don't let VirtualPages point to other VirtualPages
|
|
||||||
if($this->components['CopyContentFrom'] instanceof VirtualPage) {
|
|
||||||
$this->components['CopyContentFrom'] = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
// has_one component semantics incidate than an empty object should be returned
|
|
||||||
if(!$this->components['CopyContentFrom']) {
|
|
||||||
$this->components['CopyContentFrom'] = new SiteTree();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->components['CopyContentFrom'];
|
|
||||||
}
|
|
||||||
function setCopyContentFromID($val) {
|
|
||||||
if(DataObject::get_by_id('SiteTree', $val) instanceof VirtualPage) $val = 0;
|
|
||||||
return $this->setField("CopyContentFromID", $val);
|
|
||||||
}
|
|
||||||
|
|
||||||
function ContentSource() {
|
|
||||||
return $this->CopyContentFrom();
|
|
||||||
}
|
|
||||||
|
|
||||||
function allowedChildren() {
|
|
||||||
if($this->CopyContentFrom()) {
|
|
||||||
return $this->CopyContentFrom()->allowedChildren();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function syncLinkTracking() {
|
|
||||||
if($this->CopyContentFromID) {
|
|
||||||
$this->HasBrokenLink = !(bool) DataObject::get_by_id('SiteTree', $this->CopyContentFromID);
|
|
||||||
} else {
|
|
||||||
$this->HasBrokenLink = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* We can only publish the page if there is a published source page
|
|
||||||
*/
|
|
||||||
public function canPublish($member = null) {
|
|
||||||
return $this->isPublishable() && parent::canPublish($member);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return true if we can delete this page from the live site, which is different from can
|
|
||||||
* we publish it.
|
|
||||||
*/
|
|
||||||
public function canDeleteFromLive($member = null) {
|
|
||||||
return parent::canPublish($member);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns true if is page is publishable by anyone at all
|
|
||||||
* Return false if the source page isn't published yet.
|
|
||||||
*
|
|
||||||
* Note that isPublishable doesn't affect ete from live, only publish.
|
|
||||||
*/
|
|
||||||
public function isPublishable() {
|
|
||||||
// No source
|
|
||||||
if(!$this->CopyContentFrom() || !$this->CopyContentFrom()->ID) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Unpublished source
|
|
||||||
if(!Versioned::get_versionnumber_by_stage('SiteTree', 'Live', $this->CopyContentFromID)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Default - publishable
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate the CMS fields from the fields from the original page.
|
|
||||||
*/
|
|
||||||
function getCMSFields($cms = null) {
|
|
||||||
$fields = parent::getCMSFields($cms);
|
|
||||||
|
|
||||||
// Setup the linking to the original page.
|
|
||||||
$copyContentFromField = new TreeDropdownField(
|
|
||||||
"CopyContentFromID",
|
|
||||||
_t('VirtualPage.CHOOSE', "Choose a page to link to"),
|
|
||||||
"SiteTree"
|
|
||||||
);
|
|
||||||
// filter doesn't let you select children of virtual pages as as source page
|
|
||||||
//$copyContentFromField->setFilterFunction(create_function('$item', 'return !($item instanceof VirtualPage);'));
|
|
||||||
|
|
||||||
// Setup virtual fields
|
|
||||||
if($virtualFields = $this->getVirtualFields()) {
|
|
||||||
$roTransformation = new ReadonlyTransformation();
|
|
||||||
foreach($virtualFields as $virtualField) {
|
|
||||||
if($fields->dataFieldByName($virtualField))
|
|
||||||
$fields->replaceField($virtualField, $fields->dataFieldByName($virtualField)->transform($roTransformation));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add fields to the tab
|
|
||||||
$fields->addFieldToTab("Root.Content.Main",
|
|
||||||
new HeaderField('VirtualPageHeader',_t('VirtualPage.HEADER', "This is a virtual page")),
|
|
||||||
"Title"
|
|
||||||
);
|
|
||||||
$fields->addFieldToTab("Root.Content.Main", $copyContentFromField, "Title");
|
|
||||||
|
|
||||||
// Create links back to the original object in the CMS
|
|
||||||
if($this->CopyContentFrom()->ID) {
|
|
||||||
$linkToContent = "<a class=\"cmsEditlink\" href=\"admin/show/$this->CopyContentFromID\">" .
|
|
||||||
_t('VirtualPage.EDITCONTENT', 'click here to edit the content') . "</a>";
|
|
||||||
$fields->addFieldToTab("Root.Content.Main",
|
|
||||||
$linkToContentLabelField = new LabelField('VirtualPageContentLinkLabel', $linkToContent),
|
|
||||||
"Title"
|
|
||||||
);
|
|
||||||
$linkToContentLabelField->setAllowHTML(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $fields;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* We have to change it to copy all the content from the original page first.
|
|
||||||
*/
|
|
||||||
function onBeforeWrite() {
|
|
||||||
// On regular write, this will copy from published source. This happens on every publish
|
|
||||||
if($this->extension_instances['Versioned']->migratingVersion
|
|
||||||
&& Versioned::current_stage() == 'Live') {
|
|
||||||
if($this->CopyContentFromID) {
|
|
||||||
$performCopyFrom = true;
|
|
||||||
|
|
||||||
$stageSourceVersion = DB::query("SELECT \"Version\" FROM \"SiteTree\" WHERE \"ID\" = $this->CopyContentFromID")->value();
|
|
||||||
$liveSourceVersion = DB::query("SELECT \"Version\" FROM \"SiteTree_Live\" WHERE \"ID\" = $this->CopyContentFromID")->value();
|
|
||||||
|
|
||||||
// We're going to create a new VP record in SiteTree_versions because the published
|
|
||||||
// version might not exist, unless we're publishing the latest version
|
|
||||||
if($stageSourceVersion != $liveSourceVersion) {
|
|
||||||
$this->extension_instances['Versioned']->migratingVersion = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// On regular write, this will copy from draft source. This is only executed when the source
|
|
||||||
// page changeds
|
|
||||||
} else {
|
|
||||||
$performCopyFrom = $this->isChanged('CopyContentFromID') && $this->CopyContentFromID != 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// On publish, this will copy from published source
|
|
||||||
if($performCopyFrom && $this instanceof VirtualPage) {
|
|
||||||
// This flush is needed because the get_one cache doesn't respect site version :-(
|
|
||||||
singleton('SiteTree')->flushCache();
|
|
||||||
$source = DataObject::get_one("SiteTree",sprintf('"SiteTree"."ID" = %d', $this->CopyContentFromID));
|
|
||||||
// Leave the updating of image tracking until after write, in case its a new record
|
|
||||||
$this->copyFrom($source, false);
|
|
||||||
$this->URLSegment = $source->URLSegment;
|
|
||||||
}
|
|
||||||
|
|
||||||
parent::onBeforeWrite();
|
|
||||||
}
|
|
||||||
|
|
||||||
function onAfterWrite() {
|
|
||||||
parent::onAfterWrite();
|
|
||||||
|
|
||||||
// Don't do this stuff when we're publishing
|
|
||||||
if(!$this->extension_instances['Versioned']->migratingVersion) {
|
|
||||||
if(
|
|
||||||
$this->isChanged('CopyContentFromID')
|
|
||||||
&& $this->CopyContentFromID != 0
|
|
||||||
&& $this instanceof VirtualPage
|
|
||||||
) {
|
|
||||||
$this->updateImageTracking();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Ensure we have an up-to-date version of everything.
|
|
||||||
*/
|
|
||||||
function copyFrom($source, $updateImageTracking = true) {
|
|
||||||
if($source) {
|
|
||||||
foreach($this->getVirtualFields() as $virtualField) {
|
|
||||||
$this->$virtualField = $source->$virtualField;
|
|
||||||
}
|
|
||||||
|
|
||||||
// We also want to copy ShowInMenus, but only if we're copying the
|
|
||||||
// source page for the first time.
|
|
||||||
if($this->isChanged('CopyContentFromID')) {
|
|
||||||
$this->ShowInMenus = $source->ShowInMenus;
|
|
||||||
}
|
|
||||||
|
|
||||||
if($updateImageTracking) $this->updateImageTracking();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateImageTracking() {
|
|
||||||
// Doesn't work on unsaved records
|
|
||||||
if(!$this->ID) return;
|
|
||||||
|
|
||||||
// Remove CopyContentFrom() from the cache
|
|
||||||
unset($this->components['CopyContentFrom']);
|
|
||||||
|
|
||||||
// Update ImageTracking
|
|
||||||
$this->ImageTracking()->setByIdList($this->CopyContentFrom()->ImageTracking()->column('ID'));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Allow attributes on the master page to pass
|
|
||||||
* through to the virtual page
|
|
||||||
*
|
|
||||||
* @param string $field
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
function __get($field) {
|
|
||||||
if(parent::hasMethod($funcName = "get$field")) {
|
|
||||||
return $this->$funcName();
|
|
||||||
} else if(parent::hasField($field)) {
|
|
||||||
return $this->getField($field);
|
|
||||||
} else {
|
|
||||||
return $this->copyContentFrom()->$field;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Pass unrecognized method calls on to the original data object
|
|
||||||
*
|
|
||||||
* @param string $method
|
|
||||||
* @param string $args
|
|
||||||
*/
|
|
||||||
function __call($method, $args) {
|
|
||||||
if(parent::hasMethod($method)) {
|
|
||||||
return parent::__call($method, $args);
|
|
||||||
} else {
|
|
||||||
return call_user_func_array(array($this->copyContentFrom(), $method), $args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function hasField($field) {
|
|
||||||
return (
|
|
||||||
array_key_exists($field, $this->record)
|
|
||||||
|| $this->hasDatabaseField($field)
|
|
||||||
|| array_key_exists($field, $this->db()) // Needed for composite fields
|
|
||||||
|| parent::hasMethod("get{$field}")
|
|
||||||
|| $this->CopyContentFrom()->hasField($field)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
* Overwrite to also check for method on the original data object
|
|
||||||
*
|
|
||||||
* @param string $method
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
function hasMethod($method) {
|
|
||||||
if(parent::hasMethod($method)) return true;
|
|
||||||
return $this->copyContentFrom()->hasMethod($method);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Controller for the virtual page.
|
|
||||||
* @package cms
|
|
||||||
*/
|
|
||||||
class VirtualPage_Controller extends Page_Controller {
|
|
||||||
|
|
||||||
static $allowed_actions = array(
|
|
||||||
'loadcontentall' => 'ADMIN',
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Reloads the content if the version is different ;-)
|
|
||||||
*/
|
|
||||||
function reloadContent() {
|
|
||||||
$this->failover->copyFrom($this->failover->CopyContentFrom());
|
|
||||||
$this->failover->write();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
function getViewer($action) {
|
|
||||||
$originalClass = get_class($this->CopyContentFrom());
|
|
||||||
if ($originalClass == 'SiteTree') $name = 'Page_Controller';
|
|
||||||
else $name = $originalClass."_Controller";
|
|
||||||
$controller = new $name();
|
|
||||||
return $controller->getViewer($action);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* When the virtualpage is loaded, check to see if the versions are the same
|
|
||||||
* if not, reload the content.
|
|
||||||
* NOTE: Virtual page must have a container object of subclass of sitetree.
|
|
||||||
* We can't load the content without an ID or record to copy it from.
|
|
||||||
*/
|
|
||||||
function init(){
|
|
||||||
if(isset($this->record) && $this->record->ID){
|
|
||||||
if($this->record->VersionID != $this->failover->CopyContentFrom()->Version){
|
|
||||||
$this->reloadContent();
|
|
||||||
$this->VersionID = $this->failover->CopyContentFrom()->VersionID;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
parent::init();
|
|
||||||
}
|
|
||||||
|
|
||||||
function loadcontentall() {
|
|
||||||
$pages = DataObject::get("VirtualPage");
|
|
||||||
foreach($pages as $page) {
|
|
||||||
$page->copyFrom($page->CopyContentFrom());
|
|
||||||
$page->write();
|
|
||||||
$page->publish("Stage", "Live");
|
|
||||||
echo "<li>Published $page->URLSegment";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Also check the original object's original controller for the method
|
|
||||||
*
|
|
||||||
* @param string $method
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
function hasMethod($method) {
|
|
||||||
$haveIt = parent::hasMethod($method);
|
|
||||||
if (!$haveIt) {
|
|
||||||
$originalClass = get_class($this->CopyContentFrom());
|
|
||||||
if ($originalClass == 'SiteTree') $name = 'ContentController';
|
|
||||||
else $name = $originalClass."_Controller";
|
|
||||||
$controller = new $name($this->dataRecord->copyContentFrom());
|
|
||||||
$haveIt = $controller->hasMethod($method);
|
|
||||||
}
|
|
||||||
return $haveIt;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Pass unrecognized method calls on to the original controller
|
|
||||||
*
|
|
||||||
* @param string $method
|
|
||||||
* @param string $args
|
|
||||||
*/
|
|
||||||
function __call($method, $args) {
|
|
||||||
try {
|
|
||||||
return parent::__call($method, $args);
|
|
||||||
} catch (Exception $e) {
|
|
||||||
// Hack... detect exception type. We really should use exception subclasses.
|
|
||||||
// if the exception isn't a 'no method' error, rethrow it
|
|
||||||
if ($e->getCode() !== 2175) throw $e;
|
|
||||||
$original = $this->copyContentFrom();
|
|
||||||
$originalClass = get_class($original);
|
|
||||||
if ($originalClass == 'SiteTree') $name = 'ContentController';
|
|
||||||
else $name = $originalClass."_Controller";
|
|
||||||
$controller = new $name($this->dataRecord->copyContentFrom());
|
|
||||||
return call_user_func_array(array($controller, $method), $args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
?>
|
|
@ -1,52 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* Rewrites plain internal HTML links into shortcode form, using existing link tracking information.
|
|
||||||
*
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage tasks
|
|
||||||
*/
|
|
||||||
class MigrateSiteTreeLinkingTask extends BuildTask {
|
|
||||||
|
|
||||||
protected $title = 'Migrate SiteTree Linking Task';
|
|
||||||
|
|
||||||
protected $description = 'Rewrites plain internal HTML links into shortcode form, using existing link tracking information.';
|
|
||||||
|
|
||||||
public function run($request) {
|
|
||||||
$pages = 0;
|
|
||||||
$links = 0;
|
|
||||||
|
|
||||||
$linkedPages = DataObject::get(
|
|
||||||
'SiteTree',
|
|
||||||
null,
|
|
||||||
null,
|
|
||||||
'INNER JOIN "SiteTree_LinkTracking" ON "SiteTree_LinkTracking"."SiteTreeID" = "SiteTree"."ID"'
|
|
||||||
);
|
|
||||||
|
|
||||||
if($linkedPages) foreach($linkedPages as $page) {
|
|
||||||
$tracking = DB::query(sprintf('SELECT "ChildID", "FieldName" FROM "SiteTree_LinkTracking" WHERE "SiteTreeID" = %d', $page->ID))->map();
|
|
||||||
|
|
||||||
foreach($tracking as $childID => $fieldName) {
|
|
||||||
$linked = DataObject::get_by_id('SiteTree', $childID);
|
|
||||||
|
|
||||||
// TOOD: Replace in all HTMLText fields
|
|
||||||
$page->Content = preg_replace (
|
|
||||||
"/href *= *([\"']?){$linked->URLSegment}\/?/i",
|
|
||||||
"href=$1[sitetree_link id={$linked->ID}]",
|
|
||||||
$page->Content,
|
|
||||||
-1,
|
|
||||||
$replaced
|
|
||||||
);
|
|
||||||
|
|
||||||
if($replaced) {
|
|
||||||
$links += $replaced;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$page->write();
|
|
||||||
$pages++;
|
|
||||||
}
|
|
||||||
|
|
||||||
echo "Rewrote $links link(s) on $pages page(s) to use shortcodes.\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,354 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* Identify "orphaned" pages which point to a parent
|
|
||||||
* that no longer exists in a specific stage.
|
|
||||||
* Shows the pages to an administrator, who can then
|
|
||||||
* decide which pages to remove by ticking a checkbox
|
|
||||||
* and manually executing the removal.
|
|
||||||
*
|
|
||||||
* Caution: Pages also count as orphans if they don't
|
|
||||||
* have parents in this stage, even if the parent has a representation
|
|
||||||
* in the other stage:
|
|
||||||
* - A live child is orphaned if its parent was deleted from live, but still exists on stage
|
|
||||||
* - A stage child is orphaned if its parent was deleted from stage, but still exists on live
|
|
||||||
*
|
|
||||||
* See {@link RemoveOrphanedPagesTaskTest} for an example sitetree
|
|
||||||
* before and after orphan removal.
|
|
||||||
*
|
|
||||||
* @author Ingo Schommer (<firstname>@silverstripe.com), SilverStripe Ltd.
|
|
||||||
*
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage tasks
|
|
||||||
*/
|
|
||||||
//class RemoveOrphanedPagesTask extends BuildTask {
|
|
||||||
class RemoveOrphanedPagesTask extends Controller {
|
|
||||||
|
|
||||||
static $allowed_actions = array(
|
|
||||||
'index' => 'ADMIN',
|
|
||||||
'Form' => 'ADMIN',
|
|
||||||
'run' => 'ADMIN',
|
|
||||||
'handleAction' => 'ADMIN',
|
|
||||||
);
|
|
||||||
|
|
||||||
protected $title = 'Removed orphaned pages without existing parents from both stage and live';
|
|
||||||
|
|
||||||
protected $description = "
|
|
||||||
<p>
|
|
||||||
Identify 'orphaned' pages which point to a parent
|
|
||||||
that no longer exists in a specific stage.
|
|
||||||
</p>
|
|
||||||
<p>
|
|
||||||
Caution: Pages also count as orphans if they don't
|
|
||||||
have parents in this stage, even if the parent has a representation
|
|
||||||
in the other stage:<br />
|
|
||||||
- A live child is orphaned if its parent was deleted from live, but still exists on stage<br />
|
|
||||||
- A stage child is orphaned if its parent was deleted from stage, but still exists on live
|
|
||||||
</p>
|
|
||||||
";
|
|
||||||
|
|
||||||
protected $orphanedSearchClass = 'SiteTree';
|
|
||||||
|
|
||||||
function Link() {
|
|
||||||
return $this->class;
|
|
||||||
}
|
|
||||||
|
|
||||||
function init() {
|
|
||||||
parent::init();
|
|
||||||
|
|
||||||
if(!Permission::check('ADMIN')) {
|
|
||||||
return Security::permissionFailure($this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function index() {
|
|
||||||
Requirements::javascript(SAPPHIRE_DIR . '/thirdparty/jquery/jquery.js');
|
|
||||||
Requirements::customCSS('#OrphanIDs .middleColumn {width: auto;}');
|
|
||||||
Requirements::customCSS('#OrphanIDs label {display: inline;}');
|
|
||||||
|
|
||||||
return $this->renderWith('BlankPage');
|
|
||||||
}
|
|
||||||
|
|
||||||
function Form() {
|
|
||||||
$fields = new FieldSet();
|
|
||||||
$source = array();
|
|
||||||
|
|
||||||
$fields->push(new HeaderField(
|
|
||||||
'Header',
|
|
||||||
_t('RemoveOrphanedPagesTask.HEADER', 'Remove all orphaned pages task')
|
|
||||||
));
|
|
||||||
$fields->push(new LiteralField(
|
|
||||||
'Description',
|
|
||||||
$this->description
|
|
||||||
));
|
|
||||||
|
|
||||||
$orphans = $this->getOrphanedPages($this->orphanedSearchClass);
|
|
||||||
if($orphans) foreach($orphans as $orphan) {
|
|
||||||
$latestVersion = Versioned::get_latest_version($this->orphanedSearchClass, $orphan->ID);
|
|
||||||
$latestAuthor = DataObject::get_by_id('Member', $latestVersion->AuthorID);
|
|
||||||
$stageRecord = Versioned::get_one_by_stage(
|
|
||||||
$this->orphanedSearchClass,
|
|
||||||
'Stage',
|
|
||||||
sprintf("\"%s\".\"ID\" = %d",
|
|
||||||
ClassInfo::baseDataClass($this->orphanedSearchClass),
|
|
||||||
$orphan->ID
|
|
||||||
)
|
|
||||||
);
|
|
||||||
$liveRecord = Versioned::get_one_by_stage(
|
|
||||||
$this->orphanedSearchClass,
|
|
||||||
'Live',
|
|
||||||
sprintf("\"%s\".\"ID\" = %d",
|
|
||||||
ClassInfo::baseDataClass($this->orphanedSearchClass),
|
|
||||||
$orphan->ID
|
|
||||||
)
|
|
||||||
);
|
|
||||||
$label = sprintf(
|
|
||||||
'<a href="admin/show/%d">%s</a> <small>(#%d, Last Modified Date: %s, Last Modifier: %s, %s)</small>',
|
|
||||||
$orphan->ID,
|
|
||||||
$orphan->Title,
|
|
||||||
$orphan->ID,
|
|
||||||
DBField::create('Date', $orphan->LastEdited)->Nice(),
|
|
||||||
($latestAuthor) ? $latestAuthor->Title : 'unknown',
|
|
||||||
($liveRecord) ? 'is published' : 'not published'
|
|
||||||
);
|
|
||||||
$source[$orphan->ID] = $label;
|
|
||||||
}
|
|
||||||
|
|
||||||
if($orphans && $orphans->Count()) {
|
|
||||||
$fields->push(new CheckboxSetField('OrphanIDs', false, $source));
|
|
||||||
$fields->push(new LiteralField(
|
|
||||||
'SelectAllLiteral',
|
|
||||||
sprintf(
|
|
||||||
'<p><a href="#" onclick="javascript:jQuery(\'#Form_Form_OrphanIDs :checkbox\').attr(\'checked\', \'checked\'); return false;">%s</a> ',
|
|
||||||
_t('RemoveOrphanedPagesTask.SELECTALL', 'select all')
|
|
||||||
)
|
|
||||||
));
|
|
||||||
$fields->push(new LiteralField(
|
|
||||||
'UnselectAllLiteral',
|
|
||||||
sprintf(
|
|
||||||
'<a href="#" onclick="javascript:jQuery(\'#Form_Form_OrphanIDs :checkbox\').attr(\'checked\', \'\'); return false;">%s</a></p>',
|
|
||||||
_t('RemoveOrphanedPagesTask.UNSELECTALL', 'unselect all')
|
|
||||||
)
|
|
||||||
));
|
|
||||||
$fields->push(new OptionSetField(
|
|
||||||
'OrphanOperation',
|
|
||||||
_t('RemoveOrphanedPagesTask.CHOOSEOPERATION', 'Choose operation:'),
|
|
||||||
array(
|
|
||||||
'rebase' => _t(
|
|
||||||
'RemoveOrphanedPagesTask.OPERATION_REBASE',
|
|
||||||
sprintf(
|
|
||||||
'Rebase selected to a new holder page "%s" and unpublish. None of these pages will show up for website visitors.',
|
|
||||||
$this->rebaseHolderTitle()
|
|
||||||
)
|
|
||||||
),
|
|
||||||
'remove' => _t('RemoveOrphanedPagesTask.OPERATION_REMOVE', 'Remove selected from all stages (WARNING: Will destroy all selected pages from both stage and live)'),
|
|
||||||
),
|
|
||||||
'rebase'
|
|
||||||
));
|
|
||||||
$fields->push(new LiteralField(
|
|
||||||
'Warning',
|
|
||||||
sprintf('<p class="message">%s</p>',
|
|
||||||
_t(
|
|
||||||
'RemoveOrphanedPagesTask.DELETEWARNING',
|
|
||||||
'Warning: These operations are not reversible. Please handle with care.'
|
|
||||||
)
|
|
||||||
)
|
|
||||||
));
|
|
||||||
} else {
|
|
||||||
$fields->push(new LiteralField(
|
|
||||||
'NotFoundLabel',
|
|
||||||
sprintf(
|
|
||||||
'<p class="message">%s</p>',
|
|
||||||
_t('RemoveOrphanedPagesTask.NONEFOUND', 'No orphans found')
|
|
||||||
)
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
$form = new Form(
|
|
||||||
$this,
|
|
||||||
'Form',
|
|
||||||
$fields,
|
|
||||||
new FieldSet(
|
|
||||||
new FormAction('doSubmit', _t('RemoveOrphanedPagesTask.BUTTONRUN', 'Run'))
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
if(!$orphans || !$orphans->Count()) {
|
|
||||||
$form->makeReadonly();
|
|
||||||
}
|
|
||||||
|
|
||||||
return $form;
|
|
||||||
}
|
|
||||||
|
|
||||||
function run($request) {
|
|
||||||
// @todo Merge with BuildTask functionality
|
|
||||||
}
|
|
||||||
|
|
||||||
function doSubmit($data, $form) {
|
|
||||||
set_time_limit(60*10); // 10 minutes
|
|
||||||
|
|
||||||
if(!isset($data['OrphanIDs']) || !isset($data['OrphanOperation'])) return false;
|
|
||||||
|
|
||||||
switch($data['OrphanOperation']) {
|
|
||||||
case 'remove':
|
|
||||||
$successIDs = $this->removeOrphans($data['OrphanIDs']);
|
|
||||||
break;
|
|
||||||
case 'rebase':
|
|
||||||
$successIDs = $this->rebaseOrphans($data['OrphanIDs']);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
user_error(sprintf("Unknown operation: '%s'", $data['OrphanOperation']), E_USER_ERROR);
|
|
||||||
}
|
|
||||||
|
|
||||||
$content = '';
|
|
||||||
if($successIDs) {
|
|
||||||
$content .= "<ul>";
|
|
||||||
foreach($successIDs as $id => $label) {
|
|
||||||
$content .= sprintf('<li>%s</li>', $label);
|
|
||||||
}
|
|
||||||
$content .= "</ul>";
|
|
||||||
} else {
|
|
||||||
$content = _t('RemoveOrphanedPagesTask.NONEREMOVED', 'None removed');
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->customise(array(
|
|
||||||
'Content' => $content,
|
|
||||||
'Form' => ' '
|
|
||||||
))->renderWith('BlankPage');
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function removeOrphans($orphanIDs) {
|
|
||||||
$removedOrphans = array();
|
|
||||||
foreach($orphanIDs as $id) {
|
|
||||||
$stageRecord = Versioned::get_one_by_stage(
|
|
||||||
$this->orphanedSearchClass,
|
|
||||||
'Stage',
|
|
||||||
sprintf("\"%s\".\"ID\" = %d",
|
|
||||||
ClassInfo::baseDataClass($this->orphanedSearchClass),
|
|
||||||
$id
|
|
||||||
)
|
|
||||||
);
|
|
||||||
if($stageRecord) {
|
|
||||||
$removedOrphans[$stageRecord->ID] = sprintf('Removed %s (#%d) from Stage', $stageRecord->Title, $stageRecord->ID);
|
|
||||||
$stageRecord->delete();
|
|
||||||
$stageRecord->destroy();
|
|
||||||
unset($stageRecord);
|
|
||||||
}
|
|
||||||
$liveRecord = Versioned::get_one_by_stage(
|
|
||||||
$this->orphanedSearchClass,
|
|
||||||
'Live',
|
|
||||||
sprintf("\"%s\".\"ID\" = %d",
|
|
||||||
ClassInfo::baseDataClass($this->orphanedSearchClass),
|
|
||||||
$id
|
|
||||||
)
|
|
||||||
);
|
|
||||||
if($liveRecord) {
|
|
||||||
$removedOrphans[$liveRecord->ID] = sprintf('Removed %s (#%d) from Live', $liveRecord->Title, $liveRecord->ID);
|
|
||||||
$liveRecord->doDeleteFromLive();
|
|
||||||
$liveRecord->destroy();
|
|
||||||
unset($liveRecord);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $removedOrphans;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function rebaseHolderTitle() {
|
|
||||||
return sprintf('Rebased Orphans (%s)', date('d/m/Y g:ia', time()));
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function rebaseOrphans($orphanIDs) {
|
|
||||||
$holder = new SiteTree();
|
|
||||||
$holder->ShowInMenus = 0;
|
|
||||||
$holder->ShowInSearch = 0;
|
|
||||||
$holder->ParentID = 0;
|
|
||||||
$holder->Title = $this->rebaseHolderTitle();
|
|
||||||
$holder->write();
|
|
||||||
|
|
||||||
$removedOrphans = array();
|
|
||||||
foreach($orphanIDs as $id) {
|
|
||||||
$stageRecord = Versioned::get_one_by_stage(
|
|
||||||
$this->orphanedSearchClass,
|
|
||||||
'Stage',
|
|
||||||
sprintf("\"%s\".\"ID\" = %d",
|
|
||||||
ClassInfo::baseDataClass($this->orphanedSearchClass),
|
|
||||||
$id
|
|
||||||
)
|
|
||||||
);
|
|
||||||
if($stageRecord) {
|
|
||||||
$removedOrphans[$stageRecord->ID] = sprintf('Rebased %s (#%d)', $stageRecord->Title, $stageRecord->ID);
|
|
||||||
$stageRecord->ParentID = $holder->ID;
|
|
||||||
$stageRecord->ShowInMenus = 0;
|
|
||||||
$stageRecord->ShowInSearch = 0;
|
|
||||||
$stageRecord->write();
|
|
||||||
$stageRecord->doUnpublish();
|
|
||||||
$stageRecord->destroy();
|
|
||||||
//unset($stageRecord);
|
|
||||||
}
|
|
||||||
$liveRecord = Versioned::get_one_by_stage(
|
|
||||||
$this->orphanedSearchClass,
|
|
||||||
'Live',
|
|
||||||
sprintf("\"%s\".\"ID\" = %d",
|
|
||||||
ClassInfo::baseDataClass($this->orphanedSearchClass),
|
|
||||||
$id
|
|
||||||
)
|
|
||||||
);
|
|
||||||
if($liveRecord) {
|
|
||||||
$removedOrphans[$liveRecord->ID] = sprintf('Rebased %s (#%d)', $liveRecord->Title, $liveRecord->ID);
|
|
||||||
$liveRecord->ParentID = $holder->ID;
|
|
||||||
$liveRecord->ShowInMenus = 0;
|
|
||||||
$liveRecord->ShowInSearch = 0;
|
|
||||||
$liveRecord->write();
|
|
||||||
if(!$stageRecord) $liveRecord->doRestoreToStage();
|
|
||||||
$liveRecord->doUnpublish();
|
|
||||||
$liveRecord->destroy();
|
|
||||||
unset($liveRecord);
|
|
||||||
}
|
|
||||||
if($stageRecord) {
|
|
||||||
unset($stageRecord);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $removedOrphans;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets all orphans from "Stage" and "Live" stages.
|
|
||||||
*
|
|
||||||
* @param string $class
|
|
||||||
* @param string $filter
|
|
||||||
* @param string $sort
|
|
||||||
* @param string $join
|
|
||||||
* @param int|array $limit
|
|
||||||
* @return DataObjectSet
|
|
||||||
*/
|
|
||||||
function getOrphanedPages($class = 'SiteTree', $filter = '', $sort = null, $join = null, $limit = null) {
|
|
||||||
$filter .= ($filter) ? ' AND ' : '';
|
|
||||||
$filter .= sprintf("\"%s\".\"ParentID\" != 0 AND \"Parents\".\"ID\" IS NULL", $class);
|
|
||||||
|
|
||||||
$orphans = new DataObjectSet();
|
|
||||||
foreach(array('Stage', 'Live') as $stage) {
|
|
||||||
$joinByStage = $join;
|
|
||||||
$table = $class;
|
|
||||||
$table .= ($stage == 'Live') ? '_Live' : '';
|
|
||||||
$joinByStage .= sprintf(
|
|
||||||
"LEFT JOIN \"%s\" AS \"Parents\" ON \"%s\".\"ParentID\" = \"Parents\".\"ID\"",
|
|
||||||
$table,
|
|
||||||
$table
|
|
||||||
);
|
|
||||||
$stageOrphans = Versioned::get_by_stage(
|
|
||||||
$class,
|
|
||||||
$stage,
|
|
||||||
$filter,
|
|
||||||
$sort,
|
|
||||||
$joinByStage,
|
|
||||||
$limit
|
|
||||||
);
|
|
||||||
$orphans->merge($stageOrphans);
|
|
||||||
}
|
|
||||||
|
|
||||||
$orphans->removeDuplicates();
|
|
||||||
|
|
||||||
return $orphans;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
?>
|
|
@ -1,51 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage tasks
|
|
||||||
*/
|
|
||||||
class UpgradeSiteTreePermissionSchemaTask extends BuildTask {
|
|
||||||
static $allowed_actions = array(
|
|
||||||
'*' => 'ADMIN'
|
|
||||||
);
|
|
||||||
|
|
||||||
protected $title = 'Upgrade SiteTree Permissions Schema';
|
|
||||||
|
|
||||||
protected $description = "Move data from legacy columns to new schema introduced in SilverStripe 2.1.<br />
|
|
||||||
SiteTree->Viewers to SiteTree->CanViewType<br />
|
|
||||||
SiteTree->Editors to SiteTree->CanEditType<br />
|
|
||||||
SiteTree->ViewersGroup to SiteTree->ViewerGroups (has_one to many_many)<br />
|
|
||||||
SiteTree->Editorsroup to SiteTree->EditorGroups (has_one to many_many)<br />
|
|
||||||
See http://open.silverstripe.com/ticket/2847
|
|
||||||
";
|
|
||||||
|
|
||||||
function run($request) {
|
|
||||||
// transfer values for changed column name
|
|
||||||
foreach(array('SiteTree','SiteTree_Live','SiteTree_versions') as $table) {
|
|
||||||
DB::query("UPDATE \"{$table}\" SET \"CanViewType\" = 'Viewers';");
|
|
||||||
DB::query("UPDATE \"{$table}\" SET \"CanEditType\" = 'Editors';");
|
|
||||||
}
|
|
||||||
//Debug::message('Moved SiteTree->Viewers to SiteTree->CanViewType');
|
|
||||||
//Debug::message('Moved SiteTree->Editors to SiteTree->CanEditType');
|
|
||||||
|
|
||||||
// convert has_many to many_many
|
|
||||||
$pageIDs = DB::query("SELECT ID FROM SiteTree")->column('ID');
|
|
||||||
foreach($pageIDs as $pageID) {
|
|
||||||
$page = DataObject::get_by_id('SiteTree', $pageID);
|
|
||||||
if($page->ViewersGroup && DataObject::get_by_id("Group", $page->ViewersGroup)) $page->ViewerGroups()->add($page->ViewersGroup);
|
|
||||||
if($page->EditorsGroup && DataObject::get_by_id("Group", $page->EditorsGroup)) $page->EditorGroups()->add($page->EditorsGroup);
|
|
||||||
|
|
||||||
$page->destroy();
|
|
||||||
unset($page);
|
|
||||||
}
|
|
||||||
//Debug::message('SiteTree->ViewersGroup to SiteTree->ViewerGroups (has_one to many_many)');
|
|
||||||
//Debug::message('SiteTree->EditorsGroup to SiteTree->EditorGroups (has_one to many_many)');
|
|
||||||
|
|
||||||
// rename legacy columns
|
|
||||||
foreach(array('SiteTree','SiteTree_Live','SiteTree_versions') as $table) {
|
|
||||||
foreach(array('Viewers','Editors','ViewersGroup','EditorsGroup') as $field) {
|
|
||||||
DB::getConn()->dontRequireField($table, $field);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
?>
|
|
@ -1,64 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage tests
|
|
||||||
*/
|
|
||||||
class ErrorPageTest extends FunctionalTest {
|
|
||||||
|
|
||||||
static $fixture_file = 'sapphire/tests/ErrorPageTest.yml';
|
|
||||||
|
|
||||||
protected $orig = array();
|
|
||||||
|
|
||||||
protected $tmpAssetsPath = '';
|
|
||||||
|
|
||||||
function setUp() {
|
|
||||||
parent::setUp();
|
|
||||||
|
|
||||||
$this->orig['ErrorPage_staticfilepath'] = ErrorPage::get_static_filepath();
|
|
||||||
$this->tmpAssetsPath = sprintf('%s/_tmp_assets_%s', TEMP_FOLDER, rand());
|
|
||||||
Filesystem::makeFolder($this->tmpAssetsPath . '/ErrorPageTest');
|
|
||||||
ErrorPage::set_static_filepath($this->tmpAssetsPath . '/ErrorPageTest');
|
|
||||||
|
|
||||||
$this->orig['Director_environmenttype'] = Director::get_environment_type();
|
|
||||||
Director::set_environment_type('live');
|
|
||||||
}
|
|
||||||
|
|
||||||
function tearDown() {
|
|
||||||
parent::tearDown();
|
|
||||||
|
|
||||||
ErrorPage::set_static_filepath($this->orig['ErrorPage_staticfilepath']);
|
|
||||||
Director::set_environment_type($this->orig['Director_environmenttype']);
|
|
||||||
|
|
||||||
Filesystem::removeFolder($this->tmpAssetsPath . '/ErrorPageTest');
|
|
||||||
Filesystem::removeFolder($this->tmpAssetsPath);
|
|
||||||
}
|
|
||||||
|
|
||||||
function test404ErrorPage() {
|
|
||||||
$page = $this->objFromFixture('ErrorPage', '404');
|
|
||||||
// ensure that the errorpage exists as a physical file
|
|
||||||
$page->publish('Stage', 'Live');
|
|
||||||
|
|
||||||
$response = $this->get('nonexistent-page');
|
|
||||||
|
|
||||||
/* We have body text from the error page */
|
|
||||||
$this->assertNotNull($response->getBody(), 'We have body text from the error page');
|
|
||||||
|
|
||||||
/* Status code of the SS_HTTPResponse for error page is "404" */
|
|
||||||
$this->assertEquals($response->getStatusCode(), '404', 'Status code of the SS_HTTPResponse for error page is "404"');
|
|
||||||
|
|
||||||
/* Status message of the SS_HTTPResponse for error page is "Not Found" */
|
|
||||||
$this->assertEquals($response->getStatusDescription(), 'Not Found', 'Status message of the HTTResponse for error page is "Not found"');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testBehaviourOfShowInMenuAndShowInSearchFlags() {
|
|
||||||
$page = $this->objFromFixture('ErrorPage', '404');
|
|
||||||
|
|
||||||
/* Don't show the error page in the menus */
|
|
||||||
$this->assertEquals($page->ShowInMenus, 0, 'Don\'t show the error page in the menus');
|
|
||||||
|
|
||||||
/* Don't show the error page in the search */
|
|
||||||
$this->assertEquals($page->ShowInSearch, 0, 'Don\'t show the error page in search');
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
?>
|
|
@ -1,6 +0,0 @@
|
|||||||
ErrorPage:
|
|
||||||
404:
|
|
||||||
Title: Page Not Found
|
|
||||||
URLSegment: page-not-found
|
|
||||||
Content: My error page body
|
|
||||||
ErrorCode: 404
|
|
@ -1,53 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
class RedirectorPageTest extends FunctionalTest {
|
|
||||||
static $fixture_file = 'sapphire/tests/RedirectorPageTest.yml';
|
|
||||||
static $use_draft_site = true;
|
|
||||||
|
|
||||||
function testGoodRedirectors() {
|
|
||||||
/* For good redirectors, the final destination URL will be returned */
|
|
||||||
$this->assertEquals("http://www.google.com", $this->objFromFixture('RedirectorPage','goodexternal')->Link());
|
|
||||||
$this->assertEquals(Director::baseURL() . "redirection-dest/", $this->objFromFixture('RedirectorPage','goodinternal')->redirectionLink());
|
|
||||||
$this->assertEquals(Director::baseURL() . "redirection-dest/", $this->objFromFixture('RedirectorPage','goodinternal')->Link());
|
|
||||||
}
|
|
||||||
|
|
||||||
function testEmptyRedirectors() {
|
|
||||||
/* If a redirector page is misconfigured, then its link method will just return the usual URLSegment-generated value */
|
|
||||||
$page1 = $this->objFromFixture('RedirectorPage','badexternal');
|
|
||||||
$this->assertEquals(Director::baseURL() . 'bad-external/', $page1->Link());
|
|
||||||
|
|
||||||
/* An error message will be shown if you visit it */
|
|
||||||
$content = $this->get(Director::makeRelative($page1->Link()))->getBody();
|
|
||||||
$this->assertContains('message-setupWithoutRedirect', $content);
|
|
||||||
|
|
||||||
/* This also applies for internal links */
|
|
||||||
$page2 = $this->objFromFixture('RedirectorPage','badinternal');
|
|
||||||
$this->assertEquals(Director::baseURL() . 'bad-internal/', $page2->Link());
|
|
||||||
$content = $this->get(Director::makeRelative($page2->Link()))->getBody();
|
|
||||||
$this->assertContains('message-setupWithoutRedirect', $content);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testReflexiveAndTransitiveInternalRedirectors() {
|
|
||||||
/* Reflexive redirectors are those that point to themselves. They should behave the same as an empty redirector */
|
|
||||||
$page = $this->objFromFixture('RedirectorPage','reflexive');
|
|
||||||
$this->assertEquals(Director::baseURL() . 'reflexive/', $page->Link());
|
|
||||||
$content = $this->get(Director::makeRelative($page->Link()))->getBody();
|
|
||||||
$this->assertContains('message-setupWithoutRedirect', $content);
|
|
||||||
|
|
||||||
/* Transitive redirectors are those that point to another redirector page. They should send people to the URLSegment
|
|
||||||
* of the destination page - the middle-stop, so to speak. That should redirect to the final destination */
|
|
||||||
$page = $this->objFromFixture('RedirectorPage','transitive');
|
|
||||||
$this->assertEquals(Director::baseURL() . 'good-internal/', $page->Link());
|
|
||||||
|
|
||||||
$this->autoFollowRedirection = false;
|
|
||||||
$response = $this->get(Director::makeRelative($page->Link()));
|
|
||||||
$this->assertEquals(Director::baseURL() . "redirection-dest/", $response->getHeader("Location"));
|
|
||||||
}
|
|
||||||
|
|
||||||
function testExternalURLGetsPrefixIfNotSet() {
|
|
||||||
$page = $this->objFromFixture('RedirectorPage', 'externalnoprefix');
|
|
||||||
$this->assertEquals($page->ExternalURL, 'http://google.com', 'onBeforeWrite has prefixed with http');
|
|
||||||
$page->write();
|
|
||||||
$this->assertEquals($page->ExternalURL, 'http://google.com', 'onBeforeWrite will not double prefix if written again!');
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,39 +0,0 @@
|
|||||||
Page:
|
|
||||||
dest:
|
|
||||||
Title: Redirection Dest
|
|
||||||
URLSegment: redirection-dest
|
|
||||||
|
|
||||||
RedirectorPage:
|
|
||||||
goodexternal:
|
|
||||||
Title: Good External
|
|
||||||
URLSegment: good-external
|
|
||||||
RedirectionType: External
|
|
||||||
ExternalURL: http://www.google.com
|
|
||||||
goodinternal:
|
|
||||||
Title: Good Internal
|
|
||||||
URLSegment: good-internal
|
|
||||||
RedirectionType: Internal
|
|
||||||
LinkTo: =>Page.dest
|
|
||||||
badexternal:
|
|
||||||
Title: Bad External
|
|
||||||
RedirectionType: External
|
|
||||||
URLSegment: bad-external
|
|
||||||
externalnoprefix:
|
|
||||||
Title: External no prefix
|
|
||||||
RedirectionType: External
|
|
||||||
URLSegment: external-no-prefix
|
|
||||||
ExternalURL: google.com
|
|
||||||
badinternal:
|
|
||||||
Title: Bad Internal
|
|
||||||
RedirectionType: Internal
|
|
||||||
URLSegment: bad-internal
|
|
||||||
reflexive:
|
|
||||||
Title: Reflexive
|
|
||||||
RedirectionType: Internal
|
|
||||||
LinkTo: =>RedirectorPage.reflexive
|
|
||||||
URLSegment: reflexive
|
|
||||||
transitive:
|
|
||||||
Title: Transitive
|
|
||||||
RedirectionType: Internal
|
|
||||||
LinkTo: =>RedirectorPage.goodinternal
|
|
||||||
URLSegment: transitive
|
|
@ -1,186 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* Possible actions:
|
|
||||||
* - action_save
|
|
||||||
* - action_publish
|
|
||||||
* - action_unpublish
|
|
||||||
* - action_delete
|
|
||||||
* - action_deletefromlive
|
|
||||||
* - action_rollback
|
|
||||||
* - action_revert
|
|
||||||
*
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage tests
|
|
||||||
*/
|
|
||||||
class SiteTreeActionsTest extends FunctionalTest {
|
|
||||||
|
|
||||||
static $fixture_file = 'sapphire/tests/SiteTreeActionsTest.yml';
|
|
||||||
|
|
||||||
static function set_up_once() {
|
|
||||||
SiteTreeTest::set_up_once();
|
|
||||||
|
|
||||||
parent::set_up_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
static function tear_down_once() {
|
|
||||||
SiteTreeTest::tear_down_once();
|
|
||||||
|
|
||||||
parent::tear_down_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
function testActionsReadonly() {
|
|
||||||
if(class_exists('SiteTreeCMSWorkflow')) return true;
|
|
||||||
|
|
||||||
$readonlyEditor = $this->objFromFixture('Member', 'cmsreadonlyeditor');
|
|
||||||
$this->session()->inst_set('loggedInAs', $readonlyEditor->ID);
|
|
||||||
|
|
||||||
$page = new SiteTreeActionsTest_Page();
|
|
||||||
$page->CanEditType = 'LoggedInUsers';
|
|
||||||
$page->write();
|
|
||||||
$page->doPublish();
|
|
||||||
|
|
||||||
$actionsArr = $page->getCMSActions()->column('Name');
|
|
||||||
|
|
||||||
$this->assertNotContains('action_save',$actionsArr);
|
|
||||||
$this->assertNotContains('action_publish',$actionsArr);
|
|
||||||
$this->assertNotContains('action_unpublish',$actionsArr);
|
|
||||||
$this->assertNotContains('action_delete',$actionsArr);
|
|
||||||
$this->assertNotContains('action_deletefromlive',$actionsArr);
|
|
||||||
$this->assertNotContains('action_rollback',$actionsArr);
|
|
||||||
$this->assertNotContains('action_revert',$actionsArr);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testActionsNoDeletePublishedRecord() {
|
|
||||||
if(class_exists('SiteTreeCMSWorkflow')) return true;
|
|
||||||
|
|
||||||
$this->logInWithPermission('ADMIN');
|
|
||||||
|
|
||||||
$page = new SiteTreeActionsTest_Page();
|
|
||||||
$page->CanEditType = 'LoggedInUsers';
|
|
||||||
$page->write();
|
|
||||||
$pageID = $page->ID;
|
|
||||||
$page->doPublish();
|
|
||||||
$page->deleteFromStage('Stage');
|
|
||||||
|
|
||||||
// Get the live version of the page
|
|
||||||
$page = Versioned::get_one_by_stage("SiteTree", "Live", "\"SiteTree\".\"ID\" = $pageID");
|
|
||||||
$this->assertType("SiteTree", $page);
|
|
||||||
|
|
||||||
// Check that someone without the right permission can't delete the page
|
|
||||||
$editor = $this->objFromFixture('Member', 'cmsnodeleteeditor');
|
|
||||||
$this->session()->inst_set('loggedInAs', $editor->ID);
|
|
||||||
|
|
||||||
$actionsArr = $page->getCMSActions()->column('Name');
|
|
||||||
$this->assertNotContains('action_deletefromlive',$actionsArr);
|
|
||||||
|
|
||||||
// Check that someone with the right permission can delete the page
|
|
||||||
$this->objFromFixture('Member', 'cmseditor')->logIn();
|
|
||||||
$actionsArr = $page->getCMSActions()->column('Name');
|
|
||||||
$this->assertContains('action_deletefromlive',$actionsArr);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testActionsPublishedRecord() {
|
|
||||||
if(class_exists('SiteTreeCMSWorkflow')) return true;
|
|
||||||
|
|
||||||
$author = $this->objFromFixture('Member', 'cmseditor');
|
|
||||||
$this->session()->inst_set('loggedInAs', $author->ID);
|
|
||||||
|
|
||||||
$page = new Page();
|
|
||||||
$page->CanEditType = 'LoggedInUsers';
|
|
||||||
$page->write();
|
|
||||||
$page->doPublish();
|
|
||||||
|
|
||||||
$actionsArr = $page->getCMSActions()->column('Name');
|
|
||||||
|
|
||||||
$this->assertContains('action_save',$actionsArr);
|
|
||||||
$this->assertContains('action_publish',$actionsArr);
|
|
||||||
$this->assertContains('action_unpublish',$actionsArr);
|
|
||||||
$this->assertContains('action_delete',$actionsArr);
|
|
||||||
$this->assertNotContains('action_deletefromlive',$actionsArr);
|
|
||||||
$this->assertNotContains('action_rollback',$actionsArr);
|
|
||||||
$this->assertNotContains('action_revert',$actionsArr);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testActionsDeletedFromStageRecord() {
|
|
||||||
if(class_exists('SiteTreeCMSWorkflow')) return true;
|
|
||||||
|
|
||||||
$author = $this->objFromFixture('Member', 'cmseditor');
|
|
||||||
$this->session()->inst_set('loggedInAs', $author->ID);
|
|
||||||
|
|
||||||
$page = new Page();
|
|
||||||
$page->CanEditType = 'LoggedInUsers';
|
|
||||||
$page->write();
|
|
||||||
$pageID = $page->ID;
|
|
||||||
$page->doPublish();
|
|
||||||
$page->deleteFromStage('Stage');
|
|
||||||
|
|
||||||
// Get the live version of the page
|
|
||||||
$page = Versioned::get_one_by_stage("SiteTree", "Live", "\"SiteTree\".\"ID\" = $pageID");
|
|
||||||
$this->assertType('SiteTree', $page);
|
|
||||||
|
|
||||||
$actionsArr = $page->getCMSActions()->column('Name');
|
|
||||||
|
|
||||||
$this->assertNotContains('action_save',$actionsArr);
|
|
||||||
$this->assertNotContains('action_publish',$actionsArr);
|
|
||||||
$this->assertNotContains('action_unpublish',$actionsArr);
|
|
||||||
$this->assertNotContains('action_delete',$actionsArr);
|
|
||||||
$this->assertContains('action_deletefromlive',$actionsArr);
|
|
||||||
$this->assertNotContains('action_rollback',$actionsArr);
|
|
||||||
$this->assertContains('action_revert',$actionsArr);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testActionsChangedOnStageRecord() {
|
|
||||||
if(class_exists('SiteTreeCMSWorkflow')) return true;
|
|
||||||
|
|
||||||
$author = $this->objFromFixture('Member', 'cmseditor');
|
|
||||||
$this->session()->inst_set('loggedInAs', $author->ID);
|
|
||||||
|
|
||||||
$page = new Page();
|
|
||||||
$page->CanEditType = 'LoggedInUsers';
|
|
||||||
$page->write();
|
|
||||||
$page->doPublish();
|
|
||||||
$page->Content = 'Changed on Stage';
|
|
||||||
$page->write();
|
|
||||||
$page->flushCache();
|
|
||||||
|
|
||||||
$actionsArr = $page->getCMSActions()->column('Name');
|
|
||||||
|
|
||||||
$this->assertContains('action_save',$actionsArr);
|
|
||||||
$this->assertContains('action_publish',$actionsArr);
|
|
||||||
$this->assertContains('action_unpublish',$actionsArr);
|
|
||||||
$this->assertContains('action_delete',$actionsArr);
|
|
||||||
$this->assertNotContains('action_deletefromlive',$actionsArr);
|
|
||||||
$this->assertContains('action_rollback',$actionsArr);
|
|
||||||
$this->assertNotContains('action_revert',$actionsArr);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testActionsViewingOldVersion() {
|
|
||||||
$p = new Page();
|
|
||||||
$p->Content = 'test page first version';
|
|
||||||
$p->write();
|
|
||||||
$p->Content = 'new content';
|
|
||||||
$p->write();
|
|
||||||
|
|
||||||
// Looking at the old version, the ability to rollback to that version is available
|
|
||||||
$version = DB::query('SELECT "Version" FROM "SiteTree_versions" WHERE "Content" = \'test page first version\'')->value();
|
|
||||||
$old = Versioned::get_version('Page', $p->ID, $version);
|
|
||||||
$actions = $old->getCMSActions()->column('Name');
|
|
||||||
$this->assertNotContains('action_save', $actions);
|
|
||||||
$this->assertNotContains('action_publish', $actions);
|
|
||||||
$this->assertNotContains('action_unpublish', $actions);
|
|
||||||
$this->assertNotContains('action_delete', $actions);
|
|
||||||
$this->assertContains('action_email', $actions);
|
|
||||||
$this->assertContains('action_rollback', $actions);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
class SiteTreeActionsTest_Page extends Page implements TestOnly {
|
|
||||||
function canEdit($member = null) {
|
|
||||||
return Permission::checkMember($member, 'SiteTreeActionsTest_Page_CANEDIT');
|
|
||||||
}
|
|
||||||
|
|
||||||
function canDelete($member = null) {
|
|
||||||
return Permission::checkMember($member, 'SiteTreeActionsTest_Page_CANDELETE');
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,33 +0,0 @@
|
|||||||
Permission:
|
|
||||||
cmsmain1:
|
|
||||||
Code: CMS_ACCESS_CMSMain
|
|
||||||
cmsmain2:
|
|
||||||
Code: CMS_ACCESS_CMSMain
|
|
||||||
cmsmain3:
|
|
||||||
Code: CMS_ACCESS_CMSMain
|
|
||||||
candelete:
|
|
||||||
Code: SiteTreeActionsTest_Page_CANDELETE
|
|
||||||
canedit1:
|
|
||||||
Code: SiteTreeActionsTest_Page_CANEDIT
|
|
||||||
canedit2:
|
|
||||||
Code: SiteTreeActionsTest_Page_CANEDIT
|
|
||||||
Group:
|
|
||||||
cmseditors:
|
|
||||||
Title: CMS Editors
|
|
||||||
Permissions: =>Permission.cmsmain1,=>Permission.canedit1,=>Permission.candelete
|
|
||||||
cmsreadonly:
|
|
||||||
Title: CMS Readonly
|
|
||||||
Permissions: =>Permission.cmsmain2
|
|
||||||
cmsnodelete:
|
|
||||||
Title: CMS No Delete
|
|
||||||
Permissions: =>Permission.cmsmain3,=>Permission.canedit2
|
|
||||||
Member:
|
|
||||||
cmseditor:
|
|
||||||
Email: cmseditor@test.com
|
|
||||||
Groups: =>Group.cmseditors
|
|
||||||
cmsreadonlyeditor:
|
|
||||||
Email: cmsreadonlyeditor@test.com
|
|
||||||
Groups: =>Group.cmsreadonly
|
|
||||||
cmsnodeleteeditor:
|
|
||||||
Email: cmsnodeleteeditor@test.com
|
|
||||||
Groups: =>Group.cmsnodelete
|
|
@ -1,258 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
class SiteTreeBacklinksTest extends SapphireTest {
|
|
||||||
static $fixture_file = "sapphire/tests/SiteTreeBacklinksTest.yml";
|
|
||||||
|
|
||||||
protected $requiredExtensions = array(
|
|
||||||
'SiteTree' => array('SiteTreeBacklinksTest_DOD'),
|
|
||||||
);
|
|
||||||
|
|
||||||
static function set_up_once() {
|
|
||||||
SiteTreeTest::set_up_once();
|
|
||||||
|
|
||||||
parent::set_up_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
static function tear_down_once() {
|
|
||||||
SiteTreeTest::tear_down_once();
|
|
||||||
|
|
||||||
parent::tear_down_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
function setUp() {
|
|
||||||
parent::setUp();
|
|
||||||
|
|
||||||
// Log in as admin so that we don't run into permission issues. That's not what we're
|
|
||||||
// testing here.
|
|
||||||
$this->logInWithPermission('ADMIN');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testSavingPageWithLinkAddsBacklink() {
|
|
||||||
// load page 1
|
|
||||||
$page1 = $this->objFromFixture('Page', 'page1');
|
|
||||||
|
|
||||||
// assert backlink to page 2 doesn't exist
|
|
||||||
$page2 = $this->objFromFixture('Page', 'page2');
|
|
||||||
$this->assertFalse($page1->BackLinkTracking()->containsIDs(array($page2->ID)), 'Assert backlink to page 2 doesn\'t exist');
|
|
||||||
|
|
||||||
// add hyperlink to page 1 on page 2
|
|
||||||
$page2->Content .= '<p><a href="[sitetree_link id='.$page1->ID.']">Testing page 1 link</a></p>';
|
|
||||||
$page2->write();
|
|
||||||
|
|
||||||
// load page 1
|
|
||||||
$page1 = $this->objFromFixture('Page', 'page1');
|
|
||||||
|
|
||||||
// assert backlink to page 2 exists
|
|
||||||
$this->assertTrue($page1->BackLinkTracking()->containsIDs(array($page2->ID)), 'Assert backlink to page 2 exists');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testRemovingLinkFromPageRemovesBacklink() {
|
|
||||||
// load page 1
|
|
||||||
$page1 = $this->objFromFixture('Page', 'page1');
|
|
||||||
|
|
||||||
// assert backlink to page 3 exits
|
|
||||||
$page3 = $this->objFromFixture('Page', 'page3');
|
|
||||||
$this->assertTrue($page1->BackLinkTracking()->containsIDs(array($page3->ID)), 'Assert backlink to page 3 exists');
|
|
||||||
|
|
||||||
// remove hyperlink to page 1
|
|
||||||
$page3->Content = '<p>No links anymore!</p>';
|
|
||||||
$page3->write();
|
|
||||||
|
|
||||||
// load page 1
|
|
||||||
$page1 = $this->objFromFixture('Page', 'page1');
|
|
||||||
|
|
||||||
// assert backlink to page 3 exists
|
|
||||||
$this->assertFalse($page1->BackLinkTracking()->containsIDs(array($page3->ID)), 'Assert backlink to page 3 doesn\'t exist');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testChangingUrlOnDraftSiteRewritesLink() {
|
|
||||||
// load page 1
|
|
||||||
$page1 = $this->objFromFixture('Page', 'page1');
|
|
||||||
|
|
||||||
// assert backlink to page 3 exists
|
|
||||||
$page3 = $this->objFromFixture('Page', 'page3');
|
|
||||||
$this->assertTrue($page1->BackLinkTracking()->containsIDs(array($page3->ID)), 'Assert backlink to page 3 exists');
|
|
||||||
|
|
||||||
// assert hyperlink to page 1's current url exists on page 3
|
|
||||||
$links = HTTP::getLinksIn($page3->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'page1/', $links, 'Assert hyperlink to page 1\'s current url exists on page 3');
|
|
||||||
|
|
||||||
// change url of page 1
|
|
||||||
$page1->URLSegment = 'new-url-segment';
|
|
||||||
$page1->write();
|
|
||||||
|
|
||||||
// load page 3
|
|
||||||
$page3 = $this->objFromFixture('Page', 'page3');
|
|
||||||
|
|
||||||
// assert hyperlink to page 1's new url exists
|
|
||||||
$links = HTTP::getLinksIn($page3->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'new-url-segment/', $links, 'Assert hyperlink to page 1\'s new url exists on page 3');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testChangingUrlOnLiveSiteRewritesLink() {
|
|
||||||
// publish page 1 & 3
|
|
||||||
$page1 = $this->objFromFixture('Page', 'page1');
|
|
||||||
$page3 = $this->objFromFixture('Page', 'page3');
|
|
||||||
$this->assertTrue($page1->doPublish());
|
|
||||||
$this->assertTrue($page3->doPublish());
|
|
||||||
|
|
||||||
// load pages from live
|
|
||||||
$page1live = Versioned::get_one_by_stage('Page', 'Live', '"SiteTree"."ID" = ' . $page1->ID);
|
|
||||||
$page3live = Versioned::get_one_by_stage('Page', 'Live', '"SiteTree"."ID" = ' . $page3->ID);
|
|
||||||
|
|
||||||
// assert backlink to page 3 exists
|
|
||||||
$this->assertTrue($page1live->BackLinkTracking()->containsIDs(array($page3live->ID)), 'Assert backlink to page 3 exists');
|
|
||||||
|
|
||||||
// assert hyperlink to page 1's current url exists on page 3
|
|
||||||
$links = HTTP::getLinksIn($page3live->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'page1/', $links, 'Assert hyperlink to page 1\'s current url exists on page 3');
|
|
||||||
|
|
||||||
// change url of page 1
|
|
||||||
$page1live->URLSegment = 'new-url-segment';
|
|
||||||
$page1live->writeToStage('Live');
|
|
||||||
|
|
||||||
// load page 3 from live
|
|
||||||
$page3live = Versioned::get_one_by_stage('Page', 'Live', '"SiteTree"."ID" = ' . $page3->ID);
|
|
||||||
|
|
||||||
// assert hyperlink to page 1's new url exists
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
$links = HTTP::getLinksIn($page3live->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'new-url-segment/', $links, 'Assert hyperlink to page 1\'s new url exists on page 3');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testPublishingPageWithModifiedUrlRewritesLink() {
|
|
||||||
// publish page 1 & 3
|
|
||||||
$page1 = $this->objFromFixture('Page', 'page1');
|
|
||||||
$page3 = $this->objFromFixture('Page', 'page3');
|
|
||||||
|
|
||||||
$this->assertTrue($page1->doPublish());
|
|
||||||
$this->assertTrue($page3->doPublish());
|
|
||||||
|
|
||||||
// load page 3 from live
|
|
||||||
$page3live = Versioned::get_one_by_stage('Page', 'Live', '"SiteTree"."ID" = ' . $page3->ID);
|
|
||||||
|
|
||||||
// assert hyperlink to page 1's current url exists
|
|
||||||
$links = HTTP::getLinksIn($page3live->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'page1/', $links, 'Assert hyperlink to page 1\'s current url exists on page 3');
|
|
||||||
|
|
||||||
// rename url of page 1 on stage
|
|
||||||
$page1->URLSegment = 'new-url-segment';
|
|
||||||
$page1->write();
|
|
||||||
|
|
||||||
// assert hyperlink to page 1's current publish url exists
|
|
||||||
$page3live = Versioned::get_one_by_stage('Page', 'Live', '"SiteTree"."ID" = ' . $page3->ID);
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
$links = HTTP::getLinksIn($page3live->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'page1/', $links, 'Assert hyperlink to page 1\'s current published url exists on page 3');
|
|
||||||
|
|
||||||
|
|
||||||
// publish page 1
|
|
||||||
$this->assertTrue($page1->doPublish());
|
|
||||||
|
|
||||||
// assert hyperlink to page 1's new published url exists
|
|
||||||
$page3live = Versioned::get_one_by_stage('Page', 'Live', '"SiteTree"."ID" = ' . $page3->ID);
|
|
||||||
$links = HTTP::getLinksIn($page3live->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'new-url-segment/', $links, 'Assert hyperlink to page 1\'s new published url exists on page 3');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testPublishingPageWithModifiedLinksRewritesLinks() {
|
|
||||||
// publish page 1 & 3
|
|
||||||
$page1 = $this->objFromFixture('Page', 'page1');
|
|
||||||
$page3 = $this->objFromFixture('Page', 'page3');
|
|
||||||
$this->assertTrue($page1->doPublish());
|
|
||||||
$this->assertTrue($page3->doPublish());
|
|
||||||
|
|
||||||
// assert hyperlink to page 1's current url exists
|
|
||||||
$links = HTTP::getLinksIn($page3->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'page1/', $links, 'Assert hyperlink to page 1\'s current published url exists on page 3');
|
|
||||||
|
|
||||||
// change page 1 url on draft
|
|
||||||
$page1->URLSegment = 'new-url-segment';
|
|
||||||
|
|
||||||
// save page 1
|
|
||||||
$page1->write();
|
|
||||||
|
|
||||||
// assert page 3 on draft contains new page 1 url
|
|
||||||
$page3 = $this->objFromFixture('Page', 'page3');
|
|
||||||
$links = HTTP::getLinksIn($page3->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'new-url-segment/', $links, 'Assert hyperlink to page 1\'s current draft url exists on page 3');
|
|
||||||
|
|
||||||
// publish page 3
|
|
||||||
$this->assertTrue($page3->doPublish());
|
|
||||||
|
|
||||||
// assert page 3 on published site contains old page 1 url
|
|
||||||
$page3live = Versioned::get_one_by_stage('Page', 'Live', '"SiteTree"."ID" = ' . $page3->ID);
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
$links = HTTP::getLinksIn($page3live->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'page1/', $links, 'Assert hyperlink to page 1\'s current published url exists on page 3');
|
|
||||||
|
|
||||||
// publish page 1
|
|
||||||
$this->assertTrue($page1->doPublish());
|
|
||||||
|
|
||||||
// assert page 3 on published site contains new page 1 url
|
|
||||||
$page3live = Versioned::get_one_by_stage('Page', 'Live', '"SiteTree"."ID" = ' . $page3->ID);
|
|
||||||
$links = HTTP::getLinksIn($page3live->obj('Content')->forTemplate());
|
|
||||||
$this->assertContains(Director::baseURL().'new-url-segment/', $links, 'Assert hyperlink to page 1\'s current published url exists on page 3');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testLinkTrackingOnExtraContentFields() {
|
|
||||||
$page1 = $this->objFromFixture('Page', 'page1');
|
|
||||||
$page2 = $this->objFromFixture('Page', 'page2');
|
|
||||||
$page1->doPublish();
|
|
||||||
$page2->doPublish();
|
|
||||||
|
|
||||||
// assert backlink to page 2 doesn't exist
|
|
||||||
$this->assertFalse($page1->BackLinkTracking()->containsIDs(array($page2->ID)), 'Assert backlink to page 2 doesn\'t exist');
|
|
||||||
|
|
||||||
// add hyperlink to page 1 on page 2
|
|
||||||
$page2->ExtraContent .= '<p><a href="[sitetree_link id='.$page1->ID.']">Testing page 1 link</a></p>';
|
|
||||||
$page2->write();
|
|
||||||
$page2->doPublish();
|
|
||||||
|
|
||||||
// assert backlink to page 2 exists
|
|
||||||
$this->assertTrue($page1->BackLinkTracking()->containsIDs(array($page2->ID)), 'Assert backlink to page 2 exists');
|
|
||||||
|
|
||||||
// update page1 url
|
|
||||||
$page1 = $this->objFromFixture('Page', 'page1');
|
|
||||||
$page1->URLSegment = "page1-new-url";
|
|
||||||
$page1->write();
|
|
||||||
|
|
||||||
// confirm that draft link on page2 has been rewritten
|
|
||||||
$page2 = $this->objFromFixture('Page', 'page2');
|
|
||||||
$this->assertEquals('<p><a href="'.Director::baseURL().'page1-new-url/">Testing page 1 link</a></p>', $page2->obj('ExtraContent')->forTemplate());
|
|
||||||
|
|
||||||
// confirm that published link hasn't
|
|
||||||
$page2Live = Versioned::get_one_by_stage("Page", "Live", "\"SiteTree\".\"ID\" = $page2->ID");
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
$this->assertEquals('<p><a href="'.Director::baseURL().'page1/">Testing page 1 link</a></p>', $page2Live->obj('ExtraContent')->forTemplate());
|
|
||||||
|
|
||||||
// publish page1 and confirm that the link on the published page2 has now been updated
|
|
||||||
$page1->doPublish();
|
|
||||||
$page2Live = Versioned::get_one_by_stage("Page", "Live", "\"SiteTree\".\"ID\" = $page2->ID");
|
|
||||||
$this->assertEquals('<p><a href="'.Director::baseURL().'page1-new-url/">Testing page 1 link</a></p>', $page2Live->obj('ExtraContent')->forTemplate());
|
|
||||||
|
|
||||||
|
|
||||||
// remove hyperlink to page 1
|
|
||||||
$page2->ExtraContent = '<p>No links anymore!</p>';
|
|
||||||
$page2->write();
|
|
||||||
|
|
||||||
// assert backlink to page 2 no longer exists
|
|
||||||
$this->assertFalse($page1->BackLinkTracking()->containsIDs(array($page2->ID)), 'Assert backlink to page 2 has been removed');
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
class SiteTreeBacklinksTest_DOD extends DataObjectDecorator implements TestOnly {
|
|
||||||
function extraStatics() {
|
|
||||||
return array(
|
|
||||||
'db' => array(
|
|
||||||
'ExtraContent' => 'HTMLText',
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateCMSFields(&$fields) {
|
|
||||||
$fields->addFieldToTab("Root.Content.Main", new HTMLEditorField("ExtraContent"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
?>
|
|
@ -1,17 +0,0 @@
|
|||||||
Page:
|
|
||||||
page1:
|
|
||||||
ID: 1
|
|
||||||
Title: page1
|
|
||||||
URLSegment: page1
|
|
||||||
|
|
||||||
page2:
|
|
||||||
Title: page2
|
|
||||||
URLSegment: page2
|
|
||||||
|
|
||||||
page3:
|
|
||||||
Title: page3
|
|
||||||
URLSegment: page3
|
|
||||||
Content: '<p><a href="[sitetree_link id=1]">Testing page 1 link</a></p>'
|
|
||||||
LinkTracking: =>Page.page1
|
|
||||||
|
|
||||||
|
|
@ -1,313 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage tests
|
|
||||||
*/
|
|
||||||
class SiteTreeBrokenLinksTest extends SapphireTest {
|
|
||||||
static $fixture_file = 'sapphire/tests/SiteTreeBrokenLinksTest.yml';
|
|
||||||
|
|
||||||
static function set_up_once() {
|
|
||||||
SiteTreeTest::set_up_once();
|
|
||||||
|
|
||||||
parent::set_up_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
static function tear_down_once() {
|
|
||||||
SiteTreeTest::tear_down_once();
|
|
||||||
|
|
||||||
parent::tear_down_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
function testBrokenLinksBetweenPages() {
|
|
||||||
$obj = $this->objFromFixture('Page','content');
|
|
||||||
|
|
||||||
$obj->Content = '<a href="[sitetree_link id=3423423]">this is a broken link</a>';
|
|
||||||
$obj->syncLinkTracking();
|
|
||||||
$this->assertTrue($obj->HasBrokenLink, 'Page has a broken link');
|
|
||||||
|
|
||||||
$obj->Content = '<a href="[sitetree_link id=' . $this->idFromFixture('Page','about') .']">this is not a broken link</a>';
|
|
||||||
$obj->syncLinkTracking();
|
|
||||||
$this->assertFalse($obj->HasBrokenLink, 'Page does NOT have a broken link');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testBrokenVirtualPages() {
|
|
||||||
$obj = $this->objFromFixture('Page','content');
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
|
|
||||||
$vp->CopyContentFromID = $obj->ID;
|
|
||||||
$vp->syncLinkTracking();
|
|
||||||
$this->assertFalse($vp->HasBrokenLink, 'Working virtual page is NOT marked as broken');
|
|
||||||
|
|
||||||
$vp->CopyContentFromID = 12345678;
|
|
||||||
$vp->syncLinkTracking();
|
|
||||||
$this->assertTrue($vp->HasBrokenLink, 'Broken virtual page IS marked as such');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testBrokenInternalRedirectorPages() {
|
|
||||||
$obj = $this->objFromFixture('Page','content');
|
|
||||||
$rp = new RedirectorPage();
|
|
||||||
|
|
||||||
$rp->RedirectionType = 'Internal';
|
|
||||||
|
|
||||||
$rp->LinkToID = $obj->ID;
|
|
||||||
$rp->syncLinkTracking();
|
|
||||||
$this->assertFalse($rp->HasBrokenLink, 'Working redirector page is NOT marked as broken');
|
|
||||||
|
|
||||||
$rp->LinkToID = 12345678;
|
|
||||||
$rp->syncLinkTracking();
|
|
||||||
$this->assertTrue($rp->HasBrokenLink, 'Broken redirector page IS marked as such');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testBrokenAssetLinks() {
|
|
||||||
$obj = $this->objFromFixture('Page','content');
|
|
||||||
|
|
||||||
$obj->Content = '<a href="assets/nofilehere.pdf">this is a broken link to a pdf file</a>';
|
|
||||||
$obj->syncLinkTracking();
|
|
||||||
$this->assertTrue($obj->HasBrokenFile, 'Page has a broken file');
|
|
||||||
|
|
||||||
$obj->Content = '<a href="assets/privacypolicy.pdf">this is not a broken file link</a>';
|
|
||||||
$obj->syncLinkTracking();
|
|
||||||
$this->assertFalse($obj->HasBrokenFile, 'Page does NOT have a broken file');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testDeletingFileMarksBackedPagesAsBroken() {
|
|
||||||
// Test entry
|
|
||||||
$file = new File();
|
|
||||||
$file->Filename = 'test-file.pdf';
|
|
||||||
$file->write();
|
|
||||||
|
|
||||||
$obj = $this->objFromFixture('Page','content');
|
|
||||||
$obj->Content = '<a href="assets/test-file.pdf">link to a pdf file</a>';
|
|
||||||
$obj->write();
|
|
||||||
$this->assertTrue($obj->doPublish());
|
|
||||||
// Confirm that it isn't marked as broken to begin with
|
|
||||||
$obj->flushCache();
|
|
||||||
$obj = DataObject::get_by_id("SiteTree", $obj->ID);
|
|
||||||
$this->assertEquals(0, $obj->HasBrokenFile);
|
|
||||||
|
|
||||||
$liveObj = Versioned::get_one_by_stage("SiteTree", "Live","\"SiteTree\".\"ID\" = $obj->ID");
|
|
||||||
$this->assertEquals(0, $liveObj->HasBrokenFile);
|
|
||||||
|
|
||||||
// Delete the file
|
|
||||||
$file->delete();
|
|
||||||
|
|
||||||
// Confirm that it is marked as broken in both stage and live
|
|
||||||
$obj->flushCache();
|
|
||||||
$obj = DataObject::get_by_id("SiteTree", $obj->ID);
|
|
||||||
$this->assertEquals(1, $obj->HasBrokenFile);
|
|
||||||
|
|
||||||
$liveObj = Versioned::get_one_by_stage("SiteTree", "Live", "\"SiteTree\".\"ID\" = $obj->ID");
|
|
||||||
$this->assertEquals(1, $liveObj->HasBrokenFile);
|
|
||||||
}
|
|
||||||
function testDeletingMarksBackLinkedPagesAsBroken() {
|
|
||||||
$this->logInWithPermission('ADMIN');
|
|
||||||
|
|
||||||
// Set up two published pages with a link from content -> about
|
|
||||||
$linkDest = $this->objFromFixture('Page','about');
|
|
||||||
$linkDest->doPublish();
|
|
||||||
|
|
||||||
$linkSrc = $this->objFromFixture('Page','content');
|
|
||||||
$linkSrc->Content = "<p><a href=\"[sitetree_link id=$linkDest->ID]\">about us</a></p>";
|
|
||||||
$linkSrc->write();
|
|
||||||
|
|
||||||
$linkSrc->doPublish();
|
|
||||||
|
|
||||||
// Confirm no broken link
|
|
||||||
$this->assertEquals(0, (int)$linkSrc->HasBrokenLink);
|
|
||||||
$this->assertEquals(0, DB::query("SELECT \"HasBrokenLink\" FROM \"SiteTree_Live\"
|
|
||||||
WHERE \"ID\" = $linkSrc->ID")->value());
|
|
||||||
|
|
||||||
// Delete page from draft
|
|
||||||
$linkDestID = $linkDest->ID;
|
|
||||||
$linkDest->delete();
|
|
||||||
|
|
||||||
// Confirm draft has broken link, and published doesn't
|
|
||||||
$linkSrc->flushCache();
|
|
||||||
$linkSrc = $this->objFromFixture('Page', 'content');
|
|
||||||
|
|
||||||
$this->assertEquals(1, (int)$linkSrc->HasBrokenLink);
|
|
||||||
$this->assertEquals(0, DB::query("SELECT \"HasBrokenLink\" FROM \"SiteTree_Live\"
|
|
||||||
WHERE \"ID\" = $linkSrc->ID")->value());
|
|
||||||
|
|
||||||
// Delete from live
|
|
||||||
$linkDest = Versioned::get_one_by_stage("SiteTree", "Live", "\"SiteTree\".\"ID\" = $linkDestID");
|
|
||||||
$linkDest->doDeleteFromLive();
|
|
||||||
|
|
||||||
// Confirm both draft and published have broken link
|
|
||||||
$linkSrc->flushCache();
|
|
||||||
$linkSrc = $this->objFromFixture('Page', 'content');
|
|
||||||
|
|
||||||
$this->assertEquals(1, (int)$linkSrc->HasBrokenLink);
|
|
||||||
$this->assertEquals(1, DB::query("SELECT \"HasBrokenLink\" FROM \"SiteTree_Live\"
|
|
||||||
WHERE \"ID\" = $linkSrc->ID")->value());
|
|
||||||
}
|
|
||||||
|
|
||||||
function testPublishingSourceBeforeDestHasBrokenLink() {
|
|
||||||
$this->logInWithPermission('ADMIN');
|
|
||||||
|
|
||||||
// Set up two draft pages with a link from content -> about
|
|
||||||
$linkDest = $this->objFromFixture('Page','about');
|
|
||||||
// Ensure that it's not on the published site
|
|
||||||
$linkDest->doDeleteFromLive();
|
|
||||||
|
|
||||||
$linkSrc = $this->objFromFixture('Page','content');
|
|
||||||
$linkSrc->Content = "<p><a href=\"[sitetree_link id=$linkDest->ID]\">about us</a></p>";
|
|
||||||
$linkSrc->write();
|
|
||||||
|
|
||||||
// Publish the source of the link, while the dest is still unpublished.
|
|
||||||
$linkSrc->doPublish();
|
|
||||||
|
|
||||||
// Verify that the link isn't broken on draft but is broken on published
|
|
||||||
$this->assertEquals(0, (int)$linkSrc->HasBrokenLink);
|
|
||||||
$this->assertEquals(1, DB::query("SELECT \"HasBrokenLink\" FROM \"SiteTree_Live\"
|
|
||||||
WHERE \"ID\" = $linkSrc->ID")->value());
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
function testRestoreFixesBrokenLinks() {
|
|
||||||
// Create page and virutal page
|
|
||||||
$p = new Page();
|
|
||||||
$p->Title = "source";
|
|
||||||
$p->write();
|
|
||||||
$pageID = $p->ID;
|
|
||||||
$this->assertTrue($p->doPublish());
|
|
||||||
|
|
||||||
// Content links are one kind of link to pages
|
|
||||||
$p2 = new Page();
|
|
||||||
$p2->Title = "regular link";
|
|
||||||
$p2->Content = "<a href=\"[sitetree_link id=$p->ID]\">test</a>";
|
|
||||||
$p2->write();
|
|
||||||
$this->assertTrue($p2->doPublish());
|
|
||||||
|
|
||||||
// Virtual pages are another
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
$vp->CopyContentFromID = $p->ID;
|
|
||||||
$vp->write();
|
|
||||||
|
|
||||||
// Redirector links are a third
|
|
||||||
$rp = new RedirectorPage();
|
|
||||||
$rp->Title = "redirector";
|
|
||||||
$rp->LinkType = 'Internal';
|
|
||||||
$rp->LinkToID = $p->ID;
|
|
||||||
$rp->write();
|
|
||||||
$this->assertTrue($rp->doPublish());
|
|
||||||
|
|
||||||
// Confirm that there are no broken links to begin with
|
|
||||||
$this->assertFalse($p2->HasBrokenLink);
|
|
||||||
$this->assertFalse($vp->HasBrokenLink);
|
|
||||||
$this->assertFalse($rp->HasBrokenLink);
|
|
||||||
|
|
||||||
// Unpublish the source page, confirm that the page 2 and RP has a broken link on published
|
|
||||||
$p->doUnpublish();
|
|
||||||
$p2Live = Versioned::get_one_by_stage('SiteTree', 'Live', '"SiteTree"."ID" = ' . $p2->ID);
|
|
||||||
$rpLive = Versioned::get_one_by_stage('SiteTree', 'Live', '"SiteTree"."ID" = ' . $rp->ID);
|
|
||||||
$this->assertEquals(1, $p2Live->HasBrokenLink);
|
|
||||||
$this->assertEquals(1, $rpLive->HasBrokenLink);
|
|
||||||
|
|
||||||
// Delete the source page, confirm that the VP, RP and page 2 have broken links on draft
|
|
||||||
$p->delete();
|
|
||||||
$vp->flushCache();
|
|
||||||
$vp = DataObject::get_by_id('SiteTree', $vp->ID);
|
|
||||||
$p2->flushCache();
|
|
||||||
$p2 = DataObject::get_by_id('SiteTree', $p2->ID);
|
|
||||||
$rp->flushCache();
|
|
||||||
$rp = DataObject::get_by_id('SiteTree', $rp->ID);
|
|
||||||
$this->assertEquals(1, $p2->HasBrokenLink);
|
|
||||||
$this->assertEquals(1, $vp->HasBrokenLink);
|
|
||||||
$this->assertEquals(1, $rp->HasBrokenLink);
|
|
||||||
|
|
||||||
// Restore the page to stage, confirm that this fixes the links
|
|
||||||
$p = Versioned::get_latest_version('SiteTree', $pageID);
|
|
||||||
$p->doRestoreToStage();
|
|
||||||
|
|
||||||
$p2->flushCache();
|
|
||||||
$p2 = DataObject::get_by_id('SiteTree', $p2->ID);
|
|
||||||
$vp->flushCache();
|
|
||||||
$vp = DataObject::get_by_id('SiteTree', $vp->ID);
|
|
||||||
$rp->flushCache();
|
|
||||||
$rp = DataObject::get_by_id('SiteTree', $rp->ID);
|
|
||||||
$this->assertFalse((bool)$p2->HasBrokenLink);
|
|
||||||
$this->assertFalse((bool)$vp->HasBrokenLink);
|
|
||||||
$this->assertFalse((bool)$rp->HasBrokenLink);
|
|
||||||
|
|
||||||
// Publish and confirm that the p2 and RP broken links are fixed on published
|
|
||||||
$this->assertTrue($p->doPublish());
|
|
||||||
$p2Live = Versioned::get_one_by_stage('SiteTree', 'Live', '"SiteTree"."ID" = ' . $p2->ID);
|
|
||||||
$rpLive = Versioned::get_one_by_stage('SiteTree', 'Live', '"SiteTree"."ID" = ' . $rp->ID);
|
|
||||||
$this->assertFalse((bool)$p2Live->HasBrokenLink);
|
|
||||||
$this->assertFalse((bool)$rpLive->HasBrokenLink);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
function testRevertToLiveFixesBrokenLinks() {
|
|
||||||
// Create page and virutal page
|
|
||||||
$p = new Page();
|
|
||||||
$p->Title = "source";
|
|
||||||
$p->write();
|
|
||||||
$pageID = $p->ID;
|
|
||||||
$this->assertTrue($p->doPublish());
|
|
||||||
|
|
||||||
// Content links are one kind of link to pages
|
|
||||||
$p2 = new Page();
|
|
||||||
$p2->Title = "regular link";
|
|
||||||
$p2->Content = "<a href=\"[sitetree_link id=$p->ID]\">test</a>";
|
|
||||||
$p2->write();
|
|
||||||
$this->assertTrue($p2->doPublish());
|
|
||||||
|
|
||||||
// Virtual pages are another
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
$vp->CopyContentFromID = $p->ID;
|
|
||||||
$vp->write();
|
|
||||||
|
|
||||||
// Redirector links are a third
|
|
||||||
$rp = new RedirectorPage();
|
|
||||||
$rp->Title = "redirector";
|
|
||||||
$rp->LinkType = 'Internal';
|
|
||||||
$rp->LinkToID = $p->ID;
|
|
||||||
$rp->write();
|
|
||||||
$this->assertTrue($rp->doPublish());
|
|
||||||
|
|
||||||
// Confirm that there are no broken links to begin with
|
|
||||||
$this->assertFalse($p2->HasBrokenLink);
|
|
||||||
$this->assertFalse($vp->HasBrokenLink);
|
|
||||||
$this->assertFalse($rp->HasBrokenLink);
|
|
||||||
|
|
||||||
// Delete from draft and confirm that broken links are marked
|
|
||||||
$pID = $p->ID;
|
|
||||||
$p->delete();
|
|
||||||
|
|
||||||
$vp->flushCache();
|
|
||||||
$vp = DataObject::get_by_id('SiteTree', $vp->ID);
|
|
||||||
$p2->flushCache();
|
|
||||||
$p2 = DataObject::get_by_id('SiteTree', $p2->ID);
|
|
||||||
$rp->flushCache();
|
|
||||||
$rp = DataObject::get_by_id('SiteTree', $rp->ID);
|
|
||||||
$this->assertEquals(1, $p2->HasBrokenLink);
|
|
||||||
$this->assertEquals(1, $vp->HasBrokenLink);
|
|
||||||
$this->assertEquals(1, $rp->HasBrokenLink);
|
|
||||||
|
|
||||||
// Call doRevertToLive and confirm that broken links are restored
|
|
||||||
$pLive = Versioned::get_one_by_stage('SiteTree', 'Live', '"SiteTree"."ID" = ' . $pID);
|
|
||||||
$pLive->doRevertToLive();
|
|
||||||
|
|
||||||
$p2->flushCache();
|
|
||||||
$p2 = DataObject::get_by_id('SiteTree', $p2->ID);
|
|
||||||
$vp->flushCache();
|
|
||||||
$vp = DataObject::get_by_id('SiteTree', $vp->ID);
|
|
||||||
$rp->flushCache();
|
|
||||||
$rp = DataObject::get_by_id('SiteTree', $rp->ID);
|
|
||||||
$this->assertFalse((bool)$p2->HasBrokenLink);
|
|
||||||
$this->assertFalse((bool)$vp->HasBrokenLink);
|
|
||||||
$this->assertFalse((bool)$rp->HasBrokenLink);
|
|
||||||
|
|
||||||
// However, the page isn't marked as modified on stage
|
|
||||||
$this->assertFalse($p2->IsModifiedOnStage);
|
|
||||||
$this->assertFalse($rp->IsModifiedOnStage);
|
|
||||||
|
|
||||||
// This is something that we know to be broken
|
|
||||||
//$this->assertFalse($vp->IsModifiedOnStage);
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
?>
|
|
@ -1,27 +0,0 @@
|
|||||||
Page:
|
|
||||||
content:
|
|
||||||
Title: ContentPage
|
|
||||||
Content: This is some happy content.
|
|
||||||
about:
|
|
||||||
Title: About
|
|
||||||
URLSegment: about
|
|
||||||
Content: about us here
|
|
||||||
brokenInternalRedirector:
|
|
||||||
RedirectionType: Internal
|
|
||||||
Title: RedirectorPageToBrokenInteralPage
|
|
||||||
LinkToID: 0
|
|
||||||
workingInternalRedirector:
|
|
||||||
RedirectionType: Internal
|
|
||||||
Title: RedirectorPageToBrokenInteralPage
|
|
||||||
LinkTo: =>Page.content
|
|
||||||
|
|
||||||
File:
|
|
||||||
privacypolicy:
|
|
||||||
Name: privacypolicy.pdf
|
|
||||||
Title: privacypolicy.pdf
|
|
||||||
Filename: assets/privacypolicy.pdf
|
|
||||||
|
|
||||||
ErrorPage:
|
|
||||||
404:
|
|
||||||
Title: Page not Found
|
|
||||||
ErrorCode: 404
|
|
@ -1,445 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage tests
|
|
||||||
*
|
|
||||||
* @todo Test canAddChildren()
|
|
||||||
* @todo Test canCreate()
|
|
||||||
*/
|
|
||||||
class SiteTreePermissionsTest extends FunctionalTest {
|
|
||||||
static $fixture_file = "sapphire/tests/SiteTreePermissionsTest.yml";
|
|
||||||
|
|
||||||
protected $illegalExtensions = array(
|
|
||||||
'SiteTree' => array('SiteTreeSubsites')
|
|
||||||
);
|
|
||||||
|
|
||||||
static function set_up_once() {
|
|
||||||
SiteTreeTest::set_up_once();
|
|
||||||
|
|
||||||
parent::set_up_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
static function tear_down_once() {
|
|
||||||
SiteTreeTest::tear_down_once();
|
|
||||||
|
|
||||||
parent::tear_down_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
function setUp() {
|
|
||||||
parent::setUp();
|
|
||||||
|
|
||||||
$this->useDraftSite();
|
|
||||||
|
|
||||||
// we're testing HTTP status codes before being redirected to login forms
|
|
||||||
$this->autoFollowRedirection = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
function testAccessingStageWithBlankStage() {
|
|
||||||
$this->useDraftSite(false);
|
|
||||||
$this->autoFollowRedirection = false;
|
|
||||||
|
|
||||||
$page = $this->objFromFixture('Page', 'draftOnlyPage');
|
|
||||||
|
|
||||||
if($member = Member::currentUser()) {
|
|
||||||
$member->logOut();
|
|
||||||
}
|
|
||||||
|
|
||||||
$response = $this->get($page->URLSegment . '?stage=Live');
|
|
||||||
$this->assertEquals($response->getStatusCode(), '404');
|
|
||||||
|
|
||||||
$response = $this->get($page->URLSegment . '?stage=');
|
|
||||||
$this->assertEquals($response->getStatusCode(), '404');
|
|
||||||
|
|
||||||
// should be prompted for a login
|
|
||||||
$response = $this->get($page->URLSegment . '?stage=Stage');
|
|
||||||
$this->assertEquals($response->getStatusCode(), '302');
|
|
||||||
|
|
||||||
$this->logInWithPermission('ADMIN');
|
|
||||||
|
|
||||||
$response = $this->get($page->URLSegment . '?stage=Live');
|
|
||||||
$this->assertEquals($response->getStatusCode(), '404');
|
|
||||||
|
|
||||||
$response = $this->get($page->URLSegment . '?stage=Stage');
|
|
||||||
$this->assertEquals($response->getStatusCode(), '200');
|
|
||||||
|
|
||||||
$response = $this->get($page->URLSegment . '?stage=');
|
|
||||||
$this->assertEquals($response->getStatusCode(), '404');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testPermissionCheckingWorksOnDeletedPages() {
|
|
||||||
// Set up fixture - a published page deleted from draft
|
|
||||||
$this->logInWithPermission("ADMIN");
|
|
||||||
$page = $this->objFromFixture('Page','restrictedEditOnlySubadminGroup');
|
|
||||||
$pageID = $page->ID;
|
|
||||||
$this->assertTrue($page->doPublish());
|
|
||||||
$page->delete();
|
|
||||||
|
|
||||||
// Re-fetch the page from the live site
|
|
||||||
$page = Versioned::get_one_by_stage('SiteTree', 'Live', "\"SiteTree\".\"ID\" = $pageID");
|
|
||||||
|
|
||||||
// subadmin has edit rights on that page
|
|
||||||
$member = $this->objFromFixture('Member','subadmin');
|
|
||||||
$member->logIn();
|
|
||||||
|
|
||||||
// Test can_edit_multiple
|
|
||||||
$this->assertEquals(
|
|
||||||
array($pageID => true),
|
|
||||||
SiteTree::can_edit_multiple(array($pageID), $member->ID)
|
|
||||||
);
|
|
||||||
|
|
||||||
// Test canEdit
|
|
||||||
$member->logIn();
|
|
||||||
$this->assertTrue($page->canEdit());
|
|
||||||
}
|
|
||||||
|
|
||||||
function testPermissionCheckingWorksOnUnpublishedPages() {
|
|
||||||
// Set up fixture - an unpublished page
|
|
||||||
$this->logInWithPermission("ADMIN");
|
|
||||||
$page = $this->objFromFixture('Page','restrictedEditOnlySubadminGroup');
|
|
||||||
$pageID = $page->ID;
|
|
||||||
$page->doUnpublish();
|
|
||||||
|
|
||||||
// subadmin has edit rights on that page
|
|
||||||
$member = $this->objFromFixture('Member','subadmin');
|
|
||||||
$member->logIn();
|
|
||||||
|
|
||||||
// Test can_edit_multiple
|
|
||||||
$this->assertEquals(
|
|
||||||
array($pageID => true),
|
|
||||||
SiteTree::can_edit_multiple(array($pageID), $member->ID)
|
|
||||||
);
|
|
||||||
|
|
||||||
// Test canEdit
|
|
||||||
$member->logIn();
|
|
||||||
$this->assertTrue($page->canEdit());
|
|
||||||
}
|
|
||||||
|
|
||||||
function testCanEditOnPageDeletedFromStageAndLiveReturnsFalse() {
|
|
||||||
// Find a page that exists and delete it from both stage and published
|
|
||||||
$this->logInWithPermission("ADMIN");
|
|
||||||
$page = $this->objFromFixture('Page','restrictedEditOnlySubadminGroup');
|
|
||||||
$pageID = $page->ID;
|
|
||||||
$page->doUnpublish();
|
|
||||||
$page->delete();
|
|
||||||
|
|
||||||
// We'll need to resurrect the page from the version cache to test this case
|
|
||||||
$page = Versioned::get_latest_version('SiteTree', $pageID);
|
|
||||||
|
|
||||||
// subadmin had edit rights on that page, but now it's gone
|
|
||||||
$member = $this->objFromFixture('Member','subadmin');
|
|
||||||
$member->logIn();
|
|
||||||
|
|
||||||
$this->assertFalse($page->canEdit());
|
|
||||||
}
|
|
||||||
|
|
||||||
function testCanViewStage() {
|
|
||||||
$page = $this->objFromFixture('Page', 'standardpage');
|
|
||||||
$editor = $this->objFromFixture('Member', 'editor');
|
|
||||||
$websiteuser = $this->objFromFixture('Member', 'websiteuser');
|
|
||||||
|
|
||||||
$this->assertTrue($page->canViewStage('Live', $websiteuser));
|
|
||||||
$this->assertFalse($page->canViewStage('Stage', $websiteuser));
|
|
||||||
|
|
||||||
$this->assertTrue($page->canViewStage('Live', $editor));
|
|
||||||
$this->assertTrue($page->canViewStage('Stage', $editor));
|
|
||||||
}
|
|
||||||
|
|
||||||
function testAccessTabOnlyDisplaysWithGrantAccessPermissions() {
|
|
||||||
$page = $this->objFromFixture('Page', 'standardpage');
|
|
||||||
|
|
||||||
$subadminuser = $this->objFromFixture('Member', 'subadmin');
|
|
||||||
$this->session()->inst_set('loggedInAs', $subadminuser->ID);
|
|
||||||
$fields = $page->getCMSFields();
|
|
||||||
$this->assertFalse(
|
|
||||||
$fields->dataFieldByName('CanViewType')->isReadonly(),
|
|
||||||
'Users with SITETREE_GRANT_ACCESS permission can change "view" permissions in cms fields'
|
|
||||||
);
|
|
||||||
$this->assertFalse(
|
|
||||||
$fields->dataFieldByName('CanEditType')->isReadonly(),
|
|
||||||
'Users with SITETREE_GRANT_ACCESS permission can change "edit" permissions in cms fields'
|
|
||||||
);
|
|
||||||
|
|
||||||
$editoruser = $this->objFromFixture('Member', 'editor');
|
|
||||||
$this->session()->inst_set('loggedInAs', $editoruser->ID);
|
|
||||||
$fields = $page->getCMSFields();
|
|
||||||
$this->assertTrue(
|
|
||||||
$fields->dataFieldByName('CanViewType')->isReadonly(),
|
|
||||||
'Users without SITETREE_GRANT_ACCESS permission cannot change "view" permissions in cms fields'
|
|
||||||
);
|
|
||||||
$this->assertTrue(
|
|
||||||
$fields->dataFieldByName('CanEditType')->isReadonly(),
|
|
||||||
'Users without SITETREE_GRANT_ACCESS permission cannot change "edit" permissions in cms fields'
|
|
||||||
);
|
|
||||||
|
|
||||||
$this->session()->inst_set('loggedInAs', null);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testRestrictedViewLoggedInUsers() {
|
|
||||||
$page = $this->objFromFixture('Page', 'restrictedViewLoggedInUsers');
|
|
||||||
|
|
||||||
// unauthenticated users
|
|
||||||
$this->assertFalse(
|
|
||||||
$page->canView(FALSE),
|
|
||||||
'Unauthenticated members cant view a page marked as "Viewable for any logged in users"'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', null);
|
|
||||||
$response = $this->get($page->RelativeLink());
|
|
||||||
$this->assertEquals(
|
|
||||||
$response->getStatusCode(),
|
|
||||||
302,
|
|
||||||
'Unauthenticated members cant view a page marked as "Viewable for any logged in users"'
|
|
||||||
);
|
|
||||||
|
|
||||||
// website users
|
|
||||||
$websiteuser = $this->objFromFixture('Member', 'websiteuser');
|
|
||||||
$this->assertTrue(
|
|
||||||
$page->canView($websiteuser),
|
|
||||||
'Authenticated members can view a page marked as "Viewable for any logged in users" even if they dont have access to the CMS'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', $websiteuser->ID);
|
|
||||||
$response = $this->get($page->RelativeLink());
|
|
||||||
$this->assertEquals(
|
|
||||||
$response->getStatusCode(),
|
|
||||||
200,
|
|
||||||
'Authenticated members can view a page marked as "Viewable for any logged in users" even if they dont have access to the CMS'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', null);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testRestrictedViewOnlyTheseUsers() {
|
|
||||||
$page = $this->objFromFixture('Page', 'restrictedViewOnlyWebsiteUsers');
|
|
||||||
|
|
||||||
// unauthenticcated users
|
|
||||||
$this->assertFalse(
|
|
||||||
$page->canView(FALSE),
|
|
||||||
'Unauthenticated members cant view a page marked as "Viewable by these groups"'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', null);
|
|
||||||
$response = $this->get($page->RelativeLink());
|
|
||||||
$this->assertEquals(
|
|
||||||
$response->getStatusCode(),
|
|
||||||
302,
|
|
||||||
'Unauthenticated members cant view a page marked as "Viewable by these groups"'
|
|
||||||
);
|
|
||||||
|
|
||||||
// subadmin users
|
|
||||||
$subadminuser = $this->objFromFixture('Member', 'subadmin');
|
|
||||||
$this->assertFalse(
|
|
||||||
$page->canView($subadminuser),
|
|
||||||
'Authenticated members cant view a page marked as "Viewable by these groups" if theyre not in the listed groups'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', $subadminuser->ID);
|
|
||||||
$response = $this->get($page->RelativeLink());
|
|
||||||
$this->assertEquals(
|
|
||||||
$response->getStatusCode(),
|
|
||||||
403,
|
|
||||||
'Authenticated members cant view a page marked as "Viewable by these groups" if theyre not in the listed groups'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', null);
|
|
||||||
|
|
||||||
// website users
|
|
||||||
$websiteuser = $this->objFromFixture('Member', 'websiteuser');
|
|
||||||
$this->assertTrue(
|
|
||||||
$page->canView($websiteuser),
|
|
||||||
'Authenticated members can view a page marked as "Viewable by these groups" if theyre in the listed groups'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', $websiteuser->ID);
|
|
||||||
$response = $this->get($page->RelativeLink());
|
|
||||||
$this->assertEquals(
|
|
||||||
$response->getStatusCode(),
|
|
||||||
200,
|
|
||||||
'Authenticated members can view a page marked as "Viewable by these groups" if theyre in the listed groups'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', null);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testRestrictedEditLoggedInUsers() {
|
|
||||||
$page = $this->objFromFixture('Page', 'restrictedEditLoggedInUsers');
|
|
||||||
|
|
||||||
// unauthenticcated users
|
|
||||||
$this->assertFalse(
|
|
||||||
$page->canEdit(FALSE),
|
|
||||||
'Unauthenticated members cant edit a page marked as "Editable by logged in users"'
|
|
||||||
);
|
|
||||||
|
|
||||||
// website users
|
|
||||||
$websiteuser = $this->objFromFixture('Member', 'websiteuser');
|
|
||||||
$websiteuser->logIn();
|
|
||||||
$this->assertFalse(
|
|
||||||
$page->canEdit($websiteuser),
|
|
||||||
'Authenticated members cant edit a page marked as "Editable by logged in users" if they dont have cms permissions'
|
|
||||||
);
|
|
||||||
|
|
||||||
// subadmin users
|
|
||||||
$subadminuser = $this->objFromFixture('Member', 'subadmin');
|
|
||||||
$this->assertTrue(
|
|
||||||
$page->canEdit($subadminuser),
|
|
||||||
'Authenticated members can edit a page marked as "Editable by logged in users" if they have cms permissions and belong to any of these groups'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testRestrictedEditOnlySubadminGroup() {
|
|
||||||
$page = $this->objFromFixture('Page', 'restrictedEditOnlySubadminGroup');
|
|
||||||
|
|
||||||
// unauthenticated users
|
|
||||||
$this->assertFalse(
|
|
||||||
$page->canEdit(FALSE),
|
|
||||||
'Unauthenticated members cant edit a page marked as "Editable by these groups"'
|
|
||||||
);
|
|
||||||
|
|
||||||
// subadmin users
|
|
||||||
$subadminuser = $this->objFromFixture('Member', 'subadmin');
|
|
||||||
$this->assertTrue(
|
|
||||||
$page->canEdit($subadminuser),
|
|
||||||
'Authenticated members can view a page marked as "Editable by these groups" if theyre in the listed groups'
|
|
||||||
);
|
|
||||||
|
|
||||||
// website users
|
|
||||||
$websiteuser = $this->objFromFixture('Member', 'websiteuser');
|
|
||||||
$this->assertFalse(
|
|
||||||
$page->canEdit($websiteuser),
|
|
||||||
'Authenticated members cant edit a page marked as "Editable by these groups" if theyre not in the listed groups'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testRestrictedViewInheritance() {
|
|
||||||
$parentPage = $this->objFromFixture('Page', 'parent_restrictedViewOnlySubadminGroup');
|
|
||||||
$childPage = $this->objFromFixture('Page', 'child_restrictedViewOnlySubadminGroup');
|
|
||||||
|
|
||||||
// unauthenticated users
|
|
||||||
$this->assertFalse(
|
|
||||||
$childPage->canView(FALSE),
|
|
||||||
'Unauthenticated members cant view a page marked as "Viewable by these groups" by inherited permission'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', null);
|
|
||||||
$response = $this->get($childPage->RelativeLink());
|
|
||||||
$this->assertEquals(
|
|
||||||
$response->getStatusCode(),
|
|
||||||
302,
|
|
||||||
'Unauthenticated members cant view a page marked as "Viewable by these groups" by inherited permission'
|
|
||||||
);
|
|
||||||
|
|
||||||
// subadmin users
|
|
||||||
$subadminuser = $this->objFromFixture('Member', 'subadmin');
|
|
||||||
$this->assertTrue(
|
|
||||||
$childPage->canView($subadminuser),
|
|
||||||
'Authenticated members can view a page marked as "Viewable by these groups" if theyre in the listed groups by inherited permission'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', $subadminuser->ID);
|
|
||||||
$response = $this->get($childPage->RelativeLink());
|
|
||||||
$this->assertEquals(
|
|
||||||
$response->getStatusCode(),
|
|
||||||
200,
|
|
||||||
'Authenticated members can view a page marked as "Viewable by these groups" if theyre in the listed groups by inherited permission'
|
|
||||||
);
|
|
||||||
$this->session()->inst_set('loggedInAs', null);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testRestrictedEditInheritance() {
|
|
||||||
$parentPage = $this->objFromFixture('Page', 'parent_restrictedEditOnlySubadminGroup');
|
|
||||||
$childPage = $this->objFromFixture('Page', 'child_restrictedEditOnlySubadminGroup');
|
|
||||||
|
|
||||||
// unauthenticated users
|
|
||||||
$this->assertFalse(
|
|
||||||
$childPage->canEdit(FALSE),
|
|
||||||
'Unauthenticated members cant edit a page marked as "Editable by these groups" by inherited permission'
|
|
||||||
);
|
|
||||||
|
|
||||||
// subadmin users
|
|
||||||
$subadminuser = $this->objFromFixture('Member', 'subadmin');
|
|
||||||
$this->assertTrue(
|
|
||||||
$childPage->canEdit($subadminuser),
|
|
||||||
'Authenticated members can edit a page marked as "Editable by these groups" if theyre in the listed groups by inherited permission'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testDeleteRestrictedChild() {
|
|
||||||
$parentPage = $this->objFromFixture('Page', 'deleteTestParentPage');
|
|
||||||
$childPage = $this->objFromFixture('Page', 'deleteTestChildPage');
|
|
||||||
|
|
||||||
// unauthenticated users
|
|
||||||
$this->assertFalse(
|
|
||||||
$parentPage->canDelete(FALSE),
|
|
||||||
'Unauthenticated members cant delete a page if it doesnt have delete permissions on any of its descendants'
|
|
||||||
);
|
|
||||||
$this->assertFalse(
|
|
||||||
$childPage->canDelete(FALSE),
|
|
||||||
'Unauthenticated members cant delete a child page marked as "Editable by these groups"'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testRestrictedEditLoggedInUsersDeletedFromStage() {
|
|
||||||
$page = $this->objFromFixture('Page', 'restrictedEditLoggedInUsers');
|
|
||||||
$pageID = $page->ID;
|
|
||||||
|
|
||||||
$this->logInWithPermission("ADMIN");
|
|
||||||
|
|
||||||
$page->doPublish();
|
|
||||||
$page->deleteFromStage('Stage');
|
|
||||||
|
|
||||||
// Get the live version of the page
|
|
||||||
$page = Versioned::get_one_by_stage("SiteTree", "Live", "\"SiteTree\".\"ID\" = $pageID");
|
|
||||||
$this->assertTrue(is_object($page), 'Versioned::get_one_by_stage() is returning an object');
|
|
||||||
|
|
||||||
// subadmin users
|
|
||||||
$subadminuser = $this->objFromFixture('Member', 'subadmin');
|
|
||||||
$this->assertTrue(
|
|
||||||
$page->canEdit($subadminuser),
|
|
||||||
'Authenticated members can edit a page that was deleted from stage and marked as "Editable by logged in users" if they have cms permissions and belong to any of these groups'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testInheritCanViewFromSiteConfig() {
|
|
||||||
$page = $this->objFromFixture('Page', 'inheritWithNoParent');
|
|
||||||
$siteconfig = $this->objFromFixture('SiteConfig', 'default');
|
|
||||||
$editor = $this->objFromFixture('Member', 'editor');
|
|
||||||
$editorGroup = $this->objFromFixture('Group', 'editorgroup');
|
|
||||||
|
|
||||||
$siteconfig->CanViewType = 'Anyone';
|
|
||||||
$siteconfig->write();
|
|
||||||
$this->assertTrue($page->canView(FALSE), 'Anyone can view a page when set to inherit from the SiteConfig, and SiteConfig has canView set to LoggedInUsers');
|
|
||||||
|
|
||||||
$siteconfig->CanViewType = 'LoggedInUsers';
|
|
||||||
$siteconfig->write();
|
|
||||||
$this->assertFalse($page->canView(FALSE), 'Anonymous can\'t view a page when set to inherit from the SiteConfig, and SiteConfig has canView set to LoggedInUsers');
|
|
||||||
|
|
||||||
$siteconfig->CanViewType = 'LoggedInUsers';
|
|
||||||
$siteconfig->write();
|
|
||||||
$this->assertTrue($page->canView($editor), 'Users can view a page when set to inherit from the SiteConfig, and SiteConfig has canView set to LoggedInUsers');
|
|
||||||
|
|
||||||
$siteconfig->CanViewType = 'OnlyTheseUsers';
|
|
||||||
$siteconfig->ViewerGroups()->add($editorGroup);
|
|
||||||
$siteconfig->ViewerGroups()->write();
|
|
||||||
$siteconfig->write();
|
|
||||||
$this->assertTrue($page->canView($editor), 'Editors can view a page when set to inherit from the SiteConfig, and SiteConfig has canView set to OnlyTheseUsers');
|
|
||||||
$this->assertFalse($page->canView(FALSE), 'Anonymous can\'t view a page when set to inherit from the SiteConfig, and SiteConfig has canView set to OnlyTheseUsers');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testInheritCanEditFromSiteConfig() {
|
|
||||||
$page = $this->objFromFixture('Page', 'inheritWithNoParent');
|
|
||||||
$siteconfig = $this->objFromFixture('SiteConfig', 'default');
|
|
||||||
$editor = $this->objFromFixture('Member', 'editor');
|
|
||||||
$user = $this->objFromFixture('Member', 'websiteuser');
|
|
||||||
$editorGroup = $this->objFromFixture('Group', 'editorgroup');
|
|
||||||
|
|
||||||
$siteconfig->CanEditType = 'LoggedInUsers';
|
|
||||||
$siteconfig->write();
|
|
||||||
|
|
||||||
$this->assertFalse($page->canEdit(FALSE), 'Anonymous can\'t edit a page when set to inherit from the SiteConfig, and SiteConfig has canEdit set to LoggedInUsers');
|
|
||||||
$this->session()->inst_set('loggedInAs', $editor->ID);
|
|
||||||
$this->assertTrue($page->canEdit(), 'Users can edit a page when set to inherit from the SiteConfig, and SiteConfig has canEdit set to LoggedInUsers');
|
|
||||||
|
|
||||||
$siteconfig->CanEditType = 'OnlyTheseUsers';
|
|
||||||
$siteconfig->EditorGroups()->add($editorGroup);
|
|
||||||
$siteconfig->EditorGroups()->write();
|
|
||||||
$siteconfig->write();
|
|
||||||
$this->assertTrue($page->canEdit($editor), 'Editors can edit a page when set to inherit from the SiteConfig, and SiteConfig has canEdit set to OnlyTheseUsers');
|
|
||||||
$this->session()->inst_set('loggedInAs', null);
|
|
||||||
$this->assertFalse($page->canEdit(FALSE), 'Anonymous can\'t edit a page when set to inherit from the SiteConfig, and SiteConfig has canEdit set to OnlyTheseUsers');
|
|
||||||
$this->session()->inst_set('loggedInAs', $user->ID);
|
|
||||||
$this->assertFalse($page->canEdit($user), 'Website user can\'t edit a page when set to inherit from the SiteConfig, and SiteConfig has canEdit set to OnlyTheseUsers');
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
?>
|
|
@ -1,88 +0,0 @@
|
|||||||
SiteConfig:
|
|
||||||
default:
|
|
||||||
Title: My test site
|
|
||||||
Tagline: There is no doubt this is a great test site
|
|
||||||
CanViewType: Anyone
|
|
||||||
CanEditType: LoggedInUsers
|
|
||||||
Permission:
|
|
||||||
cmsmain1:
|
|
||||||
Code: CMS_ACCESS_CMSMain
|
|
||||||
cmsmain2:
|
|
||||||
Code: CMS_ACCESS_CMSMain
|
|
||||||
grantaccess:
|
|
||||||
Code: SITETREE_GRANT_ACCESS
|
|
||||||
Group:
|
|
||||||
subadmingroup:
|
|
||||||
Title: Create, edit and delete pages
|
|
||||||
Code: subadmingroup
|
|
||||||
Permissions: =>Permission.cmsmain1,=>Permission.grantaccess
|
|
||||||
editorgroup:
|
|
||||||
Title: Edit existing pages
|
|
||||||
Code: editorgroup
|
|
||||||
Permissions: =>Permission.cmsmain2
|
|
||||||
websiteusers:
|
|
||||||
Title: View certain restricted pages
|
|
||||||
Member:
|
|
||||||
subadmin:
|
|
||||||
Email: subadmin@test.com
|
|
||||||
Password: test
|
|
||||||
Groups: =>Group.subadmingroup
|
|
||||||
editor:
|
|
||||||
Email: editor@test.com
|
|
||||||
Password: test
|
|
||||||
Groups: =>Group.editorgroup
|
|
||||||
websiteuser:
|
|
||||||
Email: websiteuser@test.com
|
|
||||||
Password: test
|
|
||||||
Groups: =>Group.websiteusers
|
|
||||||
Page:
|
|
||||||
standardpage:
|
|
||||||
URLSegment: standardpage
|
|
||||||
restrictedViewLoggedInUsers:
|
|
||||||
CanViewType: LoggedInUsers
|
|
||||||
URLSegment: restrictedViewLoggedInUsers
|
|
||||||
restrictedViewOnlyWebsiteUsers:
|
|
||||||
CanViewType: OnlyTheseUsers
|
|
||||||
ViewerGroups: =>Group.websiteusers
|
|
||||||
URLSegment: restrictedViewOnlyWebsiteUsers
|
|
||||||
restrictedViewOnlySubadminGroup:
|
|
||||||
CanViewType: OnlyTheseUsers
|
|
||||||
ViewerGroups: =>Group.subadmingroup
|
|
||||||
URLSegment: restrictedViewOnlySubadminGroup
|
|
||||||
restrictedEditLoggedInUsers:
|
|
||||||
CanEditType: LoggedInUsers
|
|
||||||
URLSegment: restrictedEditLoggedInUsers
|
|
||||||
restrictedEditOnlySubadminGroup:
|
|
||||||
CanEditType: OnlyTheseUsers
|
|
||||||
EditorGroups: =>Group.subadmingroup
|
|
||||||
URLSegment: restrictedEditOnlySubadminGroup
|
|
||||||
inheritWithNoParent:
|
|
||||||
CanEditType: Inherit
|
|
||||||
CanViewType: Inherit
|
|
||||||
URLSegment: inheritWithNoParent
|
|
||||||
parent_restrictedViewOnlySubadminGroup:
|
|
||||||
CanViewType: OnlyTheseUsers
|
|
||||||
ViewerGroups: =>Group.subadmingroup
|
|
||||||
URLSegment: parent-restrictedViewOnlySubadminGroup
|
|
||||||
child_restrictedViewOnlySubadminGroup:
|
|
||||||
CanViewType: Inherit
|
|
||||||
Parent: =>Page.parent_restrictedViewOnlySubadminGroup
|
|
||||||
URLSegment: child-restrictedViewOnlySubadminGroup
|
|
||||||
parent_restrictedEditOnlySubadminGroup:
|
|
||||||
CanEditType: OnlyTheseUsers
|
|
||||||
EditorGroups: =>Group.subadmingroup
|
|
||||||
URLSegment: parent-restrictedEditOnlySubadminGroup
|
|
||||||
child_restrictedEditOnlySubadminGroup:
|
|
||||||
CanEditType: Inherit
|
|
||||||
Parent: =>Page.parent_restrictedEditOnlySubadminGroup
|
|
||||||
URLSegment: child-restrictedEditOnlySubadminGroup
|
|
||||||
deleteTestParentPage:
|
|
||||||
CanEditType: Inherit
|
|
||||||
URLSegment: deleteTestParentPage
|
|
||||||
deleteTestChildPage:
|
|
||||||
CanEditType: OnlyTheseUsers
|
|
||||||
EditorGroups: =>Group.subadmingroup
|
|
||||||
URLSegment: deleteTestChildPage
|
|
||||||
draftOnlyPage:
|
|
||||||
CanViewType: Anyone
|
|
||||||
URLSegment: draft-only
|
|
@ -1,756 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage tests
|
|
||||||
*/
|
|
||||||
class SiteTreeTest extends SapphireTest {
|
|
||||||
static $fixture_file = 'sapphire/tests/SiteTreeTest.yml';
|
|
||||||
|
|
||||||
protected $illegalExtensions = array(
|
|
||||||
'SiteTree' => array('SiteTreeSubsites')
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @todo Necessary because of monolithic Translatable design
|
|
||||||
*/
|
|
||||||
static protected $origTranslatableSettings = array();
|
|
||||||
|
|
||||||
static function set_up_once() {
|
|
||||||
// needs to recreate the database schema with language properties
|
|
||||||
self::kill_temp_db();
|
|
||||||
|
|
||||||
// store old defaults
|
|
||||||
self::$origTranslatableSettings['has_extension'] = singleton('SiteTree')->hasExtension('Translatable');
|
|
||||||
self::$origTranslatableSettings['default_locale'] = Translatable::default_locale();
|
|
||||||
|
|
||||||
// overwrite locale
|
|
||||||
Translatable::set_default_locale("en_US");
|
|
||||||
|
|
||||||
// refresh the decorated statics - different fields in $db with Translatable enabled
|
|
||||||
if(self::$origTranslatableSettings['has_extension']) {
|
|
||||||
Object::remove_extension('SiteTree', 'Translatable');
|
|
||||||
Object::remove_extension('SiteConfig', 'Translatable');
|
|
||||||
}
|
|
||||||
|
|
||||||
// recreate database with new settings
|
|
||||||
$dbname = self::create_temp_db();
|
|
||||||
DB::set_alternative_database_name($dbname);
|
|
||||||
|
|
||||||
parent::set_up_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
static function tear_down_once() {
|
|
||||||
if(self::$origTranslatableSettings['has_extension']) {
|
|
||||||
Object::add_extension('SiteTree', 'Translatable');
|
|
||||||
Object::add_extension('SiteConfig', 'Translatable');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
Translatable::set_default_locale(self::$origTranslatableSettings['default_locale']);
|
|
||||||
Translatable::set_current_locale(self::$origTranslatableSettings['default_locale']);
|
|
||||||
|
|
||||||
self::kill_temp_db();
|
|
||||||
self::create_temp_db();
|
|
||||||
|
|
||||||
parent::tear_down_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
function testCreateDefaultpages() {
|
|
||||||
$remove = DataObject::get('SiteTree');
|
|
||||||
if($remove) foreach($remove as $page) $page->delete();
|
|
||||||
// Make sure the table is empty
|
|
||||||
$this->assertEquals(DB::query('SELECT COUNT("ID") FROM "SiteTree"')->value(), 0);
|
|
||||||
|
|
||||||
// Disable the creation
|
|
||||||
SiteTree::set_create_default_pages(false);
|
|
||||||
singleton('SiteTree')->requireDefaultRecords();
|
|
||||||
|
|
||||||
// The table should still be empty
|
|
||||||
$this->assertEquals(DB::query('SELECT COUNT("ID") FROM "SiteTree"')->value(), 0);
|
|
||||||
|
|
||||||
// Enable the creation
|
|
||||||
SiteTree::set_create_default_pages(true);
|
|
||||||
singleton('SiteTree')->requireDefaultRecords();
|
|
||||||
|
|
||||||
// The table should now have three rows (home, about-us, contact-us)
|
|
||||||
$this->assertEquals(DB::query('SELECT COUNT("ID") FROM "SiteTree"')->value(), 3);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test generation of the URLSegment values.
|
|
||||||
* - Turns things into lowercase-hyphen-format
|
|
||||||
* - Generates from Title by default, unless URLSegment is explicitly set
|
|
||||||
* - Resolves duplicates by appending a number
|
|
||||||
* - renames classes with a class name conflict
|
|
||||||
*/
|
|
||||||
function testURLGeneration() {
|
|
||||||
$expectedURLs = array(
|
|
||||||
'home' => 'home',
|
|
||||||
'staff' => 'my-staff',
|
|
||||||
'about' => 'about-us',
|
|
||||||
'staffduplicate' => 'my-staff-2',
|
|
||||||
'product1' => '1-1-test-product',
|
|
||||||
'product2' => 'another-product',
|
|
||||||
'product3' => 'another-product-2',
|
|
||||||
'product4' => 'another-product-3',
|
|
||||||
'object' => 'object',
|
|
||||||
'controller' => 'controller-2',
|
|
||||||
'numericonly' => '1930',
|
|
||||||
);
|
|
||||||
|
|
||||||
foreach($expectedURLs as $fixture => $urlSegment) {
|
|
||||||
$obj = $this->objFromFixture('Page', $fixture);
|
|
||||||
$this->assertEquals($urlSegment, $obj->URLSegment);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test that publication copies data to SiteTree_Live
|
|
||||||
*/
|
|
||||||
function testPublishCopiesToLiveTable() {
|
|
||||||
$obj = $this->objFromFixture('Page','about');
|
|
||||||
$obj->publish('Stage', 'Live');
|
|
||||||
|
|
||||||
$createdID = DB::query("SELECT \"ID\" FROM \"SiteTree_Live\" WHERE \"URLSegment\" = '$obj->URLSegment'")->value();
|
|
||||||
$this->assertEquals($obj->ID, $createdID);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test that field which are set and then cleared are also transferred to the published site.
|
|
||||||
*/
|
|
||||||
function testPublishDeletedFields() {
|
|
||||||
$this->logInWithPermission('ADMIN');
|
|
||||||
|
|
||||||
$obj = $this->objFromFixture('Page', 'about');
|
|
||||||
$obj->MetaTitle = "asdfasdf";
|
|
||||||
$obj->write();
|
|
||||||
$this->assertTrue($obj->doPublish());
|
|
||||||
|
|
||||||
$this->assertEquals('asdfasdf', DB::query("SELECT \"MetaTitle\" FROM \"SiteTree_Live\" WHERE \"ID\" = '$obj->ID'")->value());
|
|
||||||
|
|
||||||
$obj->MetaTitle = null;
|
|
||||||
$obj->write();
|
|
||||||
$this->assertTrue($obj->doPublish());
|
|
||||||
|
|
||||||
$this->assertNull(DB::query("SELECT \"MetaTitle\" FROM \"SiteTree_Live\" WHERE \"ID\" = '$obj->ID'")->value());
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
function testParentNodeCachedInMemory() {
|
|
||||||
$parent = new SiteTree();
|
|
||||||
$parent->Title = 'Section Title';
|
|
||||||
$child = new SiteTree();
|
|
||||||
$child->Title = 'Page Title';
|
|
||||||
$child->setParent($parent);
|
|
||||||
|
|
||||||
$this->assertType("SiteTree", $child->Parent);
|
|
||||||
$this->assertEquals("Section Title", $child->Parent->Title);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testParentModelReturnType() {
|
|
||||||
$parent = new SiteTreeTest_PageNode();
|
|
||||||
$child = new SiteTreeTest_PageNode();
|
|
||||||
|
|
||||||
$child->setParent($parent);
|
|
||||||
$this->assertType('SiteTreeTest_PageNode', $child->Parent);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Confirm that DataObject::get_one() gets records from SiteTree_Live
|
|
||||||
*/
|
|
||||||
function testGetOneFromLive() {
|
|
||||||
$s = new SiteTree();
|
|
||||||
$s->Title = "V1";
|
|
||||||
$s->URLSegment = "get-one-test-page";
|
|
||||||
$s->write();
|
|
||||||
$s->publish("Stage", "Live");
|
|
||||||
$s->Title = "V2";
|
|
||||||
$s->write();
|
|
||||||
|
|
||||||
$oldMode = Versioned::get_reading_mode();
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
|
|
||||||
$checkSiteTree = DataObject::get_one("SiteTree", "\"URLSegment\" = 'get-one-test-page'");
|
|
||||||
$this->assertEquals("V1", $checkSiteTree->Title);
|
|
||||||
|
|
||||||
Versioned::set_reading_mode($oldMode);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testChidrenOfRootAreTopLevelPages() {
|
|
||||||
$pages = DataObject::get("SiteTree");
|
|
||||||
foreach($pages as $page) $page->publish('Stage', 'Live');
|
|
||||||
unset($pages);
|
|
||||||
|
|
||||||
/* If we create a new SiteTree object with ID = 0 */
|
|
||||||
$obj = new SiteTree();
|
|
||||||
/* Then its children should be the top-level pages */
|
|
||||||
$stageChildren = $obj->stageChildren()->toDropDownMap('ID','Title');
|
|
||||||
$liveChildren = $obj->liveChildren()->toDropDownMap('ID','Title');
|
|
||||||
$allChildren = $obj->AllChildrenIncludingDeleted()->toDropDownMap('ID','Title');
|
|
||||||
|
|
||||||
$this->assertContains('Home', $stageChildren);
|
|
||||||
$this->assertContains('Products', $stageChildren);
|
|
||||||
$this->assertNotContains('Staff', $stageChildren);
|
|
||||||
|
|
||||||
$this->assertContains('Home', $liveChildren);
|
|
||||||
$this->assertContains('Products', $liveChildren);
|
|
||||||
$this->assertNotContains('Staff', $liveChildren);
|
|
||||||
|
|
||||||
$this->assertContains('Home', $allChildren);
|
|
||||||
$this->assertContains('Products', $allChildren);
|
|
||||||
$this->assertNotContains('Staff', $allChildren);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testCanSaveBlankToHasOneRelations() {
|
|
||||||
/* DataObject::write() should save to a has_one relationship if you set a field called (relname)ID */
|
|
||||||
$page = new SiteTree();
|
|
||||||
$parentID = $this->idFromFixture('Page', 'home');
|
|
||||||
$page->ParentID = $parentID;
|
|
||||||
$page->write();
|
|
||||||
$this->assertEquals($parentID, DB::query("SELECT \"ParentID\" FROM \"SiteTree\" WHERE \"ID\" = $page->ID")->value());
|
|
||||||
|
|
||||||
/* You should then be able to save a null/0/'' value to the relation */
|
|
||||||
$page->ParentID = null;
|
|
||||||
$page->write();
|
|
||||||
$this->assertEquals(0, DB::query("SELECT \"ParentID\" FROM \"SiteTree\" WHERE \"ID\" = $page->ID")->value());
|
|
||||||
}
|
|
||||||
|
|
||||||
function testStageStates() {
|
|
||||||
// newly created page
|
|
||||||
$createdPage = new SiteTree();
|
|
||||||
$createdPage->write();
|
|
||||||
$this->assertFalse($createdPage->IsDeletedFromStage);
|
|
||||||
$this->assertTrue($createdPage->IsAddedToStage);
|
|
||||||
$this->assertTrue($createdPage->IsModifiedOnStage);
|
|
||||||
|
|
||||||
// published page
|
|
||||||
$publishedPage = new SiteTree();
|
|
||||||
$publishedPage->write();
|
|
||||||
$publishedPage->publish('Stage','Live');
|
|
||||||
$this->assertFalse($publishedPage->IsDeletedFromStage);
|
|
||||||
$this->assertFalse($publishedPage->IsAddedToStage);
|
|
||||||
$this->assertFalse($publishedPage->IsModifiedOnStage);
|
|
||||||
|
|
||||||
// published page, deleted from stage
|
|
||||||
$deletedFromDraftPage = new SiteTree();
|
|
||||||
$deletedFromDraftPage->write();
|
|
||||||
$deletedFromDraftPageID = $deletedFromDraftPage->ID;
|
|
||||||
$deletedFromDraftPage->publish('Stage','Live');
|
|
||||||
$deletedFromDraftPage->deleteFromStage('Stage');
|
|
||||||
$this->assertTrue($deletedFromDraftPage->IsDeletedFromStage);
|
|
||||||
$this->assertFalse($deletedFromDraftPage->IsAddedToStage);
|
|
||||||
$this->assertFalse($deletedFromDraftPage->IsModifiedOnStage);
|
|
||||||
|
|
||||||
// published page, deleted from live
|
|
||||||
$deletedFromLivePage = new SiteTree();
|
|
||||||
$deletedFromLivePage->write();
|
|
||||||
$deletedFromLivePage->publish('Stage','Live');
|
|
||||||
$deletedFromLivePage->deleteFromStage('Stage');
|
|
||||||
$deletedFromLivePage->deleteFromStage('Live');
|
|
||||||
$this->assertTrue($deletedFromLivePage->IsDeletedFromStage);
|
|
||||||
$this->assertFalse($deletedFromLivePage->IsAddedToStage);
|
|
||||||
$this->assertFalse($deletedFromLivePage->IsModifiedOnStage);
|
|
||||||
|
|
||||||
// published page, modified
|
|
||||||
$modifiedOnDraftPage = new SiteTree();
|
|
||||||
$modifiedOnDraftPage->write();
|
|
||||||
$modifiedOnDraftPage->publish('Stage','Live');
|
|
||||||
$modifiedOnDraftPage->Content = 'modified';
|
|
||||||
$modifiedOnDraftPage->write();
|
|
||||||
$this->assertFalse($modifiedOnDraftPage->IsDeletedFromStage);
|
|
||||||
$this->assertFalse($modifiedOnDraftPage->IsAddedToStage);
|
|
||||||
$this->assertTrue($modifiedOnDraftPage->IsModifiedOnStage);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test that a page can be completely deleted and restored to the stage site
|
|
||||||
*/
|
|
||||||
function testRestoreToStage() {
|
|
||||||
$page = $this->objFromFixture('Page', 'about');
|
|
||||||
$pageID = $page->ID;
|
|
||||||
$page->delete();
|
|
||||||
$this->assertTrue(!DataObject::get_by_id("Page", $pageID));
|
|
||||||
|
|
||||||
$deletedPage = Versioned::get_latest_version('SiteTree', $pageID);
|
|
||||||
$resultPage = $deletedPage->doRestoreToStage();
|
|
||||||
|
|
||||||
$requeriedPage = DataObject::get_by_id("Page", $pageID);
|
|
||||||
|
|
||||||
$this->assertEquals($pageID, $resultPage->ID);
|
|
||||||
$this->assertEquals($pageID, $requeriedPage->ID);
|
|
||||||
$this->assertEquals('About Us', $requeriedPage->Title);
|
|
||||||
$this->assertEquals('Page', $requeriedPage->class);
|
|
||||||
|
|
||||||
|
|
||||||
$page2 = $this->objFromFixture('Page', 'products');
|
|
||||||
$page2ID = $page2->ID;
|
|
||||||
$page2->doUnpublish();
|
|
||||||
$page2->delete();
|
|
||||||
|
|
||||||
// Check that if we restore while on the live site that the content still gets pushed to
|
|
||||||
// stage
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
$deletedPage = Versioned::get_latest_version('SiteTree', $page2ID);
|
|
||||||
$deletedPage->doRestoreToStage();
|
|
||||||
$this->assertTrue(!Versioned::get_one_by_stage("Page", "Live", "\"SiteTree\".\"ID\" = " . $page2ID));
|
|
||||||
|
|
||||||
Versioned::reading_stage('Stage');
|
|
||||||
$requeriedPage = DataObject::get_by_id("Page", $page2ID);
|
|
||||||
$this->assertEquals('Products', $requeriedPage->Title);
|
|
||||||
$this->assertEquals('Page', $requeriedPage->class);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testGetByLink() {
|
|
||||||
$home = $this->objFromFixture('Page', 'home');
|
|
||||||
$about = $this->objFromFixture('Page', 'about');
|
|
||||||
$staff = $this->objFromFixture('Page', 'staff');
|
|
||||||
$product = $this->objFromFixture('Page', 'product1');
|
|
||||||
$notFound = $this->objFromFixture('ErrorPage', '404');
|
|
||||||
|
|
||||||
SiteTree::disable_nested_urls();
|
|
||||||
|
|
||||||
$this->assertEquals($home->ID, SiteTree::get_by_link('/', false)->ID);
|
|
||||||
$this->assertEquals($home->ID, SiteTree::get_by_link('/home/', false)->ID);
|
|
||||||
$this->assertEquals($about->ID, SiteTree::get_by_link($about->Link(), false)->ID);
|
|
||||||
$this->assertEquals($staff->ID, SiteTree::get_by_link($staff->Link(), false)->ID);
|
|
||||||
$this->assertEquals($product->ID, SiteTree::get_by_link($product->Link(), false)->ID);
|
|
||||||
$this->assertEquals($notFound->ID, SiteTree::get_by_link($notFound->Link(), false)->ID);
|
|
||||||
|
|
||||||
SiteTree::enable_nested_urls();
|
|
||||||
|
|
||||||
$this->assertEquals($home->ID, SiteTree::get_by_link('/', false)->ID);
|
|
||||||
$this->assertEquals($home->ID, SiteTree::get_by_link('/home/', false)->ID);
|
|
||||||
$this->assertEquals($about->ID, SiteTree::get_by_link($about->Link(), false)->ID);
|
|
||||||
$this->assertEquals($staff->ID, SiteTree::get_by_link($staff->Link(), false)->ID);
|
|
||||||
$this->assertEquals($product->ID, SiteTree::get_by_link($product->Link(), false)->ID);
|
|
||||||
$this->assertEquals($notFound->ID, SiteTree::get_by_link($notFound->Link(), false)->ID);
|
|
||||||
|
|
||||||
$this->assertEquals (
|
|
||||||
$staff->ID, SiteTree::get_by_link('/my-staff/', false)->ID, 'Assert a unique URLSegment can be used for b/c.'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testRelativeLink() {
|
|
||||||
$about = $this->objFromFixture('Page', 'about');
|
|
||||||
$staff = $this->objFromFixture('Page', 'staff');
|
|
||||||
|
|
||||||
$this->assertEquals('about-us/', $about->RelativeLink(), 'Matches URLSegment on top level without parameters');
|
|
||||||
$this->assertEquals('about-us/my-staff/', $staff->RelativeLink(), 'Matches URLSegment plus parent on second level without parameters');
|
|
||||||
$this->assertEquals('about-us/edit', $about->RelativeLink('edit'), 'Matches URLSegment plus parameter on top level');
|
|
||||||
$this->assertEquals('about-us/tom&jerry', $about->RelativeLink('tom&jerry'), 'Doesnt url encode parameter');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testDeleteFromStageOperatesRecursively() {
|
|
||||||
SiteTree::set_enforce_strict_hierarchy(false);
|
|
||||||
$pageAbout = $this->objFromFixture('Page', 'about');
|
|
||||||
$pageStaff = $this->objFromFixture('Page', 'staff');
|
|
||||||
$pageStaffDuplicate = $this->objFromFixture('Page', 'staffduplicate');
|
|
||||||
|
|
||||||
$pageAbout->delete();
|
|
||||||
|
|
||||||
$this->assertFalse(DataObject::get_by_id('Page', $pageAbout->ID));
|
|
||||||
$this->assertTrue(DataObject::get_by_id('Page', $pageStaff->ID) instanceof Page);
|
|
||||||
$this->assertTrue(DataObject::get_by_id('Page', $pageStaffDuplicate->ID) instanceof Page);
|
|
||||||
SiteTree::set_enforce_strict_hierarchy(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testDeleteFromStageOperatesRecursivelyStrict() {
|
|
||||||
$pageAbout = $this->objFromFixture('Page', 'about');
|
|
||||||
$pageStaff = $this->objFromFixture('Page', 'staff');
|
|
||||||
$pageStaffDuplicate = $this->objFromFixture('Page', 'staffduplicate');
|
|
||||||
|
|
||||||
$pageAbout->delete();
|
|
||||||
|
|
||||||
$this->assertFalse(DataObject::get_by_id('Page', $pageAbout->ID));
|
|
||||||
$this->assertFalse(DataObject::get_by_id('Page', $pageStaff->ID));
|
|
||||||
$this->assertFalse(DataObject::get_by_id('Page', $pageStaffDuplicate->ID));
|
|
||||||
}
|
|
||||||
|
|
||||||
function testDeleteFromLiveOperatesRecursively() {
|
|
||||||
SiteTree::set_enforce_strict_hierarchy(false);
|
|
||||||
$this->logInWithPermission('ADMIN');
|
|
||||||
|
|
||||||
$pageAbout = $this->objFromFixture('Page', 'about');
|
|
||||||
$pageAbout->doPublish();
|
|
||||||
$pageStaff = $this->objFromFixture('Page', 'staff');
|
|
||||||
$pageStaff->doPublish();
|
|
||||||
$pageStaffDuplicate = $this->objFromFixture('Page', 'staffduplicate');
|
|
||||||
$pageStaffDuplicate->doPublish();
|
|
||||||
|
|
||||||
$parentPage = $this->objFromFixture('Page', 'about');
|
|
||||||
$parentPage->doDeleteFromLive();
|
|
||||||
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
$this->assertFalse(DataObject::get_by_id('Page', $pageAbout->ID));
|
|
||||||
$this->assertTrue(DataObject::get_by_id('Page', $pageStaff->ID) instanceof Page);
|
|
||||||
$this->assertTrue(DataObject::get_by_id('Page', $pageStaffDuplicate->ID) instanceof Page);
|
|
||||||
Versioned::reading_stage('Stage');
|
|
||||||
SiteTree::set_enforce_strict_hierarchy(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testUnpublishDoesNotDeleteChildrenWithLooseHierachyOn() {
|
|
||||||
SiteTree::set_enforce_strict_hierarchy(false);
|
|
||||||
$this->logInWithPermission('ADMIN');
|
|
||||||
|
|
||||||
$pageAbout = $this->objFromFixture('Page', 'about');
|
|
||||||
$pageAbout->doPublish();
|
|
||||||
$pageStaff = $this->objFromFixture('Page', 'staff');
|
|
||||||
$pageStaff->doPublish();
|
|
||||||
$pageStaffDuplicate = $this->objFromFixture('Page', 'staffduplicate');
|
|
||||||
$pageStaffDuplicate->doPublish();
|
|
||||||
|
|
||||||
$parentPage = $this->objFromFixture('Page', 'about');
|
|
||||||
$parentPage->doUnpublish();
|
|
||||||
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
$this->assertFalse(DataObject::get_by_id('Page', $pageAbout->ID));
|
|
||||||
$this->assertTrue(DataObject::get_by_id('Page', $pageStaff->ID) instanceof Page);
|
|
||||||
$this->assertTrue(DataObject::get_by_id('Page', $pageStaffDuplicate->ID) instanceof Page);
|
|
||||||
Versioned::reading_stage('Stage');
|
|
||||||
SiteTree::set_enforce_strict_hierarchy(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
function testDeleteFromLiveOperatesRecursivelyStrict() {
|
|
||||||
$this->logInWithPermission('ADMIN');
|
|
||||||
|
|
||||||
$pageAbout = $this->objFromFixture('Page', 'about');
|
|
||||||
$pageAbout->doPublish();
|
|
||||||
$pageStaff = $this->objFromFixture('Page', 'staff');
|
|
||||||
$pageStaff->doPublish();
|
|
||||||
$pageStaffDuplicate = $this->objFromFixture('Page', 'staffduplicate');
|
|
||||||
$pageStaffDuplicate->doPublish();
|
|
||||||
|
|
||||||
$parentPage = $this->objFromFixture('Page', 'about');
|
|
||||||
$parentPage->doDeleteFromLive();
|
|
||||||
|
|
||||||
Versioned::reading_stage('Live');
|
|
||||||
$this->assertFalse(DataObject::get_by_id('Page', $pageAbout->ID));
|
|
||||||
$this->assertFalse(DataObject::get_by_id('Page', $pageStaff->ID));
|
|
||||||
$this->assertFalse(DataObject::get_by_id('Page', $pageStaffDuplicate->ID));
|
|
||||||
Versioned::reading_stage('Stage');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Simple test to confirm that querying from a particular archive date doesn't throw
|
|
||||||
* an error
|
|
||||||
*/
|
|
||||||
function testReadArchiveDate() {
|
|
||||||
Versioned::reading_archived_date('2009-07-02 14:05:07');
|
|
||||||
|
|
||||||
DataObject::get('SiteTree', "\"ParentID\" = 0");
|
|
||||||
|
|
||||||
Versioned::reading_archived_date(null);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testEditPermissions() {
|
|
||||||
$editor = $this->objFromFixture("Member", "editor");
|
|
||||||
|
|
||||||
$home = $this->objFromFixture("Page", "home");
|
|
||||||
$products = $this->objFromFixture("Page", "products");
|
|
||||||
$product1 = $this->objFromFixture("Page", "product1");
|
|
||||||
$product4 = $this->objFromFixture("Page", "product4");
|
|
||||||
|
|
||||||
// Can't edit a page that is locked to admins
|
|
||||||
$this->assertFalse($home->canEdit($editor));
|
|
||||||
|
|
||||||
// Can edit a page that is locked to editors
|
|
||||||
$this->assertTrue($products->canEdit($editor));
|
|
||||||
|
|
||||||
// Can edit a child of that page that inherits
|
|
||||||
$this->assertTrue($product1->canEdit($editor));
|
|
||||||
|
|
||||||
// Can't edit a child of that page that has its permissions overridden
|
|
||||||
$this->assertFalse($product4->canEdit($editor));
|
|
||||||
}
|
|
||||||
|
|
||||||
function testEditPermissionsOnDraftVsLive() {
|
|
||||||
// Create an inherit-permission page
|
|
||||||
$page = new Page();
|
|
||||||
$page->write();
|
|
||||||
$page->CanEditType = "Inherit";
|
|
||||||
$page->doPublish();
|
|
||||||
$pageID = $page->ID;
|
|
||||||
|
|
||||||
// Lock down the site config
|
|
||||||
$sc = $page->SiteConfig;
|
|
||||||
$sc->CanEditType = 'OnlyTheseUsers';
|
|
||||||
$sc->EditorGroups()->add($this->idFromFixture('Group', 'admins'));
|
|
||||||
$sc->write();
|
|
||||||
|
|
||||||
// Confirm that Member.editor can't edit the page
|
|
||||||
$this->objFromFixture('Member','editor')->logIn();
|
|
||||||
$this->assertFalse($page->canEdit());
|
|
||||||
|
|
||||||
// Change the page to be editable by Group.editors, but do not publish
|
|
||||||
$this->objFromFixture('Member','admin')->logIn();
|
|
||||||
$page->CanEditType = 'OnlyTheseUsers';
|
|
||||||
$page->EditorGroups()->add($this->idFromFixture('Group', 'editors'));
|
|
||||||
$page->write();
|
|
||||||
|
|
||||||
// Confirm that Member.editor can now edit the page
|
|
||||||
$this->objFromFixture('Member','editor')->logIn();
|
|
||||||
$this->assertTrue($page->canEdit());
|
|
||||||
|
|
||||||
// Publish the changes to the page
|
|
||||||
$this->objFromFixture('Member','admin')->logIn();
|
|
||||||
$page->doPublish();
|
|
||||||
|
|
||||||
// Confirm that Member.editor can still edit the page
|
|
||||||
$this->objFromFixture('Member','editor')->logIn();
|
|
||||||
$this->assertTrue($page->canEdit());
|
|
||||||
}
|
|
||||||
|
|
||||||
function testCompareVersions() {
|
|
||||||
$page = new Page();
|
|
||||||
$page->write();
|
|
||||||
$this->assertEquals(1, $page->Version);
|
|
||||||
|
|
||||||
$page->Content = "<p>This is a test</p>";
|
|
||||||
$page->write();
|
|
||||||
$this->assertEquals(2, $page->Version);
|
|
||||||
|
|
||||||
$diff = $page->compareVersions(1, 2);
|
|
||||||
|
|
||||||
$processedContent = trim($diff->Content);
|
|
||||||
$processedContent = preg_replace('/\s*</','<',$processedContent);
|
|
||||||
$processedContent = preg_replace('/>\s*/','>',$processedContent);
|
|
||||||
$this->assertEquals("<ins><p>This is a test</p></ins>", $processedContent);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
function testAuthorIDAndPublisherIDFilledOutOnPublish() {
|
|
||||||
// Ensure that we have a member ID who is doing all this work
|
|
||||||
$member = Member::currentUser();
|
|
||||||
if($member) {
|
|
||||||
$memberID = $member->ID;
|
|
||||||
} else {
|
|
||||||
$memberID = $this->idFromFixture("Member", "admin");
|
|
||||||
Session::set("loggedInAs", $memberID);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write the page
|
|
||||||
$about = $this->objFromFixture('Page','about');
|
|
||||||
$about->Title = "Another title";
|
|
||||||
$about->write();
|
|
||||||
|
|
||||||
// Check the version created
|
|
||||||
$savedVersion = DB::query("SELECT \"AuthorID\", \"PublisherID\" FROM \"SiteTree_versions\"
|
|
||||||
WHERE \"RecordID\" = $about->ID ORDER BY \"Version\" DESC")->first();
|
|
||||||
$this->assertEquals($memberID, $savedVersion['AuthorID']);
|
|
||||||
$this->assertEquals(0, $savedVersion['PublisherID']);
|
|
||||||
|
|
||||||
// Publish the page
|
|
||||||
$about->doPublish();
|
|
||||||
$publishedVersion = DB::query("SELECT \"AuthorID\", \"PublisherID\" FROM \"SiteTree_versions\"
|
|
||||||
WHERE \"RecordID\" = $about->ID ORDER BY \"Version\" DESC")->first();
|
|
||||||
|
|
||||||
// Check the version created
|
|
||||||
$this->assertEquals($memberID, $publishedVersion['AuthorID']);
|
|
||||||
$this->assertEquals($memberID, $publishedVersion['PublisherID']);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testLinkShortcodeHandler() {
|
|
||||||
$aboutPage = $this->objFromFixture('Page', 'about');
|
|
||||||
$errorPage = $this->objFromFixture('ErrorPage', '404');
|
|
||||||
|
|
||||||
$parser = new ShortcodeParser();
|
|
||||||
$parser->register('sitetree_link', array('SiteTree', 'link_shortcode_handler'));
|
|
||||||
|
|
||||||
$aboutShortcode = sprintf('[sitetree_link id=%d]', $aboutPage->ID);
|
|
||||||
$aboutEnclosed = sprintf('[sitetree_link id=%d]Example Content[/sitetree_link]', $aboutPage->ID);
|
|
||||||
|
|
||||||
$aboutShortcodeExpected = $aboutPage->Link();
|
|
||||||
$aboutEnclosedExpected = sprintf('<a href="%s">Example Content</a>', $aboutPage->Link());
|
|
||||||
|
|
||||||
$this->assertEquals($aboutShortcodeExpected, $parser->parse($aboutShortcode), 'Test that simple linking works.');
|
|
||||||
$this->assertEquals($aboutEnclosedExpected, $parser->parse($aboutEnclosed), 'Test enclosed content is linked.');
|
|
||||||
|
|
||||||
$aboutPage->delete();
|
|
||||||
|
|
||||||
$this->assertEquals($aboutShortcodeExpected, $parser->parse($aboutShortcode), 'Test that deleted pages still link.');
|
|
||||||
$this->assertEquals($aboutEnclosedExpected, $parser->parse($aboutEnclosed));
|
|
||||||
|
|
||||||
$aboutShortcode = '[sitetree_link id="-1"]';
|
|
||||||
$aboutEnclosed = '[sitetree_link id="-1"]Example Content[/sitetree_link]';
|
|
||||||
|
|
||||||
$aboutShortcodeExpected = $errorPage->Link();
|
|
||||||
$aboutEnclosedExpected = sprintf('<a href="%s">Example Content</a>', $errorPage->Link());
|
|
||||||
|
|
||||||
$this->assertEquals($aboutShortcodeExpected, $parser->parse($aboutShortcode), 'Test link to 404 page if no suitable matches.');
|
|
||||||
$this->assertEquals($aboutEnclosedExpected, $parser->parse($aboutEnclosed));
|
|
||||||
|
|
||||||
$this->assertEquals('', $parser->parse('[sitetree_link]'), 'Test that invalid ID attributes are not parsed.');
|
|
||||||
$this->assertEquals('', $parser->parse('[sitetree_link id="text"]'));
|
|
||||||
$this->assertEquals('', $parser->parse('[sitetree_link]Example Content[/sitetree_link]'));
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testIsCurrent() {
|
|
||||||
$aboutPage = $this->objFromFixture('Page', 'about');
|
|
||||||
$errorPage = $this->objFromFixture('ErrorPage', '404');
|
|
||||||
|
|
||||||
Director::set_current_page($aboutPage);
|
|
||||||
$this->assertTrue($aboutPage->isCurrent(), 'Assert that basic isSection checks works.');
|
|
||||||
$this->assertFalse($errorPage->isCurrent());
|
|
||||||
|
|
||||||
Director::set_current_page($errorPage);
|
|
||||||
$this->assertTrue($errorPage->isCurrent(), 'Assert isSection works on error pages.');
|
|
||||||
$this->assertFalse($aboutPage->isCurrent());
|
|
||||||
|
|
||||||
Director::set_current_page($aboutPage);
|
|
||||||
$this->assertTrue (
|
|
||||||
DataObject::get_one('SiteTree', '"Title" = \'About Us\'')->isCurrent(),
|
|
||||||
'Assert that isCurrent works on another instance with the same ID.'
|
|
||||||
);
|
|
||||||
|
|
||||||
Director::set_current_page($newPage = new SiteTree());
|
|
||||||
$this->assertTrue($newPage->isCurrent(), 'Assert that isCurrent works on unsaved pages.');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testIsSection() {
|
|
||||||
$about = $this->objFromFixture('Page', 'about');
|
|
||||||
$staff = $this->objFromFixture('Page', 'staff');
|
|
||||||
$ceo = $this->objFromFixture('Page', 'ceo');
|
|
||||||
|
|
||||||
Director::set_current_page($about);
|
|
||||||
$this->assertTrue($about->isSection());
|
|
||||||
$this->assertFalse($staff->isSection());
|
|
||||||
$this->assertFalse($ceo->isSection());
|
|
||||||
|
|
||||||
Director::set_current_page($staff);
|
|
||||||
$this->assertTrue($about->isSection());
|
|
||||||
$this->assertTrue($staff->isSection());
|
|
||||||
$this->assertFalse($ceo->isSection());
|
|
||||||
|
|
||||||
Director::set_current_page($ceo);
|
|
||||||
$this->assertTrue($about->isSection());
|
|
||||||
$this->assertTrue($staff->isSection());
|
|
||||||
$this->assertTrue($ceo->isSection());
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @covers SiteTree::validURLSegment
|
|
||||||
*/
|
|
||||||
public function testValidURLSegmentURLSegmentConflicts() {
|
|
||||||
$sitetree = new SiteTree();
|
|
||||||
SiteTree::disable_nested_urls();
|
|
||||||
|
|
||||||
$sitetree->URLSegment = 'home';
|
|
||||||
$this->assertFalse($sitetree->validURLSegment(), 'URLSegment conflicts are recognised');
|
|
||||||
$sitetree->URLSegment = 'home-noconflict';
|
|
||||||
$this->assertTrue($sitetree->validURLSegment());
|
|
||||||
|
|
||||||
$sitetree->ParentID = $this->idFromFixture('Page', 'about');
|
|
||||||
$sitetree->URLSegment = 'home';
|
|
||||||
$this->assertFalse($sitetree->validURLSegment(), 'Conflicts are still recognised with a ParentID value');
|
|
||||||
|
|
||||||
SiteTree::enable_nested_urls();
|
|
||||||
|
|
||||||
$sitetree->ParentID = 0;
|
|
||||||
$sitetree->URLSegment = 'home';
|
|
||||||
$this->assertFalse($sitetree->validURLSegment(), 'URLSegment conflicts are recognised');
|
|
||||||
|
|
||||||
$sitetree->ParentID = $this->idFromFixture('Page', 'about');
|
|
||||||
$this->assertTrue($sitetree->validURLSegment(), 'URLSegments can be the same across levels');
|
|
||||||
|
|
||||||
$sitetree->URLSegment = 'my-staff';
|
|
||||||
$this->assertFalse($sitetree->validURLSegment(), 'Nested URLSegment conflicts are recognised');
|
|
||||||
$sitetree->URLSegment = 'my-staff-noconflict';
|
|
||||||
$this->assertTrue($sitetree->validURLSegment());
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @covers SiteTree::validURLSegment
|
|
||||||
*/
|
|
||||||
public function testValidURLSegmentClassNameConflicts() {
|
|
||||||
$sitetree = new SiteTree();
|
|
||||||
$sitetree->URLSegment = 'Controller';
|
|
||||||
|
|
||||||
$this->assertFalse($sitetree->validURLSegment(), 'Class name conflicts are recognised');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @covers SiteTree::validURLSegment
|
|
||||||
*/
|
|
||||||
public function testValidURLSegmentControllerConflicts() {
|
|
||||||
SiteTree::enable_nested_urls();
|
|
||||||
|
|
||||||
$sitetree = new SiteTree();
|
|
||||||
$sitetree->ParentID = $this->idFromFixture('SiteTreeTest_Conflicted', 'parent');
|
|
||||||
|
|
||||||
$sitetree->URLSegment = 'index';
|
|
||||||
$this->assertFalse($sitetree->validURLSegment(), 'index is not a valid URLSegment');
|
|
||||||
|
|
||||||
$sitetree->URLSegment = 'conflicted-action';
|
|
||||||
$this->assertFalse($sitetree->validURLSegment(), 'allowed_actions conflicts are recognised');
|
|
||||||
|
|
||||||
$sitetree->URLSegment = 'conflicted-template';
|
|
||||||
$this->assertFalse($sitetree->validURLSegment(), 'Action-specific template conflicts are recognised');
|
|
||||||
|
|
||||||
$sitetree->URLSegment = 'valid';
|
|
||||||
$this->assertTrue($sitetree->validURLSegment(), 'Valid URLSegment values are allowed');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function testVersionsAreCreated() {
|
|
||||||
$p = new Page();
|
|
||||||
$p->Content = "one";
|
|
||||||
$p->write();
|
|
||||||
$this->assertEquals(1, $p->Version);
|
|
||||||
|
|
||||||
// No changes don't bump version
|
|
||||||
$p->write();
|
|
||||||
$this->assertEquals(1, $p->Version);
|
|
||||||
|
|
||||||
$p->Content = "two";
|
|
||||||
$p->write();
|
|
||||||
$this->assertEquals(2, $p->Version);
|
|
||||||
|
|
||||||
// Only change meta-data don't bump version
|
|
||||||
$p->HasBrokenLink = true;
|
|
||||||
$p->write();
|
|
||||||
$p->HasBrokenLink = false;
|
|
||||||
$p->write();
|
|
||||||
$this->assertEquals(2, $p->Version);
|
|
||||||
|
|
||||||
$p->Content = "three";
|
|
||||||
$p->write();
|
|
||||||
$this->assertEquals(3, $p->Version);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
function testPageTypeClasses() {
|
|
||||||
$classes = SiteTree::page_type_classes();
|
|
||||||
$this->assertNotContains('SiteTree', $classes, 'Page types do not include base class');
|
|
||||||
$this->assertContains('Page', $classes, 'Page types do contain subclasses');
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**#@+
|
|
||||||
* @ignore
|
|
||||||
*/
|
|
||||||
|
|
||||||
class SiteTreeTest_PageNode extends Page implements TestOnly { }
|
|
||||||
class SiteTreeTest_PageNode_Controller extends Page_Controller implements TestOnly {
|
|
||||||
}
|
|
||||||
|
|
||||||
class SiteTreeTest_Conflicted extends Page implements TestOnly { }
|
|
||||||
class SiteTreeTest_Conflicted_Controller extends Page_Controller implements TestOnly {
|
|
||||||
|
|
||||||
public static $allowed_actions = array (
|
|
||||||
'conflicted-action'
|
|
||||||
);
|
|
||||||
|
|
||||||
public function hasActionTemplate($template) {
|
|
||||||
if($template == 'conflicted-template') {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return parent::hasActionTemplate($template);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**#@-*/
|
|
@ -1,82 +0,0 @@
|
|||||||
Group:
|
|
||||||
editors:
|
|
||||||
Title: Editors
|
|
||||||
admins:
|
|
||||||
Title: Administrators
|
|
||||||
|
|
||||||
Permission:
|
|
||||||
admins:
|
|
||||||
Code: ADMIN
|
|
||||||
Group: =>Group.admins
|
|
||||||
editors:
|
|
||||||
Code: CMS_ACCESS_CMSMain
|
|
||||||
Group: =>Group.editors
|
|
||||||
|
|
||||||
Member:
|
|
||||||
editor:
|
|
||||||
FirstName: Test
|
|
||||||
Surname: Editor
|
|
||||||
Groups: =>Group.editors
|
|
||||||
admin:
|
|
||||||
FirstName: Test
|
|
||||||
Surname: Administrator
|
|
||||||
Groups: =>Group.admins
|
|
||||||
|
|
||||||
Page:
|
|
||||||
home:
|
|
||||||
Title: Home
|
|
||||||
CanEditType: OnlyTheseUsers
|
|
||||||
EditorGroups: =>Group.admins
|
|
||||||
about:
|
|
||||||
Title: About Us
|
|
||||||
CanEditType: OnlyTheseUsers
|
|
||||||
EditorGroups: =>Group.admins
|
|
||||||
staff:
|
|
||||||
Title: Staff
|
|
||||||
URLSegment: my-staff
|
|
||||||
Parent: =>Page.about
|
|
||||||
ceo:
|
|
||||||
Title: CEO
|
|
||||||
Parent: =>Page.staff
|
|
||||||
staffduplicate:
|
|
||||||
Title: Staff
|
|
||||||
URLSegment: my-staff
|
|
||||||
Parent: =>Page.about
|
|
||||||
products:
|
|
||||||
Title: Products
|
|
||||||
CanEditType: OnlyTheseUsers
|
|
||||||
EditorGroups: =>Group.editors
|
|
||||||
product1:
|
|
||||||
Title: 1.1 Test Product
|
|
||||||
Parent: =>Page.products
|
|
||||||
CanEditType: Inherit
|
|
||||||
product2:
|
|
||||||
Title: Another Product
|
|
||||||
Parent: =>Page.products
|
|
||||||
CanEditType: Inherit
|
|
||||||
product3:
|
|
||||||
Title: Another Product
|
|
||||||
Parent: =>Page.products
|
|
||||||
CanEditType: Inherit
|
|
||||||
product4:
|
|
||||||
Title: Another Product
|
|
||||||
Parent: =>Page.products
|
|
||||||
CanEditType: OnlyTheseUsers
|
|
||||||
EditorGroups: =>Group.admins
|
|
||||||
contact:
|
|
||||||
Title: Contact Us
|
|
||||||
object:
|
|
||||||
Title: Object
|
|
||||||
controller:
|
|
||||||
Title: Controller
|
|
||||||
numericonly:
|
|
||||||
Title: 1930
|
|
||||||
|
|
||||||
SiteTreeTest_Conflicted:
|
|
||||||
parent:
|
|
||||||
Title: Parent
|
|
||||||
|
|
||||||
ErrorPage:
|
|
||||||
404:
|
|
||||||
Title: Page not Found
|
|
||||||
ErrorCode: 404
|
|
@ -1,85 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage tests
|
|
||||||
*
|
|
||||||
* Note: Most of the permission-related SiteConfig tests are located in
|
|
||||||
* SiteTreePermissionsTest
|
|
||||||
*/
|
|
||||||
class SiteConfigTest extends SapphireTest {
|
|
||||||
|
|
||||||
static $fixture_file = 'sapphire/tests/model/SiteConfigTest.yml';
|
|
||||||
|
|
||||||
protected $requiredExtensions = array(
|
|
||||||
'SiteTree' => array('Translatable'),
|
|
||||||
'SiteConfig' => array('Translatable'),
|
|
||||||
);
|
|
||||||
|
|
||||||
protected $illegalExtensions = array(
|
|
||||||
'SiteTree' => array('SiteTreeSubsites')
|
|
||||||
);
|
|
||||||
|
|
||||||
private $origLocale;
|
|
||||||
|
|
||||||
function setUp() {
|
|
||||||
parent::setUp();
|
|
||||||
|
|
||||||
$this->origLocale = Translatable::default_locale();
|
|
||||||
Translatable::set_default_locale("en_US");
|
|
||||||
}
|
|
||||||
|
|
||||||
function tearDown() {
|
|
||||||
Translatable::set_default_locale($this->origLocale);
|
|
||||||
Translatable::set_current_locale($this->origLocale);
|
|
||||||
|
|
||||||
parent::tearDown();
|
|
||||||
}
|
|
||||||
|
|
||||||
function testCurrentCreatesDefaultForLocale() {
|
|
||||||
$configEn = SiteConfig::current_site_config();
|
|
||||||
$configFr = SiteConfig::current_site_config('fr_FR');
|
|
||||||
|
|
||||||
$this->assertType('SiteConfig', $configFr);
|
|
||||||
$this->assertEquals($configFr->Locale, 'fr_FR');
|
|
||||||
$this->assertEquals($configFr->Title, $configEn->Title, 'Copies title from existing config');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testCanEditTranslatedRootPages() {
|
|
||||||
$configEn = $this->objFromFixture('SiteConfig', 'en_US');
|
|
||||||
$configDe = $this->objFromFixture('SiteConfig', 'de_DE');
|
|
||||||
|
|
||||||
$pageEn = $this->objFromFixture('Page', 'root_en');
|
|
||||||
$pageDe = $pageEn->createTranslation('de_DE');
|
|
||||||
|
|
||||||
$translatorDe = $this->objFromFixture('Member', 'translator_de');
|
|
||||||
$translatorEn = $this->objFromFixture('Member', 'translator_en');
|
|
||||||
|
|
||||||
$this->assertFalse($pageEn->canEdit($translatorDe));
|
|
||||||
$this->assertTrue($pageEn->canEdit($translatorEn));
|
|
||||||
}
|
|
||||||
|
|
||||||
function testAvailableThemes() {
|
|
||||||
$config = SiteConfig::current_site_config();
|
|
||||||
$ds = DIRECTORY_SEPARATOR;
|
|
||||||
$testThemeBaseDir = TEMP_FOLDER . $ds . 'test-themes';
|
|
||||||
|
|
||||||
if(file_exists($testThemeBaseDir)) Filesystem::removeFolder($testThemeBaseDir);
|
|
||||||
mkdir($testThemeBaseDir);
|
|
||||||
mkdir($testThemeBaseDir . $ds . 'blackcandy');
|
|
||||||
mkdir($testThemeBaseDir . $ds . 'blackcandy_blog');
|
|
||||||
mkdir($testThemeBaseDir . $ds . 'darkshades');
|
|
||||||
mkdir($testThemeBaseDir . $ds . 'darkshades_blog');
|
|
||||||
|
|
||||||
$themes = $config->getAvailableThemes($testThemeBaseDir);
|
|
||||||
$this->assertContains('blackcandy', $themes, 'Test themes contain blackcandy theme');
|
|
||||||
$this->assertContains('darkshades', $themes, 'Test themes contain darkshades theme');
|
|
||||||
|
|
||||||
SiteConfig::disable_theme('darkshades');
|
|
||||||
$themes = $config->getAvailableThemes($testThemeBaseDir);
|
|
||||||
$this->assertFalse(in_array('darkshades', $themes), 'Darkshades was disabled - it is no longer available');
|
|
||||||
|
|
||||||
Filesystem::removeFolder($testThemeBaseDir);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
?>
|
|
@ -1,318 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
class VirtualPageTest extends SapphireTest {
|
|
||||||
static $fixture_file = 'sapphire/tests/model/VirtualPageTest.yml';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test that, after you update the source page of a virtual page, all the virtual pages
|
|
||||||
* are updated
|
|
||||||
*/
|
|
||||||
function testEditingSourcePageUpdatesVirtualPages() {
|
|
||||||
$master = $this->objFromFixture('Page', 'master');
|
|
||||||
$master->Title = "New title";
|
|
||||||
$master->MenuTitle = "New menutitle";
|
|
||||||
$master->Content = "<p>New content</p>";
|
|
||||||
$master->write();
|
|
||||||
|
|
||||||
$vp1 = $this->objFromFixture('VirtualPage', 'vp1');
|
|
||||||
$vp2 = $this->objFromFixture('VirtualPage', 'vp2');
|
|
||||||
|
|
||||||
$this->assertEquals("New title", $vp1->Title);
|
|
||||||
$this->assertEquals("New title", $vp2->Title);
|
|
||||||
$this->assertEquals("New menutitle", $vp1->MenuTitle);
|
|
||||||
$this->assertEquals("New menutitle", $vp2->MenuTitle);
|
|
||||||
$this->assertEquals("<p>New content</p>", $vp1->Content);
|
|
||||||
$this->assertEquals("<p>New content</p>", $vp2->Content);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test that, after you publish the source page of a virtual page, all the already published
|
|
||||||
* virtual pages are published
|
|
||||||
*/
|
|
||||||
function testPublishingSourcePagePublishesAlreadyPublishedVirtualPages() {
|
|
||||||
$this->logInWithPermission('ADMIN');
|
|
||||||
|
|
||||||
$master = $this->objFromFixture('Page', 'master');
|
|
||||||
$master->doPublish();
|
|
||||||
|
|
||||||
$master->Title = "New title";
|
|
||||||
$master->MenuTitle = "New menutitle";
|
|
||||||
$master->Content = "<p>New content</p>";
|
|
||||||
$master->write();
|
|
||||||
|
|
||||||
$vp1 = DataObject::get_by_id("VirtualPage", $this->idFromFixture('VirtualPage', 'vp1'));
|
|
||||||
$vp2 = DataObject::get_by_id("VirtualPage", $this->idFromFixture('VirtualPage', 'vp2'));
|
|
||||||
$this->assertTrue($vp1->doPublish());
|
|
||||||
$this->assertTrue($vp2->doPublish());
|
|
||||||
|
|
||||||
$master->doPublish();
|
|
||||||
|
|
||||||
Versioned::reading_stage("Live");
|
|
||||||
$vp1 = DataObject::get_by_id("VirtualPage", $this->idFromFixture('VirtualPage', 'vp1'));
|
|
||||||
$vp2 = DataObject::get_by_id("VirtualPage", $this->idFromFixture('VirtualPage', 'vp2'));
|
|
||||||
|
|
||||||
$this->assertNotNull($vp1);
|
|
||||||
$this->assertNotNull($vp2);
|
|
||||||
|
|
||||||
$this->assertEquals("New title", $vp1->Title);
|
|
||||||
$this->assertEquals("New title", $vp2->Title);
|
|
||||||
$this->assertEquals("New menutitle", $vp1->MenuTitle);
|
|
||||||
$this->assertEquals("New menutitle", $vp2->MenuTitle);
|
|
||||||
$this->assertEquals("<p>New content</p>", $vp1->Content);
|
|
||||||
$this->assertEquals("<p>New content</p>", $vp2->Content);
|
|
||||||
Versioned::reading_stage("Stage");
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test that virtual pages get the content from the master page when they are created.
|
|
||||||
*/
|
|
||||||
function testNewVirtualPagesGrabTheContentFromTheirMaster() {
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
$vp->write();
|
|
||||||
|
|
||||||
$vp->CopyContentFromID = $this->idFromFixture('Page', 'master');
|
|
||||||
$vp->write();
|
|
||||||
|
|
||||||
$this->assertEquals("My Page", $vp->Title);
|
|
||||||
$this->assertEquals("My Page Nav", $vp->MenuTitle);
|
|
||||||
|
|
||||||
$vp->CopyContentFromID = $this->idFromFixture('Page', 'master2');
|
|
||||||
$vp->write();
|
|
||||||
|
|
||||||
$this->assertEquals("My Other Page", $vp->Title);
|
|
||||||
$this->assertEquals("My Other Page Nav", $vp->MenuTitle);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Virtual pages are always supposed to chose the same content as the published source page.
|
|
||||||
* This means that when you publish them, they should show the published content of the source
|
|
||||||
* page, not the draft content at the time when you clicked 'publish' in the CMS.
|
|
||||||
*/
|
|
||||||
function testPublishingAVirtualPageCopiedPublishedContentNotDraftContent() {
|
|
||||||
$p = new Page();
|
|
||||||
$p->Content = "published content";
|
|
||||||
$p->write();
|
|
||||||
$p->doPublish();
|
|
||||||
|
|
||||||
// Don't publish this change - published page will still say 'published content'
|
|
||||||
$p->Content = "draft content";
|
|
||||||
$p->write();
|
|
||||||
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
$vp->CopyContentFromID = $p->ID;
|
|
||||||
$vp->write();
|
|
||||||
|
|
||||||
$vp->doPublish();
|
|
||||||
|
|
||||||
// The draft content of the virtual page should say 'draft content'
|
|
||||||
$this->assertEquals('draft content',
|
|
||||||
DB::query('SELECT "Content" from "SiteTree" WHERE "ID" = ' . $vp->ID)->value());
|
|
||||||
|
|
||||||
// The published content of the virtual page should say 'published content'
|
|
||||||
$this->assertEquals('published content',
|
|
||||||
DB::query('SELECT "Content" from "SiteTree_Live" WHERE "ID" = ' . $vp->ID)->value());
|
|
||||||
}
|
|
||||||
|
|
||||||
function testCantPublishVirtualPagesBeforeTheirSource() {
|
|
||||||
// An unpublished source page
|
|
||||||
$p = new Page();
|
|
||||||
$p->Content = "test content";
|
|
||||||
$p->write();
|
|
||||||
|
|
||||||
// With no source page, we can't publish
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
$vp->write();
|
|
||||||
$this->assertFalse($vp->canPublish());
|
|
||||||
|
|
||||||
// When the source page isn't published, we can't publish
|
|
||||||
$vp->CopyContentFromID = $p->ID;
|
|
||||||
$vp->write();
|
|
||||||
$this->assertFalse($vp->canPublish());
|
|
||||||
|
|
||||||
// Once the source page gets published, then we can publish
|
|
||||||
$p->doPublish();
|
|
||||||
$this->assertTrue($vp->canPublish());
|
|
||||||
}
|
|
||||||
|
|
||||||
function testCanDeleteOrphanedVirtualPagesFromLive() {
|
|
||||||
// An unpublished source page
|
|
||||||
$p = new Page();
|
|
||||||
$p->Content = "test content";
|
|
||||||
$p->write();
|
|
||||||
$p->doPublish();
|
|
||||||
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
$vp->CopyContentFromID = $p->ID;
|
|
||||||
$vp->write();
|
|
||||||
|
|
||||||
// Delete the source page
|
|
||||||
$this->assertTrue($vp->canPublish());
|
|
||||||
$this->assertTrue($p->doDeleteFromLive());
|
|
||||||
|
|
||||||
// Confirm that we can unpublish, but not publish
|
|
||||||
$this->assertTrue($vp->canDeleteFromLive());
|
|
||||||
$this->assertFalse($vp->canPublish());
|
|
||||||
|
|
||||||
// Confirm that the action really works
|
|
||||||
$this->assertTrue($vp->doDeleteFromLive());
|
|
||||||
$this->assertNull(DB::query("SELECT \"ID\" FROM \"SiteTree_Live\" WHERE \"ID\" = $vp->ID")->value());
|
|
||||||
}
|
|
||||||
|
|
||||||
function testVirtualPagesArentInappropriatelyPublished() {
|
|
||||||
// Fixture
|
|
||||||
$p = new Page();
|
|
||||||
$p->Content = "test content";
|
|
||||||
$p->write();
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
$vp->CopyContentFromID = $p->ID;
|
|
||||||
$vp->write();
|
|
||||||
|
|
||||||
// VP is oragne
|
|
||||||
$this->assertTrue($vp->IsAddedToStage);
|
|
||||||
|
|
||||||
// VP is still orange after we publish
|
|
||||||
$p->doPublish();
|
|
||||||
$this->fixVersionNumberCache($vp);
|
|
||||||
$this->assertTrue($vp->IsAddedToStage);
|
|
||||||
|
|
||||||
// A new VP created after P's initial construction
|
|
||||||
$vp2 = new VirtualPage();
|
|
||||||
$vp2->CopyContentFromID = $p->ID;
|
|
||||||
$vp2->write();
|
|
||||||
$this->assertTrue($vp2->IsAddedToStage);
|
|
||||||
|
|
||||||
// Also remains orange after a republish
|
|
||||||
$p->Content = "new content";
|
|
||||||
$p->write();
|
|
||||||
$p->doPublish();
|
|
||||||
$this->fixVersionNumberCache($vp2);
|
|
||||||
$this->assertTrue($vp2->IsAddedToStage);
|
|
||||||
|
|
||||||
// VP is now published
|
|
||||||
$vp->doPublish();
|
|
||||||
|
|
||||||
$this->fixVersionNumberCache($vp);
|
|
||||||
$this->assertTrue($vp->ExistsOnLive);
|
|
||||||
$this->assertFalse($vp->IsModifiedOnStage);
|
|
||||||
|
|
||||||
// P edited, VP and P both go green
|
|
||||||
$p->Content = "third content";
|
|
||||||
$p->write();
|
|
||||||
|
|
||||||
$this->fixVersionNumberCache($vp, $p);
|
|
||||||
$this->assertTrue($p->IsModifiedOnStage);
|
|
||||||
$this->assertTrue($vp->IsModifiedOnStage);
|
|
||||||
|
|
||||||
// Publish, VP goes black
|
|
||||||
$p->doPublish();
|
|
||||||
$this->fixVersionNumberCache($vp);
|
|
||||||
$this->assertTrue($vp->ExistsOnLive);
|
|
||||||
$this->assertFalse($vp->IsModifiedOnStage);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testVirtualPagesCreateVersionRecords() {
|
|
||||||
$source = $this->objFromFixture('Page', 'master');
|
|
||||||
$source->Title = "T0";
|
|
||||||
$source->write();
|
|
||||||
$source->doPublish();
|
|
||||||
|
|
||||||
// Creating a new VP to ensure that Version #s are out of alignment
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
$vp->CopyContentFromID = $source->ID;
|
|
||||||
$vp->write();
|
|
||||||
|
|
||||||
$source->Title = "T1";
|
|
||||||
$source->write();
|
|
||||||
$source->Title = "T2";
|
|
||||||
$source->write();
|
|
||||||
|
|
||||||
$this->assertEquals($vp->ID, DB::query("SELECT \"RecordID\" FROM \"SiteTree_versions\"
|
|
||||||
WHERE \"RecordID\" = $vp->ID AND \"Title\" = 'T1'")->value());
|
|
||||||
$this->assertEquals($vp->ID, DB::query("SELECT \"RecordID\" FROM \"SiteTree_versions\"
|
|
||||||
WHERE \"RecordID\" = $vp->ID AND \"Title\" = 'T2'")->value());
|
|
||||||
$this->assertEquals($vp->ID, DB::query("SELECT \"RecordID\" FROM \"SiteTree_versions\"
|
|
||||||
WHERE \"RecordID\" = $vp->ID AND \"Version\" = $vp->Version")->value());
|
|
||||||
|
|
||||||
$vp->doPublish();
|
|
||||||
|
|
||||||
// Check that the published content is copied from the published page, with a legal
|
|
||||||
// version
|
|
||||||
$liveVersion = DB::query("SELECT \"Version\" FROM \"SiteTree_Live\" WHERE \"ID\" = $vp->ID")->value();
|
|
||||||
|
|
||||||
$this->assertEquals("T0", DB::query("SELECT \"Title\" FROM \"SiteTree_Live\"
|
|
||||||
WHERE \"ID\" = $vp->ID")->value());
|
|
||||||
|
|
||||||
// SiteTree_Live.Version should reference a legal entry in SiteTree_versions for the
|
|
||||||
// virtual page
|
|
||||||
$this->assertEquals("T0", DB::query("SELECT \"Title\" FROM \"SiteTree_versions\"
|
|
||||||
WHERE \"RecordID\" = $vp->ID AND \"Version\" = $liveVersion")->value());
|
|
||||||
}
|
|
||||||
|
|
||||||
function fixVersionNumberCache($page) {
|
|
||||||
$pages = func_get_args();
|
|
||||||
foreach($pages as $p) {
|
|
||||||
Versioned::prepopulate_versionnumber_cache('SiteTree', 'Stage', array($p->ID));
|
|
||||||
Versioned::prepopulate_versionnumber_cache('SiteTree', 'Live', array($p->ID));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function testUnpublishingSourcePageOfAVirtualPageAlsoUnpublishesVirtualPage() {
|
|
||||||
// Create page and virutal page
|
|
||||||
$p = new Page();
|
|
||||||
$p->Title = "source";
|
|
||||||
$p->write();
|
|
||||||
$this->assertTrue($p->doPublish());
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
$vp->CopyContentFromID = $p->ID;
|
|
||||||
$vp->write();
|
|
||||||
$this->assertTrue($vp->doPublish());
|
|
||||||
|
|
||||||
// All is fine, the virtual page doesn't have a broken link
|
|
||||||
$this->assertFalse($vp->HasBrokenLink);
|
|
||||||
|
|
||||||
// Unpublish the source page, confirm that the virtual page has also been unpublished
|
|
||||||
$p->doUnpublish();
|
|
||||||
$vpLive = Versioned::get_one_by_stage('SiteTree', 'Live', '"SiteTree"."ID" = ' . $vp->ID);
|
|
||||||
$this->assertFalse($vpLive);
|
|
||||||
|
|
||||||
// Delete from draft, confirm that the virtual page has a broken link on the draft site
|
|
||||||
$p->delete();
|
|
||||||
$vp->flushCache();
|
|
||||||
$vp = DataObject::get_by_id('SiteTree', $vp->ID);
|
|
||||||
$this->assertEquals(1, $vp->HasBrokenLink);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testDeletingFromLiveSourcePageOfAVirtualPageAlsoUnpublishesVirtualPage() {
|
|
||||||
// Create page and virutal page
|
|
||||||
$p = new Page();
|
|
||||||
$p->Title = "source";
|
|
||||||
$p->write();
|
|
||||||
$this->assertTrue($p->doPublish());
|
|
||||||
$vp = new VirtualPage();
|
|
||||||
$vp->CopyContentFromID = $p->ID;
|
|
||||||
$vp->write();
|
|
||||||
$this->assertTrue($vp->doPublish());
|
|
||||||
|
|
||||||
// All is fine, the virtual page doesn't have a broken link
|
|
||||||
$this->assertFalse($vp->HasBrokenLink);
|
|
||||||
|
|
||||||
// Delete the source page from draft, confirm that this creates a broken link
|
|
||||||
$pID = $p->ID;
|
|
||||||
$p->delete();
|
|
||||||
$vp->flushCache();
|
|
||||||
$vp = DataObject::get_by_id('SiteTree', $vp->ID);
|
|
||||||
$this->assertEquals(1, $vp->HasBrokenLink);
|
|
||||||
|
|
||||||
// Delete the source page form live, confirm that the virtual page has also been unpublished
|
|
||||||
$pLive = Versioned::get_one_by_stage('SiteTree', 'Live', '"SiteTree"."ID" = ' . $pID);
|
|
||||||
$this->assertTrue($pLive->doDeleteFromLive());
|
|
||||||
$vpLive = Versioned::get_one_by_stage('SiteTree', 'Live', '"SiteTree"."ID" = ' . $vp->ID);
|
|
||||||
$this->assertFalse($vpLive);
|
|
||||||
|
|
||||||
// Delete from draft, confirm that the virtual page has a broken link on the draft site
|
|
||||||
$pLive->delete();
|
|
||||||
$vp->flushCache();
|
|
||||||
$vp = DataObject::get_by_id('SiteTree', $vp->ID);
|
|
||||||
$this->assertEquals(1, $vp->HasBrokenLink);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,17 +0,0 @@
|
|||||||
Page:
|
|
||||||
master:
|
|
||||||
Title: My Page
|
|
||||||
MenuTitle: My Page Nav
|
|
||||||
master2:
|
|
||||||
Title: My Other Page
|
|
||||||
MenuTitle: My Other Page Nav
|
|
||||||
holder:
|
|
||||||
Title: Sub-pages
|
|
||||||
VirtualPage:
|
|
||||||
vp1:
|
|
||||||
CopyContentFrom: =>Page.master
|
|
||||||
Parent: =>Page.holder
|
|
||||||
vp2:
|
|
||||||
CopyContentFrom: =>Page.master
|
|
||||||
Parent: =>Page.holder
|
|
||||||
|
|
@ -1,105 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* <h2>Fixture tree</h2>
|
|
||||||
* <code>
|
|
||||||
* parent1_published
|
|
||||||
* child1_1_published
|
|
||||||
* grandchild1_1_1
|
|
||||||
* grandchild1_1_2_published
|
|
||||||
* grandchild1_1_3_orphaned
|
|
||||||
* grandchild1_1_4_orphaned_published
|
|
||||||
* child1_2_published
|
|
||||||
* child1_3_orphaned
|
|
||||||
* child1_4_orphaned_published
|
|
||||||
* parent2
|
|
||||||
* child2_1_published_orphaned // is orphaned because parent is not published
|
|
||||||
* </code>
|
|
||||||
*
|
|
||||||
* <h2>Cleaned up tree</h2>
|
|
||||||
* <code>
|
|
||||||
* parent1_published
|
|
||||||
* child1_1_published
|
|
||||||
* grandchild1_1_1
|
|
||||||
* grandchild1_1_2_published
|
|
||||||
* child2_1_published_orphaned
|
|
||||||
* parent2
|
|
||||||
* </code>
|
|
||||||
*
|
|
||||||
* @author Ingo Schommer (<firstname>@silverstripe.com), SilverStripe Ltd.
|
|
||||||
*
|
|
||||||
* @package sapphire
|
|
||||||
* @subpackage tests
|
|
||||||
*/
|
|
||||||
class RemoveOrphanedPagesTaskTest extends FunctionalTest {
|
|
||||||
|
|
||||||
static $fixture_file = 'sapphire/tests/tasks/RemoveOrphanedPagesTaskTest.yml';
|
|
||||||
|
|
||||||
static $use_draft_site = false;
|
|
||||||
|
|
||||||
function setUp() {
|
|
||||||
parent::setUp();
|
|
||||||
|
|
||||||
$parent1_published = $this->objFromFixture('Page', 'parent1_published');
|
|
||||||
$parent1_published->publish('Stage', 'Live');
|
|
||||||
|
|
||||||
$child1_1_published = $this->objFromFixture('Page', 'child1_1_published');
|
|
||||||
$child1_1_published->publish('Stage', 'Live');
|
|
||||||
|
|
||||||
$child1_2_published = $this->objFromFixture('Page', 'child1_2_published');
|
|
||||||
$child1_2_published->publish('Stage', 'Live');
|
|
||||||
|
|
||||||
$child1_3_orphaned = $this->objFromFixture('Page', 'child1_3_orphaned');
|
|
||||||
$child1_3_orphaned->ParentID = 9999;
|
|
||||||
$child1_3_orphaned->write();
|
|
||||||
|
|
||||||
$child1_4_orphaned_published = $this->objFromFixture('Page', 'child1_4_orphaned_published');
|
|
||||||
$child1_4_orphaned_published->ParentID = 9999;
|
|
||||||
$child1_4_orphaned_published->write();
|
|
||||||
$child1_4_orphaned_published->publish('Stage', 'Live');
|
|
||||||
|
|
||||||
$grandchild1_1_2_published = $this->objFromFixture('Page', 'grandchild1_1_2_published');
|
|
||||||
$grandchild1_1_2_published->publish('Stage', 'Live');
|
|
||||||
|
|
||||||
$grandchild1_1_3_orphaned = $this->objFromFixture('Page', 'grandchild1_1_3_orphaned');
|
|
||||||
$grandchild1_1_3_orphaned->ParentID = 9999;
|
|
||||||
$grandchild1_1_3_orphaned->write();
|
|
||||||
|
|
||||||
$grandchild1_1_4_orphaned_published = $this->objFromFixture('Page',
|
|
||||||
'grandchild1_1_4_orphaned_published'
|
|
||||||
);
|
|
||||||
$grandchild1_1_4_orphaned_published->ParentID = 9999;
|
|
||||||
$grandchild1_1_4_orphaned_published->write();
|
|
||||||
$grandchild1_1_4_orphaned_published->publish('Stage', 'Live');
|
|
||||||
|
|
||||||
$child2_1_published_orphaned = $this->objFromFixture('Page', 'child2_1_published_orphaned');
|
|
||||||
$child2_1_published_orphaned->publish('Stage', 'Live');
|
|
||||||
}
|
|
||||||
|
|
||||||
function testGetOrphansByStage() {
|
|
||||||
// all orphans
|
|
||||||
$child1_3_orphaned = $this->objFromFixture('Page', 'child1_3_orphaned');
|
|
||||||
$child1_4_orphaned_published = $this->objFromFixture('Page', 'child1_4_orphaned_published');
|
|
||||||
$grandchild1_1_3_orphaned = $this->objFromFixture('Page', 'grandchild1_1_3_orphaned');
|
|
||||||
$grandchild1_1_4_orphaned_published = $this->objFromFixture('Page',
|
|
||||||
'grandchild1_1_4_orphaned_published'
|
|
||||||
);
|
|
||||||
$child2_1_published_orphaned = $this->objFromFixture('Page', 'child2_1_published_orphaned');
|
|
||||||
|
|
||||||
$task = singleton('RemoveOrphanedPagesTask');
|
|
||||||
$orphans = $task->getOrphanedPages();
|
|
||||||
$orphanIDs = $orphans->column('ID');
|
|
||||||
sort($orphanIDs);
|
|
||||||
$compareIDs = array(
|
|
||||||
$child1_3_orphaned->ID,
|
|
||||||
$child1_4_orphaned_published->ID,
|
|
||||||
$grandchild1_1_3_orphaned->ID,
|
|
||||||
$grandchild1_1_4_orphaned_published->ID,
|
|
||||||
$child2_1_published_orphaned->ID
|
|
||||||
);
|
|
||||||
sort($compareIDs);
|
|
||||||
|
|
||||||
$this->assertEquals($orphanIDs, $compareIDs);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
?>
|
|
@ -1,32 +0,0 @@
|
|||||||
Page:
|
|
||||||
parent1_published:
|
|
||||||
Title: Parent1
|
|
||||||
child1_1_published:
|
|
||||||
Title: Child1.1
|
|
||||||
Parent: =>Page.parent1_published
|
|
||||||
child1_2_published:
|
|
||||||
Title: Child1.2
|
|
||||||
Parent: =>Page.parent1_published
|
|
||||||
child1_3_orphaned:
|
|
||||||
Title: Child1.3
|
|
||||||
Parent: =>Page.parent1_published
|
|
||||||
child1_4_orphaned_published:
|
|
||||||
Title: Child1.4
|
|
||||||
Parent: =>Page.parent1_published
|
|
||||||
grandchild1_1_1:
|
|
||||||
Title: Grandchild1.1.1
|
|
||||||
Parent: =>Page.child1_1_published
|
|
||||||
grandchild1_1_2_published:
|
|
||||||
Title: Grandchild1.1.2
|
|
||||||
Parent: =>Page.child1_1_published
|
|
||||||
grandchild1_1_3_orphaned:
|
|
||||||
Title: Grandchild1.1.3
|
|
||||||
Parent: =>Page.child1_1_published
|
|
||||||
grandchild1_1_4_orphaned_published:
|
|
||||||
Title: Grandchild1.1.4
|
|
||||||
Parent: =>Page.child1_1_published
|
|
||||||
parent2:
|
|
||||||
Title: Parent2
|
|
||||||
child2_1_published_orphaned:
|
|
||||||
Title: Child2.1
|
|
||||||
Parent: =>Page.parent2
|
|
Loading…
x
Reference in New Issue
Block a user