Merge remote-tracking branch 'origin/rel-1.6.1'

This commit is contained in:
anhefti 2024-06-05 09:21:04 +02:00
commit 00e8366fda
12 changed files with 162 additions and 35 deletions

View file

@ -16,7 +16,7 @@
<packaging>jar</packaging> <packaging>jar</packaging>
<properties> <properties>
<sebserver-version>1.6.0</sebserver-version> <sebserver-version>1.6.1</sebserver-version>
<build-version>${sebserver-version}</build-version> <build-version>${sebserver-version}</build-version>
<revision>${sebserver-version}</revision> <revision>${sebserver-version}</revision>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>

View file

@ -35,7 +35,7 @@ public class UserFeatures {
LMS_SETUP_MOODLE_PLUGIN("lms.setup.type.MOODLE_PLUGIN"), LMS_SETUP_MOODLE_PLUGIN("lms.setup.type.MOODLE_PLUGIN"),
LMS_SETUP_OPEN_EDX("lms.setup.type.OPEN_EDX"), LMS_SETUP_OPEN_EDX("lms.setup.type.OPEN_EDX"),
LMS_SETUP_ANS("lms.setup.type.ANS_DELFT"), LMS_SETUP_ANS("lms.setup.type.ANS_DELFT"),
LMS_SETUP_OPEN_OLAT("lms.setup.type.OPEN_OLAT"), LMS_SETUP_OLAT("lms.setup.type.OLAT"),
QUIZ_LOOKUP("lms.quiz.lookup"), QUIZ_LOOKUP("lms.quiz.lookup"),

View file

@ -416,7 +416,7 @@ public class SEBClientConfigForm implements TemplateComposer {
SEBClientConfig.ATTR_EXAM_SELECTION, SEBClientConfig.ATTR_EXAM_SELECTION,
FORM_EXAM_SELECTION_TEXT_KEY, FORM_EXAM_SELECTION_TEXT_KEY,
StringUtils.join(clientConfig.selectedExams, Constants.LIST_SEPARATOR), StringUtils.join(clientConfig.selectedExams, Constants.LIST_SEPARATOR),
() -> pageService.getResourceService().getExamLogSelectionResources()) () -> pageService.getResourceService().getActiveExamResources())
.withInputSpan(5)) .withInputSpan(5))
.withDefaultSpanEmptyCell(1); .withDefaultSpanEmptyCell(1);

View file

