summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--res/values/cm_strings.xml2
-rw-r--r--res/xml/privacy_settings_cyanogenmod.xml1
-rw-r--r--src/com/android/settings/cmstats/AnonymousStats.java80
-rw-r--r--src/com/android/settings/cmstats/ReportingService.java56
-rw-r--r--src/com/android/settings/cmstats/ReportingServiceManager.java51
-rw-r--r--src/com/android/settings/cmstats/StatsUploadJobService.java48
6 files changed, 119 insertions, 119 deletions
diff --git a/res/values/cm_strings.xml b/res/values/cm_strings.xml
index 8901ed0..3638d90 100644
--- a/res/values/cm_strings.xml
+++ b/res/values/cm_strings.xml
@@ -792,4 +792,6 @@
<string name="preview_version_title">Version</string>
<string name="preview_country_title">Country</string>
<string name="preview_carrier_title">Carrier</string>
+ <string name="stats_collection_title">Stats collection</string>
+ <string name="stats_collection_summary">When enabled, allows metrics collection.</string>
</resources>
diff --git a/res/xml/privacy_settings_cyanogenmod.xml b/res/xml/privacy_settings_cyanogenmod.xml
index 57e4e86..310e74b 100644
--- a/res/xml/privacy_settings_cyanogenmod.xml
+++ b/res/xml/privacy_settings_cyanogenmod.xml
@@ -33,7 +33,6 @@
android:title="@string/block_notifications_title"
android:summary="@string/block_notifications_summary"
android:fragment="com.android.settings.cyanogenmod.SpamList" />
- </Preference>
<!-- Anonymous statistics - (CMStats) -->
<PreferenceScreen
diff --git a/src/com/android/settings/cmstats/AnonymousStats.java b/src/com/android/settings/cmstats/AnonymousStats.java
index 732a579..d85e24d 100644
--- a/src/com/android/settings/cmstats/AnonymousStats.java
+++ b/src/com/android/settings/cmstats/AnonymousStats.java
@@ -20,20 +20,27 @@ import android.content.Context;
import android.content.SharedPreferences;
import android.os.Bundle;
-import android.util.ArraySet;
+import android.os.UserHandle;
+import android.preference.Preference;
+import android.preference.PreferenceScreen;
+import android.preference.SwitchPreference;
import com.android.settings.R;
import com.android.settings.SettingsPreferenceFragment;
-
-import java.util.Set;
+import cyanogenmod.providers.CMSettings;
public class AnonymousStats extends SettingsPreferenceFragment {
+
private static final String PREF_FILE_NAME = "CMStats";
/* package */ static final String ANONYMOUS_OPT_IN = "pref_anonymous_opt_in";
/* package */ static final String ANONYMOUS_LAST_CHECKED = "pref_anonymous_checked_in";
- /* package */ static final String KEY_JOB_QUEUE = "pref_job_queue";
+ /* package */ static final String KEY_LAST_JOB_ID = "last_job_id";
/* package */ static final int QUEUE_MAX_THRESHOLD = 1000;
+ public static final String KEY_STATS = "stats_collection";
+
+ SwitchPreference mStatsSwitch;
+
public static SharedPreferences getPreferences(Context context) {
return context.getSharedPreferences(PREF_FILE_NAME, 0);
}
@@ -42,62 +49,51 @@ public class AnonymousStats extends SettingsPreferenceFragment {
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
addPreferencesFromResource(R.xml.anonymous_stats);
+ mStatsSwitch = (SwitchPreference) findPreference(KEY_STATS);
}
- public static Set<String> getJobQueue(Context context) {
- return getPreferences(context).getStringSet(KEY_JOB_QUEUE, new ArraySet<String>());
+ @Override
+ public boolean onPreferenceTreeClick(PreferenceScreen preferenceScreen, Preference preference) {
+ if (preference == mStatsSwitch) {
+ boolean checked = mStatsSwitch.isChecked();
+ if (checked) {
+ // clear opt out flags
+ CMSettings.Secure.putIntForUser(getContentResolver(),
+ CMSettings.Secure.STATS_COLLECTION_REPORTED, 0, UserHandle.USER_OWNER);
+ }
+ // will initiate opt out sequence if necessary
+ ReportingServiceManager.setAlarm(getActivity());
+ return true;
+ }
+ return super.onPreferenceTreeClick(preferenceScreen, preference);
}
- public static void clearJobQueue(Context context) {
+ public static void updateLastSynced(Context context) {
getPreferences(context)
.edit()
- .remove(KEY_JOB_QUEUE)
+ .putLong(ANONYMOUS_LAST_CHECKED,System.currentTimeMillis())
.commit();
}
- public static void addJob(Context context, int jobId) {
- Set<String> jobQueue = getJobQueue(context);
- jobQueue.add(String.valueOf(jobId));
-
- getPreferences(context)
- .edit()
- .putStringSet(KEY_JOB_QUEUE, jobQueue)
- .commit();
+ private static int getLastJobId(Context context) {
+ return getPreferences(context).getInt(KEY_LAST_JOB_ID, 0);
}
- public static void removeJob(Context context, int jobId) {
- Set<String> jobQueue = getJobQueue(context);
- jobQueue.remove(String.valueOf(jobId));
+ private static void setLastJobId(Context context, int id) {
getPreferences(context)
.edit()
- .remove(KEY_JOB_QUEUE)
- .commit();
-
- getPreferences(context)
- .edit()
- .putStringSet(KEY_JOB_QUEUE, jobQueue)
+ .putInt(KEY_LAST_JOB_ID, id)
.commit();
}
- /**
- * @param context context to use to get prefs
- * @return Returns the next unused int in the job queue, up until {@link #QUEUE_MAX_THRESHOLD}
- * is reached, then it will return -1
- */
public static int getNextJobId(Context context) {
- Set<String> currentQueue = getJobQueue(context);
-
- if (currentQueue == null) {
- return 1;
- } else if (currentQueue.size() >= QUEUE_MAX_THRESHOLD) {
- return -1;
+ int lastId = getLastJobId(context);
+ if (lastId >= QUEUE_MAX_THRESHOLD) {
+ lastId = 1;
} else {
- int i = 1;
- while (currentQueue.contains(String.valueOf(i))) {
- i++;
- }
- return i;
-
+ lastId += 1;
}
+ setLastJobId(context, lastId);
+ return lastId;
}
}
diff --git a/src/com/android/settings/cmstats/ReportingService.java b/src/com/android/settings/cmstats/ReportingService.java
index 61e7af1..8410143 100644
--- a/src/com/android/settings/cmstats/ReportingService.java
+++ b/src/com/android/settings/cmstats/ReportingService.java
@@ -22,15 +22,18 @@ import android.app.job.JobScheduler;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
-import android.content.SharedPreferences;
import android.os.PersistableBundle;
+import android.os.UserHandle;
import android.util.Log;
+import cyanogenmod.providers.CMSettings;
import java.util.List;
public class ReportingService extends IntentService {
/* package */ static final String TAG = "CMStats";
- private static final boolean DEBUG = false;
+ private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG);
+
+ public static final String EXTRA_OPTING_OUT = "cmstats::opt_out";
public ReportingService() {
super(ReportingService.class.getSimpleName());
@@ -40,42 +43,21 @@ public class ReportingService extends IntentService {
protected void onHandleIntent(Intent intent) {
JobScheduler js = (JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);
- if (AnonymousStats.getNextJobId(this) == -1) {
- // if we've filled up to the threshold, we may have some stale job queue ids, purge them
- // then re-add what hasn't executed yet
- AnonymousStats.clearJobQueue(this);
-
- final List<JobInfo> allPendingJobs = js.getAllPendingJobs();
-
- // add two extra jobs to the size for what we will schedule below so we *always*
- // have room for both.
- if (js.getAllPendingJobs().size() + 2 >= AnonymousStats.QUEUE_MAX_THRESHOLD) {
- // there are still as many actual pending jobs as our threshold allows.
- // since we are past the threshold we will be losing data if we don't schedule
- // another job here, so just clear out all the old data and start fresh
- js.cancelAll();
- } else {
- for (JobInfo pendingJob : allPendingJobs) {
- AnonymousStats.addJob(this, pendingJob.getId());
- }
- }
- }
-
- int cyanogenJobId, cmOrgJobId;
- AnonymousStats.addJob(this, cyanogenJobId = AnonymousStats.getNextJobId(this));
- AnonymousStats.addJob(this, cmOrgJobId = AnonymousStats.getNextJobId(this));
-
- if (DEBUG) Log.d(TAG, "scheduling jobs id: " + cyanogenJobId + ", " + cmOrgJobId);
-
- // get snapshot and persist it
String deviceId = Utilities.getUniqueID(getApplicationContext());
String deviceName = Utilities.getDevice();
String deviceVersion = Utilities.getModVersion();
String deviceCountry = Utilities.getCountryCode(getApplicationContext());
String deviceCarrier = Utilities.getCarrier(getApplicationContext());
String deviceCarrierId = Utilities.getCarrierId(getApplicationContext());
+ boolean optOut = intent.getBooleanExtra(EXTRA_OPTING_OUT, false);
+
+ final int cyanogenJobId = AnonymousStats.getNextJobId(getApplicationContext());
+ final int cmOrgJobId = AnonymousStats.getNextJobId(getApplicationContext());
+
+ if (DEBUG) Log.d(TAG, "scheduling jobs id: " + cyanogenJobId + ", " + cmOrgJobId);
PersistableBundle cyanogenBundle = new PersistableBundle();
+ cyanogenBundle.putBoolean(StatsUploadJobService.KEY_OPT_OUT, optOut);
cyanogenBundle.putString(StatsUploadJobService.KEY_DEVICE_NAME, deviceName);
cyanogenBundle.putString(StatsUploadJobService.KEY_UNIQUE_ID, deviceId);
cyanogenBundle.putString(StatsUploadJobService.KEY_VERSION, deviceVersion);
@@ -84,6 +66,7 @@ public class ReportingService extends IntentService {
cyanogenBundle.putString(StatsUploadJobService.KEY_CARRIER_ID, deviceCarrierId);
cyanogenBundle.putLong(StatsUploadJobService.KEY_TIMESTAMP, System.currentTimeMillis());
+ // get snapshot and persist it
PersistableBundle cmBundle = new PersistableBundle(cyanogenBundle);
// set job types
@@ -110,11 +93,14 @@ public class ReportingService extends IntentService {
.setPersisted(true)
.build());
+ if (optOut) {
+ // we've successfully scheduled the opt out.
+ CMSettings.Secure.putIntForUser(getContentResolver(),
+ CMSettings.Secure.STATS_COLLECTION_REPORTED, 1, UserHandle.USER_OWNER);
+ }
+
// reschedule
- final SharedPreferences prefs = AnonymousStats.getPreferences(this);
- prefs.edit().putLong(AnonymousStats.ANONYMOUS_LAST_CHECKED,
- System.currentTimeMillis()).apply();
- ReportingServiceManager.setAlarm(this, 0);
+ AnonymousStats.updateLastSynced(this);
+ ReportingServiceManager.setAlarm(this);
}
-
}
diff --git a/src/com/android/settings/cmstats/ReportingServiceManager.java b/src/com/android/settings/cmstats/ReportingServiceManager.java
index e7ca159..bce1372 100644
--- a/src/com/android/settings/cmstats/ReportingServiceManager.java
+++ b/src/com/android/settings/cmstats/ReportingServiceManager.java
@@ -18,17 +18,22 @@ package com.android.settings.cmstats;
import android.app.AlarmManager;
import android.app.PendingIntent;
+import android.app.job.JobScheduler;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.SharedPreferences;
+import android.os.UserHandle;
import android.util.Log;
+import cyanogenmod.providers.CMSettings;
public class ReportingServiceManager extends BroadcastReceiver {
private static final long MILLIS_PER_HOUR = 60L * 60L * 1000L;
private static final long MILLIS_PER_DAY = 24L * MILLIS_PER_HOUR;
private static final long UPDATE_INTERVAL = 1L * MILLIS_PER_DAY;
+ private static final String TAG = ReportingServiceManager.class.getSimpleName();
+
public static final String ACTION_LAUNCH_SERVICE =
"com.android.settings.action.TRIGGER_REPORT_METRICS";
public static final String EXTRA_FORCE = "force";
@@ -36,33 +41,41 @@ public class ReportingServiceManager extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(Intent.ACTION_BOOT_COMPLETED)) {
- setAlarm(context, 0);
+ setAlarm(context);
} else if (intent.getAction().equals(ACTION_LAUNCH_SERVICE)){
launchService(context, intent.getBooleanExtra(EXTRA_FORCE, false));
}
}
- public static void setAlarm(Context context, long millisFromNow) {
+ /**
+ * opt out if we haven't yet
+ */
+ public static void initiateOptOut(Context context) {
+ final boolean optOutReported = CMSettings.Secure.getIntForUser(context.getContentResolver(),
+ CMSettings.Secure.STATS_COLLECTION_REPORTED, 0, UserHandle.USER_OWNER) == 1;
+ if (!optOutReported) {
+ Intent intent = new Intent();
+ intent.setClass(context, ReportingService.class);
+ intent.putExtra(ReportingService.EXTRA_OPTING_OUT, true);
+ context.startServiceAsUser(intent, UserHandle.OWNER);
+ }
+ }
+
+ public static void setAlarm(Context context) {
SharedPreferences prefs = AnonymousStats.getPreferences(context);
if (prefs.contains(AnonymousStats.ANONYMOUS_OPT_IN)) {
migrate(context, prefs);
}
if (!Utilities.isStatsCollectionEnabled(context)) {
+ initiateOptOut(context);
return;
}
-
- if (millisFromNow <= 0) {
- long lastSynced = prefs.getLong(AnonymousStats.ANONYMOUS_LAST_CHECKED, 0);
- if (lastSynced == 0) {
- // never synced, so let's fake out that the last sync was just now.
- // this will allow the user tFrame time to opt out before it will start
- // sending up anonymous stats.
- lastSynced = System.currentTimeMillis();
- prefs.edit().putLong(AnonymousStats.ANONYMOUS_LAST_CHECKED, lastSynced).apply();
- Log.d(ReportingService.TAG, "Set alarm for first sync.");
- }
- millisFromNow = (lastSynced + UPDATE_INTERVAL) - System.currentTimeMillis();
+ long lastSynced = prefs.getLong(AnonymousStats.ANONYMOUS_LAST_CHECKED, 0);
+ if (lastSynced == 0) {
+ launchService(context, true); // service will reschedule the next alarm
+ return;
}
+ long millisFromNow = (lastSynced + UPDATE_INTERVAL) - System.currentTimeMillis();
Intent intent = new Intent(ACTION_LAUNCH_SERVICE);
intent.setClass(context, ReportingServiceManager.class);
@@ -70,8 +83,8 @@ public class ReportingServiceManager extends BroadcastReceiver {
AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
alarmManager.set(AlarmManager.RTC_WAKEUP, System.currentTimeMillis() + millisFromNow,
PendingIntent.getBroadcast(context, 0, intent, 0));
- Log.d(ReportingService.TAG, "Next sync attempt in : "
- + millisFromNow / MILLIS_PER_HOUR + " hours");
+ Log.d(TAG, "Next sync attempt in : "
+ + (millisFromNow / MILLIS_PER_HOUR) + " hours");
}
public static void launchService(Context context, boolean force) {
@@ -84,13 +97,13 @@ public class ReportingServiceManager extends BroadcastReceiver {
if (!force) {
long lastSynced = prefs.getLong(AnonymousStats.ANONYMOUS_LAST_CHECKED, 0);
if (lastSynced == 0) {
- setAlarm(context, 0);
+ setAlarm(context);
return;
}
long timeElapsed = System.currentTimeMillis() - lastSynced;
if (timeElapsed < UPDATE_INTERVAL) {
long timeLeft = UPDATE_INTERVAL - timeElapsed;
- Log.d(ReportingService.TAG, "Waiting for next sync : "
+ Log.d(TAG, "Waiting for next sync : "
+ timeLeft / MILLIS_PER_HOUR + " hours");
return;
}
@@ -98,7 +111,7 @@ public class ReportingServiceManager extends BroadcastReceiver {
Intent intent = new Intent();
intent.setClass(context, ReportingService.class);
- context.startService(intent);
+ context.startServiceAsUser(intent, UserHandle.OWNER);
}
private static void migrate(Context context, SharedPreferences prefs) {
diff --git a/src/com/android/settings/cmstats/StatsUploadJobService.java b/src/com/android/settings/cmstats/StatsUploadJobService.java
index 2ff428f..77b26e6 100644
--- a/src/com/android/settings/cmstats/StatsUploadJobService.java
+++ b/src/com/android/settings/cmstats/StatsUploadJobService.java
@@ -43,7 +43,7 @@ import java.util.Map;
public class StatsUploadJobService extends JobService {
private static final String TAG = StatsUploadJobService.class.getSimpleName();
- private static final boolean DEBUG = false;
+ private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG);
public static final String KEY_JOB_TYPE = "job_type";
public static final int JOB_TYPE_CYANOGEN = 1;
@@ -56,6 +56,7 @@ public class StatsUploadJobService extends JobService {
public static final String KEY_CARRIER = "carrier";
public static final String KEY_CARRIER_ID = "carrierId";
public static final String KEY_TIMESTAMP = "timeStamp";
+ public static final String KEY_OPT_OUT = "optOut";
private final Map<JobParameters, StatsUploadTask> mCurrentJobs
= Collections.synchronizedMap(new ArrayMap<JobParameters, StatsUploadTask>());
@@ -107,16 +108,25 @@ public class StatsUploadJobService extends JobService {
String deviceCarrier = extras.getString(KEY_CARRIER);
String deviceCarrierId = extras.getString(KEY_CARRIER_ID);
long timeStamp = extras.getLong(KEY_TIMESTAMP);
+ boolean optOut = extras.getBoolean(KEY_OPT_OUT);
boolean success = false;
+ int jobType = extras.getInt(KEY_JOB_TYPE, -1);
if (!isCancelled()) {
- int jobType = extras.getInt(KEY_JOB_TYPE, -1);
-
switch (jobType) {
case JOB_TYPE_CYANOGEN:
try {
- success = uploadToCyanogen(deviceId, deviceName, deviceVersion,
- deviceCountry, deviceCarrier, deviceCarrierId, timeStamp);
+ JSONObject json = new JSONObject();
+ json.put("optOut", optOut);
+ json.put("uniqueId", deviceId);
+ json.put("deviceName", deviceName);
+ json.put("version", deviceVersion);
+ json.put("country", deviceCountry);
+ json.put("carrier", deviceCarrier);
+ json.put("carrierId", deviceCarrierId);
+ json.put("timestamp", timeStamp);
+
+ success = uploadToCyanogen(json);
} catch (IOException | JSONException e) {
Log.e(TAG, "Could not upload stats checkin to cyanogen server", e);
success = false;
@@ -126,7 +136,7 @@ public class StatsUploadJobService extends JobService {
case JOB_TYPE_CMORG:
try {
success = uploadToCM(deviceId, deviceName, deviceVersion, deviceCountry,
- deviceCarrier, deviceCarrierId);
+ deviceCarrier, deviceCarrierId, optOut);
} catch (IOException e) {
Log.e(TAG, "Could not upload stats checkin to commnity server", e);
success = false;
@@ -138,7 +148,6 @@ public class StatsUploadJobService extends JobService {
if (success) {
// we hit the server, succeed either which way.
mCurrentJobs.remove(mJobParams);
- AnonymousStats.removeJob(StatsUploadJobService.this, mJobParams.getJobId());
}
if (DEBUG)
@@ -151,10 +160,12 @@ public class StatsUploadJobService extends JobService {
private boolean uploadToCM(String deviceId, String deviceName, String deviceVersion,
- String deviceCountry, String deviceCarrier, String deviceCarrierId)
+ String deviceCountry, String deviceCarrier, String deviceCarrierId,
+ boolean optOut)
throws IOException {
final Uri uri = Uri.parse(getString(R.string.stats_cm_url)).buildUpon()
+ .appendQueryParameter("opt_out", optOut ? "1" : "0")
.appendQueryParameter("device_hash", deviceId)
.appendQueryParameter("device_name", deviceName)
.appendQueryParameter("device_version", deviceVersion)
@@ -182,9 +193,7 @@ public class StatsUploadJobService extends JobService {
}
- private boolean uploadToCyanogen(String deviceId, String deviceName, String deviceVersion,
- String deviceCountry, String carrier, String carrierId,
- long timeStamp)
+ private boolean uploadToCyanogen(JSONObject json)
throws IOException, JSONException {
String authToken = getAuthToken();
@@ -192,15 +201,6 @@ public class StatsUploadJobService extends JobService {
Log.w(TAG, "no auth token!");
}
- JSONObject json = new JSONObject();
- json.put("uniqueId", deviceId);
- json.put("deviceName", deviceName);
- json.put("version", deviceVersion);
- json.put("country", deviceCountry);
- json.put("carrier", carrier);
- json.put("carrierId", carrierId);
- json.put("timestamp", timeStamp);
-
URL url = new URL(getString(R.string.stats_cyanogen_url));
HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
try {
@@ -223,9 +223,13 @@ public class StatsUploadJobService extends JobService {
final int responseCode = urlConnection.getResponseCode();
final boolean success = responseCode == HttpURLConnection.HTTP_OK;
+
+ final String response = getResponse(urlConnection, !success);
+ if (DEBUG)
+ Log.d(TAG, "server responseCode: " + responseCode +", response=" + response);
+
if (!success) {
- Log.w(TAG, "failed sending, server returned: " + getResponse(urlConnection,
- !success));
+ Log.w(TAG, "failed sending, server returned: " + response);
}
return success;
} finally {