authorizedEmailsController = $authorizedEmailsController; $this->emoji = $emoji; $this->entityManager = $entityManager; $this->newslettersRepository = $newslettersRepository; $this->newsletterOptionsRepository = $newsletterOptionsRepository; $this->newsletterOptionFieldsRepository = $newsletterOptionFieldsRepository; $this->newsletterSegmentRepository = $newsletterSegmentRepository; $this->newsletterTemplatesRepository = $newsletterTemplatesRepository; $this->postNotificationScheduler = $postNotificationScheduler; $this->scheduledTasksRepository = $scheduledTasksRepository; $this->settings = $settings; $this->security = $security; $this->wp = $wp; $this->dataSanitizer = $dataSanitizer; $this->scheduler = $scheduler; $this->newsletterCoupon = $newsletterCoupon; } public function save(array $data = []): NewsletterEntity { if (!empty($data['template_id'])) { $template = $this->newsletterTemplatesRepository->findOneById($data['template_id']); if ($template) { $data['body'] = json_encode($template->getBody()); } } if (!empty($data['body'])) { $body = $this->dataSanitizer->sanitizeBody(json_decode($data['body'], true)); $data['body'] = $this->emoji->encodeForUTF8Column(MP_NEWSLETTERS_TABLE, 'body', json_encode($body)); } $newsletter = isset($data['id']) ? $this->getNewsletter($data) : $this->createNewsletter($data); $data = $this->sanitizeAutomationEmailData($data, $newsletter); $oldSenderAddress = $newsletter->getSenderAddress(); $this->updateNewsletter($newsletter, $data); $this->newslettersRepository->flush(); if (!empty($data['segments'])) { $this->updateSegments($newsletter, $data['segments']); } if (!empty($data['options'])) { $this->updateOptions($newsletter, $data['options']); } // fetch model with updated options (for back compatibility) $newsletterModel = Newsletter::filter('filterWithOptions', $newsletter->getType())->findOne($newsletter->getId()); if (!$newsletterModel) { throw new InvalidStateException(); } // save default sender if needed if (!$this->settings->get('sender') && !empty($data['sender_address']) && !empty($data['sender_name'])) { $this->settings->set('sender', [ 'address' => $data['sender_address'], 'name' => $data['sender_name'], ]); } $this->rescheduleIfNeeded($newsletter, $newsletterModel); $this->updateQueue($newsletter, $newsletterModel, $data['options'] ?? []); $this->authorizedEmailsController->onNewsletterSenderAddressUpdate($newsletter, $oldSenderAddress); return $newsletter; } private function sanitizeAutomationEmailData(array $data, NewsletterEntity $newsletter): array { if ($newsletter->getType() !== NewsletterEntity::TYPE_AUTOMATION) { return $data; } $data['segments'] = []; return $data; } public function duplicate(NewsletterEntity $newsletter): NewsletterEntity { $duplicate = clone $newsletter; // reset timestamps $createdAt = Carbon::createFromTimestamp($this->wp->currentTime('timestamp')); $duplicate->setCreatedAt($createdAt); $duplicate->setUpdatedAt($createdAt); $duplicate->setDeletedAt(null); // translators: %s is the subject of the mail which has been copied. $duplicate->setSubject(sprintf(__('Copy of %s', 'mailpoet'), $newsletter->getSubject())); // generate new unsubscribe token $duplicate->setUnsubscribeToken($this->security->generateUnsubscribeTokenByEntity($duplicate)); // reset status $duplicate->setStatus(NewsletterEntity::STATUS_DRAFT); // reset hash $duplicate->setHash(Security::generateHash()); // reset sent at date $duplicate->setSentAt(null); $body = $duplicate->getBody(); if ($body) { $duplicate->setBody($this->newsletterCoupon->cleanupBodySensitiveData($body)); } $this->newslettersRepository->persist($duplicate); $this->newslettersRepository->flush(); // duplicate wp post data $post = $this->wp->getPost($newsletter->getWpPostId()); if ($post instanceof \WP_Post) { $newPostId = $this->wp->wpInsertPost([ 'post_content' => $post->post_content, // @phpcs:ignore Squiz.NamingConventions.ValidVariableName.MemberNotCamelCaps 'post_type' => $post->post_type, // @phpcs:ignore Squiz.NamingConventions.ValidVariableName.MemberNotCamelCaps ]); $duplicate->setWpPostId($newPostId); } // create relationships between duplicate and segments foreach ($newsletter->getNewsletterSegments() as $newsletterSegment) { $segment = $newsletterSegment->getSegment(); if (!$segment) { continue; } $duplicateSegment = new NewsletterSegmentEntity($duplicate, $segment); $duplicate->getNewsletterSegments()->add($duplicateSegment); $this->newsletterSegmentRepository->persist($duplicateSegment); } // duplicate options $ignoredOptions = [ NewsletterOptionFieldEntity::NAME_IS_SCHEDULED, NewsletterOptionFieldEntity::NAME_SCHEDULED_AT, ]; foreach ($newsletter->getOptions() as $newsletterOption) { $optionField = $newsletterOption->getOptionField(); if (!$optionField) { continue; } if (in_array($optionField->getName(), $ignoredOptions, true)) { continue; } $duplicateOption = new NewsletterOptionEntity($duplicate, $optionField); $duplicateOption->setValue($newsletterOption->getValue()); $duplicate->getOptions()->add($duplicateOption); $this->newsletterOptionsRepository->persist($duplicateOption); } $this->newslettersRepository->flush(); return $duplicate; } private function getNewsletter(array $data): NewsletterEntity { if (!isset($data['id'])) { throw new UnexpectedValueException(); } $newsletter = $this->newslettersRepository->findOneById((int)$data['id']); if (!$newsletter) { throw new NotFoundException(); } return $newsletter; } private function createNewsletter(array $data): NewsletterEntity { $newsletter = new NewsletterEntity(); $newsletter->setUnsubscribeToken($this->security->generateUnsubscribeTokenByEntity($newsletter)); $newsletter->setHash(Security::generateHash()); // set default sender based on settings if (empty($data['sender'])) { $sender = $this->settings->get('sender', []); $data['sender_name'] = $sender['name'] ?? ''; $data['sender_address'] = $sender['address'] ?? ''; } // set default reply_to based on settings if (empty($data['reply_to'])) { $replyTo = $this->settings->get('reply_to', []); $data['reply_to_name'] = $replyTo['name'] ?? ''; $data['reply_to_address'] = $replyTo['address'] ?? ''; } $this->updateNewsletter($newsletter, $data); $this->newslettersRepository->persist($newsletter); return $newsletter; } private function updateNewsletter(NewsletterEntity $newsletter, array $data) { if (array_key_exists('type', $data)) { $newsletter->setType($data['type']); } if (array_key_exists('subject', $data)) { $newsletter->setSubject($data['subject']); } if (array_key_exists('preheader', $data)) { $newsletter->setPreheader($data['preheader']); } if (array_key_exists('body', $data)) { $newsletter->setBody(json_decode($data['body'], true)); } if (array_key_exists('ga_campaign', $data)) { $newsletter->setGaCampaign($data['ga_campaign']); } if (array_key_exists('sender_name', $data)) { $newsletter->setSenderName($data['sender_name'] ?? ''); } if (array_key_exists('sender_address', $data)) { $newsletter->setSenderAddress($data['sender_address'] ?? ''); } if (array_key_exists('reply_to_name', $data)) { $newsletter->setReplyToName($data['reply_to_name'] ?? ''); } if (array_key_exists('reply_to_address', $data)) { $newsletter->setReplyToAddress($data['reply_to_address'] ?? ''); } if ($newsletter->getStatus() === NewsletterEntity::STATUS_CORRUPT) { $newsletter->setStatus(NewsletterEntity::STATUS_SENDING); } } private function updateSegments(NewsletterEntity $newsletter, array $segments) { $newsletterSegments = []; foreach ($segments as $segmentData) { if (!is_array($segmentData) || !isset($segmentData['id'])) { continue; } $segment = $this->entityManager->getReference(SegmentEntity::class, (int)$segmentData['id']); if (!$segment) { continue; } $newsletterSegment = $this->newsletterSegmentRepository->findOneBy([ 'newsletter' => $newsletter, 'segment' => $segment, ]); if (!$newsletterSegment) { $newsletterSegment = new NewsletterSegmentEntity($newsletter, $segment); $this->entityManager->persist($newsletterSegment); } if (!$newsletter->getNewsletterSegments()->contains($newsletterSegment)) { $newsletter->getNewsletterSegments()->add($newsletterSegment); } $newsletterSegments[] = $newsletterSegment; } // on Doctrine < 2.6, when using orphan removal, we need to remove items manually instead of replacing the // whole collection (see https://github.com/doctrine/orm/commit/1587aac4ff6b0753ddd5f8b8d4558b6b40096057) foreach ($newsletter->getNewsletterSegments() as $newsletterSegment) { if (!in_array($newsletterSegment, $newsletterSegments, true)) { $newsletter->getNewsletterSegments()->removeElement($newsletterSegment); // triggers orphan removal } } $this->entityManager->flush(); } private function updateOptions(NewsletterEntity $newsletter, array $options) { $optionFields = $this->newsletterOptionFieldsRepository->findBy(['newsletterType' => $newsletter->getType()]); foreach ($optionFields as $optionField) { if (!isset($options[$optionField->getName()])) { continue; } $option = $this->newsletterOptionsRepository->findOneBy([ 'newsletter' => $newsletter, 'optionField' => $optionField, ]); if (!$option) { $option = new NewsletterOptionEntity($newsletter, $optionField); $this->newsletterOptionsRepository->persist($option); } $option->setValue($options[$optionField->getName()]); if (!$newsletter->getOptions()->contains($option)) { $newsletter->getOptions()->add($option); } } $this->entityManager->flush(); } private function rescheduleIfNeeded(NewsletterEntity $newsletter, Newsletter $newsletterModel) { if ($newsletter->getType() !== NewsletterEntity::TYPE_NOTIFICATION) { return; } // generate the new schedule from options and get the new "next run" date $schedule = $this->postNotificationScheduler->processPostNotificationSchedule($newsletter); $nextRunDateString = $this->scheduler->getNextRunDate($schedule); $nextRunDate = $nextRunDateString ? Carbon::createFromFormat('Y-m-d H:i:s', $nextRunDateString) : null; if ($nextRunDate === false) { throw InvalidStateException::create()->withMessage('Invalid next run date generated'); } // find previously scheduled jobs and reschedule them $scheduledTasks = $this->scheduledTasksRepository->findByNewsletterAndStatus($newsletter, ScheduledTaskEntity::STATUS_SCHEDULED); foreach ($scheduledTasks as $scheduledTask) { $scheduledTask->setScheduledAt($nextRunDate); } $this->entityManager->flush(); // 'processPostNotificationSchedule' modifies newsletter options by old model - let's reload them foreach ($newsletter->getOptions() as $newsletterOption) { $this->entityManager->refresh($newsletterOption); } } private function updateQueue(NewsletterEntity $newsletter, Newsletter $newsletterModel, array $options) { if ($newsletter->getType() !== NewsletterEntity::TYPE_STANDARD) { return; } $queue = $newsletter->getLatestQueue(); if (!$queue) { return; } // if newsletter was previously scheduled and is now unscheduled, set its status to DRAFT and delete associated queue record if ($newsletter->getStatus() === NewsletterEntity::STATUS_SCHEDULED && isset($options['isScheduled']) && empty($options['isScheduled'])) { $this->entityManager->remove($queue); $newsletter->setStatus(NewsletterEntity::STATUS_DRAFT); } else { $queueModel = $newsletterModel->getQueue(); $queueModel->newsletterRenderedSubject = null; $queueModel->newsletterRenderedBody = null; $newsletterQueueTask = new NewsletterQueueTask(); $newsletterQueueTask->preProcessNewsletter($newsletter, $queueModel); // 'preProcessNewsletter' modifies queue by old model - let's reload it $this->entityManager->refresh($queue); } $this->entityManager->flush(); } }