Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

adding rate limiting for command engine #10211

Merged
merged 85 commits into from
Mar 20, 2024
Merged
Show file tree
Hide file tree
Changes from 83 commits
Commits
Show all changes
85 commits
Select commit Hold shift + click to select a range
598912f
adding rate limiting for commmand engine
stevenwinship Jan 5, 2024
c657eb0
fixing tests
stevenwinship Jan 5, 2024
f5e0070
fixing tests
stevenwinship Jan 5, 2024
1b0a554
fixing tests
stevenwinship Jan 5, 2024
a534627
Update doc/release-notes/9356-rate-limiting.md
stevenwinship Jan 9, 2024
c80f74a
fixing review comments
stevenwinship Jan 9, 2024
12c15b5
review comment fixes
stevenwinship Jan 9, 2024
a178929
review comment fixes
stevenwinship Jan 9, 2024
4684384
review comment fixes
stevenwinship Jan 9, 2024
1e44206
fixes to get DatasetsIT to pass
stevenwinship Jan 11, 2024
77074cc
fix mock for redis tests
stevenwinship Jan 18, 2024
4cdba95
fix mock for redis tests
stevenwinship Jan 18, 2024
1b4f613
Update doc/sphinx-guides/source/installation/config.rst
stevenwinship Jan 24, 2024
2b603a6
fixes from comments
stevenwinship Jan 24, 2024
13e3011
Update doc/release-notes/9356-rate-limiting.md
stevenwinship Jan 24, 2024
0467f4c
Update doc/release-notes/9356-rate-limiting.md
stevenwinship Jan 24, 2024
5253de8
Update doc/release-notes/9356-rate-limiting.md
stevenwinship Jan 24, 2024
13fdd88
Update doc/release-notes/9356-rate-limiting.md
stevenwinship Jan 24, 2024
dd30c7b
Update doc/release-notes/9356-rate-limiting.md
stevenwinship Jan 24, 2024
23606a0
Update doc/release-notes/9356-rate-limiting.md
stevenwinship Jan 24, 2024
1bd2556
Update doc/release-notes/9356-rate-limiting.md
stevenwinship Jan 24, 2024
c04db0a
Update doc/release-notes/9356-rate-limiting.md
stevenwinship Jan 24, 2024
3dfc2a0
adding changes per pr comments
stevenwinship Jan 24, 2024
727cccf
remove redis and replace with jcache hazelcast
stevenwinship Jan 26, 2024
58ea032
adding cache tests
stevenwinship Jan 26, 2024
f7f9664
fixing unit tests
stevenwinship Jan 29, 2024
dbb774b
fixing unit tests
stevenwinship Jan 29, 2024
b489ec8
fixing unit tests
stevenwinship Jan 29, 2024
700e799
fixing unit tests
stevenwinship Jan 29, 2024
5a7d300
fixing unit tests
stevenwinship Jan 29, 2024
e2b5fe8
fixing unit tests
stevenwinship Jan 29, 2024
7fb8c88
fixing unit tests
stevenwinship Jan 29, 2024
0674105
fixing unit tests
stevenwinship Jan 29, 2024
a55ed93
fixing unit tests
stevenwinship Jan 29, 2024
c7b5969
fixing unit tests
stevenwinship Jan 29, 2024
a27c785
fixing unit tests
stevenwinship Jan 30, 2024
ecca881
fixing unit tests
stevenwinship Jan 30, 2024
11a37e3
fixing unit tests
stevenwinship Jan 30, 2024
c84ae14
fixing unit tests
stevenwinship Jan 30, 2024
4f8a39c
fixing unit tests
stevenwinship Jan 30, 2024
3d0e438
fix test hazelcast config
stevenwinship Jan 31, 2024
176adbc
fix test hazelcast config
stevenwinship Jan 31, 2024
403dc08
fix test hazelcast config
stevenwinship Jan 31, 2024
9e43b25
fix test hazelcast config
stevenwinship Jan 31, 2024
0771fae
fixing more review comments
stevenwinship Feb 1, 2024
252337a
fix db rate limit tier column
stevenwinship Feb 1, 2024
cc70ba7
fix db rate limit tier column
stevenwinship Feb 1, 2024
794f024
fix db rate limit tier column
stevenwinship Feb 1, 2024
605097c
getting tests to pass on Jenkins
stevenwinship Feb 2, 2024
879bc5c
testing in jenkins
stevenwinship Feb 2, 2024
27cce94
use payara instance of hazelcast
stevenwinship Feb 2, 2024
9784416
fixes for Jenkins
stevenwinship Feb 2, 2024
21b0951
fixes for Jenkins
stevenwinship Feb 2, 2024
465c5d5
fixes for Jenkins
stevenwinship Feb 2, 2024
e5fe18f
fixes for Jenkins
stevenwinship Feb 2, 2024
15ef82e
Update pom.xml
stevenwinship Feb 5, 2024
77cede2
Update src/test/java/edu/harvard/iq/dataverse/cache/CacheFactoryBeanT…
stevenwinship Feb 5, 2024
02cd0d0
Update pom.xml
stevenwinship Feb 5, 2024
9b95e4d
Update src/main/java/edu/harvard/iq/dataverse/cache/CacheFactoryBean.…
stevenwinship Feb 5, 2024
52e714b
review comments re: JCache
stevenwinship Feb 6, 2024
669d273
review comments re: JCache
stevenwinship Feb 6, 2024
9800fc1
doc change
stevenwinship Feb 6, 2024
ae0ec5a
fix bad merge
stevenwinship Feb 7, 2024
5e507a0
moving cache to util/cache
stevenwinship Feb 7, 2024
0774223
review comments fixed
stevenwinship Feb 7, 2024
d5b1fb5
rename db script
stevenwinship Feb 9, 2024
54f1077
review comments
stevenwinship Feb 20, 2024
d2d3b4a
fixing config.rst
stevenwinship Feb 20, 2024
3102b05
fixing config.rst
stevenwinship Feb 20, 2024
736c633
more review comments
stevenwinship Feb 20, 2024
ecea90c
fixing tests
stevenwinship Feb 21, 2024
9d575ed
fixing tests
stevenwinship Feb 21, 2024
692c650
more review comments
stevenwinship Feb 22, 2024
13674df
more review comments
stevenwinship Feb 22, 2024
1e4d351
review comments
stevenwinship Feb 27, 2024
8edbc04
review comments
stevenwinship Feb 27, 2024
4a0e0af
rename sql to unique
stevenwinship Feb 29, 2024
5233bf2
review comments
stevenwinship Feb 29, 2024
b66e000
Update doc/sphinx-guides/source/installation/config.rst
stevenwinship Feb 29, 2024
0b3c5e3
Cosmetics for 9356 - Rate Limiting PR (#10349)
poikilotherm Mar 1, 2024
2266225
rename sql file
stevenwinship Mar 6, 2024
a1ab6f9
change sql script name
stevenwinship Mar 18, 2024
e1f2e66
One extra phrase added to the guide
landreev Mar 19, 2024
91bb468
adding two specific commands CheckRateLimitForDatasetPage and CheckRa…
stevenwinship Mar 20, 2024
a9b2514
add check for existing cache before creating a new one
stevenwinship Mar 20, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
20 changes: 20 additions & 0 deletions doc/release-notes/9356-rate-limiting.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
## Rate Limiting using JCache (with Hazelcast as provided by Payara)
The option to rate limit has been added to prevent users from over taxing the system either deliberately or by runaway automated processes.
Rate limiting can be configured on a tier level with tier 0 being reserved for guest users and tiers 1-any for authenticated users.
Superuser accounts are exempt from rate limiting.
Rate limits can be imposed on command APIs by configuring the tier, the command, and the hourly limit in the database.
Two database settings configure the rate limiting.
stevenwinship marked this conversation as resolved.
Show resolved Hide resolved
Note: If either of these settings exist in the database rate limiting will be enabled.
If neither setting exists rate limiting is disabled.

`:RateLimitingDefaultCapacityTiers` is a comma separated list of default values for each tier.
In the following example, the default for tier `0` (guest users) is set to 10,000 calls per command per hour and tier `1` (authenticated users) is set to 20,000 calls per command per hour.
Tiers not specified in this setting will default to `-1` (No Limit). I.e., -d "10000" is equivalent to -d "10000,-1,-1,..."
`curl http://localhost:8080/api/admin/settings/:RateLimitingDefaultCapacityTiers -X PUT -d '10000,20000'`

`:RateLimitingCapacityByTierAndAction` is a JSON object specifying the rate by tier and a list of actions (commands).
This allows for more control over the rate limit of individual API command calls.
In the following example, calls made by a guest user (tier 0) for API `GetLatestPublishedDatasetVersionCommand` is further limited to only 10 calls per hour, while an authenticated user (tier 1) will be able to make 30 calls per hour to the same API.
`curl http://localhost:8080/api/admin/settings/:RateLimitingCapacityByTierAndAction -X PUT -d '[{"tier": 0, "limitPerHour": 10, "actions": ["GetLatestPublishedDatasetVersionCommand", "GetPrivateUrlCommand", "GetDatasetCommand", "GetLatestAccessibleDatasetVersionCommand"]}, {"tier": 0, "limitPerHour": 1, "actions": ["CreateGuestbookResponseCommand", "UpdateDatasetVersionCommand", "DestroyDatasetCommand", "DeleteDataFileCommand", "FinalizeDatasetPublicationCommand", "PublishDatasetCommand"]}, {"tier": 1, "limitPerHour": 30, "actions": ["CreateGuestbookResponseCommand", "GetLatestPublishedDatasetVersionCommand", "GetPrivateUrlCommand", "GetDatasetCommand", "GetLatestAccessibleDatasetVersionCommand", "UpdateDatasetVersionCommand", "DestroyDatasetCommand", "DeleteDataFileCommand", "FinalizeDatasetPublicationCommand", "PublishDatasetCommand"]}]'`

Hazelcast is configured in Payara and should not need any changes for this feature
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
[
{
"tier": 0,
"limitPerHour": 10,
"actions": [
"GetLatestPublishedDatasetVersionCommand",
"GetPrivateUrlCommand",
"GetDatasetCommand",
"GetLatestAccessibleDatasetVersionCommand"
]
},
{
"tier": 0,
"limitPerHour": 1,
"actions": [
"CreateGuestbookResponseCommand",
"UpdateDatasetVersionCommand",
"DestroyDatasetCommand",
"DeleteDataFileCommand",
"FinalizeDatasetPublicationCommand",
"PublishDatasetCommand"
]
},
{
"tier": 1,
"limitPerHour": 30,
"actions": [
"CreateGuestbookResponseCommand",
"GetLatestPublishedDatasetVersionCommand",
"GetPrivateUrlCommand",
"GetDatasetCommand",
"GetLatestAccessibleDatasetVersionCommand",
"UpdateDatasetVersionCommand",
"DestroyDatasetCommand",
"DeleteDataFileCommand",
"FinalizeDatasetPublicationCommand",
"PublishDatasetCommand"
]
}
]
41 changes: 41 additions & 0 deletions doc/sphinx-guides/source/installation/config.rst
stevenwinship marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -1373,6 +1373,33 @@ Before being moved there,
on your machine, large file uploads via API will cause RAM and/or swap usage bursts. You might want to point this to
a different location, restrict maximum size of it, and monitor for stale uploads.

.. _cache-rate-limiting:

Configure Your Dataverse Installation to Use JCache (with Hazelcast as Provided by Payara) for Rate Limiting
------------------------------------------------------------------------------------------------------------

Rate limiting has been added to prevent users from over taxing the system either deliberately or by runaway automated processes.
Rate limiting can be configured on a tier level with tier 0 being reserved for guest users and tiers 1-any for authenticated users.
Superuser accounts are exempt from rate limiting.
Rate limits can be imposed on command APIs by configuring the tier, the command, and the hourly limit in the database.
Two database settings configure the rate limiting.
Note: If either of these settings exist in the database rate limiting will be enabled (note that a Payara restart is required for the setting to take effect). If neither setting exists rate limiting is disabled.

- :RateLimitingDefaultCapacityTiers is the number of calls allowed per hour if the specific command is not configured. The values represent the number of calls per hour per user for tiers 0,1,...
A value of -1 can be used to signify no rate limit. Tiers not specified in this setting will default to `-1` (No Limit). I.e., -d "10000" is equivalent to -d "10000,-1,-1,..."

.. code-block:: bash

curl http://localhost:8080/api/admin/settings/:RateLimitingDefaultCapacityTiers -X PUT -d '10000,20000'
poikilotherm marked this conversation as resolved.
Show resolved Hide resolved

- :RateLimitingCapacityByTierAndAction is a JSON object specifying the rate by tier and a list of actions (commands). This allows for more control over the rate limit of individual API command calls.
In the following example, calls made by a guest user (tier 0) for API GetLatestPublishedDatasetVersionCommand is further limited to only 10 calls per hour, while an authenticated user (tier 1) will be able to make 30 calls per hour to the same API.

:download:`rate-limit-actions.json </_static/installation/files/examples/rate-limit-actions-setting.json>` Example json for RateLimitingCapacityByTierAndAction

.. code-block:: bash

curl http://localhost:8080/api/admin/settings/:RateLimitingCapacityByTierAndAction -X PUT -d '[{"tier": 0, "limitPerHour": 10, "actions": ["GetLatestPublishedDatasetVersionCommand", "GetPrivateUrlCommand", "GetDatasetCommand", "GetLatestAccessibleDatasetVersionCommand"]}, {"tier": 0, "limitPerHour": 1, "actions": ["CreateGuestbookResponseCommand", "UpdateDatasetVersionCommand", "DestroyDatasetCommand", "DeleteDataFileCommand", "FinalizeDatasetPublicationCommand", "PublishDatasetCommand"]}, {"tier": 1, "limitPerHour": 30, "actions": ["CreateGuestbookResponseCommand", "GetLatestPublishedDatasetVersionCommand", "GetPrivateUrlCommand", "GetDatasetCommand", "GetLatestAccessibleDatasetVersionCommand", "UpdateDatasetVersionCommand", "DestroyDatasetCommand", "DeleteDataFileCommand", "FinalizeDatasetPublicationCommand", "PublishDatasetCommand"]}]'

.. _Branding Your Installation:

Expand Down Expand Up @@ -4496,3 +4523,17 @@ tab. files saved with these headers on S3 - since they no longer have
to be generated and added to the streamed file on the fly.

The setting is ``false`` by default, preserving the legacy behavior.

:RateLimitingDefaultCapacityTiers
+++++++++++++++++++++++++++++++++
Number of calls allowed per hour if the specific command is not configured. The values represent the number of calls per hour per user for tiers 0,1,...
A value of -1 can be used to signify no rate limit. Also, by default, a tier not defined would receive a default of no limit.

:RateLimitingCapacityByTierAndAction
++++++++++++++++++++++++++++++++++++
JSON object specifying the rate by tier and a list of actions (commands). This allows for more control over the rate limit of individual API command calls.
In the following example, calls made by a guest user (tier 0) for API GetLatestPublishedDatasetVersionCommand is further limited to only 10 calls per hour, while an authenticated user (tier 1) will be able to make 30 calls per hour to the same API.
{"rateLimits":[
{"tier": 0, "limitPerHour": 10, "actions": ["GetLatestPublishedDatasetVersionCommand", "GetPrivateUrlCommand", "GetDatasetCommand", "GetLatestAccessibleDatasetVersionCommand"]},
{"tier": 0, "limitPerHour": 1, "actions": ["CreateGuestbookResponseCommand", "UpdateDatasetVersionCommand", "DestroyDatasetCommand", "DeleteDataFileCommand", "FinalizeDatasetPublicationCommand", "PublishDatasetCommand"]},
{"tier": 1, "limitPerHour": 30, "actions": ["CreateGuestbookResponseCommand", "GetLatestPublishedDatasetVersionCommand", "GetPrivateUrlCommand", "GetDatasetCommand", "GetLatestAccessibleDatasetVersionCommand", "UpdateDatasetVersionCommand", "DestroyDatasetCommand", "DeleteDataFileCommand", "FinalizeDatasetPublicationCommand", "PublishDatasetCommand"]}]}
21 changes: 21 additions & 0 deletions pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -210,6 +210,18 @@
<scope>provided</scope>
</dependency>

<!-- JSON-B -->
<dependency>
<groupId>jakarta.json.bind</groupId>
<artifactId>jakarta.json.bind-api</artifactId>
</dependency>
<!-- Rope in an implementation for unit tests - is provided at runtime by appserver -->
<dependency>
<groupId>org.eclipse</groupId>
<artifactId>yasson</artifactId>
<scope>test</scope>
</dependency>

<!-- Jakarta Faces & related -->
<dependency>
<groupId>org.glassfish</groupId>
Expand Down Expand Up @@ -542,6 +554,10 @@
<artifactId>dataverse-spi</artifactId>
<version>2.0.0</version>
</dependency>
<dependency>
<groupId>javax.cache</groupId>
<artifactId>cache-api</artifactId>
</dependency>
<!-- TESTING DEPENDENCIES -->
<dependency>
<groupId>org.junit.jupiter</groupId>
Expand Down Expand Up @@ -653,6 +669,11 @@
<version>3.9.0</version>
<scope>test</scope>
</dependency>
stevenwinship marked this conversation as resolved.
Show resolved Hide resolved
<dependency>
<groupId>com.hazelcast</groupId>
<artifactId>hazelcast</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<testResources>
Expand Down
12 changes: 10 additions & 2 deletions src/main/java/edu/harvard/iq/dataverse/EjbDataverseEngine.java
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
import edu.harvard.iq.dataverse.actionlogging.ActionLogServiceBean;
import edu.harvard.iq.dataverse.authorization.AuthenticationServiceBean;
import edu.harvard.iq.dataverse.authorization.providers.builtin.BuiltinUserServiceBean;
import edu.harvard.iq.dataverse.util.cache.CacheFactoryBean;
import edu.harvard.iq.dataverse.engine.DataverseEngine;
import edu.harvard.iq.dataverse.authorization.Permission;
import edu.harvard.iq.dataverse.authorization.groups.GroupServiceBean;
Expand All @@ -16,6 +17,7 @@
import edu.harvard.iq.dataverse.engine.command.DataverseRequest;
import edu.harvard.iq.dataverse.engine.command.exception.CommandException;
import edu.harvard.iq.dataverse.engine.command.exception.PermissionException;
import edu.harvard.iq.dataverse.engine.command.exception.RateLimitCommandException;
import edu.harvard.iq.dataverse.ingest.IngestServiceBean;
import edu.harvard.iq.dataverse.pidproviders.PidProviderFactoryBean;
import edu.harvard.iq.dataverse.privateurl.PrivateUrlServiceBean;
Expand Down Expand Up @@ -176,7 +178,9 @@ public class EjbDataverseEngine {

@EJB
EjbDataverseEngineInner innerEngine;


@EJB
CacheFactoryBean cacheFactory;

@Resource
EJBContext ejbCtxt;
Expand All @@ -202,7 +206,11 @@ public <R> R submit(Command<R> aCommand) throws CommandException {

try {
logRec.setUserIdentifier( aCommand.getRequest().getUser().getIdentifier() );

// Check for rate limit exceeded. Must be done before anything else to prevent unnecessary processing.
if (!cacheFactory.checkRate(aCommand.getRequest().getUser(), aCommand)) {
throw new RateLimitCommandException(BundleUtil.getStringFromBundle("command.exception.user.ratelimited", Arrays.asList(aCommand.getClass().getSimpleName())), aCommand);
stevenwinship marked this conversation as resolved.
Show resolved Hide resolved
}

// Check permissions - or throw an exception
Map<String, ? extends Set<Permission>> requiredMap = aCommand.getRequiredPermissions();
if (requiredMap == null) {
Expand Down
4 changes: 3 additions & 1 deletion src/main/java/edu/harvard/iq/dataverse/UserServiceBean.java
Original file line number Diff line number Diff line change
Expand Up @@ -147,6 +147,8 @@ private AuthenticatedUser createAuthenticatedUserForView (Object[] dbRowValues,
user.setMutedEmails(Type.tokenizeToSet((String) dbRowValues[15]));
user.setMutedNotifications(Type.tokenizeToSet((String) dbRowValues[15]));

user.setRateLimitTier((int)dbRowValues[17]);

user.setRoles(roles);
return user;
}
Expand Down Expand Up @@ -419,7 +421,7 @@ private List<Object[]> getUserListCore(String searchTerm,
qstr += " u.createdtime, u.lastlogintime, u.lastapiusetime, ";
qstr += " prov.id, prov.factoryalias, ";
qstr += " u.deactivated, u.deactivatedtime, ";
qstr += " u.mutedEmails, u.mutedNotifications ";
qstr += " u.mutedEmails, u.mutedNotifications, u.rateLimitTier ";
qstr += " FROM authenticateduser u,";
qstr += " authenticateduserlookup prov_lookup,";
qstr += " authenticationproviderrow prov";
Expand Down
13 changes: 10 additions & 3 deletions src/main/java/edu/harvard/iq/dataverse/api/AbstractApiBean.java
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
import edu.harvard.iq.dataverse.engine.command.impl.GetLatestAccessibleDatasetVersionCommand;
import edu.harvard.iq.dataverse.engine.command.impl.GetLatestPublishedDatasetVersionCommand;
import edu.harvard.iq.dataverse.engine.command.impl.GetSpecificPublishedDatasetVersionCommand;
import edu.harvard.iq.dataverse.engine.command.exception.RateLimitCommandException;
import edu.harvard.iq.dataverse.externaltools.ExternalToolServiceBean;
import edu.harvard.iq.dataverse.license.LicenseServiceBean;
import edu.harvard.iq.dataverse.locality.StorageSiteServiceBean;
Expand Down Expand Up @@ -421,7 +422,7 @@ public Command<DatasetVersion> handleLatestPublished() {
}));
return dsv;
}

protected DataFile findDataFileOrDie(String id) throws WrappedResponse {
DataFile datafile;
if (id.equals(PERSISTENT_ID_KEY)) {
Expand Down Expand Up @@ -575,6 +576,8 @@ protected <T> T execCommand( Command<T> cmd ) throws WrappedResponse {
try {
return engineSvc.submit(cmd);

} catch (RateLimitCommandException ex) {
throw new WrappedResponse(rateLimited(ex.getMessage()));
} catch (IllegalCommandException ex) {
//for 8859 for api calls that try to update datasets with TOA out of compliance
if (ex.getMessage().toLowerCase().contains("terms of use")){
Expand Down Expand Up @@ -776,11 +779,15 @@ protected Response notFound( String msg ) {
protected Response badRequest( String msg ) {
return error( Status.BAD_REQUEST, msg );
}

protected Response forbidden( String msg ) {
return error( Status.FORBIDDEN, msg );
}


protected Response rateLimited( String msg ) {
return error( Status.TOO_MANY_REQUESTS, msg );
}

protected Response conflict( String msg ) {
return error( Status.CONFLICT, msg );
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
import edu.harvard.iq.dataverse.util.BundleUtil;
import edu.harvard.iq.dataverse.util.json.JsonPrinter;
import static edu.harvard.iq.dataverse.util.StringUtil.nonEmpty;

import edu.harvard.iq.dataverse.util.json.NullSafeJsonBuilder;
import java.io.Serializable;
import java.sql.Timestamp;
Expand All @@ -42,6 +43,7 @@
import jakarta.persistence.PostLoad;
import jakarta.persistence.PrePersist;
import jakarta.persistence.Transient;
import jakarta.validation.constraints.Min;
import jakarta.validation.constraints.NotBlank;
import jakarta.validation.constraints.NotNull;

Expand Down Expand Up @@ -146,6 +148,10 @@ public class AuthenticatedUser implements User, Serializable {
@Transient
private Set<Type> mutedNotificationsSet = new HashSet<>();

@Column(nullable=false)
@Min(value = 1, message = "Rate Limit Tier must be greater than 0.")
private int rateLimitTier = 1;
stevenwinship marked this conversation as resolved.
Show resolved Hide resolved

@PrePersist
void prePersist() {
mutedNotifications = Type.toStringValue(mutedNotificationsSet);
Expand Down Expand Up @@ -397,6 +403,13 @@ public void setDeactivatedTime(Timestamp deactivatedTime) {
this.deactivatedTime = deactivatedTime;
}

public int getRateLimitTier() {
return rateLimitTier;
}
public void setRateLimitTier(int rateLimitTier) {
this.rateLimitTier = rateLimitTier;
}

@OneToOne(mappedBy = "authenticatedUser")
private AuthenticatedUserLookup authenticatedUserLookup;

Expand Down Expand Up @@ -435,7 +448,6 @@ public void setShibIdentityProvider(String shibIdentityProvider) {

public JsonObjectBuilder toJson() {
//JsonObjectBuilder authenicatedUserJson = Json.createObjectBuilder();

NullSafeJsonBuilder authenicatedUserJson = NullSafeJsonBuilder.jsonObjectBuilder();

authenicatedUserJson.add("id", this.id);
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
package edu.harvard.iq.dataverse.engine.command.exception;

import edu.harvard.iq.dataverse.engine.command.Command;

/**
* An exception raised when a command cannot be executed, due to the
* issuing user being rate limited.
*
* @author
*/
public class RateLimitCommandException extends CommandException {

public RateLimitCommandException(String message, Command aCommand) {
super(message, aCommand);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -239,6 +239,10 @@ public enum Key {

CVocConf,

// Default calls per hour for each tier. csv format (30,60,...)
RateLimitingDefaultCapacityTiers,
// json defined list of capacities by tier and action list. See RateLimitSetting.java
RateLimitingCapacityByTierAndAction,
/**
* A link to an installation of https://github.com/IQSS/miniverse or
* some other metrics app.
Expand Down
14 changes: 12 additions & 2 deletions src/main/java/edu/harvard/iq/dataverse/util/SystemConfig.java
Original file line number Diff line number Diff line change
Expand Up @@ -1146,11 +1146,21 @@ public Long getTestStorageQuotaLimit() {
return settingsService.getValueForKeyAsLong(SettingsServiceBean.Key.StorageQuotaSizeInBytes);
}
/**
* Should we store tab-delimited files produced during ingest *with* the
* variable name header line included?
* Should we store tab-delimited files produced during ingest *with* the
* variable name header line included?
* @return boolean - defaults to false.
*/
public boolean isStoringIngestedFilesWithHeaders() {
return settingsService.isTrueForKey(SettingsServiceBean.Key.StoreIngestedTabularFilesWithVarHeaders, false);
}

/**
* RateLimitUtil will parse the json to create a List<RateLimitSetting>
*/
public String getRateLimitsJson() {
stevenwinship marked this conversation as resolved.
Show resolved Hide resolved
return settingsService.getValueForKey(SettingsServiceBean.Key.RateLimitingCapacityByTierAndAction, "");
}
public String getRateLimitingDefaultCapacityTiers() {
return settingsService.getValueForKey(SettingsServiceBean.Key.RateLimitingDefaultCapacityTiers, "");
}
}
Loading
Loading