Files
espocrm/application/Espo/Core/Hook/GeneralInvoker.php
Taras Machyshyn 63f67e0f66 2023 year
2023-01-03 16:26:52 +02:00

189 lines
6.1 KiB
PHP

<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2023 Yurii Kuznietsov, Taras Machyshyn, Oleksii Avramenko
* Website: https://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
*
* The interactive user interfaces in modified source and object code versions
* of this program must display Appropriate Legal Notices, as required under
* Section 5 of the GNU General Public License version 3.
*
* In accordance with Section 7(b) of the GNU General Public License version 3,
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
************************************************************************/
namespace Espo\Core\Hook;
use Espo\Core\Hook\Hook\AfterMassRelate;
use Espo\Core\Hook\Hook\AfterRelate;
use Espo\Core\Hook\Hook\AfterRemove;
use Espo\Core\Hook\Hook\AfterSave;
use Espo\Core\Hook\Hook\AfterUnrelate;
use Espo\Core\Hook\Hook\BeforeRemove;
use Espo\Core\Hook\Hook\BeforeSave;
use Espo\ORM\Entity;
use Espo\ORM\Query\Select;
use Espo\ORM\Repository\Option\MassRelateOptions;
use Espo\ORM\Repository\Option\RelateOptions;
use Espo\ORM\Repository\Option\RemoveOptions;
use Espo\ORM\Repository\Option\SaveOptions;
use Espo\ORM\Repository\Option\UnrelateOptions;
use LogicException;
/**
* Invokes hook methods, prepares needed arguments for specific hooks.
*/
class GeneralInvoker
{
private const HOOK_BEFORE_SAVE = 'beforeSave';
private const HOOK_AFTER_SAVE = 'afterSave';
private const HOOK_BEFORE_REMOVE = 'beforeRemove';
private const HOOK_AFTER_REMOVE = 'afterRemove';
private const HOOK_AFTER_RELATE = 'afterRelate';
private const HOOK_AFTER_UNRELATE = 'afterUnrelate';
private const HOOK_AFTER_MASS_RELATE = 'afterMassRelate';
/**
* @param object $hook A hook object.
* @param string $name A hook name.
* @param mixed $subject A subject.
* @param array<string, mixed> $options Options.
* @param array<string, mixed> $hookData Additional data.
*/
public function invoke(
object $hook,
string $name,
mixed $subject,
array $options,
array $hookData
): void {
if ($name === self::HOOK_BEFORE_SAVE && $hook instanceof BeforeSave) {
if (!$subject instanceof Entity) {
throw new LogicException();
}
$hook->beforeSave($subject, SaveOptions::fromAssoc($options));
return;
}
if ($name === self::HOOK_AFTER_SAVE && $hook instanceof AfterSave) {
if (!$subject instanceof Entity) {
throw new LogicException();
}
$hook->afterSave($subject, SaveOptions::fromAssoc($options));
return;
}
if ($name === self::HOOK_BEFORE_REMOVE && $hook instanceof BeforeRemove) {
if (!$subject instanceof Entity) {
throw new LogicException();
}
$hook->beforeRemove($subject, RemoveOptions::fromAssoc($options));
return;
}
if ($name === self::HOOK_AFTER_REMOVE && $hook instanceof AfterRemove) {
if (!$subject instanceof Entity) {
throw new LogicException();
}
$hook->afterRemove($subject, RemoveOptions::fromAssoc($options));
return;
}
if ($name === self::HOOK_AFTER_RELATE && $hook instanceof AfterRelate) {
$relationName = $hookData['relationName'] ?? null;
$relatedEntity = $hookData['foreignEntity'] ?? null;
$columnData = $hookData['relationData'] ?? [];
if (
!$subject instanceof Entity ||
!is_string($relationName) ||
!$relatedEntity instanceof Entity
) {
throw new LogicException();
}
$hook->afterRelate(
$subject,
$relationName,
$relatedEntity,
$columnData,
RelateOptions::fromAssoc($options)
);
return;
}
if ($name === self::HOOK_AFTER_UNRELATE && $hook instanceof AfterUnrelate) {
$relationName = $hookData['relationName'] ?? null;
$relatedEntity = $hookData['foreignEntity'] ?? null;
if (
!$subject instanceof Entity ||
!is_string($relationName) ||
!$relatedEntity instanceof Entity
) {
throw new LogicException();
}
$hook->afterUnrelate(
$subject,
$relationName,
$relatedEntity,
UnrelateOptions::fromAssoc($options)
);
return;
}
if ($name === self::HOOK_AFTER_MASS_RELATE && $hook instanceof AfterMassRelate) {
$relationName = $hookData['relationName'] ?? null;
$query = $hookData['query'] ?? null;
$columnData = $hookData['relationData'] ?? []; // Not implemented currently.
if (
!$subject instanceof Entity ||
!is_string($relationName) ||
!$query instanceof Select
) {
throw new LogicException();
}
$hook->afterMassRelate(
$subject,
$relationName,
$query,
$columnData,
MassRelateOptions::fromAssoc($options)
);
return;
}
$hook->$name($subject, $options, $hookData);
}
}