Angular 8 проверка компонента с кармой не удалась - PullRequest
1 голос
/ 27 мая 2020

Я пытаюсь начать тестирование своего компонента. Первое, что я хотел проверить, это то, вызывает ли ngOnInit правильные службы.

соглашение.component.ts :

  constructor(private agreementService: AgreementService,
              private operatorService: OperatorService,
              private accountService: AccountService,
              private route: ActivatedRoute,
              private router: Router,
              private sessionService: SessionService,
              private settingsService: SettingsService) {
    this.agreementId = Number(this.route.snapshot.paramMap.get('agreementId'));
  }

  async ngOnInit() {
    this.session = await this.sessionService.getSession();
    this.settings = await this.settingsService.getSettings();

    this.operatorService.getOperators(this.session.bic).subscribe(data => {
      this.operators = data;
    });
  ...
  }

соглашение .component.spe c .ts

import {AgreementComponent} from './agreement.component';
import {async, TestBed} from '@angular/core/testing';
import {ActivatedRoute, convertToParamMap, Router} from '@angular/router';
import {RouterTestingModule} from '@angular/router/testing';
import {AgreementService} from '../../../services/agreement.service';
import {AccountService} from '../../../services/account.service';
import {SessionService} from '../../../services/session.service';
import {SettingsService} from '../../../services/settings.service';

describe('agreementComponent', () => {
  let mockAgreementService: AgreementService;
  let mockOperatorService;
  let mockAccountService: AccountService;
  let mockRoute: ActivatedRoute;
  let mockRouter: Router;
  let mockSessionService: SessionService;
  let mockSettingsService: SettingsService;
  let component: AgreementComponent;

  beforeEach(async(() => {
    mockAgreementService = jasmine.createSpyObj(['getAgreement']);
    mockOperatorService = jasmine.createSpyObj(['getOperators']);
    mockAccountService = jasmine.createSpyObj(['getFeeAccounts']);
    mockRoute = jasmine.createSpyObj(['route']);
    mockRouter = jasmine.createSpyObj(['router']);
    mockSessionService = jasmine.createSpyObj(['getSession']);
    mockSettingsService = jasmine.createSpyObj(['getSettings']);

    TestBed.configureTestingModule({
      declarations: [AgreementComponent],
      imports: [
        RouterTestingModule
      ],
      providers: [
        {
          provide: ActivatedRoute, useValue:
            {
              snapshot: {
                paramMap: convertToParamMap({agreementId: '0'})
              }
            }
        },
      ]
    });

    component = new AgreementComponent(mockAgreementService, mockOperatorService, mockAccountService,
      mockRoute, mockRouter, mockSessionService, mockSettingsService);
  }));


  it('should call operators service', () => {
    component.ngOnInit();

    expect(mockOperatorService).toHaveBeenCalled();

  });
});

В настоящее время я получаю:

Ошибка: невозможно прочитать свойство paramMap из undefined

TypeError: Невозможно прочитать свойство 'ngOnInit' из undefined

Я действительно уверен, что в этом коде много чего не хватает для нормальной работы, я просто не могу понять, что именно отсутствует и что должно быть сделано по-другому, потому что поиск моих ошибок запутал меня множеством различных решений. Я новичок в тестировании angular, поэтому хотел бы получить несколько советов, как правильно писать тесты.

1 Ответ

1 голос
/ 27 мая 2020

Воспользуйтесь другим подходом, создав заглушки, как описано в одной из моих статей .

  1. Создавайте заглушки многократного использования как:
export class MockOperatorService{
  getOperators(){
     return of({data: "someVal"})
  }
}

и так далее для других услуг.

Используйте RouterTestingModule по мере необходимости в imports

Mock ActivatedRoute и других услугах, как показано ниже:


  beforeEach(async(() => {
    TestBed.configureTestingModule({
      declarations: [AgreementComponent],
      imports: [
        RouterTestingModule
      ],
      providers: [
        {
          provide: ActivatedRoute, useValue:
            {
              snapshot: {
                paramMap: convertToParamMap({agreementId: '0'})
              }
            }
        },
        {provide: OperatorService , useClass: MockOperatorService},
        {....similarly for AgreementService etc etc}
      ]
    });
  }));

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

Я понял, что не хватает надлежащего набора статей для изучения angular тестирования, поэтому я написал сборник статей, который вы можете найти внизу этой страницы . Надеюсь, это поможет

Обновление:

Чтобы шпионить, как указано в комментарии, вы можете сделать:


  it('should call getOperators service in ngOnInit', () => {
    spyOn(component.operatorService,"getOperators").and.callThrough();
    component.ngOnInit();
    expect(component.operatorService.getOperators).toHaveBeenCalled();
    // you can also be more specific by using ".toHaveBeenCalledWith()"
  });

...