Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
91.84% covered (success)
91.84%
45 / 49
75.00% covered (warning)
75.00%
6 / 8
CRAP
0.00% covered (danger)
0.00%
0 / 1
SecureResourceCachedService
91.84% covered (success)
91.84%
45 / 49
75.00% covered (warning)
75.00%
6 / 8
19.20
0.00% covered (danger)
0.00%
0 / 1
 __construct
100.00% covered (success)
100.00%
5 / 5
100.00% covered (success)
100.00%
1 / 1
1
 getAllChildren
100.00% covered (success)
100.00%
8 / 8
100.00% covered (success)
100.00%
1 / 1
3
 validatePermissions
100.00% covered (success)
100.00%
2 / 2
100.00% covered (success)
100.00%
1 / 1
2
 validatePermission
78.57% covered (warning)
78.57%
11 / 14
0.00% covered (danger)
0.00%
0 / 1
6.35
 getService
100.00% covered (success)
100.00%
2 / 2
100.00% covered (success)
100.00%
1 / 1
1
 addToCache
100.00% covered (success)
100.00%
7 / 7
100.00% covered (success)
100.00%
1 / 1
2
 getCache
80.00% covered (warning)
80.00%
4 / 5
0.00% covered (danger)
0.00%
0 / 1
2.03
 getUser
100.00% covered (success)
100.00%
6 / 6
100.00% covered (success)
100.00%
1 / 1
2
1<?php
2
3/**
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License
6 * as published by the Free Software Foundation; under version 2
7 * of the License (non-upgradable).
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
17 *
18 * Copyright (c) 2013-2020   (original work) Open Assessment Technologies SA (under the project TAO-PRODUCT);
19 *
20 */
21
22declare(strict_types=1);
23
24namespace oat\tao\model\resources;
25
26use common_cache_Cache;
27use common_cache_NotFoundException;
28use common_exception_Error;
29use core_kernel_classes_Class;
30use core_kernel_classes_Resource;
31use oat\oatbox\session\SessionService;
32use oat\oatbox\user\User;
33use oat\tao\model\service\InjectionAwareService;
34
35class SecureResourceCachedService extends InjectionAwareService implements SecureResourceServiceInterface
36{
37    /** @var User */
38    private $user;
39    /** @var SecureResourceService */
40    private $service;
41    /** @var string */
42    private $cacheServiceId;
43    /** @var int|null */
44    private $ttl;
45    /** @var common_cache_Cache */
46    private $cache;
47    /** @var ValidatePermissionsCacheKeyFactory */
48    private $validatePermissionsCacheKeyFactory;
49    /** @var GetAllChildrenCacheKeyFactory */
50    private $getAllChildrenCacheKeyFactory;
51
52    /**
53     * @noinspection MagicMethodsValidityInspection
54     * @noinspection PhpMissingParentConstructorInspection
55     *
56     * @param SecureResourceService              $service
57     * @param ValidatePermissionsCacheKeyFactory $validatePermissionsCacheKeyFactory
58     * @param GetAllChildrenCacheKeyFactory      $getAllChildrenCacheKeyFactory
59     * @param string                             $cacheServiceId
60     * @param int                                $ttl
61     */
62    public function __construct(
63        SecureResourceService $service,
64        ValidatePermissionsCacheKeyFactory $validatePermissionsCacheKeyFactory,
65        GetAllChildrenCacheKeyFactory $getAllChildrenCacheKeyFactory,
66        string $cacheServiceId,
67        ?int $ttl
68    ) {
69        $this->service = $service;
70        $this->cacheServiceId = $cacheServiceId;
71        $this->ttl = $ttl;
72        $this->validatePermissionsCacheKeyFactory = $validatePermissionsCacheKeyFactory;
73        $this->getAllChildrenCacheKeyFactory = $getAllChildrenCacheKeyFactory;
74    }
75
76    /**
77     * @inheritDoc
78     *
79     * @throws common_exception_Error
80     * @throws common_cache_NotFoundException
81     */
82    public function getAllChildren(core_kernel_classes_Class $resource): array
83    {
84        $user = $this->getUser();
85
86        $cacheKey = $this->getAllChildrenCacheKeyFactory->create($resource, $user);
87
88        $cache = $this->getCache();
89        if ($cache && $cache->has($cacheKey)) {
90            return $cache->get($cacheKey)->getInstances();
91        }
92
93        $accessibleInstances = $this->getService()->getAllChildren($resource);
94
95        $this->addToCache($cacheKey, new SecureResourceServiceAllChildrenCacheCollection($accessibleInstances));
96
97        return $accessibleInstances;
98    }
99
100    /**
101     * @inheritDoc
102     *
103     * @throws common_exception_Error
104     * @throws common_cache_NotFoundException
105     */
106    public function validatePermissions(iterable $resources, array $permissionsToCheck): void
107    {
108        foreach ($resources as $resource) {
109            $this->validatePermission($resource, $permissionsToCheck);
110        }
111    }
112
113    /**
114     * @inheritDoc
115     *
116     * @throws common_exception_Error
117     * @throws common_cache_NotFoundException
118     */
119    public function validatePermission($resource, array $permissionsToCheck): void
120    {
121        $user = $this->getUser();
122
123        $resourceUri = $resource instanceof core_kernel_classes_Resource ? $resource->getUri() : $resource;
124
125        $cacheKey = $this->validatePermissionsCacheKeyFactory->create($resourceUri, $user);
126
127        $cache = $this->getCache();
128        if ($cache && $cache->has($cacheKey)) {
129            $hasAccess = $cache->get($cacheKey);
130
131            if (!$hasAccess) {
132                throw new ResourceAccessDeniedException($resourceUri);
133            }
134
135            return;
136        }
137
138        try {
139            $this->getService()->validatePermission($resource, $permissionsToCheck);
140        } catch (ResourceAccessDeniedException $e) {
141            $this->addToCache($cacheKey, false);
142
143            throw $e;
144        }
145
146        $this->addToCache($cacheKey, true);
147    }
148
149    /**
150     * @return SecureResourceService
151     */
152    public function getService(): SecureResourceServiceInterface
153    {
154        $this->service->setServiceLocator($this->getServiceLocator());
155
156        return $this->service;
157    }
158
159    private function addToCache(string $cacheKey, $data)
160    {
161        $cache = $this->getCache();
162
163        if ($cache) {
164            $cache->put(
165                $data,
166                $cacheKey,
167                $this->ttl
168            );
169        }
170    }
171
172    private function getCache(): ?common_cache_Cache
173    {
174        $isCacheEnabled = !empty(trim($this->cacheServiceId));
175
176        if (!$isCacheEnabled) {
177            return null;
178        }
179
180        $this->cache = $this->getServiceLocator()->get($this->cacheServiceId);
181
182        return $this->cache;
183    }
184
185    /**
186     * @return User
187     *
188     * @throws common_exception_Error
189     */
190    private function getUser(): User
191    {
192        if ($this->user === null) {
193            $this->user = $this
194                ->getServiceLocator()
195                ->get(SessionService::SERVICE_ID)
196                ->getCurrentUser();
197        }
198
199        return $this->user;
200    }
201}