Skip to content
Snippets Groups Projects
Commit 2bdbb029 authored by Adam G-H's avatar Adam G-H Committed by Chris Wells
Browse files

Issue #3507468: EnabledSourceHandler's query result caching should also...

Issue #3507468: EnabledSourceHandler's query result caching should also consider the contents of composer.lock
parent 579a4072
No related branches found
No related tags found
1 merge request!735Issue #3507468: EnabledSourceHandler's query result caching should also consider the contents of composer.lock
Pipeline #433978 passed
......@@ -2,6 +2,7 @@
namespace Drupal\project_browser;
use Composer\InstalledVersions;
use Drupal\Component\Serialization\Json;
use Drupal\Core\Config\ConfigCrudEvent;
use Drupal\Core\Config\ConfigEvents;
......@@ -109,7 +110,7 @@ class EnabledSourceHandler implements LoggerAwareInterface, EventSubscriberInter
*/
public function getProjects(string $source_id, array $query = []): ProjectsResultsPage {
// Cache only exact query, down to the page number.
$cache_key = 'query:' . md5(Json::encode($query));
$cache_key = $this->getQueryCacheKey($query);
$storage = $this->keyValue($source_id);
......@@ -144,6 +145,27 @@ class EnabledSourceHandler implements LoggerAwareInterface, EventSubscriberInter
return $results;
}
/**
* Generates a cache key for a specific query.
*
* @param array $query
* The query.
*
* @return string
* A cache key for the given query.
*/
private function getQueryCacheKey(array $query): string {
// Include a quick hash of the top-level `composer.lock` file in the hash,
// so that sources which base their queries on the state of the local site
// will be refreshed when the local site changes.
['install_path' => $project_root] = InstalledVersions::getRootPackage();
$lock_file = $project_root . DIRECTORY_SEPARATOR . 'composer.lock';
$lock_file_hash = file_exists($lock_file)
? hash_file('xxh64', $lock_file)
: '';
return 'query:' . md5(Json::encode($query) . $lock_file_hash);
}
/**
* Queries the specified source.
*
......
......@@ -28,13 +28,6 @@ class ClearStorageTest extends BrowserTestBase {
*/
protected $defaultTheme = 'stark';
/**
* The cache key under which project data is stored in key-value.
*
* @var string
*/
private readonly string $cacheKey;
/**
* The key-value storage.
*
......@@ -52,12 +45,11 @@ class ClearStorageTest extends BrowserTestBase {
->set('enabled_sources', ['project_browser_test_mock'])
->save();
$this->cacheKey = 'query:' . md5('[]');
$this->keyValue = \Drupal::service('keyvalue')->get('project_browser:project_browser_test_mock');
// Warm the project cache and confirm it is populated.
\Drupal::service(EnabledSourceHandler::class)->getProjects('project_browser_test_mock');
$this->assertNotEmpty($this->keyValue->get($this->cacheKey));
$this->assertNotEmpty($this->keyValue->getAll());
}
/**
......@@ -67,7 +59,7 @@ class ClearStorageTest extends BrowserTestBase {
*/
public function testClearCacheDirectly(): void {
\Drupal::service(EnabledSourceHandler::class)->clearStorage();
$this->assertEmpty($this->keyValue->get($this->cacheKey));
$this->assertEmpty($this->keyValue->getAll());
}
/**
......@@ -81,7 +73,7 @@ class ClearStorageTest extends BrowserTestBase {
*/
public function testClearCacheWithDrush(string $command): void {
$this->drush($command);
$this->assertEmpty($this->keyValue->get($this->cacheKey));
$this->assertEmpty($this->keyValue->getAll());
}
/**
......@@ -95,7 +87,7 @@ class ClearStorageTest extends BrowserTestBase {
$assert_session = $this->assertSession();
$assert_session->buttonExists('Clear storage')->press();
$assert_session->statusMessageContains('Storage cleared.');
$this->assertEmpty($this->keyValue->get($this->cacheKey));
$this->assertEmpty($this->keyValue->getAll());
}
}
......@@ -65,21 +65,26 @@ class EnabledSourceHandlerTest extends BrowserTestBase {
public function testErrorsAreNotStored(): void {
/** @var \Drupal\project_browser\EnabledSourceHandler $handler */
$handler = $this->container->get(EnabledSourceHandler::class);
$handler->getProjects('project_browser_test_mock');
/** @var \Drupal\Core\KeyValueStore\KeyValueStoreInterface $storage */
$storage = $this->container->get('keyvalue')
->get('project_browser:project_browser_test_mock');
$has_cached_queries = function (): bool {
$items = $this->container->get('keyvalue')
->get('project_browser:project_browser_test_mock')
->getAll();
return (bool) array_filter(
array_keys($items),
fn (string $key): bool => str_starts_with($key, 'query:'),
);
};
// Query results should have been stored.
$query_cache_key = 'query:' . md5('[]');
$this->assertTrue($storage->has($query_cache_key));
$this->assertTrue($has_cached_queries());
$handler->clearStorage();
ProjectBrowserTestMock::$resultsError = 'Nope!';
$handler->getProjects('project_browser_test_mock');
// No query results should have been stored.
$this->assertFalse($storage->has($query_cache_key));
$this->assertFalse($has_cached_queries());
}
/**
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment