processInfoList(), so look * there for examples of how to populate the array with real values. * * @see \Drupal\update\UpdateManager::getProjects() * @see \Drupal\Core\Utility\ProjectInfo::processInfoList() */ function hook_update_projects_alter(&$projects) { // Hide a site-specific module from the list. unset($projects['site_specific_module']); // Add an uninstalled module to the list. // The key for the array should be the machine-readable project "short name". $projects['uninstalled_project_name'] = [ // Machine-readable project short name (same as the array key above). 'name' => 'uninstalled_project_name', // Array of values from the main .info.yml file for this project. 'info' => [ 'name' => 'Some uninstalled module', 'description' => 'A module not installed on the site that you want to see in the available updates report.', 'version' => '8.x-1.0', 'core' => '8.x', // The maximum file change time (the "ctime" returned by the filectime() // PHP method) for all of the .info.yml files included in this project. '_info_file_ctime' => 1243888165, ], // The date stamp when the project was released, if known. If the // uninstalled project was an officially packaged release from drupal.org, // this will be included in the .info.yml file as the 'datestamp' field. // This only really matters for development snapshot releases that are // regenerated, so it can be left undefined or set to 0 in most cases. 'datestamp' => 1243888185, // Any modules (or themes) included in this project. Keyed by machine- // readable "short name", value is the human-readable project name printed // in the UI. 'includes' => [ 'uninstalled_project' => 'uninstalled module', 'uninstalled_project_helper' => 'uninstalled module helper module', 'uninstalled_project_foo' => 'uninstalled module foo add-on module', ], // Does this project contain a 'module', 'theme', 'uninstalled-module', or // 'uninstalled-theme'? 'project_type' => 'uninstalled-module', ]; } /** * Alter the information about available updates for projects. * * @param $projects * Reference to an array of information about available updates to each * project installed on the system. * * @see update_calculate_project_data() */ function hook_update_status_alter(&$projects) { $settings = \Drupal::config('update_advanced.settings')->get('projects'); foreach ($projects as $project => $project_info) { if (isset($settings[$project]) && isset($settings[$project]['check']) && ($settings[$project]['check'] == 'never' || (isset($project_info['recommended']) && $settings[$project]['check'] === $project_info['recommended']))) { $projects[$project]['status'] = UpdateFetcherInterface::NOT_CHECKED; $projects[$project]['reason'] = t('Ignored from settings'); if (!empty($settings[$project]['notes'])) { $projects[$project]['extra'][] = [ 'class' => ['admin-note'], 'label' => t('Administrator note'), 'data' => $settings[$project]['notes'], ]; } } } } /** * Verify an archive after it has been downloaded and extracted. * * @param string $project * The short name of the project that has been downloaded. * @param string $archive_file * The filename of the un-extracted archive. * @param string $directory * The directory that the archive was extracted into. * * @return array * If there are any problems, return an array of error messages. If there are * no problems, return an empty array. * * @see update_manager_archive_verify() * @ingroup update_manager_file */ function hook_verify_update_archive($project, $archive_file, $directory) { $errors = []; if (!file_exists($directory)) { $errors[] = t('The %directory does not exist.', ['%directory' => $directory]); } // Add other checks on the archive integrity here. return $errors; } /** * @} End of "addtogroup hooks". */