Karma Test: PhantomJS: ReferenceError: Can't find variable: Audio

Karma Test: PhantomJS: ReferenceError: Can't find variable: Audio

我正在尝试使用 Karma 和 PhantomJS 来测试我的应用程序,但出现错误:

   PhantomJS 1.9.8 (Windows 7) Controller: SongsCtrl should attach a list of awesomeThings to the scope FAILED
ReferenceError: Can't find variable: Audio

我的 AngularJS 服务中的违规行是:

   var audio = new Audio();

当我在浏览器中 运行 它当然 运行 没问题,但由于它是原生 HTML5 元素,PhantomJS 似乎无法识别它。

有谁知道我该如何解决这个问题?

PhantomJS 1.9.x doesn't support neither video nor audio tags from HTML5. And it seems that v2.0 does not 要么。

但您可以通过在测试中执行以下操作轻松模拟它(我假设您使用的是 Jasmine):

describe('My test', function() {
    var audioOriginal, audioMock;

    beforeEach(function() {
        audioOriginal = window.Audio;
        audioMock = {};
        window.Audio = function() { return audioMock; });
    });

    afterEach(function() {
        window.Audio = audioOriginal;
    });

    it('should call the play method', function() {
        // Arrange
        audioMock.play = jasmine.createSpy('play');

        // Act
        // Calls Audio.play() somehow

        // Assert
        expect(audioMock.play).toHaveBeenCalled();
    });
});

Working Plunker

请注意,每次测试后都会恢复原始音频值。这很重要,因为我们正在更改全局的 window 对象,如果我们不回滚该更改,那么依赖 Audio 的其他测试可能会中断。

如果您在AngularJs中使用它,请尝试使用此代码模拟 $window.Audio:

我假设您正在测试一个 class 包装音频

