/**
* <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.qti21.repository.handlers;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.net.URISyntaxException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Locale;
import org.olat.core.CoreSpringFactory;
import org.olat.core.commons.fullWebApp.LayoutMain3ColsController;
import org.olat.core.commons.persistence.DB;
import org.olat.core.commons.persistence.DBFactory;
import org.olat.core.gui.UserRequest;
import org.olat.core.gui.components.stack.TooledStackedPanel;
import org.olat.core.gui.control.Controller;
import org.olat.core.gui.control.WindowControl;
import org.olat.core.gui.control.generic.layout.MainLayoutController;
import org.olat.core.gui.control.generic.messages.MessageUIFactory;
import org.olat.core.gui.control.generic.wizard.StepsMainRunController;
import org.olat.core.gui.media.MediaResource;
import org.olat.core.gui.translator.Translator;
import org.olat.core.id.Identity;
import org.olat.core.id.OLATResourceable;
import org.olat.core.id.Roles;
import org.olat.core.logging.OLog;
import org.olat.core.logging.Tracing;
import org.olat.core.util.FileUtils;
import org.olat.core.util.PathUtils.YesMatcher;
import org.olat.core.util.Util;
import org.olat.core.util.coordinate.LockResult;
import org.olat.course.assessment.AssessmentMode;
import org.olat.course.assessment.manager.UserCourseInformationsManager;
import org.olat.fileresource.FileResourceManager;
import org.olat.fileresource.ZippedDirectoryMediaResource;
import org.olat.fileresource.types.FileResource;
import org.olat.fileresource.types.ImsQTI21Resource;
import org.olat.fileresource.types.ResourceEvaluation;
import org.olat.ims.qti.editor.QTIEditorPackage;
import org.olat.ims.qti.fileresource.TestFileResource;
import org.olat.ims.qti21.QTI21DeliveryOptions;
import org.olat.ims.qti21.QTI21Service;
import org.olat.ims.qti21.manager.AssessmentTestSessionDAO;
import org.olat.ims.qti21.model.IdentifierGenerator;
import org.olat.ims.qti21.model.QTI21QuestionType;
import org.olat.ims.qti21.model.xml.AssessmentItemFactory;
import org.olat.ims.qti21.model.xml.AssessmentTestFactory;
import org.olat.ims.qti21.model.xml.ManifestBuilder;
import org.olat.ims.qti21.pool.QTI21QPoolServiceProvider;
import org.olat.ims.qti21.ui.AssessmentTestDisplayController;
import org.olat.ims.qti21.ui.QTI21AssessmentDetailsController;
import org.olat.ims.qti21.ui.QTI21OverrideOptions;
import org.olat.ims.qti21.ui.QTI21RuntimeController;
import org.olat.ims.qti21.ui.editor.AssessmentTestComposerController;
import org.olat.modules.qpool.model.QItemList;
import org.olat.repository.ErrorList;
import org.olat.repository.RepositoryEntry;
import org.olat.repository.RepositoryManager;
import org.olat.repository.RepositoryService;
import org.olat.repository.handlers.EditionSupport;
import org.olat.repository.handlers.FileHandler;
import org.olat.repository.model.RepositoryEntrySecurity;
import org.olat.repository.ui.RepositoryEntryRuntimeController.RuntimeControllerCreator;
import org.olat.resource.OLATResource;
import org.olat.resource.OLATResourceManager;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import de.bps.onyx.plugin.OnyxModule;
import uk.ac.ed.ph.jqtiplus.node.item.AssessmentItem;
import uk.ac.ed.ph.jqtiplus.node.test.AssessmentSection;
import uk.ac.ed.ph.jqtiplus.node.test.AssessmentTest;
import uk.ac.ed.ph.jqtiplus.serialization.QtiSerializer;
/**
*
* Initial date: 08.12.2014<br>
* @author srosse, stephane.rosse@frentix.com, http://www.frentix.com
*
*/
@Service
public class QTI21AssessmentTestHandler extends FileHandler {
private static final OLog log = Tracing.createLoggerFor(QTI21AssessmentTestHandler.class);
@Autowired
private DB dbInstance;
@Autowired
private QTI21Service qtiService;
@Autowired
private RepositoryService repositoryService;
@Autowired
private QTI21QPoolServiceProvider qpoolServiceProvider;
@Autowired
private AssessmentTestSessionDAO assessmentTestSessionDao;
@Override
public String getSupportedType() {
return ImsQTI21Resource.TYPE_NAME;
}
@Override
public boolean isCreate() {
return true;
}
@Override
public String getCreateLabelI18nKey() {
return "tools.add.qti21";
}
@Override
public RepositoryEntry createResource(Identity initialAuthor, String displayname, String description, Object createObject, Locale locale) {
ImsQTI21Resource ores = new ImsQTI21Resource();
OLATResource resource = OLATResourceManager.getInstance().findOrPersistResourceable(ores);
RepositoryEntry re = repositoryService.create(initialAuthor, null, "", displayname, description, resource, RepositoryEntry.ACC_OWNERS);
dbInstance.commit();
File repositoryDir = new File(FileResourceManager.getInstance().getFileResourceRoot(re.getOlatResource()), FileResourceManager.ZIPDIR);
if(!repositoryDir.exists()) {
repositoryDir.mkdirs();
}
if(createObject instanceof QItemList) {
QItemList itemToImport = (QItemList)createObject;
qpoolServiceProvider.exportToEditorPackage(repositoryDir, itemToImport.getItems(), locale);
} else if(createObject instanceof QTIEditorPackage) {
QTIEditorPackage testToConvert = (QTIEditorPackage)createObject;
qpoolServiceProvider.convertFromEditorPackage(testToConvert, repositoryDir, locale);
} else if(createObject instanceof OLATResource) {
//convert a Onyx test in QTI 2.1
OLATResource onyxResource = (OLATResource)createObject;
RepositoryEntry onyxRe = CoreSpringFactory.getImpl(RepositoryService.class)
.loadByResourceKey(onyxResource.getKey());
if(OnyxModule.isOnyxTest((OLATResource)createObject)) {
copyOnyxResources(onyxResource, repositoryDir);
} else {
QTIEditorPackage testToConvert = TestFileResource.getQTIEditorPackageReader(onyxResource);
qpoolServiceProvider.convertFromEditorPackage(testToConvert, repositoryDir, locale);
}
copyMetadata(onyxRe, re, repositoryDir);
} else {
createMinimalAssessmentTest(displayname, repositoryDir, locale);
}
return re;
}
/**
* Copy the Onyx assessmentTest, assessmentItems, attachments and media folder.
*
* @param onyxZippedDir
* @param targetDirectory
* @return true if the copy is successful
*/
private boolean copyOnyxResources(OLATResource onyxResource, File targetDirectory) {
try {
// copy files
File onyxResourceFileroot = FileResourceManager.getInstance().getFileResourceRootImpl(onyxResource).getBasefile();
File onyxZippedDir = new File(onyxResourceFileroot, FileResourceManager.ZIPDIR);
Path path = onyxZippedDir.toPath();
Path destDir = targetDirectory.toPath();
QTI21IMSManifestExplorerVisitor visitor = new QTI21IMSManifestExplorerVisitor();
Files.walkFileTree(path, visitor);
Files.walkFileTree(path, new CopyAndConvertVisitor(path, destDir, visitor.getInfos(), new YesMatcher()));
return true;
} catch (IOException e) {
log.error("", e);
return false;
}
}
private RepositoryEntry copyMetadata(RepositoryEntry originalRe, RepositoryEntry re, File targetDirectory) {
//copy some metadata
re.setAuthors(originalRe.getAuthors());
re.setDescription(originalRe.getDescription());
re.setObjectives(originalRe.getObjectives());
re.setRequirements(originalRe.getRequirements());
re.setExpenditureOfWork(originalRe.getExpenditureOfWork());
re.setCredits(originalRe.getCredits());
re.setLocation(originalRe.getLocation());
RepositoryManager repositoryManager = CoreSpringFactory.getImpl(RepositoryManager.class);
repositoryManager.copyImage(originalRe, re);
File resourceFileroot = FileResourceManager.getInstance().getFileResourceRootImpl(originalRe.getOlatResource()).getBasefile();
FileUtils.copyDirToDir(new File(resourceFileroot, "media"), targetDirectory.getParentFile(), "copy media folder");
re = CoreSpringFactory.getImpl(RepositoryService.class).update(re);
return re;
}
public void createMinimalAssessmentTest(String displayName, File directory, Locale locale) {
ManifestBuilder manifestBuilder = ManifestBuilder.createAssessmentTestBuilder();
Translator translator = Util.createPackageTranslator(AssessmentTestComposerController.class, locale);
//single choice
File itemFile = new File(directory, IdentifierGenerator.newAsString(QTI21QuestionType.sc.getPrefix()) + ".xml");
AssessmentItem assessmentItem = AssessmentItemFactory.createSingleChoice(translator.translate("new.sc"), translator.translate("new.answer"));
QtiSerializer qtiSerializer = qtiService.qtiSerializer();
manifestBuilder.appendAssessmentItem(itemFile.getName());
//test
File testFile = new File(directory, IdentifierGenerator.newAssessmentTestFilename());
AssessmentTest assessmentTest = AssessmentTestFactory.createAssessmentTest(displayName, translator.translate("new.section"));
manifestBuilder.appendAssessmentTest(testFile.getName());
// item -> test
try {
AssessmentSection section = assessmentTest.getTestParts().get(0).getAssessmentSections().get(0);
AssessmentTestFactory.appendAssessmentItem(section, itemFile.getName());
} catch (URISyntaxException e) {
log.error("", e);
}
try(FileOutputStream out = new FileOutputStream(itemFile)) {
qtiSerializer.serializeJqtiObject(assessmentItem, out);
} catch(Exception e) {
log.error("", e);
}
try(FileOutputStream out = new FileOutputStream(testFile)) {
qtiSerializer.serializeJqtiObject(assessmentTest, out);
} catch(Exception e) {
log.error("", e);
}
manifestBuilder.write(new File(directory, "imsmanifest.xml"));
}
@Override
public boolean isPostCreateWizardAvailable() {
return false;
}
@Override
public ResourceEvaluation acceptImport(File file, String filename) {
return ImsQTI21Resource.evaluate(file, filename);
}
@Override
public RepositoryEntry importResource(Identity initialAuthor, String initialAuthorAlt, String displayname, String description,
boolean withReferences, Locale locale, File file, String filename) {
ImsQTI21Resource ores = new ImsQTI21Resource();
OLATResource resource = OLATResourceManager.getInstance().createAndPersistOLATResourceInstance(ores);
File fResourceFileroot = FileResourceManager.getInstance().getFileResourceRootImpl(resource).getBasefile();
File zipDir = new File(fResourceFileroot, FileResourceManager.ZIPDIR);
copyResource(file, filename, zipDir);
RepositoryEntry re = CoreSpringFactory.getImpl(RepositoryService.class)
.create(initialAuthor, null, "", displayname, description, resource, RepositoryEntry.ACC_OWNERS);
DBFactory.getInstance().commit();
return re;
}
private boolean copyResource(File file, String filename, File targetDirectory) {
try {
Path path = FileResource.getResource(file, filename);
if(path == null) {
return false;
}
Path destDir = targetDirectory.toPath();
QTI21IMSManifestExplorerVisitor visitor = new QTI21IMSManifestExplorerVisitor();
Files.walkFileTree(path, visitor);
Files.walkFileTree(path, new CopyAndConvertVisitor(path, destDir, visitor.getInfos(), new YesMatcher()));
return true;
} catch (IOException e) {
log.error("", e);
return false;
}
}
@Override
public MediaResource getAsMediaResource(OLATResourceable res, boolean backwardsCompatible) {
File unzippedDir = FileResourceManager.getInstance().unzipFileResource(res);
String displayName = CoreSpringFactory.getImpl(RepositoryManager.class)
.lookupDisplayNameByOLATResourceableId(res.getResourceableId());
return new ZippedDirectoryMediaResource(displayName, unzippedDir);
}
@Override
public RepositoryEntry copy(Identity author, RepositoryEntry source, RepositoryEntry target) {
File sourceRootFile = FileResourceManager.getInstance().getFileResourceRootImpl(source.getOlatResource()).getBasefile();
File targetRootDir = FileResourceManager.getInstance().getFileResourceRootImpl(target.getOlatResource()).getBasefile();
File sourceDir = new File(sourceRootFile, FileResourceManager.ZIPDIR);
File targetDir = new File(targetRootDir, FileResourceManager.ZIPDIR);
FileUtils.copyDirContentsToDir(sourceDir, targetDir, false, "Copy");
return target;
}
@Override
public boolean supportsDownload() {
return true;
}
@Override
public EditionSupport supportsEdit(OLATResourceable resource) {
return EditionSupport.yes;
}
@Override
public boolean supportsAssessmentDetails() {
return true;
}
@Override
public MainLayoutController createLaunchController(RepositoryEntry re, RepositoryEntrySecurity reSecurity,
UserRequest ureq, WindowControl wControl) {
return new QTI21RuntimeController(ureq, wControl, re, reSecurity,
new RuntimeControllerCreator() {
@Override
public Controller create(UserRequest uureq, WindowControl wwControl, TooledStackedPanel toolbarPanel,
RepositoryEntry entry, RepositoryEntrySecurity repoSecurity, AssessmentMode mode) {
QTI21DeliveryOptions deliveryOptions = qtiService.getDeliveryOptions(entry);
QTI21OverrideOptions overrideOptions = QTI21OverrideOptions.nothingOverriden();
if(!deliveryOptions.isAllowAnonym() && uureq.getUserSession().getRoles().isGuestOnly()) {
Translator translator = Util.createPackageTranslator(QTI21RuntimeController.class, uureq.getLocale());
Controller contentCtr = MessageUIFactory.createInfoMessage(uureq, wwControl,
translator.translate("anonym.not.allowed.title"),
translator.translate("anonym.not.allowed.descr"));
return new LayoutMain3ColsController(uureq, wwControl, contentCtr);
}
boolean authorMode = reSecurity.isEntryAdmin();
CoreSpringFactory.getImpl(UserCourseInformationsManager.class)
.updateUserCourseInformations(entry.getOlatResource(), uureq.getIdentity());
return new AssessmentTestDisplayController(uureq, wwControl, null, entry, entry, null,
deliveryOptions, overrideOptions, false, authorMode, false);
}
});
}
@Override
public Controller createEditorController(RepositoryEntry re, UserRequest ureq, WindowControl wControl, TooledStackedPanel toolbar) {
AssessmentTestComposerController editorCtrl = new AssessmentTestComposerController(ureq, wControl, toolbar, re);
return editorCtrl;
}
@Override
public Controller createAssessmentDetailsController(RepositoryEntry re, UserRequest ureq, WindowControl wControl,
TooledStackedPanel toolbar, Identity assessedIdentity) {
return new QTI21AssessmentDetailsController(ureq, wControl, re, assessedIdentity);
}
@Override
public StepsMainRunController createWizardController(OLATResourceable res, UserRequest ureq, WindowControl wControl) {
return null;
}
@Override
public LockResult acquireLock(OLATResourceable ores, Identity identity) {
return null;
}
@Override
public void releaseLock(LockResult lockResult) {
//
}
@Override
public boolean isLocked(OLATResourceable ores) {
return false;
}
@Override
public boolean readyToDelete(RepositoryEntry entry, Identity identity, Roles roles, Locale locale, ErrorList errors) {
return super.readyToDelete(entry, identity, roles, locale, errors);
}
@Override
public boolean cleanupOnDelete(RepositoryEntry entry, OLATResourceable res) {
boolean clean = super.cleanupOnDelete(entry, res);
assessmentTestSessionDao.deleteAllUserTestSessionsByTest(entry);
return clean;
}
@Override
protected String getDeletedFilePrefix() {
return null;
}
}