| /* |
| * Copyright (C) 2017 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.testing; |
| |
| import static org.junit.Assert.assertEquals; |
| |
| import android.content.ContentProviderClient; |
| import android.content.Context; |
| import android.os.Bundle; |
| import android.os.RemoteException; |
| import android.os.UserHandle; |
| import android.provider.Settings; |
| import android.test.mock.MockContentProvider; |
| import android.util.Log; |
| |
| import java.util.HashMap; |
| |
| /** |
| * Allows calls to android.provider.Settings to be tested easier. |
| * |
| * This provides a simple copy-on-write implementation of settings that gets cleared |
| * at the end of each test. |
| */ |
| public class TestableSettingsProvider extends MockContentProvider { |
| |
| private static final String TAG = "TestableSettingsProvider"; |
| private static final boolean DEBUG = false; |
| private static final String MY_UNIQUE_KEY = "Key_" + TestableSettingsProvider.class.getName(); |
| private static TestableSettingsProvider sInstance; |
| |
| private final ContentProviderClient mSettings; |
| |
| private final HashMap<String, String> mValues = new HashMap<>(); |
| |
| private TestableSettingsProvider(ContentProviderClient settings) { |
| mSettings = settings; |
| } |
| |
| void clearValuesAndCheck(Context context) { |
| // Ensure we swapped over to use TestableSettingsProvider |
| Settings.Global.clearProviderForTest(); |
| Settings.Secure.clearProviderForTest(); |
| Settings.System.clearProviderForTest(); |
| |
| // putString will eventually invoking the mocked call() method and update mValues |
| Settings.Global.putString(context.getContentResolver(), MY_UNIQUE_KEY, MY_UNIQUE_KEY); |
| Settings.Secure.putString(context.getContentResolver(), MY_UNIQUE_KEY, MY_UNIQUE_KEY); |
| Settings.System.putString(context.getContentResolver(), MY_UNIQUE_KEY, MY_UNIQUE_KEY); |
| // Verify that if any test is using TestableContext, they all have the correct settings |
| // provider. |
| assertEquals("Incorrect settings provider, test using incorrect Context?", MY_UNIQUE_KEY, |
| Settings.Global.getString(context.getContentResolver(), MY_UNIQUE_KEY)); |
| assertEquals("Incorrect settings provider, test using incorrect Context?", MY_UNIQUE_KEY, |
| Settings.Secure.getString(context.getContentResolver(), MY_UNIQUE_KEY)); |
| assertEquals("Incorrect settings provider, test using incorrect Context?", MY_UNIQUE_KEY, |
| Settings.System.getString(context.getContentResolver(), MY_UNIQUE_KEY)); |
| |
| mValues.clear(); |
| } |
| |
| public Bundle call(String method, String arg, Bundle extras) { |
| // Methods are "GET_system", "GET_global", "PUT_secure", etc. |
| int userId = extras.getInt(Settings.CALL_METHOD_USER_KEY, UserHandle.USER_CURRENT); |
| if (userId == UserHandle.USER_CURRENT || userId == UserHandle.USER_CURRENT_OR_SELF) { |
| userId = UserHandle.myUserId(); |
| } |
| |
| final String[] commands = method.split("_", 2); |
| final String op = commands[0]; |
| final String table = commands[1]; |
| |
| String k = key(table, arg, userId); |
| String value; |
| Bundle out = new Bundle(); |
| switch (op) { |
| case "GET": |
| if (mValues.containsKey(k)) { |
| value = mValues.get(k); |
| if (value != null) { |
| out.putString(Settings.NameValueTable.VALUE, value); |
| } |
| } else { |
| // Fall through to real settings. |
| try { |
| if (DEBUG) Log.d(TAG, "Falling through to real settings " + method); |
| // TODO: Add our own version of caching to handle this. |
| Bundle call = mSettings.call(method, arg, extras); |
| call.remove(Settings.CALL_METHOD_TRACK_GENERATION_KEY); |
| return call; |
| } catch (RemoteException e) { |
| throw new RuntimeException(e); |
| } |
| } |
| break; |
| case "PUT": |
| value = extras.getString(Settings.NameValueTable.VALUE, null); |
| mValues.put(k, value); |
| break; |
| default: |
| throw new UnsupportedOperationException("Unknown command " + method); |
| } |
| return out; |
| } |
| |
| private static String key(String table, String key, int userId) { |
| if ("global".equals(table)) { |
| return table + "_" + key; |
| } else { |
| return table + "_" + userId + "_" + key; |
| } |
| |
| } |
| |
| /** |
| * Since the settings provider is cached inside android.provider.Settings, this must |
| * be gotten statically to ensure there is only one instance referenced. |
| */ |
| static TestableSettingsProvider getFakeSettingsProvider(ContentProviderClient settings) { |
| if (sInstance == null) { |
| sInstance = new TestableSettingsProvider(settings); |
| } |
| return sInstance; |
| } |
| } |