describe("Audio wrapper", function () {
    var window = {};
    var wrapper;
    beforeEach(function () {
        module(function ($provide) {
            $provide.service('$window', function () {
                this.Audio = jasmine.createSpy('Audio');
            });
        });
        module('myapp');
    });
    beforeEach(
        inject(
            function ($window) {
                window = $window;
            }
        )
    );
    beforeEach(function () {
        wrapper = new AudioWrapper(window);
  /**
   * the wrapper has this.audio= new $window.Audio();
   *    and play method that call this.audio.play();
  */
        wrapper.audio.play = jasmine.createSpy('play');
    }
    it("call play of HtmlElement when we call play in the class", 
      function () {
        wrapper.play();
        expect(wrapper.audio.play).toHaveBeenCalled();
    });
});

PhantomJs 的音频模拟对象:

page.onInitialized = function () {
    page.evaluate(function () {
        window.Audio =  function() { 
            return {
                play: function() {},
                pause: function() {},
                canPlayType: function() {},
                accessKey: function(){},
                accessibleNode: function(){},
                assignedSlot: function(){},
                attributes: function(){},
                audioTracks: function(){},
                autoplay: function(){},
                baseURI: function(){},
                buffered: function(){},
                childElementCount: function(){},
                childNodes: function(){},
                children: function(){},
                classList: function(){},
                className: function(){},
                clientHeight: function(){},
                clientLeft: function(){},
                clientTop: function(){},
                clientWidth: function(){},
                computedName: function(){},
                computedRole: function(){},
                contentEditable: function(){},
                controls: function(){},
                controlsList: function(){},
                crossOrigin: function(){},
                currentSrc: function(){},
                currentTime: function(){},
                dataset: function(){},
                defaultMuted: function(){},
                defaultPlaybackRate: function(){},
                dir: function(){},
                disableRemotePlayback: function(){},
                draggable: function(){},
                duration: function(){},
                ended: function(){},
                error: function(){},
                firstChild: function(){},
                firstElementChild: function(){},
                hidden: function(){},
                id: function(){},
                inert: function(){},
                innerHTML: function(){},
                innerText: function(){},
                inputMode: function(){},
                isConnected: function(){},
                isContentEditable: function(){},
                lang: function(){},
                lastChild: function(){},
                lastElementChild: function(){},
                localName: function(){},
                loop: function(){},
                mediaKeys: function(){},
                muted: function(){},
                namespaceURI: function(){},
                networkState: function(){},
                nextElementSibling: function(){},
                nextSibling: function(){},
                nodeName: function(){},
                nodeType: function(){},
                nodeValue: function(){},
                nonce: function(){},
                offsetHeight: function(){},
                offsetLeft: function(){},
                offsetParent: function(){},
                offsetTop: function(){},
                offsetWidth: function(){},
                onabort: function(){},
                onauxclick: function(){},
                onbeforecopy: function(){},
                onbeforecut: function(){},
                onbeforepaste: function(){},
                onblur: function(){},
                oncancel: function(){},
                oncanplay: function(){},
                oncanplaythrough: function(){},
                onchange: function(){},
                onclick: function(){},
                onclose: function(){},
                oncontextmenu: function(){},
                oncopy: function(){},
                oncuechange: function(){},
                oncut: function(){},
                ondblclick: function(){},
                ondrag: function(){},
                ondragend: function(){},
                ondragenter: function(){},
                ondragleave: function(){},
                ondragover: function(){},
                ondragstart: function(){},
                ondrop: function(){},
                ondurationchange: function(){},
                onemptied: function(){},
                onencrypted: function(){},
                onended: function(){},
                onerror: function(){},
                onfocus: function(){},
                onfullscreenchange: function(){},
                onfullscreenerror: function(){},
                ongotpointercapture: function(){},
                oninput: function(){},
                oninvalid: function(){},
                onkeydown: function(){},
                onkeypress: function(){},
                onkeyup: function(){},
                onload: function(){},
                onloadeddata: function(){},
                onloadedmetadata: function(){},
                onloadstart: function(){},
                onlostpointercapture: function(){},
                onmousedown: function(){},
                onmouseenter: function(){},
                onmouseleave: function(){},
                onmousemove: function(){},
                onmouseout: function(){},
                onmouseover: function(){},
                onmouseup: function(){},
                onmousewheel: function(){},
                onpaste: function(){},
                onpause: function(){},
                onplay: function(){},
                onplaying: function(){},
                onpointercancel: function(){},
                onpointerdown: function(){},
                onpointerenter: function(){},
                onpointerleave: function(){},
                onpointermove: function(){},
                onpointerout: function(){},
                onpointerover: function(){},
                onpointerup: function(){},
                onprogress: function(){},
                onratechange: function(){},
                onreset: function(){},
                onresize: function(){},
                onscroll: function(){},
                onsearch: function(){},
                onseeked: function(){},
                onseeking: function(){},
                onselect: function(){},
                onselectstart: function(){},
                onstalled: function(){},
                onsubmit: function(){},
                onsuspend: function(){},
                ontimeupdate: function(){},
                ontoggle: function(){},
                ontouchcancel: function(){},
                ontouchend: function(){},
                ontouchmove: function(){},
                ontouchstart: function(){},
                onvolumechange: function(){},
                onwaiting: function(){},
                onwaitingforkey: function(){},
                onwebkitfullscreenchange: function(){},
                onwebkitfullscreenerror: function(){},
                onwheel: function(){},
                outerHTML: function(){},
                outerText: function(){},
                ownerDocument: function(){},
                parentElement: function(){},
                parentNode: function(){},
                paused: function(){},
                playbackRate: function(){},
                played: function(){},
                prefix: function(){},
                preload: function(){},
                previousElementSibling: function(){},
                previousSibling: function(){},
                readyState: function(){},
                remote: function(){},
                scrollHeight: function(){},
                scrollLeft: function(){},
                scrollTop: function(){},
                scrollWidth: function(){},
                seekable: function(){},
                seeking: function(){},
                shadowRoot: function(){},
                sinkId: function(){},
                slot: function(){},
                spellcheck: function(){},
                src: function(){},
                srcObject: function(){},
                style: function(){},
                styleMap: function(){},
                tabIndex: function(){},
                tagName: function(){},
                textContent: function(){},
                textTracks: function(){},
                title: function(){},
                translate: function(){},
                videoTracks: function(){},
                volume: function(){}

            }; 
        };
    });
};