11

我想从单元测试中测试通知是否能够从资产中播放自定义声音。该测试并不意味着验证任何内容,我将其编写为一种快速演示功能的方法,而不会弄乱主应用程序代码。

所以在测试项目中,我在里面添加了一个wav文件/res/raw。我将在通知生成器中使用此 URL:

Uri path = Uri.parse("android.resource://<main app package name>/testsound.wav");

该 URL 应该根据我在 SO 中阅读的问题起作用。让我们假设它有效。

现在因为我不想在主项目的/res/raw文件夹中包含测试 wav 文件,但在测试项目之一中,我不得不让我的单元测试扩展,InstrumentationTestCase以便我可以访问测试项目中的资源。

这是代码:

    NotificationCompat.Builder builder = new NotificationCompat.Builder(getInstrumentation().getContext());
    ...
    builder.setSound(path, AudioManager.STREAM_NOTIFICATION);
    ...
    NotificationManager notificationManager = (NotificationManager) getInstrumentation().getContext().getSystemService(Context.NOTIFICATION_SERVICE);
    notificationManager.notify(NOTIFICATION_ID, builder.build());

notify调用引发以下异常:

    java.lang.SecurityException: Calling uid 10198 gave package <main app package name> which is owned by uid 10199
    at android.os.Parcel.readException(Parcel.java:1540)
    at android.os.Parcel.readException(Parcel.java:1493)
    at android.app.INotificationManager$Stub$Proxy.enqueueNotificationWithTag(INotificationManager.java:611)
    at android.app.NotificationManager.notify(NotificationManager.java:187)
    at android.app.NotificationManager.notify(NotificationManager.java:140)
    ... 
    at android.app.Instrumentation$InstrumentationThread.run(Instrumentation.java:1873)

我已经将此异常跟踪到NotificationManagerService班级:

    void checkCallerIsSystemOrSameApp(String pkg) {
        int uid = Binder.getCallingUid();
        if (UserHandle.getAppId(uid) == Process.SYSTEM_UID || uid == 0) {
            return;
        }
        try {
            ApplicationInfo ai = AppGlobals.getPackageManager().getApplicationInfo(
                    pkg, 0, UserHandle.getCallingUserId());
            if (!UserHandle.isSameApp(ai.uid, uid)) {
                throw new SecurityException("Calling uid " + uid + " gave package"
                        + pkg + " which is owned by uid " + ai.uid);
            }
        } catch (RemoteException re) {
            throw new SecurityException("Unknown package " + pkg + "\n" + re);
        }
    }

显然,该异常与自定义声音没有任何关系,而是与我们正在从InstrumentationTestCase.

有没有办法测试这个?我记得AndroidTestCase过去曾创建过通知,但如果我这样做了,我将无法访问测试 wav 文件。我可以用 wav 创建一个 jar 并将 jar 放到测试项目的 lib 文件夹中,但这会隐藏文件,如果将来需要替换它,其他程序员可能很难找到它。

4

2 回答 2

2

事实上,我对这个问题有点困惑。所以我写了小仪器测试。

为了断言,我将测试标记为仅在 API 23 上运行(getActiveNotifications之前似乎不可用),但它在旧 API 上也能正常工作。

诀窍是使用getTargetContext()而不是getContext()

public final class MainActivityTest extends ActivityUnitTestCase<MainActivity> {

    public MainActivityTest() {
        super(MainActivity.class);
    }

    @TargetApi(Build.VERSION_CODES.M)
    public void testSendNotification() {
        final NotificationManager manager = (NotificationManager) getInstrumentation().getTargetContext().getSystemService(Context.NOTIFICATION_SERVICE);

        manager.cancel(42);
        assertEquals(0, manager.getActiveNotifications().length);

        final NotificationCompat.Builder builder = new NotificationCompat.Builder(getInstrumentation().getTargetContext());
        builder.setContentTitle("Notification test")
                .setAutoCancel(true)
                .setContentText("Hello, Mister Smith")
                .setSmallIcon(R.drawable.ic_launcher_notification);

        manager.notify(42, builder.build());

        assertEquals(1, manager.getActiveNotifications().length);
    }
}

它就像一个魅力:

在此处输入图像描述

希望能帮助到你。

于 2015-12-16T19:17:49.023 回答
0

我想出了如何让声音从AndroidTestCase. wav 文件已添加到测试项目的原始文件夹中,并且未按预期包含在主项目中。

通知的 URL 是这样的:

Uri path = Uri.parse("android.resource://<test project package name>/" + R.raw.air_horn);

生成器如下获得:

NotificationCompat.Builder builder = new NotificationCompat.Builder(getContext());
于 2016-01-12T10:38:24.470 回答