rules-8.x-3.x-dev/src/Plugin/RulesAction/SystemEmailToUsersOfRole.php
src/Plugin/RulesAction/SystemEmailToUsersOfRole.php
<?php
namespace Drupal\rules\Plugin\RulesAction;
use Drupal\Core\Language\LanguageInterface;
use Drupal\Core\Logger\LoggerChannelInterface;
use Drupal\Core\Mail\MailManagerInterface;
use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
use Drupal\Core\StringTranslation\TranslatableMarkup;
use Drupal\rules\Context\ContextDefinition;
use Drupal\rules\Core\Attribute\RulesAction;
use Drupal\rules\Core\RulesActionBase;
use Drupal\rules\TypedData\Options\LanguageOptions;
use Drupal\rules\TypedData\Options\RolesOptions;
use Drupal\user\UserStorageInterface;
use Symfony\Component\DependencyInjection\ContainerInterface;
/**
* Provides a 'Email to users of a role' action.
*
* @todo Add access callback information from Drupal 7.
*
* @RulesAction(
* id = "rules_email_to_users_of_role",
* label = @Translation("Send email to all users of a role"),
* category = @Translation("System"),
* context_definitions = {
* "roles" = @ContextDefinition("entity:user_role",
* label = @Translation("Roles"),
* description = @Translation("The roles to which to send the email."),
* options_provider = "\Drupal\rules\TypedData\Options\RolesOptions",
* multiple = TRUE
* ),
* "subject" = @ContextDefinition("string",
* label = @Translation("Subject"),
* description = @Translation("The email's subject.")
* ),
* "message" = @ContextDefinition("string",
* label = @Translation("Message"),
* description = @Translation("The email's message body. Drupal will by default remove all HTML tags. If you want to use HTML you must override this behavior by installing a contributed module such as Mime Mail.")
* ),
* "reply" = @ContextDefinition("email",
* label = @Translation("Reply to"),
* description = @Translation("The email's reply-to address. Leave it empty to use the site-wide configured address."),
* default_value = NULL,
* required = FALSE
* ),
* "language" = @ContextDefinition("language",
* label = @Translation("Language"),
* description = @Translation("If specified, the language object (not language code) used for getting the email message and subject."),
* options_provider = "\Drupal\rules\TypedData\Options\LanguageOptions",
* default_value = NULL,
* required = FALSE
* ),
* }
* )
*/
#[RulesAction(
id: "rules_email_to_users_of_role",
label: new TranslatableMarkup("Send email to all users of a role"),
category: new TranslatableMarkup("System"),
context_definitions: [
"roles" => new ContextDefinition(
data_type: "entity:user_role",
label: new TranslatableMarkup("Roles"),
description: new TranslatableMarkup("The roles to which to send the email."),
options_provider: RolesOptions::class,
multiple: TRUE
),
"subject" => new ContextDefinition(
data_type: "string",
label: new TranslatableMarkup("Subject"),
description: new TranslatableMarkup("The email's subject.")
),
"message" => new ContextDefinition(
data_type: "string",
label: new TranslatableMarkup("Message"),
description: new TranslatableMarkup("The email's message body. Drupal will by default remove all HTML tags. If you want to use HTML you must override this behavior by installing a contributed module such as Mime Mail.")
),
"reply" => new ContextDefinition(
data_type: "email",
label: new TranslatableMarkup("Reply to"),
description: new TranslatableMarkup("The email's reply-to address. Leave it empty to use the site-wide configured address."),
default_value: NULL,
required: FALSE
),
"language" => new ContextDefinition(
data_type: "language",
label: new TranslatableMarkup("Language"),
description: new TranslatableMarkup("If specified, the language object (not language code) used for getting the email message and subject."),
options_provider: LanguageOptions::class,
default_value: NULL,
required: FALSE
),
]
)]
class SystemEmailToUsersOfRole extends RulesActionBase implements ContainerFactoryPluginInterface {
/**
* @var \Drupal\Core\Logger\LoggerChannelInterface
*/
protected $logger;
/**
* @var \Drupal\Core\Mail\MailManagerInterface
*/
protected $mailManager;
/**
* @var \Drupal\user\UserStorageInterface
*/
protected $userStorage;
/**
* Constructs a SystemEmailToUsersOfRole object.
*
* @param array $configuration
* A configuration array containing information about the plugin instance.
* @param string $plugin_id
* The plugin ID for the plugin instance.
* @param mixed $plugin_definition
* The plugin implementation definition.
* @param \Drupal\Core\Logger\LoggerChannelInterface $logger
* The rules logger service.
* @param \Drupal\Core\Mail\MailManagerInterface $mail_manager
* The mail manager service.
* @param \Drupal\user\UserStorageInterface $userStorage
* The user storage service.
*/
public function __construct(array $configuration, $plugin_id, $plugin_definition, LoggerChannelInterface $logger, MailManagerInterface $mail_manager, UserStorageInterface $userStorage) {
parent::__construct($configuration, $plugin_id, $plugin_definition);
$this->logger = $logger;
$this->mailManager = $mail_manager;
$this->userStorage = $userStorage;
}
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
return new static(
$configuration,
$plugin_id,
$plugin_definition,
$container->get('logger.factory')->get('rules'),
$container->get('plugin.manager.mail'),
$container->get('entity_type.manager')->getStorage('user')
);
}
/**
* Sends an email to all users of the specified role(s).
*
* @param \Drupal\user\RoleInterface[] $roles
* Array of user roles.
* @param string $subject
* Subject of the email.
* @param string $message
* Email message text.
* @param string $reply
* (optional) Reply to email address.
* @param \Drupal\Core\Language\LanguageInterface|null $language
* (optional) Language object. If not specified, email will be sent to each
* recipient in the recipient's preferred language.
*/
protected function doExecute(array $roles, string $subject, string $message, ?string $reply = NULL, ?LanguageInterface $language = NULL): void {
if (empty($roles)) {
return;
}
$rids = array_map(function ($role) {
return $role->id();
}, $roles);
// Set a unique key for this email.
$key = 'rules_action_mail_' . $this->getPluginId();
// Select only active users, based on the roles given. We do not want to
// send email to blocked users.
$accounts = $this->userStorage->loadByProperties([
'roles' => $rids,
'status' => 1,
]);
$params = [
'subject' => $subject,
'message' => $message,
];
// Loop over users and send email to each individually using that user's
// preferred language (or a fixed language, if passed in the context).
$number = 0;
foreach ($accounts as $account) {
// Language to use. Value passed in the context takes precedence.
// ORIG.
$langcode = isset($language) ? $language->getId() : $account->getPreferredLangcode();
// @todo Is this better?
$langcode = (isset($language) && $language->getId() != LanguageInterface::LANGCODE_NOT_SPECIFIED) ? $language->getId() : $account->getPreferredLangcode();
$message = $this->mailManager->mail('rules', $key, $account->getEmail(), $langcode, $params, NULL);
$number += $message['result'] ? 1 : 0;
}
$this->logger->notice('Successfully sent email to %number out of %count users having the role(s) %roles', [
'%number' => $number,
'%count' => count($accounts),
'%roles' => implode(', ', $rids),
]);
}
}
