86

根据 Jasmine 文档,可以像这样创建模拟:

jasmine.createSpyObj(someObject, ['method1', 'method2', ... ]);

你如何存根这些方法之一?例如,如果你想测试一个方法抛出异常时会发生什么,你会怎么做?

4

4 回答 4

131

正如 EricG 评论的那样,您必须使用 chain method1,但不能使用(或在 2.0 版中)。它将委托给真正的实现method2andCallThrough()and.callThrough()

在这种情况下,您需要链接and.callFake()并传递要调用的函数(可以抛出异常或任何您想要的):

var someObject = jasmine.createSpyObj('someObject', [ 'method1', 'method2' ]);
someObject.method1.and.callFake(function() {
    throw 'an-exception';
});

然后你可以验证:

expect(yourFncCallingMethod1).toThrow('an-exception');
于 2012-11-30T15:19:17.623 回答
23

如果您使用的是 Typescript,将方法转换为Jasmine.Spy. 在上面的答案中(奇怪的是我没有代表发表评论):

(someObject.method1 as Jasmine.Spy).and.callFake(function() {
  throw 'an-exception';
});

我不知道我是否过度设计,因为我缺乏知识......

对于打字稿,我想要:

  • 来自底层类型的智能感知
  • 仅模拟函数中使用的方法的能力

我发现这很有用:

namespace Services {
    class LogService {
        info(message: string, ...optionalParams: any[]) {
            if (optionalParams && optionalParams.length > 0) {
                console.log(message, optionalParams);
                return;
            }

            console.log(message);
        }
    }
}

class ExampleSystemUnderTest {
    constructor(private log: Services.LogService) {
    }

    doIt() {
        this.log.info('done');
    }
}

// I export this in a common test file 
// with other utils that all tests import
const asSpy = f => <jasmine.Spy>f;

describe('SomeTest', () => {
    let log: Services.LogService;
    let sut: ExampleSystemUnderTest;

    // ARRANGE
    beforeEach(() => {
        log = jasmine.createSpyObj('log', ['info', 'error']);
        sut = new ExampleSystemUnderTest(log);
    });

    it('should do', () => {
        // ACT
        sut.doIt();

        // ASSERT
        expect(asSpy(log.error)).not.toHaveBeenCalled();
        expect(asSpy(log.info)).toHaveBeenCalledTimes(1);
        expect(asSpy(log.info).calls.allArgs()).toEqual([
            ['done']
        ]);
    });
});
于 2017-05-17T16:52:01.417 回答
9

角 9

在测试注入简单服务的组件时,使用jasmine.createSpyObj是理想的选择。例如:假设在我的 HomeComponent 中有一个 HomeService(注入)。HomeService 中的唯一方法是 getAddress()。创建 HomeComponent 测试套件时,我可以将组件和服务初始化为:

describe('Home Component', () => {
    let component: HomeComponent;
    let fixture: ComponentFixture<HomeComponent>;
    let element: DebugElement;
    let homeServiceSpy: any;
    let homeService: any;

    beforeEach(async(() => {
        homeServiceSpy = jasmine.createSpyObj('HomeService', ['getAddress']);

        TestBed.configureTestingModule({
           declarations: [HomeComponent],
           providers: [{ provide: HomeService, useValue: homeServiceSpy }]
        })
        .compileComponents()
        .then(() => {
            fixture = TestBed.createComponent(HomeComponent);
            component = fixture.componentInstance;
            element = fixture.debugElement;
            homeService = TestBed.get(HomeService);
            fixture.detectChanges();
        });
    }));

    it('should be created', () => {
        expect(component).toBeTruthy();
    });

    it("should display home address", () => { 
        homeService.getAddress.and.returnValue(of('1221 Hub Street'));
        fixture.detectChanges();

        const address = element.queryAll(By.css(".address"));

        expect(address[0].nativeNode.innerText).toEqual('1221 Hub Street');
    });
 });

这是使用jasmine.createSpyObj. 但是,如果您的服务有更多方法更复杂的逻辑,我会建议创建一个 mockService 而不是 createSpyObj。例如: providers: [{ provide: HomeService, useValue: MockHomeService }]

希望这可以帮助!

于 2020-06-12T17:57:14.243 回答
0

基于@Eric Swanson 的回答,我创建了一个更好的可读性和文档化的函数,用于我的测试。我还通过将参数键入为函数来添加一些类型安全性。

我建议将此代码放在通用测试类中的某个位置,以便您可以将其导入每个需要它的测试文件中。

/**
 * Transforms the given method into a jasmine spy so that jasmine functions
 * can be called on this method without Typescript throwing an error
 *
 * @example
 * `asSpy(translator.getDefaultLang).and.returnValue(null);`
 * is equal to
 * `(translator.getDefaultLang as jasmine.Spy).and.returnValue(null);`
 *
 * This function will be mostly used in combination with `jasmine.createSpyObj`, when you want
 * to add custom behavior to a by jasmine created method
 * @example
 * `const translator: TranslateService = jasmine.createSpyObj('TranslateService', ['getDefaultLang'])
 * asSpy(translator.getDefaultLang).and.returnValue(null);`
 *
 * @param {() => any} method - The method that should be types as a jasmine Spy
 * @returns {jasmine.Spy} - The newly typed method
 */
export function asSpy(method: () => any): jasmine.Spy {
  return method as jasmine.Spy;
}

用法如下:

import {asSpy} from "location/to/the/method";

const translator: TranslateService = jasmine.createSpyObj('TranslateService', ['getDefaultLang']);
asSpy(translator.getDefaultLang).and.returnValue(null);
于 2018-07-13T11:32:48.297 回答