Решимость не является функцией

В моем проекте ionic v4/angular я создал сервис для вызова API.
этот сервис выполняет некоторые действия до get или же post вроде добавления заголовка авторизации и тд.

@Injectable({
    providedIn: 'root'
})
export class ApiService {
    headers: HttpHeaders;

    constructor(private http: HttpClient,
                private storage: Storage,
                private loadingController: LoadingController,
                private alertController: AlertController,
                private router: Router) {
        this.storage.get('token').then(token => {
            this.headers = new HttpHeaders().set('Authorization', 'Bearer ' + token);
        });
    }

    async get<T>(url: string): Observable<T> {
        return this.init().then(async () => {
            this.http.get<T>(environment.api_url + url, {headers: this.headers}).subscribe(res => {
                return res;
            }, async err => {
                await this.checkError(err);
            });
        });
    }

    async post<T>(url: string, data: any): Observable<T> {
        return this.init().then(() => {
            this.http.post<T>(environment.api_url + url, data, {headers: this.headers}).subscribe(res => {
                return res;
            }, async err => {
                await this.checkError(err);
            });
        });
    }

    async checkError(err) {
        // if (err.status === 401) { }
        // if (err.status === 500) { }
        await this.alertController.create({
            header: 'Error',
            message: 'Sorry, Something bad happend on our side.',
            buttons: ['Ok']
        }).present();
    }

    async init() {
        const loading = await this.loadingController.create({
            spinner: 'crescent',
            message: 'Please Wait...'
        });

        await loading.present();
        // do some checking like version, connection, ...
        await loading.dismiss();
    }
}

но когда я подписываюсь на post Метод этого сервиса, я получаю эту ошибку: resolve is not a function

login(loginData:any) {
    this.api.post('/auth/login', loginData).subscribe(res => { });
}

3 ответа

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

Мне удалось решить мой вопрос так:

@Injectable({
    providedIn: 'root'
})
export class ApiService {
    headers: HttpHeaders;

    constructor(private http: HttpClient,
                private storage: Storage,
                private loadingController: LoadingController,
                private alertController: AlertController,
                private router: Router) {
    }

    get<T>(url: string) {
        return this.init().then(async () => {
            const loading = await this.loadingController.create({
                spinner: 'crescent',
                message: 'Please Wait'
            });

            await loading.present();
            return this.http.get<T>(environment.api_url + url, {headers: this.headers})
                .pipe(res => {
                        return res;
                    },
                    catchError((err, caught) => {
                        return this.handleError(err, caught);
                    }),
                    finalize(async () => {
                        await loading.dismiss();
                    }));
        });
    }

    post<T>(url: string, data: any) {
        return this.init().then(async () => {
            const loading = await this.loadingController.create({
                spinner: 'crescent',
                message: 'Please Wait'
            });

            await loading.present();
            return this.http.post<T>(environment.api_url + url, data, {headers: this.headers})
                .pipe(res => {
                        return res;
                    },
                    catchError((err, caught) => {
                        return this.handleError(err, caught);
                    }),
                    finalize(async () => {
                        await loading.dismiss();
                    }));
        });
    }

    async checkError(err) {
        // if (err.status === 401) { }
        // if (err.status === 500) { }
        await this.alertController.create({
            header: 'Error',
            message: 'Sorry, Something bad happend on our side.',
            buttons: ['Ok']
        }).present();
    }

    async init() {
        this.storage.get('token').then(token => {
            this.headers = new HttpHeaders().set('Authorization', 'Bearer ' + token);
        }).then(() => {
            const loading = await this.loadingController.create({
                spinner: 'crescent',
                message: 'Please Wait...'
            });

            await loading.present();
            // do some checking like version, connection, ...
            await loading.dismiss();
        });
    }
}

и в моих компонентах:

login(loginData: LoginModel) {
    return this.api.post('/auth/login', loginData)
        .then(data => {
            data.subscribe(res => {
                // json result of api call is avaiable here
            })
        });
}

Посмотрите, пожалуйста, как я справлялся с ними в своих проектах "Ионик-4". Сначала создайте сервис для обработки всех вызовов http следующим образом.

import { Injectable } from '@angular/core';
import { HttpClient, HttpHeaders } from '@angular/common/http';
import { Observable, of } from 'rxjs';
import { catchError, map, tap } from 'rxjs/operators';

@Injectable({
  providedIn: 'root'
})
export class DataService {
  /**
   *Creates an instance of DataService.
   * @param {HttpClient} http
   * @memberof DataService
   */
  constructor(private http: HttpClient) { }

  /**
   * GET data from the server
   * @param {*} url
   * @returns {Observable<any>}
   * @memberof DataService
   */
  getData(url): Observable<any> {
    return this.http.get<any>(url)
      .pipe(
        tap(response => response),
        catchError(this.handleError('getData', []))
      );
  }

  /**
   * POST call setData
   * @param {*} url
   * @param {*} data
   * @returns {Observable<any>}
   * @memberof DataService
   */
  setData(url, data): Observable<any> {
    return this.http.post<any>(url, data, { headers: new HttpHeaders({ 'Content-Type': 'application/json' }) })
      .pipe(
        tap(response => response),
        catchError(this.handleError('setData', []))
      );
  }

  updateData(url, data): Observable<any> {
    return this.http.put<any>(url, data, { headers: new HttpHeaders({ 'Content-Type': 'application/x-www-form-urlencoded' }) })
      .pipe(
        tap(response => response),
        catchError(this.handleError('getData', []))
      );
  }




  /**
   * Handle Http operation that failed.
   * Let the app continue.
   * @param operation - name of the operation that failed
   * @param result - optional value to return as the observable result
   */
  private handleError<T>(operation = 'operation', result?: T) {
    return (error: any): Observable<T> => {

      // TODO: send the error to remote logging infrastructure
      // console.error(error); // log to console instead

      // Let the app keep running by returning an empty result.
      return of(result as T);
    };
  }
}

Затем создайте другой сервис для обработки ваших вызовов, связанных модулей или коллекции модулей. Это зависит от вашего приложения. Вы должны решить это. Я создам сервис под названием AppDataservice для справки

import { Injectable } from '@angular/core';
import { environment } from 'src/environments/environment';
import { map } from 'rxjs/operators';
import { DataService } from './data.service';
/**
 * This contains all the service cales to the server
 */


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

  constructor(private dataService: DataService) { }

  /**
   * Login request with http post method
   */
  login(email, password) {
    const logindata = {
      email: email,
      password: password
    };
    return this.dataService.setData(`${environment.apiUrl}login`, logindata ).pipe(
      map((res) => res)
    );
  }

  /**
   * Fetch list with get request
   */
  getHelp() {
    return this.dataService.getData(`${environment.apiUrl}helpQuestions`).pipe(
      map((res) => res)
    );
  }

}

Наконец, вы можете использовать его в любом компоненте, как показано ниже.

 constructor(private dataService: AppDataService){
        ....
     }

    logIn() {
      this.dataService.login(this.email, this.password).subscribe(res => {
        // handle login
      }, err => {
        // handle error
      });

    }

Попробуйте это, если оно вам подходит. Хорошо работает на моих проектах.

Другие вопросы по тегам