Модульное тестирование Угловой сервис, который зависит от другого сервиса, который использует InjectionToken - PullRequest
0 голосов
/ 01 июня 2018

Я работаю над модульным тестированием службы Angular, которая выглядит следующим образом:

/* data.service.ts */
import { Injectable } from '@angular/core';

import { FooService } from './foo.service';
import { Data } from './data.model';

@Injectable({
  providedIn: 'root'
})
export class DataService {
  constructor(private fooService: FooService) {}

  addData(data: Data) {
    return this.fooService.getActiveFoo().addData(data);
  }
}

FooService - это служба в том же модуле.Он вводит некоторые значения, используя InjectionTokens.

/* foo.service.ts */
import { Inject, Injectable } from '@angular/core';

import { KEY1, KEY2 } from './tokens';
import { Foo } from './foo.ts';

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

  activeFoo: Foo; 

  constructor(@Inject(KEY1) private key1: string, @Inject(KEY2) private key2: string) {}

  ...

  getActiveFoo() {
    return this.activeFoo; 
 }
}


/* tokens.ts */
import { InjectionToken } from '@angular/core';

export const KEY1 = new InjectionToken('KEY_1');
export const KEY2 = new InjectionToken('KEY_2');

Все три файла являются частью одного модуля, который настроен следующим образом:

/* bar.module.ts */
import { ModuleWithProviders, NgModule } from '@angular/core';
import { CommonModule } from '@angular/common';

import { KEY1, KEY2 } from './tokens';
import { DataService } from './data.service';
import { FooService } from './foo.service';

@NgModule({
  imports: [CommonModule],
})
export class BarModule {
  static forRoot(key1Val: string, key2Val: string): ModuleWithProviders {
    return {
      ngModule: BarModule,
      providers: [
        DataService,
        FooService,
        { provide: KEY1, useValue: key1Val},
        { provide: KEY2, useValue: key2Val}
      ]
    };
  }
}

Это прекрасно работает в самом приложении.Сейчас я работаю над написанием модульного теста для data.service.ts и сталкиваюсь с проблемами с предоставленными InjectionTokens.

/* data.service.spec.ts */
import { TestBed, inject } from '@angular/core/testing';

import { DataService } from './data.service';
import { FooService } from './foo.service';
import { KEY1, KEY2 } from './tokens';

describe('DataService', () => {
  const mockFooService = jasmine.createSpyObj(['getActiveFoo']);

  beforeEach(() => {
    TestBed.configureTestingModule({
      providers: [
        DataService,
        { provide: FooService, useValue: mockFooService },
        { provide: KEY1, useValue: 'abc' },
        { provide: KEY2, useValue: '123' }
      ]
    });
  });

  it('should be created', inject([DataService], (service: DataService) => {
    expect(service).toBeTruthy();
  }));
});

Когда я запускаю свои тесты, я получаю сообщение об ошибке, говорящее

Error: StaticInjectorError(DynamicTestModule)[FooService -> InjectionToken KEY_1]: 
  StaticInjectorError(Platform: core)[FooService -> InjectionToken KEY_1]: 
    NullInjectorError: No provider for InjectionToken KEY_1!

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

1 Ответ

0 голосов
/ 01 июня 2018

Я решил эту проблему, добавив фабричные функции в InjectionTokens, которые возвращают значение токена по умолчанию.Когда это сделано, по умолчанию вводятся токены в корневой инжектор.

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

Мои токены теперь выглядят как

/* tokens.ts */
import { InjectionToken } from '@angular/core';

export const KEY1 = new InjectionToken<string>('KEY_1', {factory: () => '' });
export const KEY2 = new InjectionToken<string>('KEY_2', {factory: () => '' });

В качестве дополнительного преимущества токены теперь могут быть потрясены деревом.См. здесь для получения дополнительной информации.

...