simplified LMS API
This commit is contained in:
parent
0b00995aa7
commit
213cf443e1
8 changed files with 170 additions and 157 deletions
|
@ -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,
|
||||||
|
|
|
@ -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 */
|
||||||
|
|
|
@ -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);
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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());
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue