/src/com/android/permissioncontroller/permission/service/BaseSearchIndexablesProvider.java

https://github.com/android/platform_packages_apps_packageinstaller · Java · 137 lines · 77 code · 17 blank · 43 comment · 5 complexity · bd88b3dc9f7e2a6d3581fa1769b876a8 MD5 · raw file

  1. /*
  2. * Copyright (C) 2019 The Android Open Source Project
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package com.android.permissioncontroller.permission.service;
  17. import android.content.Context;
  18. import android.content.Intent;
  19. import android.content.SharedPreferences;
  20. import android.database.Cursor;
  21. import android.database.MatrixCursor;
  22. import android.provider.SearchIndexablesContract;
  23. import android.provider.SearchIndexablesProvider;
  24. import android.util.Log;
  25. import androidx.annotation.CheckResult;
  26. import androidx.annotation.NonNull;
  27. import androidx.annotation.Nullable;
  28. import com.android.permissioncontroller.Constants;
  29. import com.android.permissioncontroller.permission.utils.Utils;
  30. import java.util.Objects;
  31. import java.util.UUID;
  32. /**
  33. * Base class for {@link SearchIndexablesProvider} inside permission controller, which allows using
  34. * a password in raw data key and verifying incoming intents afterwards.
  35. */
  36. public abstract class BaseSearchIndexablesProvider extends SearchIndexablesProvider {
  37. private static final String LOG_TAG = BaseSearchIndexablesProvider.class.getSimpleName();
  38. private static final String EXTRA_SETTINGS_SEARCH_KEY = ":settings:fragment_args_key";
  39. private static final int PASSWORD_LENGTH = 36;
  40. @NonNull
  41. private static final Object sPasswordLock = new Object();
  42. @Override
  43. public boolean onCreate() {
  44. return true;
  45. }
  46. @Nullable
  47. @Override
  48. public Cursor queryXmlResources(@Nullable String[] projection) {
  49. return new MatrixCursor(SearchIndexablesContract.INDEXABLES_XML_RES_COLUMNS);
  50. }
  51. @Nullable
  52. @Override
  53. public Cursor queryNonIndexableKeys(@Nullable String[] projection) {
  54. return new MatrixCursor(SearchIndexablesContract.NON_INDEXABLES_KEYS_COLUMNS);
  55. }
  56. @NonNull
  57. private static String getPassword(@NonNull Context context) {
  58. synchronized (sPasswordLock) {
  59. SharedPreferences sharedPreferences = Utils.getDeviceProtectedSharedPreferences(
  60. context);
  61. String password = sharedPreferences.getString(
  62. Constants.SEARCH_INDEXABLE_PROVIDER_PASSWORD_KEY, null);
  63. if (password == null) {
  64. password = UUID.randomUUID().toString();
  65. sharedPreferences.edit()
  66. .putString(Constants.SEARCH_INDEXABLE_PROVIDER_PASSWORD_KEY, password)
  67. .apply();
  68. }
  69. return password;
  70. }
  71. }
  72. /**
  73. * Create a unique raw data key with password.
  74. *
  75. * @param key the original key, can be retrieved later with {@link #getOriginalKey(Intent)}
  76. * @param context the context to use
  77. * @return the created raw data key
  78. */
  79. @NonNull
  80. protected static String createRawDataKey(@NonNull String key, @NonNull Context context) {
  81. return getPassword(context) + context.getPackageName() + ',' + key;
  82. }
  83. /**
  84. * Check if the intent contains the properties expected from an intent launched from settings
  85. * search.
  86. *
  87. * @param intent the intent to check
  88. * @param context the context to get password
  89. *
  90. * @return whether the intent is valid
  91. */
  92. @CheckResult
  93. public static boolean isIntentValid(@NonNull Intent intent, @NonNull Context context) {
  94. String key = intent.getStringExtra(EXTRA_SETTINGS_SEARCH_KEY);
  95. String passwordFromIntent = key.substring(0, PASSWORD_LENGTH);
  96. String password = getPassword(context);
  97. boolean verified = Objects.equals(passwordFromIntent, password);
  98. if (!verified) {
  99. Log.w(LOG_TAG, "Invalid password: " + passwordFromIntent);
  100. }
  101. return verified;
  102. }
  103. /**
  104. * Get the original key passed to {@link #createRawDataKey(String, Context)}. Should only be
  105. * called after {@link #isIntentValid(Intent, Context)}.
  106. *
  107. * @param intent the intent to get the original key
  108. *
  109. * @return the original key from the intent, or {@code null} if none
  110. */
  111. @Nullable
  112. public static String getOriginalKey(@NonNull Intent intent) {
  113. String key = intent.getStringExtra(EXTRA_SETTINGS_SEARCH_KEY);
  114. if (key == null) {
  115. return null;
  116. }
  117. int keyStart = key.indexOf(',') + 1;
  118. return keyStart <= key.length() ? key.substring(keyStart) : null;
  119. }
  120. }