crate work-around for SEB restriction API

This commit is contained in:
anhefti 2019-11-25 11:22:21 +01:00
parent 3772ad754d
commit 2fd7ce47b8
3 changed files with 113 additions and 33 deletions

View file

@ -42,17 +42,20 @@ public class OpenEdxCourseRestriction {
private final LmsSetup lmsSetup; private final LmsSetup lmsSetup;
private final JSONMapper jsonMapper; private final JSONMapper jsonMapper;
private final OpenEdxRestTemplateFactory openEdxRestTemplateFactory; private final OpenEdxRestTemplateFactory openEdxRestTemplateFactory;
private final int restrictionAPIPushCount;
private OAuth2RestTemplate restTemplate; private OAuth2RestTemplate restTemplate;
protected OpenEdxCourseRestriction( protected OpenEdxCourseRestriction(
final LmsSetup lmsSetup, final LmsSetup lmsSetup,
final JSONMapper jsonMapper, final JSONMapper jsonMapper,
final OpenEdxRestTemplateFactory openEdxRestTemplateFactory) { final OpenEdxRestTemplateFactory openEdxRestTemplateFactory,
final int restrictionAPIPushCount) {
this.lmsSetup = lmsSetup; this.lmsSetup = lmsSetup;
this.jsonMapper = jsonMapper; this.jsonMapper = jsonMapper;
this.openEdxRestTemplateFactory = openEdxRestTemplateFactory; this.openEdxRestTemplateFactory = openEdxRestTemplateFactory;
this.restrictionAPIPushCount = restrictionAPIPushCount;
} }
LmsSetupTestResult initAPIAccess() { LmsSetupTestResult initAPIAccess() {
@ -140,24 +143,9 @@ public class OpenEdxCourseRestriction {
log.debug("PUT SEB Client restriction on course: {} : {}", courseId, restriction); log.debug("PUT SEB Client restriction on course: {} : {}", courseId, restriction);
} }
return handleSebRestriction(() -> { return handleSebRestriction(processSebRestrictionUpdate(pushSebRestrictionFunction(
final String url = this.lmsSetup.lmsApiUrl + getSebRestrictionUrl(courseId); restriction,
final HttpHeaders httpHeaders = new HttpHeaders(); courseId)));
httpHeaders.add(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE);
final OpenEdxCourseRestrictionData confirm = this.restTemplate.exchange(
url,
HttpMethod.PUT,
new HttpEntity<>(toJson(restriction), httpHeaders),
OpenEdxCourseRestrictionData.class)
.getBody();
if (log.isDebugEnabled()) {
log.debug("Successfully PUT SEB Client restriction on course: {} : {}", courseId, confirm);
}
return true;
});
} }
Result<Boolean> deleteSebRestriction(final String courseId) { Result<Boolean> deleteSebRestriction(final String courseId) {
@ -166,24 +154,97 @@ public class OpenEdxCourseRestriction {
log.debug("DELETE SEB Client restriction on course: {}", courseId); log.debug("DELETE SEB Client restriction on course: {}", courseId);
} }
return handleSebRestriction(() -> { return handleSebRestriction(processSebRestrictionUpdate(deleteSebRestrictionFunction(courseId)));
final String url = this.lmsSetup.lmsApiUrl + getSebRestrictionUrl(courseId); }
final ResponseEntity<Object> exchange = this.restTemplate.exchange(
url,
HttpMethod.DELETE,
new HttpEntity<>(new HttpHeaders()),
Object.class);
if (exchange.getStatusCode() == HttpStatus.NO_CONTENT) { private BooleanSupplier processSebRestrictionUpdate(final BooleanSupplier restrictionUpdate) {
if (log.isDebugEnabled()) { return () -> {
log.debug("Successfully PUT SEB Client restriction on course: {}", courseId); if (this.restrictionAPIPushCount > 0) {
// NOTE: This is a temporary work-around for SEB Restriction API within Open edX SEB integration plugin to
// apply on load-balanced infrastructure or infrastructure that has several layers of cache.
// The reason for this is that the API (Open edX system) internally don't apply a resource-change that is
// done within HTTP API call immediately from an outside perspective.
// After a resource-change on the API is done, the system toggles between the old and the new resource
// while constantly calling GET. This usually happens for about a minute or two then it stabilizes on the new resource
//
// This may source on load-balancing or internally caching on Open edX side.
// To mitigate this effect the SEB Server can be configured to apply a resource-change on the
// API several times in a row to flush as match caches and reach as match as possible server instances.
//
// Since this is a brute-force method to mitigate the problem, this should only be a temporary
// work-around until a better solution on Open edX SEB integration side has been found and applied.
log.warn("SEB restriction update with multiple API push "
+ "(this is a temporary work-around for SEB Restriction API within Open edX SEB integration plugin)");
for (int i = 0; i < this.restrictionAPIPushCount; i++) {
if (!restrictionUpdate.getAsBoolean()) {
Result.ofRuntimeError(
"Failed to process SEB restriction update. See logs for more information");
}
} }
return true; return true;
} else { } else {
throw new RuntimeException("Unexpected response for deletion: " + exchange); return restrictionUpdate.getAsBoolean();
} }
}); };
}
private BooleanSupplier pushSebRestrictionFunction(
final OpenEdxCourseRestrictionData restriction,
final String courseId) {
final String url = this.lmsSetup.lmsApiUrl + getSebRestrictionUrl(courseId);
final HttpHeaders httpHeaders = new HttpHeaders();
httpHeaders.add(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE);
return () -> {
try {
final OpenEdxCourseRestrictionData body = this.restTemplate.exchange(
url,
HttpMethod.PUT,
new HttpEntity<>(toJson(restriction), httpHeaders),
OpenEdxCourseRestrictionData.class)
.getBody();
if (log.isDebugEnabled()) {
log.debug("Successfully PUT SEB Client restriction on course: {} : {}", courseId, body);
}
} catch (final Exception e) {
log.error("Unexpected error while trying to call API for PUT. Course: ", courseId, e);
return false;
}
return true;
};
}
private BooleanSupplier deleteSebRestrictionFunction(final String courseId) {
final String url = this.lmsSetup.lmsApiUrl + getSebRestrictionUrl(courseId);
return () -> {
try {
final ResponseEntity<Object> exchange = this.restTemplate.exchange(
url,
HttpMethod.DELETE,
new HttpEntity<>(new HttpHeaders()),
Object.class);
if (exchange.getStatusCode() == HttpStatus.NO_CONTENT) {
if (log.isDebugEnabled()) {
log.debug("Successfully PUT SEB Client restriction on course: {}", courseId);
}
} else {
log.error("Unexpected response for deletion: {}", exchange);
return false;
}
} catch (final Exception e) {
log.error("Unexpected error while trying to call API for DELETE. Course: ", courseId, e);
return false;
}
return true;
};
} }
private Result<Boolean> handleSebRestriction(final BooleanSupplier task) { private Result<Boolean> handleSebRestriction(final BooleanSupplier task) {

View file

@ -36,6 +36,7 @@ public class OpenEdxLmsAPITemplateFactory {
private final ClientCredentialService clientCredentialService; private final ClientCredentialService clientCredentialService;
private final ClientHttpRequestFactoryService clientHttpRequestFactoryService; private final ClientHttpRequestFactoryService clientHttpRequestFactoryService;
private final String[] alternativeTokenRequestPaths; private final String[] alternativeTokenRequestPaths;
private final int restrictionAPIPushCount;
protected OpenEdxLmsAPITemplateFactory( protected OpenEdxLmsAPITemplateFactory(
final JSONMapper jsonMapper, final JSONMapper jsonMapper,
@ -43,7 +44,8 @@ public class OpenEdxLmsAPITemplateFactory {
final AsyncService asyncService, final AsyncService asyncService,
final ClientCredentialService clientCredentialService, final ClientCredentialService clientCredentialService,
final ClientHttpRequestFactoryService clientHttpRequestFactoryService, final ClientHttpRequestFactoryService clientHttpRequestFactoryService,
@Value("${sebserver.webservice.lms.openedx.api.token.request.paths}") final String alternativeTokenRequestPaths) { @Value("${sebserver.webservice.lms.openedx.api.token.request.paths}") final String alternativeTokenRequestPaths,
@Value("${sebserver.webservice.lms.openedx.seb.restriction.push-count:0}") final int restrictionAPIPushCount) {
this.jsonMapper = jsonMapper; this.jsonMapper = jsonMapper;
this.webserviceInfo = webserviceInfo; this.webserviceInfo = webserviceInfo;
@ -53,6 +55,7 @@ public class OpenEdxLmsAPITemplateFactory {
this.alternativeTokenRequestPaths = (alternativeTokenRequestPaths != null) this.alternativeTokenRequestPaths = (alternativeTokenRequestPaths != null)
? StringUtils.split(alternativeTokenRequestPaths, Constants.LIST_SEPARATOR) ? StringUtils.split(alternativeTokenRequestPaths, Constants.LIST_SEPARATOR)
: null; : null;
this.restrictionAPIPushCount = restrictionAPIPushCount;
} }
public Result<OpenEdxLmsAPITemplate> create( public Result<OpenEdxLmsAPITemplate> create(
@ -79,7 +82,8 @@ public class OpenEdxLmsAPITemplateFactory {
final OpenEdxCourseRestriction openEdxCourseRestriction = new OpenEdxCourseRestriction( final OpenEdxCourseRestriction openEdxCourseRestriction = new OpenEdxCourseRestriction(
lmsSetup, lmsSetup,
this.jsonMapper, this.jsonMapper,
openEdxRestTemplateFactory); openEdxRestTemplateFactory,
this.restrictionAPIPushCount);
return new OpenEdxLmsAPITemplate( return new OpenEdxLmsAPITemplate(
lmsSetup, lmsSetup,

View file

@ -40,6 +40,21 @@ sebserver.webservice.api.pagination.maxPageSize=500
sebserver.webservice.lms.openedx.api.token.request.paths=/oauth2/access_token sebserver.webservice.lms.openedx.api.token.request.paths=/oauth2/access_token
sebserver.webservice.lms.address.alias=lms.mockup.com=lms.address.alias sebserver.webservice.lms.address.alias=lms.mockup.com=lms.address.alias
# NOTE: This is a temporary work-around for SEB Restriction API within Open edX SEB integration plugin to
# apply on load-balanced infrastructure or infrastructure that has several layers of cache.
# The reason for this is that the API (Open edX system) internally don't apply a resource-change that is
# done within HTTP API call immediately from an outside perspective.
# After a resource-change on the API is done, the system toggles between the old and the new resource
# while constantly calling GET. This usually happens for about a minute or two then it stabilizes on the new resource
#
# This may source on load-balancing or internally caching on Open edX side.
# To mitigate this effect the SEB Server can be configured to apply a resource-change on the
# API several times in a row to flush as match caches and reach as match as possible server instances.
#
# Since this is a brute-force method to mitigate the problem, this should only be a temporary
# work-around until a better solution on Open edX SEB integration side has been found and applied.
sebserver.webservice.lms.openedx.seb.restriction.push-count=10
# actuator configuration # actuator configuration
management.endpoints.web.base-path=/actuator management.endpoints.web.base-path=/actuator
management.endpoints.web.exposure.include=logfile,loggers management.endpoints.web.exposure.include=logfile,loggers