rxjs - 如何为 combineLatest rxjs Angular 编写 Jasmine 单元测试用例

标签 rxjs angular2-services angular2-observables combinelatest

我的组件.ts

import { Component, OnInit } from '@angular/core';
import {FormGroup,FormControl} from '@angular/forms'
import { DataServiceService } from './data-service.service';
import {combineLatest,Observable,pipe} from 'rxjs';
import {map,tap} from 'rxjs/operators';
import {Model} from './mode';
@Component({
  selector: 'my-app',
  templateUrl: './app.component.html',
  styleUrls: [ './app.component.css' ]
})
export class AppComponent  {

  constructor(private dataService: DataServiceService){}
  name = 'Angular';
  myForm: FormGroup;
  observableResult$: Observable<any>;

  ngOnInit(){
    this.myForm = new FormGroup({
      localId: new FormControl()
    })

  this.observableResult$ = combineLatest(
    this.myForm.get('localId').valueChanges,
    this.dataService.getDataFromURL(),
    (localIdSelected, dataFromAPI) => ({localIdSelected,dataFromAPI})).
    pipe(map(each => this.filterData(each.dataFromAPI,each.localIdSelected)));

    this.observableResult$.subscribe(value => {
      debugger
    })

  }
  filterData(dataFromAPI,localIDSelected){
    debugger
     return dataFromAPI.filter(item => item.userId > Number(localIDSelected));
    }

}

data.service.service.ts

import { Injectable } from '@angular/core';
import {HttpClient} from '@angular/common/http'
import {Model} from './mode';
import {Observable} from 'rxjs';
@Injectable()
export class DataServiceService {

  constructor(private http:HttpClient) { }

  getDataFromURL():Observable<Model>{
    return this.http.get<Model>('https://jsonplaceholder.typicode.com/todos');
  }

}

app.component.html

<form [formGroup]="myForm" >

<select formControlName="localId">
  <option>1</option>
  <option>2</option>
</select>

</form>

app.spec.ts

const spyFilter = spyOn(component as any, filterData).and.callThrough();

const constAPIData$ = staticDataServiceMock.getAPIData();
                    spyOn(staticDataServiceMock, 'getAPIData').and.returnValue(
                        observableOf(countryStaticData$)
                    );
component.myForm.get('localId').setValue(1);

component.observableResult$.subscribe(value => {
expect(value[0].id==21).toBeTrue();
});

staticDatamock.ts

export class StaticDataMock{

static getAPIData(): Observable<StaticDataElements[]> {
    return [
  {
    "userId": 1,
    "id": 1,
    "title": "delectus aut autem",
    "completed": false
  },
  {
    "userId": 1,
    "id": 2,
    "title": "quis ut nam facilis et officia qui",
    "completed": false
  },
  {
    "userId": 1,
    "id": 3,
    "title": "fugiat veniam minus",
    "completed": false
  },
  {
    "userId": 1,
    "id": 4,
    "title": "et porro tempora",
    "completed": true
  }];
  }
}

我已添加我的测试用例以覆盖 app.spec.ts 中的 combineLatest 运算符和 filterData,但所需代码失败。我期望调用 filterData 失败了。 combineLatest 将在 valueChange 上触发事件并从 API 获取数据。我可以在规范文件中创建 mock 和 setValue,但它仍然无法正常工作。

最佳答案

好的,为了尝试帮助您继续进行此操作,我继续使用您目前提供的数据设置了 Stackblitz。这是 link .

我做了一些事情来让测试工作(某种程度上)。

  • 我将 StaticDataMock 类中 getAPIData() 的方法类型更改为 public,以便您可以从类外部调用它。
  • 我让该方法返回一个 of(),将返回值转换为您的数据的 Observable。
  • 我猜到了您对 DataServiceService 模拟的实现,Stackblitz 中有详细信息。
  • 我创建了一个 Jasmine spyObject 来拦截服务中对 getDataFromURL() 的调用,并返回您使用 StaticDataMock 创建的可观察对象。
  • 我重新安排了规范中的调用顺序,并执行了 console.log() 以表明 component.observableResult$ 永远不会发出。<

更新

根据下面的评论,上面的 Stackblitz 链接已经更新,现在可以使用了。来自 Stackblitz 的工作规范:

it('should change return of service.function1() only', () => {
    fixture.detectChanges(); 
    component.observableResult$.subscribe(value => {
        console.log('observable Emitted, value is ', value);
        expect(value[0].id==1).toBe(true); 
    });
    component.myForm.get('localId').setValue(1);
});

关键是先设置订阅,然后在表单中发出一个新值,这将更新 combineLatest() 并执行 subscribe 中的代码()

我很高兴这成功了!

关于rxjs - 如何为 combineLatest rxjs Angular 编写 Jasmine 单元测试用例,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53538362/

相关文章:

Angular RC6 - 使用自己的数据创建可注入(inject)服务

javascript - Rxjs:将数组字段中的每个元素从对象的可观察对象映射到另一个可观察对象

javascript - RxJS: BehaviorSubject 退订

javascript - 为什么 Observable 操作对每个订阅者调用一次(重复)?

angular - 如何在 angular4 中读取 APP_INITIALIZER 服务中的查询参数?

使用 Indexeddb 作为离线数据存储的 Angular5 HTTP 服务 : how to handle Observable<response> vs Object

可观察的 zip(数组)的 RXJS zip 未触发

angular2 map 数据作为特定对象类型

json - 自动将 JSON 对象映射到 TypeScript 对象(Angular 2)

angular - 从组件订阅服务中的 Observable