diff --git a/modules/common/src/DataResource.php b/modules/common/src/DataResource.php index c5081a30dd..72580ffd22 100644 --- a/modules/common/src/DataResource.php +++ b/modules/common/src/DataResource.php @@ -135,9 +135,9 @@ public function __construct($file_path, $mimeType, $perspective = self::DEFAULT_ * @return \Drupal\common\DataResource * DataResource object. * - * @deprecated Use DataResource::createFromEntity() instead. - * - * @see self::createFromEntity() + * @deprecated in dkan:8.x-2.17 and is removed from dkan:8.x-2.21. Use + * DataResource::createFromEntity() instead. + * @see https://github.com/GetDKAN/dkan/pull/4027 */ public static function createFromRecord(object $record): DataResource { $resource = new static($record->filePath, $record->mimeType, $record->perspective); @@ -236,6 +236,11 @@ public function changeMimeType($newMimeType) { * * @return \Drupal\datastore\DatastoreResource * Datastore Resource. + * + * @deprecated in dkan:8.x-2.20 and is removed from dkan:8.x-2.21. Use storage + * classes like DatabaseTable::getTableName() to determine correct table + * names, and pass true to ::getFilePath to get the resolved URL. + * @see https://github.com/GetDKAN/dkan/pull/4372 */ public function getDatastoreResource(): DatastoreResource { return new DatastoreResource( @@ -253,10 +258,16 @@ public function getIdentifier() { } /** - * Getter. + * Get the resource's filepath. + * + * @param bool|null $resolve + * Whether to resolve the URL host tokens in the file path. + * + * @return string + * The file path. */ - public function getFilePath() { - return $this->filePath; + public function getFilePath(?bool $resolve = FALSE):string { + return $resolve ? UrlHostTokenResolver::resolve($this->getFilePath()) : $this->filePath; } /** @@ -324,6 +335,11 @@ public function getUniqueIdentifierNoPerspective(): string { /** * Retrieve datastore table name for resource. + * + * @deprecated in dkan:8.x-2.20 and is removed from dkan:8.x-2.21. Use storage + * classes like DatabaseTable::getTableName() to determine correct table + * names. + * @see https://github.com/GetDKAN/dkan/pull/4372 */ public function getTableName() { return 'datastore_' . md5($this->getUniqueIdentifier()); @@ -333,7 +349,7 @@ public function getTableName() { * {@inheritDoc} */ #[\ReturnTypeWillChange] - public function jsonSerialize() { + public function jsonSerialize(): mixed { return $this->serialize(); } diff --git a/modules/common/src/Storage/AbstractDatabaseTable.php b/modules/common/src/Storage/AbstractDatabaseTable.php index e477b179e0..f7bd4e5b25 100644 --- a/modules/common/src/Storage/AbstractDatabaseTable.php +++ b/modules/common/src/Storage/AbstractDatabaseTable.php @@ -34,21 +34,13 @@ abstract class AbstractDatabaseTable implements DatabaseTableInterface { */ protected $connection; - /** - * Get the full name of datastore db table. - * - * @return string - * Table name. - */ - abstract protected function getTableName(); - /** * Prepare data. * * Transform the string data given into what should be use by the insert * query. */ - abstract protected function prepareData(string $data, string $id = NULL): array; + abstract protected function prepareData(string $data, ?string $id = NULL): array; /** * Get the primary key used in the table. @@ -113,7 +105,7 @@ public function retrieveAll(): array { /** * Store data. */ - public function store($data, string $id = NULL): string { + public function store($data, ?string $id = NULL): string { $this->setTable(); $existing = (isset($id)) ? $this->retrieve($id) : NULL; @@ -263,7 +255,7 @@ protected function sanitizedErrorMessage(string $unsanitizedMessage) { * @throws \Exception * Throws an exception if the schema was not already set. */ - protected function setTable() { + public function setTable() { if (!$this->tableExist($table_name = $this->getTableName())) { if ($schema = $this->schema) { try { diff --git a/modules/datastore/datastore.services.yml b/modules/datastore/datastore.services.yml index 3aef3143c4..e0889e2db2 100644 --- a/modules/datastore/datastore.services.yml +++ b/modules/datastore/datastore.services.yml @@ -42,6 +42,7 @@ services: - '@dkan.datastore.data_dictionary.alter_table_query_builder.mysql' - '@dkan.metastore.service' - '@dkan.metastore.data_dictionary_discovery' + - '@dkan.datastore.database_table_factory' tags: - { name: resource_processor, priority: 25 } diff --git a/modules/datastore/modules/datastore_mysql_import/src/Service/MysqlImport.php b/modules/datastore/modules/datastore_mysql_import/src/Service/MysqlImport.php index c27a8e3227..e56cf7d9d2 100644 --- a/modules/datastore/modules/datastore_mysql_import/src/Service/MysqlImport.php +++ b/modules/datastore/modules/datastore_mysql_import/src/Service/MysqlImport.php @@ -68,8 +68,12 @@ protected function runIt() { } // Attempt to resolve resource file name from file path. - if (($file_path = \Drupal::service('file_system')->realpath($this->resource->getFilePath())) === FALSE) { - return $this->setResultError(sprintf('Unable to resolve file name "%s" for resource with identifier "%s".', $this->resource->getFilePath(), $this->resource->getId())); + if (($file_path = \Drupal::service('file_system')->realpath($this->resource->getFilePath(TRUE))) === FALSE) { + return $this->setResultError(sprintf( + 'Unable to resolve file name "%s" for resource with identifier "%s".', + $this->resource->getFilePath(TRUE), + $this->resource->getUniqueIdentifier()) + ); } $size = @filesize($file_path); diff --git a/modules/datastore/modules/datastore_mysql_import/tests/src/Kernel/Storage/MySqlDatabaseTableFactoryTest.php b/modules/datastore/modules/datastore_mysql_import/tests/src/Kernel/Storage/MySqlDatabaseTableFactoryTest.php index 92f8470c40..a86ae5baaf 100644 --- a/modules/datastore/modules/datastore_mysql_import/tests/src/Kernel/Storage/MySqlDatabaseTableFactoryTest.php +++ b/modules/datastore/modules/datastore_mysql_import/tests/src/Kernel/Storage/MySqlDatabaseTableFactoryTest.php @@ -2,7 +2,7 @@ namespace Drupal\Tests\datastore_mysql_import\Kernel\Storage; -use Drupal\datastore\DatastoreResource; +use Drupal\common\DataResource; use Drupal\datastore_mysql_import\Storage\MySqlDatabaseTable; use Drupal\KernelTests\KernelTestBase; @@ -11,6 +11,7 @@ * @coversDefaultClass \Drupal\datastore_mysql_import\Storage\MySqlDatabaseTableFactory * * @group datastore_mysql_import + * @group kernel */ class MySqlDatabaseTableFactoryTest extends KernelTestBase { @@ -33,10 +34,8 @@ public function testFactoryServiceResourceException() { } public function testFactoryService() { - $identifier = 'identifier'; $file_path = dirname(__FILE__, 4) . '/data/columnspaces.csv'; - $datastore_resource = new DatastoreResource( - $identifier, + $datastore_resource = new DataResource( $file_path, 'text/csv' ); diff --git a/modules/datastore/src/DatastoreResource.php b/modules/datastore/src/DatastoreResource.php index 171d0b049e..439b114a54 100644 --- a/modules/datastore/src/DatastoreResource.php +++ b/modules/datastore/src/DatastoreResource.php @@ -5,9 +5,9 @@ /** * Basic datastore resource class. * - * Always generate this object using DataResource::getDatastoreResource(). - * - * @see \Drupal\common\DataResource::getDatastoreResource() + * @deprecated in dkan:8.x-2.20 and is removed from dkan:8.x-2.21. Use + * \Drupal\common\DataResource instead. + * @see https://github.com/GetDKAN/dkan/pull/4372 */ class DatastoreResource implements \JsonSerializable { @@ -43,6 +43,10 @@ public function __construct($id, $file_path, $mime_type) { /** * Get the resource ID. + * + * @deprecated in dkan:8.x-2.20 and is removed from dkan:8.x-2.21. Use + * \Drupal\common\DataResource::getUniqueIdentifier() instead. + * @see https://github.com/GetDKAN/dkan/pull/4372 */ public function getId(): string { return $this->id; @@ -50,6 +54,13 @@ public function getId(): string { /** * Get the file path. + * + * @return string + * The file path. + * + * @deprecated in dkan:8.x-2.20 and is removed from dkan:8.x-2.21. Use + * \Drupal\common\DataResource::getUniqueIdentifier() instead. + * @see https://github.com/GetDKAN/dkan/pull/4372 */ public function getFilePath(): string { return $this->filePath; @@ -57,6 +68,13 @@ public function getFilePath(): string { /** * Get the mimeType. + * + * @return string + * The mimeType. + * + * @deprecated in dkan:8.x-2.20 and is removed from dkan:8.x-2.21. Use + * \Drupal\common\DataResource::getMimeType() instead. + * @see https://github.com/GetDKAN/dkan/pull/4372 */ public function getMimeType(): string { return $this->mimeType; @@ -65,8 +83,7 @@ public function getMimeType(): string { /** * {@inheritdoc} */ - #[\ReturnTypeWillChange] - public function jsonSerialize() { + public function jsonSerialize(): mixed { return (object) [ 'filePath' => $this->getFilePath(), 'id' => $this->getId(), diff --git a/modules/datastore/src/Plugin/QueueWorker/ImportJob.php b/modules/datastore/src/Plugin/QueueWorker/ImportJob.php index e0b20deda0..2cbf05d561 100644 --- a/modules/datastore/src/Plugin/QueueWorker/ImportJob.php +++ b/modules/datastore/src/Plugin/QueueWorker/ImportJob.php @@ -89,7 +89,7 @@ class ImportJob extends AbstractPersistentJob { /** * Datastore resource. * - * @var \Drupal\datastore\DatastoreResource + * @var \Drupal\common\DataResource */ protected $resource; @@ -105,7 +105,7 @@ class ImportJob extends AbstractPersistentJob { * @param array|null $config * Configuration options. */ - protected function __construct(string $identifier, $storage, array $config = NULL) { + protected function __construct(string $identifier, $storage, ?array $config = NULL) { parent::__construct($identifier, $storage, $config); $this->dataStorage = $config['storage']; @@ -195,7 +195,7 @@ public function getStorage() { * {@inheritdoc} */ protected function runIt() { - $filename = $this->resource->getFilePath(); + $filename = $this->resource->getFilePath(TRUE); $size = @filesize($filename); if (!$size) { return $this->setResultError("Can't get size from file {$filename}"); diff --git a/modules/datastore/src/Service/ImportService.php b/modules/datastore/src/Service/ImportService.php index 87381d53a7..878e7dc1dd 100644 --- a/modules/datastore/src/Service/ImportService.php +++ b/modules/datastore/src/Service/ImportService.php @@ -151,10 +151,9 @@ public function import() { $data_resource = $this->getResource(); if ($result->getStatus() === Result::ERROR) { - $datastore_resource = $data_resource->getDatastoreResource(); $this->logger->error('Error importing resource id:%id path:%path message:%message', [ - '%id' => $datastore_resource->getId(), - '%path' => $datastore_resource->getFilePath(), + '%id' => $data_resource->getUniqueIdentifier(), + '%path' => $data_resource->getFilePath(TRUE), '%message' => $result->getError(), ]); } @@ -188,20 +187,20 @@ public function getImporter(): ImportJob { if ($this->importJob ?? FALSE) { return $this->importJob; } - $datastore_resource = $this->getResource()->getDatastoreResource(); + $data_resource = $this->getResource(); $delimiter = ","; - if ($datastore_resource->getMimeType() == 'text/tab-separated-values') { + if ($data_resource->getMimeType() == 'text/tab-separated-values') { $delimiter = "\t"; } $this->importJob = call_user_func([$this->importerClass, 'get'], - $datastore_resource->getId(), + $data_resource->getUniqueIdentifier(), $this->importJobStoreFactory->getInstance(), [ "storage" => $this->getStorage(), "parser" => $this->getNonRecordingParser($delimiter), - "resource" => $datastore_resource, + "resource" => $data_resource, ] ); @@ -245,8 +244,8 @@ private function getNonRecordingParser(string $delimiter) : Csv { * DatabaseTable storage object. */ public function getStorage(): DatabaseTable { - $datastore_resource = $this->getResource()->getDatastoreResource(); - return $this->databaseTableFactory->getInstance($datastore_resource->getId(), ['resource' => $datastore_resource]); + $data_resource = $this->getResource(); + return $this->databaseTableFactory->getInstance($data_resource->getUniqueIdentifier(), ['resource' => $data_resource]); } /** diff --git a/modules/datastore/src/Service/ResourceProcessor/DictionaryEnforcer.php b/modules/datastore/src/Service/ResourceProcessor/DictionaryEnforcer.php index d3515db435..494b56a569 100644 --- a/modules/datastore/src/Service/ResourceProcessor/DictionaryEnforcer.php +++ b/modules/datastore/src/Service/ResourceProcessor/DictionaryEnforcer.php @@ -5,6 +5,7 @@ use Drupal\common\DataResource; use Drupal\datastore\DataDictionary\AlterTableQueryBuilderInterface; use Drupal\datastore\Service\ResourceProcessorInterface; +use Drupal\datastore\Storage\DatabaseTableFactory; use Drupal\metastore\MetastoreService; use Drupal\metastore\DataDictionary\DataDictionaryDiscoveryInterface; @@ -43,6 +44,13 @@ class DictionaryEnforcer implements ResourceProcessorInterface { */ protected $resourceMapper; + /** + * Database table factory service. + * + * @var \Drupal\datastore\Storage\DatabaseTableFactory + */ + protected $databaseTableFactory; + /** * Constructs a \Drupal\Component\Plugin\PluginBase object. * @@ -52,15 +60,19 @@ class DictionaryEnforcer implements ResourceProcessorInterface { * The metastore service. * @param \Drupal\metastore\DataDictionary\DataDictionaryDiscoveryInterface $data_dictionary_discovery * The data-dictionary discovery service. + * @param \Drupal\datastore\Storage\DatabaseTableFactory $table_factory + * The datastore database table factory service. */ public function __construct( AlterTableQueryBuilderInterface $alter_table_query_builder, MetastoreService $metastore, - DataDictionaryDiscoveryInterface $data_dictionary_discovery + DataDictionaryDiscoveryInterface $data_dictionary_discovery, + DatabaseTableFactory $table_factory, ) { $this->metastore = $metastore; $this->dataDictionaryDiscovery = $data_dictionary_discovery; $this->alterTableQueryBuilder = $alter_table_query_builder; + $this->databaseTableFactory = $table_factory; } /** @@ -78,7 +90,8 @@ public function process(DataResource $resource): void { // Get data-dictionary for the given resource. $dictionary = $this->getDataDictionaryForResource($resource); // Retrieve name of datastore table for resource. - $datastore_table = $resource->getTableName(); + $table = $this->databaseTableFactory->getInstance('', ['resource' => $resource]); + $datastore_table = $table->getTableName(); $this->applyDictionary($dictionary, $datastore_table); } @@ -125,13 +138,13 @@ public function applyDictionary(RootedJsonData $dictionary, string $datastore_ta /** * Returning data dictionary fields from schema. * - * @param string $identifier + * @param string|null $identifier * A resource's identifier. Used when in reference mode. * * @return array|null * An array of dictionary fields or null if no dictionary is in use. */ - public function returnDataDictionaryFields(string $identifier = NULL): ?array { + public function returnDataDictionaryFields(?string $identifier = NULL): ?array { // Get data dictionary mode. $dd_mode = $this->dataDictionaryDiscovery->getDataDictionaryMode(); // Get data dictionary info. diff --git a/modules/datastore/src/Storage/DatabaseTable.php b/modules/datastore/src/Storage/DatabaseTable.php index ca7ab8fc8f..6e7a93c639 100755 --- a/modules/datastore/src/Storage/DatabaseTable.php +++ b/modules/datastore/src/Storage/DatabaseTable.php @@ -3,8 +3,8 @@ namespace Drupal\datastore\Storage; use Drupal\Core\Database\Connection; +use Drupal\common\DataResource; use Drupal\common\Storage\AbstractDatabaseTable; -use Drupal\datastore\DatastoreResource; use Psr\Log\LoggerInterface; /** @@ -17,29 +17,29 @@ class DatabaseTable extends AbstractDatabaseTable implements \JsonSerializable { /** * Datastore resource object. * - * @var \Drupal\datastore\DatastoreResource + * @var \Drupal\common\DataResource */ - private $resource; + protected $resource; /** * DKAN logger channel service. */ - private LoggerInterface $logger; + protected LoggerInterface $logger; /** * Constructor method. * * @param \Drupal\Core\Database\Connection $connection * Drupal database connection object. - * @param \Drupal\datastore\DatastoreResource $resource + * @param \Drupal\common\DataResource $resource * A resource. * @param \Psr\Log\LoggerInterface $loggerChannel * DKAN logger channel service. */ public function __construct( Connection $connection, - DatastoreResource $resource, - LoggerInterface $loggerChannel + DataResource $resource, + LoggerInterface $loggerChannel, ) { // Set resource before calling the parent constructor. The parent calls // getTableName which we implement and needs the resource to operate. @@ -71,12 +71,10 @@ public function getSummary() { } /** - * Inherited. - * * {@inheritdoc} */ #[\ReturnTypeWillChange] - public function jsonSerialize() { + public function jsonSerialize(): mixed { return (object) ['resource' => $this->resource]; } @@ -88,7 +86,7 @@ public function jsonSerialize() { */ public function getTableName() { if ($this->resource) { - return 'datastore_' . $this->resource->getId(); + return 'datastore_' . md5($this->resource->getUniqueIdentifier()); } return 'datastore_does_not_exist'; } @@ -96,7 +94,7 @@ public function getTableName() { /** * Protected. */ - protected function prepareData(string $data, string $id = NULL): array { + protected function prepareData(string $data, ?string $id = NULL): array { $decoded = json_decode($data); if ($decoded === NULL) { $this->logger->error('Error decoding id:@id, data: @data.', [ diff --git a/modules/datastore/src/Storage/DatabaseTableFactory.php b/modules/datastore/src/Storage/DatabaseTableFactory.php index a98f88e152..c7bc0098ee 100644 --- a/modules/datastore/src/Storage/DatabaseTableFactory.php +++ b/modules/datastore/src/Storage/DatabaseTableFactory.php @@ -3,6 +3,7 @@ namespace Drupal\datastore\Storage; use Contracts\FactoryInterface; +use Drupal\common\DataResource; use Drupal\Core\Database\Connection; use Psr\Log\LoggerInterface; @@ -28,16 +29,22 @@ class DatabaseTableFactory implements FactoryInterface { */ public function __construct( Connection $connection, - LoggerInterface $loggerChannel + LoggerInterface $loggerChannel, ) { $this->connection = $connection; $this->logger = $loggerChannel; } /** - * Inherited. + * Get a DatabaseTable instance. * - * @inheritdoc + * @param string $identifier + * Some way to discern between different instances of a class. + * @param array $config + * Must contain a 'resource' key, which is a DataResource object. + * + * @return \Drupal\datastore\Storage\DatabaseTable + * A DatabaseTable object. */ public function getInstance(string $identifier, array $config = []) { if (!isset($config['resource'])) { @@ -50,9 +57,15 @@ public function getInstance(string $identifier, array $config = []) { } /** - * Protected. + * Get a DatabaseTable object from a DataResource object. + * + * @param \Drupal\common\DataResource $resource + * A resource. + * + * @return \Drupal\datastore\Storage\DatabaseTable + * A DatabaseTable object. */ - protected function getDatabaseTable($resource) { + protected function getDatabaseTable(DataResource $resource) { return new DatabaseTable($this->connection, $resource, $this->logger); } diff --git a/modules/datastore/src/Storage/QueryFactory.php b/modules/datastore/src/Storage/QueryFactory.php index 2ef54e5368..e113778072 100644 --- a/modules/datastore/src/Storage/QueryFactory.php +++ b/modules/datastore/src/Storage/QueryFactory.php @@ -40,7 +40,7 @@ public function __construct(DatastoreQuery $datastoreQuery, array $storageMap) { /** * Static factory create method. * - * @param Drupal\datastore\Service\DatastoreQuery $datastoreQuery + * @param \Drupal\datastore\Service\DatastoreQuery $datastoreQuery * Datastore query request object. * @param array $storageMap * Storage map array. diff --git a/modules/datastore/tests/data/states_with_dupes_link.csv b/modules/datastore/tests/data/states_with_dupes_link.csv new file mode 120000 index 0000000000..3a2a7767d4 --- /dev/null +++ b/modules/datastore/tests/data/states_with_dupes_link.csv @@ -0,0 +1 @@ +states_with_dupes.csv \ No newline at end of file diff --git a/modules/datastore/tests/src/Kernel/Service/ResourceProcessor/DictionaryEnforcerTest.php b/modules/datastore/tests/src/Kernel/Service/ResourceProcessor/DictionaryEnforcerTest.php index bfebeb3878..cf5918dce5 100644 --- a/modules/datastore/tests/src/Kernel/Service/ResourceProcessor/DictionaryEnforcerTest.php +++ b/modules/datastore/tests/src/Kernel/Service/ResourceProcessor/DictionaryEnforcerTest.php @@ -66,6 +66,7 @@ public function testProcessModeNone() { $this->container->get('dkan.datastore.data_dictionary.alter_table_query_builder.mysql'), $this->container->get('dkan.metastore.service'), $this->container->get('dkan.metastore.data_dictionary_discovery'), + $this->container->get('dkan.datastore.database_table_factory'), ]) ->onlyMethods(['getDataDictionaryForResource', 'applyDictionary']) ->getMock(); diff --git a/modules/datastore/tests/src/Kernel/Storage/DatabaseTableTest.php b/modules/datastore/tests/src/Kernel/Storage/DatabaseTableTest.php index 2816b4e793..71994466a1 100644 --- a/modules/datastore/tests/src/Kernel/Storage/DatabaseTableTest.php +++ b/modules/datastore/tests/src/Kernel/Storage/DatabaseTableTest.php @@ -5,10 +5,10 @@ use ColinODell\PsrTestLogger\TestLogger; use Drupal\common\DataResource; use Drupal\common\Storage\ImportedItemInterface; -use Drupal\datastore\DatastoreResource; use Drupal\datastore\Plugin\QueueWorker\ImportJob; use Drupal\datastore\Service\Factory\ImportServiceFactory; use Drupal\datastore\Storage\DatabaseTable; +use Drupal\datastore\Storage\DatabaseTableFactory; use Drupal\KernelTests\KernelTestBase; use Drupal\Tests\common\Unit\Connection; use Procrastinator\Result; @@ -95,8 +95,9 @@ public function testPrepareDataLogging($expected_log, $expected_exception, $data ->onlyMethods(['tableExist']) ->setConstructorArgs([ $this->createStub(Connection::class), - $this->createStub(DatastoreResource::class), + $this->createStub(DataResource::class), $logger, + $this->createStub(DatabaseTableFactory::class), ]) ->getMock(); $database_table->expects($this->any()) diff --git a/modules/datastore/tests/src/Unit/Controller/QueryControllerTest.php b/modules/datastore/tests/src/Unit/Controller/QueryControllerTest.php index 541ed5b1cd..4f3c9a274e 100644 --- a/modules/datastore/tests/src/Unit/Controller/QueryControllerTest.php +++ b/modules/datastore/tests/src/Unit/Controller/QueryControllerTest.php @@ -2,12 +2,13 @@ namespace Drupal\Tests\datastore\Unit\Controller; +use Drupal\common\DataResource; use Drupal\Core\Cache\Context\CacheContextsManager; use Drupal\Core\Config\ConfigFactoryInterface; use Drupal\Core\Config\ImmutableConfig; use Drupal\common\DatasetInfo; +use Drupal\Component\EventDispatcher\ContainerAwareEventDispatcher; use Drupal\datastore\Controller\QueryController; -use Drupal\datastore\DatastoreResource; use Drupal\datastore\DatastoreService; use Drupal\datastore\Service\Query; use Drupal\datastore\Storage\SqliteDatabaseTable; @@ -33,23 +34,33 @@ */ class QueryControllerTest extends TestCase { + const FILE_DIR = __DIR__ . "/../../../data/"; + + /** + * Resources to be used in tests. + */ + private DataResource $resource; + protected function setUp(): void { parent::setUp(); // Set cache services. $options = (new Options) ->add('cache_contexts_manager', CacheContextsManager::class) + ->add('event_dispatcher', ContainerAwareEventDispatcher::class) ->index(0); $chain = (new Chain($this)) ->add(ContainerInterface::class, 'get', $options) ->add(CacheContextsManager::class, 'assertValidTokens', TRUE); \Drupal::setContainer($chain->getMock()); + + $this->resource = new DataResource(self::FILE_DIR . 'states_with_dupes.csv', 'text/csv'); } public function testQueryJson() { $data = json_encode([ "resources" => [ [ - "id" => "2", + "id" => $this->resource->getIdentifier(), "alias" => "t", ], ], @@ -98,7 +109,7 @@ public function testQueryRowIdProperty() { // Try simple string properties: $data = json_encode(["properties" => ["record_number", "state"]]); - $result = $this->getQueryResult($data, "2"); + $result = $this->getQueryResult($data, $this->resource->getIdentifier()); $this->assertEquals(400, $result->getStatusCode()); $this->assertStringContainsString('The record_number property is for internal use', $result->getContent()); @@ -106,7 +117,7 @@ public function testQueryRowIdProperty() { // Now try with rowIds plus an arbitrary property: $data = json_encode(["properties" => ["state"], "rowIds" => TRUE]); - $result = $this->getQueryResult($data, "2"); + $result = $this->getQueryResult($data, $this->resource->getIdentifier()); $this->assertEquals(400, $result->getStatusCode()); $this->assertStringContainsString('The rowIds property cannot be set to true', $result->getContent()); @@ -125,7 +136,7 @@ public function testQueryRowIdProperty() { ], ]); - $result = $this->getQueryResult($data, "2"); + $result = $this->getQueryResult($data, $this->resource->getIdentifier()); $this->assertEquals(400, $result->getStatusCode()); $this->assertStringContainsString('The record_number property is for internal use', $result->getContent()); @@ -147,7 +158,7 @@ public function testQueryRowIdSort() { ], ]); - $result = $this->getQueryResult($data, "2"); + $result = $this->getQueryResult($data, $this->resource->getIdentifier()); $this->assertTrue($result instanceof JsonResponse); $this->assertEquals(200, $result->getStatusCode()); @@ -195,7 +206,7 @@ public function testResourceQueryInvalidQuery() { $data = json_encode([ "conditions" => "nope", ]); - $result = $this->getQueryResult($data, "2"); + $result = $this->getQueryResult($data, $this->resource->getIdentifier()); $this->assertTrue($result instanceof JsonResponse); $this->assertEquals(400, $result->getStatusCode()); @@ -208,7 +219,7 @@ public function testResourceQueryWithJoin() { "condition" => "t.field1 = s.field1", ], ]); - $result = $this->getQueryResult($data, "2"); + $result = $this->getQueryResult($data, $this->resource->getIdentifier()); $this->assertTrue($result instanceof JsonResponse); $this->assertEquals(400, $result->getStatusCode()); @@ -221,7 +232,7 @@ public function testResourceQueryJson() { $data = json_encode([ "results" => TRUE, ]); - $result = $this->getQueryResult($data, "2"); + $result = $this->getQueryResult($data, $this->resource->getIdentifier()); $this->assertTrue($result instanceof JsonResponse); $this->assertEquals(200, $result->getStatusCode()); @@ -274,7 +285,7 @@ public function testResourceQueryJoins() { ], ], ]); - $result = $this->getQueryResult($data, "2"); + $result = $this->getQueryResult($data, $this->resource->getIdentifier()); $this->assertTrue($result instanceof JsonResponse); $this->assertEquals(400, $result->getStatusCode()); @@ -288,7 +299,7 @@ public function testQueryCsv() { $data = json_encode([ "resources" => [ [ - "id" => "2", + "id" => $this->resource->getIdentifier(), "alias" => "t", ], ], @@ -330,7 +341,7 @@ public function testResourceQueryCsv() { "results" => TRUE, "format" => "csv", ]); - $result = $this->getQueryResult($data, "2"); + $result = $this->getQueryResult($data, $this->resource->getIdentifier()); $this->assertTrue($result instanceof CsvResponse); $csv = explode("\n", $result->getContent()); $this->assertEquals('State', $csv[0]); @@ -358,7 +369,7 @@ public function testResourceExpressionQueryCsv() { "results" => TRUE, "format" => "csv", ]); - $result = $this->getQueryResult($data, "2"); + $result = $this->getQueryResult($data, $this->resource->getIdentifier()); $this->assertTrue($result instanceof CsvResponse); $csv = explode("\n", $result->getContent()); @@ -428,7 +439,7 @@ public function testQueryCsvCacheHeaders() { $data = json_encode([ "resources" => [ [ - "id" => "2", + "id" => $this->resource->getIdentifier(), "alias" => "t", ], ], @@ -524,20 +535,10 @@ public function mockRequest($data = '') { */ public function mockDatastoreTable() { $connection = new SqliteConnection(new \PDO('sqlite::memory:'), []); - $connection->query('CREATE TABLE `datastore_2` (`record_number` INTEGER NOT NULL, state TEXT, year INT);'); - - $sampleData = []; - $fp = fopen(__DIR__ . '/../../../data/states_with_dupes.csv', 'rb'); - while (!feof($fp)) { - $sampleData[] = fgetcsv($fp); - } - foreach ($sampleData as $row) { - $connection->query("INSERT INTO `datastore_2` VALUES ($row[0], '$row[1]', $row[2]);"); - } - + // $connection->query('CREATE TABLE `datastore_2` (`record_number` INTEGER NOT NULL, state TEXT, year INT);'); $storage = new SqliteDatabaseTable( $connection, - new DatastoreResource("2", "data.csv", "text/csv"), + $this->resource, $this->createStub(LoggerInterface::class) ); $storage->setSchema([ @@ -547,6 +548,20 @@ public function mockDatastoreTable() { 'year' => ['type' => 'int', 'description' => 'Year'], ], ]); + $storage->setTable(); + + $fp = fopen($this->resource->getFilePath(), 'rb'); + $sampleData = []; + while (!feof($fp)) { + $sampleData[] = fgetcsv($fp); + } + fclose($fp); + + $table_name = $storage->getTableName(); + foreach ($sampleData as $row) { + $connection->query("INSERT INTO `$table_name` VALUES ($row[0], '$row[1]', $row[2]);"); + } + return $storage; } diff --git a/modules/datastore/tests/src/Unit/Controller/QueryDownloadControllerTest.php b/modules/datastore/tests/src/Unit/Controller/QueryDownloadControllerTest.php index 94bf34fd01..fae930e3d5 100644 --- a/modules/datastore/tests/src/Unit/Controller/QueryDownloadControllerTest.php +++ b/modules/datastore/tests/src/Unit/Controller/QueryDownloadControllerTest.php @@ -2,6 +2,7 @@ namespace Drupal\Tests\datastore\Unit\Controller; +use Drupal\common\DataResource; use Drupal\Component\EventDispatcher\ContainerAwareEventDispatcher; use Drupal\Core\Cache\Context\CacheContextsManager; use Drupal\Core\Config\ConfigFactoryInterface; @@ -9,7 +10,6 @@ use Drupal\common\DatasetInfo; use Drupal\datastore\Controller\QueryController; use Drupal\datastore\Controller\QueryDownloadController; -use Drupal\datastore\DatastoreResource; use Drupal\datastore\DatastoreService; use Drupal\datastore\Service\Query; use Drupal\datastore\Storage\SqliteDatabaseTable; @@ -33,7 +33,21 @@ */ class QueryDownloadControllerTest extends TestCase { - private $buffer; + const FILE_DIR = __DIR__ . "/../../../data/"; + + /** + * Output buffer. + * + * @var string + */ + private string $buffer; + + /** + * Resources to be used in tests. + * + * @var \Drupal\common\DataResource[] + */ + private array $resources; protected function setUp(): void { parent::setUp(); @@ -46,11 +60,19 @@ protected function setUp(): void { ->add(ContainerInterface::class, 'get', $options) ->add(CacheContextsManager::class, 'assertValidTokens', TRUE); \Drupal::setContainer($chain->getMock()); + + $this->resources = [ + '2' => new DataResource(self::FILE_DIR . 'states_with_dupes.csv', 'text/csv'), + '3' => new DataResource(self::FILE_DIR . 'years_colors.csv', 'text/csv'), + '4' => new DataResource(self::FILE_DIR . 'states_with_dupes_link.csv', 'text/csv'), + ]; + + $this->buffer = ''; } protected function tearDown(): void { parent::tearDown(); - $this->buffer = NULL; + $this->buffer = ''; } /** @@ -60,13 +82,13 @@ private function queryResultCompare($data, $resource = NULL) { $request = $this->mockRequest($data); $qController = QueryController::create($this->getQueryContainer(500)); $response = $resource ? $qController->queryResource($resource, $request) : $qController->query($request); - $csv = $response->getContent(); + $csv = $response->getContent() ?? ''; $dController = QueryDownloadController::create($this->getQueryContainer(25)); ob_start([self::class, 'getBuffer']); $streamResponse = $resource ? $dController->queryResource($resource, $request) : $dController->query($request); $streamResponse->sendContent(); - $streamedCsv = $this->buffer; + $streamedCsv = $this->buffer ?? ''; ob_get_clean(); $this->assertEquals(count(explode("\n", $csv)), count(explode("\n", $streamedCsv))); @@ -80,7 +102,7 @@ public function testStreamedQueryCsv() { $data = [ "resources" => [ [ - "id" => "2", + "id" => $this->resources[2]->getIdentifier(), "alias" => "t", ], ], @@ -108,7 +130,7 @@ public function testStreamedOtherSortCsv() { $data = [ "resources" => [ [ - "id" => "2", + "id" => $this->resources[2]->getIdentifier(), "alias" => "t", ], ], @@ -138,11 +160,11 @@ public function testStreamedJoinCsv() { "schema" => TRUE, "resources" => [ [ - "id" => "2", + "id" => $this->resources[2]->getIdentifier(), "alias" => "t", ], [ - "id" => "3", + "id" => $this->resources[3]->getIdentifier(), "alias" => "j", ], ], @@ -200,7 +222,7 @@ public function testStreamedQueryJson() { $data = json_encode([ "resources" => [ [ - "id" => "2", + "id" => $this->resources[2]->getIdentifier(), "alias" => "t", ], ], @@ -223,7 +245,7 @@ public function testStreamedLimit() { $data = json_encode([ "resources" => [ [ - "id" => "2", + "id" => $this->resources[2]->getIdentifier(), "alias" => "t", ], ], @@ -258,7 +280,7 @@ public function testStreamedCsvSpecificColumns() { $data = [ "resources" => [ [ - "id" => "2", + "id" => $this->resources[2]->getIdentifier(), "alias" => "t", ], ], @@ -275,7 +297,7 @@ public function testStreamedCsvResourceColumns() { $data = [ "resources" => [ [ - "id" => "2", + "id" => $this->resources[2]->getIdentifier(), "alias" => "t", ], ], @@ -303,7 +325,7 @@ public function testStreamedCsvRowIds() { $data = [ "resources" => [ [ - "id" => "2", + "id" => $this->resources[2]->getIdentifier(), "alias" => "t", ], ], @@ -321,7 +343,7 @@ public function testStreamedBadSchema() { $data = [ "resources" => [ [ - "id" => "2", + "id" => $this->resources[2]->getIdentifier(), "alias" => "tx", ], ], @@ -342,14 +364,15 @@ public function testStreamedBadSchema() { * Create a mock object for the main container passed to the controller. * * @param int $rowLimit - * The row limit for a query. + * The row limit for a query. * @param int|null $responseStreamMaxAge - * The max age for the response stream in cache, or NULL to use the default. + * The max age for the response stream in cache, or NULL to use the default. * * @return \PHPUnit\Framework\MockObject\MockObject * MockChain mock object. */ private function getQueryContainer(int $rowLimit, ?int $responseStreamMaxAge = NULL) { + $connection = new SqliteConnection(new \PDO('sqlite::memory:'), []); $options = (new Options()) ->add("dkan.metastore.storage", DataFactory::class) ->add("dkan.datastore.service", DatastoreService::class) @@ -360,8 +383,6 @@ private function getQueryContainer(int $rowLimit, ?int $responseStreamMaxAge = N ->add('dkan.metastore.api_response', MetastoreApiResponse::class) ->index(0); - $connection = new SqliteConnection(new \PDO('sqlite::memory:'), []); - $schema2 = [ 'record_number' => [ 'type' => 'int', @@ -393,14 +414,14 @@ private function getQueryContainer(int $rowLimit, ?int $responseStreamMaxAge = N ], ]; - $storage2 = $this->mockDatastoreTable($connection, "2", 'states_with_dupes.csv', $schema2); - $storage2x = clone($storage2); + $storage2 = $this->mockDatastoreTable($this->resources[2], $schema2, $connection); + $storage2x = $this->mockDatastoreTable($this->resources[4], $schema2, $connection); $storage2x->setSchema(['fields' => []]); + $storage3 = $this->mockDatastoreTable($this->resources[3], $schema3, $connection); $storageMap = [ 't' => $storage2, 'tx' => $storage2x, - 'j' => $this->mockDatastoreTable($connection, "3", 'years_colors.csv', $schema3 - ), + 'j' => $storage3, ]; $chain = (new Chain($this)) @@ -414,7 +435,7 @@ private function getQueryContainer(int $rowLimit, ?int $responseStreamMaxAge = N ->add(Data::class, 'getCacheMaxAge', 0) ->add(ConfigFactoryInterface::class, 'get', ImmutableConfig::class) ->add(Query::class, "getQueryStorageMap", $storageMap) - ->add(Query::class, 'getDatastoreService', DatastoreService::class) + ->add(Query::class, 'getDatastoreService', DatastoreService::class) ->add(DatastoreService::class, 'getDataDictionaryFields', NULL) // @todo Use an Options or Sequence return here; this will only work for one arg at a time. ->add(ImmutableConfig::class, 'get', $rowLimit) @@ -450,46 +471,49 @@ public function mockRequest($data = '') { * @return \Drupal\common\Storage\DatabaseTableInterface * A database table storage class useable for datastore queries. */ - public function mockDatastoreTable($connection, $id, $csvFile, $fields) { - foreach ($fields as $name => $field) { + public function mockDatastoreTable(DataResource $resource, $fields, $connection) { + + $storage = new SqliteDatabaseTable( + $connection, + $resource, + $this->createStub(LoggerInterface::class) + ); + $storage->setSchema([ + 'fields' => $fields, + ]); + $storage->setTable(); + + foreach ($fields as $field) { $types[] = $field['type']; - $notNull = $field['not null'] ?? FALSE; - $createFields[] = "`$name` " . strtoupper($field['type']) . (($notNull) ? ' NOT NULL' : ''); } - $createFieldsStr = implode(", ", $createFields); - $connection->query("CREATE TABLE `datastore_$id` ($createFieldsStr);"); - + + $fp = fopen($resource->getFilePath(), 'rb'); $sampleData = []; - $fp = fopen(__DIR__ . "/../../../data/$csvFile", 'rb'); while (!feof($fp)) { $sampleData[] = fgetcsv($fp); } + fclose($fp); + + $table_name = $storage->getTableName(); foreach ($sampleData as $row) { $values = []; foreach ($row as $key => $value) { $values[] = $types[$key] == "int" ? $value : "'$value'"; $valuesStr = implode(", ", $values); } - $connection->query("INSERT INTO `datastore_$id` VALUES ($valuesStr);"); + $connection->query("INSERT INTO `$table_name` VALUES ($valuesStr);"); } - $storage = new SqliteDatabaseTable( - $connection, - new DatastoreResource($id, "data-$id.csv", "text/csv"), - $this->createStub(LoggerInterface::class) - ); - $storage->setSchema([ - 'fields' => $fields, - ]); return $storage; } /** * Callback to get output buffer. * - * @param $buffer + * @param string $buffer + * A buffer to be appended to existing buffer in memory. */ - protected function getBuffer($buffer) { + protected function getBuffer(string $buffer) { $this->buffer .= $buffer; } diff --git a/modules/datastore/tests/src/Unit/Plugin/QueueWorker/ImportJobTest.php b/modules/datastore/tests/src/Unit/Plugin/QueueWorker/ImportJobTest.php index 53c0c5bdd2..7dbd0f424a 100644 --- a/modules/datastore/tests/src/Unit/Plugin/QueueWorker/ImportJobTest.php +++ b/modules/datastore/tests/src/Unit/Plugin/QueueWorker/ImportJobTest.php @@ -5,11 +5,18 @@ use Contracts\Mock\Storage\Memory; use CsvParser\Parser\Csv; use CsvParser\Parser\ParserInterface; +use Drupal\common\DataResource; use Drupal\common\Storage\DatabaseTableInterface; -use Drupal\datastore\DatastoreResource; +use Drupal\Component\DependencyInjection\Container; +use Drupal\Core\StreamWrapper\StreamWrapperInterface; +use Drupal\Core\StreamWrapper\StreamWrapperManager; use Drupal\datastore\Plugin\QueueWorker\ImportJob; +use MockChain\Chain; +use MockChain\Options; use PHPUnit\Framework\TestCase; use Procrastinator\Result; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpFoundation\RequestStack; /** * Unit tests for Importer class. @@ -38,6 +45,17 @@ protected function setUp(): void { parent::setUp(); $this->database = new TestMemStorage(); $this->assertTrue($this->database instanceof DatabaseTableInterface); + + $options = (new Options()) + ->add('stream_wrapper_manager', StreamWrapperManager::class) + ->add('request_stack', RequestStack::class) + ->index(0); + $container = (new Chain($this)) + ->add(Container::class, 'get', $options) + ->add(StreamWrapperManager::class, 'getViaUri', StreamWrapperInterface::class) + ->add(RequestStack::class, 'getCurrentRequest', Request::class) + ->add(Request::class, 'getHost', 'web'); + \Drupal::setContainer($container->getMock()); } protected function tearDown(): void { @@ -46,24 +64,29 @@ protected function tearDown(): void { } /** + * Get an ImportJob object. + * + * @param \Drupal\common\DataResource $resource + * DataResource object. * + * @return \Drupal\datastore\Plugin\QueueWorker\ImportJob + * ImportJob object. */ - private function getImportJob(DatastoreResource $resource): ImportJob { + private function getImportJob(DataResource $resource): ImportJob { $storage = new Memory(); $config = [ 'resource' => $resource, 'storage' => $this->database, 'parser' => Csv::getParser(), ]; - return ImportJob::get('1', $storage, $config); + return ImportJob::get($resource->getUniqueIdentifier(), $storage, $config); } /** * */ public function testBasics() { - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/countries.csv', 'text/csv'); - $this->assertEquals(1, $resource->getID()); + $resource = new DataResource(__DIR__ . '/../../../../data/countries.csv', 'text/csv'); $import_job = $this->getImportJob($resource); @@ -95,7 +118,7 @@ public function testBasics() { * */ public function testFileNotFound() { - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/non-existent.csv', 'text/csv'); + $resource = new DataResource(__DIR__ . '/../../../../data/non-existent.csv', 'text/csv'); $datastore = $this->getImportJob($resource); $datastore->run(); @@ -106,7 +129,7 @@ public function testFileNotFound() { * */ public function testNonTextFile() { - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/non-text.csv', 'text/csv'); + $resource = new DataResource(__DIR__ . '/../../../../data/non-text.csv', 'text/csv'); $datastore = $this->getImportJob($resource); $datastore->run(); @@ -117,7 +140,7 @@ public function testNonTextFile() { * */ public function testDuplicateHeaders() { - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/duplicate-headers.csv', 'text/csv'); + $resource = new DataResource(__DIR__ . '/../../../../data/duplicate-headers.csv', 'text/csv'); $datastore = $this->getImportJob($resource); $datastore->run(); @@ -130,7 +153,7 @@ public function testDuplicateHeaders() { * */ public function testLongColumnName() { - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/longcolumn.csv', 'text/csv'); + $resource = new DataResource(__DIR__ . '/../../../../data/longcolumn.csv', 'text/csv'); $datastore = $this->getImportJob($resource); $truncatedLongFieldName = 'extra_long_column_name_with_tons_of_characters_that_will_ne_e872'; @@ -149,7 +172,7 @@ public function testLongColumnName() { * */ public function testColumnNameSpaces() { - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/columnspaces.csv', 'text/csv'); + $resource = new DataResource(__DIR__ . '/../../../../data/columnspaces.csv', 'text/csv'); $datastore = $this->getImportJob($resource); $noMoreSpaces = 'column_name_with_spaces_in_it'; @@ -168,8 +191,7 @@ public function testColumnNameSpaces() { */ public function testSerialization() { $timeLimit = 40; - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/countries.csv', 'text/csv'); - $this->assertEquals(1, $resource->getID()); + $resource = new DataResource(__DIR__ . '/../../../../data/countries.csv', 'text/csv'); $datastore = $this->getImportJob($resource); $datastore->setTimeLimit($timeLimit); @@ -186,7 +208,7 @@ public function testSerialization() { * Test whether a potential multi-batch import works correctly. */ public function testLargeImport() { - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/Bike_Lane.csv', 'text/csv'); + $resource = new DataResource(__DIR__ . '/../../../../data/Bike_Lane.csv', 'text/csv'); $storage = new Memory(); @@ -223,7 +245,7 @@ public function testLargeImport() { */ public function testMultiplePasses() { $this->markTestIncomplete('This does not always use more than one pass.'); - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/Bike_Lane.csv', 'text/csv'); + $resource = new DataResource(__DIR__ . '/../../../../data/Bike_Lane.csv', 'text/csv'); $storage = new Memory(); @@ -265,9 +287,9 @@ public function testMultiplePasses() { */ public function testBadStorage() { $this->expectExceptionMessage('Storage must be an instance of ' . DatabaseTableInterface::class); - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/countries.csZv', 'text/csv'); + $resource = new DataResource(__DIR__ . '/../../../../data/countries.csZv', 'text/csv'); - ImportJob::get('1', new Memory(), [ + ImportJob::get($resource->getUniqueIdentifier(), new Memory(), [ 'resource' => $resource, 'storage' => new TestMemStorageBad(), 'parser' => Csv::getParser(), @@ -279,7 +301,7 @@ public function testBadStorage() { */ public function testNonStorage() { $this->expectExceptionMessage('Storage must be an instance of Drupal\common\Storage\DatabaseTableInterface'); - $resource = new DatastoreResource(1, __DIR__ . '/../../../../data/countries.csv', 'text/csv'); + $resource = new DataResource(__DIR__ . '/../../../../data/countries.csv', 'text/csv'); ImportJob::get('1', new Memory(), [ 'resource' => $resource, 'storage' => new class() { diff --git a/modules/datastore/tests/src/Unit/Service/Info/ImportInfoTest.php b/modules/datastore/tests/src/Unit/Service/Info/ImportInfoTest.php index f1b65030d1..d7b93daebb 100644 --- a/modules/datastore/tests/src/Unit/Service/Info/ImportInfoTest.php +++ b/modules/datastore/tests/src/Unit/Service/Info/ImportInfoTest.php @@ -4,7 +4,7 @@ use Contracts\Mock\Storage\Memory; use CsvParser\Parser\Csv; -use Drupal\datastore\DatastoreResource; +use Drupal\common\DataResource; use Drupal\datastore\Plugin\QueueWorker\ImportJob; use Drupal\datastore\Service\Info\ImportInfo; use Drupal\Tests\datastore\Unit\Plugin\QueueWorker\TestMemStorage; @@ -56,7 +56,7 @@ public function testGetBytesProcessedFileFetcher() { // Make a FileFetcher object. $storage = new Memory(); $config = [ - "resource" => (new DatastoreResource('id', 'path', 'mime')), + "resource" => (new DataResource('path', 'mime')), "storage" => new TestMemStorage(), "parser" => Csv::getParser(), "filePath" => 'test', @@ -87,7 +87,7 @@ public function testGetBytesProcessedImportJob() { // Make an ImportJob object. $storage = new Memory(); $config = [ - "resource" => (new DatastoreResource('id', 'path', 'mime')), + "resource" => (new DataResource('path', 'mime')), "storage" => new TestMemStorage(), "parser" => Csv::getParser(), ]; diff --git a/modules/datastore/tests/src/Unit/Service/ResourceProcessor/DictionaryEnforcerTest.php b/modules/datastore/tests/src/Unit/Service/ResourceProcessor/DictionaryEnforcerTest.php index 84c9d2d418..ee0631a871 100644 --- a/modules/datastore/tests/src/Unit/Service/ResourceProcessor/DictionaryEnforcerTest.php +++ b/modules/datastore/tests/src/Unit/Service/ResourceProcessor/DictionaryEnforcerTest.php @@ -14,6 +14,9 @@ use Drupal\datastore\Service\PostImport; use Drupal\datastore\Service\ResourceProcessorCollector; use Drupal\datastore\Service\ResourceProcessor\DictionaryEnforcer; +use Drupal\datastore\Storage\DatabaseTable; +use Drupal\datastore\Storage\DatabaseTableFactory; +use Drupal\jsonapi\JsonApiResource\Data; use Drupal\metastore\DataDictionary\DataDictionaryDiscovery; use Drupal\metastore\DataDictionary\DataDictionaryDiscoveryInterface; use Drupal\metastore\MetastoreService; @@ -59,7 +62,17 @@ public function testProcess() { $dictionary_discovery_service = (new Chain($this)) ->add(DataDictionaryDiscoveryInterface::class, 'dictionaryIdFromResource', 'dictionary-id') ->getMock(); - $dictionary_enforcer = new DictionaryEnforcer($alter_table_query_builder, $metastore_service, $dictionary_discovery_service); + $database_table_factory = (new Chain($this)) + ->add(DatabaseTableFactory::class, 'getInstance', DatabaseTable::class) + ->add(DatabaseTable::class, 'getTableName', 'datastore_table') + ->getMock(); + + $dictionary_enforcer = new DictionaryEnforcer( + $alter_table_query_builder, + $metastore_service, + $dictionary_discovery_service, + $database_table_factory + ); $container_chain = $this->getContainerChain($resource->getVersion()) ->add(AlterTableQueryInterface::class, 'execute') @@ -96,7 +109,16 @@ public function testProcessItemExecuteException() { $dictionary_discovery_service = (new Chain($this)) ->add(DataDictionaryDiscoveryInterface::class, 'dictionaryIdFromResource', 'data-dictionary') ->getMock(); - $dictionary_enforcer = new DictionaryEnforcer($alter_table_query_builder, $metastore_service, $dictionary_discovery_service); + $database_table_factory = (new Chain($this)) + ->add(DatabaseTableFactory::class, 'getInstance', DatabaseTable::class) + ->add(DatabaseTable::class, 'getTableName', 'datastore_table') + ->getMock(); + $dictionary_enforcer = new DictionaryEnforcer( + $alter_table_query_builder, + $metastore_service, + $dictionary_discovery_service, + $database_table_factory + ); $container_chain = $this->getContainerChain($resource->getVersion()) ->add(AlterTableQueryInterface::class, 'execute') @@ -133,7 +155,17 @@ public function testReturnDataDictionaryFields() { ->add(DataDictionaryDiscoveryInterface::class, 'getDataDictionaryMode', DataDictionaryDiscoveryInterface::MODE_SITEWIDE) ->add(DataDictionaryDiscoveryInterface::class, 'getSitewideDictionaryId','2') ->getMock(); - $dictionary_enforcer = new DictionaryEnforcer($alter_table_query_builder, $metastore_service, $dictionary_discovery_service); + $database_table_factory = (new Chain($this)) + ->add(DatabaseTableFactory::class, 'getInstance', DatabaseTable::class) + ->add(DatabaseTable::class, 'getTableName', 'datastore_table') + ->getMock(); + + $dictionary_enforcer = new DictionaryEnforcer( + $alter_table_query_builder, + $metastore_service, + $dictionary_discovery_service, + $database_table_factory + ); $container_chain = $this->getContainerChain($resource->getVersion()) ->add(AlterTableQueryInterface::class, 'execute') diff --git a/modules/datastore/tests/src/Unit/Storage/DatabaseTableFactoryTest.php b/modules/datastore/tests/src/Unit/Storage/DatabaseTableFactoryTest.php index 614d60e5f7..d4f7e220bd 100644 --- a/modules/datastore/tests/src/Unit/Storage/DatabaseTableFactoryTest.php +++ b/modules/datastore/tests/src/Unit/Storage/DatabaseTableFactoryTest.php @@ -3,7 +3,7 @@ namespace Drupal\Tests\datastore\Unit\Storage; use Drupal\Core\Database\Connection; -use Drupal\datastore\DatastoreResource; +use Drupal\common\DataResource; use Drupal\datastore\Storage\DatabaseTable; use Drupal\datastore\Storage\DatabaseTableFactory; use MockChain\Chain; @@ -39,8 +39,8 @@ public function test() { $factory->method("getDatabaseTable")->willReturn($databaseTable); - $resource = new DatastoreResource("blah", "", "text/csv"); - $object = $factory->getInstance($resource->getId(), ['resource' => $resource]); + $resource = new DataResource("", "text/csv"); + $object = $factory->getInstance($resource->getUniqueIdentifier(), ['resource' => $resource]); $this->assertTrue($object instanceof DatabaseTable); } diff --git a/modules/datastore/tests/src/Unit/Storage/DatabaseTableTest.php b/modules/datastore/tests/src/Unit/Storage/DatabaseTableTest.php index e936432e80..269a7b527d 100644 --- a/modules/datastore/tests/src/Unit/Storage/DatabaseTableTest.php +++ b/modules/datastore/tests/src/Unit/Storage/DatabaseTableTest.php @@ -2,13 +2,13 @@ namespace Drupal\Tests\datastore\Storage; +use Drupal\common\DataResource; use Drupal\Core\Database\Connection; use Drupal\Core\Database\DatabaseExceptionWrapper; use Drupal\Core\Database\Query\Insert; use Drupal\Core\Database\Query\Select; use Drupal\Core\Database\StatementWrapper; use Drupal\common\Storage\Query; -use Drupal\datastore\DatastoreResource; use Drupal\datastore\Storage\DatabaseTable; use Drupal\mysql\Driver\Database\mysql\Schema; use MockChain\Chain; @@ -476,7 +476,7 @@ private function getConnectionChain() { * Private. */ private function getResource() { - return new DatastoreResource("people", "", "text/csv"); + return new DataResource("", "text/csv"); } } diff --git a/modules/json_form_widget/tests/src/Unit/StringHelperTest.php b/modules/json_form_widget/tests/src/Unit/StringHelperTest.php index 8039ee8417..2edf8f451b 100644 --- a/modules/json_form_widget/tests/src/Unit/StringHelperTest.php +++ b/modules/json_form_widget/tests/src/Unit/StringHelperTest.php @@ -102,7 +102,6 @@ public function testHandleStringElement() { $data = 'mailto:john@doe.com'; $result = $string_helper->handleStringElement($definition, $data); - print_r($result['#default_value']); $this->assertEquals('john@doe.com', $result['#default_value']); }