535 lines
17 KiB
PHP
535 lines
17 KiB
PHP
<?php
|
|
|
|
namespace MailPoet\API\JSON\v1;
|
|
|
|
use Carbon\Carbon;
|
|
use MailPoet\API\JSON\Endpoint as APIEndpoint;
|
|
use MailPoet\API\JSON\Error as APIError;
|
|
use MailPoet\Config\AccessControl;
|
|
use MailPoet\Cron\CronHelper;
|
|
use MailPoet\Cron\Workers\SendingQueue\Tasks\Newsletter as NewsletterQueueTask;
|
|
use MailPoet\Listing;
|
|
use MailPoet\Models\Newsletter;
|
|
use MailPoet\Models\NewsletterOption;
|
|
use MailPoet\Models\NewsletterOptionField;
|
|
use MailPoet\Models\NewsletterSegment;
|
|
use MailPoet\Models\NewsletterTemplate;
|
|
use MailPoet\Models\SendingQueue;
|
|
use MailPoet\Models\Setting;
|
|
use MailPoet\Models\Subscriber;
|
|
use MailPoet\Newsletter\Renderer\Renderer;
|
|
use MailPoet\Newsletter\Scheduler\Scheduler;
|
|
use MailPoet\Newsletter\Url as NewsletterUrl;
|
|
use MailPoet\Settings\SettingsController;
|
|
use MailPoet\WP\Functions as WPFunctions;
|
|
|
|
if(!defined('ABSPATH')) exit;
|
|
|
|
class Newsletters extends APIEndpoint {
|
|
|
|
/** @var Listing\BulkActionController */
|
|
private $bulk_action;
|
|
|
|
/** @var Listing\Handler */
|
|
private $listing_handler;
|
|
|
|
/** @var WPFunctions */
|
|
private $wp;
|
|
|
|
/** @var SettingsController */
|
|
private $settings;
|
|
|
|
public $permissions = array(
|
|
'global' => AccessControl::PERMISSION_MANAGE_EMAILS
|
|
);
|
|
|
|
function __construct(
|
|
Listing\BulkActionController $bulk_action,
|
|
Listing\Handler $listing_handler,
|
|
WPFunctions $wp,
|
|
SettingsController $settings
|
|
) {
|
|
$this->bulk_action = $bulk_action;
|
|
$this->listing_handler = $listing_handler;
|
|
$this->wp = $wp;
|
|
$this->settings = $settings;
|
|
}
|
|
|
|
function get($data = array()) {
|
|
$id = (isset($data['id']) ? (int)$data['id'] : false);
|
|
$newsletter = Newsletter::findOne($id);
|
|
if($newsletter === false) {
|
|
return $this->errorResponse(array(
|
|
APIError::NOT_FOUND => __('This newsletter does not exist.', 'mailpoet')
|
|
));
|
|
} else {
|
|
$newsletter = $newsletter
|
|
->withSegments()
|
|
->withOptions()
|
|
->withSendingQueue()
|
|
->asArray();
|
|
$newsletter = $this->wp->applyFilters('mailpoet_api_newsletters_get_after', $newsletter);
|
|
return $this->successResponse($newsletter);
|
|
}
|
|
}
|
|
|
|
function save($data = array()) {
|
|
$data = $this->wp->applyFilters('mailpoet_api_newsletters_save_before', $data);
|
|
|
|
$segments = array();
|
|
if(isset($data['segments'])) {
|
|
$segments = $data['segments'];
|
|
unset($data['segments']);
|
|
}
|
|
|
|
$options = array();
|
|
if(isset($data['options'])) {
|
|
$options = $data['options'];
|
|
unset($data['options']);
|
|
}
|
|
|
|
if(!empty($data['template_id'])) {
|
|
$template = NewsletterTemplate::whereEqual('id', $data['template_id'])->findOne();
|
|
if(!empty($template)) {
|
|
$template = $template->asArray();
|
|
$data['body'] = $template['body'];
|
|
}
|
|
unset($data['template_id']);
|
|
}
|
|
|
|
$newsletter = Newsletter::createOrUpdate($data);
|
|
$errors = $newsletter->getErrors();
|
|
|
|
if(!empty($errors)) return $this->badRequest($errors);
|
|
// Re-fetch newsletter to sync changes made by DB
|
|
// updated_at column use CURRENT_TIMESTAMP for update and this change is not updated automatically by ORM
|
|
$newsletter = Newsletter::findOne($newsletter->id);
|
|
|
|
if(!empty($segments)) {
|
|
NewsletterSegment::where('newsletter_id', $newsletter->id)
|
|
->deleteMany();
|
|
foreach ($segments as $segment) {
|
|
if(!is_array($segment)) continue;
|
|
$relation = NewsletterSegment::create();
|
|
$relation->segment_id = (int)$segment['id'];
|
|
$relation->newsletter_id = $newsletter->id;
|
|
$relation->save();
|
|
}
|
|
}
|
|
|
|
if(isset($data['sender_address']) && isset($data['sender_name'])) {
|
|
Setting::saveDefaultSenderIfNeeded($data['sender_address'], $data['sender_name']);
|
|
}
|
|
|
|
if(!empty($options)) {
|
|
$option_fields = NewsletterOptionField::where(
|
|
'newsletter_type',
|
|
$newsletter->type
|
|
)->findMany();
|
|
// update newsletter options
|
|
foreach ($option_fields as $option_field) {
|
|
if(isset($options[$option_field->name])) {
|
|
$newsletter_option = NewsletterOption::createOrUpdate(
|
|
array(
|
|
'newsletter_id' => $newsletter->id,
|
|
'option_field_id' => $option_field->id,
|
|
'value' => $options[$option_field->name]
|
|
)
|
|
);
|
|
}
|
|
}
|
|
// reload newsletter with updated options
|
|
$newsletter = Newsletter::filter('filterWithOptions', $newsletter->type)->findOne($newsletter->id);
|
|
// if this is a post notification, process newsletter options and update its schedule
|
|
if($newsletter->type === Newsletter::TYPE_NOTIFICATION) {
|
|
// generate the new schedule from options and get the new "next run" date
|
|
$newsletter->schedule = Scheduler::processPostNotificationSchedule($newsletter);
|
|
$next_run_date = Scheduler::getNextRunDate($newsletter->schedule);
|
|
// find previously scheduled jobs and reschedule them using the new "next run" date
|
|
SendingQueue::findTaskByNewsletterId($newsletter->id)
|
|
->where('tasks.status', SendingQueue::STATUS_SCHEDULED)
|
|
->findResultSet()
|
|
->set('scheduled_at', $next_run_date)
|
|
->save();
|
|
}
|
|
}
|
|
|
|
$queue = $newsletter->getQueue();
|
|
if($queue && !in_array($newsletter->type, array(Newsletter::TYPE_NOTIFICATION, Newsletter::TYPE_NOTIFICATION_HISTORY))) {
|
|
// if newsletter was previously scheduled and is now unscheduled, set its status to DRAFT and delete associated queue record
|
|
if($newsletter->status === Newsletter::STATUS_SCHEDULED && isset($options['isScheduled']) && empty($options['isScheduled'])) {
|
|
$queue->delete();
|
|
$newsletter->status = Newsletter::STATUS_DRAFT;
|
|
$newsletter->save();
|
|
} else {
|
|
$queue->newsletter_rendered_body = null;
|
|
$queue->newsletter_rendered_subject = null;
|
|
$newsletterQueueTask = new NewsletterQueueTask();
|
|
$newsletterQueueTask->preProcessNewsletter($newsletter, $queue);
|
|
}
|
|
}
|
|
|
|
$this->wp->doAction('mailpoet_api_newsletters_save_after', $newsletter);
|
|
|
|
$preview_url = NewsletterUrl::getViewInBrowserUrl(
|
|
NewsletterUrl::TYPE_LISTING_EDITOR,
|
|
$newsletter,
|
|
Subscriber::getCurrentWPUser()
|
|
);
|
|
|
|
return $this->successResponse($newsletter->asArray(), array('preview_url' => $preview_url));
|
|
}
|
|
|
|
function setStatus($data = array()) {
|
|
$status = (isset($data['status']) ? $data['status'] : null);
|
|
|
|
if(!$status) {
|
|
return $this->badRequest(array(
|
|
APIError::BAD_REQUEST => __('You need to specify a status.', 'mailpoet')
|
|
));
|
|
}
|
|
|
|
$id = (isset($data['id'])) ? (int)$data['id'] : false;
|
|
$newsletter = Newsletter::findOneWithOptions($id);
|
|
|
|
if($newsletter === false) {
|
|
return $this->errorResponse(array(
|
|
APIError::NOT_FOUND => __('This newsletter does not exist.', 'mailpoet')
|
|
));
|
|
}
|
|
|
|
$newsletter->setStatus($status);
|
|
$errors = $newsletter->getErrors();
|
|
|
|
if(!empty($errors)) {
|
|
return $this->errorResponse($errors);
|
|
}
|
|
|
|
// if there are past due notifications, reschedule them for the next send date
|
|
if($newsletter->type === Newsletter::TYPE_NOTIFICATION && $status === Newsletter::STATUS_ACTIVE) {
|
|
$next_run_date = Scheduler::getNextRunDate($newsletter->schedule);
|
|
$queue = $newsletter->queue()->findOne();
|
|
if($queue) {
|
|
$queue->task()
|
|
->whereLte('scheduled_at', Carbon::createFromTimestamp($this->wp->currentTime('timestamp')))
|
|
->where('status', SendingQueue::STATUS_SCHEDULED)
|
|
->findResultSet()
|
|
->set('scheduled_at', $next_run_date)
|
|
->save();
|
|
}
|
|
Scheduler::createPostNotificationSendingTask($newsletter);
|
|
}
|
|
|
|
|
|
return $this->successResponse(
|
|
Newsletter::findOne($newsletter->id)->asArray()
|
|
);
|
|
}
|
|
|
|
function restore($data = array()) {
|
|
$id = (isset($data['id']) ? (int)$data['id'] : false);
|
|
$newsletter = Newsletter::findOne($id);
|
|
if($newsletter === false) {
|
|
return $this->errorResponse(array(
|
|
APIError::NOT_FOUND => __('This newsletter does not exist.', 'mailpoet')
|
|
));
|
|
} else {
|
|
$newsletter->restore();
|
|
return $this->successResponse(
|
|
Newsletter::findOne($newsletter->id)->asArray(),
|
|
array('count' => 1)
|
|
);
|
|
}
|
|
}
|
|
|
|
function trash($data = array()) {
|
|
$id = (isset($data['id']) ? (int)$data['id'] : false);
|
|
$newsletter = Newsletter::findOne($id);
|
|
if($newsletter === false) {
|
|
return $this->errorResponse(array(
|
|
APIError::NOT_FOUND => __('This newsletter does not exist.', 'mailpoet')
|
|
));
|
|
} else {
|
|
$newsletter->trash();
|
|
return $this->successResponse(
|
|
Newsletter::findOne($newsletter->id)->asArray(),
|
|
array('count' => 1)
|
|
);
|
|
}
|
|
}
|
|
|
|
function delete($data = array()) {
|
|
$id = (isset($data['id']) ? (int)$data['id'] : false);
|
|
$newsletter = Newsletter::findOne($id);
|
|
if($newsletter === false) {
|
|
return $this->errorResponse(array(
|
|
APIError::NOT_FOUND => __('This newsletter does not exist.', 'mailpoet')
|
|
));
|
|
} else {
|
|
$newsletter->delete();
|
|
return $this->successResponse(null, array('count' => 1));
|
|
}
|
|
}
|
|
|
|
function duplicate($data = array()) {
|
|
$id = (isset($data['id']) ? (int)$data['id'] : false);
|
|
$newsletter = Newsletter::findOne($id);
|
|
|
|
if($newsletter === false) {
|
|
return $this->errorResponse(array(
|
|
APIError::NOT_FOUND => __('This newsletter does not exist.', 'mailpoet')
|
|
));
|
|
} else {
|
|
$data = array(
|
|
'subject' => sprintf(__('Copy of %s', 'mailpoet'), $newsletter->subject)
|
|
);
|
|
$duplicate = $newsletter->duplicate($data);
|
|
$errors = $duplicate->getErrors();
|
|
|
|
if(!empty($errors)) {
|
|
return $this->errorResponse($errors);
|
|
} else {
|
|
$this->wp->doAction('mailpoet_api_newsletters_duplicate_after', $newsletter, $duplicate);
|
|
return $this->successResponse(
|
|
Newsletter::findOne($duplicate->id)->asArray(),
|
|
array('count' => 1)
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
function showPreview($data = array()) {
|
|
if(empty($data['body'])) {
|
|
return $this->badRequest(array(
|
|
APIError::BAD_REQUEST => __('Newsletter data is missing.', 'mailpoet')
|
|
));
|
|
}
|
|
|
|
$id = (isset($data['id'])) ? (int)$data['id'] : false;
|
|
$newsletter = Newsletter::findOne($id);
|
|
|
|
if($newsletter === false) {
|
|
return $this->errorResponse(array(
|
|
APIError::NOT_FOUND => __('This newsletter does not exist.', 'mailpoet')
|
|
));
|
|
} else {
|
|
$newsletter->body = $data['body'];
|
|
$newsletter->save();
|
|
$subscriber = Subscriber::getCurrentWPUser();
|
|
$preview_url = NewsletterUrl::getViewInBrowserUrl(
|
|
NewsletterUrl::TYPE_LISTING_EDITOR,
|
|
$newsletter,
|
|
$subscriber
|
|
);
|
|
// strip protocol to avoid mix content error
|
|
$preview_url = preg_replace('{^https?:}i', '', $preview_url);
|
|
|
|
return $this->successResponse(
|
|
Newsletter::findOne($newsletter->id)->asArray(),
|
|
array('preview_url' => $preview_url)
|
|
);
|
|
}
|
|
}
|
|
|
|
function sendPreview($data = array()) {
|
|
if(empty($data['subscriber'])) {
|
|
return $this->badRequest(array(
|
|
APIError::BAD_REQUEST => __('Please specify receiver information.', 'mailpoet')
|
|
));
|
|
}
|
|
|
|
$id = (isset($data['id'])) ? (int)$data['id'] : false;
|
|
$newsletter = Newsletter::findOne($id);
|
|
|
|
if($newsletter === false) {
|
|
return $this->errorResponse(array(
|
|
APIError::NOT_FOUND => __('This newsletter does not exist.', 'mailpoet')
|
|
));
|
|
} else {
|
|
$renderer = new Renderer($newsletter, $preview = true);
|
|
$rendered_newsletter = $renderer->render();
|
|
$divider = '***MailPoet***';
|
|
$data_for_shortcodes = array_merge(
|
|
array($newsletter->subject),
|
|
$rendered_newsletter
|
|
);
|
|
|
|
$body = implode($divider, $data_for_shortcodes);
|
|
|
|
$subscriber = Subscriber::getCurrentWPUser();
|
|
$subscriber = ($subscriber) ? $subscriber : false;
|
|
|
|
$shortcodes = new \MailPoet\Newsletter\Shortcodes\Shortcodes(
|
|
$newsletter,
|
|
$subscriber,
|
|
$queue = false,
|
|
$wp_user_preview = true
|
|
);
|
|
|
|
list(
|
|
$rendered_newsletter['subject'],
|
|
$rendered_newsletter['body']['html'],
|
|
$rendered_newsletter['body']['text']
|
|
) = explode($divider, $shortcodes->replace($body));
|
|
|
|
try {
|
|
$mailer = (!empty($data['mailer'])) ?
|
|
$data['mailer'] :
|
|
new \MailPoet\Mailer\Mailer(
|
|
$mailer = false,
|
|
$sender = false,
|
|
$reply_to = false
|
|
);
|
|
$extra_params = array('unsubscribe_url' => home_url());
|
|
$result = $mailer->send($rendered_newsletter, $data['subscriber'], $extra_params);
|
|
|
|
if($result['response'] === false) {
|
|
$error = sprintf(
|
|
__('The email could not be sent: %s', 'mailpoet'),
|
|
$result['error']->getMessage()
|
|
);
|
|
return $this->errorResponse(array(APIError::BAD_REQUEST => $error));
|
|
} else {
|
|
return $this->successResponse(
|
|
Newsletter::findOne($id)->asArray()
|
|
);
|
|
}
|
|
} catch(\Exception $e) {
|
|
return $this->errorResponse(array(
|
|
$e->getCode() => $e->getMessage()
|
|
));
|
|
}
|
|
}
|
|
}
|
|
|
|
function listing($data = array()) {
|
|
$listing_data = $this->listing_handler->get('\MailPoet\Models\Newsletter', $data);
|
|
|
|
$data = array();
|
|
foreach ($listing_data['items'] as $newsletter) {
|
|
$queue = false;
|
|
|
|
if($newsletter->type === Newsletter::TYPE_STANDARD) {
|
|
$newsletter
|
|
->withSegments(true)
|
|
->withSendingQueue()
|
|
->withStatistics();
|
|
} else if($newsletter->type === Newsletter::TYPE_WELCOME || $newsletter->type === Newsletter::TYPE_AUTOMATIC) {
|
|
$newsletter
|
|
->withOptions()
|
|
->withTotalSent()
|
|
->withScheduledToBeSent()
|
|
->withStatistics();
|
|
} else if($newsletter->type === Newsletter::TYPE_NOTIFICATION) {
|
|
$newsletter
|
|
->withOptions()
|
|
->withSegments(true)
|
|
->withChildrenCount();
|
|
} else if($newsletter->type === Newsletter::TYPE_NOTIFICATION_HISTORY) {
|
|
$newsletter
|
|
->withSegments(true)
|
|
->withSendingQueue()
|
|
->withStatistics();
|
|
}
|
|
|
|
if($newsletter->status === Newsletter::STATUS_SENT ||
|
|
$newsletter->status === Newsletter::STATUS_SENDING
|
|
) {
|
|
$queue = $newsletter->getQueue();
|
|
}
|
|
|
|
// get preview url
|
|
$newsletter->preview_url = NewsletterUrl::getViewInBrowserUrl(
|
|
NewsletterUrl::TYPE_LISTING_EDITOR,
|
|
$newsletter,
|
|
$subscriber = null,
|
|
$queue
|
|
);
|
|
|
|
$data[] = $this->wp->applyFilters('mailpoet_api_newsletters_listing_item', $newsletter->asArray());
|
|
}
|
|
|
|
return $this->successResponse($data, array(
|
|
'count' => $listing_data['count'],
|
|
'filters' => $listing_data['filters'],
|
|
'groups' => $listing_data['groups'],
|
|
'mta_log' => $this->settings->get('mta_log'),
|
|
'mta_method' => $this->settings->get('mta.method'),
|
|
'cron_accessible' => CronHelper::isDaemonAccessible(),
|
|
'current_time' => $this->wp->currentTime('mysql')
|
|
));
|
|
}
|
|
|
|
function bulkAction($data = array()) {
|
|
try {
|
|
$meta = $this->bulk_action->apply('\MailPoet\Models\Newsletter', $data);
|
|
return $this->successResponse(null, $meta);
|
|
} catch(\Exception $e) {
|
|
return $this->errorResponse(array(
|
|
$e->getCode() => $e->getMessage()
|
|
));
|
|
}
|
|
}
|
|
|
|
function create($data = array()) {
|
|
$options = array();
|
|
if(isset($data['options'])) {
|
|
$options = $data['options'];
|
|
unset($data['options']);
|
|
}
|
|
|
|
$newsletter = Newsletter::createOrUpdate($data);
|
|
$errors = $newsletter->getErrors();
|
|
|
|
if(!empty($errors)) {
|
|
return $this->badRequest($errors);
|
|
} else {
|
|
// try to load template data
|
|
$template_id = (isset($data['template']) ? (int)$data['template'] : false);
|
|
$template = NewsletterTemplate::findOne($template_id);
|
|
if($template === false) {
|
|
$newsletter->body = array();
|
|
} else {
|
|
$newsletter->body = $template->body;
|
|
}
|
|
}
|
|
|
|
$newsletter->save();
|
|
$errors = $newsletter->getErrors();
|
|
if(!empty($errors)) {
|
|
return $this->badRequest($errors);
|
|
} else {
|
|
if(!empty($options)) {
|
|
$option_fields = NewsletterOptionField::where(
|
|
'newsletter_type', $newsletter->type
|
|
)->findArray();
|
|
|
|
foreach ($option_fields as $option_field) {
|
|
if(isset($options[$option_field['name']])) {
|
|
$relation = NewsletterOption::create();
|
|
$relation->newsletter_id = $newsletter->id;
|
|
$relation->option_field_id = $option_field['id'];
|
|
$relation->value = $options[$option_field['name']];
|
|
$relation->save();
|
|
}
|
|
}
|
|
}
|
|
|
|
if(
|
|
empty($data['id'])
|
|
&&
|
|
isset($data['type'])
|
|
&&
|
|
$data['type'] === Newsletter::TYPE_NOTIFICATION
|
|
) {
|
|
$newsletter = Newsletter::filter('filterWithOptions', $data['type'])->findOne($newsletter->id);
|
|
Scheduler::processPostNotificationSchedule($newsletter);
|
|
}
|
|
|
|
return $this->successResponse(
|
|
Newsletter::findOne($newsletter->id)->asArray()
|
|
);
|
|
}
|
|
}
|
|
}
|