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); | |
| } | |
| } | |