syncStorage = $sync_storage; $this->activeStorage = $active_storage; $this->snapshotStorage = $snapshot_storage; $this->lock = $lock; $this->eventDispatcher = $event_dispatcher; $this->configManager = $config_manager; $this->typedConfigManager = $typed_config; $this->moduleHandler = $module_handler; $this->moduleInstaller = $module_installer; $this->themeHandler = $theme_handler; $this->renderer = $renderer; } /** * {@inheritdoc} */ public static function create(ContainerInterface $container) { return new static( $container->get('config.storage.sync'), $container->get('config.storage'), $container->get('config.storage.snapshot'), $container->get('lock.persistent'), $container->get('event_dispatcher'), $container->get('config.manager'), $container->get('config.typed'), $container->get('module_handler'), $container->get('module_installer'), $container->get('theme_handler'), $container->get('renderer') ); } /** * {@inheritdoc} */ public function getFormId() { return 'config_admin_import_form'; } /** * {@inheritdoc} */ public function buildForm(array $form, FormStateInterface $form_state) { $form['actions'] = ['#type' => 'actions']; $form['actions']['submit'] = [ '#type' => 'submit', '#value' => $this->t('Import all'), ]; $source_list = $this->syncStorage->listAll(); $storage_comparer = new StorageComparer($this->syncStorage, $this->activeStorage, $this->configManager); if (empty($source_list) || !$storage_comparer->createChangelist()->hasChanges()) { $form['no_changes'] = [ '#type' => 'table', '#header' => [$this->t('Name'), $this->t('Operations')], '#rows' => [], '#empty' => $this->t('There are no configuration changes to import.'), ]; $form['actions']['#access'] = FALSE; return $form; } elseif (!$storage_comparer->validateSiteUuid()) { $this->messenger()->addError($this->t('The staged configuration cannot be imported, because it originates from a different site than this site. You can only synchronize configuration between cloned instances of this site.')); $form['actions']['#access'] = FALSE; return $form; } // A list of changes will be displayed, so check if the user should be // warned of potential losses to configuration. if ($this->snapshotStorage->exists('core.extension')) { $snapshot_comparer = new StorageComparer($this->activeStorage, $this->snapshotStorage, $this->configManager); if (!$form_state->getUserInput() && $snapshot_comparer->createChangelist()->hasChanges()) { $change_list = []; foreach ($snapshot_comparer->getAllCollectionNames() as $collection) { foreach ($snapshot_comparer->getChangelist(NULL, $collection) as $config_names) { if (empty($config_names)) { continue; } foreach ($config_names as $config_name) { $change_list[] = $config_name; } } } sort($change_list); $message = [ [ '#markup' => $this->t('The following items in your active configuration have changes since the last import that may be lost on the next import.'), ], [ '#theme' => 'item_list', '#items' => $change_list, ], ]; $this->messenger()->addWarning($this->renderer->renderPlain($message)); } } // Store the comparer for use in the submit. $form_state->set('storage_comparer', $storage_comparer); // Add the AJAX library to the form for dialog support. $form['#attached']['library'][] = 'core/drupal.dialog.ajax'; foreach ($storage_comparer->getAllCollectionNames() as $collection) { if ($collection != StorageInterface::DEFAULT_COLLECTION) { $form[$collection]['collection_heading'] = [ '#type' => 'html_tag', '#tag' => 'h2', '#value' => $this->t('@collection configuration collection', ['@collection' => $collection]), ]; } foreach ($storage_comparer->getChangelist(NULL, $collection) as $config_change_type => $config_names) { if (empty($config_names)) { continue; } // @todo A table caption would be more appropriate, but does not have the // visual importance of a heading. $form[$collection][$config_change_type]['heading'] = [ '#type' => 'html_tag', '#tag' => 'h3', ]; switch ($config_change_type) { case 'create': $form[$collection][$config_change_type]['heading']['#value'] = $this->formatPlural(count($config_names), '@count new', '@count new'); break; case 'update': $form[$collection][$config_change_type]['heading']['#value'] = $this->formatPlural(count($config_names), '@count changed', '@count changed'); break; case 'delete': $form[$collection][$config_change_type]['heading']['#value'] = $this->formatPlural(count($config_names), '@count removed', '@count removed'); break; case 'rename': $form[$collection][$config_change_type]['heading']['#value'] = $this->formatPlural(count($config_names), '@count renamed', '@count renamed'); break; } $form[$collection][$config_change_type]['list'] = [ '#type' => 'table', '#header' => [$this->t('Name'), $this->t('Operations')], ]; foreach ($config_names as $config_name) { if ($config_change_type == 'rename') { $names = $storage_comparer->extractRenameNames($config_name); $route_options = ['source_name' => $names['old_name'], 'target_name' => $names['new_name']]; $config_name = $this->t('@source_name to @target_name', ['@source_name' => $names['old_name'], '@target_name' => $names['new_name']]); } else { $route_options = ['source_name' => $config_name]; } if ($collection != StorageInterface::DEFAULT_COLLECTION) { $route_name = 'config.diff_collection'; $route_options['collection'] = $collection; } else { $route_name = 'config.diff'; } $links['view_diff'] = [ 'title' => $this->t('View differences'), 'url' => Url::fromRoute($route_name, $route_options), 'attributes' => [ 'class' => ['use-ajax'], 'data-dialog-type' => 'modal', 'data-dialog-options' => json_encode([ 'width' => 700, ]), ], ]; $form[$collection][$config_change_type]['list']['#rows'][] = [ 'name' => $config_name, 'operations' => [ 'data' => [ '#type' => 'operations', '#links' => $links, ], ], ]; } } } return $form; } /** * {@inheritdoc} */ public function submitForm(array &$form, FormStateInterface $form_state) { $config_importer = new ConfigImporter( $form_state->get('storage_comparer'), $this->eventDispatcher, $this->configManager, $this->lock, $this->typedConfigManager, $this->moduleHandler, $this->moduleInstaller, $this->themeHandler, $this->getStringTranslation() ); if ($config_importer->alreadyImporting()) { $this->messenger()->addStatus($this->t('Another request may be synchronizing configuration already.')); } else { try { $sync_steps = $config_importer->initialize(); $batch = [ 'operations' => [], 'finished' => [ConfigImporterBatch::class, 'finish'], 'title' => t('Synchronizing configuration'), 'init_message' => t('Starting configuration synchronization.'), 'progress_message' => t('Completed step @current of @total.'), 'error_message' => t('Configuration synchronization has encountered an error.'), ]; foreach ($sync_steps as $sync_step) { $batch['operations'][] = [[ConfigImporterBatch::class, 'process'], [$config_importer, $sync_step]]; } batch_set($batch); } catch (ConfigImporterException $e) { // There are validation errors. $this->messenger()->addError($this->t('The configuration cannot be imported because it failed validation for the following reasons:')); foreach ($config_importer->getErrors() as $message) { $this->messenger()->addError($message); } } } } /** * Processes the config import batch and persists the importer. * * @param \Drupal\Core\Config\ConfigImporter $config_importer * The batch config importer object to persist. * @param string $sync_step * The synchronization step to do. * @param array $context * The batch context. * * @deprecated in Drupal 8.6.0 and will be removed before 9.0.0. Use * \Drupal\Core\Config\Importer\ConfigImporterBatch::process() instead. * * @see https://www.drupal.org/node/2897299 */ public static function processBatch(ConfigImporter $config_importer, $sync_step, &$context) { @trigger_error('\Drupal\config\Form\ConfigSync::processBatch() deprecated in Drupal 8.6.0 and will be removed before 9.0.0. Use \Drupal\Core\Config\Importer\ConfigImporterBatch::process() instead. See https://www.drupal.org/node/2897299'); ConfigImporterBatch::process($config_importer, $sync_step, $context); } /** * Finish batch. * * This function is a static function to avoid serializing the ConfigSync * object unnecessarily. * * @deprecated in Drupal 8.6.0 and will be removed before 9.0.0. Use * \Drupal\Core\Config\Importer\ConfigImporterBatch::finish() instead. * * @see https://www.drupal.org/node/2897299 */ public static function finishBatch($success, $results, $operations) { @trigger_error('\Drupal\config\Form\ConfigSync::finishBatch() deprecated in Drupal 8.6.0 and will be removed before 9.0.0. Use \Drupal\Core\Config\Importer\ConfigImporterBatch::finish() instead. See https://www.drupal.org/node/2897299'); ConfigImporterBatch::finish($success, $results, $operations); } }