simplified LMS API

This commit is contained in:
anhefti 2021-05-17 21:19:01 +02:00
parent 0b00995aa7
commit 213cf443e1
8 changed files with 170 additions and 157 deletions

View file

@ -806,11 +806,9 @@ public class ExamDAOImpl implements ExamDAO {
// get and map quizzes // get and map quizzes
final Map<String, QuizData> quizzes = this.lmsAPIService final Map<String, QuizData> quizzes = this.lmsAPIService
.getLmsAPITemplate(lmsSetupId) .getLmsAPITemplate(lmsSetupId)
.map(template -> getQuizzesFromLMS(template, recordMapping.keySet())) .flatMap(template -> template.getQuizzes(recordMapping.keySet()))
.onError(error -> log.error("Failed to get quizzes for exams: ", error)) .getOrElse(() -> Collections.emptyList())
.getOr(Collections.emptyList())
.stream() .stream()
.flatMap(Result::skipOnError)
.collect(Collectors.toMap(q -> q.id, Function.identity())); .collect(Collectors.toMap(q -> q.id, Function.identity()));
if (records.size() != quizzes.size()) { if (records.size() != quizzes.size()) {
@ -858,18 +856,6 @@ public class ExamDAOImpl implements ExamDAO {
}); });
} }
private Collection<Result<QuizData>> getQuizzesFromLMS(
final LmsAPITemplate template,
final Set<String> ids) {
try {
return template.getQuizzes(ids);
} catch (final Exception e) {
log.error("Unexpected error while using LmsAPITemplate to get quizzes: ", e);
return Collections.emptyList();
}
}
private QuizData getQuizData( private QuizData getQuizData(
final Map<String, QuizData> quizzes, final Map<String, QuizData> quizzes,
final String externalId, final String externalId,

View file

@ -12,7 +12,7 @@ import java.util.Collection;
import java.util.List; import java.util.List;
import java.util.Set; import java.util.Set;
import ch.ethz.seb.sebserver.gbl.async.MemoizingCircuitBreaker; import ch.ethz.seb.sebserver.gbl.async.CircuitBreaker;
import ch.ethz.seb.sebserver.gbl.model.exam.Chapters; import ch.ethz.seb.sebserver.gbl.model.exam.Chapters;
import ch.ethz.seb.sebserver.gbl.model.exam.Exam; import ch.ethz.seb.sebserver.gbl.model.exam.Exam;
import ch.ethz.seb.sebserver.gbl.model.exam.QuizData; import ch.ethz.seb.sebserver.gbl.model.exam.QuizData;
@ -22,6 +22,7 @@ import ch.ethz.seb.sebserver.gbl.model.institution.LmsSetupTestResult;
import ch.ethz.seb.sebserver.gbl.model.user.ExamineeAccountDetails; import ch.ethz.seb.sebserver.gbl.model.user.ExamineeAccountDetails;
import ch.ethz.seb.sebserver.gbl.util.Result; import ch.ethz.seb.sebserver.gbl.util.Result;
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.FilterMap; import ch.ethz.seb.sebserver.webservice.servicelayer.dao.FilterMap;
import ch.ethz.seb.sebserver.webservice.servicelayer.lms.impl.AbstractCachedCourseAccess;
import ch.ethz.seb.sebserver.webservice.servicelayer.lms.impl.AbstractCourseAccess; import ch.ethz.seb.sebserver.webservice.servicelayer.lms.impl.AbstractCourseAccess;
/** Defines an LMS API access template to build SEB Server LMS integration. /** Defines an LMS API access template to build SEB Server LMS integration.
@ -41,28 +42,18 @@ import ch.ethz.seb.sebserver.webservice.servicelayer.lms.impl.AbstractCourseAcce
* All course API requests of this template shall not block and return as fast as possible * All course API requests of this template shall not block and return as fast as possible
* with the best result it can provide for the time on that the request was made. * with the best result it can provide for the time on that the request was made.
* </p> * </p>
* Each request to a remote LMS shall be executed within a protected call such that the
* request don't block the API call as well as do not attack the remote LMS with endless
* requests on failure.</br>
* Therefore the abstract class {@link AbstractCourseAccess} defines protected calls
* for different API calls by using {@link CircuitBreaker}. documentation on the class for
* more information.
* </p>
* Since the course API requests course data from potentially thousands of existing and * Since the course API requests course data from potentially thousands of existing and
* active courses, the course API can implement some caches if needed.</br> * active courses, the course API can implement some short time caches if needed.</br>
* A cache in the course API has manly two purposes; The first and prior purpose is to * The abstract class {@link AbstractCachedCourseAccess} defines such a short time
* be able to provide course data as fast as possible even if the LMS is not available or * cache for all implementing classes using EH-Cache. See documentation on the class for
* busy at the time. The second purpose is to guarantee fast data access for the system * more information.
* if this is needed and data actuality has second priority.</br>
* Therefore usual get quiz data functions like {@link #getQuizzes(FilterMap filterMap) },
* {@link #getQuizzes(Set<String> ids) } and {@link #getQuiz(final String id) }
* shall always first try to connect to the LMS and request the specified data from the LMS.
* If this succeeds the cache shall be updated with the received quizzes data and return them.
* If this is not possible within a certain time, the implementation shall get as much of the
* requested data from the cache and return them to the caller to not block the call too long
* and allow the caller to return fast and present as much data as possible.</br>
* This can be done with a {@link MemoizingCircuitBreaker} or a simple {@link CircuitBreaker}
* with a separated cache, for example. The abstract implementation; {@link AbstractCourseAccess}
* provides already defined wrapped circuit breaker for each call. To use it, just extend the
* abstract class and implement the needed suppliers.</br>
* On the other hand, dedicated cache access functions like {@link #getQuizzesFromCache(Set<String> ids) }
* shall always first look into the cache to geht the requested data and if not
* available, call the LMS and request the data from the LMS. If partial data is needed to get
* be requested from the LMS, this functions shall also update the catch with the requested
* and cache missed data afterwards.
* </p> * </p>
* <b>SEB restriction API</b></br> * <b>SEB restriction API</b></br>
* For this API we need no caching since this is mostly about pushing data to the LMS for the LMS * For this API we need no caching since this is mostly about pushing data to the LMS for the LMS
@ -120,15 +111,14 @@ public interface LmsAPITemplate {
Result<List<QuizData>> getQuizzes(FilterMap filterMap); Result<List<QuizData>> getQuizzes(FilterMap filterMap);
/** Get all {@link QuizData } for the set of {@link QuizData } identifiers from LMS API in a collection /** Get all {@link QuizData } for the set of {@link QuizData } identifiers from LMS API in a collection
* of Result. If particular quiz cannot be loaded because of errors or deletion, * of Result. If particular quizzes cannot be loaded because of errors or deletion,
* the Result will have an error reference. * the the referencing QuizData will not be in the resulting list and an error is logged.
* *
* @param ids the Set of Quiz identifiers to get the {@link QuizData } for * @param ids the Set of Quiz identifiers to get the {@link QuizData } for
* @return Collection of all {@link QuizData } from the given id set */ * @return Collection of all {@link QuizData } from the given id set */
Collection<Result<QuizData>> getQuizzes(Set<String> ids); Result<Collection<QuizData>> getQuizzes(Set<String> ids);
/** Get the quiz data with specified identifier. /** Get the quiz data with specified identifier.
*
* *
* @param id the quiz data identifier * @param id the quiz data identifier
* @return Result refer to the quiz data or to an error when happened */ * @return Result refer to the quiz data or to an error when happened */

View file

@ -8,6 +8,7 @@
package ch.ethz.seb.sebserver.webservice.servicelayer.lms.impl; package ch.ethz.seb.sebserver.webservice.servicelayer.lms.impl;
import java.util.Collection;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.Set; import java.util.Set;
@ -44,7 +45,11 @@ public abstract class AbstractCourseAccess {
} }
/** CircuitBreaker for protected quiz and course data requests */ /** CircuitBreaker for protected quiz and course data requests */
protected final CircuitBreaker<List<QuizData>> quizzesRequest; protected final CircuitBreaker<List<QuizData>> allQuizzesRequest;
/** CircuitBreaker for protected quiz and course data requests */
protected final CircuitBreaker<Collection<QuizData>> quizzesRequest;
/** CircuitBreaker for protected quiz and course data requests */
protected final CircuitBreaker<QuizData> quizRequest;
/** CircuitBreaker for protected chapter data requests */ /** CircuitBreaker for protected chapter data requests */
protected final CircuitBreaker<Chapters> chaptersRequest; protected final CircuitBreaker<Chapters> chaptersRequest;
/** CircuitBreaker for protected examinee account details requests */ /** CircuitBreaker for protected examinee account details requests */
@ -54,6 +59,20 @@ public abstract class AbstractCourseAccess {
final AsyncService asyncService, final AsyncService asyncService,
final Environment environment) { final Environment environment) {
this.allQuizzesRequest = asyncService.createCircuitBreaker(
environment.getProperty(
"sebserver.webservice.circuitbreaker.quizzesRequest.attempts",
Integer.class,
3),
environment.getProperty(
"sebserver.webservice.circuitbreaker.quizzesRequest.blockingTime",
Long.class,
Constants.MINUTE_IN_MILLIS),
environment.getProperty(
"sebserver.webservice.circuitbreaker.quizzesRequest.timeToRecover",
Long.class,
Constants.MINUTE_IN_MILLIS));
this.quizzesRequest = asyncService.createCircuitBreaker( this.quizzesRequest = asyncService.createCircuitBreaker(
environment.getProperty( environment.getProperty(
"sebserver.webservice.circuitbreaker.quizzesRequest.attempts", "sebserver.webservice.circuitbreaker.quizzesRequest.attempts",
@ -68,6 +87,20 @@ public abstract class AbstractCourseAccess {
Long.class, Long.class,
Constants.MINUTE_IN_MILLIS)); Constants.MINUTE_IN_MILLIS));
this.quizRequest = asyncService.createCircuitBreaker(
environment.getProperty(
"sebserver.webservice.circuitbreaker.quizzesRequest.attempts",
Integer.class,
3),
environment.getProperty(
"sebserver.webservice.circuitbreaker.quizzesRequest.blockingTime",
Long.class,
Constants.MINUTE_IN_MILLIS),
environment.getProperty(
"sebserver.webservice.circuitbreaker.quizzesRequest.timeToRecover",
Long.class,
Constants.MINUTE_IN_MILLIS));
this.chaptersRequest = asyncService.createCircuitBreaker( this.chaptersRequest = asyncService.createCircuitBreaker(
environment.getProperty( environment.getProperty(
"sebserver.webservice.circuitbreaker.chaptersRequest.attempts", "sebserver.webservice.circuitbreaker.chaptersRequest.attempts",
@ -97,8 +130,18 @@ public abstract class AbstractCourseAccess {
Constants.SECOND_IN_MILLIS * 10)); Constants.SECOND_IN_MILLIS * 10));
} }
public Result<List<QuizData>> getQuizzes(final FilterMap filterMap) { public Result<List<QuizData>> protectedQuizzesRequest(final FilterMap filterMap) {
return this.quizzesRequest.protectedRun(allQuizzesSupplier(filterMap)); return this.allQuizzesRequest.protectedRun(allQuizzesSupplier(filterMap));
}
public Collection<QuizData> protectedQuizzesRequest(final Set<String> ids) {
return this.quizzesRequest.protectedRun(quizzesSupplier(ids))
.onError(error -> log.error("Failed to get QuizData for ids: ", error))
.getOrElse(() -> Collections.emptyList());
}
public Result<QuizData> protectedQuizRequest(final String id) {
return this.quizRequest.protectedRun(quizSupplier(id));
} }
public Result<ExamineeAccountDetails> getExamineeAccountDetails(final String examineeSessionId) { public Result<ExamineeAccountDetails> getExamineeAccountDetails(final String examineeSessionId) {
@ -116,7 +159,7 @@ public abstract class AbstractCourseAccess {
.getOr(examineeSessionId); .getOr(examineeSessionId);
} }
protected Result<Chapters> getCourseChapters(final String courseId) { public Result<Chapters> getCourseChapters(final String courseId) {
return this.chaptersRequest.protectedRun(getCourseChaptersSupplier(courseId)); return this.chaptersRequest.protectedRun(getCourseChaptersSupplier(courseId));
} }
@ -134,12 +177,15 @@ public abstract class AbstractCourseAccess {
Collections.emptyMap()); Collections.emptyMap());
} }
/** Provides a supplier for the quiz data request to use within the circuit breaker */
protected abstract Supplier<List<QuizData>> quizzesSupplier(final Set<String> ids);
/** Provides a supplier to supply request to use within the circuit breaker */ /** Provides a supplier to supply request to use within the circuit breaker */
protected abstract Supplier<List<QuizData>> allQuizzesSupplier(final FilterMap filterMap); protected abstract Supplier<List<QuizData>> allQuizzesSupplier(final FilterMap filterMap);
/** Provides a supplier for the quiz data request to use within the circuit breaker */
protected abstract Supplier<Collection<QuizData>> quizzesSupplier(final Set<String> ids);
/** Provides a supplier for the quiz data request to use within the circuit breaker */
protected abstract Supplier<QuizData> quizSupplier(final String id);
/** Provides a supplier for the course chapter data request to use within the circuit breaker */ /** Provides a supplier for the course chapter data request to use within the circuit breaker */
protected abstract Supplier<Chapters> getCourseChaptersSupplier(final String courseId); protected abstract Supplier<Chapters> getCourseChaptersSupplier(final String courseId);

