Skip to content

Commit

Permalink
NEW Added dev/task to initialise DataObjects for existing datasets
Browse files Browse the repository at this point in the history
New task is based on the original task built for SiteTree but is now extendable for any DataObject enabled with Fluent. To prevent duplicate code, the original task now extends the new task.
  • Loading branch information
nathanbrauer committed Apr 2, 2024
1 parent 9e650e6 commit 9aea31d
Show file tree
Hide file tree
Showing 2 changed files with 308 additions and 87 deletions.
287 changes: 287 additions & 0 deletions src/Task/InitialDataObjectLocalisationTask.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,287 @@
<?php

namespace TractorCow\Fluent\Task;

use SilverStripe\Control\Director;
use SilverStripe\Control\HTTPRequest;
use SilverStripe\Core\ClassInfo;
use SilverStripe\Dev\BuildTask;
use SilverStripe\ORM\DataObject;
use SilverStripe\Versioned\Versioned;
use TractorCow\Fluent\Extension\FluentExtension;
use TractorCow\Fluent\Extension\FluentVersionedExtension;
use TractorCow\Fluent\Model\Locale;
use TractorCow\Fluent\State\FluentState;

class InitialDataObjectLocalisationTask extends BuildTask
{
/**
* @var string
*/
private static $segment = 'initial-dataobject-localisation-task';

/**
* @var string
*/
protected $title = 'Initial DataObject localisation (excludes SiteTree)';

/**
* @var string
*/
protected $description = 'Intended for projects which already have data when Fluent module is added.' .
' This dev task will localise / publish all DataObjects in the default locale. Locale setup has to be done before running this task.' .
' Pass limit=N to limit number of records to localise. Pass publish=1 to enable publishing of localised Versioned DataObjects.' .
' Regardless, Versioned DataObjects which were not already published will not be published, only localised. DataObjects which were already localised will always be skipped.' .
' This class may be extended to create custom initialization tasks targeting or excluding specific classes.';

/**
* When extending this class, you may choose to include only these specific classes.
* Adding any classes here will disable `$this->exclude_classes`.
* @var string[]
*/
protected $include_only_classes = [];

/**
* When extending this class, you may choose to exclude these specific classes.
* This is IGNORED if `$this->include_only_classes` is not empty.
* @var string[]
*/
protected $exclude_classes = [
SiteTree::class
];

/**
* @param HTTPRequest $request
* @return void
* @throws \ReflectionException
* @throws \SilverStripe\ORM\ValidationException
*/
public function run($request)
{
if (!Director::is_cli()) {
echo '<pre>' . PHP_EOL;
}

$publish = (bool)$request->getVar('publish');
$limit = (int)$request->getVar('limit');

$total_results = [
'localisable' => 0,
'localised' => 0,
'publishable' => 0,
'published' => 0,
];

/** @var Locale $globalLocale */
$globalLocale = Locale::get()
->filter(['IsGlobalDefault' => 1])
->sort('ID', 'ASC')
->first();

if (!$globalLocale) {
echo 'Please set global locale first!' . PHP_EOL;

return;
}

if ($this->include_only_classes && is_array($this->include_only_classes)) {
$classesWithFluent = $this->include_only_classes;
foreach ($this->include_only_classes as $key => $dataClass) {
if (!$this->isClassNamePermitted($dataClass)) {
echo sprintf('ERROR: `%s` does not have FluentExtension installed. Continuing without it...', $dataClass) . PHP_EOL;
unset($classesWithFluent[$key]);
}
}
} else {
$dataClasses = static::getDirectSubclassesRecursivelyFor(DataObject::class);
$classesWithFluent = $this->filterPermittedClassesRecursively($dataClasses);
}

foreach ($classesWithFluent as $classWithFluent) {
if (!$this->isClassNamePermitted($classWithFluent)) {
continue;
}

$results = $this->doLocaliseClass($classWithFluent, $globalLocale, $limit, $publish);
foreach ($results as $key => $value) {
$total_results[$key] += $value;
}

echo sprintf('Processing %s objects...', $classWithFluent) . PHP_EOL;
echo sprintf('└─ Localised %d of %d objects.', $results['localised'], $results['localisable']) . PHP_EOL;
if ($results['publishable']) {
echo sprintf('└─ Published %d of %d objects.', $results['published'], $results['publishable']) . PHP_EOL;
}
}

echo PHP_EOL;
echo sprintf('Completed %d classes.', count($classesWithFluent)) . PHP_EOL;
echo sprintf('└─ Localised %d of %d objects in total.', $total_results['localised'], $total_results['localisable']) . PHP_EOL;
echo PHP_EOL;

if ($total_results['publishable']) {
echo sprintf('└─ Published %d of %d objects in total.', $total_results['published'], $total_results['publishable']) . PHP_EOL;
echo PHP_EOL;
}

if (!Director::is_cli()) {
echo '</pre>';
}
}

/**
* @param $className
* @param $globalLocale
* @param $limit
* @param $publish
* @return array{localisable: int, localised: int, publishable: int, published: int}
* @throws \SilverStripe\ORM\ValidationException
*/
protected function doLocaliseClass($className, $globalLocale, $limit, $publish): array
{
$dataObjectIDs = FluentState::singleton()->withState(static function (FluentState $state) use ($className, $limit): array {
$state->setLocale(null);
$dataObjects = $className::get()->sort('ID', 'ASC');

if ($limit > 0) {
$dataObjects = $dataObjects->limit($limit);
}

return $dataObjects->column('ID');
});

return FluentState::singleton()->withState(
static function (FluentState $state) use ($className, $globalLocale, $publish, $dataObjectIDs): array {
$state->setLocale($globalLocale->Locale);
$return = [
'localisable' => 0,
'localised' => 0,
'publishable' => 0,
'published' => 0,
];

foreach ($dataObjectIDs as $dataObjectID) {
/** @var DataObject|FluentExtension $dataObject */
$dataObject = $className::get()->byID($dataObjectID);
$return['localisable'] += 1;

if (!$dataObject->hasExtension(FluentVersionedExtension::class)) {
if ($dataObject->existsInLocale()) {
continue;
}
$dataObject->write();
$return['localised'] += 1;
continue;
}

// We have versioned data, so start tracking how many have been published
$return['publishable'] += 1;

/** @var DataObject|Versioned|FluentVersionedExtension $dataObject */
if ($dataObject->isDraftedInLocale()) {
continue;
}
$dataObject->writeToStage(Versioned::DRAFT);

$return['localised'] += 1;

if (!$publish) {
continue;
}

// Check if the base record was published - if not then we don't need to publish
// as this would leak draft content, we only want to publish pages which were published
// before Fluent module was added
$dataObjectID = $dataObject->ID;
$isBaseRecordPublished = FluentState::singleton()->withState(
static function (FluentState $state) use ($className, $dataObjectID): bool {
$state->setLocale(null);
$page = $className::get_by_id($dataObjectID);

if ($page === null) {
return false;
}

return $page->isPublished();
}
);

if (!$isBaseRecordPublished) {
continue;
}

$dataObject->publishRecursive();
$return['published'] += 1;
}

return $return;
}
);
}

/**
* @param string $className
* @return array[]
* @throws \ReflectionException
*/
protected static function getDirectSubclassesRecursivelyFor(string $className): array
{
$directSubclasses = [];
foreach (ClassInfo::subclassesFor($className, false) as $subclassName) {
$actualParentClass = get_parent_class($subclassName);
if ($className === $actualParentClass) {
$directSubclasses[$subclassName] = static::getDirectSubclassesRecursivelyFor($subclassName);
}
}

return $directSubclasses;
}

/**
* @param array $classes
* @return array
*/
protected function filterPermittedClassesRecursively(array $classes): array
{
$permittedClasses = [];
foreach ($classes as $parentClassName => $subclassNames) {
if ($this->isClassNamePermitted($parentClassName)) {
$permittedClasses[] = $parentClassName;
// We will skip all subclasses since the ORM will automatically
// pull them in when this parent is referenced
continue;
}

$permittedClasses = array_merge($permittedClasses, $this->filterPermittedClassesRecursively($subclassNames));
}

return $permittedClasses;
}

/**
* @param string $className
* @return bool
*/
protected function isClassNamePermitted(string $className): bool
{
// Do a simple (inexpensive) text comparison against the exclusion list before we create an object
if (!$this->include_only_classes && is_array($this->exclude_classes) && in_array($className, $this->exclude_classes)) {
return false;
}

/** @var DataObject $dataObject */
$dataObject = singleton($className);

// Now we'll do a full comparison against the exclusion list
// This important step will, for example, match (refuse) a BlogPost if Page is listed as excluded
if (is_array($this->exclude_classes)) {
foreach ($this->exclude_classes as $excluded_class) {
if ($dataObject instanceof $excluded_class) {
return false;
}
}
}

return $dataObject->hasExtension(FluentExtension::class);
}
}
Loading

0 comments on commit 9aea31d

Please sign in to comment.