@ -717,6 +717,24 @@ public class ResourceService {
.collect(Collectors.toList()); .collect(Collectors.toList());
} }
public List<Tuple<String>> getActiveExamResources() {
final UserInfo userInfo = this.currentUser.get();
return this.restService.getBuilder(GetExams.class)
.withQueryParam(Entity.FILTER_ATTR_INSTITUTION, String.valueOf(userInfo.getInstitutionId()))
.withQueryParam(Exam.FILTER_CACHED_QUIZZES, Constants.TRUE_STRING)
.call()
.getOr(Collections.emptyList())
.stream()
.filter(exam -> exam != null &&
exam.getStatus() != ExamStatus.FINISHED &&
exam.getStatus() != ExamStatus.ARCHIVED)
.map(exam -> new Tuple<>(
exam.getModelId(),
StringUtils.isBlank(exam.name) ? exam.externalId : exam.name))
.sorted(RESOURCE_COMPARATOR)
.collect(Collectors.toList());
}
public List<Tuple<String>> getExamResources() { public List<Tuple<String>> getExamResources() {
final UserInfo userInfo = this.currentUser.get(); final UserInfo userInfo = this.currentUser.get();
return this.restService.getBuilder(GetExamNames.class) return this.restService.getBuilder(GetExamNames.class)

View file

@ -0,0 +1,92 @@
/*
* Copyright (c) 2019 ETH Zürich, IT Services
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/
package ch.ethz.seb.sebserver.webservice.datalayer.checks;
import static ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.OrientationRecordDynamicSqlSupport.*;
import static ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.OrientationRecordDynamicSqlSupport.title;
import ch.ethz.seb.sebserver.gbl.profile.WebServiceProfile;
import ch.ethz.seb.sebserver.gbl.util.Result;
import ch.ethz.seb.sebserver.webservice.DBIntegrityCheck;
import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.OrientationRecordDynamicSqlSupport;
import ch.ethz.seb.sebserver.webservice.datalayer.batis.mapper.OrientationRecordMapper;
import ch.ethz.seb.sebserver.webservice.datalayer.batis.model.OrientationRecord;
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.OrientationDAO;
import org.mybatis.dynamic.sql.SqlBuilder;
import org.mybatis.dynamic.sql.update.UpdateDSL;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Component;
@Lazy
@Component
@WebServiceProfile
public class ClipboardSEBSettingsGUICheck implements DBIntegrityCheck {
public static final Logger INIT_LOGGER = LoggerFactory.getLogger("ch.ethz.seb.SEB_SERVER_INIT");
private final OrientationRecordMapper orientationRecordMapper;
public ClipboardSEBSettingsGUICheck(final OrientationRecordMapper orientationRecordMapper) {
this.orientationRecordMapper = orientationRecordMapper;
}
@Override
public String name() {
return "ClipboardSEBSettingsGUICheck";
}
@Override
public String description() {
return "Check if clipboardPolicy SEB Setting is missing in the GUI and if so add it to GUI";
}
@Override
public Result<String> applyCheck(boolean tryFix) {
return Result.tryCatch(() -> {
// check if clipboardPolicy SEB Setting is missing
final Long count = orientationRecordMapper.countByExample()
.where(templateId, SqlBuilder.isEqualTo(0L))
.and(configAttributeId, SqlBuilder.isEqualTo(1201L))
.build()
.execute();
if (count != null && count.intValue() > 0) {
return "clipboardPolicy SEB Setting detected in GUI";
}
INIT_LOGGER.info("--------> Missing clipboardPolicy SEB Setting in GUI detected. Add it");
// move allowedSEBVersion setting
UpdateDSL.updateWithMapper(orientationRecordMapper::update, orientationRecord)
.set(yPosition).equalTo(21)
.where(templateId, SqlBuilder.isEqualTo(0L))
.and(configAttributeId, SqlBuilder.isEqualTo(1578L))
.build()
.execute();
// add clipboardPolicy setting
orientationRecordMapper.insert(new OrientationRecord(
null,
1201L,
0L,
9L,
"clipboardPolicy",
7,
18,
5,
2,
"NONE"
));
return "Missing clipboardPolicy SEB Setting in GUI successfully added";
});
}
}

View file

@ -103,7 +103,7 @@ public class DowngradeSEBSettingsCheck implements DBIntegrityCheck {
.map(OrientationRecord::getConfigAttributeId) .map(OrientationRecord::getConfigAttributeId)
.collect(Collectors.toList()); .collect(Collectors.toList());
if (attributeIds.isEmpty()) { if (attributeIds.isEmpty() || attributeIds.get(0).intValue() == 1201) {
return "No additional SEB Settings orientations for downgrading found."; return "No additional SEB Settings orientations for downgrading found.";
} }

View file

@ -56,13 +56,23 @@ public abstract class MoodleUtils {
private static String maskShortName(final String shortname) { private static String maskShortName(final String shortname) {
return shortname return shortname
.replace(Constants.SEMICOLON.toString(), "_SC_") .replace(Constants.SEMICOLON.toString(), "_SC_")
.replace(Constants.COLON.toString(), "_COLON_"); .replace(Constants.COLON.toString(), "_COLON_")
.replace(Constants.SLASH.toString(), "_SL_")
.replace(Constants.BACKSLASH.toString(), "_BSL_")
.replace(Constants.AMPERSAND.toString(), "_AMP_")
.replace(Constants.ANGLE_BRACE_OPEN.toString(), "_AO_")
.replace(Constants.ANGLE_BRACE_CLOSE.toString(), "_AC_");
} }
private static String unmaskShortName(final String shortname) { private static String unmaskShortName(final String shortname) {
return shortname return shortname
.replace("_SC_", Constants.SEMICOLON.toString()) .replace("_SC_", Constants.SEMICOLON.toString())
.replace("_COLON_", Constants.COLON.toString()); .replace("_COLON_", Constants.COLON.toString())
.replace("_SL_", Constants.SLASH.toString())
.replace("_BSL_", Constants.BACKSLASH.toString())
.replace("_AMP_", Constants.AMPERSAND.toString())
.replace("_AO_", Constants.ANGLE_BRACE_OPEN.toString())
.replace("_AC_", Constants.ANGLE_BRACE_CLOSE.toString());
} }
public static String getQuizId(final String internalQuizId) { public static String getQuizId(final String internalQuizId) {

View file

@ -275,7 +275,7 @@ public class OlatLmsAPITemplate extends AbstractCachedCourseAccess implements Lm
final DateTime quizFromTime = (filterMap != null) ? filterMap.getQuizFromTime() : null; final DateTime quizFromTime = (filterMap != null) ? filterMap.getQuizFromTime() : null;
final long fromCutTime = (quizFromTime != null) ? Utils.toUnixTimeInSeconds(quizFromTime) : -1; final long fromCutTime = (quizFromTime != null) ? Utils.toUnixTimeInSeconds(quizFromTime) : -1;
String url = "/restapi/assessment_modes/seb?"; String url = "/restapi/repo/assessmentmodes?";
if (fromCutTime != -1) { if (fromCutTime != -1) {
url = String.format("%sdateFrom=%s&", url, fromCutTime); url = String.format("%sdateFrom=%s&", url, fromCutTime);
} }
@ -295,8 +295,8 @@ public class OlatLmsAPITemplate extends AbstractCachedCourseAccess implements Lm
lmsSetup.getLmsType(), lmsSetup.getLmsType(),
a.name, a.name,
a.description, a.description,
Utils.toDateTimeUTC(a.dateFrom), Utils.toDateTimeUTC(a.begin - a.leadTime * 1000 * 60),
Utils.toDateTimeUTC(a.dateTo), Utils.toDateTimeUTC(a.end + a.followupTime * 1000 * 60),
examUrl(a.repositoryEntryKey), examUrl(a.repositoryEntryKey),
new HashMap<String, String>()); new HashMap<String, String>());
}) })
@ -316,7 +316,7 @@ public class OlatLmsAPITemplate extends AbstractCachedCourseAccess implements Lm
private QuizData quizById(final OlatLmsRestTemplate restTemplate, final String id) { private QuizData quizById(final OlatLmsRestTemplate restTemplate, final String id) {
final LmsSetup lmsSetup = this.apiTemplateDataSupplier.getLmsSetup(); final LmsSetup lmsSetup = this.apiTemplateDataSupplier.getLmsSetup();
final String url = String.format("/restapi/assessment_modes/%s", id); final String url = String.format("/restapi/repo/assessmentmodes/%s", id);
final AssessmentData a = this.apiGet(restTemplate, url, AssessmentData.class); final AssessmentData a = this.apiGet(restTemplate, url, AssessmentData.class);
return new QuizData( return new QuizData(
String.format("%d", a.key), String.format("%d", a.key),
@ -325,26 +325,26 @@ public class OlatLmsAPITemplate extends AbstractCachedCourseAccess implements Lm
lmsSetup.getLmsType(), lmsSetup.getLmsType(),
a.name, a.name,
a.description, a.description,
Utils.toDateTimeUTC(a.dateFrom), Utils.toDateTimeUTC(a.begin - a.leadTime * 1000 * 60),
Utils.toDateTimeUTC(a.dateTo), Utils.toDateTimeUTC(a.end + a.followupTime * 1000 * 60),
examUrl(a.repositoryEntryKey), examUrl(a.repositoryEntryKey),
new HashMap<String, String>()); new HashMap<String, String>());
} }
private ExamineeAccountDetails getExamineeById(final RestTemplate restTemplate, final String id) { private ExamineeAccountDetails getExamineeById(final RestTemplate restTemplate, final String id) {
final String url = String.format("/restapi/users/%s/name_username", id); final String url = String.format("/restapi/users/%s", id);
final UserData u = this.apiGet(restTemplate, url, UserData.class); final UserData u = this.apiGet(restTemplate, url, UserData.class);
final Map<String, String> attrs = new HashMap<>(); final Map<String, String> attrs = new HashMap<>();
return new ExamineeAccountDetails( return new ExamineeAccountDetails(
String.valueOf(u.key), String.valueOf(u.key),
u.lastName + ", " + u.firstName, u.lastName + ", " + u.firstName,
u.username, u.login,
"OLAT API does not provide email addresses", u.email,
attrs); attrs);
} }
private SEBRestriction getRestrictionForAssignmentId(final RestTemplate restTemplate, final String id) { private SEBRestriction getRestrictionForAssignmentId(final RestTemplate restTemplate, final String id) {
final String url = String.format("/restapi/assessment_modes/%s/seb_restriction", id); final String url = String.format("/restapi/repo/assessmentmodes/%s/seb", id);
final RestrictionData r = this.apiGet(restTemplate, url, RestrictionData.class); final RestrictionData r = this.apiGet(restTemplate, url, RestrictionData.class);
final HashMap<String, String> additionalAttributes = new HashMap<>(); final HashMap<String, String> additionalAttributes = new HashMap<>();
if (StringUtils.isNotBlank(r.quitLink)) { if (StringUtils.isNotBlank(r.quitLink)) {
@ -362,7 +362,7 @@ public class OlatLmsAPITemplate extends AbstractCachedCourseAccess implements Lm
final String id, final String id,
final SEBRestriction restriction) { final SEBRestriction restriction) {
final String url = String.format("/restapi/assessment_modes/%s/seb_restriction", id); final String url = String.format("/restapi/repo/assessmentmodes/%s/seb", id);
final RestrictionDataPost post = new RestrictionDataPost(); final RestrictionDataPost post = new RestrictionDataPost();
post.browserExamKeys = new ArrayList<>(restriction.browserExamKeys); post.browserExamKeys = new ArrayList<>(restriction.browserExamKeys);
post.configKeys = new ArrayList<>(restriction.configKeys); post.configKeys = new ArrayList<>(restriction.configKeys);
@ -377,7 +377,7 @@ public class OlatLmsAPITemplate extends AbstractCachedCourseAccess implements Lm
} }
private SEBRestriction deleteRestrictionForAssignmentId(final RestTemplate restTemplate, final String id) { private SEBRestriction deleteRestrictionForAssignmentId(final RestTemplate restTemplate, final String id) {
final String url = String.format("/restapi/assessment_modes/%s/seb_restriction", id); final String url = String.format("/restapi/repo/assessmentmodes/%s/seb", id);
final RestrictionData r = this.apiDelete(restTemplate, url, RestrictionData.class); final RestrictionData r = this.apiDelete(restTemplate, url, RestrictionData.class);
// OLAT returns RestrictionData with null values upon deletion. // OLAT returns RestrictionData with null values upon deletion.
// We return it here for consistency, even though SEB server does not need it // We return it here for consistency, even though SEB server does not need it
@ -473,7 +473,7 @@ public class OlatLmsAPITemplate extends AbstractCachedCourseAccess implements Lm
.getOrThrow(); .getOrThrow();
final ClientCredentialsResourceDetails details = new ClientCredentialsResourceDetails(); final ClientCredentialsResourceDetails details = new ClientCredentialsResourceDetails();
details.setAccessTokenUri(lmsSetup.lmsApiUrl + "/restapi/auth/"); details.setAccessTokenUri(lmsSetup.lmsApiUrl + "/restapi/auth/{username}?password={password}");
details.setClientId(plainClientId.toString()); details.setClientId(plainClientId.toString());
details.setClientSecret(plainClientSecret.toString()); details.setClientSecret(plainClientSecret.toString());
@ -489,4 +489,5 @@ public class OlatLmsAPITemplate extends AbstractCachedCourseAccess implements Lm
}); });
} }
} }

View file

@ -20,22 +20,24 @@ public final class OlatLmsData {
/* /*
* OLAT API example: * OLAT API example:
* { * {
* "courseName": "course 1", * "begin": 1624420800000,
* "dateFrom": 1624420800000, * "end": 1624658400000,
* "dateTo": 1624658400000,
* "description": "", * "description": "",
* "key": 6356992, * "key": 6356992,
* repositoryEntryKey: 462324, * repositoryEntryKey: 462324,
* "name": "SEB test" * "name": "SEB test",
* "leadTime": 15,
* "followupTime", 5
* } * }
*/ */
public long key; public long key;
public long repositoryEntryKey; public long repositoryEntryKey;
public String name; public String name;
public String description; public String description;
public String courseName; public Long begin;
public long dateFrom; public Long end;
public long dateTo; public long leadTime;
public long followupTime;
} }
@JsonIgnoreProperties(ignoreUnknown = true) @JsonIgnoreProperties(ignoreUnknown = true)
@ -46,13 +48,15 @@ public final class OlatLmsData {
* "firstName": "OpenOLAT", * "firstName": "OpenOLAT",
* "key": 360448, * "key": 360448,
* "lastName": "Administrator", * "lastName": "Administrator",
* "username": "administrator" * "login": "administrator",
* "email": "admin@example.org"
* } * }
*/ */
public long key; public long key;
public String firstName; public String firstName;
public String lastName; public String lastName;
public String username; public String login;
public String email;
} }
@JsonIgnoreProperties(ignoreUnknown = true) @JsonIgnoreProperties(ignoreUnknown = true)

