Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
2.88% covered (danger)
2.88%
3 / 104
13.33% covered (danger)
13.33%
2 / 15
CRAP
0.00% covered (danger)
0.00%
0 / 1
LocalItemSource
2.88% covered (danger)
2.88%
3 / 104
13.33% covered (danger)
13.33%
2 / 15
1290.91
0.00% covered (danger)
0.00%
0 / 1
 __construct
100.00% covered (success)
100.00%
2 / 2
100.00% covered (success)
100.00%
1 / 1
3
 getItem
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 getDirectories
0.00% covered (danger)
0.00%
0 / 6
0.00% covered (danger)
0.00%
0 / 1
2
 getDirectory
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 getFileInfo
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 getInfoFromSource
0.00% covered (danger)
0.00%
0 / 10
0.00% covered (danger)
0.00%
0 / 1
6
 getInfoFromFile
0.00% covered (danger)
0.00%
0 / 8
0.00% covered (danger)
0.00%
0 / 1
2
 download
0.00% covered (danger)
0.00%
0 / 11
0.00% covered (danger)
0.00%
0 / 1
12
 getBaseName
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 add
0.00% covered (danger)
0.00%
0 / 11
0.00% covered (danger)
0.00%
0 / 1
30
 delete
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 getFileStream
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 getFile
0.00% covered (danger)
0.00%
0 / 6
0.00% covered (danger)
0.00%
0 / 1
12
 getItemDirectory
0.00% covered (danger)
0.00%
0 / 3
0.00% covered (danger)
0.00%
0 / 1
6
 searchDirectories
