如何在ngOnInit中使用条件测试功能

时间:2019-08-29 15:49:02

标签: javascript angular typescript karma-jasmine angular8

我想使用来自ngOnInit中的服务的条件来测试功能。我尝试了很多方法但没有成功。我有各种各样的错误。

我的组件

export class MainSectionComponent implements OnInit {    

  propertiesFrDb: PropertyPost[];

  constructor(
    private getPropertiesFrDbService: GetPropertiesFrDbService,
    private propertyWarehouseService: PropertyWarehouseService,
    private router: Router,
    config: NgbCarouselConfig,
    private userService: UserService,
    private sharedFunctionService: SharedFunctionService,
    private returnResponseAfterUserLoginService: ReturnResponseAfterUserLoginService,
    private localStorageService: LocalStorageServiceService,
    private dialogLoginService: DialogLoginService,
    @Inject(PLATFORM_ID) private platformId: Object
  ) { 
    // this.isBrowser = isPlatformBrowser(platformIds);

  }

  ngOnInit() {
    this.getPropertiesFrDb();

  }

  getPropertiesFrDb() {
    if (this.propertyWarehouseService.currentValuesProperties) {
      this.propertyWarehouseService.getPropertyHome$.subscribe(
        prop => {
          console.log(prop);

         return this.propertiesFrDb = prop
        }
      )
    } else {

      this.getPropertiesFrDbService.getHomeProperties()
        .subscribe(property => {
          // console.log(property['results']);
          this.propertyWarehouseService.setPropertiesHome(property['results'])
         return   this.propertiesFrDb = property['results']

        },

        )
    }
  }

我要在this.getPropertiesFrDb()中测试ngOnInit 我想用this.propertyWarehouseService.currentValuesProperties !== ''测试案例,并检查是否调用了this.getPropertiesFrDbService.getHomeProperties()并检查了propertiesFrDb的值

和我的 spec.ts 文件

import { async, ComponentFixture, TestBed, fakeAsync, tick } from '@angular/core/testing';

import { MainSectionComponent } from './home-properties.component';
import { CUSTOM_ELEMENTS_SCHEMA } from '@angular/core';
import { MaterialModule } from 'src/app/material/material.module';
import { HttpClientTestingModule } from '@angular/common/http/testing';
import { RouterTestingModule } from '@angular/router/testing';
import { GetPropertiesFrDbService } from 'src/app/services/getPropertiesFromDb/get-properties-fr-db.service';
import { MOCKPROPERTIES, MockPropertyWarehouseService } from 'src/app/mocks/property-post';
import { NgxPaginationModule, PaginatePipe } from 'ngx-pagination';
import { PropertyWarehouseService } from 'src/app/services/propertyWarehouse/property-warehouse.service';
import { BsDropdownModule } from 'ngx-bootstrap';
import { NgbModule } from '@ng-bootstrap/ng-bootstrap';
import { StorageServiceModule } from 'angular-webstorage-service';
import { of } from 'rxjs/internal/observable/of';

fdescribe('MainSectionComponent', () => {
  let component: MainSectionComponent;
  let fixture: ComponentFixture<MainSectionComponent>;
  const PROPERTYMODEL = MOCKPROPERTIES;
  const spyPropertyWarehouseService = jasmine.createSpyObj('spypropertyWarehouseService', ['currentValuesProperties', 'getPropertyHome$']);


  beforeEach(async(() => {
    TestBed.configureTestingModule({
      imports: [
        MaterialModule,
        HttpClientTestingModule,
        RouterTestingModule.withRoutes([]),
        NgxPaginationModule,
        BsDropdownModule.forRoot(),
        NgbModule,
        StorageServiceModule,


      ],
      declarations: [
        MainSectionComponent,

      ],
      providers: [
        {
          provide: PropertyWarehouseService,
          useValue:  spyPropertyWarehouseService
      }

      ],
      schemas: [CUSTOM_ELEMENTS_SCHEMA],

    })
      .compileComponents();

  }));

  beforeEach(() => {
    fixture = TestBed.createComponent(MainSectionComponent);
    component = fixture.componentInstance;
    fixture.detectChanges();

  });


  it('should create', (() => {
    // console.log('properties', component);
    expect(component).toBeTruthy();
  }));

  it('Should get  propertiesFrDb from GetPropertiesFrDbService', async(() => {
    spyPropertyWarehouseService.currentValuesProperties.and.returnValue(PROPERTYMODEL);
    spyPropertyWarehouseService.getPropertyHome$.and.returnValue(of(PROPERTYMODEL));
    expect(component.propertiesFrDb).toBe(PROPERTYMODEL);

    console.log('spy',spyPropertyWarehouseService);

  }));

});

1 个答案:

答案 0 :(得分:1)

尝试如下创建存根:

export class PropertyWarehouseServiceStub{
   currentValuesProperties = '';
   getPropertyHome$ = new BaheviorSubject<any>('someObj');
   setPropertiesHome(){  }

}

export class GetPropertiesFrDbServiceStub{
   getHomeProperties(){
     return of({results: 'val'})
   }
}

component文件中的

将该服务在构造函数中公开,以便我们可以覆盖其某些行为:

constructor(...,
   public propertyWarehouseService: PropertyWarehouseService, 
   public getPropertiesFrDbService: GetPropertiesFrDbService,
   ....)

,并在spec文件中为:

      providers: [
        {
          provide: PropertyWarehouseService,
          useClass:  PropertyWarehouseServiceStub
        },{
          provide: GetPropertiesFrDbService,
          useClass:  GetPropertiesFrDbServiceStub
        }
      ],

......
....
..
it('should call getPropertiesFrDb() in ngOnInit',()=>{
     spyOn(component,'getPropertiesFrDb').and.callThrough();
     component.ngOnInit();
     expect(component.getPropertiesFrDb).toHaveBeenCalled();
})

it('inside getPropertiesFrDb() should call getPropertiesFrDbService.getHomeProperties() when "propertyWarehouseService.currentValuesProperties" is empty,()=>{
  spyOn(component.getPropertiesFrDbService,'getHomeProperties').and.callThrough();
  spyOn(component.propertyWarehouseService,'setPropertiesHome').and.callThrough();
  component.getPropertiesFrDb();
  expect(component.getPropertiesFrDbService.getHomeProperties).toHaveBeenCalled();
  expect(component.propertyWarehouseService.setPropertiesHome).toHaveBeenCalledWith('val');
  expect(component.propertiesFrDb).toBe('someObj'); 
})

it('inside getPropertiesFrDb() should not call getPropertiesFrDbService.getHomeProperties() when "propertyWarehouseService.currentValuesProperties" is NOT empty,()=>{
  component.propertyWarehouseService.currentValuesProperties = 'Not empty';
  spyOn(component.getPropertiesFrDbService,'getHomeProperties').and.callThrough();
  spyOn(component.propertyWarehouseService,'setPropertiesHome').and.callThrough();
  component.getPropertiesFrDb();
  expect(component.getPropertiesFrDbService.getHomeProperties).not.toHaveBeenCalled();
  expect(component.propertyWarehouseService.setPropertiesHome).not.toHaveBeenCalledWith('val');
  expect(component.propertiesFrDb).toBe('val');
})

您可以参考this intro article written by me on Karma-jasmine,其中包含有关多个测试用例的更多文章参考。

This one is very much similar to what you are looking for.我打算写更多的文章,以备您关注。


此外,我不知道您为什么在return中使用getPropertiesFrDb()

return this.propertiesFrDb = prop

这是没有用的,因为尚未将此函数的值分配给ngOnInit中的任何变量。