Mockowanie właściwości Subject mockowanej usługi, która ma być subskrybowana w teście jednostkowym Angular
W moich testach jednostkowych Angular moja fałszywa usługa ma dwie właściwości:
public messageChange: Subject<ChatMessage> = new Subject<ChatMessage>();
public gameChange: Subject<GameState> = new Subject<GameState>();
A w moim SUT używam ich w moim konstruktorze:
this._subChat = this.hub.messageChange.subscribe((message: ChatMessage) => {
this.message = message;
this.messageChange.next(this.message);
});
this._subGame = this.hub.gameChange.subscribe((game: GameState) => {
this.game.setGame(game);
});
W tej chwili próbuję użyć tych 2 podejść do mockowania właściwości z usługi mocked:
- Temat próbnej powtórki testu jednostkowego kątowego
- Jak szpiegować wartość właściwości (zamiast metody) za pomocą Jasmine
Riht teraz moje testy wyglądają tak:
describe('SignalRService', () => {
let signalrService: SignalRService;
const hubServiceMock = jasmine.createSpyObj('HubConnectionService', [
'isConnectionStarted',
]);
const messageChange = new Subject();
const gameChange = new Subject();
beforeEach(() => {
signalrService = new SignalRService(
hubServiceMock
);
let msg = {} as ChatMessage;
let gam = {} as GameState;
messageChange.next(msg);
gameChange.next(gam);
});
it('Service_ShouldBeCreated', () => {
spyOnProperty(hubServiceMock, 'messageChange', 'get').and.returnValue(
messageChange
);
spyOnProperty(hubServiceMock, 'gameChange', 'get').and.returnValue(
gameChange
);
expect(signalrService).toBeTruthy();
});
}
Więc w teście tworzę:
próba usługi
hubServiceMock
,fałszywy
messageChange = new Subject();
,fałszywy
gameChange = new Subject();
,biegam dla obu
.next
,i konfiguruję szpiega dla właściwości:
spyOnProperty(hubServiceMock, 'messageChange', 'get').and.returnValue( messageChange );
spyOnProperty(hubServiceMock, 'gameChange', 'get').and.returnValue( gameChange );
Dlaczego mi to nie działa? Otrzymuję błąd:
Nie można odczytać właściwości „subskrybuj” wartości undefined
Odpowiedzi
To nie działa, ponieważ hubServiceMock
nie zawiera fałszywych tematów messageChange
i gameChange
musisz je ustawić przed wywołaniem new SignalRService(hubServiceMock)
.
const hubServiceMock = jasmine.createSpyObj('HubConnectionService', [
'isConnectionStarted',
]);
const messageChange = new Subject();
const gameChange = new Subject();
// add this
hubServiceMock.messageChange = messageChange;
hubServiceMock.gameChange = gameChange;
to powinno działać, być może potrzebne są drobne poprawki.
W takich przypadkach sugerowałbym użycie kpiącej libelli, aby uniknąć bólu.
Na przykład w przypadku ng-mocks test może wyglądać następująco:
describe('SignalRService', () => {
beforeEach(() => MockBuilder(SignalRService, ITS_MODULE));
const hubServiceMock = {
messageChange: new Subject(),
gameChange: new Subject(),
};
beforeEach(() => MockInstance(HubConnectionService, hubServiceMock));
it('Service_ShouldBeCreated', () => {
const signalrService = MockRender(SignalRService).point.componentInstance;
expect(signalrService).toBeTruthy();
hubServiceMock.messageChange.next({});
hubServiceMock.gameChange.next({});
// next assertions.
});
}
Skończyło się na podejściu, w którym wstrzykuję fałszywą usługę w następujący sposób:
hubServiceMock = TestBed.inject(HubConnectionService);
Potem kpię z moich w Subject
ten sposób:
it('Service_ShouldBeCreated', () => {
spyOn(hubServiceMock.messageChange, 'next');
spyOn(hubServiceMock.gameChange, 'next');
expect(signalrService).toBeTruthy();
});
A w innych testach mogę używać metod mockowanej obsługi, takich jak:
let spyIsConnectionStarted = spyOn(hubServiceMock, 'isConnectionStarted');
spyIsConnectionStarted.and.returnValue(true);