Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
13.43% covered (danger)
13.43%
9 / 67
0.00% covered (danger)
0.00%
0 / 5
CRAP
0.00% covered (danger)
0.00%
0 / 1
AssemblyExporterService
13.43% covered (danger)
13.43%
9 / 67
0.00% covered (danger)
0.00%
0 / 5
204.48
0.00% covered (danger)
0.00%
0 / 1
 __construct
56.25% covered (warning)
56.25%
9 / 16
0.00% covered (danger)
0.00%
0 / 1
3.75
 exportCompiledDelivery
0.00% covered (danger)
0.00%
0 / 14
0.00% covered (danger)
0.00%
0 / 1
20
 doExportCompiledDelivery
0.00% covered (danger)
0.00%
0 / 30
0.00% covered (danger)
0.00%
0 / 1
42
 setupCompiledTestConverter
0.00% covered (danger)
0.00%
0 / 4
0.00% covered (danger)
0.00%
0 / 1
6
 getAssemblyFilesReader
0.00% covered (danger)
0.00%
0 / 3
0.00% covered (danger)
0.00%
0 / 1
6
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) 2019  (original work) Open Assessment Technologies SA;
19 */
20
21namespace oat\taoDeliveryRdf\model\export;
22
23use oat\taoDeliveryRdf\model\assembly\CompiledTestConverterFactory;
24use ZipArchive;
25use Exception;
26use InvalidArgumentException;
27use common_Exception;
28use core_kernel_classes_EmptyProperty;
29use tao_helpers_Display;
30use tao_helpers_Export;
31use tao_helpers_File;
32use core_kernel_classes_Resource;
33use common_ext_ExtensionsManager;
34use tao_models_classes_service_ServiceCall;
35use tao_models_classes_export_RdfExporter;
36use oat\generis\model\OntologyAwareTrait;
37use oat\oatbox\log\LoggerAwareTrait;
38use oat\oatbox\service\ConfigurableService;
39use oat\tao\model\service\ServiceFileStorage;
40use oat\taoDeliveryRdf\model\assembly\AssemblyFilesReaderInterface;
41use oat\taoDeliveryRdf\model\assembly\UnsupportedCompiledTestFormatException;
42use oat\taoDeliveryRdf\model\DeliveryAssemblyService;
43
44class AssemblyExporterService extends ConfigurableService
45{
46    use LoggerAwareTrait;
47    use OntologyAwareTrait;
48
49
50    public const SERVICE_ID = 'taoDeliveryRdf/AssemblyExporterService';
51    public const OPTION_ASSEMBLY_FILES_READER = 'assembly_files_reader';
52    public const OPTION_RDF_EXPORTER = 'rdf_exporter';
53    public const MANIFEST_FILENAME = 'manifest.json';
54    public const DELIVERY_RDF_FILENAME = 'delivery.rdf';
55    /**
56         * @var AssemblyFilesReaderInterface
57         */
58    private $assemblyFilesReader;
59    /**
60         * @var tao_models_classes_export_RdfExporter
61         */
62    private $rdfExporter;
63    /**
64         * AssemblyExporterService constructor.
65         * @param array $options
66         */
67    public function __construct($options = [])
68    {
69        parent::__construct($options);
70        if (!$this->getOption(self::OPTION_ASSEMBLY_FILES_READER) instanceof AssemblyFilesReaderInterface) {
71            throw new InvalidArgumentException(
72                sprintf(
73                    '%s option value must be an instance of %s',
74                    self::OPTION_ASSEMBLY_FILES_READER,
75                    AssemblyFilesReaderInterface::class
76                )
77            );
78        }
79
80        $this->rdfExporter = $this->getOption(self::OPTION_RDF_EXPORTER);
81        if (!$this->rdfExporter instanceof tao_models_classes_export_RdfExporter) {
82            throw new InvalidArgumentException(
83                '%s option value must be an instance of %s',
84                self::OPTION_RDF_EXPORTER,
85                tao_models_classes_export_RdfExporter::class
86            );
87        }
88    }
89
90    /**
91     * Export Compiled Delivery
92     *
93     * Exports a delivery into its compiled form. In case of the $fsExportPath argument is set,
94     * the compiled delivery will be stored in the 'taoDelivery' shared file system, at $fsExportPath location.
95     *
96     * @param core_kernel_classes_Resource $compiledDelivery
97     * @param string $outputTestFormat Format compiled test file in output assembly package.
98     *
99     * @return string The path to the compiled delivery on the local file system OR the 'taoDelivery' shared file
100     *                system, depending on whether $fsExportPath is set.
101     *
102     * @throws common_Exception
103     * @throws core_kernel_classes_EmptyProperty
104     */
105    public function exportCompiledDelivery(core_kernel_classes_Resource $compiledDelivery, $outputTestFormat)
106    {
107        $this->logDebug("Exporting Delivery Assembly '" . $compiledDelivery->getUri() . "'...");
108        $fileName = tao_helpers_Display::textCleaner($compiledDelivery->getLabel()) . '.zip';
109        $path = tao_helpers_File::concat([tao_helpers_Export::getExportPath(), $fileName]);
110        if (!tao_helpers_File::securityCheck($path, true)) {
111            throw new Exception('Unauthorized file name');
112        }
113
114        // If such a target zip file exists, remove it from local filesystem. It prevents some synchronicity issues
115        // to occur while dealing with ZIP Archives (not explained yet).
116        if (file_exists($path)) {
117            unlink($path);
118        }
119
120        $zipArchive = new ZipArchive();
121        if ($zipArchive->open($path, ZipArchive::CREATE) !== true) {
122            throw new Exception('Unable to create archive at ' . $path);
123        }
124
125        $this->setupCompiledTestConverter($outputTestFormat);
126        $this->doExportCompiledDelivery($path, $compiledDelivery, $zipArchive);
127        $zipArchive->close();
128        return $path;
129    }
130
131    /**
132     * Do Export Compiled Delivery
133     *
134     * Method containing the main behavior of exporting a compiled delivery into a ZIP archive.
135     *
136     * For developers wanting to override this method, the following information has to be taken into account:
137     *
138     * - The value of the $zipArgive argument is an already open ZipArchive object.
139     * - The method must keep the archive open after its execution (calling code will take care of it).
140     *
141     * @param $path
142     * @param core_kernel_classes_Resource $compiledDelivery
143     * @param ZipArchive $zipArchive
144     * @throws common_Exception
145     * @throws core_kernel_classes_EmptyProperty
146     */
147    protected function doExportCompiledDelivery(
148        $path,
149        core_kernel_classes_Resource $compiledDelivery,
150        ZipArchive $zipArchive
151    ) {
152        $taoDeliveryVersion = common_ext_ExtensionsManager::singleton()->getInstalledVersion('taoDelivery');
153        $data = [
154            'dir' => [],
155            'label' => $compiledDelivery->getLabel(),
156            'version' => $taoDeliveryVersion
157        ];
158        $directories = $compiledDelivery->getPropertyValues(
159            $this->getProperty(DeliveryAssemblyService::PROPERTY_DELIVERY_DIRECTORY)
160        );
161        foreach ($directories as $id) {
162            $directory = $this->getServiceLocator()->get(ServiceFileStorage::SERVICE_ID)->getDirectoryById($id);
163            foreach ($this->getAssemblyFilesReader()->getFiles($directory) as $filePath => $fileStream) {
164                tao_helpers_File::addFilesToZip($zipArchive, $fileStream, $filePath);
165            }
166            $data['dir'][$id] = $directory->getPrefix();
167        }
168
169        $runtime = $compiledDelivery->getOnePropertyValue(
170            $this->getProperty(DeliveryAssemblyService::PROPERTY_DELIVERY_RUNTIME)
171        );
172        $serviceCall = $runtime instanceof core_kernel_classes_Resource
173            ? tao_models_classes_service_ServiceCall::fromResource($runtime)
174            : tao_models_classes_service_ServiceCall::fromString((string)$runtime);
175        $data['runtime'] = base64_encode(json_encode($serviceCall));
176        $rdfData = $this->rdfExporter->getRdfString([$compiledDelivery]);
177        if (!$zipArchive->addFromString(self::DELIVERY_RDF_FILENAME, $rdfData)) {
178            throw new common_Exception('Unable to add metadata to exported delivery assembly');
179        }
180        $data['meta'] = self::DELIVERY_RDF_FILENAME;
181        $content = json_encode($data);
182        if (!$zipArchive->addFromString(self::MANIFEST_FILENAME, $content)) {
183            $zipArchive->close();
184            unlink($path);
185            throw new common_Exception('Unable to add manifest to exported delivery assembly');
186        }
187    }
188
189    /**
190     * @param string $outputTestFormat
191     * @return void
192     *
193     * @throws UnsupportedCompiledTestFormatException
194     */
195    private function setupCompiledTestConverter($outputTestFormat)
196    {
197        /** @var CompiledTestConverterFactory $compiledTestConverterFactory */
198        $compiledTestConverterFactory = $this->getServiceLocator()->get(CompiledTestConverterFactory::class);
199        $converter = $compiledTestConverterFactory->createConverter($outputTestFormat);
200
201        if ($converter) {
202            $this->getAssemblyFilesReader()->setCompiledTestConverter($converter);
203        }
204    }
205
206    /**
207     * @return AssemblyFilesReaderInterface
208     */
209    private function getAssemblyFilesReader()
210    {
211        if (!$this->assemblyFilesReader instanceof AssemblyFilesReaderInterface) {
212            $this->assemblyFilesReader = $this->getOption(self::OPTION_ASSEMBLY_FILES_READER);
213        }
214
215        return $this->assemblyFilesReader;
216    }
217}