/* * Copyright 2010 Kevin Gaudin * * 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 org.acra.collector; import java.lang.reflect.Field; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import org.acra.ACRA; import android.content.ContentResolver; import android.content.Context; import android.provider.Settings; import android.provider.Settings.Secure; import android.provider.Settings.System; import static org.acra.ACRA.LOG_TAG; /** * Helper to collect data from {@link System} and {@link Secure} Settings * classes. * * @author Kevin Gaudin * */ final class SettingsCollector { /** * Collect data from {@link System}. This * collector uses reflection to be sure to always get the most accurate data * whatever Android API level it runs on. * * @param ctx * Application context. * @return A human readable String containing one key=value pair per line. */ public static String collectSystemSettings(Context ctx) { final StringBuilder result = new StringBuilder(); final Field[] keys = System.class.getFields(); for (final Field key : keys) { // Avoid retrieving deprecated fields... it is useless, has an // impact on perfs, and the system writes many warnings in the // logcat. if (!key.isAnnotationPresent(Deprecated.class) && key.getType() == String.class) { try { final Object value = System.getString(ctx.getContentResolver(), (String) key.get(null)); if (value != null) { result.append(key.getName()).append("=").append(value).append("\n"); } } catch (IllegalArgumentException e) { ACRA.log.w(LOG_TAG, "Error : ", e); } catch (IllegalAccessException e) { ACRA.log.w(LOG_TAG, "Error : ", e); } } } return result.toString(); } /** * Collect data from {@link Secure}. This * collector uses reflection to be sure to always get the most accurate data * whatever Android API level it runs on. * * @param ctx * Application context. * @return A human readable String containing one key=value pair per line. */ public static String collectSecureSettings(Context ctx) { final StringBuilder result = new StringBuilder(); final Field[] keys = Secure.class.getFields(); for (final Field key : keys) { if (!key.isAnnotationPresent(Deprecated.class) && key.getType() == String.class && isAuthorized(key)) { try { final Object value = Secure.getString(ctx.getContentResolver(), (String) key.get(null)); if (value != null) { result.append(key.getName()).append("=").append(value).append("\n"); } } catch (IllegalArgumentException e) { ACRA.log.w(LOG_TAG, "Error : ", e); } catch (IllegalAccessException e) { ACRA.log.w(LOG_TAG, "Error : ", e); } } } return result.toString(); } /** * Collect data from {@link Settings.Global}. This * collector uses reflection to be sure to always get the most accurate data * whatever Android API level it runs on. * * @param ctx * Application context. * @return A human readable String containing one key=value pair per line. */ public static String collectGlobalSettings(Context ctx) { if (Compatibility.getAPILevel() < Compatibility.VERSION_CODES.JELLY_BEAN_MR1) { return ""; } final StringBuilder result = new StringBuilder(); try { final Class<?> globalClass = Class.forName("android.provider.Settings$Global"); final Field[] keys = globalClass.getFields(); final Method getString = globalClass.getMethod("getString", ContentResolver.class, String.class); for (final Field key : keys) { if (!key.isAnnotationPresent(Deprecated.class) && key.getType() == String.class && isAuthorized(key)) { final Object value = getString.invoke(null, ctx.getContentResolver(), (String) key.get(null)); if (value != null) { result.append(key.getName()).append("=").append(value).append("\n"); } } } } catch (IllegalArgumentException e) { ACRA.log.w(LOG_TAG, "Error : ", e); } catch (IllegalAccessException e) { ACRA.log.w(LOG_TAG, "Error : ", e); } catch (ClassNotFoundException e) { ACRA.log.w(LOG_TAG, "Error : ", e); } catch (SecurityException e) { ACRA.log.w(LOG_TAG, "Error : ", e); } catch (NoSuchMethodException e) { ACRA.log.w(LOG_TAG, "Error : ", e); } catch (InvocationTargetException e) { ACRA.log.w(LOG_TAG, "Error : ", e); } return result.toString(); } private static boolean isAuthorized(Field key) { if (key == null || key.getName().startsWith("WIFI_AP")) { return false; } for (String regex : ACRA.getConfig().excludeMatchingSettingsKeys()) { if(key.getName().matches(regex)) { return false; } } return true; } }