Reputation: 6119
I have a service SoundPanelService which is used in service isolation scenario ( like https://angular.io/guide/hierarchical-dependency-injection#scenario-service-isolation )
@Injectable()
export class SoundPanelService {
recorded = new Subject<Sound>();
constructor() {
}
}
and I have SoundPanelComponent
Component({
selector: 'app-sound-panel',
templateUrl: './sound-panel.component.html',
styleUrls: ['./sound-panel.component.css'],
providers: [SoundPanelService] // Service isolation
})
export class SoundPanelComponent implements OnInit {
recorded = new Subject<Sound>();
constructor(private soundPanelService: SoundPanelService) {
this.soundPanelService.recorded.subscribe((data) => {
this.recorded.next(data);
});
}
ngOnInit() {
}
}
sound-panel.component.html
<app-sound-player></app-sound-player>
<app-sound-recorder></app-sound-recorder>
SoundPlayer and SoundRecorder communicate with soundpanel through service SoundPanelService.
I want to test SoundPanelComponent
import { async, ComponentFixture, TestBed } from '@angular/core/testing';
import { By } from '@angular/platform-browser';
import { DebugElement } from '@angular/core';
import { SoundPanelComponent } from './sound-panel.component';
import { SoundRecorderComponent } from '../sound-recorder/sound-recorder.component';
import { SoundPlayerComponent } from '../sound-player/sound-player.component';
import { HttpClientTestingModule } from '@angular/common/http/testing';
import { SoundPanelService } from 'src/app/_services/sound-panel.service';
import { Sound } from 'src/app/_models/Sound';
describe('SoundPanelComponent', () => {
let component: SoundPanelComponent;
let fixture: ComponentFixture<SoundPanelComponent>;
beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [
SoundPanelComponent,
SoundPlayerComponent,
SoundRecorderComponent,
SafeHtmlPipe
],
imports: [HttpClientTestingModule],
providers: [
{
provide: SoundPanelService, useClass: SoundPanelService
}
]
})
.compileComponents();
}));
beforeEach(() => {
fixture = TestBed.createComponent(SoundPanelComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should emit sound on new record from sound panel service', async () => {
const s: Sound = {base64: 'data:base64', mimeType: 'audio/wmw'};
spyOn(component.recorded, 'next').and.callThrough();
sps = TestBed.get(SoundPanelService);
sps.recorded.next(s);
fixture.detectChanges();
fixture.whenStable().then(res => {
expect(component.recorded.next).toHaveBeenCalledTimes(1);
});
});
});
but I get error
SoundPanelComponent > should emit sound on new record from sound panel service Expected spy next to have been called once. It was called 0 times.
If I make SoundPanelService providedIn: 'root' I manage to pass tests, but this is not what I want since I want SoundPanelService to be isolated to each SoundPanelComponent and it's children (I intend to have have many SoundPanelComponents on the same page).
How to test this?
Upvotes: 13
Views: 3640
Reputation: 6119
SOLVED
Used this Override component providers
Had to change code to this:
beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [
SoundPanelComponent,
SoundPlayerComponent,
SoundRecorderComponent,
SafeHtmlPipe
],
imports: [HttpClientTestingModule]
})
.overrideComponent(SoundPanelComponent, {
set: {
providers: [
{ provide: SoundPanelService, useClass: SoundPanelService}
]
}
})
.compileComponents();
}));
it('should emit sound on new record from sound panel service', async () => {
const s: Sound = {base64: 'data:base64', mimeType: 'audio/wmw'};
spyOn(component.recorded, 'next').and.callThrough();
sps = fixture.debugElement.injector.get(SoundPanelService) as SoundPanelService;
sps.recorded.next(s);
fixture.detectChanges();
fixture.whenStable().then(res => {
expect(component.recorded.next).toHaveBeenCalledTimes(1);
});
});
Test passed!
Upvotes: 31