diff --git a/src/main/java/org/olat/ims/qti21/QTI21Service.java b/src/main/java/org/olat/ims/qti21/QTI21Service.java index 2cdea6e1e4e9305ed61cbdbd1512157a4f1665f8..11dea632620b5bca2d744ee6926b2ebef317e23d 100644 --- a/src/main/java/org/olat/ims/qti21/QTI21Service.java +++ b/src/main/java/org/olat/ims/qti21/QTI21Service.java @@ -49,6 +49,7 @@ import uk.ac.ed.ph.jqtiplus.reading.QtiXmlReader; import uk.ac.ed.ph.jqtiplus.resolution.ResolvedAssessmentItem; import uk.ac.ed.ph.jqtiplus.resolution.ResolvedAssessmentObject; import uk.ac.ed.ph.jqtiplus.resolution.ResolvedAssessmentTest; +import uk.ac.ed.ph.jqtiplus.running.TestSessionController; import uk.ac.ed.ph.jqtiplus.serialization.QtiSerializer; import uk.ac.ed.ph.jqtiplus.state.ItemSessionState; import uk.ac.ed.ph.jqtiplus.state.TestPlanNodeKey; @@ -512,5 +513,10 @@ public interface QTI21Service { * @return A number of seconds, 0 if nothing found */ public Long getMetadataCorrectionTimeInSeconds(RepositoryEntry testEntry, AssessmentTestSession candidateSession); + + + public void putCachedTestSessionController(AssessmentTestSession testSession, TestSessionController testSessionController); + + public TestSessionController getCachedTestSessionController(AssessmentTestSession testSession, TestSessionController testSessionController); } diff --git a/src/main/java/org/olat/ims/qti21/manager/QTI21ServiceImpl.java b/src/main/java/org/olat/ims/qti21/manager/QTI21ServiceImpl.java index d0e8084957ecf3543c4fe7b67c05676797a58292..571020725b0b6fe2b30a8054b9b9a0595ec8abce 100644 --- a/src/main/java/org/olat/ims/qti21/manager/QTI21ServiceImpl.java +++ b/src/main/java/org/olat/ims/qti21/manager/QTI21ServiceImpl.java @@ -149,6 +149,7 @@ import uk.ac.ed.ph.jqtiplus.reading.QtiXmlReader; import uk.ac.ed.ph.jqtiplus.resolution.ResolvedAssessmentItem; import uk.ac.ed.ph.jqtiplus.resolution.ResolvedAssessmentObject; import uk.ac.ed.ph.jqtiplus.resolution.ResolvedAssessmentTest; +import uk.ac.ed.ph.jqtiplus.running.TestSessionController; import uk.ac.ed.ph.jqtiplus.serialization.QtiSerializer; import uk.ac.ed.ph.jqtiplus.serialization.SaxFiringOptions; import uk.ac.ed.ph.jqtiplus.state.AssessmentSectionSessionState; @@ -229,6 +230,7 @@ public class QTI21ServiceImpl implements QTI21Service, UserDataDeletable, Initia private InfinispanXsltStylesheetCache xsltStylesheetCache; private CacheWrapper<File,ResolvedAssessmentTest> assessmentTestsCache; private CacheWrapper<File,ResolvedAssessmentItem> assessmentItemsCache; + private CacheWrapper<AssessmentTestSession,TestSessionController> testSessionControllersCache; private final ConcurrentMap<String,URI> resourceToTestURI = new ConcurrentHashMap<>(); @@ -255,6 +257,7 @@ public class QTI21ServiceImpl implements QTI21Service, UserDataDeletable, Initia Cacher cacher = coordinatorManager.getInstance().getCoordinator().getCacher(); assessmentTestsCache = cacher.getCache("QTIWorks", "assessmentTests"); assessmentItemsCache = cacher.getCache("QTIWorks", "assessmentItems"); + testSessionControllersCache = cacher.getCache("QTIWorks", "testSessionControllers"); } @Override @@ -593,6 +596,8 @@ public class QTI21ServiceImpl implements QTI21Service, UserDataDeletable, Initia @Override public AssessmentTestSession reloadAssessmentTestSession(AssessmentTestSession session) { + if(session == null) return null; + if(session.getKey() == null) return session; return testSessionDao.loadByKey(session.getKey()); } @@ -1582,4 +1587,18 @@ public class QTI21ServiceImpl implements QTI21Service, UserDataDeletable, Initia return Long.valueOf(timeInMinutes * 60l); } + + @Override + public void putCachedTestSessionController(AssessmentTestSession testSession, TestSessionController testSessionController) { + if(testSession == null || testSessionController == null) return; + testSessionControllersCache.put(testSession, testSessionController); + } + + @Override + public TestSessionController getCachedTestSessionController(AssessmentTestSession testSession, TestSessionController testSessionController) { + if(testSession == null) return null; + + TestSessionController result = testSessionControllersCache.get(testSession); + return result == null ? testSessionController : result; + } } diff --git a/src/main/java/org/olat/ims/qti21/ui/AssessmentTestDisplayController.java b/src/main/java/org/olat/ims/qti21/ui/AssessmentTestDisplayController.java index d8c73c8413beb067bb4b4f035b3aa29392a0d30f..a87cafc4c26ffafad83d16c25022494bacd47283 100644 --- a/src/main/java/org/olat/ims/qti21/ui/AssessmentTestDisplayController.java +++ b/src/main/java/org/olat/ims/qti21/ui/AssessmentTestDisplayController.java @@ -415,8 +415,11 @@ public class AssessmentTestDisplayController extends BasicController implements resourcesList.deregisterResourceable(entry, subIdent, getWindow()); } try { - suspendAssessmentTest(new Date()); + candidateSession = qtiService.reloadAssessmentTestSession(candidateSession); if(candidateSession != null) { + testSessionController = qtiService.getCachedTestSessionController(candidateSession, testSessionController); + suspendAssessmentTest(new Date()); + OLATResourceable sessionOres = OresHelper .createOLATResourceableInstance(AssessmentTestSession.class, candidateSession.getKey()); CoordinatorManager.getInstance().getCoordinator().getEventBus().deregisterFor(this, sessionOres); @@ -570,6 +573,8 @@ public class AssessmentTestDisplayController extends BasicController implements } private void doSuspend(UserRequest ureq) { + testSessionController = qtiService.getCachedTestSessionController(candidateSession, testSessionController); + VelocityContainer suspendedVC = createVelocityContainer("suspended"); mainPanel.setContent(suspendedVC); suspendAssessmentTest(ureq.getRequestTimestamp()); @@ -648,6 +653,17 @@ public class AssessmentTestDisplayController extends BasicController implements return sessionDeleted; } + private boolean sessionEndedOrSuspended() { + TestSessionState testSessionState = testSessionController.getTestSessionState(); + if(testSessionState.isEnded() || testSessionState.isSuspended()) { + candidateSession = qtiService.reloadAssessmentTestSession(candidateSession); + showWarning("warning.suspended.ended.assessmenttest"); + logAudit("Try to work on an ended/suspended test"); + return true; + } + return false; + } + /** * This method maintains the assessment test in cache during * a test session. This controller doesn't need the cache, the @@ -765,7 +781,9 @@ public class AssessmentTestDisplayController extends BasicController implements } private void processQTIEvent(UserRequest ureq, QTIWorksAssessmentTestEvent qe) { - if(timeLimitBarrier(ureq) || sessionReseted(ureq)) { + testSessionController = qtiService.getCachedTestSessionController(candidateSession, testSessionController); + + if(timeLimitBarrier(ureq) || sessionReseted(ureq) || sessionEndedOrSuspended()) { return; } @@ -1720,6 +1738,7 @@ public class AssessmentTestDisplayController extends BasicController implements if (notificationRecorder!=null) { result.addNotificationListener(notificationRecorder); } + qtiService.putCachedTestSessionController(candidateSession, result); return result; } @@ -1727,9 +1746,9 @@ public class AssessmentTestDisplayController extends BasicController implements Date requestTimestamp = ureq.getRequestTimestamp(); final NotificationRecorder notificationRecorder = new NotificationRecorder(NotificationLevel.INFO); - TestSessionController controller = createTestSessionController(notificationRecorder); + TestSessionController controller = createTestSessionController(notificationRecorder); if(!controller.getTestSessionState().isEnded() && !controller.getTestSessionState().isExited()) { - controller.unsuspendTestSession(requestTimestamp); + controller.unsuspendTestSession(requestTimestamp); TestSessionState testSessionState = controller.getTestSessionState(); TestPlanNodeKey currentItemKey = testSessionState.getCurrentItemKey(); @@ -1749,8 +1768,13 @@ public class AssessmentTestDisplayController extends BasicController implements } private TestSessionController createTestSessionController(NotificationRecorder notificationRecorder) { - final TestSessionState testSessionState = qtiService.loadTestSessionState(candidateSession); - return createTestSessionController(testSessionState, notificationRecorder); + TestSessionController result = qtiService.getCachedTestSessionController(candidateSession, null); + if(result == null) { + final TestSessionState testSessionState = qtiService.loadTestSessionState(candidateSession); + result = createTestSessionController(testSessionState, notificationRecorder); + qtiService.putCachedTestSessionController(candidateSession, result); + } + return result; } public TestSessionController createTestSessionController(TestSessionState testSessionState, NotificationRecorder notificationRecorder) { diff --git a/src/main/java/org/olat/ims/qti21/ui/_i18n/LocalStrings_de.properties b/src/main/java/org/olat/ims/qti21/ui/_i18n/LocalStrings_de.properties index 2e72dd256b4f72997ee16cdec6de359866a5c4bd..8c46f0e812e47375af1aac606f7bf6f486b029ca 100644 --- a/src/main/java/org/olat/ims/qti21/ui/_i18n/LocalStrings_de.properties +++ b/src/main/java/org/olat/ims/qti21/ui/_i18n/LocalStrings_de.properties @@ -282,5 +282,6 @@ validate.xml.signature.ok=Testquittung und Datei konnte erfolgreich validiert we warning.download.log=Es gibt leider kein Logdatei f\u00FCr diesen Test. warning.reset.assessmenttest.data=Die Test-Resultate wurden von einem Administrator oder Kursbesitzer zur\u00FCckgesetzt. Sie k\u00F6nnen den Test nicht fortsetzen und m\u00FCssen ihn erneut starten. warning.reset.test.data.nobody=Es gibt kein Teilnehmer zu zur\u00FCcksetzen +warning.suspended.ended.assessmenttest=Sie haben schon den Test unterbrochen oder beendet, wahrscheinlich in einem anderen Fenster. Bitte, diese Fenster jetzt schliessen. warning.xml.signature.notok=Unterschrift und Datei konnte nicht validiert werden. warning.xml.signature.session.not.found=Die Resultaten konnte nicht gefunden werden. diff --git a/src/main/java/org/olat/ims/qti21/ui/_i18n/LocalStrings_en.properties b/src/main/java/org/olat/ims/qti21/ui/_i18n/LocalStrings_en.properties index cef4e7c927f12b0af4185b413089f01aab2dc801..aa5a166cb944cadb34030ff5b28fefb9fac1443c 100644 --- a/src/main/java/org/olat/ims/qti21/ui/_i18n/LocalStrings_en.properties +++ b/src/main/java/org/olat/ims/qti21/ui/_i18n/LocalStrings_en.properties @@ -281,6 +281,7 @@ validate.xml.signature.file=XML file validate.xml.signature.ok=Test receipt and results was successfully validated. warning.download.log=There is not a log file for this test. warning.reset.assessmenttest.data=The test results were reset by an administrator or course owner. You cannot continue the test and have to restart it. +warning.suspended.ended.assessmenttest=You have already suspended or ended this test, probably in an other window. Please close this window. warning.reset.test.data.nobody=There isn't any participant which data can be reseted. warning.xml.signature.notok=Signature and results cannot be validate each other. warning.xml.signature.session.not.found=Tests results cannot be found. diff --git a/src/main/resources/infinispan-config.xml b/src/main/resources/infinispan-config.xml index f01b674ceb698b7d90059dbf867b3f99231a88f8..004bc3b9d9c1531918b63b01b4b0f748c14e073a 100644 --- a/src/main/resources/infinispan-config.xml +++ b/src/main/resources/infinispan-config.xml @@ -77,6 +77,13 @@ <expiration max-idle="1800000" interval="15000" /> </local-cache> + <local-cache name="QTIWorks@testSessionControllers" simple-cache="true" statistics="true" statistics-available="true"> + <locking isolation="READ_COMMITTED" concurrency-level="1000" acquire-timeout="15000" striping="false" /> + <transaction mode="NONE" auto-commit="true" /> + <memory max-count="50000" when-full="REMOVE" /> + <expiration max-idle="7200000" interval="15000" /> + </local-cache> + <local-cache name="WebDAVManager@webdav" simple-cache="true" statistics="true" statistics-available="true"> <locking isolation="READ_COMMITTED" concurrency-level="1000" acquire-timeout="15000" striping="false" /> <transaction mode="NONE" auto-commit="true" /> diff --git a/src/test/java/org/olat/selenium/page/qti/QTI21Page.java b/src/test/java/org/olat/selenium/page/qti/QTI21Page.java index 46a4052e5513a0b837c2353e79200c417b36b3fa..f8df2ff40ce0450b643f2183f59ff54d915534b6 100644 --- a/src/test/java/org/olat/selenium/page/qti/QTI21Page.java +++ b/src/test/java/org/olat/selenium/page/qti/QTI21Page.java @@ -517,13 +517,14 @@ public class QTI21Page { */ public QTI21Page answerGraphicGapClick(String item, String gap) { By sourceBy = By.xpath("//div[contains(@class,'gap_container')]/div[contains(@class,'o_gap_item')][@data-qti-id='" + item + "']"); - OOGraphene.waitElement(sourceBy, 5, browser); + OOGraphene.waitElement(sourceBy, browser); browser.findElement(sourceBy).click(); By areaBy = By.xpath("//div[@class='graphicGapMatchInteraction']//map/area[@data-qti-id='" + gap + "']"); WebElement areaEl = browser.findElement(areaBy); String coords = areaEl.getAttribute("coords"); - By imgBy = By.xpath("//div[contains(@class,'graphicGapMatchInteraction')]/div/div/img"); - WebElement element = browser.findElement(imgBy); + By canvasBy = By.xpath("//div[contains(@class,'graphicGapMatchInteraction')]/div/div/canvas"); + OOGraphene.waitElement(canvasBy, browser); + WebElement element = browser.findElement(canvasBy); Dimension dim = element.getSize(); Position pos = Position.valueOf(coords, dim); new Actions(browser)