- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
class ExpiredTokenRetryPolicy extends RetryPolicy {
@override
bool shouldAttemptRetryOnResponse(Response response) {
if (response.statusCode == 401) {
// Perform your token refresh here.
// refreshToken();
return true;
}
return false;
}
}
如何调用这个方法?
如何集成 Flutter 的 Http 拦截器来重试请求?基本上我想在那里实现刷新 token 。
最佳答案
您可以复制粘贴运行下面的完整代码
我使用https://s0pub0dev.icopy.site/packages/http_interceptor的官方示例
为了模拟这种情况,我不设置 API key ,这将产生 401
我将 maxRetryAttempts
设置为 10
,因此您可以看到输出产生错误消息 10
次
您可以像这样注入(inject)retryPolicy
HttpClientWithInterceptor.build(interceptors: [ WeatherApiInterceptor(),
], retryPolicy: ExpiredTokenRetryPolicy()),
代码片段
class ExpiredTokenRetryPolicy extends RetryPolicy {
@override
int maxRetryAttempts = 10;
@override
bool shouldAttemptRetryOnResponse(http.Response response) {
print(response.statusCode);
if (response.statusCode == 401) {
print("Perform your token refresh here in 401");
return true;
}
return false;
}
}
class _HomeScreenState extends State<HomeScreen> {
WeatherRepository repository = WeatherRepository(
HttpClientWithInterceptor.build(interceptors: [
WeatherApiInterceptor(),
], retryPolicy: ExpiredTokenRetryPolicy()),
);
输出
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
I/flutter (30233): 401
I/flutter (30233): Perform your token refresh here in 401
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
I/flutter (30233): 401
I/flutter (30233): Perform your token refresh here in 401
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
I/flutter (30233): 401
I/flutter (30233): Perform your token refresh here in 401
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
I/flutter (30233): 401
I/flutter (30233): Perform your token refresh here in 401
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
I/flutter (30233): 401
I/flutter (30233): Perform your token refresh here in 401
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
I/flutter (30233): 401
I/flutter (30233): Perform your token refresh here in 401
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
I/flutter (30233): 401
I/flutter (30233): Perform your token refresh here in 401
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
I/flutter (30233): 401
I/flutter (30233): Perform your token refresh here in 401
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
I/flutter (30233): 401
I/flutter (30233): Perform your token refresh here in 401
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
I/flutter (30233): 401
I/flutter (30233): Perform your token refresh here in 401
I/flutter (30233): {id: 707860, appid: YOUR-KEY-HERE, units: metric}
工作演示
完整代码
import 'dart:convert';
import 'package:flutter/material.dart';
import 'package:http_interceptor/http_interceptor.dart';
import 'package:http/http.dart' as http;
const cities = [
{
"id": 707860,
"name": "Hurzuf",
"country": "UA",
"coord": {"lon": 34.283333, "lat": 44.549999}
},
{
"id": 519188,
"name": "Novinki",
"country": "RU",
"coord": {"lon": 37.666668, "lat": 55.683334}
},
{
"id": 1283378,
"name": "Gorkhā",
"country": "NP",
"coord": {"lon": 84.633331, "lat": 28}
},
{
"id": 1270260,
"name": "State of Haryāna",
"country": "IN",
"coord": {"lon": 76, "lat": 29}
},
];
const String OPEN_WEATHER_API_KEY = "YOUR-KEY-HERE";
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
home: HomeScreen(),
);
}
}
class ExpiredTokenRetryPolicy extends RetryPolicy {
@override
int maxRetryAttempts = 10;
@override
bool shouldAttemptRetryOnResponse(http.Response response) {
print(response.statusCode);
if (response.statusCode == 401) {
print("Perform your token refresh here in 401");
return true;
}
return false;
}
}
class HomeScreen extends StatefulWidget {
@override
_HomeScreenState createState() => _HomeScreenState();
}
class _HomeScreenState extends State<HomeScreen> {
WeatherRepository repository = WeatherRepository(
HttpClientWithInterceptor.build(interceptors: [
WeatherApiInterceptor(),
], retryPolicy: ExpiredTokenRetryPolicy()),
);
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
centerTitle: true,
title: const Text('Weather App'),
actions: <Widget>[
IconButton(
icon: Icon(Icons.search),
onPressed: () {
showSearch(
context: context,
delegate: WeatherSearch(repository),
);
},
)
],
),
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
Icon(
Icons.wb_sunny,
size: 64,
color: Colors.grey,
),
Container(
height: 16,
),
Text(
"Search for a city",
style: TextStyle(
fontSize: 24.0,
fontWeight: FontWeight.w300,
),
textAlign: TextAlign.center,
),
],
),
),
);
}
}
class WeatherSearch extends SearchDelegate<String> {
int selected = -1;
WeatherRepository repo;
WeatherSearch(this.repo);
@override
List<Widget> buildActions(BuildContext context) {
return [
IconButton(
icon: Icon(Icons.clear),
onPressed: () {
selected = -1;
query = "";
},
)
];
}
@override
Widget buildLeading(BuildContext context) {
return IconButton(
icon: AnimatedIcon(
icon: AnimatedIcons.menu_arrow,
progress: transitionAnimation,
),
onPressed: () {
close(context, null);
},
);
}
@override
Widget buildResults(BuildContext context) {
final city = selected == -1 ? null : cities[selected];
return city != null ? buildWeatherCard(city) : buildEmptyCard();
}
@override
Widget buildSuggestions(BuildContext context) {
final suggestionList = query.isEmpty
? cities
: cities.where((p) => p["name"].toString().startsWith(query)).toList();
return ListView.builder(
itemCount: suggestionList.length,
itemBuilder: (context, index) {
return ListTile(
onTap: () {
selected = index;
query = cities[selected]["name"];
showResults(context);
},
title: Text(suggestionList[index]['name']),
subtitle: Text(suggestionList[index]['country']),
);
},
);
}
Widget buildWeatherCard(final city) {
return FutureBuilder(
future: repo.fetchCityWeather(city["id"]),
builder: (context, snapshot) {
if (snapshot.hasError) {
return Center(
child: Text(snapshot.error),
);
}
if (!snapshot.hasData) {
return Center(
child: CircularProgressIndicator(),
);
}
final weather = snapshot.data;
final iconWeather = weather["weather"][0]["icon"];
final main = weather["main"];
final wind = weather["wind"];
return Card(
margin: EdgeInsets.all(16.0),
child: Container(
width: Size.infinite.width,
padding: EdgeInsets.all(16.0),
child: Column(
mainAxisSize: MainAxisSize.min,
children: <Widget>[
ListTile(
leading: Tooltip(
child: Image.network(
"https://openweathermap.org/img/w/$iconWeather.png"),
message: weather["weather"][0]["main"],
),
title: Text(city["name"]),
subtitle: Text(city["country"]),
),
ListTile(
title: Text("${main["temp"]} °C"),
subtitle: Text("Temperature"),
),
ListTile(
title: Text("${main["temp_min"]} °C"),
subtitle: Text("Min Temperature"),
),
ListTile(
title: Text("${main["temp_max"]} °C"),
subtitle: Text("Max Temperature"),
),
ListTile(
title: Text("${main["humidity"]} %"),
subtitle: Text("Humidity"),
),
ListTile(
title: Text("${main["pressure"]} hpa"),
subtitle: Text("Pressure"),
),
ListTile(
title: Text("${wind["speed"]} m/s"),
subtitle: Text("Wind Speed"),
),
],
),
),
);
},
);
}
Widget buildEmptyCard() {
return Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
Icon(
Icons.wb_sunny,
size: 64,
color: Colors.grey,
),
Container(
height: 16,
),
Text(
"Search for a city",
style: TextStyle(
fontSize: 24.0,
fontWeight: FontWeight.w300,
),
textAlign: TextAlign.center,
),
],
),
);
}
}
const baseUrl = "https://api.openweathermap.org/data/2.5";
class WeatherRepository {
HttpClientWithInterceptor client;
WeatherRepository(this.client);
// Alternatively you can forget about using the Client and just doing the HTTP request with
// the HttpWithInterceptor.build() call.
// Future<Map<String, dynamic>> fetchCityWeather(int id) async {
// var parsedWeather;
// try {
// var response = await HttpWithInterceptor.build(
// interceptors: [WeatherApiInterceptor()])
// .get("$baseUrl/weather", params: {'id': "$id"});
// if (response.statusCode == 200) {
// parsedWeather = json.decode(response.body);
// } else {
// throw Exception("Error while fetching. \n ${response.body}");
// }
// } catch (e) {
// print(e);
// }
// return parsedWeather;
// }
Future<Map<String, dynamic>> fetchCityWeather(int id) async {
var parsedWeather;
try {
final response =
await client.get("$baseUrl/weather", params: {'id': "$id"});
if (response.statusCode == 200) {
parsedWeather = json.decode(response.body);
} else {
return Future.error(
"Error while fetching.",
StackTrace.fromString("${response.body}"),
);
}
} catch (e) {
print(e);
}
return parsedWeather;
}
}
class WeatherApiInterceptor implements InterceptorContract {
@override
Future<RequestData> interceptRequest({RequestData data}) async {
try {
data.params['appid'] = OPEN_WEATHER_API_KEY;
data.params['units'] = 'metric';
data.headers["Content-Type"] = "application/json";
} catch (e) {
print(e);
}
print(data.params);
return data;
}
@override
Future<ResponseData> interceptResponse({ResponseData data}) async => data;
}
关于flutter - 如何为flutter集成Http拦截器以重试请求?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61579974/
是否可以重试网络客户端请求?在奇怪的情况下,我的应用程序在尝试连接到 xml Web 服务时会抛出错误,但如果我重试,它就可以正常工作。我希望它在抛出错误之前重试 2 次,除非有人有更好的解决方案:)
我在一本书中找到了这段代码片段: int ival; // read cin and test only for EOF; loop is executed even if there are oth
是否可以使用 for lop 来设置对象的条件。如果该条件未通过测试(if 语句),则更改条件直到它通过测试?这是我的伪代码尝试,但我怀疑它是否有效: for (int i = 0; i < myAr
我有以下问题。我的主要 Activity 由一个 ListView 组成,其中填充了从 Web 服务获得的数据。首次加载主要 Activity 时,以防无法从网络检索数据,我想显示一个带有 2 个按钮
我有微服务应用程序。为了协作,每个服务都使用异步消息传递。我知道,spring data jpa 默认使用乐观锁。但是如果这种锁定不是由用户调用,而是由另一个服务调用的方法(在我的示例中有验证服务,可
我希望能够在 F# 中编写一个计算表达式,如果它抛出异常,它将能够重试操作。现在我的代码看起来像: let x = retry (fun() -> GetResourceX()) let y = re
是否可以在 NServiceBus 版本 3.2.2 中禁用重试? 使用以下配置,可以禁用重试: 但当线程数设置为 20 时则不会。在这种情况下,消息会重试两次: 这看起来很像
我在 failed_jobs 上有多个失败的作业。我尝试重新排队 MaxAttemptsExceededException 但总是失败。如何重试那里的工作类型? 注意:每次我通过 php artisa
下面的 sproc 尝试向表中插入一行并生成一个随机 ID,用于在相应表上进行 PK。与随机生成的 ID 的冲突在 catch 块中处理,在那里再次重试/调用该过程。现在,这需要很长时间并导致死锁,因
我试图实现代码以使用“mocha-retry”重试失败的测试以下是示例。 describe(retries,' retries-',function () { var self = this;
我正在尝试通过 Azure 数据工厂将数据从 Azure 数据湖存储插入到 Azure 表。 Azure Data Lake 文件中的数据与最终 Azure 表接收器的架构相同。 ADF 管道包含从
是http.RoundTripper在 Go 中基于 HTTP 状态代码(例如 429)实现 http GET 请求重试机制的正确位置? 它在某种程度上“感觉正确”( Go Playground )并
使用 spring reactive WebClient,我使用了一个 API,如果响应状态为 500,我需要使用指数退避重试。但是在 Mono 类中,我没有看到任何以 Predicate 作为输入参
我一直在尝试编写 react native 的快速入门指南,但一直收到此错误 There appears to be trouble with your network connection. Ret
我正在尝试使用从我们心爱的堆栈溢出中获取的 Retry Monad: type RetryBuilder(max, sleep : TimeSpan) = member x.Return(
使用 spring reactive WebClient,我使用了一个 API,如果响应状态为 500,我需要使用指数退避重试。但是在 Mono 类中,我没有看到任何以 Predicate 作为输入参
我有一个由 C#.NET 4.0 开发的两层 Windows 窗体应用程序。在这个应用程序中,我读取文件内容并在数据访问层中创建实体列表,并将其返回到 GUI 层以在 GridView 中显示。在我当
如果有人问过这个问题,我深表歉意,但我已经做了很多搜索,但还没有找到与我的问题类似的问题。 在我的应用程序中,我有一个密码更改页面,如果某人是新用户或重置了他/她的密码,该页面就会触发。 问题是,一旦
想知道为什么我的 promise 正在解决但试图重试。 var getResultsStream = url => Rx.Observable.onErrorResumeNext( Rx.O
假设我有以下 Promise 链: var result = Promise.resolve(filename) .then(unpackDataFromFile) .then(tra
我是一名优秀的程序员,十分优秀!