Я решил эту проблему, сделав следующее:

м приложении Angular2, которое использует AngularFire2, у меня естьAuthService который пытается аутентифицироваться анонимно с Firebase.

Я пытаюсь написать тест, который ожидаетAngularFireAuth«ssignInAnonymously вернуть отклоненное обещание; заauthState бытьnull и ошибка, которая будет брошена.

Я новичок в Жасмин и тестирование в целом, но я думаю, что мне, возможно, придется использовать асинхронные тесты, но я застреваю.

Вот упрощенныйAuthService:

import { Injectable } from '@angular/core';

import { AngularFireAuth } from 'angularfire2/auth';
import * as firebase from 'firebase/app';
import { Observable } from 'rxjs/Rx';

@Injectable()
export class AuthService {
  private authState: firebase.User;

  constructor(private afAuth: AngularFireAuth) { this.init(); }

  private init (): void {
    this.afAuth.authState.subscribe((authState: firebase.User) => {
      if (authState === null) {
        this.afAuth.auth.signInAnonymously()
          .then((authState) => {
            this.authState = authState;
          })
          .catch((error) => {
            throw new Error(error.message);
          });
      } else {
        this.authState = authState;
      }
    }, (error) => {
      throw new Error(error.message);
    });
  }
}

И вот мои тестовые характеристики:

import { TestBed, inject } from '@angular/core/testing';

import { AngularFireAuth } from 'angularfire2/auth';
import 'rxjs/add/observable/of';
import { Observable } from 'rxjs/Rx';

import { AuthService } from './auth.service';
import { environment } from '../environments/environment';

describe('AuthService', () => {
  const mockAngularFireAuth: any = {
    auth: jasmine.createSpyObj('auth', {
      'signInAnonymously': Promise.resolve('foo'),
      // 'signInWithPopup': Promise.reject(),
      // 'signOut': Promise.reject()
    }),
    authState: Observable.of(null)
  };

  beforeEach(() => {
    TestBed.configureTestingModule({
      providers: [
        { provide: AngularFireAuth, useValue: mockAngularFireAuth },
        { provide: AuthService, useClass: AuthService }
      ]
    });
  });

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

  //
  //
  //
  //
  //

  describe('when we can’t authenticate', () => {
    beforeEach(() => {
      mockAngularFireAuth.auth.signInAnonymously.and.returnValue(Promise.reject('bar'));
    });

    it('should thow', inject([ AuthService ], (service: AuthService) => {
      expect(mockAngularFireAuth.auth.signInAnonymously).toThrow();
    }));
  });

  //
  //
  //
  //
  //

});

Спасибо за помощь!

 jonrsharpe16 сент. 2017 г., 12:52
Да, но эта проверка - единственная вещь, которую, кажется, выполняет ваш тест.
 Jonathon Oates16 сент. 2017 г., 12:51
Привет @jonrsharpe, я издеваюсьAngularFireAuth«ssignInAnonymously в противном случае он будет делать реальные вызовы в Firebase каждый раз, когда запускаются тесты. Я склонен полагать, что я должен издеваться над такого рода сторонними поставщиками?
 jonrsharpe16 сент. 2017 г., 12:45
Почему ты пишешь тест на макет? Что должен делать тестируемый код?
 Jonathon Oates16 сент. 2017 г., 12:55
Это только потому, что я не написал остальную частьAuthService или любые другие тесты для этих будущих методов ;-) Я думаю, что мне удобно писать эти тесты, поэтому, чтобы быть кратким, я включил только те тесты, с которыми мне действительно нужна помощь. ура

Ответы на вопрос(2)

Решение Вопроса

Оказывается, я издевалсяmockAngularFireAuth правильно. Мне нужно было отказатьсяmockAngularFireAuth.auth signInAnonymously()Обещаю с ошибкой и ожидаю, что ее поймают, а-ля:

import { TestBed, async, inject } from '@angular/core/testing';

import { AngularFireAuth } from 'angularfire2/auth';
import 'rxjs/add/observable/of';
import { Observable } from 'rxjs/Rx';

import { AuthService } from './auth.service';
import { MockUser} from './mock-user';
import { environment } from '../environments/environment';

describe('AuthService', () => {
  // An anonymous user
  const authState: MockUser = {
    displayName: null,
    isAnonymous: true,
    uid: '17WvU2Vj58SnTz8v7EqyYYb0WRc2'
  };

  const mockAngularFireAuth: any = {
    auth: jasmine.createSpyObj('auth', {
      'signInAnonymously': Promise.reject({
        code: 'auth/operation-not-allowed'
      }),
      // 'signInWithPopup': Promise.reject(),
      // 'signOut': Promise.reject()
    }),
    authState: Observable.of(authState)
  };

  beforeEach(() => {
    TestBed.configureTestingModule({
      providers: [
        { provide: AngularFireAuth, useValue: mockAngularFireAuth },
        { provide: AuthService, useClass: AuthService }
      ]
    });
  });

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

  describe('can authenticate anonymously', () => {
    describe('AngularFireAuth.auth.signInAnonymously()', () => {
      it('should return a resolved promise', () => {
        mockAngularFireAuth.auth.signInAnonymously()
          .then((data: MockUser) => {
            expect(data).toEqual(authState);
          });
      });
    });
  });

  describe('can’t authenticate anonymously', () => {
    describe('AngularFireAuth.auth.signInAnonymously()', () => {
      it('should return a rejected promise', () => {
        mockAngularFireAuth.auth.signInAnonymously()
          .catch((error: { code: string }) => {
            expect(error.code).toEqual('auth/operation-not-allowed');
          });
      });
    });
  });
  …
});
 André Werlang07 дек. 2017 г., 20:13
Когда вы ожидаете, что обещание будет отклонено, вы пропустите проверку, было ли обещание выполнено. А также наоборот.

Я решил эту проблему, сделав следующее:

    describe('should reject promise', () => {

        let resolved: boolean;
        let rejected: boolean;
        let _e: any;

        beforeEach(function (done) {
            resolved = false;
            rejected = false;
            // ensure conditions here are such that myFn() should return a rejected promise
            service.myFn().then(() => {
                resolved = true;
                done();
            }).catch((e) => {
                rejected = true;
                _e = e;
                done();
            });
        })

        it('should reject', () => {
            expect(resolved).toEqual(false);
            expect(rejected).toEqual(true);
            expect(_e.name).toEqual("MyCustomErrorName");
        });
    });

Ваш ответ на вопрос