diff --git a/pom.xml b/pom.xml
index 57d43678..bff33c62 100644
--- a/pom.xml
+++ b/pom.xml
@@ -341,6 +341,11 @@
commons-text
1.8
+
+ com.github.vladimir-bukhtoyarov
+ bucket4j-core
+ 4.10.0
+
diff --git a/src/main/java/ch/ethz/seb/sebserver/gbl/api/TooManyRegistrations.java b/src/main/java/ch/ethz/seb/sebserver/gbl/api/TooManyRegistrations.java
new file mode 100644
index 00000000..55cacedd
--- /dev/null
+++ b/src/main/java/ch/ethz/seb/sebserver/gbl/api/TooManyRegistrations.java
@@ -0,0 +1,13 @@
+/*
+ * Copyright (c) 2021 ETH Zürich, Educational Development and Technology (LET)
+ *
+ * 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.gbl.api;
+
+public class TooManyRegistrations {
+
+}
diff --git a/src/main/java/ch/ethz/seb/sebserver/gbl/api/TooManyRequests.java b/src/main/java/ch/ethz/seb/sebserver/gbl/api/TooManyRequests.java
new file mode 100644
index 00000000..bf29ba48
--- /dev/null
+++ b/src/main/java/ch/ethz/seb/sebserver/gbl/api/TooManyRequests.java
@@ -0,0 +1,32 @@
+/*
+ * Copyright (c) 2021 ETH Zürich, Educational Development and Technology (LET)
+ *
+ * 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.gbl.api;
+
+public class TooManyRequests extends RuntimeException {
+
+ private static final long serialVersionUID = 3303246002774619224L;
+
+ public static enum Code {
+ INCOMMING,
+ REGISTRATION
+ }
+
+ public final Code code;
+
+ public TooManyRequests() {
+ super("TooManyRequests");
+ this.code = Code.INCOMMING;
+ }
+
+ public TooManyRequests(final Code code) {
+ super("TooManyRequests");
+ this.code = code;
+ }
+
+}
diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/content/RegisterPage.java b/src/main/java/ch/ethz/seb/sebserver/gui/content/RegisterPage.java
index c034f6c7..00433b02 100644
--- a/src/main/java/ch/ethz/seb/sebserver/gui/content/RegisterPage.java
+++ b/src/main/java/ch/ethz/seb/sebserver/gui/content/RegisterPage.java
@@ -232,7 +232,7 @@ public class RegisterPage implements TemplateComposer {
registerButton.addListener(SWT.Selection, event -> {
registerForm.getForm().clearErrors();
- final Result onError = this.pageService
+ final Result result = this.pageService
.getRestService()
.getBuilder(RegisterNewUser.class)
.withRestTemplate(this.restTemplate)
@@ -240,7 +240,7 @@ public class RegisterPage implements TemplateComposer {
.call()
.onError(registerForm::handleError);
- if (onError.hasError()) {
+ if (result.hasError()) {
return;
}
diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/form/FormHandle.java b/src/main/java/ch/ethz/seb/sebserver/gui/form/FormHandle.java
index cb8e12c5..438ea007 100644
--- a/src/main/java/ch/ethz/seb/sebserver/gui/form/FormHandle.java
+++ b/src/main/java/ch/ethz/seb/sebserver/gui/form/FormHandle.java
@@ -18,6 +18,7 @@ import ch.ethz.seb.sebserver.gbl.Constants;
import ch.ethz.seb.sebserver.gbl.api.API;
import ch.ethz.seb.sebserver.gbl.api.APIMessage;
import ch.ethz.seb.sebserver.gbl.api.EntityType;
+import ch.ethz.seb.sebserver.gbl.api.TooManyRequests;
import ch.ethz.seb.sebserver.gbl.model.Entity;
import ch.ethz.seb.sebserver.gbl.util.Result;
import ch.ethz.seb.sebserver.gbl.util.Utils;
@@ -38,6 +39,8 @@ public class FormHandle {
private static final Logger log = LoggerFactory.getLogger(FormHandle.class);
public static final String FIELD_VALIDATION_LOCTEXT_PREFIX = "sebserver.form.validation.fieldError.";
+ static final LocTextKey MESSAGE_TOO_MANY_REQUESTS_TEXT =
+ new LocTextKey("sebserver.error.tooManyRequests");
private final PageService pageService;
private final PageContext pageContext;
@@ -147,14 +150,32 @@ public class FormHandle {
fieldAccessor -> showValidationError(fieldAccessor, fve)));
return true;
} else {
+
log.error("Unexpected error while trying to post form: {}", error.getMessage());
- final EntityType resultType = this.post.getEntityType();
- if (resultType != null) {
- this.pageContext.notifySaveError(resultType, error);
- } else {
- this.pageContext.notifyError(
- new LocTextKey(PageContext.GENERIC_SAVE_ERROR_TEXT_KEY, Constants.EMPTY_NOTE),
- error);
+
+ try {
+
+ if (error instanceof TooManyRequests) {
+ final TooManyRequests.Code code = ((TooManyRequests) error).code;
+ if (code != null) {
+ this.pageContext.publishInfo(new LocTextKey(MESSAGE_TOO_MANY_REQUESTS_TEXT.name + "." + code));
+ } else {
+ this.pageContext.publishInfo(MESSAGE_TOO_MANY_REQUESTS_TEXT);
+ }
+
+ return false;
+ }
+
+ final EntityType resultType = this.post.getEntityType();
+ if (resultType != null) {
+ this.pageContext.notifySaveError(resultType, error);
+ } else {
+ this.pageContext.notifyError(
+ new LocTextKey(PageContext.GENERIC_SAVE_ERROR_TEXT_KEY, Constants.EMPTY_NOTE),
+ error);
+ }
+ } catch (final Exception e) {
+ log.error("Failed to handle error: ", e);
}
return false;
diff --git a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCall.java b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCall.java
index 575b139b..0932f3cc 100644
--- a/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCall.java
+++ b/src/main/java/ch/ethz/seb/sebserver/gui/service/remote/webservice/api/RestCall.java
@@ -40,6 +40,7 @@ import ch.ethz.seb.sebserver.gbl.Constants;
import ch.ethz.seb.sebserver.gbl.api.APIMessage;
import ch.ethz.seb.sebserver.gbl.api.EntityType;
import ch.ethz.seb.sebserver.gbl.api.JSONMapper;
+import ch.ethz.seb.sebserver.gbl.api.TooManyRequests;
import ch.ethz.seb.sebserver.gbl.model.Entity;
import ch.ethz.seb.sebserver.gbl.model.Page;
import ch.ethz.seb.sebserver.gbl.model.PageSortOrder;
@@ -134,6 +135,15 @@ public abstract class RestCall {
}
} catch (final RestClientResponseException responseError) {
+ if (responseError.getRawStatusCode() == HttpStatus.TOO_MANY_REQUESTS.value()) {
+ final String code = responseError.getResponseBodyAsString();
+ if (StringUtils.isNotBlank(code)) {
+ return Result.ofError(new TooManyRequests(TooManyRequests.Code.valueOf(code)));
+ } else {
+ return Result.ofError(new TooManyRequests());
+ }
+ }
+
final RestCallError restCallError = new RestCallError("Unexpected error while rest call", responseError);
try {
diff --git a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/APIExceptionHandler.java b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/APIExceptionHandler.java
index 2b56b838..12087843 100644
--- a/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/APIExceptionHandler.java
+++ b/src/main/java/ch/ethz/seb/sebserver/webservice/weblayer/api/APIExceptionHandler.java
@@ -33,6 +33,7 @@ import org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExcep
import ch.ethz.seb.sebserver.gbl.api.APIMessage;
import ch.ethz.seb.sebserver.gbl.api.APIMessage.APIMessageException;
import ch.ethz.seb.sebserver.gbl.api.APIMessage.FieldValidationException;
+import ch.ethz.seb.sebserver.gbl.api.TooManyRequests;
import ch.ethz.seb.sebserver.gbl.util.Utils;
import ch.ethz.seb.sebserver.webservice.servicelayer.authorization.PermissionDeniedException;
import ch.ethz.seb.sebserver.webservice.servicelayer.dao.ResourceNotFoundException;
@@ -87,6 +88,15 @@ public class APIExceptionHandler extends ResponseEntityExceptionHandler {
HttpStatus.BAD_REQUEST);
}
+ @ExceptionHandler(TooManyRequests.class)
+ public ResponseEntity