PHPackages                             dmstr/yii2-active-record-search - PHPackages - PHPackages  [Skip to content](#main-content)[PHPackages](/)[Directory](/)[Categories](/categories)[Trending](/trending)[Leaderboard](/leaderboard)[Changelog](/changelog)[Analyze](/analyze)[Collections](/collections)[Log in](/login)[Sign up](/register)

1. [Directory](/)
2. /
3. [Search &amp; Filtering](/categories/search)
4. /
5. dmstr/yii2-active-record-search

ActiveYii2-extension[Search &amp; Filtering](/categories/search)

dmstr/yii2-active-record-search
===============================

Highly customizable ActiveRecord search indexer

1.0.6(2y ago)311.3k—0%1BSD-3-ClausePHP

Since Feb 16Pushed 9mo ago3 watchersCompare

[ Source](https://github.com/dmstr/yii2-active-record-search)[ Packagist](https://packagist.org/packages/dmstr/yii2-active-record-search)[ RSS](/packages/dmstr-yii2-active-record-search/feed)WikiDiscussions master Synced 1mo ago

READMEChangelogDependencies (3)Versions (10)Used By (0)

yii2-active-record-search
=========================

[](#yii2-active-record-search)

This module provides a simple but very flexible way to create a search index of almost any ActiveRecord entries.

The base idea is:

- we want a search index for every (configured) app language
- we want to group different Types of Models in search results
- we want to be able to get simple strings for different Types of Models where we can execute simple LIKE SQL queries on. No matter where the data for these strings come from.
- we want a fast frontend for the search module, so all required info to build links for the results should be pre-defined

Module Config
-------------

[](#module-config)

see [Module](src/Module.php) for available module params. These should allow to customize almost all aspects of the module and its behavior.

simple example config:

```
    'modules' => [
        'search' => [
            'class' => \dmstr\activeRecordSearch\Module::class,
            'layout' => '@backend/views/layouts/box',
            'frontendLayout' => '@app/views/layouts/container',
        ],
    ]
```

The module provides 2 types of controllers:

- frontend:
    - `\dmstr\activeRecordSearch\controllers\FrontendController`
- backend:
    - `\dmstr\activeRecordSearch\controllers\SearchGroupController` to manage (translate, en/disable) search groups which are created by the indexer
    - `\dmstr\activeRecordSearch\controllers\SearchController` to manage search items, attentions these will be overwritten by next indexer run. Should usually not be required
    - `\dmstr\activeRecordSearch\controllers\SearchGroupTranslationController` search group translations. Should usually not be required

Additionally the module provide a simple Search-Input widget:

- `\dmstr\activeRecordSearch\widgets\SearchInput`which will be used in frontend controller if not overwritten via modul `searchInputWidget` property

Indexer
-------

[](#indexer)

The heart of this module is the [SearchIndexer](src/components/SearchIndexer.php).

The Indexer has to be configured for all Type of Models that should be indexed.

see [SearchIndexer](src/components/SearchIndexer.php) for available module params and example.

### run the indexer

[](#run-the-indexer)

The Indexer should be defined and called as yii cli cmd:

```
    $config['controllerMap']['search-index'] = \dmstr\activeRecordSearch\commands\IndexController::class;

```

To automate indexer runs create cron jobs.

Example script which can run as cron

```
#!/bin/bash

. /root/export-env
LOG=/tmp/search-index-update
date > $LOG
yii search-index/update >> $LOG
date >> $LOG

```

### Simple Indexer Config example

[](#simple-indexer-config-example)

- In this example we index 2 types of models (products and accessories)
- For both types we define the AR model classes. These will be used to get the "data" by calling their `find()->all()` Methods
- The string that we will use for the search will be build (concatenation) from the values of the defined `attributes`
- For both types we definie the route that should be used to build the URL in results
- We also define which url\_params should be used to build the result URL
- `link_text` defines the text for the result Link

```
 $config['components']['searchIndexer'] = [
     'class' => \dmstr\activeRecordSearch\components\SearchIndexer::class,
     'languages' => function() {
         return project\components\CountryHelper::activeLanguages();
     },
     'fallbackLanguage' => 'en',
     'searchItems' => [
         'products'  => [
             'model_class'      => Product::class,
             'route'      => '/frontend/product/detail',
             'attributes' => [
                 'name',
                 'desc'
             ],
             'url_params' => ['productId' => 'id'],
             'link_text'  => 'name',
             'group'      => 'Products',
         ],
         'accessories'  => [
             'model_class'      => Accessory::class,
             'route'      => '/frontend/accessory/detail',
             'attributes' => [
                 'name',
                 'desc'
             ],
             'url_params' => ['accessoryId' => 'id'],
             'link_text'  => 'name',
             'group'      => 'P&A',
         ],
     ],
 ];

```

### Complex Indexer Config example

[](#complex-indexer-config-example)

- Here we define a bunch of different types where you can see that almost every param can be a callback so that one is able to define "non-static" results.
- The find\_method param can be used to overwrite the default find(). Useful to filter models e.g. by their status flags
- Callbacks can be used in almost any place e.g. to generate link\_text values from more than one attribute or even from attributes of different models (see tags where we use name prefixed by the name from tagGroup relation model)
- `products['attributes']` is an example where you can see how to define virtual attributes from relation models with simple array notation
- `news['attributes']['content']` is an example how to get parts of a json struct as 'content'
- if you have SEO url rules, you can define all required `url_params`
- ...

```
 $config['components']['searchIndexer'] = [
     'class' => \dmstr\activeRecordSearch\components\SearchIndexer::class,
     'languages' => function() {
         return project\components\CountryHelper::activeLanguages();
     },
     'searchItems' => [
         'products'        => [
             'model_class'      => Product::class,
             'find_method' => function ($item) {
                 return $item['model_class']::find()->andWhere(['archived' => 0]);
             },
             'route'      => '/frontend/product/detail',
             'attributes' => [
                 'name',
                 'frame',
                 'tags' => ['name'],
             ],
             'url_params' => ['productId' => 'id', 'productName' => 'name'],
             'link_text'  => function ($item) {
                 $parts = [];
                 if ($item->getClassificationTag()) {
                     $cTag = $item->getClassificationTag();
                     $cTag !== null && $parts[] = $cTag->name;
                 }
                 $parts[] = $item->name;
                 return implode(': ', array_filter($parts));
             },
             'group'      => 'Products',
         ],
         'product-archive' => [
             'model_class'      => Product::class,
             'find_method' => function ($item) {
                 return $item['model_class']::find()->andWhere(['archived' => 1]);
             },
             'route'      => '/frontend/product/detail',
             'attributes' => [
                 'name',
                 'frame',
                 'tags' => ['name'],
             ],
             'url_params' => ['productId' => 'id', 'productName' => 'name'],
             'link_text'  => function ($item) {
                 $parts = [];
                 if ($item->getClassificationTag()) {
                     $cTag = $item->getClassificationTag();
                     $cTag !== null && $parts[] = $cTag->name;
                 }
                 $parts[] = $item->name;
                 return implode(': ', array_filter($parts));
             },
             'group'      => 'Products Archive',
         ],
         'accessories'  => [
             'model_class'      => Accessory::class,
             'route'      => '/frontend/accessory/detail',
             'attributes' => [
                 'name',
                 'tags' => ['name'],
             ],
             'url_params' => ['accessoryId' => 'id'],
             'link_text'  => 'productName',
             'group'      => 'P&A',
         ],
         'news'         => [
             'model_class'      => PublicationItem::class,
             'route'      => '/publication/default/detail',
             'attributes' => [
                 'title',
                 'content' => function ($item) {
                     $content = Json::decode($item->content_widget_json);
                     return html_entity_decode(strip_tags($content['text_html']));
                 }
             ],
             'url_params' => [
                 'itemId' => 'id',
                 'title'  => function ($item) {
                     return $item->title;
                 }
             ],
             'link_text'  => 'title',
             'group'      => 'News',
         ],
         'tags'         => [
             'model_class'      => \project\modules\cruds\models\Tag::class,
             'route'      => '/productfinder/default/index',
             'find_method' => function ($item) {
                 // get used tagIds from finder
                 $tag_ids = \project\modules\productfinder\models\Productfinder::getFacetIdList('tag_ids');
                 return $item['model_class']::find()->andWhere(['id' => $tag_ids]);
             },
             'attributes' => [
                 'name',
             ],
             'url_params' => [
                 'mainTag'     => 'id',
                 'mainTagName' => 'name',
             ],
             'link_text'  => function ($item) {
                 return implode(': ', [$item->tagGroup->name, $item->name]);
             },
             'group'      => 'Product Tags',
         ],
     ],
 ];
```

Host info for URL Requests in Console Applications
--------------------------------------------------

[](#host-info-for-url-requests-in-console-applications)

If there is the need to make HTTP requests to fetch content, a Yii2 console application must be able to generate absolute URLs. Since console apps lack an HTTP context, you need to set the UrlManager's hostInfo property to enable proper URL generation. You can ie. set an ENV `CONSOLE_HOST_INFO` to the current URL of your page `https://example.com/`.

```
'urlManager' => [
    'hostInfo' => getenv('CONSOLE_HOST_INFO'),
]
```

```
'pages' => [
    // ...
    'attributes' => [
        'content' => function (Page $item) {
            // This method will fetch data using http requests
            return $item->fetchContentForSearchIndexer();
        }
    ],
    // ...
],
```

###  Health Score

37

—

LowBetter than 83% of packages

Maintenance40

Moderate activity, may be stable

Popularity28

Limited adoption so far

Community14

Small or concentrated contributor base

Maturity55

Maturing project, gaining track record

 Bus Factor2

2 contributors hold 50%+ of commits

How is this calculated?**Maintenance (25%)** — Last commit recency, latest release date, and issue-to-star ratio. Uses a 2-year decay window.

**Popularity (30%)** — Total and monthly downloads, GitHub stars, and forks. Logarithmic scaling prevents top-heavy scores.

**Community (15%)** — Contributors, dependents, forks, watchers, and maintainers. Measures real ecosystem engagement.

**Maturity (30%)** — Project age, version count, PHP version support, and release stability.

###  Release Activity

Cadence

Every ~135 days

Recently: every ~202 days

Total

7

Last Release

734d ago

### Community

Maintainers

![](https://www.gravatar.com/avatar/3b4ab2b6685ec71887908ec8cff261a16cd5bb24c69bb8ab52840e220f175f9d?d=identicon)[schmunk](/maintainers/schmunk)

---

Top Contributors

[![eluhr](https://avatars.githubusercontent.com/u/13000805?v=4)](https://github.com/eluhr "eluhr (11 commits)")[![handcode](https://avatars.githubusercontent.com/u/825574?v=4)](https://github.com/handcode "handcode (8 commits)")[![gbisurgi](https://avatars.githubusercontent.com/u/18024942?v=4)](https://github.com/gbisurgi "gbisurgi (2 commits)")[![schmunk42](https://avatars.githubusercontent.com/u/649031?v=4)](https://github.com/schmunk42 "schmunk42 (2 commits)")

---

Tags

yii2extension

### Embed Badge

![Health badge](/badges/dmstr-yii2-active-record-search/health.svg)

```
[![Health](https://phpackages.com/badges/dmstr-yii2-active-record-search/health.svg)](https://phpackages.com/packages/dmstr-yii2-active-record-search)
```

###  Alternatives

[skeeks/cms

SkeekS CMS — control panel and tools based on php framework Yii2

13825.6k47](/packages/skeeks-cms)[wbraganca/yii2-tagsinput

The yii2-tagsinput is a Yii 2 wrapper for \[bootstrap-tagsinput\](https://github.com/timschlechter/bootstrap-tagsinput). A user interface for managing tags.

32160.2k](/packages/wbraganca-yii2-tagsinput)[thrieu/yii2-grid-view-state

Save filters from GridView to session, keep the filter state between pages.

1313.1k1](/packages/thrieu-yii2-grid-view-state)[keygenqt/yii2-autocomplete-ajax

A simple way to search model id of the attributes model

1015.8k](/packages/keygenqt-yii2-autocomplete-ajax)[sammaye/yii2-solr

Solr plugin for the Yii2 framework built ontop of Solarium

1063.1k](/packages/sammaye-yii2-solr)[vxm/yii2-searchable

Simple full-text search for Yii2 active record

181.4k](/packages/vxm-yii2-searchable)

PHPackages © 2026

[Directory](/)[Categories](/categories)[Trending](/trending)[Changelog](/changelog)[Analyze](/analyze)
