summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
author Susi Kharraz-Post <susikp@google.com> 2019-04-05 11:35:38 +0000
committer Android (Google) Code Review <android-gerrit@google.com> 2019-04-05 11:35:38 +0000
commit884e901cf46c2de047f11c6e4b44f8da980eaa26 (patch)
tree9a8d5faaf3a24ced99374acbdcc26b83111eea68
parente0283ded0a7b2a2010f614943f939e1adc81ebeb (diff)
parenta7f614519e571f6ef6d72b3c36abba13de78138f (diff)
Merge "Fast follow-on unit tests for HashedStringCache" into qt-dev
-rw-r--r--core/java/android/util/HashedStringCache.java15
-rw-r--r--core/tests/coretests/src/android/util/HashedStringCacheTest.java189
2 files changed, 199 insertions, 5 deletions
diff --git a/core/java/android/util/HashedStringCache.java b/core/java/android/util/HashedStringCache.java
index 8ce85148c7c2..1f2b95650288 100644
--- a/core/java/android/util/HashedStringCache.java
+++ b/core/java/android/util/HashedStringCache.java
@@ -22,6 +22,8 @@ import android.os.Environment;
import android.os.storage.StorageManager;
import android.text.TextUtils;
+import com.android.internal.annotations.VisibleForTesting;
+
import java.io.File;
import java.nio.charset.Charset;
import java.security.MessageDigest;
@@ -32,7 +34,6 @@ import java.security.SecureRandom;
* HashedStringCache provides hashing functionality with an underlying LRUCache and expiring salt.
* Salt and expiration time are being stored under the tag passed in by the calling package --
* intended usage is the calling package name.
- * TODO: Add unit tests b/129870147
* @hide
*/
public class HashedStringCache {
@@ -40,9 +41,12 @@ public class HashedStringCache {
private static final Charset UTF_8 = Charset.forName("UTF-8");
private static final int HASH_CACHE_SIZE = 100;
private static final int HASH_LENGTH = 8;
- private static final String HASH_SALT = "_hash_salt";
- private static final String HASH_SALT_DATE = "_hash_salt_date";
- private static final String HASH_SALT_GEN = "_hash_salt_gen";
+ @VisibleForTesting
+ static final String HASH_SALT = "_hash_salt";
+ @VisibleForTesting
+ static final String HASH_SALT_DATE = "_hash_salt_date";
+ @VisibleForTesting
+ static final String HASH_SALT_GEN = "_hash_salt_gen";
// For privacy we need to rotate the salt regularly
private static final long DAYS_TO_MILLIS = 1000 * 60 * 60 * 24;
private static final int MAX_SALT_DAYS = 100;
@@ -94,7 +98,8 @@ public class HashedStringCache {
*/
public HashResult hashString(Context context, String tag, String clearText,
int saltExpirationDays) {
- if (TextUtils.isEmpty(clearText) || saltExpirationDays == -1) {
+ if (saltExpirationDays == -1 || context == null
+ || TextUtils.isEmpty(clearText) || TextUtils.isEmpty(tag)) {
return null;
}
diff --git a/core/tests/coretests/src/android/util/HashedStringCacheTest.java b/core/tests/coretests/src/android/util/HashedStringCacheTest.java
new file mode 100644
index 000000000000..333db246d637
--- /dev/null
+++ b/core/tests/coretests/src/android/util/HashedStringCacheTest.java
@@ -0,0 +1,189 @@
+/*
+ * Copyright (C) 2019 The Android Open Source Project
+ *
+ * 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 android.util;
+
+import static org.hamcrest.CoreMatchers.is;
+import static org.hamcrest.CoreMatchers.not;
+import static org.hamcrest.CoreMatchers.notNullValue;
+import static org.hamcrest.CoreMatchers.nullValue;
+import static org.hamcrest.MatcherAssert.assertThat;
+
+import android.content.Context;
+import android.content.SharedPreferences;
+import android.os.Environment;
+import android.os.storage.StorageManager;
+
+import androidx.test.InstrumentationRegistry;
+
+import org.junit.Before;
+import org.junit.Test;
+
+import java.io.File;
+
+
+/**
+ * Unit tests for {@link HashedStringCache}.
+ */
+public class HashedStringCacheTest {
+ private static final String TAG = "HashedStringCacheTest";
+ private Context mContext;
+ private static final String TEST_STRING = "test_string";
+
+ @Before
+ public void setup() {
+ mContext = null;
+ mContext = InstrumentationRegistry.getContext();
+ clearSharedPreferences();
+ }
+
+ @Test
+ public void testInstanceNotNull() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ assertThat(cache, is(notNullValue()));
+ }
+
+ @Test
+ public void testInstanceMatchesOnSecondCall() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ assertThat(HashedStringCache.getInstance(), is(cache));
+ }
+
+ @Test
+ public void testHashedStringNotOriginalString() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(mContext, TAG, TEST_STRING, 7);
+ assertThat(cachedResult.hashedString, is(not(TEST_STRING)));
+ }
+
+ @Test
+ public void testThatMultipleCallsResultInSameHash() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(mContext, TAG, TEST_STRING, 7);
+ HashedStringCache.HashResult cachedResult2 =
+ cache.hashString(mContext, TAG, TEST_STRING, 7);
+ assertThat(cachedResult2.hashedString, is(cachedResult.hashedString));
+ }
+
+ @Test
+ public void testThatZeroDaysResultsInNewHash() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(mContext, TAG, TEST_STRING, 7);
+ HashedStringCache.HashResult cachedResult2 =
+ cache.hashString(mContext, TAG, TEST_STRING, 0);
+ assertThat(cachedResult2.hashedString, is(not(cachedResult.hashedString)));
+ }
+
+ @Test
+ public void testThatNegativeDaysResultsInNewHash() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(mContext, TAG, TEST_STRING, 7);
+ HashedStringCache.HashResult cachedResult2 =
+ cache.hashString(mContext, TAG, TEST_STRING, -10);
+ assertThat(cachedResult2.hashedString, is(not(cachedResult.hashedString)));
+ }
+
+ @Test
+ public void testThatDaysGreater365ResultsInSameResult() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(mContext, TAG, TEST_STRING, 7);
+ HashedStringCache.HashResult cachedResult2 =
+ cache.hashString(mContext, TAG, TEST_STRING, 400);
+ assertThat(cachedResult2.hashedString, is(cachedResult.hashedString));
+ }
+
+ /**
+ * -1 is treated as a special input to short-circuit out of doing the hashing to give us
+ * the option to turn this feature off if need be while incurring as little computational cost
+ * as possible.
+ */
+ @Test
+ public void testMinusOneResultsInNull() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(mContext, TAG, TEST_STRING, -1);
+ assertThat(cachedResult, is(nullValue()));
+ }
+
+ @Test
+ public void testEmptyStringInput() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(mContext, TAG, "", -1);
+ assertThat(cachedResult, is(nullValue()));
+ }
+
+ @Test
+ public void testNullInput() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(mContext, TAG, null, -1);
+ assertThat(cachedResult, is(nullValue()));
+ }
+
+ @Test
+ public void testEmptyStringTag() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(mContext, "", TEST_STRING, -1);
+ assertThat(cachedResult, is(nullValue()));
+ }
+
+ @Test
+ public void testNullTag() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(mContext, null, TEST_STRING, -1);
+ assertThat(cachedResult, is(nullValue()));
+ }
+
+ @Test
+ public void testNullContext() {
+ HashedStringCache cache = HashedStringCache.getInstance();
+ HashedStringCache.HashResult cachedResult =
+ cache.hashString(null, TAG, TEST_STRING, -1);
+ assertThat(cachedResult, is(nullValue()));
+ }
+
+ private void clearSharedPreferences() {
+ SharedPreferences preferences = getTestSharedPreferences(mContext);
+ preferences.edit()
+ .remove(TAG + HashedStringCache.HASH_SALT)
+ .remove(TAG + HashedStringCache.HASH_SALT_DATE)
+ .remove(TAG + HashedStringCache.HASH_SALT_GEN).apply();
+ }
+
+ /**
+ * Android:ui doesn't have persistent preferences, so need to fall back on this hack originally
+ * from ChooserActivity.java
+ * @param context
+ * @return
+ */
+ private SharedPreferences getTestSharedPreferences(Context context) {
+ final File prefsFile = new File(new File(
+ Environment.getDataUserCePackageDirectory(
+ StorageManager.UUID_PRIVATE_INTERNAL,
+ context.getUserId(), context.getPackageName()),
+ "shared_prefs"),
+ "hashed_cache_test.xml");
+ return context.getSharedPreferences(prefsFile, Context.MODE_PRIVATE);
+ }
+}