0.00% covered (danger)
0.00%
0 / 41
0.00% covered (danger)
0.00%
0 / 1
132
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) 2014 (original work) Open Assessment Technologies SA;
19 *
20 *
21 */
22
23namespace oat\taoItems\model\media;
24
25use common_exception_Error;
26use oat\oatbox\filesystem\Directory;
27use oat\oatbox\filesystem\File;
28use oat\tao\model\media\MediaManagement;
29use oat\tao\model\media\mediaSource\DirectorySearchQuery;
30use Psr\Http\Message\StreamInterface;
31use tao_helpers_File;
32use taoItems_models_classes_ItemsService;
33
34/**
35 * This media source gives access to files that are part of the item
36 * and are addressed in a relative way
37 */
38class LocalItemSource implements MediaManagement
39{
40    private $item;
41
42    private $lang;
43
44    protected $itemService;
45
46    public function __construct($data)
47    {
48        $this->item = (isset($data['item'])) ? $data['item'] : null;
49        $this->lang = (isset($data['lang'])) ? $data['lang'] : '';
50    }
51
52    /**
53     * @return \core_kernel_classes_Resource
54     */
55    public function getItem()
56    {
57        return $this->item;
58    }
59
60    /**
61     * @throws \common_Exception
62     * @throws \tao_models_classes_FileNotFoundException
63     * @throws common_exception_Error
64     */
65    public function getDirectories(DirectorySearchQuery $params): array
66    {
67        return $this->searchDirectories(
68            $params->getParentLink(),
69            $params->getFilter(),
70            $params->getDepth(),
71            $params->getChildrenLimit()
72        );
73    }
74
75    /**
76     * Get a array representing the tree of directory
77     * @deprecated  in favor of MediaBrowser::getDirectories
78     * @throws \common_Exception
79     * @throws \tao_models_classes_FileNotFoundException
80     * @throws common_exception_Error
81     */
82    public function getDirectory($parentLink = '', $acceptableMime = [], $depth = 1)
83    {
84        return $this->searchDirectories($parentLink, $acceptableMime, $depth, 0);
85    }
86
87    /**
88     * Return file info regarding a file
89     *
90     * @see \oat\tao\model\media\MediaBrowser::getFileInfo
91     * @param string $link
92     * @return array
93     * @throws \tao_models_classes_FileNotFoundException
94     * @throws common_exception_Error
95     */
96    public function getFileInfo($link)
97    {
98        return $this->getInfoFromFile($this->getFile($link));
99    }
100
101    /**
102     * Method should be used only after uploading local file, when metadata like file size and mime type
103     * of remote uploaded file are not critical. To get more precise info about remote file use getInfoFromFile().
104     *
105     * In case of using S3 Flysystem adapter there might be latency and file cannot be accessed right after upload.
106     *
107     * @param File $file
108     * @param string $sourceFile
109     * @return array
110     * @throws \common_Exception
111     * @throws \tao_models_classes_FileNotFoundException
112     */
113    protected function getInfoFromSource(File $file, string $sourceFile): array
114    {
115        if (file_exists($sourceFile)) {
116            $link = $this->getItemDirectory()->getRelPath($file);
117            return [
118                'name'     => $file->getBasename(),
119                'uri'      => $link,
120                'mime'     => tao_helpers_File::getMimeType($sourceFile),
121                'filePath' => $link,
122                'size'     => filesize($sourceFile),
123            ];
124        } else {
125            return $this->getInfoFromFile($file);
126        }
127    }
128
129    protected function getInfoFromFile(File $file)
130    {
131        $link = $this->getItemDirectory()->getRelPath($file);
132        return [
133            'name'     => $file->getBasename(),
134            'uri'      => $link,
135            'mime'     => $file->getMimeType(),
136            'filePath' => $link,
137            'size'     => $file->getSize(),
138        ];
139    }
140
141    /**
142     * Copy file content to temp file. Path is return to be downloaded
143     *
144     * @see \oat\tao\model\media\MediaBrowser::download
145     * @param string $link
146     * @return string
147     * @throws \common_Exception
148     * @throws \tao_models_classes_FileNotFoundException
149     * @throws common_exception_Error
150     */
151    public function download($link)
152    {
153        $file = $this->getFile($link);
154
155        $relPath = $this->getItemDirectory()->getRelPath($file);
156
157        $tmpDir = rtrim(\tao_helpers_File::createTempDir(), '/');
158        $tmpFile = $tmpDir . $relPath;
159        if (! is_dir(dirname($tmpFile))) {
160            mkdir(dirname($tmpFile), 0755, true);
161        }
162
163        if (($resource = fopen($tmpFile, 'w')) !== false) {
164            stream_copy_to_stream($file->readStream(), $resource);
165            fclose($resource);
166            return $tmpFile;
167        }
168
169        throw new \common_Exception('Unable to write "' . $link . '" into tmp folder("' . $tmpFile . '").');
170    }
171
172    /**
173     * Return filename of given path
174     *
175     * @see \oat\tao\model\media\MediaBrowser::getBaseName
176     * @param string $link
177     * @return string
178     */
179    public function getBaseName($link)
180    {
181        return basename($link);
182    }
183
184    /**
185     * Add content to file
186     *
187     * @see \oat\tao\model\media\MediaManagement::add
188     * @param string $source
189     * @param string $fileName
190     * @param string $parent
191     * @return array
192     * @throws \common_Exception
193     * @throws \tao_models_classes_FileNotFoundException
194     */
195    public function add($source, $fileName, $parent)
196    {
197        if (! \tao_helpers_File::securityCheck($fileName, true)) {
198            throw new \common_Exception('Unsecured filename "' . $fileName . '"');
199        }
200
201        if (($resource = fopen($source, 'r')) === false) {
202            throw new \common_Exception('Unable to read content of file ("' . $source . '")');
203        }
204
205        $file = $this->getItemDirectory()->getDirectory($parent)->getFile($fileName);
206        $writeSuccess = $file->put($resource);
207        if (is_resource($resource)) {
208            fclose($resource);
209        }
210        if (! $writeSuccess) {
211            throw new \common_Exception('Unable to write file ("' . $fileName . '")');
212        }
213
214        return $this->getInfoFromSource($file, $source);
215    }
216
217    /**
218     * Delete the file located at $link
219     *
220     * @see \oat\tao\model\media\MediaManagement::delete
221     * @param $link
222     * @return bool
223     * @throws \tao_models_classes_FileNotFoundException
224     * @throws common_exception_Error
225     */
226    public function delete($link)
227    {
228        return $this->getFile($link)->delete();
229    }
230
231    /**
232     * Return file stream of file located at $link
233     *
234     * @see tao/models/classes/media/MediaBrowser.php:getFileStream
235     * @param string $link
236     * @return StreamInterface
237     * @throws \tao_models_classes_FileNotFoundException
238     * @throws common_exception_Error
239     */
240    public function getFileStream($link)
241    {
242        return $this->getFile($link)->readPsrStream();
243    }
244
245    /**
246     * Get file object associated to $link, search in main item directory or specific item directory
247     *
248     * @param $link
249     * @return File
250     * @throws \common_Exception
251     * @throws \tao_models_classes_FileNotFoundException
252     * @throws common_exception_Error
253     */
254    private function getFile($link)
255    {
256        if (!tao_helpers_File::securityCheck($link)) {
257            throw new common_exception_Error(__('Your path contains error'));
258        }
259
260        $file = $this->getItemDirectory()->getFile($link);
261        if ($file->exists()) {
262            return $file;
263        }
264        throw new \tao_models_classes_FileNotFoundException($link);
265    }
266
267    /**
268     * Get item directory based on $this->item & $this->lang
269     *
270     * @return Directory
271     * @throws \common_Exception
272     */
273    protected function getItemDirectory()
274    {
275        if (! $this->itemService) {
276            $this->itemService = taoItems_models_classes_ItemsService::singleton();
277        }
278        return $this->itemService->getItemDirectory($this->item, $this->lang);
279    }
280
281    /**
282     * @throws \common_Exception
283     * @throws \tao_models_classes_FileNotFoundException
284     * @throws common_exception_Error
285     */
286    private function searchDirectories(
287        string $parentLink,
288        array $acceptableMime,
289        int $depth,
290        int $childrenLimit
291    ): array {
292        if (!tao_helpers_File::securityCheck($parentLink)) {
293            throw new common_exception_Error(__('Your path contains error'));
294        }
295
296        $label = rtrim($parentLink, '/');
297        if (strrpos($parentLink, '/') !== false && substr($parentLink, -1) !== '/') {
298            $label = substr($parentLink, strrpos($parentLink, '/') + 1);
299            $parentLink = $parentLink . '/';
300        }
301
302        if (in_array($parentLink, ['', '/'])) {
303            $label = $this->getItem()->getLabel();
304            $parentLink = '/';
305        }
306
307        $data = [
308            'path' => $parentLink,
309            'label' => $label,
310            'childrenLimit' => $childrenLimit,
311            'total' => 0,
312        ];
313
314        if ($depth <= 0) {
315            $data['parent'] = $parentLink;
316            return $data;
317        }
318
319        $children = [];
320
321        /** @var Directory $directory */
322        $itemDirectory = $this->getItemDirectory();
323        if ($parentLink != '/') {
324            $directory = $itemDirectory->getDirectory($parentLink);
325        } else {
326            $directory = $itemDirectory;
327        }
328
329        $iterator = $directory->getFlyIterator();
330        $total = 0;
331        foreach ($iterator as $content) {
332            if ($content instanceof Directory) {
333                $children[] = $this->searchDirectories(
334                    $itemDirectory->getRelPath($content),
335                    $acceptableMime,
336                    $depth - 1,
337                    $childrenLimit
338                );
339                continue;
340            }
341
342            $fileInfo = $this->getInfoFromFile($content);
343            if (empty($acceptableMime) || in_array($fileInfo['mime'], $acceptableMime)) {
344                $children[] = $fileInfo;
345                $total++;
346            }
347        }
348
349        $data['children'] = $children;
350        $data['total'] = $total;
351
352        return $data;
353    }
354}