commit 3bce4bd2007cc432244fbef886e48848e52d1c8b
Author: Ilias Dimopoulos <idimopoulos@hotmail.com>
Date:   Tue Apr 12 16:30:17 2016 +0200

    Support for arbitary string ID.

diff --git a/src/Plugin/search_api/datasource/ContentEntity.php b/src/Plugin/search_api/datasource/ContentEntity.php
index 8d5993b..ab63cb5 100644
--- a/src/Plugin/search_api/datasource/ContentEntity.php
+++ b/src/Plugin/search_api/datasource/ContentEntity.php
@@ -368,7 +368,7 @@ class ContentEntity extends DatasourcePluginBase {
 
     $entity_ids = array();
     foreach ($ids as $item_id) {
-      list($entity_id, $langcode) = explode(':', $item_id, 2);
+      list($entity_id, $langcode) = preg_split('/:(?=[a-zA-Z0-9-]+$)/', $item_id);
       if (isset($allowed_languages[$langcode])) {
         $entity_ids[$entity_id][$item_id] = $langcode;
       }
diff --git a/src/Utility.php b/src/Utility.php
index 2cf7521..fd4a588 100644
--- a/src/Utility.php
+++ b/src/Utility.php
@@ -365,7 +365,8 @@ class Utility {
    *   might be NULL. If $separate_last is TRUE it's the exact other way round.
    */
   public static function splitPropertyPath($property_path, $separate_last = TRUE, $separator = ':') {
-    $function = $separate_last ? 'strrpos' : 'strpos';
+    $is_default_separator = $separator != IndexInterface::DATASOURCE_ID_SEPARATOR;
+    $function = $separate_last && $is_default_separator ? 'strrpos' : 'strpos';
     $pos = $function($property_path, $separator);
     if ($pos !== FALSE) {
       return array(
diff --git a/tests/src/Kernel/EntityStringIdTest.php b/tests/src/Kernel/EntityStringIdTest.php
new file mode 100644
index 0000000..cf32d14
--- /dev/null
+++ b/tests/src/Kernel/EntityStringIdTest.php
@@ -0,0 +1,160 @@
+<?php
+
+namespace Drupal\Tests\search_api\Kernel;
+
+use Drupal\entity_test\Entity\EntityTestStringId;
+use Drupal\KernelTests\KernelTestBase;
+use Drupal\search_api\Entity\Index;
+use Drupal\search_api\Entity\Server;
+
+/**
+ * Tests entity indexing that are using string IDs.
+ *
+ * Current limit of the search_api id is 50 characters. The format of the
+ * saved item is entity:<entity_type_id>:<entity_id>:<language_code>.
+ * For this test we are using:
+ *
+ * entity:entity_test_string_id:<string_id>:und
+ * entity = 6 characters.
+ * entity_test_string_id = 21 characters.
+ * und = 3 characters.
+ * 3 x ':' = 3 characters.
+ * 50 - 6 - 21 - 3 - 3 = 17 characters left for the ID.
+ *
+ * @group search_api
+ */
+class EntityStringIdTest extends KernelTestBase {
+
+  /**
+   * The test entity type used in the test.
+   *
+   * @var string
+   */
+  protected $testEntityTypeId = 'entity_test_string_id';
+
+  /**
+   * The search server used for testing.
+   *
+   * @var \Drupal\search_api\ServerInterface
+   */
+  protected $server;
+
+  /**
+   * The search index used for testing.
+   *
+   * @var \Drupal\search_api\IndexInterface
+   */
+  protected $index;
+
+  /**
+   * Modules to enable for this test.
+   *
+   * @var string[]
+   */
+  public static $modules = array(
+    'search_api',
+    'search_api_test_backend',
+    'language',
+    'user',
+    'system',
+    'entity_test',
+  );
+
+  /**
+   * An array of langcodes.
+   *
+   * @var string[]
+   */
+  protected $langcodes;
+
+  /**
+   * {@inheritdoc}
+   */
+  public function setUp() {
+    parent::setUp();
+
+    // Enable translation for the entity_test module.
+    \Drupal::state()->set('entity_test_string_id.translation', FALSE);
+
+    $this->installSchema('search_api', array('search_api_item', 'search_api_task'));
+    $this->installEntitySchema('entity_test_string_id');
+
+    // Do not use a batch for tracking the initial items after creating an
+    // index when running the tests via the GUI. Otherwise, it seems Drupal's
+    // Batch API gets confused and the test fails.
+    \Drupal::state()->set('search_api_use_tracking_batch', FALSE);
+
+    // Create a test server.
+    $this->server = Server::create(array(
+      'name' => 'Test Server',
+      'id' => 'test_server',
+      'status' => 1,
+      'backend' => 'search_api_test_backend',
+    ));
+    $this->server->save();
+
+    // Create a test index.
+    $this->index = Index::create(array(
+      'name' => 'Test Index',
+      'id' => 'test_index',
+      'status' => 1,
+      'datasource_settings' => array(
+        'entity:' . $this->testEntityTypeId => array(
+          'plugin_id' => 'entity:' . $this->testEntityTypeId,
+          'settings' => array(),
+        ),
+      ),
+      'tracker_settings' => array(
+        'default' => array(
+          'plugin_id' => 'default',
+          'settings' => array(),
+        )
+      ),
+      'server' => $this->server->id(),
+      'options' => array('index_directly' => FALSE),
+    ));
+    $this->index->save();
+  }
+
+  /**
+   * Tests Uris as Ids.
+   *
+   * @dataProvider entityStringIdList
+   */
+  public function testUriStringId($entity_id) {
+    $entity_1 = EntityTestStringId::create(array(
+      'id' => $entity_id,
+      'name' => 'String Test',
+      'user_id' => $this->container->get('current_user')->id(),
+    ));
+    $entity_1->save();
+
+    // Test that the datasource returns the correct item IDs.
+    $datasource = $this->index->getDatasource('entity:' . $this->testEntityTypeId);
+    $datasource_item_ids = $datasource->getItemIds();
+    sort($datasource_item_ids);
+    $expected = array(
+      $entity_id . ':und',
+    );
+    $this->assertEquals($expected, $datasource_item_ids, 'Datasource returns correct item ids.');
+
+    // Test indexing the new entities. One should fail so only one should be indexed.
+    $this->assertEquals(0, $this->index->getTrackerInstance()->getIndexedItemsCount(), 'The index is empty.');
+    $this->assertEquals(1, $this->index->getTrackerInstance()->getTotalItemsCount(), 'There is one item to be indexed.');
+    $this->index->indexItems();
+    $this->assertEquals(1, $this->index->getTrackerInstance()->getIndexedItemsCount(), 'One item has been indexed.');
+  }
+
+  /**
+   * Provides string ids to test.
+   *
+   * @return array An array of arrays which contain a list of parameters to be
+   *   passed to the appropriate tests.
+   */
+  function entityStringIdList(){
+    return[
+      ['short_string_id'], // Normal string matched by \w regex code.
+      ['http://drupal.org'], // Uri string not matched by \w regex code.
+    ];
+  }
+}
