Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Better error-correction in HarvestUtility #4268

Open
wants to merge 12 commits into
base: 2.x
Choose a base branch
from
46 changes: 32 additions & 14 deletions modules/harvest/src/HarvestUtility.php
Original file line number Diff line number Diff line change
Expand Up @@ -185,16 +185,7 @@ public function convertHashTable(string $plan_id) {
* Outdated tables will be removed.
*/
public function harvestHashUpdate() {
$plan_ids = array_merge(
$this->harvestService->getAllHarvestIds(),
array_values($this->findOrphanedHarvestDataIds())
);
foreach ($plan_ids as $plan_id) {
$this->logger->notice('Converting hashes for ' . $plan_id);
$this->convertHashTable($plan_id);
$this->storeFactory->getInstance('harvest_' . $plan_id . '_hashes')
->destruct();
}
$this->convertTables('hashes');
}

/**
Expand All @@ -219,15 +210,42 @@ public function convertRunTable(string $plan_id) {
* Outdated tables will be removed.
*/
public function harvestRunsUpdate() {
$this->convertTables('runs');
}

/**
* Convert the hashes or runs tables to use entities.
*
* Outdated tables will be removed.
*
* @param string $type
* The type of table to convert. Must be one of 'hashes' or 'runs'.
*/
protected function convertTables(string $type) {
// Types mapped to conversion methods.
$types = [
'hashes' => 'convertHashTable',
'runs' => 'convertRunTable',
];
if (!in_array($type, array_keys($types))) {
throw new \InvalidArgumentException('Type must be one of: ' . implode(', ', array_keys($types)));
}
$plan_ids = array_merge(
$this->harvestService->getAllHarvestIds(),
array_values($this->findOrphanedHarvestDataIds())
);
foreach ($plan_ids as $plan_id) {
$this->logger->notice('Converting runs for ' . $plan_id);
$this->convertRunTable($plan_id);
$this->storeFactory->getInstance('harvest_' . $plan_id . '_runs')
->destruct();
$this->logger->notice('Converting ' . $type . ' for ' . $plan_id);
try {
$this->{$types[$type]}($plan_id);
$this->storeFactory->getInstance('harvest_' . $plan_id . '_' . $type)
->destruct();
}
catch (\Exception $e) {
$this->logger->error('Unable to convert ' . $type . ' for ' . $plan_id);
$this->logger->error($e->getMessage());
$this->logger->debug($e->getTraceAsString());
}
}
}

Expand Down
65 changes: 65 additions & 0 deletions modules/harvest/tests/src/Kernel/HarvestUtilityTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

namespace Drupal\Tests\harvest\Kernel;

use Drupal\harvest\HarvestUtility;
use Drupal\KernelTests\KernelTestBase;

/**
Expand Down Expand Up @@ -197,4 +198,68 @@ public function testHarvestRunsUpdate() {
$this->assertEquals('AWESOME', $run_entity->get('extract_status')->getString());
}

/**
* Ensure an exception during DB handling does not stop further updates.
*
* @covers ::harvestRunsUpdate
*/
public function testHarvestRunException() {
// Mock HarvestUtility so we can have one method throw an exception.
$utility = $this->getMockBuilder(HarvestUtility::class)
->setConstructorArgs([
$this->container->get('dkan.harvest.service'),
$this->container->get('dkan.harvest.storage.database_table'),
$this->container->get('dkan.harvest.storage.hashes_database_table'),
$this->container->get('dkan.harvest.storage.harvest_run_repository'),
$this->container->get('database'),
$this->container->get('dkan.harvest.logger_channel'),
])
->onlyMethods(['findOrphanedHarvestDataIds', 'convertRunTable'])
->getMock();

// findOrphanedHarvestDataIds will return two fake IDs to loop over.
$utility->expects($this->atLeast(1))
->method('findOrphanedHarvestDataIds')
->willReturn(['FAKE_PLAN_ID', 'ANOTHER_FAKE_PLAN_ID']);
// DB work will throw an exception. We expect it to be called twice because
// there are two plan IDs.
$utility->expects($this->atLeast(2))
->method('convertRunTable')
->willThrowException(new \Exception('Hello.'));

$utility->harvestRunsUpdate();
}

/**
* Ensure an exception during DB handling does not stop further updates.
*
* @covers ::harvestHashUpdate
*/
public function testHarvestHashException() {
// Mock HarvestUtility so we can mock some methods.
$utility = $this->getMockBuilder(HarvestUtility::class)
->setConstructorArgs([
$this->container->get('dkan.harvest.service'),
$this->container->get('dkan.harvest.storage.database_table'),
$this->container->get('dkan.harvest.storage.hashes_database_table'),
$this->container->get('dkan.harvest.storage.harvest_run_repository'),
$this->container->get('database'),
$this->container->get('dkan.harvest.logger_channel'),
])
->onlyMethods(['findOrphanedHarvestDataIds', 'convertHashTable'])
->getMock();

// findOrphanedHarvestDataIds will return two fake IDs to loop over.
$utility->expects($this->atLeast(1))
->method('findOrphanedHarvestDataIds')
->willReturn(['FAKE_PLAN_ID', 'ANOTHER_FAKE_PLAN_ID']);
// DB work will throw an exception. We expect it to be called twice because
// there are two plan IDs.
$utility->expects($this->atLeast(2))
->method('convertHashTable')
->willThrowException(new \Exception('Hello.'));

$utility->harvestHashUpdate();
}

}