tao-test/app/taoQtiItem/model/qti/Parser.php

190 lines
6.0 KiB
PHP

<?php
/*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; under version 2
* of the License (non-upgradable).
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* Copyright (c) 2013 (original work) Open Assessment Technologies SA (under the project TAO-PRODUCT);
*
*
*/
namespace oat\taoQtiItem\model\qti;
use oat\oatbox\service\ServiceManager;
use oat\taoQtiItem\model\qti\ParserFactory;
use oat\taoQtiItem\model\qti\exception\UnsupportedQtiElement;
use oat\taoQtiItem\model\ValidationService;
use \tao_models_classes_Parser;
use \DOMDocument;
use \tao_helpers_Request;
use oat\oatbox\log\LoggerAwareTrait;
/**
* The QTI Parser enables you to parse QTI item xml files and build the
* objects
*
* @access public
* @author Jerome Bogaerts, <jerome.bogaerts@tudor.lu>
* @package taoQTI
* @see http://www.imsglobal.org/question/qti_v2p0/imsqti_infov2p0.html#element10010
*/
class Parser extends tao_models_classes_Parser
{
use LoggerAwareTrait;
/**
* Run the validation process
*
* @access public
* @author Jerome Bogaerts, <jerome.bogaerts@tudor.lu>
*
* @param string $schema
*
* @return bool
* @throws \Exception
* @throws \common_Exception
*/
public function validate($schema = '')
{
if (empty($schema)) {
// Let's detect NS in use...
$dom = new DOMDocument('1.0', 'UTF-8');
switch ($this->sourceType) {
case self::SOURCE_FLYFILE:
if ($this->source->exists()) {
$dom->load($this->source->read());
}
break;
case self::SOURCE_FILE:
$dom->load($this->source);
break;
case self::SOURCE_URL:
$xmlContent = tao_helpers_Request::load($this->source, true);
$dom->loadXML($xmlContent);
break;
case self::SOURCE_STRING:
$dom->loadXML($this->source);
break;
}
// Retrieve Root's namespace.
if ($dom->documentElement == null) {
$this->addError('dom is null and could not be validate');
$returnValue = false;
} else {
$ns = $dom->documentElement->lookupNamespaceUri(null);
$servicemanager = $this->getServiceManager();
$validationService = $servicemanager->get(ValidationService::SERVICE_ID);
$schemas = $validationService->getContentValidationSchema($ns);
$this->logDebug("The following schema will be used to validate: '" . $schemas[0] . "'.");
$validSchema = $this->validateMultiple($schemas);
$returnValue = $validSchema !== '';
}
} elseif (!file_exists($schema)) {
throw new \common_Exception('no schema found in the location ' . $schema);
} else {
$returnValue = parent::validate($schema);
}
return (bool) $returnValue;
}
/**
* load the file content, parse it and build the a QTI_Item instance
*
* @access public
* @author Jerome Bogaerts, <jerome.bogaerts@tudor.lu>
* @param boolean resolveXInclude
* @return \oat\taoQtiItem\model\qti\Item
*/
public function load($resolveXInclude = false)
{
$returnValue = null;
if (!$this->valid) {
libxml_use_internal_errors(true); //retrieve errors if no validation has been done previously
}
//load it using the DOMDocument library
$xml = new DOMDocument();
switch ($this->sourceType) {
case self::SOURCE_FLYFILE:
if ($this->source->exists()) {
$xml->load($this->source->read());
}
break;
case self::SOURCE_FILE:
$xml->load($this->source);
break;
case self::SOURCE_URL:
$xmlContent = tao_helpers_Request::load($this->source, true);
$xml->loadXML($xmlContent);
break;
case self::SOURCE_STRING:
$xml->loadXML($this->source);
break;
}
if ($xml !== false) {
$basePath = '';
if ($this->sourceType == self::SOURCE_FILE || $this->sourceType == self::SOURCE_URL) {
$basePath = dirname($this->source) . '/';
}
//build the item from the xml
$parserFactory = new ParserFactory($xml, $basePath);
try {
$returnValue = $parserFactory->load();
} catch (UnsupportedQtiElement $e) {
$this->addError($e);
}
if (!$this->valid) {
$this->valid = true;
libxml_clear_errors();
}
} elseif (!$this->valid) {
$this->addErrors(libxml_get_errors());
libxml_clear_errors();
}
return $returnValue;
}
protected function addError($error)
{
$this->valid = false;
if ($error instanceof UnsupportedQtiElement) {
$this->errors[] = [
'message' => '[Unsupported Qti Type] ' . $error->getUserMessage()
];
} else {
parent::addError($error);
}
}
protected function getServiceManager()
{
return ServiceManager::getServiceManager();
}
}