定时器(@Scheduled)使用feign夸服务调用,Oauth2 客户端client_credentials模式

1 篇文章 0 订阅
1 篇文章 0 订阅

定时器(@Scheduled)使用feign夸服务调用,Oauth2 客户端client_credentials模式


应用场景

定时器(@Scheduled)使用feign夸服务调用(A——>B)。因为不是web请求,所以没有HTTP 请求信息,所以:ServletRequestAttributes attributes = (ServletRequestAttributes)RequestContextHolder.getRequestAttributes(); 获取不到token。报null值异常。
(PS:我的A服务是专门做定时器的服务,不走任何web请求!

解决办法(仅代表个人)

我在A服务调用B服务之前加了一个私有的feign拦截器(写在了A服务里面):使用Oauth2的客户端(client_credentials)模式获取到token并塞进了请求头中。

参数说明
/oauth/token 是Oauth2 提供的token获取地址
grant_type=client_credentials 代表客户端模式请求token
client_id=schedule “schedule”数据库设置的
client_secret=123456 “123456 ”数据库设置的
scope 代表权限,不填代表所有

在这里插入图片描述

package cn.config;
import feign.RequestInterceptor;
import feign.RequestTemplate;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.client.RestTemplate;

import javax.annotation.Resource;
import java.util.Map;

/**
 * 实现RequestInterceptor接口:
 * feign全局设置求头信息
 */
@Configuration
public class MyFeignRequestInterceptor implements RequestInterceptor {

    @Resource
    RestTemplate restTemplate;

    @Override
    public void apply(RequestTemplate requestTemplate) {
        Map<String,Object> objectMap = restTemplate.getForObject("http://jhcloud-portal/oauth/token?grant_type=client_credentials&client_id=schedule&client_secret=123456",Map.class);
        String token = objectMap.get("access_token").toString();
        requestTemplate.header("Authorization", "bearer "+token);
    }
}

然后在全局的feignConfig中加个判断
package cn.jhcloud_common.config;
@Configuration
public class FeignConfig implements RequestInterceptor {

    @Override
    public void apply(RequestTemplate requestTemplate) {
        ServletRequestAttributes attributes = (ServletRequestAttributes)RequestContextHolder.getRequestAttributes();
        if (attributes == null) {
        //因为私有的feign拦截器,所以请求头里已经有了token信息了,就不需要在这里在做处理了!
        }else {
            HttpServletRequest request = attributes.getRequest();
            requestTemplate.header(HttpHeaders.AUTHORIZATION, request.getHeader(HttpHeaders.AUTHORIZATION));
        }
        //seata分布式事务
          requestTemplate.header(RootContext.KEY_XID, RootContext.getXID());
    }
}
使用 RestTemplate调用的时候最好设置一个全局的:RestTemplateConfig,不然可能会报错。
package cn.jhcloud_common.config;

import org.springframework.cloud.client.loadbalancer.LoadBalanced;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.client.RestTemplate;

@Configuration
public class RestTemplateConfig {
    @Bean
    @LoadBalanced //让这个RestTemplate在请求时拥有客户端负载均衡的能力
    public RestTemplate getRestTemplate() {
        return new RestTemplate();
    }

}
yml 里面最好也设置一下,如果不设置的话。会报:I/O ERROR ON GET REQUEST FOR "http://jhcloud-portal/auth/currentUser":
security:
  oauth2:
    resource:
      #每次调用接口都获取一下当前登录人的信息
      user-info-uri: http://jhcloud-portal/auth/currentUser 
      loadBalanced: true #让这个RestTemplate在请求时拥有客户端负载均衡的能力

在这里插入图片描述

  • [ 1、] 可能存在的问题,因为yml里面设置了每次调用接口之前都需要先获取一下当前登录人的信息,然而客户端模式获取到的token,不是用户token,所以没有用户信息,相应的也没有用户权限。所以postmain可能会"无效的token"错误。根据控制台报错信息定位一下会发现是:FixedAuthoritiesExtractor.extractAuthorities 报错
  • 在这里插入图片描述
    所以你可能需要在获取用户信息的地方,“/currentUser”。自己封装一下权限
    在这里插入图片描述
    然后正常调用就可以了。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

因为时间有点长,遇到的其他问题也忘了。先写这么多吧。想起来再补充.

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: spring定时器@scheduled是一种基于注解的定时任务调度方式,可以在指定的时间间隔或固定时间点执行任务。通过在方法上添加注解@scheduled,可以指定任务的执行时间、执行频率、执行条件等。这种方式简单易用,适用于各种场景下的定时任务调度。 ### 回答2: Spring定时器@scheduled是一种非常常用和方便的定时任务调度方式,它是Spring框架中的一部分。使用@scheduled注解可以让方法按照一定的时间间隔或者某个特定时间点去执行,可以说是非常方便实用了。 @scheduled注解必须放在方法上,可以设置多个属性来指定任务的执行时间和间隔。其中,cron属性是一个非常重要的属性,它使用了系统的cron表达式来指定任务的执行时间,这样我们可以实现非常精细的任务调度,实现复杂的任务流程。 除了cron之外,还有fixedDelay和fixedRate属性,分别表示延迟多少毫秒后再次执行和间隔多少毫秒再次执行,这两个属性适用于固定间隔执行的任务。 @scheduled注解还支持设置initialDelay属性,表示任务延迟多少秒后开始执行。这在某些情况下非常有用,如需要在系统启动后延迟一定时间再执行的任务。 除了定时任务之外,@scheduled还支持retry注解,可以在任务执行失败时自动重新执行几次,这对于保证任务的可靠性非常重要。 需要注意的是,@scheduled注解只能用于Spring Bean中的方法,也就是说,这样的方法必须是由Spring容器来管理的,只有被纳入了Spring容器的Bean才能被@scheduled注解所标注。 总之,Spring定时器@scheduled的使用非常广泛,实用性非常高。需要我们根据业务需求来灵活运用,并注意任务的可靠性和性能问题。 ### 回答3: Spring框架提供了一个方便的定时任务处理的功能模块:@Scheduled。这个注解就是用来创建一个定时任务方法的。 基本使用方式 在一个方法上添加 @Scheduled 注解, 并设置触发执行时间。例如,下面代码是定义了一个每5秒执行一次的定时任务: @Scheduled(fixedRate = 5000) public void doSomething() { //TODO: 实现定时任务所需的业务逻辑... } 同时要注意,要开启对 @Scheduled 的支持,需要在 Spring 配置文件中添加以下配置: <task:annotation-driven/> 其他属性 除了 fixedRate 属性,还有另外两个常用属性: - fixedDelay:在两次执行之间存在延迟(即两次执行之间需要等等),单位是毫秒; - cron:使用 Cron 表达式支持,更加灵活,更精准地设置执行的时间,类似于Linux中的Crontab。比如: - @Scheduled(cron="0 15 10 ? * *") // 每天10点15分执行 - @Scheduled(cron="0 0/5 14,18 * * ?") // 每天下午2点到3点,每个五分钟执行一次 总结 @Scheduled 可以让程序员非常方便地实现定时任务的功能,使用起来也很容易理解。 同时因为 Spring 框架支持的任务类型非常丰富,包括定时任务、异步执行、并发运行、任务队列等等,可谓是应对各种需求场景的优秀工具。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值