Почему асинхронный угловой блок-тест не находит элемент DOM?
У меня есть неудачный асинхронный угловой компонент DOM-теста, но он синхронный эквивалент терпит неудачу, и я не понимаю, почему.
Здесь тест жасмина:
describe('Availability Component', () => {
let fixture: ComponentFixture<AvailabilityComponent>;
const absenceService = jasmine.createSpyObj('AbsenceService', ['findAbsences']);
absenceService.findAbsences.and.returnValue(of([{}]));
beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [AvailabilityComponent],
imports: [CalendarModule.forRoot()],
providers: [{provide: AbsenceService, useValue: absenceService}]
}).compileComponents();
}));
beforeEach(() => {
fixture = TestBed.createComponent(AvailabilityComponent);
});
const printAbsenceReasons = function () {
console.log(fixture.debugElement.queryAll(By.css('.calendarAbsence'))
.map(e => e.nativeElement.textContent)
.join(','));
};
it('should synchronously find absences in calendar view', () => {
fixture.detectChanges();
console.log('synchronous test');
printAbsenceReasons();
});
it('should asynchronously find absences in calendar view', fakeAsync(() => {
fixture.detectChanges();
tick();
fixture.detectChanges();
tick();
console.log('asynchronous test');
printAbsenceReasons();
}));
});
Что создает правильный вывод в синхронном случае, но неверно в асинхронном случае:
LOG: 'processing absences'
HeadlessChrome 0.0.0 (Mac OS X 10.13.4): Executed 0 of 51 SUCCESS (0 secs / 0 secs)
LOG: 'synchronous test'
HeadlessChrome 0.0.0 (Mac OS X 10.13.4): Executed 0 of 51 SUCCESS (0 secs / 0 secs)
LOG: 'A,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,'
HeadlessChrome 0.0.0 (Mac OS X 10.13.4): Executed 0 of 51 SUCCESS (0 secs / 0 secs)
ERROR: 'Spec 'Availability Component should synchronously find absences in calendar view' has no expectations.'
HeadlessChrome 0.0.0 (Mac OS X 10.13.4): Executed 0 of 51 SUCCESS (0 secs / 0 secs)
LOG: 'processing absences'
HeadlessChrome 0.0.0 (Mac OS X 10.13.4): Executed 1 of 51 SUCCESS (0 secs / 0 secs)
LOG: 'asynchronous test'
HeadlessChrome 0.0.0 (Mac OS X 10.13.4): Executed 1 of 51 SUCCESS (0 secs / 0 secs)
LOG: ''
HeadlessChrome 0.0.0 (Mac OS X 10.13.4): Executed 1 of 51 SUCCESS (0 secs / 0 secs)
ERROR: 'Spec 'Availability Component should asynchronously find absences in calendar view' has no expectations.'
Я не уверен, что это имеет какое-то отношение к компоненту углового календаря, который я использую, или если это более простая проблема с моим тестовым кодом.
Для справки здесь приведен код компонента, шаблона и сервиса:
@Component({
selector: 'app-availability',
templateUrl: './availability.component.html',
styleUrls: ['availability.component.scss']
})
export class AvailabilityComponent implements OnInit {
viewDate: Date = new Date();
absenceEvents: CalendarEvent[] = [];
constructor(private absenceService: AbsenceService) {
}
ngOnInit() {
this.getAbsences();
}
getAbsences() {
this.absenceService.findAbsences()
.subscribe(ignored => {
console.log('processing absences');
this.absenceEvents = [{
start: new Date(2018, 3, 29), title: 'A'
}];
});
}
getAbsence(events: CalendarEvent[]) {
return events[0] ? events[0].title : '';
}
}
код шаблона:
<div>
<div>
<mwl-calendar-month-view
[viewDate]="viewDate"
[events]="absenceEvents"
[cellTemplate]="availabilityCellTemplate">
</mwl-calendar-month-view>
</div>
<ng-template #availabilityCellTemplate let-day="day">
<div class="calendarAbsence">{{ getAbsence(day.events) }}</div>
</ng-template>
</div>
служебный код:
@Injectable()
export class AbsenceService {
private url = environment.APP_SHIFT_SERVICE_BASE_URL + '/api/absences';
constructor(private http: HttpClient) {
}
findAbsences(): Observable<Absence[]> {
console.error('Actual findAbsences() called');
return this.http.get<Absence[]>(this.url);
}
}
Ответы
Ответ 1
К сожалению, это, по-видимому, имеет какое-то отношение к зоне fakeAsync, в частности, а не к асинхронным тестам в частности.
Мне удалось получить рабочий асинхронный тест для работы с async
вместо fakeAsync
:
it('should asynchronously find absences in calendar view', async(() => {
fixture.detectChanges();
fixture.whenStable().then(() => {
console.log('asynchronous test');
printAbsenceReasons(fixture);
expect(getAbsenceElements(fixture).length).toEqual(35);
});
});
}));
Я отлаживал как неудачные fakeAsync
и успешные синхронные тесты. Они оба называют метод под названием getMonthView
в библиотеке "calendar-utils" того же автора, что и "угловой календарь": https://github.com/mattlewis92/calendar-utils/blob/master/src/calendar-utils. Т.С.
В этом методе как параметры Date, так и другие сами вычисления Date, похоже, очень ошибочны.
Я могу только предположить, что это связано с этой известной проблемой в zone.js. Я нахожусь на Angular 5.2, но я предполагаю, что он все еще связан. В любом случае я уверен, что мой тестовый код в основном правильный, но проблема кроется в другом месте.
Ответ 2
Я почти уверен, что причина, по которой это не работает, - это то, как вы определили своего шпиона. Он определяется вне блока beforeEach
.
Теперь, когда у вас есть это, ваш шпион создается один раз, когда прибор начинает работу. И жасмин работает, удаляя всех старых шпионов в конце каждого теста. Итак, во втором тесте ваш шпион больше не существует.
Я бы поспорил, что если вы включили свой тестовый заказ, вы увидите, что тест асинхронный работает, но синхронизация отсутствует.
Чтобы исправить это, просто переместите это в блок beforeEach:
const absenceService = jasmine.createSpyObj('AbsenceService', ['findAbsences']);
absenceService.findAbsences.and.returnValue(of([{}]));