SendBug.java revision 818b58c3cc3d9f99d65bc9e4e7ff5b80021079ec
18c658e058446ad69fb056b2160340d708582b9eeGuang Zhu/*
28c658e058446ad69fb056b2160340d708582b9eeGuang Zhu * Copyright (C) 2011 The Android Open Source Project
38c658e058446ad69fb056b2160340d708582b9eeGuang Zhu *
48c658e058446ad69fb056b2160340d708582b9eeGuang Zhu * Licensed under the Apache License, Version 2.0 (the "License");
58c658e058446ad69fb056b2160340d708582b9eeGuang Zhu * you may not use this file except in compliance with the License.
68c658e058446ad69fb056b2160340d708582b9eeGuang Zhu * You may obtain a copy of the License at
78c658e058446ad69fb056b2160340d708582b9eeGuang Zhu *
88c658e058446ad69fb056b2160340d708582b9eeGuang Zhu *      http://www.apache.org/licenses/LICENSE-2.0
98c658e058446ad69fb056b2160340d708582b9eeGuang Zhu *
108c658e058446ad69fb056b2160340d708582b9eeGuang Zhu * Unless required by applicable law or agreed to in writing, software
118c658e058446ad69fb056b2160340d708582b9eeGuang Zhu * distributed under the License is distributed on an "AS IS" BASIS,
128c658e058446ad69fb056b2160340d708582b9eeGuang Zhu * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
138c658e058446ad69fb056b2160340d708582b9eeGuang Zhu * See the License for the specific language governing permissions and
148c658e058446ad69fb056b2160340d708582b9eeGuang Zhu * limitations under the License.
158c658e058446ad69fb056b2160340d708582b9eeGuang Zhu */
168c658e058446ad69fb056b2160340d708582b9eeGuang Zhupackage com.android.commands.sendbug;
178c658e058446ad69fb056b2160340d708582b9eeGuang Zhu
188c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport android.accounts.Account;
198c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport android.accounts.IAccountManager;
208c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport android.app.ActivityManagerNative;
218c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport android.app.IActivityManager;
228c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport android.content.Context;
238c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport android.content.Intent;
24d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhuimport android.content.pm.IPackageManager;
25d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhuimport android.content.pm.ResolveInfo;
268c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport android.net.Uri;
278c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport android.os.RemoteException;
288c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport android.os.ServiceManager;
298c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport android.os.SystemProperties;
309ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhuimport android.util.Log;
318c658e058446ad69fb056b2160340d708582b9eeGuang Zhu
328c658e058446ad69fb056b2160340d708582b9eeGuang Zhuimport java.io.File;
3301cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albertimport java.util.ArrayList;
34d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhuimport java.util.List;
359ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhuimport java.util.regex.Pattern;
368c658e058446ad69fb056b2160340d708582b9eeGuang Zhu
378c658e058446ad69fb056b2160340d708582b9eeGuang Zhupublic class SendBug {
388c658e058446ad69fb056b2160340d708582b9eeGuang Zhu
399ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu    private static final String LOG_TAG = SendBug.class.getSimpleName();
409ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu    private static final Pattern EMAIL_REGEX = Pattern.compile(
419ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu            "^[\\w.%+-]+@[\\w.-]+\\.[a-zA-Z]{2,}$");
42d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu    private static final String SEND_BUG_INTENT_ACTION = "android.testing.SEND_BUG";
438c658e058446ad69fb056b2160340d708582b9eeGuang Zhu
448c658e058446ad69fb056b2160340d708582b9eeGuang Zhu    public static void main(String[] args) {
458beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu        if (args.length == 1) {
468c658e058446ad69fb056b2160340d708582b9eeGuang Zhu            new SendBug().run(args[0]);
478beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu        } else if (args.length == 2) {
488beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu            new SendBug().run(args[0], args[1]);
498c658e058446ad69fb056b2160340d708582b9eeGuang Zhu        }
508c658e058446ad69fb056b2160340d708582b9eeGuang Zhu    }
518c658e058446ad69fb056b2160340d708582b9eeGuang Zhu
528c658e058446ad69fb056b2160340d708582b9eeGuang Zhu    private void run(String bugreportPath) {
538beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu        run(bugreportPath, null);
548beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu    }
558beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu
568beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu    private void run(String bugreportPath, String screenShotPath) {
5701cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert        final File bugreport = new File(bugreportPath);
588beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu        File screenShot = null;
598beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu        if (screenShotPath != null) {
608beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu            screenShot = new File(screenShotPath);
618beb39758fe046fc19ebee7317949b9ba7ee97ddGuang Zhu        }
626186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey        final Uri bugreportUri = Uri.fromFile(bugreport);
636186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey        // todo (aalbert): investigate adding a screenshot to BugReporter
646186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey        Intent intent = tryBugReporter(bugreportUri);
656186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey        if (intent == null) {
666186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey            final Uri screenshotUri = screenShot != null
676186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey                    ? Uri.fromFile(screenShot) : null;
686186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey            intent = getSendMailIntent(bugreportUri, screenshotUri);
696186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey        }
706186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey        if (intent != null) {
71818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            final IActivityManager am = ActivityManagerNative.getDefault();
72818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            if (am == null) {
73818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu                Log.e(LOG_TAG, "Cannot get ActivityManager, is the system running?");
74818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu                return;
75818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            }
766186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey            try {
77818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu                am.startActivity(null, intent, intent.getType(), null, null, 0, 0,
786186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey                        null, null, null);
796186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey            } catch (RemoteException e) {
806186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey                // ignore
818c658e058446ad69fb056b2160340d708582b9eeGuang Zhu            }
826186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey        } else {
836186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey            Log.w(LOG_TAG, "Cannot find account to send bugreport, local path: "
846186a9cb55dae698f048616bb82887565d5779dfJeff Sharkey                    + bugreportPath);
858c658e058446ad69fb056b2160340d708582b9eeGuang Zhu        }
868c658e058446ad69fb056b2160340d708582b9eeGuang Zhu    }
878c658e058446ad69fb056b2160340d708582b9eeGuang Zhu
88d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu    private Intent tryBugReporter(Uri bugreportUri) {
8901cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert        final Intent intent = new Intent(SEND_BUG_INTENT_ACTION);
90d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu        intent.setData(bugreportUri);
91818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        final IPackageManager pm = IPackageManager.Stub.asInterface(
92d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu                ServiceManager.getService("package"));
93818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        if (pm == null) {
94818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            Log.e(LOG_TAG, "Cannot get PackageManager, is the system running?");
95818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            return null;
96818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        }
97818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        final List<ResolveInfo> results;
98818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        try {
99818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            results = pm.queryIntentActivities(intent, null, 0, 0);
100818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        } catch (RemoteException e) {
101818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            return null;
102818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        }
103818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        if (results != null && results.size() > 0) {
104818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            final ResolveInfo info = results.get(0);
105818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
106818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            intent.setClassName(info.activityInfo.applicationInfo.packageName,
107818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu                    info.activityInfo.name);
108818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            return intent;
109818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        } else {
110818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            return null;
111d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu        }
112d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu    }
113d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu
11401cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert    private Intent getSendMailIntent(Uri bugreportUri, Uri screenshotUri) {
11501cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert        final Account sendToAccount = findSendToAccount();
11601cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert        final Intent intent = new Intent(Intent.ACTION_SEND);
11746899d1b5d606e9349a40079b60034e3cf6eaafbGuang Zhu        intent.addCategory(Intent.CATEGORY_DEFAULT);
118d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu        intent.setType("application/octet-stream");
1199ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu        intent.putExtra(Intent.EXTRA_SUBJECT, bugreportUri.getLastPathSegment());
12001cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert        final StringBuilder sb = new StringBuilder();
121d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu        sb.append(SystemProperties.get("ro.build.description"));
122d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu        sb.append("\n(Sent from BugMailer)");
1239ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu        intent.putExtra(Intent.EXTRA_TEXT, (CharSequence)sb);
12401cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert        if (screenshotUri != null) {
12501cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert            final ArrayList<Uri> attachments = new ArrayList<Uri>();
12601cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert            attachments.add(bugreportUri);
12701cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert            attachments.add(screenshotUri);
12801cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert            intent.setAction(Intent.ACTION_SEND_MULTIPLE);
12901cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert            intent.putParcelableArrayListExtra(Intent.EXTRA_STREAM, attachments);
13001cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert        } else {
13101cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert            intent.putExtra(Intent.EXTRA_STREAM, bugreportUri);
13201cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert        }
133d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu        if (sendToAccount != null) {
1349ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu            intent.putExtra(Intent.EXTRA_EMAIL, new String[]{sendToAccount.name});
1359ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu            return intent;
136d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu        }
1379ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu        return null;
138d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu    }
139d4cd3249e37e5689fc4a8c2858351ec9bebb467cGuang Zhu
1408c658e058446ad69fb056b2160340d708582b9eeGuang Zhu    private Account findSendToAccount() {
14101cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert        final IAccountManager accountManager = IAccountManager.Stub.asInterface(ServiceManager
1428c658e058446ad69fb056b2160340d708582b9eeGuang Zhu                .getService(Context.ACCOUNT_SERVICE));
143818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        if (accountManager == null) {
144818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            Log.e(LOG_TAG, "Cannot get AccountManager, is the system running?");
145818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu            return null;
146818b58c3cc3d9f99d65bc9e4e7ff5b80021079ecGuang Zhu        }
1478c658e058446ad69fb056b2160340d708582b9eeGuang Zhu        Account[] accounts = null;
1488c658e058446ad69fb056b2160340d708582b9eeGuang Zhu        Account foundAccount = null;
1499ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu        String preferredDomain = SystemProperties.get("sendbug.preferred.domain");
1509ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu        if (!preferredDomain.startsWith("@")) {
1519ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu            preferredDomain = "@" + preferredDomain;
1529ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu        }
1538c658e058446ad69fb056b2160340d708582b9eeGuang Zhu        try {
1548c658e058446ad69fb056b2160340d708582b9eeGuang Zhu            accounts = accountManager.getAccounts(null);
1558c658e058446ad69fb056b2160340d708582b9eeGuang Zhu        } catch (RemoteException e) {
15601cef7d875c77ad5efed9e73d0e3d60a258f79deAlon Albert            // ignore
1578c658e058446ad69fb056b2160340d708582b9eeGuang Zhu        }
1588c658e058446ad69fb056b2160340d708582b9eeGuang Zhu        if (accounts != null) {
1598c658e058446ad69fb056b2160340d708582b9eeGuang Zhu            for (Account account : accounts) {
1609ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                if (EMAIL_REGEX.matcher(account.name).matches()) {
1619ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                    if (!preferredDomain.isEmpty()) {
1629ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                        // if we have a preferred domain and it matches, return; otherwise keep
1639ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                        // looking
1649ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                        if (account.name.endsWith(preferredDomain)) {
1659ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                            return account;
1669ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                        } else {
1679ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                            foundAccount = account;
1689ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                        }
1699ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                        // if we don't have a preferred domain, just return since it looks like
1709ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                        // an email address
1719ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                    } else {
1729ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                        return account;
1739ebc639a313d143ade1293cb5932b446d5e623d5Guang Zhu                    }
1748c658e058446ad69fb056b2160340d708582b9eeGuang Zhu                }
1758c658e058446ad69fb056b2160340d708582b9eeGuang Zhu            }
1768c658e058446ad69fb056b2160340d708582b9eeGuang Zhu        }
1778c658e058446ad69fb056b2160340d708582b9eeGuang Zhu        return foundAccount;
1788c658e058446ad69fb056b2160340d708582b9eeGuang Zhu    }
1798c658e058446ad69fb056b2160340d708582b9eeGuang Zhu}
180