Skip to content
Snippets Groups Projects
Commit ffcee5fb authored by srosse's avatar srosse
Browse files

Merge OpenOLAT 10.0 to OpenOLAT default branch with 5c3556b2a97a7b9f717fb832801d8d4ea416b090

parents ee4cee63 7e734341
No related branches found
No related tags found
No related merge requests found
......@@ -21,7 +21,7 @@ encoding.same=Gleich wie Inhalt
height.auto=Automatisch
height.label=H\u00F6he Anzeigefl\u00E4che
inherit.label=Standardwerte übernehmen
inherit=Aus Lernressourcenverwaltung übernehmen
inherit=Aus Layouteinstellungen der Lernressource \u00FCbernehmen
custom=Anpassen
glossary.need.jQuery=Glossar braucht jQuery
automatic.need.js=Automatic braucht javascript
......
......@@ -21,7 +21,7 @@ automatic.need.js=Automatic needs javascript
height.auto=Automatic
height.label=Display height
iframe.content=Page content within iframe
inherit=Use from learning resource configuration
inherit=Use from learning resource layout configuration
inherit.label=Use standard configuration
mode=Display mode
mode.configured=Optimized for OpenOLAT
......
......@@ -31,6 +31,10 @@ import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import org.apache.commons.io.IOUtils;
import org.dom4j.Document;
import org.dom4j.DocumentType;
import org.dom4j.io.SAXReader;
import org.dom4j.io.SAXValidator;
import org.olat.core.logging.OLog;
import org.olat.core.logging.Tracing;
import org.olat.core.util.vfs.VFSLeaf;
......@@ -38,7 +42,6 @@ import org.olat.ims.resources.IMSEntityResolver;
import org.olat.search.service.document.file.utils.ShieldInputStream;
import org.xml.sax.Attributes;
import org.xml.sax.ErrorHandler;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.xml.sax.SAXParseException;
import org.xml.sax.XMLReader;
......@@ -114,6 +117,30 @@ public class ItemFileResourceValidator {
}
private boolean validateXml(InputStream in) {
boolean valid = false;
Document doc = readDocument(in);
if(doc != null) {
DocumentType docType = doc.getDocType();
if(docType == null) {
doc.addDocType("questestinterop", null, "ims_qtiasiv1p2p1.dtd");
}
valid = validateDocument(doc);
}
return valid;
}
private Document readDocument(InputStream in) {
try {
SAXReader reader = new SAXReader();
reader.setEntityResolver(new IMSEntityResolver());
reader.setValidation(false);
return reader.read(in, "");
} catch (Exception e) {
return null;
}
}
private boolean validateDocument(Document in) {
try {
SAXParserFactory factory = SAXParserFactory.newInstance();
factory.setValidating(true);
......@@ -127,7 +154,10 @@ public class ItemFileResourceValidator {
reader.setEntityResolver(new IMSEntityResolver());
reader.setErrorHandler(errorHandler);
reader.setContentHandler(contentHandler);
reader.parse(new InputSource(in));
SAXValidator validator = new SAXValidator(reader);
validator.validate(in);
return errorHandler.isValid() && contentHandler.isItem();
} catch (ParserConfigurationException e) {
return false;
......@@ -176,7 +206,7 @@ public class ItemFileResourceValidator {
if("questestinterop".equals(qName)) {
interop = true;
} else if("item".equals(localName)) {
} else if("item".equals(localName) || "item".equals(qName)) {
if(interop) {
item = true;
}
......
......@@ -636,11 +636,12 @@ class QTIImportProcessor {
}
private boolean processSidecarMetadata(QuestionItemImpl item, DocInfos docInfos) {
InputStream metadataIn = null;
try {
Path path = docInfos.root;
if(path != null) {
if(path != null && path.getFileName() != null) {
Path metadata = path.resolve(path.getFileName().toString() + "_metadata.xml");
InputStream metadataIn = Files.newInputStream(metadata);
metadataIn = Files.newInputStream(metadata);
SAXReader reader = new SAXReader();
Document document = reader.read(metadataIn);
Element rootElement = document.getRootElement();
......@@ -654,6 +655,8 @@ class QTIImportProcessor {
} catch (Exception e) {
log.error("", e);
return false;
} finally {
IOUtils.closeQuietly(metadataIn);
}
}
......
package org.olat.ims.qti.qpool;
import java.io.File;
import java.net.URISyntaxException;
import java.net.URL;
import junit.framework.Assert;
import org.junit.Test;
/**
*
* Initial date: 20.11.2014<br>
* @author srosse, stephane.rosse@frentix.com, http://www.frentix.com
*
*/
public class ItemFileResourceValidatorTest {
@Test
public void validate() throws URISyntaxException {
URL itemUrl = ItemFileResourceValidatorTest.class.getResource("fibi_i_001.xml");
Assert.assertNotNull(itemUrl);
File itemFile = new File(itemUrl.toURI());
ItemFileResourceValidator validator = new ItemFileResourceValidator();
boolean validate = validator.validate("fibi_i_001.xml", itemFile);
Assert.assertTrue(validate);
}
@Test
public void validate_missingDoctype() throws URISyntaxException {
URL itemUrl = ItemFileResourceValidatorTest.class.getResource("oo_item_without_doctype.xml");
Assert.assertNotNull(itemUrl);
File itemFile = new File(itemUrl.toURI());
ItemFileResourceValidator validator = new ItemFileResourceValidator();
boolean validate = validator.validate("oo_item_without_doctype.xml", itemFile);
Assert.assertTrue(validate);
}
@Test
public void validate_missingDoctype_invalid() throws URISyntaxException {
URL itemUrl = ItemFileResourceValidatorTest.class.getResource("oo_item_without_doctype_invalid.xml");
Assert.assertNotNull(itemUrl);
File itemFile = new File(itemUrl.toURI());
ItemFileResourceValidator validator = new ItemFileResourceValidator();
boolean validate = validator.validate("oo_item_without_doctype_invalid.xml", itemFile);
Assert.assertFalse(validate);
}
}
<questestinterop>
<item ident="QTIEDIT:MCQ:1000002286" title="New question">
<presentation label="New question">
<material>
<mattext texttype="text/html"><![CDATA[New question]]></mattext>
</material>
<response_lid ident="1000002288" rcardinality="Multiple" rtiming="No">
<render_choice shuffle="No" minnumber="0" maxnumber="1">
<flow_label class="List">
<response_label ident="1000002291" rshuffle="Yes">
<material>
<mattext texttype="text/html"><![CDATA[New answer]]></mattext>
</material>
</response_label>
</flow_label>
</render_choice>
</response_lid>
</presentation>
<resprocessing>
<outcomes>
<decvar varname="SCORE" vartype="Decimal" defaultval="0" minvalue="0.0" maxvalue="1.0" cutvalue="1.0"/>
</outcomes>
<respcondition title="Mastery" continue="Yes">
<conditionvar>
<and>
<varequal respident="1000002288" case="Yes">1000002291</varequal>
</and>
</conditionvar>
<setvar varname="SCORE" action="Set">1.0</setvar>
<displayfeedback feedbacktype="Response" linkrefid="Mastery"/>
</respcondition>
<respcondition title="_olat_resp_feedback" continue="Yes">
<conditionvar>
<varequal respident="1000002288" case="Yes">1000002291</varequal>
</conditionvar>
<displayfeedback feedbacktype="Response" linkrefid="1000002291"/>
</respcondition>
<respcondition title="Fail" continue="Yes">
<conditionvar>
<other/>
</conditionvar>
<setvar varname="SCORE" action="Set">0</setvar>
<displayfeedback feedbacktype="Response" linkrefid="Fail"/>
<displayfeedback feedbacktype="Solution" linkrefid="Solution"/>
<displayfeedback feedbacktype="Hint" linkrefid="Hint"/>
</respcondition>
</resprocessing>
</item>
</questestinterop>
<questestinterop>
<itemop ident="QTIEDIT:MCQ:1000002286" title="New question">
<presentation label="New question">
<material>
<mattext texttype="text/html"><![CDATA[New question]]></mattext>
</material>
<response_lid ident="1000002288" rcardinality="Multiple" rtiming="No">
<render_choice shuffle="No" minnumber="0" maxnumber="1">
<flow_label class="List">
<response_label ident="1000002291" rshuffle="Yes">
<material>
<mattext texttype="text/html"><![CDATA[New answer]]></mattext>
</material>
</response_label>
</flow_label>
</render_choice>
</response_lid>
</presentation>
<resprocessing>
<outcomes>
<decvar varname="SCORE" vartype="Decimal" defaultval="0" minvalue="0.0" maxvalue="1.0" cutvalue="1.0"/>
</outcomes>
<respcondition title="Mastery" continue="Yes">
<conditionvar>
<and>
<varequal respident="1000002288" case="Yes">1000002291</varequal>
</and>
</conditionvar>
<setvar varname="SCORE" action="Set">1.0</setvar>
<displayfeedback feedbacktype="Response" linkrefid="Mastery"/>
</respcondition>
<respcondition title="_olat_resp_feedback" continue="Yes">
<conditionvar>
<varequal respident="1000002288" case="Yes">1000002291</varequal>
</conditionvar>
<displayfeedback feedbacktype="Response" linkrefid="1000002291"/>
</respcondition>
<respcondition title="Fail" continue="Yes">
<conditionvar>
<other/>
</conditionvar>
<setvar varname="SCORE" action="Set">0</setvar>
<displayfeedback feedbacktype="Response" linkrefid="Fail"/>
<displayfeedback feedbacktype="Solution" linkrefid="Solution"/>
<displayfeedback feedbacktype="Hint" linkrefid="Hint"/>
</respcondition>
</resprocessing>
</itemop>
</questestinterop>
......@@ -145,6 +145,7 @@ import org.junit.runners.Suite;
org.olat.ims.qti.QTIResultManagerTest.class,
org.olat.ims.qti.qpool.QTIImportProcessorTest.class,
org.olat.ims.qti.qpool.QTIExportProcessorTest.class,
org.olat.ims.qti.qpool.ItemFileResourceValidatorTest.class,
org.olat.ims.qti.questionimport.CSVToQuestionConverterTest.class,
org.olat.ims.qti.statistics.manager.QTIStatisticsManagerLargeTest.class,
org.olat.ims.qti.statistics.manager.QTIStatisticsManagerTest.class,
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment