使用Angular 2注入依赖项的单元测试服务

时间:2017-06-16 10:06:26

标签: angular unit-testing jasmine karma-runner

我为具有依赖关系的服务创建一个简单的单元测试。这是一项简单的服务,我开始帮助我了解最佳流程和结构。

我创建的单元测试有效,但TypeScript会抛出错误。

Argument of type '{ order_id: string; user: string; }' is not assignable to parameter of type 'Expected<Observable<any>>'.
Object literal may only specify known properties, and 'order_id' does not exist in type 'Expected<Observable<any>>'.

该服务在注入的服务上执行方法sendBody,该服务发出HTTP post请求。我理解应该在这里返回一个Observable,但我认为模拟服务应该已经解决了这个TypeScript类型错误。

在仅测试服务时是否有更好的方法来注入和模拟依赖项?

服务

export class ReturnOrderService {

    constructor(private _ReturnOrderResource: ReturnOrderResource) { }

    updateOrderStatus(orderId: string, user: string) {
        let body = {};

        if (orderId !== undefined && user !== undefined) {
            body["order_id"] = orderId;
            body["user"] = user;
        }

        return this._ReturnOrderResource.sendBody(body);
    }

}

规格

const MockReturnOrderResource = {
sendBody: function(body) {
    return body;
  }
}

describe('ReturnsComponent', () => {

  beforeEach(() => {
    TestBed.configureTestingModule({ providers: [
       ReturnOrderService,
       { provide: ReturnOrderResource, useValue: MockReturnOrderResource }
      ] });
  });

  it('#updateOrderStatus should create an object literal from arguments',
    async(inject([ReturnOrderService], (service: ReturnOrderService) => {
      expect(service.updateOrderStatus("10006886","Mr Bungle")).toEqual({ order_id: '10006886', user: 'Mr Bungle' });
  })));

});

2 个答案:

答案 0 :(得分:0)

如果必须返回一个Observable,则返回一个Observable!

试试这个:

const MockReturnOrderResource = {
    sendBody: (any: any) => Observable.of(any);
}

答案 1 :(得分:0)

我使用不同的方法为此解决了这个问题。虽然它可能无法解决我遇到的类型错误,但它允许我监视注入的服务,这正是我所追求的。

注入服务后,我使用TestBed.get()获取服务并将其分配给变量。

然后我可以窥探注入的服务并测试该方法被调用的参数。

以下更新的规范。

const MockReturnOrderResource = {
  sendBody: function(body) {
    return body;
  }
}

describe('ReturnsComponent', () => {
  let _ReturnOrderResource;

  beforeEach(() => {
    TestBed.configureTestingModule({ providers: [
       ReturnOrderService,
       { provide: ReturnOrderResource, useValue: MockReturnOrderResource }
      ] });
    _ReturnOrderResource = TestBed.get(ReturnOrderResource);
  });

  it('#updateOrderStatus should pass an object literal as an argument',
    async(inject([ReturnOrderService], (service: ReturnOrderService) => {
    spyOn(service, "updateOrderStatus").and.callThrough();
    spyOn(_ReturnOrderResource, "sendBody").and.callThrough();
    service.updateOrderStatus("10006886","Mr Bungle");
    expect(_ReturnOrderResource.sendBody).toHaveBeenCalledWith({ order_id: '10006886', user: 'Mr Bungle' })
  })));

});

有关此解决方案的进一步阅读:https://angular.io/guide/testing#testbedget