/** * <a href="http://www.openolat.org"> * OpenOLAT - Online Learning and Training</a><br> * <p> * Licensed under the Apache License, Version 2.0 (the "License"); <br> * you may not use this file except in compliance with the License.<br> * You may obtain a copy of the License at the * <a href="http://www.apache.org/licenses/LICENSE-2.0">Apache homepage</a> * <p> * Unless required by applicable law or agreed to in writing,<br> * software distributed under the License is distributed on an "AS IS" BASIS, <br> * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. <br> * See the License for the specific language governing permissions and <br> * limitations under the License. * <p> * Initial code contributed and copyrighted by<br> * frentix GmbH, http://www.frentix.com * <p> */ package org.olat.ims.qti.qpool; import java.io.File; import java.io.InputStream; import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.Locale; import java.util.Set; import java.util.zip.ZipOutputStream; import org.dom4j.Document; import org.dom4j.Element; import org.olat.core.commons.persistence.DB; import org.olat.core.gui.UserRequest; import org.olat.core.gui.control.Controller; import org.olat.core.gui.control.WindowControl; import org.olat.core.gui.media.MediaResource; import org.olat.core.gui.translator.Translator; import org.olat.core.helpers.Settings; import org.olat.core.id.Identity; import org.olat.core.id.OLATResourceable; import org.olat.core.logging.OLog; import org.olat.core.logging.Tracing; import org.olat.core.util.FileUtils; import org.olat.core.util.Util; import org.olat.core.util.i18n.I18nModule; import org.olat.core.util.vfs.VFSContainer; import org.olat.core.util.vfs.VFSItem; import org.olat.core.util.vfs.VFSLeaf; import org.olat.core.util.vfs.VFSManager; import org.olat.fileresource.FileResourceManager; import org.olat.ims.qti.QTI12EditorController; import org.olat.ims.qti.QTI12PreviewController; import org.olat.ims.qti.QTIConstants; import org.olat.ims.qti.editor.QTIEditHelper; import org.olat.ims.qti.editor.QTIEditorMainController; import org.olat.ims.qti.editor.QTIEditorPackageImpl; import org.olat.ims.qti.editor.beecom.objects.Item; import org.olat.ims.qti.editor.beecom.objects.Section; import org.olat.ims.qti.editor.beecom.parser.ParserManager; import org.olat.ims.qti.fileresource.TestFileResource; import org.olat.ims.qti.qpool.QTI12ItemFactory.Type; import org.olat.ims.qti.questionimport.ItemAndMetadata; import org.olat.ims.resources.IMSEntityResolver; import org.olat.ims.qti21.pool.QTI12And21PoolWordExport; import org.olat.modules.qpool.ExportFormatOptions; import org.olat.modules.qpool.ExportFormatOptions.Outcome; import org.olat.modules.qpool.QItemFactory; import org.olat.modules.qpool.QPoolSPI; import org.olat.modules.qpool.QuestionItem; import org.olat.modules.qpool.QuestionItemFull; import org.olat.modules.qpool.QuestionItemShort; import org.olat.modules.qpool.manager.QEducationalContextDAO; import org.olat.modules.qpool.manager.QItemTypeDAO; import org.olat.modules.qpool.manager.QLicenseDAO; import org.olat.modules.qpool.manager.QPoolFileStorage; import org.olat.modules.qpool.manager.QuestionItemDAO; import org.olat.modules.qpool.manager.TaxonomyLevelDAO; import org.olat.modules.qpool.model.DefaultExportFormat; import org.olat.modules.qpool.model.QuestionItemImpl; import org.olat.repository.RepositoryEntry; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.xml.sax.InputSource; import org.xml.sax.XMLReader; import org.xml.sax.helpers.XMLReaderFactory; /** * * Initial date: 21.02.2013<br> * @author srosse, stephane.rosse@frentix.com, http://www.frentix.com * */ @Service("qtiPoolServiceProvider") public class QTIQPoolServiceProvider implements QPoolSPI { private static final OLog log = Tracing.createLoggerFor(QTIQPoolServiceProvider.class); public static final String QTI_12_OO_TEST = "OpenOLAT Test"; @Autowired private DB dbInstance; @Autowired private QPoolFileStorage qpoolFileStorage; @Autowired private QLicenseDAO qLicenseDao; @Autowired private QItemTypeDAO qItemTypeDao; @Autowired private QuestionItemDAO questionItemDao; @Autowired private QEducationalContextDAO qEduContextDao; @Autowired private TaxonomyLevelDAO taxonomyLevelDao; private static final List<ExportFormatOptions> formats = new ArrayList<ExportFormatOptions>(2); static { formats.add(DefaultExportFormat.ZIP_EXPORT_FORMAT); formats.add(DefaultExportFormat.DOCX_EXPORT_FORMAT); formats.add(new DefaultExportFormat(QTIConstants.QTI_12_FORMAT, Outcome.download, null)); formats.add(new DefaultExportFormat(QTIConstants.QTI_12_FORMAT, Outcome.repository, TestFileResource.TYPE_NAME)); } public QTIQPoolServiceProvider() { // } @Override public int getPriority() { return 10; } @Override public String getFormat() { return QTIConstants.QTI_12_FORMAT; } @Override public List<ExportFormatOptions> getTestExportFormats() { return Collections.unmodifiableList(formats); } @Override public boolean isCompatible(String filename, File file) { boolean ok = new ItemFileResourceValidator().validate(filename, file); return ok; } @Override public boolean isCompatible(String filename, VFSLeaf file) { boolean ok = new ItemFileResourceValidator().validate(filename, file); return ok; } @Override public boolean isConversionPossible(QuestionItemShort question) { return false; } @Override public List<QItemFactory> getItemfactories() { List<QItemFactory> factories = new ArrayList<QItemFactory>(); factories.add(new QTI12ItemFactory(Type.sc)); factories.add(new QTI12ItemFactory(Type.mc)); factories.add(new QTI12ItemFactory(Type.kprim)); factories.add(new QTI12ItemFactory(Type.fib)); factories.add(new QTI12ItemFactory(Type.essay)); return factories; } @Override public String extractTextContent(QuestionItemFull item) { String content = null; if(item.getRootFilename() != null) { String dir = item.getDirectory(); VFSContainer container = qpoolFileStorage.getContainer(dir); VFSItem file = container.resolve(item.getRootFilename()); if(file instanceof VFSLeaf) { VFSLeaf leaf = (VFSLeaf)file; InputStream is = leaf.getInputStream(); QTI12SAXHandler handler = new QTI12SAXHandler(); try { XMLReader parser = XMLReaderFactory.createXMLReader(); parser.setContentHandler(handler); parser.setEntityResolver(new IMSEntityResolver()); parser.setFeature("http://xml.org/sax/features/validation", false); parser.parse(new InputSource(is)); } catch (Exception e) { log.error("", e); } finally { FileUtils.closeSafely(is); } return handler.toString(); } } return content; } @Override public List<QuestionItem> importItems(Identity owner, Locale defaultLocale, String filename, File file) { QTIImportProcessor processor = new QTIImportProcessor(owner, defaultLocale, filename, file, questionItemDao, qItemTypeDao, qEduContextDao, taxonomyLevelDao, qLicenseDao, qpoolFileStorage, dbInstance); return processor.process(); } public List<QuestionItem> importRepositoryEntry(Identity owner, RepositoryEntry repositoryEntry, Locale defaultLocale) { OLATResourceable ores = repositoryEntry.getOlatResource(); FileResourceManager frm = FileResourceManager.getInstance(); File testFile = frm.getFileResource(ores); List<QuestionItem> importedItem = importItems(owner, defaultLocale, testFile.getName(), testFile); if(importedItem != null && importedItem.size() > 0) { dbInstance.getCurrentEntityManager().flush(); } return importedItem; } public QuestionItem createItem(Identity owner, QTI12ItemFactory.Type type, String title, Locale defaultLocale) { Translator trans = Util.createPackageTranslator(QTIEditorMainController.class, defaultLocale); Item item; switch(type) { case sc: item = QTIEditHelper.createSCItem(trans); break; case mc: item = QTIEditHelper.createMCItem(trans); break; case kprim: item = QTIEditHelper.createKPRIMItem(trans); break; case fib: item = QTIEditHelper.createFIBItem(trans); break; case essay: item = QTIEditHelper.createEssayItem(trans); break; default: return null; } item.setLabel(title); item.setTitle(title); QTIImportProcessor processor = new QTIImportProcessor(owner, defaultLocale, questionItemDao, qItemTypeDao, qEduContextDao, taxonomyLevelDao, qLicenseDao, qpoolFileStorage, dbInstance); Document doc = QTIEditHelper.itemToXml(item); Element itemEl = (Element)doc.selectSingleNode("questestinterop/item"); QuestionItemImpl qitem = processor.processItem(itemEl, "", null, "OpenOLAT", Settings.getVersion(), null, null); //save to file System VFSContainer baseDir = qpoolFileStorage.getContainer(qitem.getDirectory()); VFSLeaf leaf = baseDir.createChildLeaf(qitem.getRootFilename()); QTIEditHelper.serialiazeDoc(doc, leaf); return qitem; } public QuestionItemImpl importBeecomItem(Identity owner, ItemAndMetadata itemAndMetadata, VFSContainer sourceDir, Locale defaultLocale) { QTIImportProcessor processor = new QTIImportProcessor(owner, defaultLocale, questionItemDao, qItemTypeDao, qEduContextDao, taxonomyLevelDao, qLicenseDao, qpoolFileStorage, dbInstance); String editor = null; String editorVersion = null; Item item = itemAndMetadata.getItem(); if(!item.isAlient()) { editor = "OpenOLAT"; editorVersion = Settings.getVersion(); } Document doc = QTIEditHelper.itemToXml(item); Element itemEl = (Element)doc.selectSingleNode("questestinterop/item"); QuestionItemImpl qitem = processor.processItem(itemEl, "", null, editor, editorVersion, null, itemAndMetadata); //save to file System VFSContainer baseDir = qpoolFileStorage.getContainer(qitem.getDirectory()); VFSLeaf leaf = baseDir.createChildLeaf(qitem.getRootFilename()); QTIEditHelper.serialiazeDoc(doc, leaf); //process materials if(sourceDir != null) { List<String> materials = processor.getMaterials(itemEl); //copy materials for(String material:materials) { VFSItem sourceItem = sourceDir.resolve(material); if(sourceItem instanceof VFSLeaf) { VFSLeaf targetItem = baseDir.createChildLeaf(material); VFSManager.copyContent((VFSLeaf)sourceItem, targetItem); } } } return qitem; } public List<QuestionItem> importBeecomItem(Identity owner, List<ItemAndMetadata> items, Locale defaultLocale) { int count = 0; List<QuestionItem> qItems = new ArrayList<>(items.size()); for(ItemAndMetadata item:items) { QuestionItem qItem = importBeecomItem(owner, item, null, defaultLocale); qItems.add(qItem); if(++count % 10 == 0) { dbInstance.commitAndCloseSession(); } } return qItems; } public void exportToEditorPackage(QTIEditorPackageImpl editorPackage, List<QuestionItemShort> items, boolean newTest) { VFSContainer editorContainer = editorPackage.getBaseDir(); List<Long> itemKeys = toKeys(items); List<QuestionItemFull> fullItems = questionItemDao.loadByIds(itemKeys); Section section = editorPackage.getQTIDocument().getAssessment().getSections().get(0); if(newTest) { //remove autogenerated question section.getItems().clear(); } QTIExportProcessor processor = new QTIExportProcessor(qpoolFileStorage); for(QuestionItemFull fullItem:fullItems) { Element itemEl = processor.exportToQTIEditor(fullItem, editorContainer); Item item = (Item)new ParserManager().parse(itemEl); item.setIdent(QTIEditHelper.generateNewIdent(item.getIdent())); section.getItems().add(item); } } private List<Long> toKeys(List<? extends QuestionItemShort> items) { List<Long> keys = new ArrayList<Long>(items.size()); for(QuestionItemShort item:items) { keys.add(item.getKey()); } return keys; } @Override public MediaResource exportTest(List<QuestionItemShort> items, ExportFormatOptions format, Locale locale) { if(QTIConstants.QTI_12_FORMAT.equals(format.getFormat())) { return new QTIExportTestResource("UTF-8", locale, items, this); } else if(DefaultExportFormat.DOCX_EXPORT_FORMAT.getFormat().equals(format.getFormat())) { return new QTI12And21PoolWordExport(items, I18nModule.getDefaultLocale(), "UTF-8", questionItemDao, qpoolFileStorage); } return null; } @Override public void exportItem(QuestionItemFull item, ZipOutputStream zout, Locale locale, Set<String> names) { QTIExportProcessor processor = new QTIExportProcessor(qpoolFileStorage); processor.process(item, zout, names); } public void assembleTest(List<QuestionItemShort> items, ZipOutputStream zout) { List<Long> itemKeys = new ArrayList<Long>(); for(QuestionItemShort item:items) { itemKeys.add(item.getKey()); } List<QuestionItemFull> fullItems = questionItemDao.loadByIds(itemKeys); QTIExportProcessor processor = new QTIExportProcessor(qpoolFileStorage); processor.assembleTest(fullItems, zout); } /** * Export to QTI editor an item from the pool. The ident of the item * is always regenerated as an UUID. * @param qitem * @param editorContainer * @return */ public Item exportToQTIEditor(QuestionItemShort qitem, VFSContainer editorContainer) { QTIExportProcessor processor = new QTIExportProcessor(qpoolFileStorage); QuestionItemFull fullItem = questionItemDao.loadById(qitem.getKey()); Element itemEl = processor.exportToQTIEditor(fullItem, editorContainer); Item exportedItem = (Item)new ParserManager().parse(itemEl); exportedItem.setIdent(QTIEditHelper.generateNewIdent(exportedItem.getIdent())); return exportedItem; } @Override public void copyItem(QuestionItemFull original, QuestionItemFull copy) { VFSContainer originalDir = qpoolFileStorage.getContainer(original.getDirectory()); VFSContainer copyDir = qpoolFileStorage.getContainer(copy.getDirectory()); VFSManager.copyContent(originalDir, copyDir); } @Override public QuestionItem convert(Identity identity, QuestionItemShort question, Locale locale) { return null; } @Override public Controller getPreviewController(UserRequest ureq, WindowControl wControl, QuestionItem item, boolean summary) { QTI12PreviewController previewCtrl = new QTI12PreviewController(ureq, wControl, item, summary); return previewCtrl; } @Override public boolean isTypeEditable() { return true; } @Override public Controller getEditableController(UserRequest ureq, WindowControl wControl, QuestionItem item) { QTI12EditorController previewCtrl = new QTI12EditorController(ureq, wControl, item); return previewCtrl; } }