From 92f54a94b1061b822bff8bcae8592ab2af1c7369 Mon Sep 17 00:00:00 2001
From: mikeryan <mikeryan@4420.no-reply.drupal.org>
Date: Tue, 5 Nov 2019 20:00:10 -0600
Subject: [PATCH] Issue #2949271 by webflo, mikeryan, chriscalip, heddn: Add
 feature id_mapping for destination plugin table

---
 src/Plugin/migrate/destination/Table.php      |  35 ++++-
 .../src/Kernel/MigrateTableIncrementTest.php  | 145 ++++++++++++++++++
 2 files changed, 172 insertions(+), 8 deletions(-)
 create mode 100755 tests/src/Kernel/MigrateTableIncrementTest.php

diff --git a/src/Plugin/migrate/destination/Table.php b/src/Plugin/migrate/destination/Table.php
index f04d9f66..9bac28c6 100755
--- a/src/Plugin/migrate/destination/Table.php
+++ b/src/Plugin/migrate/destination/Table.php
@@ -110,9 +110,14 @@ class Table extends DestinationBase implements ContainerFactoryPluginInterface {
    * {@inheritdoc}
    */
   public function import(Row $row, array $old_destination_id_values = []) {
-    $id = $row->getSourceIdValues();
-    if (count($id) != count($this->idFields)) {
-      throw new MigrateSkipProcessException('All the id fields are required for a table migration.');
+    $ids = [];
+    foreach ($this->idFields as $field => $fieldInfo) {
+      if ($row->hasDestinationProperty($field)) {
+        $ids[$field] = $row->getDestinationProperty($field);
+      }
+      elseif (!$row->hasDestinationProperty($field) && empty($fieldInfo['use_auto_increment'])) {
+        throw new MigrateSkipProcessException('All the id fields are required for a table migration.');
+      }
     }
 
     $values = $row->getDestination();
@@ -121,12 +126,26 @@ class Table extends DestinationBase implements ContainerFactoryPluginInterface {
       $values = array_intersect_key($values, $this->fields);
     }
 
-    $status = $this->dbConnection->merge($this->tableName)
-      ->key($id)
-      ->fields($values)
-      ->execute();
+    // Row contains empty id field with use_auto_increment enabled.
+    if (count($ids) < count($this->idFields)) {
+      $status = $id = $this->dbConnection->insert($this->tableName)
+        ->fields($values)
+        ->execute();
+      foreach ($this->idFields as $field => $fieldInfo) {
+        if (isset($fieldInfo['use_auto_increment']) && $fieldInfo['use_auto_increment'] === TRUE && !$row->hasDestinationProperty($field)) {
+          $row->setDestinationProperty($field, $id);
+          $ids[$field] = $id;
+        }
+      }
+    }
+    else {
+      $status = $this->dbConnection->merge($this->tableName)
+        ->key($ids)
+        ->fields($values)
+        ->execute();
+    }
 
-    return $status ? $id : NULL;
+    return $status ? $ids : NULL;
   }
 
   /**
diff --git a/tests/src/Kernel/MigrateTableIncrementTest.php b/tests/src/Kernel/MigrateTableIncrementTest.php
new file mode 100755
index 00000000..d4c96e10
--- /dev/null
+++ b/tests/src/Kernel/MigrateTableIncrementTest.php
@@ -0,0 +1,145 @@
+<?php
+
+namespace Drupal\Tests\migrate_plus\Kernel;
+
+use Drupal\Core\Database\Database;
+use Drupal\migrate\MigrateExecutable;
+use Drupal\Tests\migrate\Kernel\MigrateTestBase;
+
+/**
+ * Tests migration destination table with auto-increment keys.
+ *
+ * @group migrate
+ */
+class MigrateTableIncrementTest extends MigrateTestBase {
+
+  const TABLE_NAME = 'migrate_test_destination_table';
+
+  /**
+   * The database connection.
+   *
+   * @var \Drupal\Core\Database\Connection
+   */
+  protected $connection;
+
+  /**
+   * {@inheritdoc}
+   */
+  public static $modules = ['migrate_plus'];
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp() {
+    parent::setUp();
+    $this->connection = $this->container->get('database');
+    $this->connection->schema()->createTable(static::TABLE_NAME, [
+      'description' => 'Test table',
+      'fields' => [
+        'id' => [
+          'type' => 'serial',
+          'not null' => TRUE,
+        ],
+        'data1' => [
+          'type' => 'varchar',
+          'length' => '32',
+          'not null' => TRUE,
+        ],
+        'data2' => [
+          'type' => 'varchar',
+          'length' => '32',
+          'not null' => TRUE,
+        ],
+      ],
+      'primary key' => ['id'],
+    ]);
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function tearDown() {
+    $this->connection->schema()->dropTable(static::TABLE_NAME);
+    parent::tearDown();
+  }
+
+  /**
+   * Create a minimally valid migration with some source data.
+   *
+   * @return array
+   *   The migration definition.
+   */
+  public function tableDestinationMigration() {
+    return [
+      'dummy table' => [
+        [
+          'id' => 'migration_table_test',
+          'migration_tags' => ['Testing'],
+          'source' => [
+            'plugin' => 'embedded_data',
+            'data_rows' => [
+              [
+                'data1' => 'dummy1 value1',
+                'data2' => 'dummy2 value1',
+              ],
+              [
+                'data1' => 'dummy1 value2',
+                'data2' => 'dummy2 value2',
+              ],
+              [
+                'data1' => 'dummy1 value3',
+                'data2' => 'dummy2 value3',
+              ],
+            ],
+            'ids' => [
+              'data1' => ['type' => 'string'],
+            ],
+          ],
+          'destination' => [
+            'plugin' => 'table',
+            'table_name' => static::TABLE_NAME,
+            'id_fields' => [
+              'id' => [
+                'type' => 'integer',
+                'use_auto_increment' => TRUE,
+              ],
+            ],
+          ],
+          'process' => [
+            'data1' => 'data1',
+            'data2' => 'data2',
+          ],
+        ],
+      ],
+    ];
+  }
+
+  /**
+   * Tests table destination.
+   *
+   * @param array $definition
+   *   The migration definition.
+   *
+   * @dataProvider tableDestinationMigration
+   *
+   * @throws \Drupal\migrate\MigrateException
+   */
+  public function testTableDestination(array $definition) {
+    $migration = \Drupal::service('plugin.manager.migration')->createStubMigration($definition);
+
+    $executable = new MigrateExecutable($migration, $this);
+    $executable->import();
+
+    $values = $this->connection->select(static::TABLE_NAME)
+      ->fields(static::TABLE_NAME)
+      ->execute()
+      ->fetchAllAssoc('data1');
+
+    $this->assertEquals(1, $values['dummy1 value1']->id);
+    $this->assertEquals(2, $values['dummy1 value2']->id);
+    $this->assertEquals(3, $values['dummy1 value3']->id);
+    $this->assertEquals('dummy2 value3', $values['dummy1 value3']->data2);
+    $this->assertCount(3, $values);
+  }
+
+}
-- 
GitLab