SEBSERV-363 use SDKKey for proctoring session

This commit is contained in:
anhefti 2023-01-16 08:42:02 +01:00
parent c03141c619
commit 826a0a3bfa
5 changed files with 182 additions and 146 deletions

View file

@ -21,7 +21,6 @@ import java.util.stream.Collectors;
import javax.crypto.Mac; import javax.crypto.Mac;
import javax.crypto.spec.SecretKeySpec; import javax.crypto.spec.SecretKeySpec;
import javax.xml.bind.DatatypeConverter;
import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.StringUtils;
import org.joda.time.DateTime; import org.joda.time.DateTime;
@ -42,10 +41,6 @@ import org.springframework.web.client.RestClientResponseException;
import org.springframework.web.client.RestTemplate; import org.springframework.web.client.RestTemplate;
import org.springframework.web.util.UriComponentsBuilder; import org.springframework.web.util.UriComponentsBuilder;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonProperty;
import ch.ethz.seb.sebserver.ClientHttpRequestFactoryService; import ch.ethz.seb.sebserver.ClientHttpRequestFactoryService;
import ch.ethz.seb.sebserver.gbl.Constants; import ch.ethz.seb.sebserver.gbl.Constants;
import ch.ethz.seb.sebserver.gbl.api.API; import ch.ethz.seb.sebserver.gbl.api.API;
@ -56,7 +51,6 @@ import ch.ethz.seb.sebserver.gbl.api.JSONMapper;
import ch.ethz.seb.sebserver.gbl.async.AsyncService; import ch.ethz.seb.sebserver.gbl.async.AsyncService;
import ch.ethz.seb.sebserver.gbl.async.CircuitBreaker; import ch.ethz.seb.sebserver.gbl.async.CircuitBreaker;
import ch.ethz.seb.sebserver.gbl.client.ClientCredentials; import ch.ethz.seb.sebserver.gbl.client.ClientCredentials;
import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringRoomConnection; import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringRoomConnection;
import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringServiceSettings; import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringServiceSettings;
import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringServiceSettings.ProctoringServerType; import ch.ethz.seb.sebserver.gbl.model.exam.ProctoringServiceSettings.ProctoringServerType;
@ -75,10 +69,12 @@ import ch.ethz.seb.sebserver.webservice.servicelayer.dao.RemoteProctoringRoomDAO
import ch.ethz.seb.sebserver.webservice.servicelayer.session.ExamProctoringService; import ch.ethz.seb.sebserver.webservice.servicelayer.session.ExamProctoringService;
import ch.ethz.seb.sebserver.webservice.servicelayer.session.ExamSessionService; import ch.ethz.seb.sebserver.webservice.servicelayer.session.ExamSessionService;
import ch.ethz.seb.sebserver.webservice.servicelayer.session.SEBClientInstructionService; import ch.ethz.seb.sebserver.webservice.servicelayer.session.SEBClientInstructionService;
import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.AdditionalZoomRoomData;
import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.ApplyUserSettingsRequest; import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.ApplyUserSettingsRequest;
import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.CreateMeetingRequest; import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.CreateMeetingRequest;
import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.CreateUserRequest; import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.CreateUserRequest;
import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.MeetingResponse; import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.MeetingResponse;
import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.SDKJWTPayload;
import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.UserResponse; import ch.ethz.seb.sebserver.webservice.servicelayer.session.impl.proctoring.ZoomRoomRequestResponse.UserResponse;
@Lazy @Lazy
@ -90,10 +86,13 @@ public class ZoomProctoringService implements ExamProctoringService {
private static final String TOKEN_ENCODE_ALG = "HmacSHA256"; private static final String TOKEN_ENCODE_ALG = "HmacSHA256";
@Deprecated
private static final String ZOOM_ACCESS_TOKEN_HEADER = private static final String ZOOM_ACCESS_TOKEN_HEADER =
"{\"alg\":\"HS256\",\"typ\":\"JWT\"}"; "{\"alg\":\"HS256\",\"typ\":\"JWT\"}";
@Deprecated
private static final String ZOOM_API_ACCESS_TOKEN_PAYLOAD = private static final String ZOOM_API_ACCESS_TOKEN_PAYLOAD =
"{\"iss\":\"%s\",\"exp\":%s}"; "{\"iss\":\"%s\",\"exp\":%s}";
@Deprecated
private static final String ZOOM_SDK_ACCESS_TOKEN_PAYLOAD = private static final String ZOOM_SDK_ACCESS_TOKEN_PAYLOAD =
"{\"appKey\":\"%s\",\"iat\":%s,\"exp\":%s,\"tokenExp\":%s}"; "{\"appKey\":\"%s\",\"iat\":%s,\"exp\":%s,\"tokenExp\":%s}";
@ -262,28 +261,17 @@ public class ZoomProctoringService implements ExamProctoringService {
remoteProctoringRoom.additionalRoomData, remoteProctoringRoom.additionalRoomData,
AdditionalZoomRoomData.class); AdditionalZoomRoomData.class);
final ClientCredentials credentials = new ClientCredentials( // Note: since SEB Server version 1.5 we work only with SDKKey instead of AppKey which is deprecated
proctoringSettings.appKey,
proctoringSettings.appSecret,
remoteProctoringRoom.joinKey);
final String jwt = this.createSignatureForMeetingAccess(
credentials,
String.valueOf(additionalZoomRoomData.meeting_id),
true);
String sdkJWT = null;
if (StringUtils.isNotBlank(proctoringSettings.sdkKey)) {
final ClientCredentials sdkCredentials = new ClientCredentials( final ClientCredentials sdkCredentials = new ClientCredentials(
proctoringSettings.sdkKey, proctoringSettings.sdkKey,
proctoringSettings.sdkSecret, proctoringSettings.sdkSecret,
remoteProctoringRoom.joinKey); remoteProctoringRoom.joinKey);
sdkJWT = this.createJWTForSDKAccess( final String sdkJWT = this.createSDKJWT(
sdkCredentials, sdkCredentials,
forExam(proctoringSettings)); expiryTimeforExam(proctoringSettings),
} additionalZoomRoomData.meeting_id,
true);
return new ProctoringRoomConnection( return new ProctoringRoomConnection(
ProctoringServerType.ZOOM, ProctoringServerType.ZOOM,
@ -292,10 +280,10 @@ public class ZoomProctoringService implements ExamProctoringService {
additionalZoomRoomData.join_url, additionalZoomRoomData.join_url,
roomName, roomName,
subject, subject,
jwt,
sdkJWT, sdkJWT,
credentials.accessToken, sdkJWT,
credentials.clientId, sdkCredentials.accessToken,
sdkCredentials.clientId,
String.valueOf(additionalZoomRoomData.meeting_id), String.valueOf(additionalZoomRoomData.meeting_id),
this.authorizationService.getUserService().getCurrentUser().getUsername(), this.authorizationService.getUserService().getCurrentUser().getUsername(),
remoteProctoringRoom.additionalRoomData); remoteProctoringRoom.additionalRoomData);
@ -319,32 +307,21 @@ public class ZoomProctoringService implements ExamProctoringService {
remoteProctoringRoom.additionalRoomData, remoteProctoringRoom.additionalRoomData,
AdditionalZoomRoomData.class); AdditionalZoomRoomData.class);
final ClientCredentials credentials = new ClientCredentials(
proctoringSettings.appKey,
proctoringSettings.appSecret,
remoteProctoringRoom.joinKey);
final String signature = this.createSignatureForMeetingAccess(
credentials,
String.valueOf(additionalZoomRoomData.meeting_id),
false);
final ClientConnectionData clientConnection = this.examSessionService final ClientConnectionData clientConnection = this.examSessionService
.getConnectionData(connectionToken) .getConnectionData(connectionToken)
.getOrThrow(); .getOrThrow();
String sdkJWT = null; // Note: since SEB Server version 1.5 we work only with SDKKey instead of AppKey which is deprecated
if (StringUtils.isNotBlank(proctoringSettings.sdkKey)) {
final ClientCredentials sdkCredentials = new ClientCredentials( final ClientCredentials sdkCredentials = new ClientCredentials(
proctoringSettings.sdkKey, proctoringSettings.sdkKey,
proctoringSettings.sdkSecret, proctoringSettings.sdkSecret,
remoteProctoringRoom.joinKey); remoteProctoringRoom.joinKey);
sdkJWT = this.createJWTForSDKAccess( final String sdkJWT = this.createSDKJWT(
sdkCredentials, sdkCredentials,
forExam(proctoringSettings)); expiryTimeforExam(proctoringSettings),
} additionalZoomRoomData.meeting_id,
true);
return new ProctoringRoomConnection( return new ProctoringRoomConnection(
ProctoringServerType.ZOOM, ProctoringServerType.ZOOM,
@ -353,10 +330,10 @@ public class ZoomProctoringService implements ExamProctoringService {
additionalZoomRoomData.join_url, additionalZoomRoomData.join_url,
roomName, roomName,
subject, subject,
signature,
sdkJWT, sdkJWT,
credentials.accessToken, sdkJWT,
credentials.clientId, sdkCredentials.accessToken,
sdkCredentials.clientId,
String.valueOf(additionalZoomRoomData.meeting_id), String.valueOf(additionalZoomRoomData.meeting_id),
clientConnection.clientConnection.userSessionId, clientConnection.clientConnection.userSessionId,
remoteProctoringRoom.additionalRoomData); remoteProctoringRoom.additionalRoomData);
@ -666,9 +643,11 @@ public class ZoomProctoringService implements ExamProctoringService {
} }
} }
private String createJWTForSDKAccess( private String createSDKJWT(
final ClientCredentials sdkCredentials, final ClientCredentials sdkCredentials,
final Long expTime) { final Long expTime,
final long meetingNumber,
final boolean host) {
try { try {
@ -684,16 +663,21 @@ public class ZoomProctoringService implements ExamProctoringService {
// epoch time in seconds // epoch time in seconds
final long secondsNow = Utils.getSecondsNow(); final long secondsNow = Utils.getSecondsNow();
final String sdkKey = sdkCredentials.clientIdAsString();
final String jwtPayload = String.format( final SDKJWTPayload sdkjwtPayload = new SDKJWTPayload(
ZOOM_SDK_ACCESS_TOKEN_PAYLOAD sdkKey,
.replaceAll(" ", "") sdkKey,
.replaceAll("\n", ""), meetingNumber,
sdkCredentials.clientIdAsString(), host ? 1 : 0,
secondsNow, secondsNow,
expTime, expTime,
expTime); expTime);
final String jwtPayload = this.jsonMapper
.writeValueAsString(sdkjwtPayload)
.replaceAll(" ", "")
.replaceAll("\n", "");
if (log.isTraceEnabled()) { if (log.isTraceEnabled()) {
log.trace("Zoom SDK Token payload: {}", jwtPayload); log.trace("Zoom SDK Token payload: {}", jwtPayload);
} }
@ -722,70 +706,93 @@ public class ZoomProctoringService implements ExamProctoringService {
} }
} }
private String createSignatureForMeetingAccess( // private String createJWTForSDKAccess(
final ClientCredentials credentials, // final ClientCredentials sdkCredentials,
final String meetingId, // final Long expTime) {
final boolean host) { //
// try {
//
// final CharSequence decryptedSecret = this.cryptor
// .decrypt(sdkCredentials.secret)
// .getOrThrow();
//
// final StringBuilder builder = new StringBuilder();
// final Encoder urlEncoder = Base64.getUrlEncoder().withoutPadding();
//
// final String jwtHeaderPart = urlEncoder
// .encodeToString(ZOOM_ACCESS_TOKEN_HEADER.getBytes(StandardCharsets.UTF_8));
//
// // epoch time in seconds
// final long secondsNow = Utils.getSecondsNow();
//
// final String jwtPayload = String.format(
// ZOOM_SDK_ACCESS_TOKEN_PAYLOAD
// .replaceAll(" ", "")
// .replaceAll("\n", ""),
// sdkCredentials.clientIdAsString(),
// secondsNow,
// expTime,
// expTime);
//
// if (log.isTraceEnabled()) {
// log.trace("Zoom SDK Token payload: {}", jwtPayload);
// }
//
// final String jwtPayloadPart = urlEncoder
// .encodeToString(jwtPayload.getBytes(StandardCharsets.UTF_8));
//
// final String message = jwtHeaderPart + "." + jwtPayloadPart;
//
// final Mac sha256_HMAC = Mac.getInstance(TOKEN_ENCODE_ALG);
// final SecretKeySpec secret_key = new SecretKeySpec(
// Utils.toByteArray(decryptedSecret),
// TOKEN_ENCODE_ALG);
//
// sha256_HMAC.init(secret_key);
// final String hash = urlEncoder
// .encodeToString(sha256_HMAC.doFinal(Utils.toByteArray(message)));
//
// builder.append(message)
// .append(".")
// .append(hash);
//
// return builder.toString();
// } catch (final Exception e) {
// throw new RuntimeException("Failed to create JWT for Zoom API access: ", e);
// }
// }
try { private long expiryTimeforExam(final ProctoringServiceSettings examProctoring) {
final String apiKey = credentials.clientIdAsString();
final int status = host ? 1 : 0;
final CharSequence decryptedSecret = this.cryptor
.decrypt(credentials.secret)
.getOrThrow();
final Mac hasher = Mac.getInstance("HmacSHA256");
final String ts = Long.toString(System.currentTimeMillis() - 30000);
final String msg = String.format("%s%s%s%d", apiKey, meetingId, ts, status);
hasher.init(new SecretKeySpec(Utils.toByteArray(decryptedSecret), "HmacSHA256"));
final String message = Base64.getEncoder().encodeToString(msg.getBytes());
final byte[] hash = hasher.doFinal(message.getBytes());
final String hashBase64Str = DatatypeConverter.printBase64Binary(hash);
final String tmpString = String.format("%s.%s.%s.%d.%s", apiKey, meetingId, ts, status, hashBase64Str);
final String encodedString = Base64.getEncoder().encodeToString(tmpString.getBytes());
if (log.isTraceEnabled()) {
log.trace("Zoom Meeting signature payload: {}", tmpString);
}
return encodedString.replaceAll("\\=+$", "");
} catch (final Exception e) {
throw new RuntimeException("Failed to create JWT for Zoom meeting access: ", e);
}
}
private long forExam(final ProctoringServiceSettings examProctoring) {
// NOTE: following is the original code that includes the exam end time but seems to make trouble for OLAT // NOTE: following is the original code that includes the exam end time but seems to make trouble for OLAT
final long nowInSeconds = Utils.getSecondsNow(); final long nowInSeconds = Utils.getSecondsNow();
final long nowPlus30MinInSeconds = nowInSeconds + Utils.toSeconds(30 * Constants.MINUTE_IN_MILLIS); // final long nowPlus30MinInSeconds = nowInSeconds + Utils.toSeconds(30 * Constants.MINUTE_IN_MILLIS);
final long nowPlusOneDayInSeconds = nowInSeconds + Utils.toSeconds(Constants.DAY_IN_MILLIS); final long nowPlusOneDayInSeconds = nowInSeconds + Utils.toSeconds(Constants.DAY_IN_MILLIS);
final long nowPlusTwoDayInSeconds = nowInSeconds + Utils.toSeconds(2 * Constants.DAY_IN_MILLIS); // final long nowPlusTwoDayInSeconds = nowInSeconds + Utils.toSeconds(2 * Constants.DAY_IN_MILLIS);
long expTime = nowPlusOneDayInSeconds;
if (examProctoring.examId == null && this.examSessionService.isExamRunning(examProctoring.examId)) {
final Exam exam = this.examSessionService.getRunningExam(examProctoring.examId)
.getOrThrow();
if (exam.endTime != null) {
expTime = Utils.toSeconds(exam.endTime.getMillis());
}
}
// refer to https://marketplace.zoom.us/docs/sdk/native-sdks/auth
// "exp": 0, //JWT expiration date (Min:1800 seconds greater than iat value, Max: 48 hours greater than iat value) in epoch format.
if (expTime > nowPlusTwoDayInSeconds) {
expTime = nowPlusTwoDayInSeconds - 10; // Do not set to max because it is not well defined if max is included or not
} else if (expTime < nowPlus30MinInSeconds) {
expTime = nowPlusOneDayInSeconds;
}
log.debug("**** SDK Token exp time with exam-end-time inclusion would be: {}", expTime);
// long expTime = nowPlusOneDayInSeconds;
// if (examProctoring.examId == null && this.examSessionService.isExamRunning(examProctoring.examId)) {
// final Exam exam = this.examSessionService.getRunningExam(examProctoring.examId)
// .getOrThrow();
// if (exam.endTime != null) {
// expTime = Utils.toSeconds(exam.endTime.getMillis());
// }
// }
// // refer to https://marketplace.zoom.us/docs/sdk/native-sdks/auth
// // "exp": 0, //JWT expiration date (Min:1800 seconds greater than iat value, Max: 48 hours greater than iat value) in epoch format.
// if (expTime >= nowPlusTwoDayInSeconds) {
// expTime = nowPlusTwoDayInSeconds - 10; // Do not set to max because it is not well defined if max is included or not
// } else if (expTime < nowPlus30MinInSeconds) {
// expTime = nowPlusOneDayInSeconds;
// }
//
// log.debug("**** SDK Token exp time with exam-end-time inclusion would be: {}", expTime);
//
//
// NOTE: Set this to the maximum according to https://marketplace.zoom.us/docs/sdk/native-sdks/auth // NOTE: Set this to the maximum according to https://marketplace.zoom.us/docs/sdk/native-sdks/auth
return nowPlusTwoDayInSeconds - 10; // Do not set to max because it is not well defined if max is included or not; //return nowPlusTwoDayInSeconds - 1000; // Do not set to max because it is not well defined if max is included or not;
// NOTE: It seems that since the update of web sdk to SDKToken to 1.7.0, the max is new + one day
return nowPlusOneDayInSeconds - 10;
} }
private final static class ZoomRestTemplate { private final static class ZoomRestTemplate {
@ -1063,30 +1070,4 @@ public class ZoomProctoringService implements ExamProctoringService {
} }
} }
@JsonIgnoreProperties(ignoreUnknown = true)
public static final class AdditionalZoomRoomData {
@JsonProperty("meeting_id")
private final Long meeting_id;
@JsonProperty("user_id")
public final String user_id;
@JsonProperty("start_url")
public final String start_url;
@JsonProperty("join_url")
public final String join_url;
@JsonCreator
public AdditionalZoomRoomData(
@JsonProperty("meeting_id") final Long meeting_id,
@JsonProperty("user_id") final String user_id,
@JsonProperty("start_url") final String start_url,
@JsonProperty("join_url") final String join_url) {
this.meeting_id = meeting_id;
this.user_id = user_id;
this.start_url = start_url;
this.join_url = join_url;
}
}
} }

View file

@ -20,6 +20,61 @@ import com.fasterxml.jackson.annotation.JsonProperty;
public interface ZoomRoomRequestResponse { public interface ZoomRoomRequestResponse {
// @formatter:off // @formatter:off
@JsonIgnoreProperties(ignoreUnknown = true)
public static final class SDKJWTPayload {
public final String appKey;
public final String sdkKey;
public final long mn;
public final int role;
public final long iat;
public final long exp;
public final long tokenExp;
public SDKJWTPayload(
final String appKey,
final String sdkKey,
final long mn,
final int
role,
final long iat,
final long exp,
final long tokenExp) {
this.appKey = appKey;
this.sdkKey = sdkKey;
this.mn = mn;
this.role = role;
this.iat = iat;
this.exp = exp;
this.tokenExp = tokenExp;
}
}
@JsonIgnoreProperties(ignoreUnknown = true)
public static final class AdditionalZoomRoomData {
@JsonProperty("meeting_id")
public final Long meeting_id;
@JsonProperty("user_id")
public final String user_id;
@JsonProperty("start_url")
public final String start_url;
@JsonProperty("join_url")
public final String join_url;
@JsonCreator
public AdditionalZoomRoomData(
@JsonProperty("meeting_id") final Long meeting_id,
@JsonProperty("user_id") final String user_id,
@JsonProperty("start_url") final String start_url,
@JsonProperty("join_url") final String join_url) {
this.meeting_id = meeting_id;
this.user_id = user_id;
this.start_url = start_url;
this.join_url = join_url;
}
}
//https://marketplace.zoom.us/docs/api-reference/zoom-api/users/users //https://marketplace.zoom.us/docs/api-reference/zoom-api/users/users
@JsonIgnoreProperties(ignoreUnknown = true) @JsonIgnoreProperties(ignoreUnknown = true)
static class UserPageResponse { static class UserPageResponse {

View file

@ -82,7 +82,7 @@
console.log('INIT SUCCESS') console.log('INIT SUCCESS')
ZoomMtg.join({ ZoomMtg.join({
signature: signature, signature: signature,
apiKey: API_KEY, sdkKey: API_KEY,
meetingNumber: config.meetingNumber, meetingNumber: config.meetingNumber,
userName: config.userName, userName: config.userName,
passWord: config.passWord, passWord: config.passWord,

View file

@ -38,7 +38,7 @@ sebserver.gui.webservice.edx-lms-enabled=true
sebserver.gui.webservice.moodle-lms-enabled=true sebserver.gui.webservice.moodle-lms-enabled=true
sebserver.gui.seb.client.config.download.filename=SEBServerSettings.seb sebserver.gui.seb.client.config.download.filename=SEBServerSettings.seb
sebserver.gui.seb.exam.config.download.filename=SEBExamSettings.seb sebserver.gui.seb.exam.config.download.filename=SEBExamSettings.seb
sebserver.gui.proctoring.zoom.websdk.version=1.9.8 sebserver.gui.proctoring.zoom.websdk.version=2.8.0
sebserver.gui.filter.date.from.years=2 sebserver.gui.filter.date.from.years=2
# remote proctoring # remote proctoring

View file

@ -153,7 +153,7 @@ public class ZoomWindowScriptResolverTest {
+ " console.log('INIT SUCCESS')\r\n" + " console.log('INIT SUCCESS')\r\n"
+ " ZoomMtg.join({\r\n" + " ZoomMtg.join({\r\n"
+ " signature: signature,\r\n" + " signature: signature,\r\n"
+ " apiKey: API_KEY,\r\n" + " sdkKey: API_KEY,\r\n"
+ " meetingNumber: config.meetingNumber,\r\n" + " meetingNumber: config.meetingNumber,\r\n"
+ " userName: config.userName,\r\n" + " userName: config.userName,\r\n"
+ " passWord: config.passWord,\r\n" + " passWord: config.passWord,\r\n"