738 lines
19 KiB
PHP
738 lines
19 KiB
PHP
<?php
|
|
namespace MailPoet\Models;
|
|
use MailPoet\Newsletter\Renderer\Renderer;
|
|
use MailPoet\Util\Helpers;
|
|
use MailPoet\Util\Security;
|
|
|
|
if(!defined('ABSPATH')) exit;
|
|
|
|
class Newsletter extends Model {
|
|
public static $_table = MP_NEWSLETTERS_TABLE;
|
|
const TYPE_STANDARD = 'standard';
|
|
const TYPE_WELCOME = 'welcome';
|
|
const TYPE_NOTIFICATION = 'notification';
|
|
const TYPE_NOTIFICATION_HISTORY = 'notification_history';
|
|
// standard newsletters
|
|
const STATUS_DRAFT = 'draft';
|
|
const STATUS_SCHEDULED = 'scheduled';
|
|
const STATUS_SENDING = 'sending';
|
|
const STATUS_SENT = 'sent';
|
|
// automatic newsletters status
|
|
const STATUS_ACTIVE = 'active';
|
|
const NEWSLETTER_HASH_LENGTH = 6;
|
|
|
|
function __construct() {
|
|
parent::__construct();
|
|
$this->addValidations('type', array(
|
|
'required' => __('Please specify a type.', 'mailpoet')
|
|
));
|
|
}
|
|
|
|
function queue() {
|
|
return $this->has_one(__NAMESPACE__ . '\SendingQueue', 'newsletter_id', 'id');
|
|
}
|
|
|
|
function save() {
|
|
if(is_string($this->deleted_at) && strlen(trim($this->deleted_at)) === 0) {
|
|
$this->set_expr('deleted_at', 'NULL');
|
|
}
|
|
|
|
$this->set('body',
|
|
is_array($this->body)
|
|
? json_encode($this->body)
|
|
: $this->body
|
|
);
|
|
|
|
$this->set('hash',
|
|
($this->hash)
|
|
? $this->hash
|
|
: self::generateHash()
|
|
);
|
|
return parent::save();
|
|
}
|
|
|
|
function setStatus($status = null) {
|
|
if(in_array($status, array(
|
|
self::STATUS_DRAFT,
|
|
self::STATUS_SCHEDULED,
|
|
self::STATUS_SENDING,
|
|
self::STATUS_SENT,
|
|
self::STATUS_ACTIVE
|
|
))) {
|
|
$this->set('status', $status);
|
|
$this->save();
|
|
}
|
|
return $this;
|
|
}
|
|
|
|
function duplicate($data = array()) {
|
|
$newsletter_data = $this->asArray();
|
|
|
|
// remove id so that it creates a new record
|
|
unset($newsletter_data['id']);
|
|
|
|
// merge data with newsletter data (allows override)
|
|
$data = array_merge($newsletter_data, $data);
|
|
|
|
$duplicate = self::create();
|
|
$duplicate->hydrate($data);
|
|
|
|
// reset timestamps
|
|
$duplicate->set_expr('created_at', 'NOW()');
|
|
$duplicate->set_expr('updated_at', 'NOW()');
|
|
$duplicate->set_expr('deleted_at', 'NULL');
|
|
|
|
// reset status
|
|
$duplicate->set('status', self::STATUS_DRAFT);
|
|
|
|
// reset hash
|
|
$duplicate->set('hash', null);
|
|
|
|
$duplicate->save();
|
|
|
|
if($duplicate->getErrors() === false) {
|
|
// create relationships between duplicate and segments
|
|
$segments = $this->segments()->findArray();
|
|
|
|
if(!empty($segments)) {
|
|
foreach($segments as $segment) {
|
|
$relation = NewsletterSegment::create();
|
|
$relation->segment_id = $segment['id'];
|
|
$relation->newsletter_id = $duplicate->id;
|
|
$relation->save();
|
|
}
|
|
}
|
|
|
|
// duplicate options
|
|
$options = NewsletterOption::where('newsletter_id', $this->id)
|
|
->findArray();
|
|
|
|
if(!empty($options)) {
|
|
foreach($options as $option) {
|
|
$relation = NewsletterOption::create();
|
|
$relation->newsletter_id = $duplicate->id;
|
|
$relation->option_field_id = $option['option_field_id'];
|
|
$relation->value = $option['value'];
|
|
$relation->save();
|
|
}
|
|
}
|
|
}
|
|
|
|
return $duplicate;
|
|
}
|
|
|
|
function createNotificationHistory() {
|
|
$newsletter_data = $this->asArray();
|
|
|
|
// remove id so that it creates a new record
|
|
unset($newsletter_data['id']);
|
|
|
|
$data = array_merge(
|
|
$newsletter_data,
|
|
array(
|
|
'parent_id' => $this->id,
|
|
'type' => self::TYPE_NOTIFICATION_HISTORY,
|
|
'status' => self::STATUS_SENDING
|
|
)
|
|
);
|
|
|
|
$notification_history = self::create();
|
|
$notification_history->hydrate($data);
|
|
|
|
// reset timestamps
|
|
$notification_history->set_expr('created_at', 'NOW()');
|
|
$notification_history->set_expr('updated_at', 'NOW()');
|
|
$notification_history->set_expr('deleted_at', 'NULL');
|
|
|
|
// reset hash
|
|
$notification_history->set('hash', null);
|
|
|
|
$notification_history->save();
|
|
|
|
if($notification_history->getErrors() === false) {
|
|
// create relationships between notification history and segments
|
|
$segments = $this->segments()->findArray();
|
|
|
|
if(!empty($segments)) {
|
|
foreach($segments as $segment) {
|
|
$relation = NewsletterSegment::create();
|
|
$relation->segment_id = $segment['id'];
|
|
$relation->newsletter_id = $notification_history->id;
|
|
$relation->save();
|
|
}
|
|
}
|
|
}
|
|
|
|
return $notification_history;
|
|
}
|
|
|
|
function asArray() {
|
|
$model = parent::asArray();
|
|
|
|
if(isset($model['body'])) {
|
|
$model['body'] = json_decode($model['body'], true);
|
|
}
|
|
return $model;
|
|
}
|
|
|
|
function delete() {
|
|
// delete all relations to segments
|
|
NewsletterSegment::where('newsletter_id', $this->id)->deleteMany();
|
|
|
|
return parent::delete();
|
|
}
|
|
|
|
function children() {
|
|
return $this->has_many(
|
|
__NAMESPACE__.'\Newsletter',
|
|
'parent_id',
|
|
'id'
|
|
);
|
|
}
|
|
|
|
function segments() {
|
|
return $this->has_many_through(
|
|
__NAMESPACE__.'\Segment',
|
|
__NAMESPACE__.'\NewsletterSegment',
|
|
'newsletter_id',
|
|
'segment_id'
|
|
);
|
|
}
|
|
|
|
function withSegments($incl_deleted = false) {
|
|
$this->segments = $this->segments()->findArray();
|
|
if($incl_deleted) {
|
|
$this->withDeletedSegments();
|
|
}
|
|
return $this;
|
|
}
|
|
|
|
// Intermediary table only
|
|
function segmentLinks() {
|
|
return $this->has_many(
|
|
__NAMESPACE__.'\NewsletterSegment',
|
|
'newsletter_id',
|
|
'id'
|
|
);
|
|
}
|
|
|
|
function withDeletedSegments() {
|
|
if(!empty($this->segments)) {
|
|
$segment_ids = Helpers::arrayColumn($this->segments, 'id');
|
|
$links = $this->segmentLinks()
|
|
->whereNotIn('segment_id', $segment_ids)->findArray();
|
|
$deleted_segments = array();
|
|
|
|
foreach($links as $link) {
|
|
$deleted_segments[] = array(
|
|
'id' => $link['segment_id'],
|
|
'name' => __('Deleted list', 'mailpoet')
|
|
);
|
|
}
|
|
$this->segments = array_merge($this->segments, $deleted_segments);
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
function withChildrenCount() {
|
|
$this->children_count = $this->children()->count();
|
|
return $this;
|
|
}
|
|
|
|
function options() {
|
|
return $this->has_many_through(
|
|
__NAMESPACE__.'\NewsletterOptionField',
|
|
__NAMESPACE__.'\NewsletterOption',
|
|
'newsletter_id',
|
|
'option_field_id'
|
|
)->select_expr(MP_NEWSLETTER_OPTION_TABLE.'.value');
|
|
}
|
|
|
|
function getQueue($columns = '*') {
|
|
return SendingQueue::select($columns)
|
|
->where('newsletter_id', $this->id)
|
|
->orderByDesc('updated_at')
|
|
->findOne();
|
|
}
|
|
|
|
function withSendingQueue() {
|
|
$queue = $this->getQueue(array(
|
|
'id',
|
|
'newsletter_id',
|
|
'newsletter_rendered_subject',
|
|
'status',
|
|
'count_processed',
|
|
'count_total',
|
|
'scheduled_at'
|
|
));
|
|
if($queue === false) {
|
|
$this->queue = false;
|
|
} else {
|
|
$this->queue = $queue->asArray();
|
|
}
|
|
return $this;
|
|
}
|
|
|
|
function withOptions() {
|
|
$options = $this->options()->findArray();
|
|
if(empty($options)) {
|
|
$this->options = array();
|
|
} else {
|
|
$this->options = Helpers::arrayColumn($options, 'value', 'name');
|
|
}
|
|
return $this;
|
|
}
|
|
|
|
function withTotalSent() {
|
|
// total of subscribers who received the email
|
|
$this->total_sent = (int)SendingQueue::where('newsletter_id', $this->id)
|
|
->where('status', SendingQueue::STATUS_COMPLETED)
|
|
->sum('count_processed');
|
|
return $this;
|
|
}
|
|
|
|
function withStatistics() {
|
|
$statistics = $this->getStatistics();
|
|
$this->statistics = $statistics;
|
|
return $this;
|
|
}
|
|
|
|
function render() {
|
|
$renderer = new Renderer($this);
|
|
return $renderer->render();
|
|
}
|
|
|
|
function getStatistics() {
|
|
if($this->type === self::TYPE_WELCOME) {
|
|
$where = '`queue_id` IN (SELECT `id` FROM ' . MP_SENDING_QUEUES_TABLE . '
|
|
WHERE `newsletter_id` = ? AND `status` = ?)';
|
|
$where_params = array($this->id, SendingQueue::STATUS_COMPLETED);
|
|
} else {
|
|
if($this->queue === false) {
|
|
return false;
|
|
} else {
|
|
$where = '`queue_id` = ?';
|
|
$where_params = array($this->queue['id']);
|
|
}
|
|
}
|
|
|
|
$clicks = StatisticsClicks::selectExpr(
|
|
'COUNT(DISTINCT subscriber_id) as cnt'
|
|
)
|
|
->whereRaw($where, $where_params)
|
|
->findOne();
|
|
|
|
$opens = StatisticsOpens::selectExpr(
|
|
'COUNT(DISTINCT subscriber_id) as cnt'
|
|
)
|
|
->whereRaw($where, $where_params)
|
|
->findOne();
|
|
|
|
$unsubscribes = StatisticsUnsubscribes::selectExpr(
|
|
'COUNT(DISTINCT subscriber_id) as cnt'
|
|
)
|
|
->whereRaw($where, $where_params)
|
|
->findOne();
|
|
|
|
return array(
|
|
'clicked' => !empty($clicks->cnt) ? $clicks->cnt : 0,
|
|
'opened' => !empty($opens->cnt) ? $opens->cnt : 0,
|
|
'unsubscribed' => !empty($unsubscribes->cnt) ? $unsubscribes->cnt : 0
|
|
);
|
|
}
|
|
|
|
static function search($orm, $search = '') {
|
|
if(strlen(trim($search)) > 0) {
|
|
$orm->whereLike('subject', '%' . $search . '%');
|
|
}
|
|
return $orm;
|
|
}
|
|
|
|
static function filters($data = array()) {
|
|
$type = isset($data['params']['type']) ? $data['params']['type'] : null;
|
|
|
|
// newsletter types without filters
|
|
if(in_array($type, array(
|
|
self::TYPE_NOTIFICATION_HISTORY
|
|
))) {
|
|
return false;
|
|
}
|
|
|
|
$segments = Segment::orderByAsc('name')->findMany();
|
|
$segment_list = array();
|
|
$segment_list[] = array(
|
|
'label' => __('All Lists', 'mailpoet'),
|
|
'value' => ''
|
|
);
|
|
|
|
foreach($segments as $segment) {
|
|
$newsletters = $segment->newsletters()
|
|
->filter('filterType', $type)
|
|
->filter('groupBy', $data);
|
|
|
|
$newsletters_count = $newsletters->count();
|
|
|
|
if($newsletters_count > 0) {
|
|
$segment_list[] = array(
|
|
'label' => sprintf('%s (%d)', $segment->name, $newsletters_count),
|
|
'value' => $segment->id
|
|
);
|
|
}
|
|
}
|
|
|
|
$filters = array(
|
|
'segment' => $segment_list
|
|
);
|
|
|
|
return $filters;
|
|
}
|
|
|
|
static function filterBy($orm, $data = array()) {
|
|
// apply filters
|
|
if(!empty($data['filter'])) {
|
|
foreach($data['filter'] as $key => $value) {
|
|
if($key === 'segment') {
|
|
$segment = Segment::findOne($value);
|
|
if($segment !== false) {
|
|
$orm = $segment->newsletters();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// filter by type
|
|
$type = isset($data['params']['type']) ? $data['params']['type'] : null;
|
|
if($type !== null) {
|
|
$orm->filter('filterType', $type);
|
|
}
|
|
|
|
// filter by parent id
|
|
$parent_id = isset($data['params']['parent_id'])
|
|
? (int)$data['params']['parent_id']
|
|
: null;
|
|
if($parent_id !== null) {
|
|
$orm->where('parent_id', $parent_id);
|
|
}
|
|
|
|
return $orm;
|
|
}
|
|
|
|
static function filterWithOptions($orm) {
|
|
$orm = $orm->select(MP_NEWSLETTERS_TABLE.'.*');
|
|
$optionFields = NewsletterOptionField::findArray();
|
|
foreach($optionFields as $optionField) {
|
|
$orm = $orm->select_expr(
|
|
'IFNULL(GROUP_CONCAT(CASE WHEN ' .
|
|
MP_NEWSLETTER_OPTION_FIELDS_TABLE . '.id=' . $optionField['id'] . ' THEN ' .
|
|
MP_NEWSLETTER_OPTION_TABLE . '.value END), NULL) as "' . $optionField['name'].'"');
|
|
}
|
|
$orm = $orm
|
|
->left_outer_join(
|
|
MP_NEWSLETTER_OPTION_TABLE,
|
|
array(
|
|
MP_NEWSLETTERS_TABLE.'.id',
|
|
'=',
|
|
MP_NEWSLETTER_OPTION_TABLE.'.newsletter_id'
|
|
)
|
|
)
|
|
->left_outer_join(
|
|
MP_NEWSLETTER_OPTION_FIELDS_TABLE,
|
|
array(
|
|
MP_NEWSLETTER_OPTION_FIELDS_TABLE.'.id',
|
|
'=',
|
|
MP_NEWSLETTER_OPTION_TABLE.'.option_field_id'
|
|
)
|
|
)
|
|
->group_by(MP_NEWSLETTERS_TABLE.'.id');
|
|
return $orm;
|
|
}
|
|
|
|
static function groups($data = array()) {
|
|
$type = isset($data['params']['type']) ? $data['params']['type'] : null;
|
|
|
|
// newsletter types without groups
|
|
if(in_array($type, array(
|
|
self::TYPE_NOTIFICATION_HISTORY
|
|
))) {
|
|
return false;
|
|
}
|
|
|
|
$groups = array(
|
|
array(
|
|
'name' => 'all',
|
|
'label' => __('All', 'mailpoet'),
|
|
'count' => Newsletter::getPublished()
|
|
->filter('filterType', $type)
|
|
->count()
|
|
)
|
|
);
|
|
|
|
switch($type) {
|
|
case self::TYPE_STANDARD:
|
|
$groups = array_merge($groups, array(
|
|
array(
|
|
'name' => self::STATUS_DRAFT,
|
|
'label' => __('Draft', 'mailpoet'),
|
|
'count' => Newsletter::getPublished()
|
|
->filter('filterType', $type)
|
|
->filter('filterStatus', self::STATUS_DRAFT)
|
|
->count()
|
|
),
|
|
array(
|
|
'name' => self::STATUS_SCHEDULED,
|
|
'label' => __('Scheduled', 'mailpoet'),
|
|
'count' => Newsletter::getPublished()
|
|
->filter('filterType', $type)
|
|
->filter('filterStatus', self::STATUS_SCHEDULED)
|
|
->count()
|
|
),
|
|
array(
|
|
'name' => self::STATUS_SENDING,
|
|
'label' => __('Sending', 'mailpoet'),
|
|
'count' => Newsletter::getPublished()
|
|
->filter('filterType', $type)
|
|
->filter('filterStatus', self::STATUS_SENDING)
|
|
->count()
|
|
),
|
|
array(
|
|
'name' => self::STATUS_SENT,
|
|
'label' => __('Sent', 'mailpoet'),
|
|
'count' => Newsletter::getPublished()
|
|
->filter('filterType', $type)
|
|
->filter('filterStatus', self::STATUS_SENT)
|
|
->count()
|
|
)
|
|
));
|
|
break;
|
|
|
|
case self::TYPE_WELCOME:
|
|
case self::TYPE_NOTIFICATION:
|
|
$groups = array_merge($groups, array(
|
|
array(
|
|
'name' => self::STATUS_ACTIVE,
|
|
'label' => __('Active', 'mailpoet'),
|
|
'count' => Newsletter::getPublished()
|
|
->filter('filterType', $type)
|
|
->filter('filterStatus', self::STATUS_ACTIVE)
|
|
->count()
|
|
),
|
|
array(
|
|
'name' => self::STATUS_DRAFT,
|
|
'label' => __('Not active', 'mailpoet'),
|
|
'count' => Newsletter::getPublished()
|
|
->filter('filterType', $type)
|
|
->filter('filterStatus', self::STATUS_DRAFT)
|
|
->count()
|
|
)
|
|
));
|
|
break;
|
|
}
|
|
|
|
$groups[] = array(
|
|
'name' => 'trash',
|
|
'label' => __('Trash', 'mailpoet'),
|
|
'count' => Newsletter::getTrashed()
|
|
->filter('filterType', $type)
|
|
->count()
|
|
);
|
|
|
|
return $groups;
|
|
}
|
|
|
|
static function groupBy($orm, $data = array()) {
|
|
$group = (!empty($data['group'])) ? $data['group'] : 'all';
|
|
|
|
switch($group) {
|
|
case self::STATUS_DRAFT:
|
|
case self::STATUS_SCHEDULED:
|
|
case self::STATUS_SENDING:
|
|
case self::STATUS_SENT:
|
|
case self::STATUS_ACTIVE:
|
|
$orm
|
|
->whereNull('deleted_at')
|
|
->filter('filterStatus', $group);
|
|
break;
|
|
|
|
case 'trash':
|
|
$orm->whereNotNull('deleted_at');
|
|
break;
|
|
|
|
default:
|
|
$orm->whereNull('deleted_at');
|
|
}
|
|
return $orm;
|
|
}
|
|
|
|
static function filterStatus($orm, $status = false) {
|
|
if(in_array($status, array(
|
|
self::STATUS_DRAFT,
|
|
self::STATUS_SCHEDULED,
|
|
self::STATUS_SENDING,
|
|
self::STATUS_SENT,
|
|
self::STATUS_ACTIVE
|
|
))) {
|
|
$orm->where('status', $status);
|
|
}
|
|
return $orm;
|
|
}
|
|
|
|
static function filterType($orm, $type = false) {
|
|
if(in_array($type, array(
|
|
self::TYPE_STANDARD,
|
|
self::TYPE_WELCOME,
|
|
self::TYPE_NOTIFICATION,
|
|
self::TYPE_NOTIFICATION_HISTORY
|
|
))) {
|
|
$orm->where('type', $type);
|
|
}
|
|
return $orm;
|
|
}
|
|
|
|
static function listingQuery($data = array()) {
|
|
return self::select(array(
|
|
'id',
|
|
'subject',
|
|
'hash',
|
|
'type',
|
|
'status',
|
|
'updated_at',
|
|
'deleted_at'
|
|
))
|
|
->filter('filterBy', $data)
|
|
->filter('groupBy', $data)
|
|
->filter('search', $data['search']);
|
|
}
|
|
|
|
static function createOrUpdate($data = array()) {
|
|
$newsletter = false;
|
|
|
|
if(isset($data['id']) && (int)$data['id'] > 0) {
|
|
$newsletter = self::findOne((int)$data['id']);
|
|
}
|
|
|
|
if($newsletter === false) {
|
|
$newsletter = self::create();
|
|
|
|
// set default sender based on settings
|
|
if(empty($data['sender'])) {
|
|
$sender = Setting::getValue('sender', array());
|
|
$data['sender_name'] = (
|
|
!empty($sender['name'])
|
|
? $sender['name']
|
|
: ''
|
|
);
|
|
$data['sender_address'] = (
|
|
!empty($sender['address'])
|
|
? $sender['address']
|
|
: ''
|
|
);
|
|
}
|
|
|
|
// set default reply_to based on settings
|
|
if(empty($data['reply_to'])) {
|
|
$reply_to = Setting::getValue('reply_to', array());
|
|
$data['reply_to_name'] = (
|
|
!empty($reply_to['name'])
|
|
? $reply_to['name']
|
|
: ''
|
|
);
|
|
$data['reply_to_address'] = (
|
|
!empty($reply_to['address'])
|
|
? $reply_to['address']
|
|
: ''
|
|
);
|
|
}
|
|
|
|
$newsletter->hydrate($data);
|
|
} else {
|
|
unset($data['id']);
|
|
$newsletter->set($data);
|
|
}
|
|
|
|
$newsletter->save();
|
|
return $newsletter;
|
|
}
|
|
|
|
static function getWelcomeNotificationsForSegments($segments) {
|
|
return NewsletterOption::table_alias('options')
|
|
->select('options.newsletter_id')
|
|
->select('options.value', 'segment_id')
|
|
->join(
|
|
self::$_table,
|
|
'newsletters.id = options.newsletter_id',
|
|
'newsletters'
|
|
)
|
|
->join(
|
|
MP_NEWSLETTER_OPTION_FIELDS_TABLE,
|
|
'option_fields.id = options.option_field_id',
|
|
'option_fields'
|
|
)
|
|
->whereNull('newsletters.deleted_at')
|
|
->where('newsletters.type', 'welcome')
|
|
->where('option_fields.name', 'segment')
|
|
->whereIn('options.value', $segments)
|
|
->findMany();
|
|
}
|
|
|
|
static function getArchives($segment_ids = array()) {
|
|
$orm = self::table_alias('newsletters')
|
|
->distinct()->select('newsletters.*')
|
|
->whereIn('newsletters.type', array(
|
|
self::TYPE_STANDARD,
|
|
self::TYPE_NOTIFICATION_HISTORY
|
|
))
|
|
->join(
|
|
MP_SENDING_QUEUES_TABLE,
|
|
'queues.newsletter_id = newsletters.id',
|
|
'queues'
|
|
)
|
|
->where('queues.status', SendingQueue::STATUS_COMPLETED)
|
|
->whereNull('newsletters.deleted_at')
|
|
->select('queues.processed_at')
|
|
->orderByDesc('queues.processed_at');
|
|
|
|
if(!empty($segment_ids)) {
|
|
$orm->join(
|
|
MP_NEWSLETTER_SEGMENT_TABLE,
|
|
'newsletter_segments.newsletter_id = newsletters.id',
|
|
'newsletter_segments'
|
|
)
|
|
->whereIn('newsletter_segments.segment_id', $segment_ids);
|
|
}
|
|
return $orm->findMany();
|
|
}
|
|
|
|
static function getByHash($hash) {
|
|
return parent::where('hash', $hash)
|
|
->findOne();
|
|
}
|
|
|
|
static function generateHash() {
|
|
return substr(
|
|
md5(AUTH_KEY . Security::generateRandomString(15)),
|
|
0,
|
|
self::NEWSLETTER_HASH_LENGTH
|
|
);
|
|
}
|
|
|
|
function restore() {
|
|
if($this->status == self::STATUS_SENDING) {
|
|
$this->set('status', self::STATUS_DRAFT);
|
|
$this->save();
|
|
}
|
|
return parent::restore();
|
|
}
|
|
|
|
static function bulkRestore($orm) {
|
|
parent::bulkAction($orm, function($ids) {
|
|
Newsletter::whereIn('id', $ids)
|
|
->where('status', Newsletter::STATUS_SENDING)
|
|
->findResultSet()
|
|
->set('status', Newsletter::STATUS_DRAFT)
|
|
->save();
|
|
});
|
|
return parent::bulkRestore($orm);
|
|
}
|
|
}
|