mirror of
https://github.com/silverstripe/silverstripe-cms
synced 2024-10-22 06:05:56 +00:00
bbc3aaaf9f
The main benefit of this is so that authors who make use of .editorconfig don't end up with whitespace changes in their PRs. Spaces vs. tabs has been left alone, although that could do with a tidy-up in SS4 after the switch to PSR-1/2. The command used was this: for match in '*.ss' '*.css' '*.scss' '*.html' '*.yml' '*.php' '*.js' '*.csv' '*.inc' '*.php5'; do find . -path ./thirdparty -prune -o -type f -name "$match" -exec sed -i '' 's/[[:space:]]\+$//' {} \+ find . -path ./thirdparty -prune -o -type f -name "$match" | xargs perl -pi -e 's/ +$//' done
338 lines
9.9 KiB
PHP
338 lines
9.9 KiB
PHP
<?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 cms
|
|
* @subpackage tasks
|
|
*/
|
|
//class RemoveOrphanedPagesTask extends BuildTask {
|
|
class RemoveOrphanedPagesTask extends Controller {
|
|
|
|
private 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';
|
|
|
|
public function Link() {
|
|
return $this->class;
|
|
}
|
|
|
|
public function init() {
|
|
parent::init();
|
|
|
|
if(!Permission::check('ADMIN')) {
|
|
return Security::permissionFailure($this);
|
|
}
|
|
}
|
|
|
|
public function index() {
|
|
Requirements::javascript(FRAMEWORK_DIR . '/thirdparty/jquery/jquery.js');
|
|
Requirements::customCSS('#OrphanIDs .middleColumn {width: auto;}');
|
|
Requirements::customCSS('#OrphanIDs label {display: inline;}');
|
|
|
|
return $this->renderWith('BlankPage');
|
|
}
|
|
|
|
public function Form() {
|
|
$fields = new FieldList();
|
|
$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);
|
|
$orphanBaseClass = ClassInfo::baseDataClass($this->orphanedSearchClass);
|
|
$stageRecord = Versioned::get_one_by_stage(
|
|
$this->orphanedSearchClass,
|
|
'Stage',
|
|
array("\"$orphanBaseClass\".\"ID\"" => $orphan->ID)
|
|
);
|
|
$liveRecord = Versioned::get_one_by_stage(
|
|
$this->orphanedSearchClass,
|
|
'Live',
|
|
array("\"$orphanBaseClass\".\"ID\"" => $orphan->ID)
|
|
);
|
|
$label = sprintf(
|
|
'<a href="admin/pages/edit/show/%d">%s</a> <small>(#%d, Last Modified Date: %s, Last Modifier: %s, %s)</small>',
|
|
$orphan->ID,
|
|
$orphan->Title,
|
|
$orphan->ID,
|
|
Date::create($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 FieldList(
|
|
new FormAction('doSubmit', _t('RemoveOrphanedPagesTask.BUTTONRUN', 'Run'))
|
|
)
|
|
);
|
|
|
|
if(!$orphans || !$orphans->Count()) {
|
|
$form->makeReadonly();
|
|
}
|
|
|
|
return $form;
|
|
}
|
|
|
|
public function run($request) {
|
|
// @todo Merge with BuildTask functionality
|
|
}
|
|
|
|
public 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();
|
|
$orphanBaseClass = ClassInfo::baseDataClass($this->orphanedSearchClass);
|
|
foreach($orphanIDs as $id) {
|
|
$stageRecord = Versioned::get_one_by_stage(
|
|
$this->orphanedSearchClass,
|
|
'Stage',
|
|
array("\"$orphanBaseClass\".\"ID\"" => $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',
|
|
array("\"$orphanBaseClass\".\"ID\"" => $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();
|
|
$orphanBaseClass = ClassInfo::baseDataClass($this->orphanedSearchClass);
|
|
foreach($orphanIDs as $id) {
|
|
$stageRecord = Versioned::get_one_by_stage(
|
|
$this->orphanedSearchClass,
|
|
'Stage',
|
|
array("\"$orphanBaseClass\".\"ID\"" => $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',
|
|
array("\"$orphanBaseClass\".\"ID\"" => $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 array $filter
|
|
* @param string $sort
|
|
* @param string $join
|
|
* @param int|array $limit
|
|
* @return SS_List
|
|
*/
|
|
public function getOrphanedPages($class = 'SiteTree', $filter = array(), $sort = null, $join = null, $limit = null) {
|
|
// Alter condition
|
|
if(empty($filter)) $where = array();
|
|
elseif(is_array($filter)) $where = $filter;
|
|
else $where = array($filter);
|
|
$where[] = array("\"$class\".\"ParentID\" != ?" => 0);
|
|
$where[] = '"Parents"."ID" IS NULL';
|
|
|
|
$orphans = new ArrayList();
|
|
foreach(array('Stage', 'Live') as $stage) {
|
|
$joinByStage = $join;
|
|
$table = $class;
|
|
$table .= ($stage == 'Live') ? '_Live' : '';
|
|
$stageOrphans = Versioned::get_by_stage(
|
|
$class,
|
|
$stage,
|
|
$where,
|
|
$sort,
|
|
null,
|
|
$limit
|
|
)->leftJoin($table, "\"$table\".\"ParentID\" = \"Parents\".\"ID\"", "Parents");
|
|
$orphans->merge($stageOrphans);
|
|
}
|
|
|
|
$orphans->removeDuplicates();
|
|
|
|
return $orphans;
|
|
}
|
|
}
|