com.azure.spring.cloud.appconfiguration.config.implementation.StateHolder Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of spring-cloud-azure-appconfiguration-config Show documentation
Show all versions of spring-cloud-azure-appconfiguration-config Show documentation
Integration of Spring Cloud Config and Azure App Configuration Service
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.
package com.azure.spring.cloud.appconfiguration.config.implementation;
import java.security.SecureRandom;
import java.time.Duration;
import java.time.Instant;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.concurrent.ConcurrentHashMap;
import com.azure.data.appconfiguration.models.ConfigurationSetting;
import com.azure.spring.cloud.appconfiguration.config.implementation.feature.FeatureFlagState;
import com.azure.spring.cloud.appconfiguration.config.implementation.feature.FeatureFlags;
final class StateHolder {
private static final int MAX_JITTER = 15;
private static StateHolder currentState;
private final Map state = new ConcurrentHashMap<>();
private final Map featureFlagState = new ConcurrentHashMap<>();
private final Map loadState = new ConcurrentHashMap<>();
private Integer clientRefreshAttempts = 1;
private Instant nextForcedRefresh;
StateHolder() {
}
static StateHolder getCurrentState() {
return currentState;
}
static StateHolder updateState(StateHolder newState) {
currentState = newState;
return currentState;
}
/**
* @param originEndpoint the endpoint for the origin config store
* @return the state
*/
static State getState(String originEndpoint) {
return currentState.getFullState().get(originEndpoint);
}
private Map getFullState() {
return state;
}
private Map getFullFeatureFlagState() {
return featureFlagState;
}
private Map getFullLoadState() {
return loadState;
}
/**
* @param originEndpoint the endpoint for the origin config store
* @return the state
*/
static FeatureFlagState getStateFeatureFlag(String originEndpoint) {
return currentState.getFullFeatureFlagState().get(originEndpoint);
}
/**
* @param originEndpoint the stores origin endpoint
* @param watchKeys list of configuration watch keys that can trigger a refresh event
* @param duration refresh duration.
*/
void setState(String originEndpoint, List watchKeys, Duration duration) {
state.put(originEndpoint, new State(watchKeys, Math.toIntExact(duration.getSeconds()), originEndpoint));
}
/**
* @param originEndpoint the stores origin endpoint
* @param watchKeys list of configuration watch keys that can trigger a refresh event
* @param duration refresh duration.
*/
void setStateFeatureFlag(String originEndpoint, List watchKeys,
Duration duration) {
featureFlagState.put(originEndpoint, new FeatureFlagState(watchKeys, Math.toIntExact(duration.getSeconds()), originEndpoint));
}
/**
* @param state previous state to base off
* @param duration nextRefreshPeriod
*/
void setState(State state, Duration duration) {
this.state.put(state.getOriginEndpoint(),
new State(state, Instant.now().plusSeconds(Math.toIntExact(duration.getSeconds()))));
}
void updateStateRefresh(State state, Duration duration) {
this.state.put(state.getOriginEndpoint(),
new State(state, Instant.now().plusSeconds(Math.toIntExact(duration.getSeconds()))));
}
void updateFeatureFlagStateRefresh(FeatureFlagState state, Duration duration) {
this.featureFlagState.put(state.getOriginEndpoint(),
new FeatureFlagState(state, Instant.now().plusSeconds(Math.toIntExact(duration.getSeconds()))));
}
void expireState(String originEndpoint) {
State oldState = state.get(originEndpoint);
long wait = (long) (new SecureRandom().nextDouble() * MAX_JITTER);
long timeLeft = (int) ((oldState.getNextRefreshCheck().toEpochMilli() - (Instant.now().toEpochMilli())) / 1000);
if (wait < timeLeft) {
state.put(originEndpoint, new State(oldState, Instant.now().plusSeconds(wait)));
}
}
/**
* @return the loadState
*/
static boolean getLoadState(String originEndpoint) {
return currentState.getFullLoadState().getOrDefault(originEndpoint, false);
}
Map getLoadState() {
return loadState;
}
/**
* @param originEndpoint the configuration store connected to.
* @param loaded true if the configuration store was loaded.
* @param failFast application started after it failed to load from a store.
*/
void setLoadState(String originEndpoint, Boolean loaded, Boolean failFast) {
if (loaded || !failFast) {
loadState.put(originEndpoint, true);
} else {
loadState.put(originEndpoint, false);
}
}
/**
* @return the nextForcedRefresh
*/
public static Instant getNextForcedRefresh() {
return currentState.nextForcedRefresh;
}
/**
* Set after load or refresh is successful.
* @param refreshPeriod the refreshPeriod to set
*/
public void setNextForcedRefresh(Duration refreshPeriod) {
if (refreshPeriod != null) {
nextForcedRefresh = Instant.now().plusSeconds(refreshPeriod.getSeconds());
}
}
/**
* Sets a minimum value until the next refresh. If a refresh interval has passed or is smaller than the calculated
* backoff time, the refresh interval is set to the backoff time.
* @param refreshInterval period between refresh checks.
* @param defaultMinBackoff min backoff between checks
*/
void updateNextRefreshTime(Duration refreshInterval, Long defaultMinBackoff) {
if (refreshInterval != null) {
Instant newForcedRefresh = getNextRefreshCheck(nextForcedRefresh,
clientRefreshAttempts, refreshInterval.getSeconds(), defaultMinBackoff);
if (newForcedRefresh.compareTo(nextForcedRefresh) != 0) {
clientRefreshAttempts += 1;
}
nextForcedRefresh = newForcedRefresh;
}
for (Entry entry : state.entrySet()) {
State state = entry.getValue();
Instant newRefresh = getNextRefreshCheck(state.getNextRefreshCheck(),
state.getRefreshAttempt(), (long) state.getRefreshInterval(), defaultMinBackoff);
if (newRefresh.compareTo(entry.getValue().getNextRefreshCheck()) != 0) {
state.incrementRefreshAttempt();
}
State updatedState = new State(state, newRefresh);
this.state.put(entry.getKey(), updatedState);
}
}
/**
* Calculates the amount of time to the next refresh, if a refresh fails. Takes current Refresh date into account
* for watch keys. Used for checking client refresh-interval only.
* @param nextRefreshCheck next refresh for the whole client
* @param attempt refresh attempt for the client
* @param interval the Refresh Interval
* @param defaultMinBackoff min backoff between checks
* @return new Refresh Date
*/
private Instant getNextRefreshCheck(Instant nextRefreshCheck, Integer attempt, Long interval,
Long defaultMinBackoff) {
// The refresh interval is only updated if it is expired.
if (!Instant.now().isAfter(nextRefreshCheck)) {
return nextRefreshCheck;
}
int durationPeriod = Math.toIntExact(interval);
Instant now = Instant.now();
if (durationPeriod <= defaultMinBackoff) {
return now.plusSeconds(interval);
}
return now.plusNanos(
BackoffTimeCalculator.calculateBackoff(attempt));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy