aboutsummaryrefslogtreecommitdiff
path: root/lib/classes/JsonApi/Routes/Resources/ResourceIndex.php
blob: 07f5b12be016f23f1b54ea50bc147b73b2c52c8d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
<?php
namespace JsonApi\Routes\Resources;

use JsonApi\Schemas\ResourceSchema;
use JsonApi\Errors\BadRequestException;
use JsonApi\JsonApiController;
use Psr\Http\Message\{
    RequestInterface as Request,
    ResponseInterface as Response
};

final class ResourceIndex extends JsonApiController
{
    protected $allowedFilteringParameters = ['level', 'class'];
    protected $allowedIncludePaths = [ResourceSchema::REL_CATEGORY];
    protected $allowedPagingParameters = ['offset', 'limit'];

    public function __invoke(Request $request, Response $response, array $args): Response
    {
        [$offset, $limit] = $this->getOffsetAndLimit();
        [$condition, $parameters] = $this->getConditionAndParameters(
            $this->getFilters()
        );

        $total = \Resource::countBySql($condition, $parameters);
        $resources = \Resource::findBySQL(
            "{$condition} LIMIT {$offset}, {$limit}",
            $parameters
        );

        return $this->getPaginatedContentResponse($resources, $total);
    }

    private function getFilters()
    {
        $filters = [];

        $filtering = $this->getQueryParameters()->getFilteringParameters() ?? [];

        if (array_key_exists('level', $filtering)) {
            if (!ctype_digit($filtering['level'])) {
                throw new BadRequestException('Level filter must be an int.');
            }

            $filters['level'] = (int) $filtering['level'];
        }

        if (array_key_exists('class', $filtering)) {
            if (empty($filtering['class'])) {
                throw new BadRequestException('Class filter must be not be empty.');
            }

            $filters['class'] = $filtering['class'];
        }

        return $filters;
    }

    private function getConditionAndParameters(array $filters): array
    {
        $joins = [];
        $conditions = [];
        $parameters = [];

        if (array_key_exists('level', $filters)) {
            $conditions[] = '`resources`.`level` = :level';
            $parameters[':level'] = $filters['level'];
        }

        if (array_key_exists('class', $filters)) {
            $joins[] = 'JOIN `resource_categories`
                        ON `resources`.`category_id` = `resource_categories`.`id`';
            $conditions[] = '`resource_categories`.`class_name` = :class';
            $parameters[':class'] = $filters['class'];
        }

        // Build condition
        $condition = implode(' ', $joins);
        if ($condition) {
            $condition .= ' WHERE ';
        }

        if (count($conditions) === 0) {
            $conditions[] = '1';
        }

        $condition .= implode(' AND ', array_map(
            function ($condition): string {
                return "({$condition})";
            },
            $conditions
        ));

        return [
            $condition,
            $parameters,
        ];
    }
}