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