| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250 |
- <?php
- namespace MailPoet\Subscribers;
- if (!defined('ABSPATH')) exit;
- use MailPoet\Entities\FormEntity;
- use MailPoet\Form\FormsRepository;
- use MailPoet\Form\Util\FieldNameObfuscator;
- use MailPoet\NotFoundException;
- use MailPoet\Settings\SettingsController;
- use MailPoet\Statistics\StatisticsFormsRepository;
- use MailPoet\Subscription\Captcha;
- use MailPoet\Subscription\CaptchaSession;
- use MailPoet\Subscription\SubscriptionUrlFactory;
- use MailPoet\Subscription\Throttling as SubscriptionThrottling;
- use MailPoet\UnexpectedValueException;
- use MailPoet\WP\Functions as WPFunctions;
- class SubscriberSubscribeController {
- /** @var FormsRepository */
- private $formsRepository;
- /** @var Captcha */
- private $subscriptionCaptcha;
- /** @var CaptchaSession */
- private $captchaSession;
- /** @var SubscriptionUrlFactory */
- private $subscriptionUrlFactory;
- /** @var FieldNameObfuscator */
- private $fieldNameObfuscator;
- /** @var SettingsController */
- private $settings;
- /** @var RequiredCustomFieldValidator */
- private $requiredCustomFieldValidator;
- /** @var SubscriberActions */
- private $subscriberActions;
- /** @var WPFunctions */
- private $wp;
- /** @var SubscriptionThrottling */
- private $throttling;
- /** @var StatisticsFormsRepository */
- private $statisticsFormsRepository;
- public function __construct(
- Captcha $subscriptionCaptcha,
- CaptchaSession $captchaSession,
- SubscriberActions $subscriberActions,
- SubscriptionUrlFactory $subscriptionUrlFactory,
- SubscriptionThrottling $throttling,
- FieldNameObfuscator $fieldNameObfuscator,
- RequiredCustomFieldValidator $requiredCustomFieldValidator,
- SettingsController $settings,
- FormsRepository $formsRepository,
- StatisticsFormsRepository $statisticsFormsRepository,
- WPFunctions $wp
- ) {
- $this->formsRepository = $formsRepository;
- $this->subscriptionCaptcha = $subscriptionCaptcha;
- $this->captchaSession = $captchaSession;
- $this->subscriptionUrlFactory = $subscriptionUrlFactory;
- $this->requiredCustomFieldValidator = $requiredCustomFieldValidator;
- $this->fieldNameObfuscator = $fieldNameObfuscator;
- $this->settings = $settings;
- $this->subscriberActions = $subscriberActions;
- $this->wp = $wp;
- $this->throttling = $throttling;
- $this->statisticsFormsRepository = $statisticsFormsRepository;
- }
- public function subscribe(array $data): array {
- $form = $this->getForm($data);
- if (!empty($data['email'])) {
- throw new UnexpectedValueException(__('Please leave the first field empty.', 'mailpoet'));
- }
- $captchaSettings = $this->settings->get('captcha');
- $data = $this->initCaptcha($captchaSettings, $form, $data);
- $data = $this->deobfuscateFormPayload($data);
- try {
- $this->requiredCustomFieldValidator->validate($data, $form);
- } catch (\Exception $e) {
- throw new UnexpectedValueException($e->getMessage());
- }
- $segmentIds = $this->getSegmentIds($form, $data['segments'] ?? []);
- unset($data['segments']);
- $meta = $this->validateCaptcha($captchaSettings, $data);
- if (isset($meta['error'])) {
- return $meta;
- }
- // only accept fields defined in the form
- $formFieldIds = array_filter(array_map(function (array $formField): ?string {
- if (!isset($formField['id'])) {
- return null;
- }
- return is_numeric($formField['id']) ? "cf_{$formField['id']}" : $formField['id'];
- }, $form->getBlocksByTypes(FormEntity::FORM_FIELD_TYPES)));
- $data = array_intersect_key($data, array_flip($formFieldIds));
- // make sure we don't allow too many subscriptions with the same ip address
- $timeout = $this->throttling->throttle();
- if ($timeout > 0) {
- $timeToWait = $this->throttling->secondsToTimeString($timeout);
- $meta['refresh_captcha'] = true;
- $meta['error'] = sprintf(__('You need to wait %s before subscribing again.', 'mailpoet'), $timeToWait);
- return $meta;
- }
- $subscriber = $this->subscriberActions->subscribe($data, $segmentIds);
- if (!empty($captchaSettings['type']) && $captchaSettings['type'] === Captcha::TYPE_BUILTIN) {
- // Captcha has been verified, invalidate the session vars
- $this->captchaSession->reset();
- }
- // record form statistics
- $this->statisticsFormsRepository->record($form, $subscriber);
- $formSettings = $form->getSettings();
- if (!empty($formSettings['on_success'])) {
- if ($formSettings['on_success'] === 'page') {
- // redirect to a page on a success, pass the page url in the meta
- $meta['redirect_url'] = $this->wp->getPermalink($formSettings['success_page']);
- } else if ($formSettings['on_success'] === 'url') {
- $meta['redirect_url'] = $formSettings['success_url'];
- }
- }
- return $meta;
- }
- private function deobfuscateFormPayload($data): array {
- return $this->fieldNameObfuscator->deobfuscateFormPayload($data);
- }
- private function initCaptcha(?array $captchaSettings, FormEntity $form, array $data): array {
- if (!$captchaSettings || !isset($captchaSettings['type'])) {
- return $data;
- }
- if ($captchaSettings['type'] === Captcha::TYPE_BUILTIN) {
- $captchaSessionId = isset($data['captcha_session_id']) ? $data['captcha_session_id'] : null;
- $this->captchaSession->init($captchaSessionId);
- if (!isset($data['captcha'])) {
- // Save form data to session
- $this->captchaSession->setFormData(array_merge($data, ['form_id' => $form->getId()]));
- } elseif ($this->captchaSession->getFormData()) {
- // Restore form data from session
- $data = array_merge($this->captchaSession->getFormData(), ['captcha' => $data['captcha']]);
- }
- // Otherwise use the post data
- }
- return $data;
- }
- private function validateCaptcha($captchaSettings, $data): array {
- if (empty($captchaSettings['type'])) {
- return [];
- }
- $meta = [];
- $isBuiltinCaptchaRequired = false;
- if ($captchaSettings['type'] === Captcha::TYPE_BUILTIN) {
- $isBuiltinCaptchaRequired = $this->subscriptionCaptcha->isRequired(isset($data['email']) ? $data['email'] : '');
- if ($isBuiltinCaptchaRequired && empty($data['captcha'])) {
- $meta['redirect_url'] = $this->subscriptionUrlFactory->getCaptchaUrl($this->captchaSession->getId());
- $meta['error'] = __('Please fill in the CAPTCHA.', 'mailpoet');
- return $meta;
- }
- }
- if ($captchaSettings['type'] === Captcha::TYPE_RECAPTCHA && empty($data['recaptcha'])) {
- return ['error' => __('Please check the CAPTCHA.', 'mailpoet')];
- }
- if ($captchaSettings['type'] === Captcha::TYPE_RECAPTCHA) {
- $response = empty($data['recaptcha']) ? $data['recaptcha-no-js'] : $data['recaptcha'];
- $response = $this->wp->wpRemotePost('https://www.google.com/recaptcha/api/siteverify', [
- 'body' => [
- 'secret' => $captchaSettings['recaptcha_secret_token'],
- 'response' => $response,
- ],
- ]);
- if (is_wp_error($response)) {
- return ['error' => __('Error while validating the CAPTCHA.', 'mailpoet')];
- }
- $response = json_decode(wp_remote_retrieve_body($response));
- if (empty($response->success)) {
- return ['error' => __('Error while validating the CAPTCHA.', 'mailpoet')];
- }
- } elseif ($captchaSettings['type'] === Captcha::TYPE_BUILTIN && $isBuiltinCaptchaRequired) {
- $captchaHash = $this->captchaSession->getCaptchaHash();
- if (empty($captchaHash)) {
- $meta['error'] = __('Please regenerate the CAPTCHA.', 'mailpoet');
- } elseif (!hash_equals(strtolower($data['captcha']), strtolower($captchaHash))) {
- $this->captchaSession->setCaptchaHash(null);
- $meta['refresh_captcha'] = true;
- $meta['error'] = __('The characters entered do not match with the previous CAPTCHA.', 'mailpoet');
- }
- }
- return $meta;
- }
- private function getSegmentIds(FormEntity $form, array $segmentIds): array {
- // If form contains segment selection blocks allow only segments ids configured in those blocks
- $segmentBlocksSegmentIds = $form->getSegmentBlocksSegmentIds();
- if (!empty($segmentBlocksSegmentIds)) {
- $segmentIds = array_intersect($segmentIds, $segmentBlocksSegmentIds);
- } else {
- $segmentIds = $form->getSettingsSegmentIds();
- }
- if (empty($segmentIds)) {
- throw new UnexpectedValueException(__('Please select a list.', 'mailpoet'));
- }
- return $segmentIds;
- }
- private function getForm(array $data): FormEntity {
- $formId = (isset($data['form_id']) ? (int)$data['form_id'] : false);
- $form = $this->formsRepository->findOneById($formId);
- if (!$form) {
- throw new NotFoundException(__('Please specify a valid form ID.', 'mailpoet'));
- }
- return $form;
- }
- }
|