Skip to content
Snippets Groups Projects

Issue #3173210 - Refactor properties field mapping plugins

Merged Aaron Bauman requested to merge issue/salesforce-3173210:3173210 into 8.x-4.x
3 files
+ 244
502
Compare changes
  • Side-by-side
  • Inline
Files
3
@@ -2,27 +2,11 @@
namespace Drupal\salesforce_mapping\Plugin\SalesforceMappingField;
use Drupal\Core\Datetime\DateFormatterInterface;
use Drupal\Core\Entity\EntityFieldManagerInterface;
use Drupal\Core\Entity\EntityInterface;
use Drupal\Core\Entity\EntityTypeBundleInfoInterface;
use Drupal\Core\Entity\EntityTypeManagerInterface;
use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\TypedData\ComplexDataDefinitionInterface;
use Drupal\Core\TypedData\ComplexDataInterface;
use Drupal\Core\TypedData\DataDefinitionInterface;
use Drupal\Core\TypedData\Exception\MissingDataException;
use Drupal\Core\TypedData\ListDataDefinitionInterface;
use Drupal\Core\TypedData\TypedDataInterface;
use Drupal\field\Entity\FieldConfig;
use Drupal\salesforce\Rest\RestClientInterface;
use Drupal\salesforce\SObject;
use Drupal\salesforce_mapping\SalesforceMappingFieldPluginBase;
use Drupal\salesforce_mapping\Entity\SalesforceMappingInterface;
use Drupal\typed_data\DataFetcherInterface;
use Drupal\typed_data\Exception\InvalidArgumentException;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\EventDispatcher\EventDispatcherInterface;
/**
* Adapter for entity properties and fields.
@@ -32,61 +16,7 @@ use Symfony\Component\EventDispatcher\EventDispatcherInterface;
* label = @Translation("Properties")
* )
*/
class Properties extends SalesforceMappingFieldPluginBase {
/**
* Data fetcher service.
*
* @var \Drupal\typed_data\DataFetcherInterface
*/
protected $dataFetcher;
/**
* Properties constructor.
*
* @param array $configuration
* Plugin config.
* @param string $plugin_id
* Plugin id.
* @param array $plugin_definition
* Plugin definition.
* @param \Drupal\Core\Entity\EntityTypeBundleInfoInterface $entity_type_bundle_info
* Entity type bundle info service.
* @param \Drupal\Core\Entity\EntityFieldManagerInterface $entity_field_manager
* Entity field manager.
* @param \Drupal\salesforce\Rest\RestClientInterface $rest_client
* Salesforce client.
* @param \Drupal\Core\Entity\EntityTypeManagerInterface $etm
* ETM service.
* @param \Drupal\Core\Datetime\DateFormatterInterface $dateFormatter
* Date formatter service.
* @param \Symfony\Component\EventDispatcher\EventDispatcherInterface $event_dispatcher
* Event dispatcher service.
* @param \Drupal\typed_data\DataFetcherInterface $dataFetcher
* Data fetcher.
*
* @throws \Drupal\Component\Plugin\Exception\InvalidPluginDefinitionException
* @throws \Drupal\Component\Plugin\Exception\PluginNotFoundException
*/
public function __construct(array $configuration, $plugin_id, array $plugin_definition, EntityTypeBundleInfoInterface $entity_type_bundle_info, EntityFieldManagerInterface $entity_field_manager, RestClientInterface $rest_client, EntityTypeManagerInterface $etm, DateFormatterInterface $dateFormatter, EventDispatcherInterface $event_dispatcher, DataFetcherInterface $dataFetcher) {
parent::__construct($configuration, $plugin_id, $plugin_definition, $entity_type_bundle_info, $entity_field_manager, $rest_client, $etm, $dateFormatter, $event_dispatcher);
$this->dataFetcher = $dataFetcher;
}
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
return new static($configuration, $plugin_id, $plugin_definition,
$container->get('entity_type.bundle.info'),
$container->get('entity_field.manager'),
$container->get('salesforce.client'),
$container->get('entity_type.manager'),
$container->get('date.formatter'),
$container->get('event_dispatcher'),
$container->get('typed_data.data_fetcher')
);
}
class Properties extends PropertiesBase {
/**
* Implementation of PluginFormInterface::buildConfigurationForm.
@@ -115,125 +45,10 @@ class Properties extends SalesforceMappingFieldPluginBase {
return $pluginForm;
}
/**
* {@inheritdoc}
*/
public function validateConfigurationForm(array &$form, FormStateInterface $form_state) {
parent::validateConfigurationForm($form, $form_state);
$vals = $form_state->getValues();
$config = $vals['config'];
if (empty($config['salesforce_field'])) {
$form_state->setError($form['config']['salesforce_field'], $this->t('Salesforce field is required.'));
}
if (empty($config['drupal_field_value'])) {
$form_state->setError($form['config']['drupal_field_value'], $this->t('Drupal field is required.'));
}
}
/**
* {@inheritdoc}
*/
public function value(EntityInterface $entity, SalesforceMappingInterface $mapping) {
// No error checking here. If a property is not defined, it's a
// configuration bug that needs to be solved elsewhere.
// Multipicklist is the only target type that handles multi-valued fields.
$describe = $this
->salesforceClient
->objectDescribe($mapping->getSalesforceObjectType());
$field_definition = $describe->getField($this->config('salesforce_field'));
if ($field_definition['type'] == 'multipicklist') {
$values = [];
foreach ($entity->get($this->config('drupal_field_value')) as $value) {
$values[] = $value->value;
}
return implode(';', $values);
}
else {
return $this->getStringValue($entity, $this->config('drupal_field_value'));
}
}
/**
* {@inheritdoc}
*/
public function pullValue(SObject $sf_object, EntityInterface $entity, SalesforceMappingInterface $mapping) {
$field_selector = $this->config('drupal_field_value');
$pullValue = parent::pullValue($sf_object, $entity, $mapping);
try {
// Fetch the TypedData property and set its value.
$data = $this->dataFetcher->fetchDataByPropertyPath($entity->getTypedData(), $field_selector);
$data->setValue($pullValue);
return $data;
}
catch (MissingDataException $e) {
}
catch (InvalidArgumentException $e) {
}
// Allow any other exception types to percolate.
// If the entity doesn't have any value in the field, data fetch will
// throw an exception. We must attempt to create the field.
// Typed Data API doesn't provide any good way to initialize a field value
// given a selector. Instead we have to do it ourselves.
// We descend only to the first-level fields on the entity. Cascading pull
// values to entity references is not supported.
$parts = explode('.', $field_selector, 4);
switch (count($parts)) {
case 1:
$entity->set($field_selector, $pullValue);
return $entity->getTypedData()->get($field_selector);
case 2:
$field_name = $parts[0];
$delta = 0;
$property = $parts[1];
break;
case 3:
$field_name = $parts[0];
$delta = $parts[1];
$property = $parts[2];
if (!is_numeric($delta)) {
return;
}
break;
case 4:
return;
}
/** @var \Drupal\Core\TypedData\ListInterface $list_data */
$list_data = $entity->get($field_name);
// If the given delta has not been initialized, initialize it.
if (!$list_data->get($delta) instanceof TypedDataInterface) {
$list_data->set($delta, []);
}
/** @var \Drupal\Core\TypedData\TypedDataInterface|\Drupal\Core\TypedData\ComplexDataInterface $typed_data */
$typed_data = $list_data->get($delta);
if ($typed_data instanceof ComplexDataInterface && $property) {
// If the given property has not been initialized, initialize it.
if (!$typed_data->get($property) instanceof TypedDataInterface) {
$typed_data->set($property, []);
}
/** @var \Drupal\Core\TypedData\TypedDataInterface $typed_data */
$typed_data = $typed_data->get($property);
}
if (!$typed_data instanceof TypedDataInterface) {
return;
}
$typed_data->setValue($pullValue);
return $typed_data->getParent();
}
/**
* Form options helper.
*/
private function getConfigurationOptions(SalesforceMappingInterface $mapping) {
protected function getConfigurationOptions(SalesforceMappingInterface $mapping) {
/** @var \Drupal\Core\Field\FieldDefinitionInterface[] $field_definitions */
$field_definitions = $this->entityFieldManager->getFieldDefinitions(
$mapping->get('drupal_entity_type'),
@@ -249,7 +64,8 @@ class Properties extends SalesforceMappingFieldPluginBase {
}
else {
// Get a list of property definitions.
$property_definitions = $field_definition->getFieldStorageDefinition()->getPropertyDefinitions();
$property_definitions = $field_definition->getFieldStorageDefinition()
->getPropertyDefinitions();
if (count($property_definitions) > 1) {
foreach ($property_definitions as $property => $property_definition) {
$options[(string) $label][$field_name . '.' . $property] = $label . ': ' . $property_definition->getLabel();
@@ -275,69 +91,4 @@ class Properties extends SalesforceMappingFieldPluginBase {
return $definition;
}
/**
* {@inheritdoc}
*/
public function checkFieldMappingDependency(array $dependencies) {
$definition = $this->getPluginDefinition();
foreach ($definition['config_dependencies'] as $type => $dependency) {
foreach ($dependency as $item) {
if (!empty($dependencies[$type][$item])) {
return TRUE;
}
}
}
}
/**
* Helper Method to check for and retrieve field data.
*
* If it is just a regular field/property of the entity, the data is
* retrieved with ->value(). If this is a property referenced using the
* typed_data module's extension, use typed_data module's DataFetcher class
* to retrieve the value.
*
* @param \Drupal\Core\Entity\EntityInterface $entity
* The entity to search the Typed Data for.
* @param string $drupal_field_value
* The Typed Data property to get.
*
* @return string
* The String representation of the Typed Data property value.
*/
protected function getStringValue(EntityInterface $entity, $drupal_field_value) {
try {
return $this->dataFetcher->fetchDataByPropertyPath($entity->getTypedData(), $drupal_field_value)->getString();
}
catch (\Exception $e) {
return NULL;
}
}
/**
* {@inheritdoc}
*/
protected function getFieldDataDefinition(EntityInterface $entity) {
if (!strpos($this->config('drupal_field_value'), '.')) {
return parent::getFieldDataDefinition($entity);
}
$data_definition = $this->dataFetcher->fetchDefinitionByPropertyPath($entity->getTypedData()->getDataDefinition(), $this->config('drupal_field_value'));
if ($data_definition instanceof ListDataDefinitionInterface) {
$data_definition = $data_definition->getItemDefinition();
}
return $data_definition;
}
/**
* {@inheritdoc}
*/
protected function getDrupalFieldType(DataDefinitionInterface $data_definition) {
$field_main_property = $data_definition;
if ($data_definition instanceof ComplexDataDefinitionInterface) {
$field_main_property = $data_definition
->getPropertyDefinition($data_definition->getMainPropertyName());
}
return $field_main_property ? $field_main_property->getDataType() : NULL;
}
}
Loading