Skip to content

Commit

Permalink
Create Atlas Search example
Browse files Browse the repository at this point in the history
  • Loading branch information
GromNaN committed Sep 6, 2023
1 parent f6ce211 commit 8714314
Show file tree
Hide file tree
Showing 3 changed files with 189 additions and 0 deletions.
129 changes: 129 additions & 0 deletions examples/atlas-search.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,129 @@
<?php

/**
* This example demonstrates how to create an Atlas Search index and perform a search query.
* It requires a MongoDB Atlas M10+ cluster with Sample Dataset loaded.
*
* Use the MONGODB_URI environment variable to specify the connection string from the Atlas UI.
*/

declare(strict_types=1);

namespace MongoDB\Examples;

use Closure;
use MongoDB\Client;
use RuntimeException;

use function define;
use function getenv;
use function hrtime;
use function iterator_to_array;
use function sleep;
use function str_contains;

require __DIR__ . '/../vendor/autoload.php';

$uri = getenv('MONGODB_URI');
if (! $uri || ! str_contains($uri, '.mongodb.net')) {
echo 'This example requires a MongoDB Atlas cluster.', "\n";
echo 'Make sure you set the MONGODB_URI environment variable.', "\n";
exit(1);
}

// Atlas Search index management operations are asynchronous.
// They usually take less than 5 minutes to complete.
define('WAIT_TIMEOUT_SEC', 300);

$client = new Client($uri);
$collection = $client->selectCollection('sample_airbnb', 'listingsAndReviews');

$count = $collection->estimatedDocumentCount();
if ($count === 0) {
echo 'This example requires the sample_airbnb database with the listingsAndReviews collection.', "\n";
echo 'Load the sample dataset in your MongoDB Atlas cluster before running this example:', "\n";
echo ' https://www.mongodb.com/docs/atlas/sample-data/', "\n";
exit(1);
}

// Delete the index if it already exists
$indexes = iterator_to_array($collection->listSearchIndexes());
foreach ($indexes as $index) {
if ($index->name === 'default') {
echo "The index already exists. Dropping it.\n";
$collection->dropSearchIndex($index->name);

// Wait for the index to be deleted.
wait(function () use ($collection) {
echo '.';
foreach ($collection->listSearchIndexes() as $index) {
if ($index->name === 'default') {
return false;
}
}

return true;
});
}
}

// Create the search index
echo "\nCreating the index.\n";
$collection->createSearchIndex(
/* The index definition requires a mapping
* See: https://www.mongodb.com/docs/atlas/atlas-search/define-field-mappings/ */
['mappings' => ['dynamic' => true]],
// "default" is the default index name, this config can be omitted.
['name' => 'default'],
);

// Wait for the index to be ready.
wait(function () use ($collection) {
echo '.';
foreach ($collection->listSearchIndexes() as $index) {
if ($index->name === 'default') {
return $index->queryable;
}
}

return false;
});

// Perform a text search
echo "\n", 'Performing a text search...', "\n";
$results = $collection->aggregate([
[
'$search' => [
'index' => 'default',
'text' => [
'query' => 'view beach ocean',
'path' => ['name'],
],
],
],
['$project' => ['name' => 1, 'description' => 1]],
['$limit' => 10],
])->toArray();

foreach ($results as $document) {
echo ' - ', $document['name'], "\n";
}

echo "\n", 'Enjoy MongoDB Atlas Search!', "\n\n";

/**
* This function waits until the callback returns true or the timeout is reached.
*/
function wait(Closure $callback): void
{
$timeout = hrtime()[0] + WAIT_TIMEOUT_SEC;
while (hrtime()[0] < $timeout) {
if ($callback()) {
return;
}

sleep(5);
}

throw new RuntimeException('Time out');
}
24 changes: 24 additions & 0 deletions psalm-baseline.xml
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,30 @@
<code><![CDATA[$clientEncryption->decrypt($document->encryptedField)]]></code>
</MixedArgument>
</file>
<file src="examples/atlas-search.php">
<MixedArgument>
<code><![CDATA[$document['name']]]></code>
<code><![CDATA[$index->name]]></code>
</MixedArgument>
<MixedArrayAccess>
<code><![CDATA[$document['name']]]></code>
</MixedArrayAccess>
<MixedAssignment>
<code>$document</code>
<code>$index</code>
<code>$index</code>
<code>$index</code>
</MixedAssignment>
<MixedPropertyFetch>
<code><![CDATA[$index->name]]></code>
<code><![CDATA[$index->name]]></code>
<code><![CDATA[$index->name]]></code>
<code><![CDATA[$index->queryable]]></code>
</MixedPropertyFetch>
<PossiblyFalseArgument>
<code>$uri</code>
</PossiblyFalseArgument>
</file>
<file src="src/ChangeStream.php">
<DeprecatedConstant>
<code>self::CURSOR_NOT_FOUND</code>
Expand Down
36 changes: 36 additions & 0 deletions tests/ExamplesTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,9 @@
namespace MongoDB\Tests;

use Generator;
use MongoDB\Client;

use function getenv;

/** @runTestsInSeparateProcesses */
final class ExamplesTest extends FunctionalTestCase
Expand Down Expand Up @@ -179,6 +182,39 @@ public static function provideExamples(): Generator
];
}

/**
* MongoDB Atlas Search example requires a MongoDB Atlas M10+ cluster with MongoDB 7.0+ and sample data loaded.
* Tips for insiders: if using a cloud-dev server, append ".mongodb.net" to the MONGODB_URI.
*
* @group atlas
*/
public function testAtlasSearch(): void
{
$uri = getenv('MONGODB_URI') ?? '';
if (! self::isAtlas($uri)) {
$this->markTestSkipped('Atlas Search examples are only supported on MongoDB Atlas');
}

$this->skipIfServerVersion('<', '7.0', 'Atlas Search examples require MongoDB 7.0 or later');

$client = new Client($uri);
$collection = $client->selectCollection('sample_airbnb', 'listingsAndReviews');
$count = $collection->estimatedDocumentCount();
if ($count === 0) {
$this->markTestSkipped('Atlas Search examples require the sample_airbnb database with the listingsAndReviews collection');
}

// Clean variables to avoid conflict with example
unset($uri, $client, $collection, $count);

require __DIR__ . '/../examples/atlas-search.php';

$output = $this->getActualOutputForAssertion();
$this->assertStringContainsString("\nCreating the index.\n...", $output);
$this->assertStringContainsString("\nPerforming a text search...\n - ", $output);
$this->assertStringContainsString("\nEnjoy MongoDB Atlas Search!\n", $output);
}

public function testChangeStream(): void
{
$this->skipIfChangeStreamIsNotSupported();
Expand Down

0 comments on commit 8714314

Please sign in to comment.