better seb restriction update handling

This commit is contained in:
anhefti 2019-11-03 15:05:48 +01:00
parent 810b6dc8c2
commit 2d016ed7fc
6 changed files with 96 additions and 42 deletions

View file

@ -87,6 +87,13 @@ public interface LmsAPITemplate {
* @return Result refer to the given SebRestrictionData if restriction was successful or to an error if not */
Result<SebRestrictionData> applySebClientRestriction(SebRestrictionData sebRestrictionData);
/** Updates a SEB Client restriction within the LMS with the given attributes.
*
* @param sebRestrictionData containing all data for SEB Client restriction
* @return Result refer to the given SebRestrictionData if updating restriction was successful or to an error if
* not */
Result<SebRestrictionData> updateSebClientRestriction(SebRestrictionData sebRestrictionData);
/** Releases an already applied SEB Client restriction within the LMS for a given Exam.
* This completely removes the SEB Client restriction on LMS side.
*

View file

@ -149,6 +149,12 @@ final class MockupLmsAPITemplate implements LmsAPITemplate {
return Result.of(sebRestrictionData);
}
@Override
public Result<SebRestrictionData> updateSebClientRestriction(final SebRestrictionData sebRestrictionData) {
log.info("Update SEB Client restriction: {}", sebRestrictionData);
return Result.of(sebRestrictionData);
}
@Override
public Result<Exam> releaseSebClientRestriction(final Exam exam) {
log.info("Release SEB Client restriction for Exam: {}", exam);

View file

@ -191,6 +191,20 @@ final class OpenEdxLmsAPITemplate implements LmsAPITemplate {
});
}
@Override
public Result<SebRestrictionData> updateSebClientRestriction(final SebRestrictionData sebRestrictionData) {
return Result.tryCatch(() -> {
if (log.isDebugEnabled()) {
log.debug("Update SEB Client restriction: {}", sebRestrictionData);
}
// TODO
return sebRestrictionData;
});
}
@Override
public Result<Exam> releaseSebClientRestriction(final Exam exam) {
return Result.tryCatch(() -> {

View file

@ -18,7 +18,6 @@ import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import ch.ethz.seb.sebserver.gbl.Constants;
import ch.ethz.seb.sebserver.gbl.api.APIMessage;
@ -74,7 +73,6 @@ public class ExamConfigUpdateServiceImpl implements ExamConfigUpdateService {
// generate the new Config Key and update the Config Key within the LMSSetup API for each exam (delete old Key and add new Key)
// evict each Exam from cache and release the update-lock on DB
@Override
@Transactional
public Result<Collection<Long>> processExamConfigurationChange(final Long configurationNodeId) {
final String updateId = this.examUpdateHandler.createUpdateId();
@ -172,7 +170,6 @@ public class ExamConfigUpdateServiceImpl implements ExamConfigUpdateService {
}
@Override
@Transactional
public <T> Result<T> processExamConfigurationMappingChange(
final ExamConfigurationMap mapping,
final Function<ExamConfigurationMap, Result<T>> changeAction) {
@ -205,29 +202,35 @@ public class ExamConfigUpdateServiceImpl implements ExamConfigUpdateService {
// check again if there are no new active client connections in the meantime
checkActiveClientConnections(exam);
// apply the referenced change action
// apply the referenced change action. On error the change is rolled back and
// this processing returns immediately with the error
final T result = changeAction.apply(mapping)
.getOrThrow();
// flush the exam cache
this.examSessionService.flushCache(exam)
.onError(t -> log.error("Fauled to save exam configuration: {}",
mapping.configurationNodeId))
.getOrThrow();
// update seb client restriction if the feature is activated
if (exam.lmsSebRestriction) {
final Result<Exam> updateSebClientRestriction = this.updateSebClientRestriction(exam);
// if there was an error during update, it is logged but this process goes on
// and the saved changes are not rolled back
if (updateSebClientRestriction.hasError()) {
log.error("Failed to update SEB Client restriction on LMS for exam: {}", exam);
}
}
// release the lock
// flush the exam cache. If there was an error during flush, it is logged but this process goes on
// and the saved changes are not rolled back
this.examSessionService.flushCache(exam)
.onError(t -> log.error("Failed to flush cache for exam: {}", exam));
// release the exam lock
this.examDAO.releaseLock(exam.id, updateId)
.getOrThrow();
.onError(t -> log.error("Failed to release lock for exam: {}", exam));
return result;
})
.onError(TransactionHandler::rollback);
.onError(t -> this.examDAO.forceUnlock(mapping.examId));
}
@ -280,20 +283,17 @@ public class ExamConfigUpdateServiceImpl implements ExamConfigUpdateService {
@Override
public Result<Exam> applySebClientRestriction(final Exam exam) {
return this.examUpdateHandler.applySebClientRestriction(exam)
.onError(error -> log.error("Failed to apply SEB Client restriction for Exam: {}", exam, error));
return this.examUpdateHandler.applySebClientRestriction(exam);
}
@Override
public Result<Exam> updateSebClientRestriction(final Exam exam) {
return this.examUpdateHandler.updateSebClientRestriction(exam)
.onError(error -> log.error("Failed to update SEB Client restriction for Exam: {}", exam, error));
return this.examUpdateHandler.updateSebClientRestriction(exam);
}
@Override
public Result<Exam> releaseSebClientRestriction(final Exam exam) {
return this.examUpdateHandler.releaseSebClientRestriction(exam)
.onError(error -> log.error("Failed to release SEB Client restriction for Exam: {}", exam, error));
return this.examUpdateHandler.releaseSebClientRestriction(exam);
}
private void checkIntegrityDoubleCheck(

View file

@ -126,24 +126,7 @@ class ExamUpdateHandler {
return Result.tryCatch(() -> {
final Collection<String> configKeys = this.sebExamConfigService
.generateConfigKeys(exam.institutionId, exam.id)
.getOrThrow();
final Collection<String> browserExamKeys = new ArrayList<>();
final String browserExamKeysString = exam.getBrowserExamKeys();
if (StringUtils.isNotBlank(browserExamKeysString)) {
browserExamKeys.addAll(Arrays.asList(StringUtils.split(
browserExamKeysString,
Constants.LIST_SEPARATOR)));
}
final SebRestrictionData sebRestrictionData = new SebRestrictionData(
exam,
configKeys,
browserExamKeys,
// TODO when we have more restriction details available form the Exam, put it to the map
Collections.emptyMap());
final SebRestrictionData sebRestrictionData = createSebRestrictionData(exam);
if (log.isDebugEnabled()) {
log.debug("Appling SEB Client restriction on LMS with: {}", sebRestrictionData);
@ -166,11 +149,20 @@ class ExamUpdateHandler {
return Result.of(exam);
}
if (log.isDebugEnabled()) {
log.debug("Update SEB Client restrictions for exam: {}", exam);
}
// TODO Auto-generated method stub
return null;
return Result.tryCatch(() -> {
final SebRestrictionData sebRestrictionData = createSebRestrictionData(exam);
if (log.isDebugEnabled()) {
log.debug("Update SEB Client restrictions for exam: {}", exam);
}
return this.lmsAPIService
.getLmsAPITemplate(exam.lmsSetupId)
.flatMap(lmsTemplate -> lmsTemplate.updateSebClientRestriction(sebRestrictionData))
.map(data -> exam)
.getOrThrow();
});
}
Result<Exam> releaseSebClientRestriction(final Exam exam) {
@ -184,4 +176,39 @@ class ExamUpdateHandler {
.flatMap(template -> template.releaseSebClientRestriction(exam));
}
// @Transactional
// public <T> Result<T> updateExamConfigurationMappingChange(
// final ExamConfigurationMap mapping,
// final Function<ExamConfigurationMap, Result<T>> changeAction,
// final Exam exam) {
//
// return Result.tryCatch(() -> {
//
// return result;
// })
// .onError(TransactionHandler::rollback);
// }
private SebRestrictionData createSebRestrictionData(final Exam exam) {
final Collection<String> configKeys = this.sebExamConfigService
.generateConfigKeys(exam.institutionId, exam.id)
.getOrThrow();
final Collection<String> browserExamKeys = new ArrayList<>();
final String browserExamKeysString = exam.getBrowserExamKeys();
if (StringUtils.isNotBlank(browserExamKeysString)) {
browserExamKeys.addAll(Arrays.asList(StringUtils.split(
browserExamKeysString,
Constants.LIST_SEPARATOR)));
}
final SebRestrictionData sebRestrictionData = new SebRestrictionData(
exam,
configKeys,
browserExamKeys,
// TODO when we have more restriction details available form the Exam, put it to the map
Collections.emptyMap());
return sebRestrictionData;
}
}

View file

@ -194,7 +194,7 @@ public class ExamConfigurationMappingController extends EntityController<ExamCon
null,
null,
null,
ConfigurationStatus.IN_USE))
ConfigurationStatus.READY_TO_USE))
.map(id -> pair);
}