From ea4f8ec2686a97b192a1ff1fc7a38f3e352858c0 Mon Sep 17 00:00:00 2001 From: Jae Kim Date: Thu, 4 Jan 2024 16:13:53 -0800 Subject: [PATCH 1/3] remove support for legacy ups --- android-sdk/build.gradle | 4 +- datafile-handler/build.gradle | 4 +- event-handler/build.gradle | 4 +- odp/build.gradle | 4 +- shared/build.gradle | 4 +- test-app/build.gradle | 4 +- user-profile/build.gradle | 4 +- .../DefaultUserProfileServiceTest.java | 4 +- .../user_profile/LegacyDiskCacheTest.java | 132 ------------------ .../user_profile/UserProfileCacheTest.java | 4 +- .../DefaultUserProfileService.java | 17 ++- .../user_profile/UserProfileCache.java | 121 +--------------- 12 files changed, 29 insertions(+), 277 deletions(-) delete mode 100644 user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/LegacyDiskCacheTest.java diff --git a/android-sdk/build.gradle b/android-sdk/build.gradle index 845746bf..4f4b8af6 100644 --- a/android-sdk/build.gradle +++ b/android-sdk/build.gradle @@ -45,8 +45,8 @@ android { } compileOptions { - sourceCompatibility JavaVersion.VERSION_1_8 - targetCompatibility JavaVersion.VERSION_1_8 + sourceCompatibility JavaVersion.VERSION_11 + targetCompatibility JavaVersion.VERSION_11 } } diff --git a/datafile-handler/build.gradle b/datafile-handler/build.gradle index 6cdbfc68..e1f46765 100644 --- a/datafile-handler/build.gradle +++ b/datafile-handler/build.gradle @@ -39,8 +39,8 @@ android { } } compileOptions { - sourceCompatibility JavaVersion.VERSION_1_8 - targetCompatibility JavaVersion.VERSION_1_8 + sourceCompatibility JavaVersion.VERSION_11 + targetCompatibility JavaVersion.VERSION_11 } } diff --git a/event-handler/build.gradle b/event-handler/build.gradle index 4e361dd1..e276a421 100644 --- a/event-handler/build.gradle +++ b/event-handler/build.gradle @@ -39,8 +39,8 @@ android { } } compileOptions { - sourceCompatibility JavaVersion.VERSION_1_8 - targetCompatibility JavaVersion.VERSION_1_8 + sourceCompatibility JavaVersion.VERSION_11 + targetCompatibility JavaVersion.VERSION_11 } buildToolsVersion build_tools_version } diff --git a/odp/build.gradle b/odp/build.gradle index 97c81043..5270d7f2 100644 --- a/odp/build.gradle +++ b/odp/build.gradle @@ -42,8 +42,8 @@ android { } } compileOptions { - sourceCompatibility JavaVersion.VERSION_1_8 - targetCompatibility JavaVersion.VERSION_1_8 + sourceCompatibility JavaVersion.VERSION_11 + targetCompatibility JavaVersion.VERSION_11 } buildToolsVersion build_tools_version diff --git a/shared/build.gradle b/shared/build.gradle index 06203d4e..4732e5fa 100644 --- a/shared/build.gradle +++ b/shared/build.gradle @@ -41,8 +41,8 @@ android { } } compileOptions { - sourceCompatibility JavaVersion.VERSION_1_8 - targetCompatibility JavaVersion.VERSION_1_8 + sourceCompatibility JavaVersion.VERSION_11 + targetCompatibility JavaVersion.VERSION_11 } buildToolsVersion build_tools_version } diff --git a/test-app/build.gradle b/test-app/build.gradle index 1e0d1508..efda1ec2 100644 --- a/test-app/build.gradle +++ b/test-app/build.gradle @@ -29,8 +29,8 @@ android { } } compileOptions { - sourceCompatibility JavaVersion.VERSION_1_8 - targetCompatibility JavaVersion.VERSION_1_8 + sourceCompatibility JavaVersion.VERSION_11 + targetCompatibility JavaVersion.VERSION_11 } packagingOptions { resources { diff --git a/user-profile/build.gradle b/user-profile/build.gradle index cfc7fee3..4d0d54e5 100644 --- a/user-profile/build.gradle +++ b/user-profile/build.gradle @@ -39,8 +39,8 @@ android { } } compileOptions { - sourceCompatibility JavaVersion.VERSION_1_8 - targetCompatibility JavaVersion.VERSION_1_8 + sourceCompatibility JavaVersion.VERSION_11 + targetCompatibility JavaVersion.VERSION_11 } } diff --git a/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/DefaultUserProfileServiceTest.java b/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/DefaultUserProfileServiceTest.java index 0390922d..f10968be 100644 --- a/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/DefaultUserProfileServiceTest.java +++ b/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/DefaultUserProfileServiceTest.java @@ -51,7 +51,6 @@ public class DefaultUserProfileServiceTest { private UserProfileCache.DiskCache diskCache; private ExecutorService executor; private Logger logger; - private UserProfileCache.LegacyDiskCache legacyDiskCache; private Map> memoryCache; private String projectId; private UserProfileCache userProfileCache; @@ -66,11 +65,10 @@ public void setup() { logger = mock(Logger.class); cache = new Cache(InstrumentationRegistry.getInstrumentation().getTargetContext(), logger); executor =Executors.newSingleThreadExecutor(); - legacyDiskCache = new UserProfileCache.LegacyDiskCache(cache, executor, logger, projectId); memoryCache = new ConcurrentHashMap<>(); projectId = "123"; diskCache = new UserProfileCache.DiskCache(cache, executor, logger, projectId); - userProfileCache = new UserProfileCache(diskCache, logger, memoryCache, legacyDiskCache); + userProfileCache = new UserProfileCache(diskCache, logger, memoryCache); userProfileService = new DefaultUserProfileService(userProfileCache, logger); // Test data. diff --git a/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/LegacyDiskCacheTest.java b/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/LegacyDiskCacheTest.java deleted file mode 100644 index 1cfa7001..00000000 --- a/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/LegacyDiskCacheTest.java +++ /dev/null @@ -1,132 +0,0 @@ -/**************************************************************************** - * Copyright 2021, Optimizely, Inc. and contributors * - * * - * 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 com.optimizely.ab.android.user_profile; - -import androidx.test.ext.junit.runners.AndroidJUnit4; -import androidx.test.platform.app.InstrumentationRegistry; - -import com.optimizely.ab.android.shared.Cache; - -import org.json.JSONException; -import org.junit.After; -import org.junit.Before; -import org.junit.Test; -import org.junit.runner.RunWith; -import org.slf4j.Logger; - -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.TimeUnit; - -import static junit.framework.Assert.assertEquals; -import static junit.framework.Assert.assertNull; -import static junit.framework.Assert.fail; -import static org.mockito.Matchers.any; -import static org.mockito.Matchers.eq; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.verify; -import static org.mockito.Mockito.when; - -/** - * Tests for {@link UserProfileCache.LegacyDiskCache} - */ -@RunWith(AndroidJUnit4.class) -public class LegacyDiskCacheTest { - - // Runs tasks serially on the calling thread - private ExecutorService executor = Executors.newSingleThreadExecutor(); - private Cache cache; - private Logger logger; - private UserProfileCache.LegacyDiskCache legacyDiskCache; - private String projectId; - - @Before - public void setup() { - logger = mock(Logger.class); - cache = new Cache(InstrumentationRegistry.getInstrumentation().getTargetContext(), logger); - projectId = "123"; - legacyDiskCache = new UserProfileCache.LegacyDiskCache(cache, executor, logger, projectId); - } - - @After - public void teardown() { - cache.delete(legacyDiskCache.getFileName()); - } - - @Test - public void testGetFileName() { - assertEquals("optly-user-profile-123.json", legacyDiskCache.getFileName()); - } - - @Test - public void testLoadWhenNoFile() throws JSONException { - assertNull(legacyDiskCache.load()); - verify(logger).warn("Unable to load file {}.", legacyDiskCache.getFileName()); - verify(logger).info("Legacy user profile cache not found."); - } - - @Test - public void testLoadMalformedCache() throws JSONException { - cache = mock(Cache.class); - when(cache.load(legacyDiskCache.getFileName())).thenReturn("{?}"); - when(cache.delete(legacyDiskCache.getFileName())).thenReturn(true); - legacyDiskCache = new UserProfileCache.LegacyDiskCache(cache, executor, logger, projectId); - - assertNull(legacyDiskCache.load()); - try { - executor.awaitTermination(5, TimeUnit.SECONDS); - } catch (InterruptedException e) { - fail("Timed out"); - } - - verify(logger).info("Deleted legacy user profile from disk."); - verify(logger).warn(eq("Unable to parse legacy user profiles. Will delete legacy user profile cache file."), - any(Exception.class)); - } - - @Test - public void testDelete() throws JSONException { - cache = mock(Cache.class); - when(cache.delete(legacyDiskCache.getFileName())).thenReturn(true); - legacyDiskCache = new UserProfileCache.LegacyDiskCache(cache, executor, logger, projectId); - - legacyDiskCache.delete(); - try { - executor.awaitTermination(5, TimeUnit.SECONDS); - } catch (InterruptedException e) { - fail("Timed out"); - } - - verify(logger).info("Deleted legacy user profile from disk."); - } - - @Test - public void testDeleteFailed() throws JSONException { - cache = mock(Cache.class); - when(cache.delete(legacyDiskCache.getFileName())).thenReturn(false); - legacyDiskCache = new UserProfileCache.LegacyDiskCache(cache, executor, logger, projectId); - - legacyDiskCache.delete(); - try { - executor.awaitTermination(5, TimeUnit.SECONDS); - } catch (InterruptedException e) { - fail("Timed out"); - } - - verify(logger).warn("Unable to delete legacy user profile from disk."); - } -} diff --git a/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/UserProfileCacheTest.java b/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/UserProfileCacheTest.java index acc13b4a..ee4fde74 100644 --- a/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/UserProfileCacheTest.java +++ b/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/UserProfileCacheTest.java @@ -53,7 +53,6 @@ public class UserProfileCacheTest { private Logger logger; private Cache cache; private UserProfileCache.DiskCache diskCache; - private UserProfileCache.LegacyDiskCache legacyDiskCache; private Map> memoryCache; private String projectId; private UserProfileCache userProfileCache; @@ -69,9 +68,8 @@ public void setup() throws JSONException { projectId = "1"; cache = new Cache(InstrumentationRegistry.getInstrumentation().getTargetContext(), logger); diskCache = new UserProfileCache.DiskCache(cache, executor, logger, projectId); - legacyDiskCache = new UserProfileCache.LegacyDiskCache(cache, executor, logger, projectId); memoryCache = new ConcurrentHashMap<>(); - userProfileCache = new UserProfileCache(diskCache, logger, memoryCache, legacyDiskCache); + userProfileCache = new UserProfileCache(diskCache, logger, memoryCache); // Test data. userId1 = "user_1"; diff --git a/user-profile/src/main/java/com/optimizely/ab/android/user_profile/DefaultUserProfileService.java b/user-profile/src/main/java/com/optimizely/ab/android/user_profile/DefaultUserProfileService.java index 11cd64b5..70df153d 100644 --- a/user-profile/src/main/java/com/optimizely/ab/android/user_profile/DefaultUserProfileService.java +++ b/user-profile/src/main/java/com/optimizely/ab/android/user_profile/DefaultUserProfileService.java @@ -62,14 +62,17 @@ public class DefaultUserProfileService implements UserProfileService { */ public static UserProfileService newInstance(@NonNull String projectId, @NonNull Context context) { UserProfileCache userProfileCache = new UserProfileCache( - new UserProfileCache.DiskCache(new Cache(context, LoggerFactory.getLogger(Cache.class)), - Executors.newSingleThreadExecutor(), LoggerFactory.getLogger(UserProfileCache.DiskCache.class), - projectId), + new UserProfileCache.DiskCache( + new Cache( + context, + LoggerFactory.getLogger(Cache.class) + ), + Executors.newSingleThreadExecutor(), + LoggerFactory.getLogger(UserProfileCache.DiskCache.class), + projectId + ), LoggerFactory.getLogger(UserProfileCache.class), - new ConcurrentHashMap>(), - new UserProfileCache.LegacyDiskCache(new Cache(context, LoggerFactory.getLogger(Cache.class)), - Executors.newSingleThreadExecutor(), - LoggerFactory.getLogger(UserProfileCache.LegacyDiskCache.class), projectId)); + new ConcurrentHashMap>()); return new DefaultUserProfileService(userProfileCache, LoggerFactory.getLogger(DefaultUserProfileService.class)); diff --git a/user-profile/src/main/java/com/optimizely/ab/android/user_profile/UserProfileCache.java b/user-profile/src/main/java/com/optimizely/ab/android/user_profile/UserProfileCache.java index 8e90f271..671fd98c 100644 --- a/user-profile/src/main/java/com/optimizely/ab/android/user_profile/UserProfileCache.java +++ b/user-profile/src/main/java/com/optimizely/ab/android/user_profile/UserProfileCache.java @@ -47,15 +47,13 @@ class UserProfileCache { @NonNull @VisibleForTesting protected final DiskCache diskCache; @NonNull private final Logger logger; @NonNull private final Map> memoryCache; - @NonNull private final LegacyDiskCache legacyDiskCache; - UserProfileCache(@NonNull DiskCache diskCache, @NonNull Logger logger, - @NonNull Map> memoryCache, - @NonNull LegacyDiskCache legacyDiskCache) { + UserProfileCache(@NonNull DiskCache diskCache, + @NonNull Logger logger, + @NonNull Map> memoryCache) { this.logger = logger; this.diskCache = diskCache; this.memoryCache = memoryCache; - this.legacyDiskCache = legacyDiskCache; } /** @@ -85,49 +83,6 @@ Map lookup(String userId) { return memoryCache.get(userId); } - /** - * Migrate legacy user profiles if found. - *

- * Note: this will overwrite a newer `UserProfile` cache in the unlikely event that a legacy cache and new cache - * both exist on disk. - */ - @VisibleForTesting - void migrateLegacyUserProfiles() { - JSONObject legacyUserProfilesJson = legacyDiskCache.load(); - - if (legacyUserProfilesJson == null) { - logger.info("No legacy user profiles to migrate."); - return; - } - - try { - Iterator userIdIterator = legacyUserProfilesJson.keys(); - while (userIdIterator.hasNext()) { - String userId = userIdIterator.next(); - JSONObject legacyUserProfileJson = legacyUserProfilesJson.getJSONObject(userId); - - Map> experimentBucketMap = new ConcurrentHashMap<>(); - Iterator experimentIdIterator = legacyUserProfileJson.keys(); - while (experimentIdIterator.hasNext()) { - String experimentId = experimentIdIterator.next(); - String variationId = legacyUserProfileJson.getString(experimentId); - Map decisionMap = new ConcurrentHashMap<>(); - decisionMap.put(variationIdKey, variationId); - experimentBucketMap.put(experimentId, decisionMap); - } - - Map userProfileMap = new ConcurrentHashMap<>(); - userProfileMap.put(userIdKey, userId); - userProfileMap.put(experimentBucketMapKey, experimentBucketMap); - save(userProfileMap); - } - } catch (JSONException e) { - logger.warn("Unable to deserialize legacy user profiles. Will delete legacy user profile cache file.", e); - } finally { - legacyDiskCache.delete(); - } - } - /** * Remove a user profile. * @@ -218,9 +173,6 @@ void save(Map userProfileMap) { * Load the cache from disk to memory. */ void start() { - // Migrate legacy user profiles if found. - migrateLegacyUserProfiles(); - try { JSONObject userProfilesJson = diskCache.load(); Map> userProfilesMap = UserProfileCacheUtils.convertJSONObjectToMap @@ -295,71 +247,4 @@ protected Boolean doInBackground(Void[] params) { task.executeOnExecutor(executor); } } - - /** - * Stores a map of userIds to a map of expIds to variationIds in a file. - * - * @deprecated This class is only used to migrate legacy user profiles to the new {@link UserProfileCache}. - */ - static class LegacyDiskCache { - - private static final String FILE_NAME = "optly-user-profile-%s.json"; - @NonNull private final Cache cache; - @NonNull private final Executor executor; - @NonNull private final Logger logger; - @NonNull private final String projectId; - - LegacyDiskCache(@NonNull Cache cache, @NonNull Executor executor, @NonNull Logger logger, - @NonNull String projectId) { - this.cache = cache; - this.executor = executor; - this.logger = logger; - this.projectId = projectId; - } - - @VisibleForTesting - String getFileName() { - return String.format(FILE_NAME, projectId); - } - - /** - * Load legacy user profiles from disk if found. - */ - @Nullable - JSONObject load() { - String cacheString = cache.load(getFileName()); - - if (cacheString == null) { - logger.info("Legacy user profile cache not found."); - return null; - } - - try { - return new JSONObject(cacheString); - } catch (JSONException e) { - logger.warn("Unable to parse legacy user profiles. Will delete legacy user profile cache file.", e); - delete(); - return null; - } - } - - /** - * Delete the legacy user profile cache from disk in a background thread. - */ - void delete() { - AsyncTask task = new AsyncTask() { - @Override - protected Boolean doInBackground(Void[] params) { - Boolean deleted = cache.delete(getFileName()); - if (deleted) { - logger.info("Deleted legacy user profile from disk."); - } else { - logger.warn("Unable to delete legacy user profile from disk."); - } - return deleted; - } - }; - task.executeOnExecutor(executor); - } - } } From c740c0337d0809b8c66c0a06f5926c8448fada02 Mon Sep 17 00:00:00 2001 From: Jae Kim Date: Fri, 5 Jan 2024 11:41:41 -0800 Subject: [PATCH 2/3] add returnInManThread for async init --- .../ab/android/sdk/OptimizelyManagerTest.java | 72 ++++++++++++++++- .../ab/android/sdk/OptimizelyManager.java | 30 +++++++- .../test_app/Samples/APISamplesInJava.java | 20 ++++- .../test_app/Samples/APISamplesInKotlin.kt | 28 ++++--- .../android/test_app/SplashScreenActivity.kt | 3 +- .../DefaultUserProfileServiceTest.java | 17 ++++ .../DefaultUserProfileService.java | 77 +++++++++++-------- 7 files changed, 197 insertions(+), 50 deletions(-) diff --git a/android-sdk/src/androidTest/java/com/optimizely/ab/android/sdk/OptimizelyManagerTest.java b/android-sdk/src/androidTest/java/com/optimizely/ab/android/sdk/OptimizelyManagerTest.java index 3052612e..1687761c 100644 --- a/android-sdk/src/androidTest/java/com/optimizely/ab/android/sdk/OptimizelyManagerTest.java +++ b/android-sdk/src/androidTest/java/com/optimizely/ab/android/sdk/OptimizelyManagerTest.java @@ -21,6 +21,7 @@ import android.content.pm.PackageInfo; import android.content.pm.PackageManager; import android.os.Build; +import android.util.Log; import androidx.test.ext.junit.runners.AndroidJUnit4; import androidx.test.filters.SdkSuppress; @@ -48,6 +49,8 @@ import org.mockito.stubbing.Answer; import org.slf4j.Logger; +import java.util.Map; +import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.TimeUnit; @@ -58,6 +61,7 @@ import static junit.framework.Assert.assertNull; import static junit.framework.Assert.assertTrue; import static junit.framework.Assert.fail; +import static org.junit.Assert.assertNotEquals; import static org.mockito.Matchers.any; import static org.mockito.Matchers.eq; import static org.mockito.Mockito.doAnswer; @@ -359,7 +363,7 @@ public void injectOptimizely() { UserProfileService userProfileService = mock(UserProfileService.class); OptimizelyStartListener startListener = mock(OptimizelyStartListener.class); - optimizelyManager.setOptimizelyStartListener(startListener); + optimizelyManager.setOptimizelyStartListener(startListener, true); optimizelyManager.injectOptimizely(context, userProfileService, minDatafile); try { executor.awaitTermination(5, TimeUnit.SECONDS); @@ -750,6 +754,72 @@ public void initializeSyncWithResourceDatafileNoCacheWithDefaultParams() { verify(manager).initialize(eq(context), eq(defaultDatafile), eq(true), eq(false)); } + @Test + public void initializeAsyncCallbackInBackgroundThread() throws InterruptedException { + OptimizelyManager optimizelyManager = OptimizelyManager.builder(testProjectId) + .build(InstrumentationRegistry.getInstrumentation().getTargetContext()); + + CountDownLatch latch = new CountDownLatch(1); + + // by default, async init returns in main thread. + // this parameter should be set to false to overrule it. + boolean returnInMainThread = false; + + optimizelyManager.initialize( + InstrumentationRegistry.getInstrumentation().getContext(), + null, + returnInMainThread, + (client) -> { + Log.d("Optly", "[TESTING] " + Thread.currentThread().getName()); + try { + assertNotEquals( + "OptimizelyStartListener should be called in a background thread", + "main", Thread.currentThread().getName() + ); + latch.countDown(); + } catch (AssertionError e) { + // we need catch and silence this assertion error, otherwise it will be caught in OptimizeManager, + // and give a wrong error message. The failure will be detected with the latch timeout below. + } + } + ); + + boolean completed = latch.await(1, TimeUnit.SECONDS); + if (!completed) { + fail("OptimizelyStartListener thread checking failed"); + } + } + + @Test + public void initializeAsyncCallbackInMainThread() throws InterruptedException { + OptimizelyManager optimizelyManager = OptimizelyManager.builder(testProjectId) + .build(InstrumentationRegistry.getInstrumentation().getTargetContext()); + + CountDownLatch latch = new CountDownLatch(1); + + optimizelyManager.initialize( + InstrumentationRegistry.getInstrumentation().getContext(), + null, + (client) -> { + Log.d("Optly", "[TESTING] " + Thread.currentThread().getName()); + try { + assertEquals( + "OptimizelyStartListener should be called in a background thread", + "main", Thread.currentThread().getName() + ); + latch.countDown(); + } catch (AssertionError e) { + // we need catch and silence this assertion error, otherwise it will be caught in OptimizeManager, + // and give a wrong error message. The failure will be detected with the latch timeout below. + } + } + ); + + boolean completed = latch.await(1, TimeUnit.SECONDS); + if (!completed) { + fail("OptimizelyStartListener thread checking failed"); + } + } // Utils diff --git a/android-sdk/src/main/java/com/optimizely/ab/android/sdk/OptimizelyManager.java b/android-sdk/src/main/java/com/optimizely/ab/android/sdk/OptimizelyManager.java index 5776cbf6..3aa6958f 100644 --- a/android-sdk/src/main/java/com/optimizely/ab/android/sdk/OptimizelyManager.java +++ b/android-sdk/src/main/java/com/optimizely/ab/android/sdk/OptimizelyManager.java @@ -97,6 +97,7 @@ public class OptimizelyManager { @Nullable private final String vuid; @Nullable private OptimizelyStartListener optimizelyStartListener; + private boolean returnInMainThreadFromAsyncInit = true; @Nullable private final List defaultDecideOptions; private String sdkVersion = null; @@ -175,8 +176,13 @@ OptimizelyStartListener getOptimizelyStartListener() { return optimizelyStartListener; } - void setOptimizelyStartListener(@Nullable OptimizelyStartListener optimizelyStartListener) { + void setOptimizelyStartListener(@Nullable OptimizelyStartListener optimizelyStartListener, boolean returnInMainThread) { this.optimizelyStartListener = optimizelyStartListener; + this.returnInMainThreadFromAsyncInit = returnInMainThread; + } + + void setOptimizelyStartListener(@Nullable OptimizelyStartListener optimizelyStartListener) { + setOptimizelyStartListener(optimizelyStartListener, true); } private void notifyStartListener() { @@ -398,11 +404,27 @@ public void initialize(@NonNull final Context context, @NonNull OptimizelyStartL * @see #initialize(Context, Integer, OptimizelyStartListener) */ @TargetApi(Build.VERSION_CODES.ICE_CREAM_SANDWICH) - public void initialize(@NonNull final Context context, @RawRes final Integer datafileRes, @NonNull OptimizelyStartListener optimizelyStartListener) { + public void initialize( + @NonNull final Context context, + @RawRes final Integer datafileRes, + @NonNull OptimizelyStartListener optimizelyStartListener) + { + // return in main thread after async completed (backward compatible) + boolean returnInMainThread = true; + initialize(context, datafileRes, returnInMainThread, optimizelyStartListener); + } + + @TargetApi(Build.VERSION_CODES.ICE_CREAM_SANDWICH) + public void initialize( + @NonNull final Context context, + @RawRes final Integer datafileRes, + final boolean returnInMainThread, + @NonNull OptimizelyStartListener optimizelyStartListener) + { if (!isAndroidVersionSupported()) { return; } - setOptimizelyStartListener(optimizelyStartListener); + setOptimizelyStartListener(optimizelyStartListener, returnInMainThread); datafileHandler.downloadDatafile(context, datafileConfig, getDatafileLoadedListener(context,datafileRes)); } @@ -553,7 +575,7 @@ public void onStartComplete(UserProfileService userProfileService) { logger.info("No listener to send Optimizely to"); } } - }); + }, returnInMainThreadFromAsyncInit); } else { if (optimizelyStartListener != null) { diff --git a/test-app/src/main/java/com/optimizely/ab/android/test_app/Samples/APISamplesInJava.java b/test-app/src/main/java/com/optimizely/ab/android/test_app/Samples/APISamplesInJava.java index 3b7016dd..ccf09dfe 100644 --- a/test-app/src/main/java/com/optimizely/ab/android/test_app/Samples/APISamplesInJava.java +++ b/test-app/src/main/java/com/optimizely/ab/android/test_app/Samples/APISamplesInJava.java @@ -90,7 +90,8 @@ static public void samplesAll(Context context) { samplesForDoc_NotificatonListener(context); samplesForDoc_OlderVersions(context); samplesForDoc_ForcedDecision(context); - samplesForDoc_ODP(context); + samplesForDoc_ODP_async(context); + samplesForDoc_ODP_sync(context); } static public void samplesForDecide(Context context) { @@ -859,7 +860,7 @@ static public void samplesForDoc_ForcedDecision(Context context) { success = user.removeAllForcedDecisions(); } - static public void samplesForDoc_ODP(Context context) { + static public void samplesForDoc_ODP_async(Context context) { OptimizelyManager optimizelyManager = OptimizelyManager.builder().withSDKKey("VivZyCGPHY369D4z8T9yG").build(context); optimizelyManager.initialize(context, null, (OptimizelyClient client) -> { OptimizelyUserContext userContext = client.createUserContext("user_123"); @@ -871,4 +872,19 @@ static public void samplesForDoc_ODP(Context context) { }); } + static public void samplesForDoc_ODP_sync(Context context) { + OptimizelyManager optimizelyManager = OptimizelyManager.builder().withSDKKey("VivZyCGPHY369D4z8T9yG").build(context); + + boolean returnInMainThread = false; + + optimizelyManager.initialize(context, null, returnInMainThread, (OptimizelyClient client) -> { + OptimizelyUserContext userContext = client.createUserContext("user_123"); + userContext.fetchQualifiedSegments(); + + Log.d("Optimizely", "[ODP] segments = " + userContext.getQualifiedSegments()); + OptimizelyDecision optDecision = userContext.decide("odp-flag-1"); + Log.d("Optimizely", "[ODP] decision = " + optDecision.toString()); + }); + } + } diff --git a/test-app/src/main/java/com/optimizely/ab/android/test_app/Samples/APISamplesInKotlin.kt b/test-app/src/main/java/com/optimizely/ab/android/test_app/Samples/APISamplesInKotlin.kt index 342bbef9..d2112caf 100644 --- a/test-app/src/main/java/com/optimizely/ab/android/test_app/Samples/APISamplesInKotlin.kt +++ b/test-app/src/main/java/com/optimizely/ab/android/test_app/Samples/APISamplesInKotlin.kt @@ -18,8 +18,6 @@ package com.optimizely.ab.android.test_app import android.content.Context import android.content.IntentFilter import android.net.wifi.WifiManager -import android.os.Parcel -import android.os.Parcelable import android.util.Log import com.optimizely.ab.OptimizelyDecisionContext import com.optimizely.ab.OptimizelyForcedDecision @@ -29,7 +27,6 @@ import com.optimizely.ab.android.event_handler.EventRescheduler import com.optimizely.ab.android.sdk.OptimizelyClient import com.optimizely.ab.android.sdk.OptimizelyManager import com.optimizely.ab.bucketing.UserProfileService -import com.optimizely.ab.config.Variation import com.optimizely.ab.config.parser.JsonParseException import com.optimizely.ab.error.ErrorHandler import com.optimizely.ab.error.RaiseExceptionErrorHandler @@ -40,12 +37,8 @@ import com.optimizely.ab.notification.DecisionNotification import com.optimizely.ab.notification.NotificationHandler import com.optimizely.ab.notification.TrackNotification import com.optimizely.ab.notification.UpdateConfigNotification -import com.optimizely.ab.optimizelyconfig.OptimizelyConfig import com.optimizely.ab.optimizelydecision.OptimizelyDecideOption import com.optimizely.ab.optimizelydecision.OptimizelyDecision -import com.optimizely.ab.optimizelyjson.OptimizelyJSON -import org.slf4j.LoggerFactory -import java.lang.Exception import java.util.* import java.util.concurrent.TimeUnit @@ -76,7 +69,8 @@ object APISamplesInKotlin { samplesForDoc_NotificatonListener(context) samplesForDoc_OlderVersions(context) samplesForDoc_ForcedDecision(context) - samplesForDoc_ODP(context) + samplesForDoc_ODP_async(context) + samplesForDoc_ODP_sync(context) } fun samplesForDecide(context: Context) { @@ -829,7 +823,7 @@ object APISamplesInKotlin { success = user.removeAllForcedDecisions() } - fun samplesForDoc_ODP(context: Context?) { + fun samplesForDoc_ODP_async(context: Context?) { val optimizelyManager = OptimizelyManager.builder().withSDKKey("VivZyCGPHY369D4z8T9yG").build(context) optimizelyManager.initialize(context!!, null) { client: OptimizelyClient -> @@ -842,6 +836,22 @@ object APISamplesInKotlin { } } + fun samplesForDoc_ODP_sync(context: Context?) { + val optimizelyManager = + OptimizelyManager.builder().withSDKKey("VivZyCGPHY369D4z8T9yG").build(context) + + val returnInMainThread = false; + + optimizelyManager.initialize(context!!, null, returnInMainThread) { client: OptimizelyClient -> + val userContext = client.createUserContext("user_123") + userContext!!.fetchQualifiedSegments() + + Log.d("Optimizely", "[ODP] segments = " + userContext.qualifiedSegments) + val optDecision = userContext.decide("odp-flag-1") + Log.d("Optimizely", "[ODP] decision = $optDecision") + } + } + } diff --git a/test-app/src/main/java/com/optimizely/ab/android/test_app/SplashScreenActivity.kt b/test-app/src/main/java/com/optimizely/ab/android/test_app/SplashScreenActivity.kt index 3ee7f38b..36ea2e1a 100644 --- a/test-app/src/main/java/com/optimizely/ab/android/test_app/SplashScreenActivity.kt +++ b/test-app/src/main/java/com/optimizely/ab/android/test_app/SplashScreenActivity.kt @@ -25,6 +25,7 @@ import com.optimizely.ab.android.event_handler.EventRescheduler import com.optimizely.ab.android.sdk.OptimizelyClient import com.optimizely.ab.android.sdk.OptimizelyManager import com.optimizely.ab.android.shared.CountingIdlingResourceManager +import com.optimizely.ab.android.test_app.Samples.APISamplesInJava import com.optimizely.ab.notification.DecisionNotification import com.optimizely.ab.notification.TrackNotification import com.optimizely.ab.notification.UpdateConfigNotification @@ -131,4 +132,4 @@ class SplashScreenActivity : AppCompatActivity() { // The Idling Resource which will be null in production. private val countingIdlingResourceManager: CountingIdlingResourceManager? = null } -} \ No newline at end of file +} diff --git a/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/DefaultUserProfileServiceTest.java b/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/DefaultUserProfileServiceTest.java index f10968be..54448543 100644 --- a/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/DefaultUserProfileServiceTest.java +++ b/user-profile/src/androidTest/java/com/optimizely/ab/android/user_profile/DefaultUserProfileServiceTest.java @@ -29,6 +29,7 @@ import java.util.Map; import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.TimeUnit; @@ -39,6 +40,8 @@ import static junit.framework.Assert.assertTrue; import static junit.framework.Assert.fail; import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.spy; +import static org.mockito.Mockito.verify; /** * Tests for {@link DefaultUserProfileService} @@ -101,6 +104,20 @@ public void teardown() { cache.delete(diskCache.getFileName()); } + @Test + public void startInBackground() throws InterruptedException { + DefaultUserProfileService ups = spy(DefaultUserProfileService.class); + + CountDownLatch latch = new CountDownLatch(1); + ups.startInBackground((u) -> { + latch.countDown(); + }); + + latch.await(3, TimeUnit.SECONDS); + + verify(ups).start(); + } + @Test public void saveAndStartAndLookup() { userProfileService.save(userProfileMap1); diff --git a/user-profile/src/main/java/com/optimizely/ab/android/user_profile/DefaultUserProfileService.java b/user-profile/src/main/java/com/optimizely/ab/android/user_profile/DefaultUserProfileService.java index 70df153d..d5f3aa33 100644 --- a/user-profile/src/main/java/com/optimizely/ab/android/user_profile/DefaultUserProfileService.java +++ b/user-profile/src/main/java/com/optimizely/ab/android/user_profile/DefaultUserProfileService.java @@ -20,6 +20,9 @@ import android.content.Context; import android.os.AsyncTask; import android.annotation.TargetApi; +import android.os.Handler; +import android.os.Looper; + import androidx.annotation.NonNull; import androidx.annotation.Nullable; @@ -32,6 +35,7 @@ import java.util.Map; import java.util.Set; import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; /** @@ -44,8 +48,10 @@ */ public class DefaultUserProfileService implements UserProfileService { - @NonNull private final UserProfileCache userProfileCache; - @NonNull private final Logger logger; + @NonNull + private final UserProfileCache userProfileCache; + @NonNull + private final Logger logger; DefaultUserProfileService(@NonNull UserProfileCache userProfileCache, @NonNull Logger logger) { this.userProfileCache = userProfileCache; @@ -62,20 +68,20 @@ public class DefaultUserProfileService implements UserProfileService { */ public static UserProfileService newInstance(@NonNull String projectId, @NonNull Context context) { UserProfileCache userProfileCache = new UserProfileCache( - new UserProfileCache.DiskCache( - new Cache( - context, - LoggerFactory.getLogger(Cache.class) - ), - Executors.newSingleThreadExecutor(), - LoggerFactory.getLogger(UserProfileCache.DiskCache.class), - projectId + new UserProfileCache.DiskCache( + new Cache( + context, + LoggerFactory.getLogger(Cache.class) ), - LoggerFactory.getLogger(UserProfileCache.class), - new ConcurrentHashMap>()); + Executors.newSingleThreadExecutor(), + LoggerFactory.getLogger(UserProfileCache.DiskCache.class), + projectId + ), + LoggerFactory.getLogger(UserProfileCache.class), + new ConcurrentHashMap>()); return new DefaultUserProfileService(userProfileCache, - LoggerFactory.getLogger(DefaultUserProfileService.class)); + LoggerFactory.getLogger(DefaultUserProfileService.class)); } public interface StartCallback { @@ -83,30 +89,35 @@ public interface StartCallback { } public void startInBackground(final StartCallback callback) { - final DefaultUserProfileService userProfileService = this; + startInBackground(callback, true); + } - AsyncTask initUserProfileTask = new AsyncTask() { - @Override - protected UserProfileService doInBackground(Void[] params) { - userProfileService.start(); - return userProfileService; - } + public void startInBackground(final StartCallback callback, boolean returnOnMainThread) { + final DefaultUserProfileService userProfileService = this; + Handler mainHandler = new Handler(Looper.getMainLooper()); + + Runnable initUserProfileTask = new Runnable() { @Override - protected void onPostExecute(UserProfileService userProfileService) { + public void run() { + userProfileService.start(); + if (callback != null) { - callback.onStartComplete(userProfileService); + if (returnOnMainThread) { + mainHandler.post(new Runnable() { + @Override + public void run() { + callback.onStartComplete(userProfileService); + } + }); + } else { + callback.onStartComplete(userProfileService); + } } } }; - try { - initUserProfileTask.executeOnExecutor(Executors.newSingleThreadExecutor()); - } - catch (Exception e) { - logger.error("Error loading user profile service from AndroidUserProfileServiceDefault"); - callback.onStartComplete(null); - } - + ExecutorService executor = Executors.newSingleThreadExecutor(); + executor.submit(initUserProfileTask); } /** @@ -146,15 +157,15 @@ public void remove(String userId) { public void removeInvalidExperiments(Set validExperiments) { try { userProfileCache.removeInvalidExperiments(validExperiments); - } - catch (Exception e) { + } catch (Exception e) { logger.error("Error calling userProfileCache to remove invalid experiments", e); } } + /** * Remove a decision from a user profile. * - * @param userId the user ID of the decision to remove + * @param userId the user ID of the decision to remove * @param experimentId the experiment ID of the decision to remove */ public void remove(String userId, String experimentId) { From 56d52b791b177920c127d2eec301c2e22743b1cd Mon Sep 17 00:00:00 2001 From: Jae Kim Date: Thu, 11 Jan 2024 09:56:11 -0800 Subject: [PATCH 3/3] clean up --- .../java/com/optimizely/ab/android/sdk/OptimizelyManager.java | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/android-sdk/src/main/java/com/optimizely/ab/android/sdk/OptimizelyManager.java b/android-sdk/src/main/java/com/optimizely/ab/android/sdk/OptimizelyManager.java index 3aa6958f..286881c2 100644 --- a/android-sdk/src/main/java/com/optimizely/ab/android/sdk/OptimizelyManager.java +++ b/android-sdk/src/main/java/com/optimizely/ab/android/sdk/OptimizelyManager.java @@ -182,7 +182,8 @@ void setOptimizelyStartListener(@Nullable OptimizelyStartListener optimizelyStar } void setOptimizelyStartListener(@Nullable OptimizelyStartListener optimizelyStartListener) { - setOptimizelyStartListener(optimizelyStartListener, true); + boolean returnInMainThread = true; + setOptimizelyStartListener(optimizelyStartListener, returnInMainThread); } private void notifyStartListener() {