用茉莉花大理石测试对象随时间的行为

时间:2019-04-09 15:13:08

标签: unit-testing jasmine rxjs jasmine-marbles

因此,我正在尝试测试某个主题的行为,并且该行为不起作用,似乎有些事情我没有正确理解。请考虑以下测试:

it('marble testing subject test', () => {
    const obs$: Subject<number> = new Subject<number>();

    obs$.next(42);
    obs$.next(24);

    expect(obs$.asObservable()).toBeObservable(hot('xy', { x: 42, y: 24 }));
  });

此操作失败,并显示以下错误消息:

Expected $.length = 0 to equal 2.
Expected $[0] = undefined to equal Object({ frame: 0, notification: Notification({ kind: 'N', value: 42, error: undefined, hasValue: true }) }).
Expected $[1] = undefined to equal Object({ frame: 10, notification: Notification({ kind: 'N', value: 24, error: undefined, hasValue: true }) }).

我想我有点理解为什么:Subject(每个文档)仅在订阅开始后才发出值。 toBeObservable()函数(我假设)订阅了Subject,因此我的2 next个调用在此之前发生,因此它们不会发出。

所以我的问题是,我将如何测试类似的东西?即随时间测试对象的一系列排放?这可能与大理石测试有关吗?我可以通过将其更改为ReplaySubject来使其工作,但是如果我这样做,则大理石图必须为(xy)而不是xy

谢谢。

1 个答案:

答案 0 :(得分:1)

我可以在Angular应用程序中使用它

我的服务,关键是我将自己的事情定义为吸气剂,如果我只是将asObservable定义为属性,那么我就有机会实际监视things$,而间谍却没有工作:

@Injectable({
  providedIn: 'root'
})
export class ApiService {

  private things = new BehaviorSubject<Array<string>>([]);

  public get things$(): Observable<boolean> {
    return this.things.asObservable().pipe(map((things) => things.length > 0))
  }
}

然后在测试中,我想关键部分是我在asObservable things上监视BehaviourSubject方法。像这样:

describe('#things$', () => {

    it('should ', () => {
      const things.   = 'a-b-a';
      const expected =  't-f-t';
      // spy on
      spyOn(service['things'], 'asObservable').and.returnValue(hot(things, {
        a: ['a', 'b'],
        b: []
      }));

      expect(service.things$).toBeObservable(cold(expected, {
        t: true,
        f: false
      }));
    });
  });