Spaces:
No application file
No application file
namespace Mautic\FormBundle\Controller\Api; | |
use Doctrine\Persistence\ManagerRegistry; | |
use Mautic\ApiBundle\Controller\CommonApiController; | |
use Mautic\ApiBundle\Helper\EntityResultHelper; | |
use Mautic\CoreBundle\Entity\CommonEntity; | |
use Mautic\CoreBundle\Factory\MauticFactory; | |
use Mautic\CoreBundle\Factory\ModelFactory; | |
use Mautic\CoreBundle\Helper\AppVersion; | |
use Mautic\CoreBundle\Helper\CoreParametersHelper; | |
use Mautic\CoreBundle\Security\Permissions\CorePermissions; | |
use Mautic\CoreBundle\Translation\Translator; | |
use Mautic\FormBundle\Entity\Action; | |
use Mautic\FormBundle\Entity\Field; | |
use Mautic\FormBundle\Entity\Form; | |
use Mautic\FormBundle\Model\ActionModel; | |
use Mautic\FormBundle\Model\FieldModel; | |
use Mautic\FormBundle\Model\FormModel; | |
use Symfony\Component\EventDispatcher\EventDispatcherInterface; | |
use Symfony\Component\Form\Exception\InvalidArgumentException; | |
use Symfony\Component\Form\FormFactoryInterface; | |
use Symfony\Component\Form\FormInterface; | |
use Symfony\Component\HttpFoundation\Request; | |
use Symfony\Component\HttpFoundation\RequestStack; | |
use Symfony\Component\HttpFoundation\Response; | |
use Symfony\Component\Routing\RouterInterface; | |
/** | |
* @extends CommonApiController<Form> | |
*/ | |
class FormApiController extends CommonApiController | |
{ | |
/** | |
* @var FormModel|null | |
*/ | |
protected $model; | |
public function __construct( | |
CorePermissions $security, | |
Translator $translator, | |
EntityResultHelper $entityResultHelper, | |
RouterInterface $router, | |
FormFactoryInterface $formFactory, | |
AppVersion $appVersion, | |
RequestStack $requestStack, | |
ManagerRegistry $doctrine, | |
ModelFactory $modelFactory, | |
EventDispatcherInterface $dispatcher, | |
CoreParametersHelper $coreParametersHelper, | |
MauticFactory $factory | |
) { | |
$formModel = $modelFactory->getModel('form'); | |
\assert($formModel instanceof FormModel); | |
$this->model = $formModel; | |
$this->entityClass = Form::class; | |
$this->entityNameOne = 'form'; | |
$this->entityNameMulti = 'forms'; | |
$this->serializerGroups = ['formDetails', 'categoryList', 'publishDetails']; | |
$this->dataInputMasks = [ | |
'text' => 'html', | |
'message' => 'html', | |
]; | |
parent::__construct($security, $translator, $entityResultHelper, $router, $formFactory, $appVersion, $requestStack, $doctrine, $modelFactory, $dispatcher, $coreParametersHelper, $factory); | |
} | |
/** | |
* Delete fields from a form. | |
* | |
* @return Response | |
*/ | |
public function deleteFieldsAction(Request $request, $formId) | |
{ | |
if (!$this->security->isGranted(['form:forms:editown', 'form:forms:editother'], 'MATCH_ONE')) { | |
return $this->accessDenied(); | |
} | |
$entity = $this->model->getEntity($formId); | |
if (null === $entity) { | |
return $this->notFound(); | |
} | |
$fieldsToDelete = $request->get('fields'); | |
if (!is_array($fieldsToDelete)) { | |
return $this->badRequest('The fields attribute must be array.'); | |
} | |
$this->model->deleteFields($entity, $fieldsToDelete); | |
$view = $this->view([$this->entityNameOne => $entity]); | |
return $this->handleView($view); | |
} | |
/** | |
* Delete fields from a form. | |
* | |
* @return Response | |
*/ | |
public function deleteActionsAction(Request $request, $formId) | |
{ | |
if (!$this->security->isGranted(['form:forms:editown', 'form:forms:editother'], 'MATCH_ONE')) { | |
return $this->accessDenied(); | |
} | |
$entity = $this->model->getEntity($formId); | |
if (null === $entity) { | |
return $this->notFound(); | |
} | |
$actionsToDelete = $request->get('actions'); | |
if (!is_array($actionsToDelete)) { | |
return $this->badRequest('The actions attribute must be array.'); | |
} | |
$this->model->deleteActions($entity, $actionsToDelete); | |
$view = $this->view([$this->entityNameOne => $entity]); | |
return $this->handleView($view); | |
} | |
protected function preSaveEntity(&$entity, $form, $parameters, $action = 'edit') | |
{ | |
$fieldModel = $this->getModel('form.field'); | |
\assert($fieldModel instanceof FieldModel); | |
$actionModel = $this->getModel('form.action'); | |
\assert($actionModel instanceof ActionModel); | |
$method = $this->getCurrentRequest()->getMethod(); | |
$isNew = false; | |
$alias = $entity->getAlias(); | |
if (empty($alias)) { | |
// Set clean alias to prevent SQL errors | |
$alias = $this->model->cleanAlias($entity->getName(), '', 10); | |
$entity->setAlias($alias); | |
} | |
// Set timestamps | |
$this->model->setTimestamps($entity, true, false); | |
$connection = $this->doctrine->getConnection(); | |
$connection->beginTransaction(); | |
if (!$entity->getId()) { | |
$isNew = true; | |
// Save the form first to get the form ID. | |
// Using the repository function to not trigger the listeners twice. | |
$this->model->getRepository()->saveEntity($entity); | |
} | |
$formId = $entity->getId(); | |
$requestFieldIds = []; | |
$requestActionIds = []; | |
$requestUsedAliases = []; | |
$currentFields = $entity->getFields(); | |
$currentActions = $entity->getActions(); | |
try { | |
// Add fields from the request | |
if (!empty($parameters['fields']) && is_array($parameters['fields'])) { | |
$aliases = $entity->getFieldAliases(); | |
foreach ($parameters['fields'] as &$fieldParams) { | |
if (empty($fieldParams['id'])) { | |
// Create an unique ID if not set - the following code requires one | |
$fieldParams['id'] = 'new'.hash('sha1', uniqid(mt_rand())); | |
/** @var ?Field $fieldEntity */ | |
$fieldEntity = $fieldModel->getEntity(); | |
} else { | |
/** @var ?Field $fieldEntity */ | |
$fieldEntity = $fieldModel->getEntity($fieldParams['id']); | |
$requestFieldIds[] = $fieldParams['id']; | |
} | |
if (is_null($fieldEntity)) { | |
$msg = $this->translator->trans( | |
'mautic.core.error.entity.not.found', | |
[ | |
'%entity%' => $this->translator->trans('mautic.form.field'), | |
'%id%' => $fieldParams['id'], | |
], | |
'flashes' | |
); | |
throw new InvalidArgumentException($msg, Response::HTTP_NOT_FOUND); | |
} | |
$fieldEntityArray = $fieldEntity->convertToArray(); | |
$fieldEntityArray['formId'] = $formId; | |
if (!empty($fieldParams['alias'])) { | |
$fieldParams['alias'] = $fieldModel->cleanAlias($fieldParams['alias'], 'f_', 25); | |
if (!in_array($fieldParams['alias'], $aliases)) { | |
$fieldEntityArray['alias'] = $fieldParams['alias']; | |
} | |
} | |
if (empty($fieldEntityArray['alias'])) { | |
$fieldEntityArray['alias'] = $fieldParams['alias'] = $fieldModel->generateAlias($fieldEntityArray['label'] ?? '', $aliases); | |
} | |
// Check that the alias is not already in use by another field | |
if (in_array($fieldEntityArray['alias'], $requestUsedAliases)) { | |
$msg = $this->translator->trans('mautic.form.field.alias.unique', ['%alias%' => $fieldEntityArray['alias']], 'validators'); | |
throw new InvalidArgumentException($msg, Response::HTTP_BAD_REQUEST); | |
} else { | |
$requestUsedAliases[] = $fieldEntityArray['alias']; | |
} | |
$fieldForm = $this->createFieldEntityForm($fieldEntityArray); | |
$fieldForm->submit($fieldParams, 'PATCH' !== $method); | |
if (!$fieldForm->isValid()) { | |
$formErrors = $this->getFormErrorMessages($fieldForm); | |
$msg = $this->getFormErrorMessage($formErrors); | |
throw new InvalidArgumentException($msg, Response::HTTP_BAD_REQUEST); | |
} | |
} | |
$this->model->setFields($entity, $parameters['fields']); | |
} | |
// Remove fields which weren't in the PUT request | |
if (!$isNew && 'PUT' === $method) { | |
$fieldsToDelete = []; | |
foreach ($currentFields as $currentField) { | |
if (!in_array($currentField->getId(), $requestFieldIds)) { | |
$fieldsToDelete[] = $currentField->getId(); | |
} | |
} | |
if ($fieldsToDelete) { | |
$this->model->deleteFields($entity, $fieldsToDelete); | |
} | |
} | |
// Add actions from the request | |
if (!empty($parameters['actions']) && is_array($parameters['actions'])) { | |
$actions = []; | |
foreach ($parameters['actions'] as &$actionParams) { | |
if (empty($actionParams['id'])) { | |
$actionParams['id'] = 'new'.hash('sha1', uniqid(mt_rand())); | |
$actionEntity = $actionModel->getEntity(); | |
} else { | |
$actionEntity = $actionModel->getEntity($actionParams['id']); | |
$requestActionIds[] = $actionParams['id']; | |
} | |
$actionEntity->setForm($entity); | |
$actionForm = $this->createActionEntityForm($actionEntity, $actionParams); | |
$actionForm->submit($actionParams, 'PATCH' !== $method); | |
if (!$actionForm->isValid()) { | |
$formErrors = $this->getFormErrorMessages($actionForm); | |
$msg = $this->getFormErrorMessage($formErrors); | |
throw new InvalidArgumentException($msg, Response::HTTP_BAD_REQUEST); | |
} | |
$actions[] = $actionForm->getNormData(); | |
} | |
// Save the form first and new actions so that new fields are available to actions. | |
// Using the repository function to not trigger the listeners twice. | |
$this->model->getRepository()->saveEntity($entity); | |
$this->model->setActions($entity, $actions); | |
} | |
$connection->commit(); | |
} catch (InvalidArgumentException $e) { | |
$connection->rollback(); | |
return $this->returnError($e->getMessage(), $e->getCode()); | |
} | |
// Remove actions which weren't in the PUT request | |
if (!$isNew && 'PUT' === $method) { | |
$actionsToDelete = []; | |
foreach ($currentActions as $currentAction) { | |
if (!in_array($currentAction->getId(), $requestActionIds)) { | |
$actionsToDelete[] = $currentAction->getId(); | |
} | |
} | |
if ($actionsToDelete) { | |
$this->model->deleteActions($entity, $actionsToDelete); | |
} | |
} | |
} | |
/** | |
* Creates the form instance. | |
* | |
* @return FormInterface<mixed> | |
*/ | |
protected function createActionEntityForm(Action $entity, array $action) | |
{ | |
/** @var FormModel $formModel */ | |
$formModel = $this->getModel('form'); | |
$components = $formModel->getCustomComponents(); | |
$type = $action['type'] ?? $entity->getType(); | |
$formActionModel = $this->getModel('form.action'); | |
\assert($formActionModel instanceof ActionModel); | |
return $formActionModel->createForm( | |
$entity, | |
$this->formFactory, | |
null, | |
[ | |
'csrf_protection' => false, | |
'allow_extra_fields' => true, | |
'settings' => $components['actions'][$type], | |
] | |
); | |
} | |
/** | |
* Creates the form instance. | |
* | |
* @return FormInterface<mixed> | |
*/ | |
protected function createFieldEntityForm($entity) | |
{ | |
$formFieldModel = $this->getModel('form.field'); | |
\assert($formFieldModel instanceof FieldModel); | |
return $formFieldModel->createForm( | |
$entity, | |
$this->formFactory, | |
null, | |
[ | |
'csrf_protection' => false, | |
'allow_extra_fields' => true, | |
] | |
); | |
} | |
/** | |
* @param CommonEntity $entity | |
* @param array<mixed> $parameters | |
* | |
* @return mixed | |
*/ | |
protected function processForm(Request $request, $entity, $parameters = null, $method = 'PUT') | |
{ | |
if (!isset($parameters['postAction'])) { | |
$parameters['postAction'] = 'return'; | |
} | |
return parent::processForm($request, $entity, $parameters, $method); | |
} | |
public function newEntityAction(Request $request): Response | |
{ | |
$parameters = $request->request->all(); | |
if (!isset($parameters['postAction'])) { | |
$request->request->add(['postAction' => 'return']); | |
} | |
return parent::newEntityAction($request); | |
} | |
} | |