2018-05-18 13:41:02 +12:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace SilverStripe\UserForms\Extension;
|
|
|
|
|
|
|
|
use SilverStripe\Control\Director;
|
|
|
|
use SilverStripe\Core\Injector\Injector;
|
2024-08-20 16:25:59 +12:00
|
|
|
use SilverStripe\Core\Extension;
|
2018-05-18 13:41:02 +12:00
|
|
|
use SilverStripe\ORM\DataList;
|
|
|
|
use SilverStripe\ORM\DataObject;
|
2024-09-23 14:40:23 +12:00
|
|
|
use SilverStripe\Core\Validation\ValidationException;
|
2018-05-18 13:41:02 +12:00
|
|
|
use SilverStripe\UserForms\Model\EditableFormField;
|
|
|
|
use SilverStripe\UserForms\Model\Recipient\EmailRecipient;
|
|
|
|
use SilverStripe\UserForms\Model\Submission\SubmittedForm;
|
|
|
|
use SilverStripe\UserForms\Model\UserDefinedForm;
|
|
|
|
use SilverStripe\UserForms\UserForm;
|
|
|
|
|
|
|
|
/**
|
2024-09-06 14:56:21 +12:00
|
|
|
* This extension provides a hook that runs when building the db which will check for existing data in various
|
2018-05-18 13:41:02 +12:00
|
|
|
* polymorphic relationship fields for userforms models, and ensure that the data is correct.
|
|
|
|
*
|
|
|
|
* Various `Parent` relationships in silverstripe/userforms for SilverStripe 3 were mapped directly to UserDefinedForm
|
|
|
|
* instances, and were made polymorphic in SilverStripe 4 (which also requires a class name). This means that a
|
|
|
|
* certain amount of manual checking is required to ensure that upgrades are performed smoothly.
|
|
|
|
*
|
|
|
|
* @internal This API is likely to be removed in later major versions of silverstripe/userforms
|
2024-01-19 10:29:08 +13:00
|
|
|
*
|
2024-08-20 16:25:59 +12:00
|
|
|
* @extends Extension<UserDefinedForm>
|
2018-05-18 13:41:02 +12:00
|
|
|
*/
|
2024-08-20 16:25:59 +12:00
|
|
|
class UpgradePolymorphicExtension extends Extension
|
2018-05-18 13:41:02 +12:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* A list of userforms classes that have had polymorphic relationships added in SilverStripe 4, and the fields
|
|
|
|
* on them that are polymorphic
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $targets = [
|
|
|
|
EditableFormField::class => ['ParentClass'],
|
|
|
|
EmailRecipient::class => ['FormClass'],
|
|
|
|
SubmittedForm::class => ['ParentClass'],
|
|
|
|
];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The default class name that will be used to replace values with
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $defaultReplacement = UserDefinedForm::class;
|
|
|
|
|
2024-08-27 15:39:30 +12:00
|
|
|
protected function onRequireDefaultRecords()
|
2018-05-18 13:41:02 +12:00
|
|
|
{
|
|
|
|
if (!UserDefinedForm::config()->get('upgrade_on_build')) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$updated = 0;
|
|
|
|
foreach ($this->targets as $className => $fieldNames) {
|
|
|
|
foreach ($fieldNames as $fieldName) {
|
|
|
|
/** @var DataList $list */
|
|
|
|
$list = $className::get();
|
|
|
|
|
|
|
|
foreach ($list as $entry) {
|
|
|
|
/** @var DataObject $relationshipObject */
|
|
|
|
$relationshipObject = Injector::inst()->get($entry->$fieldName);
|
|
|
|
if (!$relationshipObject) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the defined data class doesn't have the UserForm trait applied, it's probably wrong. Re-map
|
|
|
|
// it to a default value that does
|
|
|
|
$classTraits = class_uses($relationshipObject);
|
2022-04-13 13:52:56 +12:00
|
|
|
if (in_array(UserForm::class, $classTraits ?? [])) {
|
2018-05-18 13:41:02 +12:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-09-07 12:50:34 +02:00
|
|
|
// Don't rewrite class values when an existing value is set and is an instance of UserDefinedForm
|
|
|
|
if ($relationshipObject instanceof UserDefinedForm) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-05-18 13:41:02 +12:00
|
|
|
$entry->$fieldName = $this->defaultReplacement;
|
|
|
|
try {
|
|
|
|
$entry->write();
|
|
|
|
$updated++;
|
|
|
|
} catch (ValidationException $ex) {
|
2024-09-06 14:56:21 +12:00
|
|
|
// no-op, allow the rest of the db build to continue. There may be an error indicating that the
|
2018-05-18 13:41:02 +12:00
|
|
|
// object's class doesn't exist, which can be fixed by {@link DatabaseAdmin::doBuild} and this
|
2024-09-06 14:56:21 +12:00
|
|
|
// logic will work the next time the db is built.
|
2018-05-18 13:41:02 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($updated) {
|
|
|
|
$message = "Corrected {$updated} default polymorphic class names to {$this->defaultReplacement}";
|
|
|
|
if (Director::is_cli()) {
|
|
|
|
echo sprintf(" * %s\n", $message);
|
|
|
|
} else {
|
|
|
|
echo sprintf("<li>%s</li>\n", $message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|