View file

@ -16,6 +16,7 @@ import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import org.springframework.http.HttpEntity; import org.springframework.http.HttpEntity;
import org.springframework.http.HttpHeaders; import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpMethod;
import org.springframework.http.HttpRequest; import org.springframework.http.HttpRequest;
import org.springframework.http.HttpStatus; import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity; import org.springframework.http.ResponseEntity;
@ -116,14 +117,14 @@ public class OlatLmsRestTemplate extends RestTemplate {
// Authenticate with OLAT and store the received X-OLAT-TOKEN // Authenticate with OLAT and store the received X-OLAT-TOKEN
this.token = "authenticating"; this.token = "authenticating";
final String authUrl = this.details.getAccessTokenUri(); final String authUrl = this.details.getAccessTokenUri();
final Map<String, String> credentials = new HashMap<>(); final Map<String, String> parameters = new HashMap<>();
credentials.put("username", this.details.getClientId()); parameters.put("username", this.details.getClientId());
credentials.put("password", this.details.getClientSecret()); parameters.put("password", this.details.getClientSecret());
final HttpHeaders httpHeaders = new HttpHeaders(); final HttpHeaders httpHeaders = new HttpHeaders();
httpHeaders.set("content-type", "application/json"); httpHeaders.set("content-type", "application/json");
final HttpEntity<Map<String, String>> requestEntity = new HttpEntity<>(credentials, httpHeaders); final HttpEntity<Map<String, String>> requestEntity = new HttpEntity<>(httpHeaders);
try { try {
final ResponseEntity<String> response = this.postForEntity(authUrl, requestEntity, String.class); final ResponseEntity<String> response = this.exchange(authUrl, HttpMethod.GET, requestEntity, String.class, parameters);
final HttpHeaders responseHeaders = response.getHeaders(); final HttpHeaders responseHeaders = response.getHeaders();
if (log.isDebugEnabled()) { if (log.isDebugEnabled()) {

View file

@ -89,7 +89,7 @@ sebserver.feature.exam.seb.screenProctoring.bundled.sebserveraccount.username=SE
#sebserver.feature.config.certificate.enabled=false #sebserver.feature.config.certificate.enabled=false
# #
#sebserver.feature.lms.setup.type.ANS_DELFT.enabled=false #sebserver.feature.lms.setup.type.ANS_DELFT.enabled=false
#sebserver.feature.lms.setup.type.OPEN_OLAT.enabled=false #sebserver.feature.lms.setup.type.OLAT.enabled=false
# #
#sebserver.feature.exam.ask.enabled=false #sebserver.feature.exam.ask.enabled=false
#sebserver.feature.exam.seb.restriction.enabled=false #sebserver.feature.exam.seb.restriction.enabled=false

View file

@ -344,7 +344,7 @@ sebserver.lmssetup.type.MOODLE_PLUGIN=Moodle with SEB Server Plugin
sebserver.lmssetup.type.MOODLE_PLUGIN.tooltip=Moodle with SEB Server integration plugin installed sebserver.lmssetup.type.MOODLE_PLUGIN.tooltip=Moodle with SEB Server integration plugin installed
sebserver.lmssetup.type.OPEN_EDX=Open edX sebserver.lmssetup.type.OPEN_EDX=Open edX
sebserver.lmssetup.type.ANS_DELFT=Ans Delft sebserver.lmssetup.type.ANS_DELFT=Ans Delft
sebserver.lmssetup.type.OPEN_OLAT=Open OLAT sebserver.lmssetup.type.OPEN_OLAT=OLAT
sebserver.lmssetup.list.actions= sebserver.lmssetup.list.actions=
sebserver.lmssetup.list.action.no.modify.privilege=No Access: A Assessment Tool Setup from other institution cannot be modified. sebserver.lmssetup.list.action.no.modify.privilege=No Access: A Assessment Tool Setup from other institution cannot be modified.
@ -1942,6 +1942,7 @@ sebserver.examconfig.props.validation.SEBVersionValidator=At least one SEB Versi
sebserver.examconfig.props.label.signature=Signature sebserver.examconfig.props.label.signature=Signature
sebserver.examconfig.props.label.signature.tooltip=The hash / thumbprint of the certificate used to sign the executable. sebserver.examconfig.props.label.signature.tooltip=The hash / thumbprint of the certificate used to sign the executable.
sebserver.examconfig.props.group.clipboardPolicy=Clipboard Policy
sebserver.examconfig.props.label.clipboardPolicy=Clipboard Policy sebserver.examconfig.props.label.clipboardPolicy=Clipboard Policy
sebserver.examconfig.props.label.clipboardPolicy.tooltip= sebserver.examconfig.props.label.clipboardPolicy.tooltip=
sebserver.examconfig.props.label.clipboardPolicy.0=Allow sebserver.examconfig.props.label.clipboardPolicy.0=Allow