A full text search engine with tokenization, stemming, typo tolerance, filters and geo support based on only PHP and SQLite.
Loupe…
"
quotation marks-
as modifierLIKE %...%
queries with a way better searchIf this is your first encounter with Loupe, you might want to read the blog post on Medium or as
Markdown file that should give you more information about the reasons and what you can do with it.
Note that some implementation details (e.g. libraries used) referenced in this blog post are not up-to-date anymore.
Performance depends on many factors but here are some ballpark numbers based on indexing the
~32k movies fixture provided by MeiliSearch.
Amakin Dkywalker
with typo tolerance and relevance ranking takes about 100 msNote that anything above 50k documents is probably not a use case for Loupe. You can run your own benchmarks
using the scripts in the bin/bench
folder: index.php
for indexing and search.php
for searching.
Please, also read the Performance chapter in the docs. You may report your own performance
measurements and more details in the respective discussion.
If you are familiar with MeiliSearch, you will notice that the API is very much inspired by it. The
reasons for this are simple:
I even took the liberty to copy some of their test data to feed Loupe for functional tests.
pdo_sqlite
available and your installed SQLite version is at least 3.16.0. This is whencomposer require loupe/loupe
.The first step is configuring and creating a client.
use Loupe\Loupe\Config\TypoTolerance;
use Loupe\Loupe\Configuration;
use Loupe\Loupe\LoupeFactory;
use Loupe\Loupe\SearchParameters;
$configuration = Configuration::create()
->withPrimaryKey('uuid') // optional, by default it's 'id'
->withSearchableAttributes(['firstname', 'lastname']) // optional, by default it's ['*'] - everything is indexed
->withFilterableAttributes(['departments', 'age'])
->withSortableAttributes(['lastname'])
->withTypoTolerance(TypoTolerance::create()->withFirstCharTypoCountsDouble(false)) // can be further fine-tuned but is enabled by default
;
$loupe = (new LoupeFactory())->create('path/to/my_loupe_data_dir', $configuration);
To create an in-memory search client:
$loupe = (new LoupeFactory())->createInMemory($configuration);
$loupe->addDocuments([
[
'uuid' => 2,
'firstname' => 'Uta',
'lastname' => 'Koertig',
'departments' => [
'Development',
'Backoffice',
],
'age' => 29,
],
[
'uuid' => 6,
'firstname' => 'Huckleberry',
'lastname' => 'Finn',
'departments' => [
'Backoffice',
],
'age' => 18,
],
]);
$searchParameters = SearchParameters::create()
->withQuery('Gucleberry')
->withAttributesToRetrieve(['uuid', 'firstname'])
->withFilter("(departments = 'Backoffice' OR departments = 'Project Management') AND age > 17")
->withSort(['lastname:asc'])
;
$results = $loupe->search($searchParameters);
foreach ($results->getHits() as $hit) {
echo $hit['title'] . PHP_EOL;
}
The $results
array contains a list of search hits and metadata about the query.
print_r($results->toArray());
[
'hits' => [
[
'uuid' => 6,
'firstname' => 'Huckleberry'
]
],
'query' => 'Gucleberry',
'processingTimeMs' => 4,
'hitsPerPage' => 20,
'page' => 1,
'totalPages' => 1,
'totalHits' => 1
]
“Why Loupe?” you ask? “Loupe” means “magnifier” in French and I felt like this was the appropriate choice for this
library after having given my PHP crawler library a French name 😃