102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey/* 202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * Copyright (C) 2013 The Android Open Source Project 302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * 402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * Licensed under the Apache License, Version 2.0 (the "License"); 502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * you may not use this file except in compliance with the License. 602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * You may obtain a copy of the License at 702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * 802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * http://www.apache.org/licenses/LICENSE-2.0 902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * 1002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * Unless required by applicable law or agreed to in writing, software 1102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * distributed under the License is distributed on an "AS IS" BASIS, 1202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 1302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * See the License for the specific language governing permissions and 1402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * limitations under the License. 1502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey */ 1602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 1702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeypackage com.android.shell; 1802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 1902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport static com.android.shell.BugreportPrefs.STATE_SHOW; 2002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport static com.android.shell.BugreportPrefs.getWarningState; 2102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 2202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.accounts.Account; 2302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.accounts.AccountManager; 2402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.app.Notification; 2502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.app.NotificationManager; 2602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.app.PendingIntent; 2702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.content.BroadcastReceiver; 2802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.content.Context; 2902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.content.Intent; 3002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.net.Uri; 3102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.os.AsyncTask; 32d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkeyimport android.os.FileUtils; 3302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.os.SystemProperties; 3402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.support.v4.content.FileProvider; 35d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkeyimport android.text.format.DateUtils; 3602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport android.util.Patterns; 3702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 3802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport com.google.android.collect.Lists; 3902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 4002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport java.io.File; 4102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeyimport java.util.ArrayList; 4202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 4302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey/** 4402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * Receiver that handles finished bugreports, usually by attaching them to an 4502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * {@link Intent#ACTION_SEND}. 4602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey */ 4702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkeypublic class BugreportReceiver extends BroadcastReceiver { 4802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey private static final String TAG = "Shell"; 4902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 5002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey private static final String AUTHORITY = "com.android.shell"; 5102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 5202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey private static final String EXTRA_BUGREPORT = "android.intent.extra.BUGREPORT"; 5302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey private static final String EXTRA_SCREENSHOT = "android.intent.extra.SCREENSHOT"; 5402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 5502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey /** 56d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkey * Always keep the newest 8 bugreport files; 4 reports and 4 screenshots are 57d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkey * roughly 17MB of disk space. 5802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey */ 59d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkey private static final int MIN_KEEP_COUNT = 8; 60d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkey 61d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkey /** 62d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkey * Always keep bugreports taken in the last week. 63d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkey */ 64d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkey private static final long MIN_KEEP_AGE = DateUtils.WEEK_IN_MILLIS; 6502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 6602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey @Override 6702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey public void onReceive(Context context, Intent intent) { 6802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final File bugreportFile = getFileExtra(intent, EXTRA_BUGREPORT); 6902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final File screenshotFile = getFileExtra(intent, EXTRA_SCREENSHOT); 7002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 7102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey // Files are kept on private storage, so turn into Uris that we can 7202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey // grant temporary permissions for. 7302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final Uri bugreportUri = FileProvider.getUriForFile(context, AUTHORITY, bugreportFile); 7402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final Uri screenshotUri = FileProvider.getUriForFile(context, AUTHORITY, screenshotFile); 7502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 7602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey Intent sendIntent = buildSendIntent(context, bugreportUri, screenshotUri); 7702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey Intent notifIntent; 7802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 7902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey // Send through warning dialog by default 8002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey if (getWarningState(context, STATE_SHOW) == STATE_SHOW) { 8102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey notifIntent = buildWarningIntent(context, sendIntent); 8202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } else { 8302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey notifIntent = sendIntent; 8402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 8502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey notifIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); 8602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 87255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek final Notification.Builder builder = new Notification.Builder(context) 88255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek .setSmallIcon(com.android.internal.R.drawable.stat_sys_adb) 89255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek .setContentTitle(context.getString(R.string.bugreport_finished_title)) 90255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek .setTicker(context.getString(R.string.bugreport_finished_title)) 91255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek .setContentText(context.getString(R.string.bugreport_finished_text)) 92255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek .setContentIntent(PendingIntent.getActivity( 93255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek context, 0, notifIntent, PendingIntent.FLAG_CANCEL_CURRENT)) 94255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek .setAutoCancel(true) 95255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek .setColor(context.getResources().getColor( 96255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek com.android.internal.R.color.system_notification_accent_color)); 97255dd04271088590fedc46c8e22b2fd4ab142d39Selim Cinek 9802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey NotificationManager.from(context).notify(TAG, 0, builder.build()); 9902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 10002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey // Clean up older bugreports in background 10102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final PendingResult result = goAsync(); 10202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey new AsyncTask<Void, Void, Void>() { 10302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey @Override 10402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey protected Void doInBackground(Void... params) { 105d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkey FileUtils.deleteOlderFiles( 106d9526907d1a51ef0b35bfbbeee43fa209d8b5bbfJeff Sharkey bugreportFile.getParentFile(), MIN_KEEP_COUNT, MIN_KEEP_AGE); 10702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey result.finish(); 10802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey return null; 10902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 11002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey }.execute(); 11102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 11202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 11302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey private static Intent buildWarningIntent(Context context, Intent sendIntent) { 11402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final Intent intent = new Intent(context, BugreportWarningActivity.class); 11502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey intent.putExtra(Intent.EXTRA_INTENT, sendIntent); 11602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey return intent; 11702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 11802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 11902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey /** 12002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * Build {@link Intent} that can be used to share the given bugreport. 12102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey */ 12202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey private static Intent buildSendIntent(Context context, Uri bugreportUri, Uri screenshotUri) { 12302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final Intent intent = new Intent(Intent.ACTION_SEND_MULTIPLE); 12402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); 12502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey intent.addCategory(Intent.CATEGORY_DEFAULT); 12602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey intent.setType("application/vnd.android.bugreport"); 12702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 12802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey intent.putExtra(Intent.EXTRA_SUBJECT, bugreportUri.getLastPathSegment()); 12902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey intent.putExtra(Intent.EXTRA_TEXT, SystemProperties.get("ro.build.description")); 13002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 13102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final ArrayList<Uri> attachments = Lists.newArrayList(bugreportUri, screenshotUri); 13202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey intent.putParcelableArrayListExtra(Intent.EXTRA_STREAM, attachments); 13302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 13402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final Account sendToAccount = findSendToAccount(context); 13502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey if (sendToAccount != null) { 13602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey intent.putExtra(Intent.EXTRA_EMAIL, new String[] { sendToAccount.name }); 13702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 13802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 13902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey return intent; 14002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 14102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 14202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey /** 14302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey * Find the best matching {@link Account} based on build properties. 14402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey */ 14502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey private static Account findSendToAccount(Context context) { 14602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final AccountManager am = (AccountManager) context.getSystemService( 14702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey Context.ACCOUNT_SERVICE); 14802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 14902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey String preferredDomain = SystemProperties.get("sendbug.preferred.domain"); 15002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey if (!preferredDomain.startsWith("@")) { 15102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey preferredDomain = "@" + preferredDomain; 15202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 15302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 15402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final Account[] accounts = am.getAccounts(); 15502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey Account foundAccount = null; 15602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey for (Account account : accounts) { 15702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey if (Patterns.EMAIL_ADDRESS.matcher(account.name).matches()) { 15802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey if (!preferredDomain.isEmpty()) { 15902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey // if we have a preferred domain and it matches, return; otherwise keep 16002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey // looking 16102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey if (account.name.endsWith(preferredDomain)) { 16202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey return account; 16302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } else { 16402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey foundAccount = account; 16502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 16602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey // if we don't have a preferred domain, just return since it looks like 16702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey // an email address 16802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } else { 16902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey return account; 17002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 17102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 17202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 17302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey return foundAccount; 17402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 17502ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey 17602ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey private static File getFileExtra(Intent intent, String key) { 17702ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey final String path = intent.getStringExtra(key); 17802ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey if (path != null) { 17902ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey return new File(path); 18002ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } else { 18102ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey return null; 18202ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 18302ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey } 18402ffba940ca96988ed3e7774c606b43c58373b5eJeff Sharkey} 185