Решимость не является функцией
В моем проекте 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
});
}
Попробуйте это, если оно вам подходит. Хорошо работает на моих проектах.