eca-1.0.x-dev/modules/base/src/Plugin/Action/TokenSetValue.php
modules/base/src/Plugin/Action/TokenSetValue.php
<?php
namespace Drupal\eca_base\Plugin\Action;
use Drupal\Core\Access\AccessResult;
use Drupal\Core\Action\Attribute\Action;
use Drupal\Core\Field\EntityReferenceFieldItemListInterface;
use Drupal\Core\Field\FieldItemListInterface;
use Drupal\Core\Field\Plugin\Field\FieldType\EntityReferenceItem;
use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\Session\AccountInterface;
use Drupal\Core\StringTranslation\TranslatableMarkup;
use Drupal\Core\TypedData\TypedDataInterface;
use Drupal\eca\Attribute\EcaAction;
use Drupal\eca\Plugin\Action\ConfigurableActionBase;
use Drupal\eca\Plugin\DataType\DataTransferObject;
use Drupal\eca\Plugin\FormFieldYamlTrait;
use Drupal\eca\Service\YamlParser;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\Yaml\Exception\ParseException;
/**
* Action to set an arbitrary token value.
*/
#[Action(
id: 'eca_token_set_value',
label: new TranslatableMarkup('Token: set value'),
)]
#[EcaAction(
description: new TranslatableMarkup('Define a locally available token by a specific name and value.'),
version_introduced: '1.1.0',
)]
class TokenSetValue extends ConfigurableActionBase {
use FormFieldYamlTrait;
/**
* The YAML parser.
*
* @var \Drupal\eca\Service\YamlParser
*/
protected YamlParser $yamlParser;
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition): static {
$instance = parent::create($container, $configuration, $plugin_id, $plugin_definition);
$instance->setYamlParser($container->get('eca.service.yaml_parser'));
return $instance;
}
/**
* {@inheritdoc}
*/
public function access($object, ?AccountInterface $account = NULL, $return_as_object = FALSE) {
$result = parent::access($object, $account, TRUE);
if ($result->isAllowed() && $this->configuration['use_yaml'] && $this->configuration['validate_yaml']) {
try {
$this->yamlParser->parse($this->configuration['token_value']);
}
catch (ParseException) {
$result = AccessResult::forbidden('YAML data is not valid.');
}
}
return $return_as_object ? $result : $result->isAllowed();
}
/**
* {@inheritdoc}
*/
public function execute(): void {
$name = $this->configuration['token_name'];
$value = $this->configuration['token_value'];
if ($this->configuration['use_yaml']) {
try {
$value = $this->yamlParser->parse($value);
}
catch (ParseException) {
$this->logger->error('Tried parsing a Token value in action "eca_token_set_value" as YAML format, but parsing failed.');
return;
}
}
else {
// Allow direct assignment of available data from the Token environment.
$value = $this->tokenService->getOrReplace($value);
if ($value instanceof DataTransferObject) {
// Wrap the values with a new DTO, for example for having a new list.
$value = DataTransferObject::create($value->getValue());
}
elseif ($value instanceof EntityReferenceFieldItemListInterface) {
// Extract all referenced entities.
$referenced = $value->referencedEntities();
if ((count($referenced) === 1) && ($value->getFieldDefinition()->getFieldStorageDefinition()->getCardinality() === 1)) {
$referenced = reset($referenced);
}
$value = $referenced;
}
elseif ($value instanceof EntityReferenceItem) {
// Extract the single targeted referenced entity.
if (isset($value->entity)) {
$referenced = $value->entity;
}
else {
$referenced = NULL;
$items = $value->getParent();
if (($items instanceof EntityReferenceFieldItemListInterface) && ($entities = $items->referencedEntities())) {
foreach ($items as $delta => $item) {
if (($item === $value) || ($item->getValue() === $value->getValue())) {
$referenced = $entities[$delta] ?? NULL;
break;
}
}
}
}
$value = $referenced;
}
elseif ($value instanceof TypedDataInterface) {
$use_first_item = ($value instanceof FieldItemListInterface) && ($value->getFieldDefinition()->getFieldStorageDefinition()->getCardinality() === 1);
// Extract the value from typed data, such as field item lists.
$value = $value->getValue();
if ($use_first_item) {
$value = reset($value);
}
}
}
$this->setToken($name, $value);
}
/**
* Sets the token.
*
* @param string $name
* The token name.
* @param mixed $value
* The token value.
*/
protected function setToken(string $name, mixed $value): void {
$name = trim($name);
if ($name === '') {
// Without a token name specified, a token cannot be set.
return;
}
$this->tokenService->addTokenData($name, $value);
}
/**
* {@inheritdoc}
*/
public function defaultConfiguration(): array {
return [
'token_name' => '',
'token_value' => '',
'use_yaml' => FALSE,
'validate_yaml' => FALSE,
] + parent::defaultConfiguration();
}
/**
* {@inheritdoc}
*/
public function buildConfigurationForm(array $form, FormStateInterface $form_state): array {
$form['token_name'] = [
'#type' => 'textfield',
'#title' => $this->t('Name of token'),
'#default_value' => $this->configuration['token_name'],
'#weight' => -30,
'#description' => $this->t('Provide the name of a token where the value should be stored.'),
'#eca_token_reference' => TRUE,
];
$form['token_value'] = [
'#type' => 'textarea',
'#title' => $this->t('Value of the token'),
'#default_value' => $this->configuration['token_value'],
'#weight' => -20,
'#description' => $this->t('The value of the token.'),
'#eca_token_replacement' => TRUE,
];
$this->buildYamlFormFields(
$form,
$this->t('Interpret above value as YAML format'),
$this->t('Nested data can be set using YAML format, for example <em>mykey: "My value"</em>. When using this format, this option needs to be enabled. When using tokens and YAML altogether, make sure that tokens are wrapped as a string. Example: <em>title: "[node:title]"</em>'),
-10,
);
return parent::buildConfigurationForm($form, $form_state);
}
/**
* {@inheritdoc}
*/
public function submitConfigurationForm(array &$form, FormStateInterface $form_state): void {
$this->configuration['token_name'] = $form_state->getValue('token_name');
$this->configuration['token_value'] = $form_state->getValue('token_value');
$this->configuration['use_yaml'] = !empty($form_state->getValue('use_yaml'));
$this->configuration['validate_yaml'] = !empty($form_state->getValue('validate_yaml'));
parent::submitConfigurationForm($form, $form_state);
}
/**
* Set the YAML parser.
*
* @param \Drupal\eca\Service\YamlParser $yaml_parser
* The YAML parser.
*/
public function setYamlParser(YamlParser $yaml_parser): void {
$this->yamlParser = $yaml_parser;
}
}
