class UpgradeStatusForm in Upgrade Status 8
Same name and namespace in other branches
- 8.3 src/Form/UpgradeStatusForm.php \Drupal\upgrade_status\Form\UpgradeStatusForm
- 8.2 src/Form/UpgradeStatusForm.php \Drupal\upgrade_status\Form\UpgradeStatusForm
Hierarchy
- class \Drupal\Core\Form\FormBase implements ContainerInjectionInterface, FormInterface uses DependencySerializationTrait, LoggerChannelTrait, MessengerTrait, LinkGeneratorTrait, RedirectDestinationTrait, UrlGeneratorTrait, StringTranslationTrait
- class \Drupal\upgrade_status\Form\UpgradeStatusForm
Expanded class hierarchy of UpgradeStatusForm
1 string reference to 'UpgradeStatusForm'
File
- src/
Form/ UpgradeStatusForm.php, line 20
Namespace
Drupal\upgrade_status\FormView source
class UpgradeStatusForm extends FormBase {
/**
* The project collector service.
*
* @var \Drupal\upgrade_status\ProjectCollector
*/
protected $projectCollector;
/**
* Available releases store.
*
* @var \Drupal\Core\KeyValueStore\KeyValueStoreInterface|mixed
*/
protected $releaseStore;
/**
* The scan result formatter service.
*
* @var \Drupal\upgrade_status\ScanResultFormatter
*/
protected $resultFormatter;
/**
* The renderer service.
*
* @var \Drupal\Core\Render\RendererInterface
*/
protected $renderer;
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container) {
return new static($container
->get('upgrade_status.project_collector'), $container
->get('keyvalue.expirable'), $container
->get('upgrade_status.result_formatter'), $container
->get('renderer'));
}
/**
* Constructs a Drupal\upgrade_status\Form\UpgradeStatusForm.
*
* @param \Drupal\upgrade_status\ProjectCollector $project_collector
* The project collector service.
* @param \Drupal\Core\KeyValueStore\KeyValueExpirableFactory $key_value_expirable
* The expirable key/value storage.
* @param \Drupal\upgrade_status\ScanResultFormatter $result_formatter
* The scan result formatter service.
* @param \Drupal\Core\Render\RendererInterface $renderer
* The renderer service.
*/
public function __construct(ProjectCollector $project_collector, KeyValueExpirableFactory $key_value_expirable, ScanResultFormatter $result_formatter, RendererInterface $renderer) {
$this->projectCollector = $project_collector;
$this->releaseStore = $key_value_expirable
->get('update_available_releases');
$this->resultFormatter = $result_formatter;
$this->renderer = $renderer;
}
/**
* {@inheritdoc}
*/
public function getFormId() {
return 'drupal_upgrade_status_form';
}
/**
* Form constructor.
*
* @param array $form
* An associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*
* @return array
* The form structure.
*/
public function buildForm(array $form, FormStateInterface $form_state) {
$form['#attached']['library'][] = 'upgrade_status/upgrade_status.admin';
// Gather project list grouped by custom and contrib projects.
$projects = $this->projectCollector
->collectProjects();
// List custom project status first.
$custom = [
'#type' => 'markup',
'#markup' => '<br /><strong>' . $this
->t('No custom projects found.') . '</strong>',
];
if (count($projects['custom'])) {
$custom = $this
->buildProjectList($projects['custom']);
}
$form['custom'] = [
'#type' => 'details',
'#title' => $this
->t('Custom projects'),
'#description' => $this
->t('Custom code is specific to your site, and must be upgraded manually. <a href=":upgrade">Read more about how developers can upgrade their code to Drupal 9</a>.', [
':upgrade' => 'https://www.drupal.org/docs/9/how-drupal-9-is-made-and-what-is-included/how-and-why-we-deprecate-on-the-way-to-drupal-9',
]),
'#open' => TRUE,
'#attributes' => [
'class' => [
'upgrade-status-summary upgrade-status-summary-custom',
],
],
'data' => $custom,
'#tree' => TRUE,
];
// List contrib project status second.
$contrib = [
'#type' => 'markup',
'#markup' => '<br /><strong>' . $this
->t('No contributed projects found.') . '</strong>',
];
if (count($projects['contrib'])) {
$contrib = $this
->buildProjectList($projects['contrib'], TRUE);
}
$form['contrib'] = [
'#type' => 'details',
'#title' => $this
->t('Contributed projects'),
'#description' => $this
->t('Contributed code is available from drupal.org. Problems here may be partially resolved by updating to the latest version. <a href=":update">Read more about how to update contributed projects</a>.', [
':update' => 'https://www.drupal.org/docs/8/update/update-modules',
]),
'#open' => TRUE,
'#attributes' => [
'class' => [
'upgrade-status-summary upgrade-status-summary-contrib',
],
],
'data' => $contrib,
'#tree' => TRUE,
];
$form['drupal_upgrade_status_form']['action']['submit'] = [
'#type' => 'submit',
'#value' => $this
->t('Scan selected'),
'#weight' => 2,
'#button_type' => 'primary',
];
$form['drupal_upgrade_status_form']['action']['export'] = [
'#type' => 'submit',
'#value' => $this
->t('Export as HTML'),
'#weight' => 5,
'#submit' => [
[
$this,
'exportReportHTML',
],
],
];
$form['drupal_upgrade_status_form']['action']['export_ascii'] = [
'#type' => 'submit',
'#value' => $this
->t('Export as ASCII'),
'#weight' => 6,
'#submit' => [
[
$this,
'exportReportASCII',
],
],
];
return $form;
}
/**
* Builds a list and status summary of projects.
*
* @param \Drupal\Core\Extension\Extension[] $projects
* Array of extensions representing projects.
* @param bool $isContrib
* (Optional) Whether the list to be produced is for contributed projects.
*
* @return array
* Build array.
*/
protected function buildProjectList(array $projects, bool $isContrib = FALSE) {
$counters = [
'not-scanned' => 0,
'no-known-error' => 0,
'known-errors' => 0,
'known-warnings' => 0,
'known-error-projects' => 0,
'known-warning-projects' => 0,
];
$header = [
'project' => [
'data' => $this
->t('Project'),
'class' => 'project-label',
],
];
if ($isContrib) {
$header['update'] = [
'data' => $this
->t('Available update'),
'class' => 'update-info',
];
}
$header['status'] = [
'data' => $this
->t('Status'),
'class' => 'status-info',
];
$build['data'] = [
'#type' => 'tableselect',
'#header' => $header,
'#weight' => 20,
'#options' => [],
];
foreach ($projects as $name => $extension) {
// Always use a fresh service. An injected service could get stale results
// because scan result saving happens in different HTTP requests for most
// cases (when analysis was successful).
$scan_result = \Drupal::service('keyvalue')
->get('upgrade_status_scan_results')
->get($name);
$info = $extension->info;
$label = $info['name'] . (!empty($info['version']) ? ' ' . $info['version'] : '');
$update_cell = [
'class' => 'update-info',
'data' => $isContrib ? $this
->t('Up to date') : '',
];
$label_cell = [
'data' => [
'label' => [
'#type' => 'html_tag',
'#tag' => 'label',
'#value' => $label,
'#attributes' => [
'for' => 'edit-' . ($isContrib ? 'contrib' : 'custom') . '-data-data-' . str_replace('_', '-', $name),
],
],
],
'class' => 'project-label',
];
if ($isContrib) {
$projectUpdateData = $this->releaseStore
->get($name);
// @todo: trigger update information fetch.
if (!isset($projectUpdateData['releases']) || is_null($projectUpdateData['releases'])) {
$update_cell = [
'class' => 'update-info',
'data' => $this
->t('N/A'),
];
}
else {
$latestRelease = reset($projectUpdateData['releases']);
$latestVersion = $latestRelease['version'];
if ($info['version'] !== $latestVersion) {
$link = $projectUpdateData['link'] . '/releases/' . $latestVersion;
$update_cell = [
'class' => 'update-info',
'data' => [
'#type' => 'link',
'#title' => $latestVersion,
'#url' => Url::fromUri($link),
],
];
}
}
}
// If this project was not found in our keyvalue storage, it is not yet scanned, report that.
if (empty($scan_result)) {
$build['data']['#options'][$name] = [
'#attributes' => [
'class' => [
'not-scanned',
'project-' . $name,
],
],
'project' => $label_cell,
'update' => $update_cell,
'status' => [
'class' => 'status-info',
'data' => $this
->t('Not scanned'),
],
];
$counters['not-scanned']++;
continue;
}
// Unpack JSON of deprecations to display results.
$report = json_decode($scan_result, TRUE);
if (!empty($report['plans'])) {
$label_cell['data']['plans'] = [
'#type' => 'markup',
'#markup' => '<div>' . $report['plans'] . '</div>',
];
}
if (isset($report['data']['totals'])) {
$project_error_count = $report['data']['totals']['file_errors'];
}
else {
$project_error_count = 0;
}
// If this project had no known issues found, report that.
if ($project_error_count === 0) {
$build['data']['#options'][$name] = [
'#attributes' => [
'class' => [
'no-known-error',
'project-' . $name,
],
],
'project' => $label_cell,
'update' => $update_cell,
'status' => [
'class' => 'status-info',
'data' => $this
->t('No known errors'),
],
];
$counters['no-known-error']++;
continue;
}
// Finally this project had errors found, display them.
$error_label = [];
$error_class = 'known-warnings';
if (!empty($report['data']['totals']['upgrade_status_split']['error'])) {
$counters['known-errors'] += $report['data']['totals']['upgrade_status_split']['error'];
$counters['known-error-projects']++;
$error_class = 'known-errors';
$error_label[] = $this
->formatPlural($report['data']['totals']['upgrade_status_split']['error'], '@count error', '@count errors');
}
if (!empty($report['data']['totals']['upgrade_status_split']['warning'])) {
$counters['known-warnings'] += $report['data']['totals']['upgrade_status_split']['warning'];
$counters['known-warning-projects']++;
$error_label[] = $this
->formatPlural($report['data']['totals']['upgrade_status_split']['warning'], '@count warning', '@count warnings');
}
$build['data']['#options'][$name] = [
'#attributes' => [
'class' => [
$error_class,
'project-' . $name,
],
],
'project' => $label_cell,
'update' => $update_cell,
'status' => [
'class' => 'status-info',
'data' => [
'#type' => 'link',
'#title' => join(', ', $error_label),
'#url' => Url::fromRoute('upgrade_status.project', [
'type' => $extension
->getType(),
'project_machine_name' => $name,
]),
'#attributes' => [
'class' => [
'use-ajax',
],
'data-dialog-type' => 'modal',
'data-dialog-options' => Json::encode([
'width' => 1024,
'height' => 568,
]),
],
],
],
];
}
if (!$isContrib) {
// If the list is not for contrib, remove the update placeholder.
foreach ($build['data']['#options'] as $name => &$row) {
if (is_array($row)) {
unset($row['update']);
}
}
}
$summary = [];
if ($counters['known-errors'] > 0) {
$summary[] = [
'type' => $this
->formatPlural($counters['known-errors'], '1 error', '@count errors'),
'class' => 'error',
'message' => $this
->formatPlural($counters['known-error-projects'], 'Found in one project.', 'Found in @count projects.'),
];
}
if ($counters['known-warnings'] > 0) {
$summary[] = [
'type' => $this
->formatPlural($counters['known-warnings'], '1 warning', '@count warnings'),
'class' => 'warning',
'message' => $this
->formatPlural($counters['known-warning-projects'], 'Found in one project.', 'Found in @count projects.'),
];
}
if ($counters['no-known-error'] > 0) {
$summary[] = [
'type' => $this
->formatPlural($counters['no-known-error'], '1 checked', '@count checked'),
'class' => 'checked',
'message' => $this
->t('No known errors found.'),
];
}
if ($counters['not-scanned'] > 0) {
$summary[] = [
'type' => $this
->formatPlural($counters['not-scanned'], '1 not scanned', '@count not scanned'),
'class' => 'not-scanned',
'message' => $this
->t('Scan to find errors.'),
];
}
$build['summary'] = [
'#theme' => 'upgrade_status_summary_counter',
'#summary' => $summary,
];
return $build;
}
/**
* Form submission handler.
*
* @param array $form
* An associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*/
public function submitForm(array &$form, FormStateInterface $form_state) {
$operations = [];
$projects = $this->projectCollector
->collectProjects();
$submitted = $form_state
->getValues();
foreach ([
'custom',
'contrib',
] as $type) {
if (!empty($submitted[$type])) {
foreach ($submitted[$type]['data']['data'] as $project => $checked) {
if ($checked !== 0) {
// If the checkbox was checked, add a batch operation.
$operations[] = [
static::class . '::parseProject',
[
$projects[$type][$project],
],
];
}
}
}
}
if (!empty($operations)) {
$batch = [
'title' => $this
->t('Scanning projects'),
'operations' => $operations,
];
batch_set($batch);
}
}
/**
* Form submission handler.
*
* @param array $form
* An associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*/
public function exportReportHTML(array &$form, FormStateInterface $form_state) {
$selected = $form_state
->getValues();
$form_state
->setResponse($this
->exportReport($selected, 'html'));
}
/**
* Form submission handler.
*
* @param array $form
* An associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*/
public function exportReportASCII(array &$form, FormStateInterface $form_state) {
$selected = $form_state
->getValues();
$form_state
->setResponse($this
->exportReport($selected, 'ascii'));
}
/**
* Export generator.
*
* @param array $selected
* Selected projects from the form.
* @param string $format
* The format of export to do: html or ascii.
*
* @return \Symfony\Component\HttpFoundation\Response
* Response object for this export.
*/
public function exportReport(array $selected, string $format) {
$extensions = [];
$projects = $this->projectCollector
->collectProjects();
foreach ([
'custom',
'contrib',
] as $type) {
foreach ($selected[$type]['data']['data'] as $project => $checked) {
if ($checked !== 0) {
// If the checkbox was checked, add it to the list.
$extensions[$type][$project] = $format == 'html' ? $this->resultFormatter
->formatResult($projects[$type][$project]) : $this->resultFormatter
->formatAsciiResult($projects[$type][$project]);
}
}
}
$build = [
'#theme' => 'upgrade_status_' . $format . '_export',
'#projects' => $extensions,
];
$fileDate = $this->resultFormatter
->formatDateTime(0, 'html_datetime');
$extension = $format == 'html' ? '.html' : '.txt';
$filename = 'upgrade-status-export-' . $fileDate . $extension;
$response = new Response($this->renderer
->renderRoot($build));
$response->headers
->set('Content-Disposition', 'attachment; filename="' . $filename . '"');
return $response;
}
/**
* Batch callback to analyse a project.
*
* @param \Drupal\Core\Extension\Extension $extension
* The extension to analyse.
* @param array $context
* Batch context.
*/
public static function parseProject(Extension $extension, &$context) {
$context['message'] = t('Completed @project.', [
'@project' => $extension
->getName(),
]);
$error = $file_name = FALSE;
// Prepare for a POST request to scan this project. The separate HTTP
// request is used to separate any PHP errors found from this batch process.
// We can store any errors and gracefully continue if there was any PHP
// errors in parsing.
$url = Url::fromRoute('upgrade_status.analyse', [
'type' => $extension
->getType(),
'project_machine_name' => $extension
->getName(),
]);
// Pass over authentication information because access to this functionality
// requires administrator privileges.
/** @var \Drupal\Core\Session\SessionConfigurationInterface $session_config */
$session_config = \Drupal::service('session_configuration');
$request = \Drupal::request();
$session_options = $session_config
->getOptions($request);
// Unfortunately DrupalCI testbot does not have a domain that would normally
// be considered valid for cookie setting, so we need to work around that
// by manually setting the cookie domain in case there was none. What we
// care about is we get actual results, and cookie on the host level should
// suffice for that.
$cookie_domain = empty($session_options['cookie_domain']) ? '.' . $request
->getHost() : $session_options['cookie_domain'];
$cookie_jar = new CookieJar();
$cookie = new SetCookie([
'Name' => $session_options['name'],
'Value' => $request->cookies
->get($session_options['name']),
'Domain' => $cookie_domain,
'Secure' => $session_options['cookie_secure'],
]);
$cookie_jar
->setCookie($cookie);
$options = [
'cookies' => $cookie_jar,
'timeout' => 0,
];
// Try a POST request with the session cookie included. We expect valid JSON
// back. In case there was a PHP error before that, we log that.
try {
$response = \Drupal::httpClient()
->post($url
->setAbsolute()
->toString(), $options);
$data = json_decode((string) $response
->getBody(), TRUE);
if (!$data) {
$error = (string) $response
->getBody();
$file_name = 'PHP Fatal Error';
}
} catch (\Exception $e) {
$error = $e
->getMessage();
$file_name = 'Scanning exception';
}
if ($error !== FALSE) {
/** @var \Drupal\Core\KeyValueStore\KeyValueStoreInterface $key_value */
$key_value = \Drupal::service('keyvalue')
->get('upgrade_status_scan_results');
$result = [];
$result['date'] = REQUEST_TIME;
$result['data'] = [
'totals' => [
'errors' => 1,
'file_errors' => 1,
'upgrade_status_split' => [
'warning' => 1,
],
],
'files' => [],
];
$result['data']['files'][$file_name] = [
'errors' => 1,
'messages' => [
[
'message' => $error,
'line' => 0,
],
],
];
$key_value
->set($extension
->getName(), json_encode($result));
}
}
}
Members
Name | Modifiers | Type | Description | Overrides |
---|---|---|---|---|
DependencySerializationTrait:: |
protected | property | An array of entity type IDs keyed by the property name of their storages. | |
DependencySerializationTrait:: |
protected | property | An array of service IDs keyed by property name used for serialization. | |
DependencySerializationTrait:: |
public | function | 1 | |
DependencySerializationTrait:: |
public | function | 2 | |
FormBase:: |
protected | property | The config factory. | 1 |
FormBase:: |
protected | property | The request stack. | 1 |
FormBase:: |
protected | property | The route match. | |
FormBase:: |
protected | function | Retrieves a configuration object. | |
FormBase:: |
protected | function | Gets the config factory for this form. | 1 |
FormBase:: |
private | function | Returns the service container. | |
FormBase:: |
protected | function | Gets the current user. | |
FormBase:: |
protected | function | Gets the request object. | |
FormBase:: |
protected | function | Gets the route match. | |
FormBase:: |
protected | function | Gets the logger for a specific channel. | |
FormBase:: |
protected | function |
Returns a redirect response object for the specified route. Overrides UrlGeneratorTrait:: |
|
FormBase:: |
public | function | Resets the configuration factory. | |
FormBase:: |
public | function | Sets the config factory for this form. | |
FormBase:: |
public | function | Sets the request stack object to use. | |
FormBase:: |
public | function |
Form validation handler. Overrides FormInterface:: |
62 |
LinkGeneratorTrait:: |
protected | property | The link generator. | 1 |
LinkGeneratorTrait:: |
protected | function | Returns the link generator. | |
LinkGeneratorTrait:: |
protected | function | Renders a link to a route given a route name and its parameters. | |
LinkGeneratorTrait:: |
public | function | Sets the link generator service. | |
LoggerChannelTrait:: |
protected | property | The logger channel factory service. | |
LoggerChannelTrait:: |
protected | function | Gets the logger for a specific channel. | |
LoggerChannelTrait:: |
public | function | Injects the logger channel factory. | |
MessengerTrait:: |
protected | property | The messenger. | 29 |
MessengerTrait:: |
public | function | Gets the messenger. | 29 |
MessengerTrait:: |
public | function | Sets the messenger. | |
RedirectDestinationTrait:: |
protected | property | The redirect destination service. | 1 |
RedirectDestinationTrait:: |
protected | function | Prepares a 'destination' URL query parameter for use with \Drupal\Core\Url. | |
RedirectDestinationTrait:: |
protected | function | Returns the redirect destination service. | |
RedirectDestinationTrait:: |
public | function | Sets the redirect destination service. | |
StringTranslationTrait:: |
protected | property | The string translation service. | 1 |
StringTranslationTrait:: |
protected | function | Formats a string containing a count of items. | |
StringTranslationTrait:: |
protected | function | Returns the number of plurals supported by a given language. | |
StringTranslationTrait:: |
protected | function | Gets the string translation service. | |
StringTranslationTrait:: |
public | function | Sets the string translation service to use. | 2 |
StringTranslationTrait:: |
protected | function | Translates a string to the current language or to a given language. | |
UpgradeStatusForm:: |
protected | property | The project collector service. | |
UpgradeStatusForm:: |
protected | property | Available releases store. | |
UpgradeStatusForm:: |
protected | property | The renderer service. | |
UpgradeStatusForm:: |
protected | property | The scan result formatter service. | |
UpgradeStatusForm:: |
public | function |
Form constructor. Overrides FormInterface:: |
|
UpgradeStatusForm:: |
protected | function | Builds a list and status summary of projects. | |
UpgradeStatusForm:: |
public static | function |
Instantiates a new instance of this class. Overrides FormBase:: |
|
UpgradeStatusForm:: |
public | function | Export generator. | |
UpgradeStatusForm:: |
public | function | Form submission handler. | |
UpgradeStatusForm:: |
public | function | Form submission handler. | |
UpgradeStatusForm:: |
public | function |
Returns a unique string identifying the form. Overrides FormInterface:: |
|
UpgradeStatusForm:: |
public static | function | Batch callback to analyse a project. | |
UpgradeStatusForm:: |
public | function |
Form submission handler. Overrides FormInterface:: |
|
UpgradeStatusForm:: |
public | function | Constructs a Drupal\upgrade_status\Form\UpgradeStatusForm. | |
UrlGeneratorTrait:: |
protected | property | The url generator. | |
UrlGeneratorTrait:: |
protected | function | Returns the URL generator service. | |
UrlGeneratorTrait:: |
public | function | Sets the URL generator service. | |
UrlGeneratorTrait:: |
protected | function | Generates a URL or path for a specific route based on the given parameters. |