Add misc logging:

- Button clicks in app detail page:
  uninstall/disable/clear data/clear cache
- More logging in search:
  log inline search result click, and its metadata such as
  name/value/rank.

Fix: 32652772
Fix: 32798201
Test: make RunSettingsRoboTests
Change-Id: I90cef7c9ea9ea69ea9cf083013a38417097f8594
This commit is contained in:
Fan Zhang
2017-03-13 14:37:59 -07:00
parent 26a5a71d31
commit c49ab99007
6 changed files with 76 additions and 37 deletions

View File

@@ -30,7 +30,6 @@ import android.content.pm.IPackageDataObserver;
import android.content.pm.PackageManager; import android.content.pm.PackageManager;
import android.content.pm.ProviderInfo; import android.content.pm.ProviderInfo;
import android.os.Bundle; import android.os.Bundle;
import android.os.Environment;
import android.os.Handler; import android.os.Handler;
import android.os.Message; import android.os.Message;
import android.os.RemoteException; import android.os.RemoteException;
@@ -39,7 +38,6 @@ import android.os.storage.StorageManager;
import android.os.storage.VolumeInfo; import android.os.storage.VolumeInfo;
import android.support.v7.preference.Preference; import android.support.v7.preference.Preference;
import android.support.v7.preference.PreferenceCategory; import android.support.v7.preference.PreferenceCategory;
import android.text.format.Formatter;
import android.util.Log; import android.util.Log;
import android.util.MutableInt; import android.util.MutableInt;
import android.view.View; import android.view.View;
@@ -51,8 +49,6 @@ import com.android.settings.R;
import com.android.settings.Utils; import com.android.settings.Utils;
import com.android.settings.deviceinfo.StorageWizardMoveConfirm; import com.android.settings.deviceinfo.StorageWizardMoveConfirm;
import com.android.settingslib.RestrictedLockUtils; import com.android.settingslib.RestrictedLockUtils;
import com.android.settingslib.applications.ApplicationsState;
import com.android.settingslib.applications.ApplicationsState.AppEntry;
import com.android.settingslib.applications.ApplicationsState.Callbacks; import com.android.settingslib.applications.ApplicationsState.Callbacks;
import com.android.settingslib.applications.StorageStatsSource; import com.android.settingslib.applications.StorageStatsSource;
import com.android.settingslib.applications.StorageStatsSource.AppStorageStats; import com.android.settingslib.applications.StorageStatsSource.AppStorageStats;
@@ -196,6 +192,8 @@ public class AppStorageSettings extends AppInfoWithHeader
} else if (mClearCacheObserver == null) { // Lazy initialization of observer } else if (mClearCacheObserver == null) { // Lazy initialization of observer
mClearCacheObserver = new ClearCacheObserver(); mClearCacheObserver = new ClearCacheObserver();
} }
mMetricsFeatureProvider.action(getContext(),
MetricsEvent.ACTION_SETTINGS_CLEAR_APP_CACHE);
mPm.deleteApplicationCacheFiles(mPackageName, mClearCacheObserver); mPm.deleteApplicationCacheFiles(mPackageName, mClearCacheObserver);
} else if (v == mClearDataButton) { } else if (v == mClearDataButton) {
if (mAppsControlDisallowedAdmin != null && !mAppsControlDisallowedBySystem) { if (mAppsControlDisallowedAdmin != null && !mAppsControlDisallowedBySystem) {
@@ -344,6 +342,7 @@ public class AppStorageSettings extends AppInfoWithHeader
* button for a system package * button for a system package
*/ */
private void initiateClearUserData() { private void initiateClearUserData() {
mMetricsFeatureProvider.action(getContext(), MetricsEvent.ACTION_SETTINGS_CLEAR_APP_DATA);
mClearDataButton.setEnabled(false); mClearDataButton.setEnabled(false);
// Invoke uninstall or clear user data based on sysPackage // Invoke uninstall or clear user data based on sysPackage
String packageName = mAppEntry.info.packageName; String packageName = mAppEntry.info.packageName;

View File

@@ -157,8 +157,6 @@ public class InstalledAppDetails extends AppInfoBase
private static final String KEY_MEMORY = "memory"; private static final String KEY_MEMORY = "memory";
private static final String KEY_VERSION = "app_version"; private static final String KEY_VERSION = "app_version";
private static final String NOTIFICATION_TUNER_SETTING = "show_importance_slider";
private final HashSet<String> mHomePackages = new HashSet<>(); private final HashSet<String> mHomePackages = new HashSet<>();
private boolean mInitialized; private boolean mInitialized;
@@ -735,6 +733,8 @@ public class InstalledAppDetails extends AppInfoBase
new DialogInterface.OnClickListener() { new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int which) { public void onClick(DialogInterface dialog, int which) {
// Disable the app // Disable the app
mMetricsFeatureProvider.action(getContext(),
MetricsEvent.ACTION_SETTINGS_DISABLE_APP);
new DisableChanger(InstalledAppDetails.this, mAppEntry.info, new DisableChanger(InstalledAppDetails.this, mAppEntry.info,
PackageManager.COMPONENT_ENABLED_STATE_DISABLED_USER) PackageManager.COMPONENT_ENABLED_STATE_DISABLED_USER)
.execute((Object)null); .execute((Object)null);
@@ -749,6 +749,8 @@ public class InstalledAppDetails extends AppInfoBase
new DialogInterface.OnClickListener() { new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int which) { public void onClick(DialogInterface dialog, int which) {
// Disable the app and ask for uninstall // Disable the app and ask for uninstall
mMetricsFeatureProvider.action(getContext(),
MetricsEvent.ACTION_SETTINGS_DISABLE_APP);
uninstallPkg(mAppEntry.info.packageName, uninstallPkg(mAppEntry.info.packageName,
false, true); false, true);
} }
@@ -777,13 +779,14 @@ public class InstalledAppDetails extends AppInfoBase
Uri packageURI = Uri.parse("package:"+packageName); Uri packageURI = Uri.parse("package:"+packageName);
Intent uninstallIntent = new Intent(Intent.ACTION_UNINSTALL_PACKAGE, packageURI); Intent uninstallIntent = new Intent(Intent.ACTION_UNINSTALL_PACKAGE, packageURI);
uninstallIntent.putExtra(Intent.EXTRA_UNINSTALL_ALL_USERS, allUsers); uninstallIntent.putExtra(Intent.EXTRA_UNINSTALL_ALL_USERS, allUsers);
mMetricsFeatureProvider.action(
getContext(), MetricsEvent.ACTION_SETTINGS_UNINSTALL_APP);
startActivityForResult(uninstallIntent, REQUEST_UNINSTALL); startActivityForResult(uninstallIntent, REQUEST_UNINSTALL);
mDisableAfterUninstall = andDisable; mDisableAfterUninstall = andDisable;
} }
private void forceStopPackage(String pkgName) { private void forceStopPackage(String pkgName) {
FeatureFactory.getFactory(getContext()).getMetricsFeatureProvider().action(getContext(), mMetricsFeatureProvider.action(getContext(), MetricsEvent.ACTION_APP_FORCE_STOP, pkgName);
MetricsEvent.ACTION_APP_FORCE_STOP, pkgName);
ActivityManager am = (ActivityManager) getActivity().getSystemService( ActivityManager am = (ActivityManager) getActivity().getSystemService(
Context.ACTIVITY_SERVICE); Context.ACTIVITY_SERVICE);
Log.d(LOG_TAG, "Stopping package " + pkgName); Log.d(LOG_TAG, "Stopping package " + pkgName);
@@ -802,7 +805,7 @@ public class InstalledAppDetails extends AppInfoBase
mForceStopButton.setEnabled(false); mForceStopButton.setEnabled(false);
} else { } else {
mForceStopButton.setEnabled(enabled); mForceStopButton.setEnabled(enabled);
mForceStopButton.setOnClickListener(InstalledAppDetails.this); mForceStopButton.setOnClickListener(this);
} }
} }
@@ -875,6 +878,8 @@ public class InstalledAppDetails extends AppInfoBase
Intent uninstallDAIntent = new Intent(activity, DeviceAdminAdd.class); Intent uninstallDAIntent = new Intent(activity, DeviceAdminAdd.class);
uninstallDAIntent.putExtra(DeviceAdminAdd.EXTRA_DEVICE_ADMIN_PACKAGE_NAME, uninstallDAIntent.putExtra(DeviceAdminAdd.EXTRA_DEVICE_ADMIN_PACKAGE_NAME,
mPackageName); mPackageName);
mMetricsFeatureProvider.action(
activity, MetricsEvent.ACTION_SETTINGS_UNINSTALL_DEVICE_ADMIN);
activity.startActivityForResult(uninstallDAIntent, REQUEST_REMOVE_DEVICE_ADMIN); activity.startActivityForResult(uninstallDAIntent, REQUEST_REMOVE_DEVICE_ADMIN);
return; return;
} }
@@ -895,6 +900,11 @@ public class InstalledAppDetails extends AppInfoBase
showDialogInner(DLG_DISABLE, 0); showDialogInner(DLG_DISABLE, 0);
} }
} else { } else {
mMetricsFeatureProvider.action(
getActivity(),
mAppEntry.info.enabled
? MetricsEvent.ACTION_SETTINGS_DISABLE_APP
: MetricsEvent.ACTION_SETTINGS_ENABLE_APP);
new DisableChanger(this, mAppEntry.info, new DisableChanger(this, mAppEntry.info,
PackageManager.COMPONENT_ENABLED_STATE_DEFAULT) PackageManager.COMPONENT_ENABLED_STATE_DEFAULT)
.execute((Object) null); .execute((Object) null);

View File

@@ -18,10 +18,11 @@
package com.android.settings.search2; package com.android.settings.search2;
import android.content.Context; import android.content.Context;
import android.util.Pair;
import android.view.View; import android.view.View;
import android.widget.CompoundButton;
import android.widget.Switch; import android.widget.Switch;
import com.android.internal.logging.nano.MetricsProto.MetricsEvent;
import com.android.settings.R; import com.android.settings.R;
/** /**
@@ -36,7 +37,7 @@ public class InlineSwitchViewHolder extends SearchViewHolder {
public InlineSwitchViewHolder(View view, Context context) { public InlineSwitchViewHolder(View view, Context context) {
super(view); super(view);
mContext = context; mContext = context;
switchView = (Switch) view.findViewById(R.id.switchView); switchView = view.findViewById(R.id.switchView);
} }
@Override @Override
@@ -47,12 +48,21 @@ public class InlineSwitchViewHolder extends SearchViewHolder {
} }
final InlineSwitchPayload payload = (InlineSwitchPayload) result.payload; final InlineSwitchPayload payload = (InlineSwitchPayload) result.payload;
switchView.setChecked(payload.getSwitchValue(mContext)); switchView.setChecked(payload.getSwitchValue(mContext));
switchView.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { switchView.setOnCheckedChangeListener((buttonView, isChecked) -> {
@Override final Pair<Integer, Object> name = Pair.create(
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { MetricsEvent.FIELD_SETTINGS_SEARCH_INLINE_RESULT_NAME, payload.settingsUri);
final Pair<Integer, Object> value = Pair.create(
MetricsEvent.FIELD_SETTINGS_SEARCH_INLINE_RESULT_VALUE, isChecked
? "checked"
: "not-checked");
final Pair<Integer, Object> rank = Pair.create(
MetricsEvent.FIELD_SETTINGS_SERACH_RESULT_RANK, getAdapterPosition());
mMetricsFeatureProvider.action(mContext,
MetricsEvent.ACTION_CLICK_SETTINGS_SEARCH_INLINE_RESULT,
name, value, rank);
fragment.onSearchResultClicked(); fragment.onSearchResultClicked();
payload.setSwitchValue(mContext, isChecked); payload.setSwitchValue(mContext, isChecked);
}
}); });
} }
} }

View File

@@ -23,8 +23,6 @@ import android.view.View;
import com.android.internal.logging.nano.MetricsProto.MetricsEvent; import com.android.internal.logging.nano.MetricsProto.MetricsEvent;
import com.android.settings.SettingsActivity; import com.android.settings.SettingsActivity;
import com.android.settings.core.instrumentation.MetricsFeatureProvider;
import com.android.settings.overlay.FeatureFactory;
/** /**
* ViewHolder for intent based search results. * ViewHolder for intent based search results.
@@ -32,12 +30,8 @@ import com.android.settings.overlay.FeatureFactory;
*/ */
public class IntentSearchViewHolder extends SearchViewHolder { public class IntentSearchViewHolder extends SearchViewHolder {
private final MetricsFeatureProvider mMetricsFeatureProvider;
public IntentSearchViewHolder(View view) { public IntentSearchViewHolder(View view) {
super(view); super(view);
mMetricsFeatureProvider = FeatureFactory.getFactory(view.getContext())
.getMetricsFeatureProvider();
} }
@Override @Override

View File

@@ -17,8 +17,6 @@ package com.android.settings.search2;
import android.content.Context; import android.content.Context;
import android.content.pm.PackageManager; import android.content.pm.PackageManager;
import android.graphics.drawable.Drawable;
import android.os.UserHandle;
import android.support.v7.widget.RecyclerView; import android.support.v7.widget.RecyclerView;
import android.text.TextUtils; import android.text.TextUtils;
import android.view.View; import android.view.View;
@@ -26,6 +24,8 @@ import android.widget.ImageView;
import android.widget.TextView; import android.widget.TextView;
import com.android.settings.R; import com.android.settings.R;
import com.android.settings.core.instrumentation.MetricsFeatureProvider;
import com.android.settings.overlay.FeatureFactory;
/** /**
* The ViewHolder for the Search RecyclerView. * The ViewHolder for the Search RecyclerView.
@@ -39,12 +39,16 @@ public abstract class SearchViewHolder extends RecyclerView.ViewHolder {
public final TextView breadcrumbView; public final TextView breadcrumbView;
public final ImageView iconView; public final ImageView iconView;
protected final MetricsFeatureProvider mMetricsFeatureProvider;
public SearchViewHolder(View view) { public SearchViewHolder(View view) {
super(view); super(view);
titleView = (TextView) view.findViewById(android.R.id.title); mMetricsFeatureProvider = FeatureFactory.getFactory(view.getContext())
summaryView = (TextView) view.findViewById(android.R.id.summary); .getMetricsFeatureProvider();
iconView = (ImageView) view.findViewById(android.R.id.icon); titleView = view.findViewById(android.R.id.title);
breadcrumbView = (TextView) view.findViewById(R.id.breadcrumb); summaryView = view.findViewById(android.R.id.summary);
iconView = view.findViewById(android.R.id.icon);
breadcrumbView = view.findViewById(R.id.breadcrumb);
} }
public void onBind(SearchFragment fragment, SearchResult result) { public void onBind(SearchFragment fragment, SearchResult result) {

View File

@@ -19,9 +19,10 @@ package com.android.settings.search;
import android.content.Context; import android.content.Context;
import android.graphics.drawable.Drawable; import android.graphics.drawable.Drawable;
import android.util.Pair;
import android.view.LayoutInflater; import android.view.LayoutInflater;
import android.view.View;
import com.android.internal.logging.nano.MetricsProto;
import com.android.settings.R; import com.android.settings.R;
import com.android.settings.SettingsRobolectricTestRunner; import com.android.settings.SettingsRobolectricTestRunner;
import com.android.settings.TestConfig; import com.android.settings.TestConfig;
@@ -29,19 +30,24 @@ import com.android.settings.search2.InlineSwitchPayload;
import com.android.settings.search2.InlineSwitchViewHolder; import com.android.settings.search2.InlineSwitchViewHolder;
import com.android.settings.search2.SearchFragment; import com.android.settings.search2.SearchFragment;
import com.android.settings.search2.SearchResult; import com.android.settings.search2.SearchResult;
import com.android.settings.testutils.FakeFeatureFactory;
import org.junit.Before; import org.junit.Before;
import org.junit.Test; import org.junit.Test;
import org.junit.runner.RunWith; import org.junit.runner.RunWith;
import org.mockito.Answers;
import org.mockito.Mock; import org.mockito.Mock;
import org.mockito.MockitoAnnotations; import org.mockito.MockitoAnnotations;
import org.robolectric.RuntimeEnvironment;
import org.robolectric.annotation.Config; import org.robolectric.annotation.Config;
import org.robolectric.shadows.ShadowApplication; import org.robolectric.util.ReflectionHelpers;
import java.util.ArrayList; import java.util.ArrayList;
import static com.google.common.truth.Truth.assertThat; import static com.google.common.truth.Truth.assertThat;
import static org.mockito.Matchers.any; import static org.mockito.Matchers.any;
import static org.mockito.Matchers.eq;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when; import static org.mockito.Mockito.when;
@RunWith(SettingsRobolectricTestRunner.class) @RunWith(SettingsRobolectricTestRunner.class)
@@ -51,22 +57,31 @@ public class InlineSwitchViewHolderTest {
private static final String TITLE = "title"; private static final String TITLE = "title";
private static final String SUMMARY = "summary"; private static final String SUMMARY = "summary";
@Mock(answer = Answers.RETURNS_DEEP_STUBS)
private Context mContext;
@Mock @Mock
private SearchFragment mFragment; private SearchFragment mFragment;
@Mock @Mock
private InlineSwitchPayload mPayload; private InlineSwitchPayload mPayload;
private FakeFeatureFactory mFeatureFactory;
private InlineSwitchViewHolder mHolder; private InlineSwitchViewHolder mHolder;
private Drawable mIcon; private Drawable mIcon;
@Before @Before
public void setUp() { public void setUp() {
MockitoAnnotations.initMocks(this); MockitoAnnotations.initMocks(this);
final Context context = ShadowApplication.getInstance().getApplicationContext(); final Context context = RuntimeEnvironment.application;
View view = LayoutInflater.from(context).inflate(R.layout.search_inline_switch_item, null);
mHolder = new InlineSwitchViewHolder(view, context);
mIcon = context.getDrawable(R.drawable.ic_search_history); mIcon = context.getDrawable(R.drawable.ic_search_history);
FakeFeatureFactory.setupForTest(mContext);
mFeatureFactory = (FakeFeatureFactory) FakeFeatureFactory.getFactory(mContext);
mHolder = new InlineSwitchViewHolder(
LayoutInflater.from(context).inflate(R.layout.search_inline_switch_item, null),
context);
ReflectionHelpers.setField(mHolder, "mMetricsFeatureProvider",
mFeatureFactory.metricsFeatureProvider);
} }
@Test @Test
@@ -82,12 +97,19 @@ public class InlineSwitchViewHolderTest {
when(mPayload.getSwitchValue(any(Context.class))).thenReturn(true); when(mPayload.getSwitchValue(any(Context.class))).thenReturn(true);
SearchResult result = getSearchResult(); SearchResult result = getSearchResult();
mHolder.onBind(mFragment, result); mHolder.onBind(mFragment, result);
mHolder.switchView.setChecked(true); // Precondition: switch is on.
assertThat(mHolder.switchView.isChecked()).isTrue();
mHolder.switchView.performClick();
verify(mFeatureFactory.metricsFeatureProvider).action(
any(Context.class),
eq(MetricsProto.MetricsEvent.ACTION_CLICK_SETTINGS_SEARCH_INLINE_RESULT),
any(Pair.class), any(Pair.class), any(Pair.class));
assertThat(mHolder.titleView.getText()).isEqualTo(TITLE); assertThat(mHolder.titleView.getText()).isEqualTo(TITLE);
assertThat(mHolder.summaryView.getText()).isEqualTo(SUMMARY); assertThat(mHolder.summaryView.getText()).isEqualTo(SUMMARY);
assertThat(mHolder.iconView.getDrawable()).isEqualTo(mIcon); assertThat(mHolder.iconView.getDrawable()).isEqualTo(mIcon);
assertThat(mHolder.switchView.isChecked()).isTrue(); assertThat(mHolder.switchView.isChecked()).isFalse();
} }
private SearchResult getSearchResult() { private SearchResult getSearchResult() {