silverstripe-framework/admin/code/CMSBatchActionHandler.php

312 lines
8.5 KiB
PHP
Raw Normal View History

<?php
2016-08-11 11:14:02 +12:00
namespace SilverStripe\Admin;
use SilverStripe\Control\Controller;
2016-09-09 18:43:05 +12:00
use SilverStripe\Control\HTTPRequest;
use SilverStripe\Control\HTTPResponse;
use SilverStripe\Control\RequestHandler;
use SilverStripe\Core\Config\Config;
use SilverStripe\ORM\ArrayList;
use SilverStripe\ORM\DB;
2016-06-23 11:37:22 +12:00
use SilverStripe\ORM\SS_List;
use SilverStripe\ORM\Versioning\Versioned;
use SilverStripe\ORM\DataObject;
2016-06-23 11:37:22 +12:00
use SilverStripe\Security\SecurityToken;
use SilverStripe\View\ArrayData;
2016-08-11 11:14:02 +12:00
use InvalidArgumentException;
use Translatable;
/**
* Special request handler for admin/batchaction
*/
class CMSBatchActionHandler extends RequestHandler {
2014-08-15 18:53:05 +12:00
/** @config */
private static $batch_actions = array();
2014-08-15 18:53:05 +12:00
private static $url_handlers = array(
'$BatchAction/applicablepages' => 'handleApplicablePages',
'$BatchAction/confirmation' => 'handleConfirmation',
'$BatchAction' => 'handleBatchAction',
);
private static $allowed_actions = array(
'handleBatchAction',
'handleApplicablePages',
'handleConfirmation',
);
2014-08-15 18:53:05 +12:00
2016-06-23 11:37:22 +12:00
/**
* @var Controller
*/
protected $parentController;
2014-08-15 18:53:05 +12:00
/**
* @var String
*/
protected $urlSegment;
2014-08-15 18:53:05 +12:00
/**
* @var String $recordClass The classname that should be affected
* by any batch changes. Needs to be set in the actual {@link CMSBatchAction}
* implementations as well.
*/
protected $recordClass = 'SilverStripe\\CMS\\Model\\SiteTree';
2014-08-15 18:53:05 +12:00
/**
2016-06-23 11:37:22 +12:00
* @param Controller $parentController
* @param string $urlSegment
* @param string $recordClass
*/
public function __construct($parentController, $urlSegment, $recordClass = null) {
$this->parentController = $parentController;
$this->urlSegment = $urlSegment;
if($recordClass) {
$this->recordClass = $recordClass;
}
parent::__construct();
}
/**
* Register a new batch action. Each batch action needs to be represented by a subclass
* of {@link CMSBatchAction}.
2014-08-15 18:53:05 +12:00
*
2016-06-23 11:37:22 +12:00
* @param string $urlSegment The URL Segment of the batch action - the URL used to process this
* action will be admin/batchactions/(urlSegment)
2016-06-23 11:37:22 +12:00
* @param string $batchActionClass The name of the CMSBatchAction subclass to register
* @param string $recordClass
*/
public static function register($urlSegment, $batchActionClass, $recordClass = 'SilverStripe\\CMS\\Model\\SiteTree') {
2016-08-11 11:14:02 +12:00
if(is_subclass_of($batchActionClass, 'SilverStripe\\Admin\\CMSBatchAction')) {
Config::inst()->update(
2016-08-11 11:14:02 +12:00
'SilverStripe\\Admin\\CMSBatchActionHandler',
'batch_actions',
array(
$urlSegment => array(
'class' => $batchActionClass,
'recordClass' => $recordClass
)
)
);
} else {
user_error("CMSBatchActionHandler::register() - Bad class '$batchActionClass'", E_USER_ERROR);
}
}
2014-08-15 18:53:05 +12:00
public function Link() {
return Controller::join_links($this->parentController->Link(), $this->urlSegment);
}
/**
* Invoke a batch action
*
2016-09-09 18:43:05 +12:00
* @param HTTPRequest $request
* @return HTTPResponse
*/
public function handleBatchAction($request) {
// This method can't be called without ajax.
if(!$request->isAjax()) {
return $this->parentController->redirectBack();
}
2014-08-15 18:53:05 +12:00
// Protect against CSRF on destructive action
if(!SecurityToken::inst()->checkRequest($request)) {
return $this->httpError(400);
}
// Find the action handler
$action = $request->param('BatchAction');
$actionHandler = $this->actionByName($action);
2014-08-15 18:53:05 +12:00
// Sanitise ID list and query the database for apges
$csvIDs = $request->requestVar('csvIDs');
$ids = $this->cleanIDs($csvIDs);
2014-08-15 18:53:05 +12:00
// Filter ids by those which are applicable to this action
// Enforces front end filter in LeftAndMain.BatchActions.js:refreshSelected
$ids = $actionHandler->applicablePages($ids);
2014-08-15 18:53:05 +12:00
// Query ids and pass to action to process
$pages = $this->getPages($ids);
return $actionHandler->run($pages);
2014-08-15 18:53:05 +12:00
}
/**
* Respond with the list of applicable pages for a given filter
*
2016-09-09 18:43:05 +12:00
* @param HTTPRequest $request
* @return HTTPResponse
*/
public function handleApplicablePages($request) {
// Find the action handler
$action = $request->param('BatchAction');
$actionHandler = $this->actionByName($action);
// Sanitise ID list and query the database for apges
$csvIDs = $request->requestVar('csvIDs');
$ids = $this->cleanIDs($csvIDs);
2014-08-15 18:53:05 +12:00
// Filter by applicable pages
if($ids) {
$applicableIDs = $actionHandler->applicablePages($ids);
} else {
$applicableIDs = array();
}
2014-08-15 18:53:05 +12:00
2016-09-09 18:43:05 +12:00
$response = new HTTPResponse(json_encode($applicableIDs));
$response->addHeader("Content-type", "application/json");
return $response;
}
2014-08-15 18:53:05 +12:00
/**
* Check if this action has a confirmation step
*
2016-09-09 18:43:05 +12:00
* @param HTTPRequest $request
* @return HTTPResponse
*/
public function handleConfirmation($request) {
// Find the action handler
$action = $request->param('BatchAction');
$actionHandler = $this->actionByName($action);
// Sanitise ID list and query the database for apges
$csvIDs = $request->requestVar('csvIDs');
$ids = $this->cleanIDs($csvIDs);
2014-08-15 18:53:05 +12:00
// Check dialog
if($actionHandler->hasMethod('confirmationDialog')) {
2016-09-09 18:43:05 +12:00
$response = new HTTPResponse(json_encode($actionHandler->confirmationDialog($ids)));
} else {
2016-09-09 18:43:05 +12:00
$response = new HTTPResponse(json_encode(array('alert' => false)));
}
2014-08-15 18:53:05 +12:00
$response->addHeader("Content-type", "application/json");
return $response;
}
2014-08-15 18:53:05 +12:00
/**
* Get an action for a given name
*
* @param string $name Name of the action
* @return CMSBatchAction An instance of the given batch action
* @throws InvalidArgumentException if invalid action name is passed.
*/
protected function actionByName($name) {
// Find the action handler
$actions = $this->batchActions();
if(!isset($actions[$name]['class'])) {
throw new InvalidArgumentException("Invalid batch action: {$name}");
}
return $this->buildAction($actions[$name]['class']);
}
/**
* Return a SS_List of ArrayData objects containing the following pieces of info
* about each batch action:
* - Link
* - Title
*
* @return ArrayList
*/
public function batchActionList() {
$actions = $this->batchActions();
$actionList = new ArrayList();
2014-08-15 18:53:05 +12:00
foreach($actions as $urlSegment => $action) {
$actionObj = $this->buildAction($action['class']);
if($actionObj->canView()) {
$actionDef = new ArrayData(array(
"Link" => Controller::join_links($this->Link(), $urlSegment),
"Title" => $actionObj->getActionTitle(),
));
$actionList->push($actionDef);
}
}
2014-08-15 18:53:05 +12:00
return $actionList;
}
2014-08-15 18:53:05 +12:00
/**
* Safely generate batch action object for a given classname
*
* @param string $class Class name to check
* @return CMSBatchAction An instance of the given batch action
* @throws InvalidArgumentException if invalid action class is passed.
*/
protected function buildAction($class) {
2016-08-11 11:14:02 +12:00
if(!is_subclass_of($class, 'SilverStripe\\Admin\\CMSBatchAction')) {
throw new InvalidArgumentException("{$class} is not a valid subclass of CMSBatchAction");
}
return CMSBatchAction::singleton($class);
}
/**
* Sanitise ID list from string input
*
* @param string $csvIDs
* @return array List of IDs as ints
*/
protected function cleanIDs($csvIDs) {
$ids = preg_split('/ *, */', trim($csvIDs));
foreach($ids as $k => $id) {
$ids[$k] = (int)$id;
}
return array_filter($ids);
}
/**
* Get all registered actions through the static defaults set by {@link register()}.
* Filters for the currently set {@link recordClass}.
2014-08-15 18:53:05 +12:00
*
* @return array See {@link register()} for the returned format.
*/
public function batchActions() {
$actions = $this->config()->batch_actions;
$recordClass = $this->recordClass;
$actions = array_filter($actions, function($action) use ($recordClass) {
return $action['recordClass'] === $recordClass;
});
return $actions;
}
/**
* Safely query and return all pages queried
*
* @param array $ids
* @return SS_List
*/
protected function getPages($ids) {
// Check empty set
if(empty($ids)) {
return new ArrayList();
}
2014-08-15 18:53:05 +12:00
$recordClass = $this->recordClass;
// Bypass translatable filter
if(class_exists('Translatable') && $recordClass::has_extension('Translatable')) {
Translatable::disable_locale_filter();
}
// Bypass versioned filter
if($recordClass::has_extension('SilverStripe\\ORM\\Versioning\\Versioned')) {
// Workaround for get_including_deleted not supporting byIDs filter very well
// Ensure we select both stage / live records
$pages = Versioned::get_including_deleted($recordClass, array(
'"RecordID" IN ('.DB::placeholders($ids).')' => $ids
));
} else {
$pages = DataObject::get($recordClass)->byIDs($ids);
}
if(class_exists('Translatable') && $recordClass::has_extension('Translatable')) {
Translatable::enable_locale_filter();
}
return $pages;
}
}