io.engineblock.activityapi.ratelimits.RateSpec Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of eb-api Show documentation
Show all versions of eb-api Show documentation
The driver API for engineblock;
Provides the interfaces needed to build drivers that can be loaded by engineblock core
/*
*
* Copyright 2016 jshook
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
* /
*/
package io.engineblock.activityapi.ratelimits;
import io.engineblock.activityimpl.ParameterMap;
import io.engineblock.util.Unit;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Rate Limiter Specifications
*
* A rate spec represent the configuration of a rate limiter. It is the event carrier
* for applying changes to a rate limiter. For scripting purposes, rate limiters can be
* controlled via assignment of a configuration string. A future version of the scripting
* API will support direct method access. For now, the following semantics will apply
* to the configuration value assigned to any of the rate parameters like cyclerate and striderate.
*
*
* Controlling Rate Limiters
*
* Rate limiters specifiers can be easily constructed programmatically. However, in scripting,
* these will often be controlled by assigning a configuration string.
*
*
* When a rate limiter is configured
* using the configuration String, it can be in one of the following forms:
*
*
* - <rate>
* - <rate>,<burst ratio>
* - <rate>,<burst ratio>,<verb>
*
*
* Where:
*
* rate is the ops per second, expressed as any positive floating point value.
* burst ratio is a floating point value greater than 1.0 which determines how much faster
* the rate limiter may go to catch up to the overall.
* verb is one of configure, start, or restart, as explained below.
*
* For example:
*
* - 200 - allow up to 200 ops to start per second, with the default burst ratio of 1.1.
* Start the rate limiter automatically.
* - 3.6,2 - Allow up to 3.6 ops to start per second, but allow up to 7.2 ops per second
* if needed to catch up. Start the rate limiter automatically
* - 1000,1.05,restart - Allow up to 1000 ops per second on average, but allow 1050 ops per second
* if the workload gets behind. If the rate limiter was already running, restart it, clearing any
* previous backlog (wait time) and resource pools.
*
*
* Rate Limiter Life Cycle
*
* A rate limiter can be directly configured, started, restarted. Indirectly, rate
* limiter construction and initialization occurs as needed.
*
*
*
* - (constructed)
* - Rate limiters are constructed and assigned automatically when they are first configured. A rate
* limiter may not be explicitly constructed by the user.
*
* - configure
* - When a rate limiter is configured, the rate spec is applied to it. If the rate limiters is
* already running, then this new rate is applied to it. Applying the same rate parameters a
* second time to a given rate limiter is a no-op.
*
* - (initialization)
* - A rate limiter is initialized immediately before it is initially started. Rate limiters are also
* re-initialized when they are restarted. When a rate limiter is initialized, the start time is set
* to the current time, and the resource pools and accumulated wait time are zeroed. A rate limiter may
* not be explicitly initialized by the user.
*
*
* - start
* - Starting a rate limiter activates the rate metering logic according to the current time. From
* the time a rate limiter is started, any unused time will accumulate as wait time. A rate limiter is required
* to be reset immediately before it is started for the first time.
*
* - restart
* - Restarting a rate limiter is the same as starting it initially. The only difference is that
* restarting forces a re-initialization as part of the configuration.
*
*
*/
public class RateSpec {
private final static Logger logger = LoggerFactory.getLogger(RateSpec.class);
public static final double DEFAULT_RATE_OPS_S = 1.0D;
public static final double DEFAULT_BURST_RATIO = 1.1D;
public static Verb DEFAULT_VERB = Verb.start;
/**
* Target rate in Operations Per Second
*/
public double opsPerSec = DEFAULT_RATE_OPS_S;
public double burstRatio = DEFAULT_BURST_RATIO;
public Verb verb = Verb.start;
public static enum Verb {
/**
* Specify that a rate limiter should only be configured without affecting its running state.
* If the rate limiter is already running, then the configuration should take effect immediately.
* A rate limiter will be created automatically if needed. Configurations that do not effectively
* change the rate limiter are ignored.
*/
configure,
/**
* The default SetAction is start. This means that the rate limiter should be started at the time
* this rate spec is applied. The start time of a rate limiter is significant in that it determines
* both the calculated average rate as well as the accumulated wait time from slow callers. In order
* to start, a rate limiter will be configured automatically, if the provided rate spec would cause
* a change to the configuration. If a rate limiter is started that is already running, an error should
* be thrown. If it is desired to ignore this condition, the restart should be used instead.
*/
start,
/**
* The restart action on a rate limiter causes it to be re-initialized as if it were just being
* started for the first time. This causes any accumulated wait time or time resources to be zeroed.
* This type of specifier can be useful, for example, when iterating on a workload with different
* target rates, where each iteration is independent of the others. In order to restart, a rate
* limiter will be configured if necessary.
*/
restart
}
public RateSpec(double opsPerSec, double burstRatio) {
this(opsPerSec, burstRatio, DEFAULT_VERB);
}
public RateSpec(double opsPerSec, double burstRatio, Verb type) {
this.opsPerSec = opsPerSec;
this.burstRatio = burstRatio;
this.verb = type;
}
public RateSpec(ParameterMap.NamedParameter tuple) {
this(tuple.value);
if (tuple.name.startsWith("co_")) {
logger.warn("The co_ prefix on " + tuple.name + " is no longer needed. All rate limiters now provide standard coordinated omission metrics.");
}
}
public RateSpec(String spec) {
String[] specs = spec.split("[,:;]");
switch (specs.length) {
case 3:
verb = Verb.valueOf(specs[2].toLowerCase());
logger.debug("selected rate limiter type: " + verb);
case 2:
burstRatio = Double.valueOf(specs[1]);
if (burstRatio < 1.0) {
throw new RuntimeException("burst ratios less than 1.0 are invalid.");
}
case 1:
opsPerSec = Unit.doubleCountFor(specs[0]).orElseThrow(() -> new RuntimeException("Unparsable:" + specs[0]));
break;
default:
throw new RuntimeException("Rate specs must be either '' or ':' as in 5000.0 or 5000.0:1.0");
}
}
public String toString() {
StringBuilder sb = new StringBuilder();
double ratePortion = Math.abs(opsPerSec - ((long) opsPerSec));
String ratefmt = (ratePortion > 0.001D) ? String.format("%,.3f", opsPerSec) : String.format("%,d", (long) opsPerSec);
double br = burstRatio * opsPerSec;
double burstPortion = Math.abs(br - ((long) br));
String burstfmt = (burstPortion > 0.001D) ? String.format("%,.3f", br) : String.format("%,d", (long) br);
return String.format("rate=%s burstRatio=%.3f (%s SOPSS %s BOPSS)", ratefmt, burstRatio, ratefmt, burstfmt);
}
public RateSpec withOpsPerSecond(double rate) {
return new RateSpec(rate, this.burstRatio);
}
public RateSpec withBurstRatio(double burstRatio) {
return new RateSpec(this.opsPerSec, burstRatio);
}
public RateSpec withVerb(Verb verb) {
return new RateSpec(this.opsPerSec, this.burstRatio, verb);
}
public long getNanosPerOp() {
return (long) (1E9 / opsPerSec);
}
public double getRate() {
return this.opsPerSec;
}
public double getBurstRatio() {
return this.burstRatio;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
RateSpec rateSpec = (RateSpec) o;
if (Double.compare(rateSpec.opsPerSec, opsPerSec) != 0) return false;
if (Double.compare(rateSpec.burstRatio, burstRatio) != 0) return false;
return true;
}
@Override
public int hashCode() {
int result;
long temp;
temp = Double.doubleToLongBits(opsPerSec);
result = (int) (temp ^ (temp >>> 32));
temp = Double.doubleToLongBits(burstRatio);
result = 31 * result + (int) (temp ^ (temp >>> 32));
return result;
}
public Verb getVerb() {
return this.verb;
}
public boolean isAutoStart() {
return this.verb == Verb.start || this.verb == Verb.restart;
}
public boolean isRestart() {
return this.verb == Verb.restart;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy