Passed
Push — master ( 676de8...b86364 )
by Alex
57s queued 12s
created

QueryFilterManager::applyFilter()   B

Complexity

Conditions 6
Paths 6

Size

Total Lines 33
Code Lines 23

Duplication

Lines 0
Ratio 0 %

Importance

Changes 1
Bugs 0 Features 0
Metric Value
eloc 23
c 1
b 0
f 0
dl 0
loc 33
rs 8.9297
cc 6
nc 6
nop 3
1
<?php
2
3
declare(strict_types=1);
4
5
namespace Arp\DoctrineQueryFilter;
6
7
use Arp\DoctrineQueryFilter\Exception\QueryFilterManagerException;
8
use Arp\DoctrineQueryFilter\Filter\Exception\FilterException;
9
use Arp\DoctrineQueryFilter\Filter\FilterFactoryInterface;
10
use Arp\DoctrineQueryFilter\Filter\FilterInterface;
11
use Arp\DoctrineQueryFilter\Metadata\Metadata;
12
use Arp\DoctrineQueryFilter\Metadata\MetadataInterface;
13
use Doctrine\ORM\EntityManager;
14
use Doctrine\ORM\QueryBuilder as DoctrineQueryBuilder;
15
16
/**
17
 * @author  Alex Patterson <[email protected]>
18
 * @package Arp\DoctrineQueryFilter
19
 */
20
class QueryFilterManager implements QueryFilterManagerInterface
21
{
22
    /**
23
     * @var FilterFactoryInterface
24
     */
25
    private FilterFactoryInterface $filterFactory;
26
27
    /**
28
     * @param FilterFactoryInterface $filterFactory
29
     */
30
    public function __construct(FilterFactoryInterface $filterFactory)
31
    {
32
        $this->filterFactory = $filterFactory;
33
    }
34
35
    /**
36
     * Apply the query filters to the provided query builder instance
37
     *
38
     * @param DoctrineQueryBuilder|QueryBuilderInterface $queryBuilder
39
     * @param string                                     $entityName
40
     * @param array                                      $criteria
41
     *
42
     * @return QueryBuilderInterface
43
     *
44
     * @throws QueryFilterManagerException
45
     */
46
    public function filter($queryBuilder, string $entityName, array $criteria): QueryBuilderInterface
47
    {
48
        $queryBuilder = $this->getQueryBuilder($queryBuilder);
49
50
        if (!empty($criteria['filters']) && is_array($criteria['filters'])) {
51
            $metadata = $this->createMetadataProxy($queryBuilder->getEntityManager(), $entityName);
52
            foreach ($criteria['filters'] as $data) {
53
                $this->applyFilter($queryBuilder, $metadata, $data);
54
            }
55
        }
56
57
        return $queryBuilder;
58
    }
59
60
    /**
61
     * Create a new filter matching $name with the provided $options
62
     *
63
     * @param string $name
64
     * @param array  $options
65
     *
66
     * @return FilterInterface
67
     *
68
     * @throws QueryFilterManagerException
69
     */
70
    public function createFilter(string $name, array $options = []): FilterInterface
71
    {
72
        try {
73
            return $this->filterFactory->create($this, $name, $options);
74
        } catch (\Throwable $e) {
75
            throw new QueryFilterManagerException(
76
                sprintf('Failed to create filter \'%s\': %s', $name, $e->getMessage()),
77
                $e->getCode(),
78
                $e
79
            );
80
        }
81
    }
82
83
    /**
84
     * @param QueryBuilderInterface|DoctrineQueryBuilder $queryBuilder
85
     *
86
     * @return QueryBuilderInterface
87
     *
88
     * @throws QueryFilterManagerException
89
     */
90
    private function getQueryBuilder($queryBuilder): QueryBuilderInterface
91
    {
92
        if ($queryBuilder instanceof DoctrineQueryBuilder) {
93
            $queryBuilder = $this->createQueryBuilderProxy($queryBuilder);
94
        }
95
96
        if (!$queryBuilder instanceof QueryBuilderInterface) {
0 ignored issues
show
introduced by
$queryBuilder is always a sub-type of Arp\DoctrineQueryFilter\QueryBuilderInterface.
Loading history...
97
            throw new QueryFilterManagerException(
98
                sprintf(
99
                    'The \'queryBuilder\' argument must be an object of type \'%s\' or \'%s\'; '
100
                    . '\'%s\' provided in \'%s\'',
101
                    QueryBuilderInterface::class,
102
                    DoctrineQueryBuilder::class,
103
                    is_object($queryBuilder) ? get_class($queryBuilder) : gettype($queryBuilder),
104
                    static::class
105
                )
106
            );
107
        }
108
109
        return $queryBuilder;
110
    }
111
112
    /**
113
     * @param QueryBuilderInterface $queryBuilder
114
     * @param MetadataInterface     $metadata
115
     * @param array|FilterInterface $data
116
     *
117
     * @throws QueryFilterManagerException
118
     */
119
    private function applyFilter(QueryBuilderInterface $queryBuilder, MetadataInterface $metadata, $data): void
120
    {
121
        if ($data instanceof FilterInterface) {
122
            $filter = $data;
123
            $data = [];
124
        } elseif (is_array($data)) {
0 ignored issues
show
introduced by
The condition is_array($data) is always true.
Loading history...
125
            $filterName = $data['name'] ?? null;
126
127
            if (empty($filterName)) {
128
                throw new QueryFilterManagerException(
129
                    sprintf('The required \'name\' configuration option is missing in \'%s\'', static::class)
130
                );
131
            }
132
133
            $filter = $this->createFilter($filterName, $data['options'] ?? []);
134
        } else {
135
            throw new QueryFilterManagerException(
136
                sprintf(
137
                    'The \'data\' argument must be an \'array\' or object of type \'%s\'; \'%s\' provided in \'%s\'',
138
                    FilterInterface::class,
139
                    is_object($data) ? get_class($data) : gettype($data),
140
                    static::class
141
                )
142
            );
143
        }
144
145
        try {
146
            $filter->filter($queryBuilder, $metadata, $data);
147
        } catch (FilterException $e) {
148
            throw new QueryFilterManagerException(
149
                sprintf('Failed to apply query filter for entity \'%s\': %s', $metadata->getName(), $e->getMessage()),
150
                $e->getCode(),
151
                $e
152
            );
153
        }
154
    }
155
156
    /**
157
     * @param EntityManager $entityManager
158
     * @param string        $entityName
159
     *
160
     * @return MetadataInterface
161
     *
162
     * @throws QueryFilterManagerException
163
     */
164
    private function createMetadataProxy(EntityManager $entityManager, string $entityName): MetadataInterface
165
    {
166
        try {
167
            return new Metadata($entityManager->getClassMetadata($entityName));
168
        } catch (\Throwable $e) {
169
            throw new QueryFilterManagerException(
170
                sprintf('Failed to fetch entity metadata for class \'%s\': %s', $entityName, $e->getMessage()),
171
                $e->getCode(),
172
                $e
173
            );
174
        }
175
    }
176
177
    /**
178
     * @param DoctrineQueryBuilder $queryBuilder
179
     *
180
     * @return QueryBuilderInterface
181
     */
182
    private function createQueryBuilderProxy(DoctrineQueryBuilder $queryBuilder): QueryBuilderInterface
183
    {
184
        return new QueryBuilder($queryBuilder);
185
    }
186
}
187