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

feat: polling data source now supports one shot configuration #285

Merged
merged 4 commits into from
Jan 8, 2025
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
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
Original file line number Diff line number Diff line change
Expand Up @@ -264,16 +264,27 @@ public DataSource build(ClientContext clientContext) {
}

// To avoid unnecessarily frequent polling requests due to process or application lifecycle, we have added
// this initial delay logic. Calculate how much time has passed since the last update, if that is less than
// the polling interval, delay by the difference, otherwise 0 delay.
// this rate limiting logic. Calculate how much time has passed since the last update, if that is less than
// the polling interval, delay to when the next poll would have occurred, otherwise 0 delay.
long elapsedSinceUpdate = System.currentTimeMillis() - lastUpdated;
long initialDelayMillis = Math.max(pollInterval - elapsedSinceUpdate, 0);

int maxNumPolls = -1; // negative for unlimited number of polls
if (oneShot) {
if (initialDelayMillis > 0) {
clientContext.getBaseLogger().info("One shot polling attempt will be blocked by rate limiting.");
maxNumPolls = 0; // one shot was blocked by rate limiting logic
} else {
maxNumPolls = 1; // one shot was not blocked by rate limiting logic
}
}

return new PollingDataSource(
clientContextImpl.getEvaluationContext(),
clientContextImpl.getDataSourceUpdateSink(),
initialDelayMillis,
pollInterval,
maxNumPolls,
clientContextImpl.getFetcher(),
clientContextImpl.getPlatformState(),
clientContextImpl.getTaskExecutor(),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,11 +17,18 @@
*/
public interface LDClientInterface extends Closeable {
/**
* Checks whether the client is ready to return feature flag values. This is true if either
* the client has successfully connected to LaunchDarkly and received feature flags, or the
* client has been put into offline mode (in which case it will return only default flag values).
* Returns true if the client has successfully connected to LaunchDarkly and received feature flags after
* {@link LDClient#init(Application, LDConfig, LDContext, int)} was called.
*
* @return true if the client is initialized or offline
* Also returns true if the SDK knows it will never be able to fetch flag data (such as when the client is set
* to offline mode or if in one shot configuration, the one shot fails).
*
* Otherwise this returns false until the client is able to retrieve latest feature flag data from
* LaunchDarkly services. This includes not connecting to LaunchDarkly within the start wait time provided to
* {@link LDClient#init(Application, LDConfig, LDContext, int)} even if the SDK has cached feature flags.
*
* @return true if the client is able to retrieve flag data from LaunchDarkly or offline, false if the client has been
* unable to up to this point.
tanderson-ld marked this conversation as resolved.
Show resolved Hide resolved
*/
boolean isInitialized();

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,12 +22,13 @@ final class PollingDataSource implements DataSource {
private final DataSourceUpdateSink dataSourceUpdateSink;
final long initialDelayMillis; // visible for testing
final long pollIntervalMillis; // visible for testing
private final int maxNumberOfPolls;
int numberOfPollsRemaining; // visible for testing
private final FeatureFetcher fetcher;
private final PlatformState platformState;
private final TaskExecutor taskExecutor;
private final LDLogger logger;
private final AtomicReference<ScheduledFuture<?>> currentPollTask =
new AtomicReference<>();
final AtomicReference<ScheduledFuture<?>> currentPollTask = new AtomicReference<>(); // visible for testing

/**
* @param context that this data source will fetch data for
Expand All @@ -36,6 +37,7 @@ final class PollingDataSource implements DataSource {
* source will report success immediately as it is now running even if data has not been
* fetched.
* @param pollIntervalMillis interval in millis between each polling request
* @param maxNumberOfPolls the maximum number of polling attempts, unlimited if negative is provided
* @param fetcher that will be used for each fetch
* @param platformState used for making decisions based on platform state
* @param taskExecutor that will be used to schedule the polling tasks
Expand All @@ -46,6 +48,7 @@ final class PollingDataSource implements DataSource {
DataSourceUpdateSink dataSourceUpdateSink,
long initialDelayMillis,
long pollIntervalMillis,
int maxNumberOfPolls,
FeatureFetcher fetcher,
PlatformState platformState,
TaskExecutor taskExecutor,
Expand All @@ -55,6 +58,8 @@ final class PollingDataSource implements DataSource {
this.dataSourceUpdateSink = dataSourceUpdateSink;
this.initialDelayMillis = initialDelayMillis;
this.pollIntervalMillis = pollIntervalMillis;
this.maxNumberOfPolls = maxNumberOfPolls;
this.numberOfPollsRemaining = maxNumberOfPolls;
this.fetcher = fetcher;
this.platformState = platformState;
this.taskExecutor = taskExecutor;
Expand All @@ -63,15 +68,16 @@ final class PollingDataSource implements DataSource {

@Override
public void start(final Callback<Boolean> resultCallback) {

if (initialDelayMillis > 0) {
// if there is an initial delay, we will immediately report the successful start of the data source
if (maxNumberOfPolls == 0) {
// If there are no polls to be made, we will immediately report the successful start of the data source. This
// may seem strange, but one can think of this data source as behaving like a no-op in this configuration.
resultCallback.onSuccess(true);
return;
}
tanderson-ld marked this conversation as resolved.
Show resolved Hide resolved

Runnable pollRunnable = () -> poll(resultCallback);
logger.debug("Scheduling polling task with interval of {}ms, starting after {}ms",
pollIntervalMillis, initialDelayMillis);
logger.debug("Scheduling polling task with interval of {}ms, starting after {}ms, with max number of polls of {}",
pollIntervalMillis, initialDelayMillis, maxNumberOfPolls);
ScheduledFuture<?> task = taskExecutor.startRepeatingTask(pollRunnable,
initialDelayMillis, pollIntervalMillis);
currentPollTask.set(task);
Expand All @@ -87,7 +93,19 @@ public void stop(Callback<Void> completionCallback) {
}

private void poll(Callback<Boolean> resultCallback) {
ConnectivityManager.fetchAndSetData(fetcher, context, dataSourceUpdateSink,
resultCallback, logger);
// poll if there is no max (negative number) or there are polls remaining
if (maxNumberOfPolls < 0 || numberOfPollsRemaining > 0) {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

From reviewer comment: Update to use just Long numberOfPollsRemaining and unlimited will set it to Long.MAX

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done

ConnectivityManager.fetchAndSetData(fetcher, context, dataSourceUpdateSink,
resultCallback, logger);
numberOfPollsRemaining--; // decrementing even when we have unlimited polls has no consequence
}
tanderson-ld marked this conversation as resolved.
Show resolved Hide resolved

// terminate if we have a max number of polls and no polls remaining
if (maxNumberOfPolls >= 0 && numberOfPollsRemaining <= 0) {
ScheduledFuture<?> task = currentPollTask.getAndSet(null);
if (task != null) {
task.cancel(true);
}
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,13 @@ public abstract class PollingDataSourceBuilder implements ComponentConfigurer<Da
*/
protected int pollIntervalMillis = DEFAULT_POLL_INTERVAL_MILLIS;

/**
* If true, the polling data source will make at most one poll attempt to get
* feature flag updates. The one shot poll may be blocked by rate limiting logic
* and will not be retried if that occurs.
*/
protected boolean oneShot = false;

/**
* Sets the interval between feature flag updates when the application is running in the background.
* <p>
Expand Down Expand Up @@ -80,4 +87,15 @@ public PollingDataSourceBuilder pollIntervalMillis(int pollIntervalMillis) {
DEFAULT_POLL_INTERVAL_MILLIS : pollIntervalMillis;
return this;
}

/**
* Sets the data source to make one and only one attempt to get feature flag updates. The one shot
* poll may be blocked by rate limiting logic and will not be retried if that occurs.
*
* @return the builder
*/
public PollingDataSourceBuilder oneShot() {
this.oneShot = true;
return this;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,9 @@
import static com.launchdarkly.sdk.android.AssertHelpers.requireNoMoreValues;
import static com.launchdarkly.sdk.android.AssertHelpers.requireValue;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotEquals;
import static org.junit.Assert.assertTrue;

import com.launchdarkly.sdk.LDContext;
import com.launchdarkly.sdk.LDValue;
Expand All @@ -23,6 +25,7 @@
import java.util.Map;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;

public class PollingDataSourceTest {
Expand Down Expand Up @@ -82,7 +85,7 @@ public void pollsAreRepeatedAtRegularPollIntervalInForeground() throws Exception
ClientContext clientContext = makeClientContext(false, null);
PollingDataSourceBuilder builder = Components.pollingDataSource()
.backgroundPollIntervalMillis(100000);
((ComponentsImpl.PollingDataSourceBuilderImpl)builder).pollIntervalMillisNoMinimum(200);
((ComponentsImpl.PollingDataSourceBuilderImpl) builder).pollIntervalMillisNoMinimum(200);
DataSource ds = builder.build(clientContext);

fetcher.setupSuccessResponse("{}");
Expand Down Expand Up @@ -147,12 +150,46 @@ public void pollingIntervalHonoredAcrossMultipleBuildCalls() throws Exception {
assertNotEquals(0, ds2.initialDelayMillis);
}

@Test
public void oneShotPollingSetsMaxNumberOfPollsTo1() throws Exception {
ClientContextImpl clientContext = makeClientContext(true, null);
PollingDataSourceBuilder builder = Components.pollingDataSource().oneShot();

PollingDataSource ds = (PollingDataSource) builder.build(clientContext);
assertEquals(1, ds.numberOfPollsRemaining);
}

@Test
public void oneShotIsPreventByRateLimiting() throws Exception {
ClientContextImpl clientContext = makeClientContext(true, null);
PollingDataSourceBuilder builder = Components.pollingDataSource()
.pollIntervalMillis(100000).oneShot();

// first build should have no delay
PollingDataSource ds1 = (PollingDataSource) builder.build(clientContext);
assertEquals(1, ds1.numberOfPollsRemaining);
assertEquals(0, ds1.initialDelayMillis);

// simulate successful update of context index timestamp
String hashedContextId = LDUtil.urlSafeBase64HashedContextId(CONTEXT);
String fingerPrint = LDUtil.urlSafeBase64Hash(CONTEXT);
PersistentDataStoreWrapper.PerEnvironmentData perEnvironmentData = clientContext.getPerEnvironmentData();
perEnvironmentData.setContextData(hashedContextId, fingerPrint, new EnvironmentData());
ContextIndex newIndex = perEnvironmentData.getIndex().updateTimestamp(hashedContextId, System.currentTimeMillis());
perEnvironmentData.setIndex(newIndex);

// second build should have a non-zero delay and so one shot is prevented by max number of polls being 0.
PollingDataSource ds2 = (PollingDataSource) builder.build(clientContext);
assertEquals(0, ds2.numberOfPollsRemaining);
assertNotEquals(0, ds2.initialDelayMillis);
}

@Test
public void pollsAreRepeatedAtBackgroundPollIntervalInBackground() throws Exception {
ClientContext clientContext = makeClientContext(true, null);
PollingDataSourceBuilder builder = Components.pollingDataSource()
.pollIntervalMillis(100000);
((ComponentsImpl.PollingDataSourceBuilderImpl)builder).backgroundPollIntervalMillisNoMinimum(200);
((ComponentsImpl.PollingDataSourceBuilderImpl) builder).backgroundPollIntervalMillisNoMinimum(200);
DataSource ds = builder.build(clientContext);

fetcher.setupSuccessResponse("{}");
Expand Down Expand Up @@ -180,7 +217,7 @@ public void dataIsUpdatedAfterEachPoll() throws Exception {
ClientContext clientContext = makeClientContext(false, null);
PollingDataSourceBuilder builder = Components.pollingDataSource()
.backgroundPollIntervalMillis(100000);
((ComponentsImpl.PollingDataSourceBuilderImpl)builder).pollIntervalMillisNoMinimum(200);
((ComponentsImpl.PollingDataSourceBuilderImpl) builder).pollIntervalMillisNoMinimum(200);
DataSource ds = builder.build(clientContext);

EnvironmentData data1 = new DataSetBuilder()
Expand Down Expand Up @@ -213,6 +250,43 @@ public void dataIsUpdatedAfterEachPoll() throws Exception {
}
}

@Test
public void terminatesAfterMaxNumberOfPolls() throws Exception {
ClientContextImpl clientContext = makeClientContext(false, null);
PollingDataSource ds = new PollingDataSource(
clientContext.getEvaluationContext(),
clientContext.getDataSourceUpdateSink(),
0,
50,
2, // maximum number of requests is 2
clientContext.getFetcher(),
clientContext.getPlatformState(),
clientContext.getTaskExecutor(),
clientContext.getBaseLogger()
);

fetcher.setupSuccessResponse("{}");
fetcher.setupSuccessResponse("{}");
fetcher.setupSuccessResponse("{}"); // need a third response to detect if the third request is sent which is a failure

try {
ds.start(LDUtil.noOpCallback());
ScheduledFuture pollTask = ds.currentPollTask.get();
assertFalse(pollTask.isCancelled());

LDContext context1 = requireValue(fetcher.receivedContexts, 5, TimeUnit.MILLISECONDS);
Thread.sleep(50);

LDContext context2 = requireValue(fetcher.receivedContexts, 5, TimeUnit.MILLISECONDS);

// if a third request is sent, this will fail here
requireNoMoreValues(fetcher.receivedContexts, 100, TimeUnit.MILLISECONDS);
assertTrue(pollTask.isCancelled());
} finally {
ds.stop(LDUtil.noOpCallback());
}
}

private class MockFetcher implements FeatureFetcher {
BlockingQueue<LDContext> receivedContexts = new LinkedBlockingQueue<>();
BlockingQueue<MockResponse> responses = new LinkedBlockingQueue<>();
Expand Down
Loading