View file

@ -12,12 +12,10 @@ import java.net.URL;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
import java.util.Collection; import java.util.Collection;
import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set; import java.util.Set;
import java.util.function.Supplier; import java.util.function.Supplier;
import java.util.stream.Collectors; import java.util.stream.Collectors;
@ -198,41 +196,9 @@ final class OpenEdxCourseAccess extends AbstractCachedCourseAccess {
.getOrThrow(); .getOrThrow();
} }
public Collection<Result<QuizData>> getQuizzesFromCache(final Set<String> ids) { @Override
final HashSet<String> leftIds = new HashSet<>(ids); protected Supplier<QuizData> quizSupplier(final String id) {
final Collection<Result<QuizData>> result = new ArrayList<>(); return () -> {
ids.stream()
.map(this::getQuizFromCache)
.forEach(q -> {
if (q != null) {
leftIds.remove(q.id);
result.add(Result.of(q));
}
});
if (!leftIds.isEmpty()) {
super.quizzesRequest.protectedRun(this.quizzesSupplier(leftIds))
.onError(error -> log.error("Failed to get quizzes by ids: ", error))
.getOrElse(() -> Collections.emptyList())
.stream()
.forEach(q -> {
leftIds.remove(q.id);
result.add(Result.of(q));
});
}
if (!leftIds.isEmpty()) {
leftIds.forEach(q -> result.add(Result.ofError(new NoSuchElementException())));
}
return result;
}
public QuizData getQuizFromCache(final String id) {
return super.getFromCache(id);
}
public QuizData getQuizFromLMS(final String id) {
final LmsSetup lmsSetup = getApiTemplateDataSupplier().getLmsSetup(); final LmsSetup lmsSetup = getApiTemplateDataSupplier().getLmsSetup();
final String externalStartURI = getExternalLMSServerAddress(lmsSetup); final String externalStartURI = getExternalLMSServerAddress(lmsSetup);
final QuizData quizData = quizDataOf( final QuizData quizData = quizDataOf(
@ -244,11 +210,11 @@ final class OpenEdxCourseAccess extends AbstractCachedCourseAccess {
super.putToCache(quizData); super.putToCache(quizData);
} }
return quizData; return quizData;
};
} }
@Override @Override
protected Supplier<List<QuizData>> quizzesSupplier(final Set<String> ids) { protected Supplier<Collection<QuizData>> quizzesSupplier(final Set<String> ids) {
if (ids.size() == 1) { if (ids.size() == 1) {
return () -> { return () -> {
@ -295,6 +261,31 @@ final class OpenEdxCourseAccess extends AbstractCachedCourseAccess {
}; };
} }
public Result<Collection<QuizData>> getQuizzesFromCache(final Set<String> ids) {
return Result.tryCatch(() -> {
final HashSet<String> leftIds = new HashSet<>(ids);
final Collection<QuizData> result = new ArrayList<>();
ids.stream()
.map(this::getQuizFromCache)
.forEach(q -> {
if (q != null) {
leftIds.remove(q.id);
result.add(q);
}
});
if (!leftIds.isEmpty()) {
result.addAll(super.protectedQuizzesRequest(leftIds));
}
return result;
});
}
public QuizData getQuizFromCache(final String id) {
return super.getFromCache(id);
}
private ArrayList<QuizData> collectQuizzes(final OAuth2RestTemplate restTemplate, final Set<String> ids) { private ArrayList<QuizData> collectQuizzes(final OAuth2RestTemplate restTemplate, final Set<String> ids) {
final LmsSetup lmsSetup = getApiTemplateDataSupplier().getLmsSetup(); final LmsSetup lmsSetup = getApiTemplateDataSupplier().getLmsSetup();
final String externalStartURI = getExternalLMSServerAddress(lmsSetup); final String externalStartURI = getExternalLMSServerAddress(lmsSetup);

View file

@ -74,7 +74,7 @@ final class OpenEdxLmsAPITemplate implements LmsAPITemplate {
@Override @Override
public Result<List<QuizData>> getQuizzes(final FilterMap filterMap) { public Result<List<QuizData>> getQuizzes(final FilterMap filterMap) {
return this.openEdxCourseAccess return this.openEdxCourseAccess
.getQuizzes(filterMap) .protectedQuizzesRequest(filterMap)
.map(quizzes -> quizzes.stream() .map(quizzes -> quizzes.stream()
.filter(LmsAPIService.quizFilterPredicate(filterMap)) .filter(LmsAPIService.quizFilterPredicate(filterMap))
.collect(Collectors.toList())); .collect(Collectors.toList()));
@ -82,18 +82,16 @@ final class OpenEdxLmsAPITemplate implements LmsAPITemplate {
@Override @Override
public Result<QuizData> getQuiz(final String id) { public Result<QuizData> getQuiz(final String id) {
return Result.tryCatch(() -> {
final QuizData quizFromCache = this.openEdxCourseAccess.getQuizFromCache(id); final QuizData quizFromCache = this.openEdxCourseAccess.getQuizFromCache(id);
if (quizFromCache != null) { if (quizFromCache != null) {
return quizFromCache; return Result.of(quizFromCache);
} }
return this.openEdxCourseAccess.getQuizFromLMS(id); return this.openEdxCourseAccess.protectedQuizRequest(id);
});
} }
@Override @Override
public Collection<Result<QuizData>> getQuizzes(final Set<String> ids) { public Result<Collection<QuizData>> getQuizzes(final Set<String> ids) {
return this.openEdxCourseAccess.getQuizzesFromCache(ids); return this.openEdxCourseAccess.getQuizzesFromCache(ids);
} }

View file

@ -172,7 +172,8 @@ public class MockupLmsAPITemplate implements LmsAPITemplate {
} }
@Override @Override
public Collection<Result<QuizData>> getQuizzes(final Set<String> ids) { public Result<Collection<QuizData>> getQuizzes(final Set<String> ids) {
return Result.tryCatch(() -> {
if (!authenticate()) { if (!authenticate()) {
throw new IllegalArgumentException("Wrong clientId or secret"); throw new IllegalArgumentException("Wrong clientId or secret");
} }
@ -181,8 +182,8 @@ public class MockupLmsAPITemplate implements LmsAPITemplate {
.stream() .stream()
.map(this::getExternalAddressAlias) .map(this::getExternalAddressAlias)
.filter(mock -> ids.contains(mock.id)) .filter(mock -> ids.contains(mock.id))
.map(Result::of)
.collect(Collectors.toList()); .collect(Collectors.toList());
});
} }
@Override @Override

View file

@ -14,7 +14,6 @@ import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set; import java.util.Set;
import java.util.function.Function; import java.util.function.Function;
import java.util.function.Supplier; import java.util.function.Supplier;
@ -245,16 +244,13 @@ public class MoodleCourseAccess extends AbstractCourseAccess {
} }
} }
// get from LMS // get from LMS in protected request
final Set<String> ids = Stream.of(id).collect(Collectors.toSet()); return super.protectedQuizRequest(id).getOrThrow();
return super.quizzesRequest
.protectedRun(quizzesSupplier(ids))
.getOrThrow()
.get(0);
}); });
} }
public Collection<Result<QuizData>> getQuizzesFromCache(final Set<String> ids) { public Result<Collection<QuizData>> getQuizzesFromCache(final Set<String> ids) {
return Result.tryCatch(() -> {
final List<QuizData> cached = getCached(); final List<QuizData> cached = getCached();
final List<QuizData> available = (cached != null) final List<QuizData> available = (cached != null)
? cached ? cached
@ -278,19 +274,24 @@ public class MoodleCourseAccess extends AbstractCourseAccess {
} }
} }
return ids return quizMapping.values();
.stream()
.map(id -> { });
final QuizData q = quizMapping.get(id);
return (q == null)
? Result.<QuizData> ofError(new NoSuchElementException("Quiz with id: " + id))
: Result.of(q);
})
.collect(Collectors.toList());
} }
@Override @Override
protected Supplier<List<QuizData>> quizzesSupplier(final Set<String> ids) { protected Supplier<QuizData> quizSupplier(final String id) {
return () -> {
final Set<String> ids = Stream.of(id).collect(Collectors.toSet());
return getRestTemplate()
.map(template -> getQuizzesForIds(template, ids))
.getOr(Collections.emptyList())
.get(0);
};
}
@Override
protected Supplier<Collection<QuizData>> quizzesSupplier(final Set<String> ids) {
return () -> getRestTemplate() return () -> getRestTemplate()
.map(template -> getQuizzesForIds(template, ids)) .map(template -> getQuizzesForIds(template, ids))
.getOr(Collections.emptyList()); .getOr(Collections.emptyList());

View file

@ -86,7 +86,7 @@ public class MoodleLmsAPITemplate implements LmsAPITemplate {
@Override @Override
public Result<List<QuizData>> getQuizzes(final FilterMap filterMap) { public Result<List<QuizData>> getQuizzes(final FilterMap filterMap) {
return this.moodleCourseAccess return this.moodleCourseAccess
.getQuizzes(filterMap) .protectedQuizzesRequest(filterMap)
.map(quizzes -> quizzes.stream() .map(quizzes -> quizzes.stream()
.filter(LmsAPIService.quizFilterPredicate(filterMap)) .filter(LmsAPIService.quizFilterPredicate(filterMap))
.collect(Collectors.toList())); .collect(Collectors.toList()));
@ -98,7 +98,7 @@ public class MoodleLmsAPITemplate implements LmsAPITemplate {
} }
@Override @Override
public Collection<Result<QuizData>> getQuizzes(final Set<String> ids) { public Result<Collection<QuizData>> getQuizzes(final Set<String> ids) {
return this.moodleCourseAccess.getQuizzesFromCache(ids); return this.moodleCourseAccess.getQuizzesFromCache(ids);
} }