如何实现java调用接口后发现token失效或过期,自动刷新token并存入redis,如果服务端返回过新token,旧token将会失效,无法确认当前token是否为最新的
时间: 2024-04-23 22:11:44 浏览: 224
可以在调用接口前,先从 Redis 中获取当前 token,然后再进行接口调用。如果接口返回了过期或失效的错误信息,说明当前的 token 已经过期或失效了,此时可以使用刷新 token 的接口获取最新的 token,然后将最新的 token 存入 Redis 中,并更新当前使用的 token。
为了避免并发问题,可以使用 Redis 的分布式锁来确保只有一个线程可以进行 token 的刷新和存储操作。如果服务端返回的 token 与 Redis 中存储的 token 不一致,则说明当前的 token 已经失效并被服务端更新,需要重新获取最新的 token 并重试接口调用。
另外,为了避免频繁刷新 token,可以在获取到最新 token 后,将其过期时间减少一定的时间,例如将其过期时间设置为原来过期时间的一半,这样可以有效减少 token 刷新的次数。
相关问题
如何实现java调用接口后发现token失效或过期,自动刷新token并存入redis
在Java中,可以使用拦截器(Interceptor)或过滤器(Filter)来实现在调用接口前和调用接口后的一些操作,比如检查token是否失效或过期,自动刷新token并存入Redis。
以下是一个基本的实现过程:
1. 编写拦截器或过滤器,在调用接口前和调用接口后进行操作。
2. 在调用接口前,从Redis中获取token,并检查token是否失效或过期。
3. 如果token失效或过期,则根据刷新token的接口重新获取新的token,并将新的token存入Redis。
4. 在调用接口时,将新的token传递给接口。
下面是一个简单的实现示例:
```java
public class TokenInterceptor implements HandlerInterceptor {
@Autowired
private RedisTemplate<String, String> redisTemplate;
@Value("${token.expireTime}")
private long expireTime;
@Value("${token.refreshUrl}")
private String refreshUrl;
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
// 从请求头中获取token
String token = request.getHeader("token");
if (StringUtils.isEmpty(token)) {
throw new RuntimeException("token不能为空");
}
// 从Redis中获取token对应的value值
String value = redisTemplate.opsForValue().get(token);
if (StringUtils.isEmpty(value)) {
throw new RuntimeException("token已失效,请重新登录");
}
// 检查token是否过期
long expire = redisTemplate.getExpire(token, TimeUnit.SECONDS);
if (expire <= 0) {
// token已过期,重新获取并存入Redis
String refreshedToken = refresh(token);
redisTemplate.opsForValue().set(refreshedToken, value, expireTime, TimeUnit.SECONDS);
redisTemplate.delete(token);
// 将新的token传递给接口
request.setAttribute("token", refreshedToken);
} else {
// 将token传递给接口
request.setAttribute("token", token);
}
return true;
}
private String refresh(String token) {
// 调用刷新token的接口,获取新的token
String refreshedToken = restTemplate.getForObject(refreshUrl + "?token=" + token, String.class);
if (StringUtils.isEmpty(refreshedToken)) {
throw new RuntimeException("刷新token失败");
}
return refreshedToken;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
// do nothing
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
// do nothing
}
}
```
在上面的代码中,我们定义了一个TokenInterceptor拦截器,并实现了preHandle方法,在该方法中进行了token的检查和刷新操作。
我们使用了Spring的RedisTemplate来操作Redis,其中expireTime和refreshUrl是从配置文件中读取的参数。如果token失效或过期,则调用refresh方法获取新的token,并将新的token存入Redis中,同时将新的token传递给接口。
我们还可以在配置文件中配置该拦截器:
```java
@Configuration
public class WebMvcConfig implements WebMvcConfigurer {
@Autowired
private TokenInterceptor tokenInterceptor;
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(tokenInterceptor).addPathPatterns("/**");
}
}
```
这样,当我们调用接口时,TokenInterceptor拦截器会自动检查token并刷新token,确保我们的接口调用是有效的。
5)应用系统通过code(授权码)调用统一门户提供的接口来获取access_token访问令牌。应用系统应该保存access_token。access_token在下2小时后失效,应用系统也可以调用注销功能提前让access_token失效 java如何实现
在Java中,应用系统可以按照以下步骤实现通过code获取access_token并管理其有效期:
1. **调用接口获取access_token**:
使用HTTP客户端库(如Apache HttpClient、OkHttp或Spring的RestTemplate),构造一个请求发送到统一门户提供的API端点。这个端点通常接受授权码(code)作为参数,并返回新的access_token。
```java
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.StringEntity;
String apiUrl = "统一门户接口地址";
HttpPost httpPost = new HttpPost(apiUrl);
httpPost.setHeader("Content-Type", "application/x-www-form-urlencoded");
StringEntity params = new StringEntity("grant_type=authorization_code&code=your_code");
httpPost.setEntity(params);
HttpClient httpClient = HttpClientBuilder.create().build();
CloseableHttpResponse response = null;
try {
response = httpClient.execute(httpPost);
// ...处理响应,获取access_token
} finally {
if (response != null) {
response.close();
}
}
```
2. **保存access_token**:
将获取到的access_token存储在一个安全的地方,如数据库、内存缓存(例如Redis)或者JWT(JSON Web Token)中,设置合理的过期时间。
```java
import com.fasterxml.jackson.databind.ObjectMapper;
import net.minidev.json.JSONObject; // JSON库
// 假设我们使用HashMap存储
Map<String, String> tokenCache = new HashMap<>();
JSONObject accessTokenObj = new JSONObject();
accessTokenObj.put("token", accessToken);
accessTokenObj.put("expires_at", String.valueOf(System.currentTimeMillis() + (2 * 60 * 60 * 1000))); // 过期时间(2小时)
tokenCache.put(appId, accessTokenObj.toString());
```
3. **检查并刷新access_token**:
当需要访问受保护的资源之前,先检查access_token是否已过期。如果未过期,则直接使用;如果已过期,调用刷新接口(通常需要使用refresh_token)获取新token,然后更新缓存。
```java
public synchronized String getTokenForApp(String appId) {
String accessTokenFromCache = tokenCache.get(appId);
if (accessTokenFromCache == null || isAccessTokenExpired(accessTokenFromCache)) {
// 刷新access_token
String refreshToken = getRefreshToken(appId); // 获取refresh_token
accessToken = refreshTokenAccessToken(refreshToken);
updateTokenCache(appId, accessToken);
}
return accessToken;
}
private boolean isAccessTokenExpired(String accessTokenJson) {
try {
JSONObject accessTokenObject = new JSONObject(accessTokenJson);
long expiresIn = Long.parseLong(accessTokenObject.get("expires_in").toString());
long currentTime = System.currentTimeMillis();
return currentTime > Long.parseLong(accessTokenObject.get("expires_at").toString()) - expiresIn * 1000;
} catch (Exception e) {
return true; // 认定已过期
}
}
private String refreshTokenAccessToken(String refreshToken) {
// 调用刷新接口并替换access_token
// ...
return newAccessToken;
}
private void updateTokenCache(String appId, String accessToken) {
tokenCache.put(appId, accessToken);
}
```
4. **注销功能**:
实现一个方法用于主动清除access_token,这通常涉及到向服务端发送一个注销请求,服务端在接收到请求后会从其记录中删除该用户的access_token。
```java
public void logoutUser(String appId) {
String refreshToken = getTokenForApp(appId).get("refresh_token");
// 发送注销请求
revokeTokenApi(refreshToken);
// 清除本地缓存
tokenCache.remove(appId);
}
```
阅读全文