diff --git a/docs/reference/search.rst b/docs/reference/search.rst
index 8bf2869816..141961bc59 100644
--- a/docs/reference/search.rst
+++ b/docs/reference/search.rst
@@ -5,6 +5,23 @@ The admin comes with a basic global search available in the upper navigation men
and look for filter with the option ``global_search`` set to true. If you are using the ``SonataDoctrineORMBundle``
any text filter will be set to ``true`` by default.
+Disabling the search by admin
+-----------------------------
+
+You can disable the search for a whole admin by setting the ``global_search`` attribute
+to ``false`` at your admin definition using the tag ``sonata.admin``.
+
+.. configuration-block::
+
+ .. code-block:: xml
+
+
+
+
+ App\Entity\Post
+
+
+
Customization
-------------
diff --git a/src/DependencyInjection/Compiler/AdminSearchCompilerPass.php b/src/DependencyInjection/Compiler/AdminSearchCompilerPass.php
new file mode 100644
index 0000000000..4211dc3f41
--- /dev/null
+++ b/src/DependencyInjection/Compiler/AdminSearchCompilerPass.php
@@ -0,0 +1,106 @@
+
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Sonata\AdminBundle\DependencyInjection\Compiler;
+
+use Sonata\AdminBundle\Admin\AdminInterface;
+use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
+use Symfony\Component\DependencyInjection\ContainerBuilder;
+use Symfony\Component\DependencyInjection\Exception\LogicException;
+
+/**
+ * This class configures which admins must be considered for global search at `SearchHandler`.
+ *
+ * @author Javier Spagnoletti
+ */
+final class AdminSearchCompilerPass implements CompilerPassInterface
+{
+ public const TAG_ATTRIBUTE_TOGGLE_SEARCH = 'global_search';
+
+ private const TAG_ADMIN = 'sonata.admin';
+
+ public function process(ContainerBuilder $container): void
+ {
+ if (!$container->hasDefinition('sonata.admin.search.handler')) {
+ return;
+ }
+
+ $adminSearch = [];
+
+ foreach ($container->findTaggedServiceIds(self::TAG_ADMIN) as $id => $tags) {
+ $this->validateAdminClass($container, $id);
+
+ foreach ($tags as $attributes) {
+ $globalSearch = $this->getGlobalSearchValue($attributes, $id);
+
+ if (null === $globalSearch) {
+ continue;
+ }
+
+ $adminSearch[$id] = $globalSearch;
+ }
+ }
+
+ $searchHandlerDefinition = $container->getDefinition('sonata.admin.search.handler');
+ $searchHandlerDefinition->addMethodCall('configureAdminSearch', [$adminSearch]);
+ }
+
+ /**
+ * @throws LogicException if the class in the given service definition is not
+ * a subclass of `AdminInterface`
+ */
+ private function validateAdminClass(ContainerBuilder $container, string $id): void
+ {
+ $definition = $container->getDefinition($id);
+
+ // Trim possible parameter delimiters ("%") from the class name.
+ $adminClass = trim($definition->getClass(), '%');
+ if (!class_exists($adminClass) && $container->hasParameter($adminClass)) {
+ $adminClass = $container->getParameter($adminClass);
+ }
+
+ if (!is_subclass_of($adminClass, AdminInterface::class)) {
+ throw new LogicException(sprintf(
+ 'Service "%s" must implement `%s`.',
+ $id,
+ AdminInterface::class
+ ));
+ }
+ }
+
+ /**
+ * @param array $attributes
+ *
+ * @throws LogicException if the attribute value is not of type boolean
+ */
+ private function getGlobalSearchValue(array $attributes, string $id): ?bool
+ {
+ $globalSearch = $attributes[self::TAG_ATTRIBUTE_TOGGLE_SEARCH] ?? null;
+
+ if (null === $globalSearch) {
+ return null;
+ }
+
+ if (!\is_bool($globalSearch)) {
+ throw new LogicException(sprintf(
+ 'Attribute "%s" in tag "%s" at service "%s" must be of type boolean, "%s" given.',
+ self::TAG_ATTRIBUTE_TOGGLE_SEARCH,
+ self::TAG_ADMIN,
+ $id,
+ \gettype($globalSearch)
+ ));
+ }
+
+ return $globalSearch;
+ }
+}
diff --git a/src/Search/SearchHandler.php b/src/Search/SearchHandler.php
index cb113b8620..619da7b1f7 100644
--- a/src/Search/SearchHandler.php
+++ b/src/Search/SearchHandler.php
@@ -35,6 +35,11 @@ class SearchHandler
*/
private $caseSensitive;
+ /**
+ * @var array
+ */
+ private $adminsSearchConfig = [];
+
/**
* NEXT_MAJOR: Change signature to __construct(bool $caseSensitive) and remove pool property.
*
@@ -69,6 +74,11 @@ public function __construct($deprecatedPoolOrCaseSensitive, $caseSensitive = tru
*/
public function search(AdminInterface $admin, $term, $page = 0, $offset = 20)
{
+ // If the search is disabled for the whole admin, skip any further processing.
+ if (false === ($this->adminsSearchConfig[$admin->getCode()] ?? true)) {
+ return false;
+ }
+
$datagrid = $admin->getDatagrid();
$found = false;
@@ -95,4 +105,15 @@ public function search(AdminInterface $admin, $term, $page = 0, $offset = 20)
return $pager;
}
+
+ /**
+ * Sets whether the search must be enabled or not for the passed admin codes.
+ * Receives an array with the admin code as key and a boolean as value.
+ *
+ * @param array $adminsSearchConfig
+ */
+ final public function configureAdminSearch(array $adminsSearchConfig): void
+ {
+ $this->adminsSearchConfig = $adminsSearchConfig;
+ }
}
diff --git a/src/SonataAdminBundle.php b/src/SonataAdminBundle.php
index 5c99f7235b..4aa333d8d2 100644
--- a/src/SonataAdminBundle.php
+++ b/src/SonataAdminBundle.php
@@ -16,6 +16,7 @@
use Mopa\Bundle\BootstrapBundle\Form\Type\TabType;
use Sonata\AdminBundle\DependencyInjection\Compiler\AddDependencyCallsCompilerPass;
use Sonata\AdminBundle\DependencyInjection\Compiler\AddFilterTypeCompilerPass;
+use Sonata\AdminBundle\DependencyInjection\Compiler\AdminSearchCompilerPass;
use Sonata\AdminBundle\DependencyInjection\Compiler\ExtensionCompilerPass;
use Sonata\AdminBundle\DependencyInjection\Compiler\GlobalVariablesCompilerPass;
use Sonata\AdminBundle\DependencyInjection\Compiler\ModelManagerCompilerPass;
@@ -50,6 +51,7 @@ public function build(ContainerBuilder $container)
{
$container->addCompilerPass(new AddDependencyCallsCompilerPass());
$container->addCompilerPass(new AddFilterTypeCompilerPass());
+ $container->addCompilerPass(new AdminSearchCompilerPass());
$container->addCompilerPass(new ExtensionCompilerPass());
$container->addCompilerPass(new GlobalVariablesCompilerPass());
$container->addCompilerPass(new ModelManagerCompilerPass());
diff --git a/tests/DependencyInjection/Compiler/AdminSearchCompilerPassTest.php b/tests/DependencyInjection/Compiler/AdminSearchCompilerPassTest.php
new file mode 100644
index 0000000000..015e4c1064
--- /dev/null
+++ b/tests/DependencyInjection/Compiler/AdminSearchCompilerPassTest.php
@@ -0,0 +1,57 @@
+
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Sonata\AdminBundle\Tests\DependencyInjection\Compiler;
+
+use Matthias\SymfonyDependencyInjectionTest\PhpUnit\AbstractCompilerPassTestCase;
+use Sonata\AdminBundle\DependencyInjection\Compiler\AdminSearchCompilerPass;
+use Sonata\AdminBundle\Tests\Fixtures\Admin\PostAdmin;
+use Symfony\Component\DependencyInjection\ContainerBuilder;
+use Symfony\Component\DependencyInjection\Definition;
+
+/**
+ * @author Javier Spagnoletti
+ */
+final class AdminSearchCompilerPassTest extends AbstractCompilerPassTestCase
+{
+ public function testProcess(): void
+ {
+ $adminFooDefinition = new Definition(PostAdmin::class);
+ $adminFooDefinition->addTag('sonata.admin', ['global_search' => true]);
+ $this->setDefinition('admin.foo', $adminFooDefinition);
+
+ $adminBarDefinition = new Definition(PostAdmin::class);
+ $adminBarDefinition->addTag('sonata.admin', ['global_search' => false]);
+ $this->setDefinition('admin.bar', $adminBarDefinition);
+
+ $adminBazDefinition = new Definition(PostAdmin::class);
+ $adminBazDefinition->addTag('sonata.admin', ['some_attribute' => 42]);
+ $this->setDefinition('admin.baz', $adminBazDefinition);
+
+ $searchHandlerDefinition = new Definition();
+ $this->setDefinition('sonata.admin.search.handler', $searchHandlerDefinition);
+
+ $this->compile();
+
+ $this->assertContainerBuilderHasServiceDefinitionWithMethodCall(
+ 'sonata.admin.search.handler',
+ 'configureAdminSearch',
+ [['admin.foo' => true, 'admin.bar' => false]]
+ );
+ }
+
+ protected function registerCompilerPass(ContainerBuilder $container): void
+ {
+ $container->addCompilerPass(new AdminSearchCompilerPass());
+ }
+}
diff --git a/tests/Search/SearchHandlerTest.php b/tests/Search/SearchHandlerTest.php
index a04ac4275d..b2b6ead498 100644
--- a/tests/Search/SearchHandlerTest.php
+++ b/tests/Search/SearchHandlerTest.php
@@ -24,14 +24,14 @@ class SearchHandlerTest extends TestCase
{
public function testBuildPagerWithNoGlobalSearchField(): void
{
- $filter = $this->getMockForAbstractClass(FilterInterface::class);
- $filter->expects($this->once())->method('getOption')->willReturn(false);
+ $filter = $this->createMock(FilterInterface::class);
+ $filter->expects($this->once())->method('getOption')->with('global_search')->willReturn(false);
$filter->expects($this->never())->method('setOption');
- $datagrid = $this->getMockForAbstractClass(DatagridInterface::class);
+ $datagrid = $this->createMock(DatagridInterface::class);
$datagrid->expects($this->once())->method('getFilters')->willReturn([$filter]);
- $admin = $this->getMockForAbstractClass(AdminInterface::class);
+ $admin = $this->createMock(AdminInterface::class);
$admin->expects($this->once())->method('getDatagrid')->willReturn($datagrid);
$handler = new SearchHandler(true);
@@ -43,20 +43,20 @@ public function testBuildPagerWithNoGlobalSearchField(): void
*/
public function testBuildPagerWithGlobalSearchField(bool $caseSensitive): void
{
- $filter = $this->getMockForAbstractClass(FilterInterface::class);
- $filter->expects($this->once())->method('getOption')->willReturn(true);
+ $filter = $this->createMock(FilterInterface::class);
+ $filter->expects($this->once())->method('getOption')->with('global_search')->willReturn(true);
$filter->expects($this->once())->method('setOption')->with('case_sensitive', $caseSensitive);
- $pager = $this->getMockForAbstractClass(PagerInterface::class);
+ $pager = $this->createMock(PagerInterface::class);
$pager->expects($this->once())->method('setPage');
$pager->expects($this->once())->method('setMaxPerPage');
- $datagrid = $this->getMockForAbstractClass(DatagridInterface::class);
+ $datagrid = $this->createMock(DatagridInterface::class);
$datagrid->expects($this->once())->method('getFilters')->willReturn([$filter]);
$datagrid->expects($this->once())->method('setValue');
$datagrid->expects($this->once())->method('getPager')->willReturn($pager);
- $admin = $this->getMockForAbstractClass(AdminInterface::class);
+ $admin = $this->createMock(AdminInterface::class);
$admin->expects($this->once())->method('getDatagrid')->willReturn($datagrid);
$handler = new SearchHandler($caseSensitive);
@@ -70,4 +70,46 @@ public function buildPagerWithGlobalSearchFieldProvider(): array
[false],
];
}
+
+ /**
+ * @dataProvider provideAdminSearchConfigurations
+ */
+ public function testAdminSearch($expected, $filterCallsCount, ?bool $enabled, string $adminCode): void
+ {
+ $filter = $this->createMock(FilterInterface::class);
+ $filter->expects($this->exactly($filterCallsCount))->method('getOption')->with('global_search')->willReturn(true);
+ $filter->expects($this->exactly($filterCallsCount))->method('setOption')->with('case_sensitive', true);
+
+ $pager = $this->createMock(PagerInterface::class);
+ $pager->expects($this->exactly($filterCallsCount))->method('setPage');
+ $pager->expects($this->exactly($filterCallsCount))->method('setMaxPerPage');
+
+ $datagrid = $this->createMock(DatagridInterface::class);
+ $datagrid->expects($this->exactly($filterCallsCount))->method('getFilters')->willReturn([$filter]);
+ $datagrid->expects($this->exactly($filterCallsCount))->method('setValue');
+ $datagrid->expects($this->exactly($filterCallsCount))->method('getPager')->willReturn($pager);
+
+ $admin = $this->createMock(AdminInterface::class);
+ $admin->expects($this->exactly($filterCallsCount))->method('getDatagrid')->willReturn($datagrid);
+ $admin->expects($this->once())->method('getCode')->willReturn($adminCode);
+
+ $handler = new SearchHandler(true);
+
+ if (null !== $enabled) {
+ $handler->configureAdminSearch([$adminCode => $enabled]);
+ }
+
+ if (false === $expected) {
+ $this->assertFalse($handler->search($admin, 'myservice'));
+ } else {
+ $this->assertInstanceOf($expected, $handler->search($admin, 'myservice'));
+ }
+ }
+
+ public function provideAdminSearchConfigurations(): iterable
+ {
+ yield 'admin_search_enabled' => [PagerInterface::class, 1, true, 'admin.foo'];
+ yield 'admin_search_disabled' => [false, 0, false, 'admin.bar'];
+ yield 'admin_search_omitted' => [PagerInterface::class, 1, null, 'admin.baz'];
+ }
}
diff --git a/tests/SonataAdminBundleTest.php b/tests/SonataAdminBundleTest.php
index 74bd1d1553..c77d787dcc 100644
--- a/tests/SonataAdminBundleTest.php
+++ b/tests/SonataAdminBundleTest.php
@@ -16,6 +16,7 @@
use PHPUnit\Framework\TestCase;
use Sonata\AdminBundle\DependencyInjection\Compiler\AddDependencyCallsCompilerPass;
use Sonata\AdminBundle\DependencyInjection\Compiler\AddFilterTypeCompilerPass;
+use Sonata\AdminBundle\DependencyInjection\Compiler\AdminSearchCompilerPass;
use Sonata\AdminBundle\DependencyInjection\Compiler\ExtensionCompilerPass;
use Sonata\AdminBundle\DependencyInjection\Compiler\GlobalVariablesCompilerPass;
use Sonata\AdminBundle\DependencyInjection\Compiler\ModelManagerCompilerPass;
@@ -35,7 +36,7 @@ public function testBuild(): void
{
$containerBuilder = $this->createMock(ContainerBuilder::class);
- $containerBuilder->expects($this->exactly(7))
+ $containerBuilder->expects($this->exactly(8))
->method('addCompilerPass')
->willReturnCallback(function (CompilerPassInterface $pass, $type = PassConfig::TYPE_BEFORE_OPTIMIZATION): void {
if ($pass instanceof AddDependencyCallsCompilerPass) {
@@ -46,6 +47,10 @@ public function testBuild(): void
return;
}
+ if ($pass instanceof AdminSearchCompilerPass) {
+ return;
+ }
+
if ($pass instanceof ExtensionCompilerPass) {
return;
}
@@ -70,6 +75,7 @@ public function testBuild(): void
'CompilerPass is not one of the expected types. Expects "%s", "%s", "%s", "%s", "%s", "%s" or "%s", but got "%s".',
AddDependencyCallsCompilerPass::class,
AddFilterTypeCompilerPass::class,
+ AdminSearchCompilerPass::class,
ExtensionCompilerPass::class,
GlobalVariablesCompilerPass::class,
ModelManagerCompilerPass::class,