服务内部调用api鉴权忽略token(动态放权)

通常情况下微服务之间调用或开放接口的时候,我们会在资源服务器的security配置当中配置路径,觉得比较麻烦,参考了网上一些案例,写了下面的应用,感觉比较实用的是可以不用谢那么多重复的代码permitAll()方法。灵活配置动态添加删除,只需要一个注解就可以配置接口是否完全开放,是否内部开放,是否外部开放。希望能帮助到有需要的人。

创建一个注解

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface IgnoreToken {
    /**
     * 是否AOP统一处理
     */
    boolean value() default true;
}

注解配置一个切面

@Aspect
@Component
@Slf4j
public class IgnoreTokenAspect implements Ordered {
    @Autowired
    private HttpServletRequest request;

    @Around("@annotation(ignoreToken)")
    public Object around(ProceedingJoinPoint point, IgnoreToken ignoreToken) throws Throwable {
        String header = request.getHeader(SecurityConstants.FROM);
        if (ignoreToken.value() && !StringUtils.equals(SecurityConstants.FROM_IN, header)){
            log.warn("访问接口 {} 没有权限", point.getSignature().getName());
            throw new AccessDeniedException("Access is denied");
        }
        return point.proceed();
    }

    @Override
    public int getOrder() {
        return Ordered.HIGHEST_PRECEDENCE + 1;
    }
}

常量配置

public final class SecurityConstants {
    public static final String FROM = "from";
    public static final String FROM_IN = "in";
}

初始化忽略token的链接

@Configuration
public class PermitAllUrlProperties implements InitializingBean {

    private static final Pattern PATTERN = Pattern.compile("\\\\{(.*?)\\\\}");

    @Autowired
    private ApplicationContext applicationContext;

    private List<String> urls = new ArrayList<>();

    public static final String ASTERISK = "*";

    @Override
    public void afterPropertiesSet() {
        RequestMappingHandlerMapping mapping = applicationContext.getBean("requestMappingHandlerMapping",RequestMappingHandlerMapping.class);
        Map<RequestMappingInfo, HandlerMethod> map = mapping.getHandlerMethods();

        map.keySet().forEach(info -> {
            HandlerMethod handlerMethod = map.get(info);

            // 获取@IgnoreToke注解的替代path variable 为 *
            IgnoreToken method = AnnotationUtils.findAnnotation(handlerMethod.getMethod(), IgnoreToken.class);
            Optional.ofNullable(method).ifPresent(inner -> info.getPatternsCondition().getPatterns()
                    .forEach(url -> urls.add(ReUtil.replaceAll(url, PATTERN, ASTERISK))));

            // 获取@IgnoreToke注解的访问路径替代path variable 为 *
            IgnoreToken controller = AnnotationUtils.findAnnotation(handlerMethod.getBeanType(), IgnoreToken.class);
            Optional.ofNullable(controller).ifPresent(inner -> info.getPatternsCondition().getPatterns()
                    .forEach(url -> urls.add(ReUtil.replaceAll(url, PATTERN, ASTERISK))));
        });
    }

    public List<String> getUrls() {
        return urls;
    }

    public void setUrls(List<String> urls) {
        this.urls = urls;
    }
}

资源服务器配置

@Configuration
@EnableResourceServer
public class EcmContentResourceServerConfigure extends ResourceServerConfigurerAdapter {
    @Autowired
    PermitAllUrlProperties permitAllUrlProperties;

    @Override
    public void configure(HttpSecurity http) throws Exception {
        http.headers().frameOptions().disable();
        http.csrf().disable()
                .requestMatchers().antMatchers("/**")
                .and()
                .headers().frameOptions().disable()
                .and()
                .authorizeRequests()
                //添加注解动态添加放权
                .antMatchers(permitAllUrlProperties.getUrls().stream().distinct().toArray(String[]::new)).permitAll()
                .antMatchers("/**").authenticated();
    }
}

注解使用,直接添加到controller 方法上value = false表示可外部网关访问true表示内部接口外部无法访问。


    @GetMapping(value = "/getSite/{siteId}")
    @IgnoreToken(value = false)
    public EcmResponse getSite(@ApiParam(value = "siteid", required = true) @PathVariable("siteId") String siteId) {

        return siteServiceI.getSiteById(siteId);
    }

内部feign调佣需要请求头添加from参数

@GetMapping(value = "/public/lvanecm/v1/site/getSite/{siteId}") 
EcmResponse getSiteBySiteId(@PathVariable(value="siteId")String siteId,@RequestHeader(SecurityConstants.FROM) String from);

保证内部接口安全,网关拦截器全局处理,外部请求删除请求头中from参数

// 清洗请求头中from 参数
request = exchange.getRequest().mutate().headers(httpHeaders -> httpHeaders.remove(SecurityConstants.FROM)).build();
### 如何在Sa-Token中实现服务内部调用 #### 背景说明 在分布式架构下,微服务之间通常需要进行相互调用来完成业务逻辑。为了保障安全性,在服务间的通信过程中也需要进行身份认证和授操作。对于基于Spring Cloud的应用场景,可以通过集成Sa-Token来解决这一需求。 --- #### 解决方案概述 以下是几种常见的解决方案及其具体实施方法: 1. **通过Feign拦截器传递Token** Feign提供了`RequestInterceptor`接口,允许开发者自定义HTTP请求头的内容。利用该功能可以在每次发起远程调用时自动附加有效的Token[^2]。 ```java import feign.RequestInterceptor; import feign.RequestTemplate; @Component public class TokenInterceptor implements RequestInterceptor { private final static String TOKEN_HEADER_NAME = "Authorization"; @Override public void apply(RequestTemplate template) { // 获取当前线程中的 token (假设已存入 ThreadLocal 或其他共享上下文中) String currentToken = SaHolder.getRequest().getHeader(TOKEN_HEADER_NAME); if(currentToken != null && !currentToken.isEmpty()) { template.header(TOKEN_HEADER_NAME, currentToken); } } } ``` 2. **配置全局过滤器跳过特定路径校验** 如果某些API仅用于内部调用而不对外暴露,则可以直接设置这些端点无需经过Token验证流程。这可通过注解或者统一配置的方式达成[^3]。 示例代码如下: ```java @RestController public class InternalController { /** * 此接口只允许内部访问 */ @GetMapping("/internal-only") @IgnoreToken(value=true) // 设置为 true 表示此接口不允许外部直接访问 public ResponseEntity<String> internalEndpoint() { return ResponseEntity.ok("This is an internal endpoint."); } } ``` 3. **借助Redis同步Session状态** 当多个独立部署的服务都需要维护同一个用户的登录态时,推荐使用集中式的缓存机制比如Redis来保存会话信息。这样即使客户端切换到了不同的服务器实例上也能保持一致性的认证体验[^4]。 - 导入必要的依赖项并调整application.yml文件内的参数; ```yaml spring: redis: host: localhost port: 6379 lettuce: pool: max-active: 8 max-idle: 8 min-idle: 0 sa-token: token-name: satoken timeout: 2592000 # 单位秒,默认一个月有效期 is-share: true # 开启跨域资源共享模式 cache-type: redis # 使用 Redis 存储令牌数据 ``` - 初始化完成后记得重启整个项目集群以便生效新的改动; 4. **增强型安全策略——双重签名技术** 对于更高层次的安全防护要求来说,除了单纯依靠access_token之外还可以额外增加一层随机数nonce以及时间戳timestamp作为辅助因子参与计算最终提交给目标系统的哈希值hashcode。接收方收到消息后再重新按照相同算法还原一遍看两者是否匹配即可判断来源的真实性。 --- ### 总结 综上所述,针对不同实际应用场景可以选择合适的技术手段去满足各自的需求。无论是简单的header复制还是复杂的多维度组合加密措施都能有效提升整体系统的健壮性和抗攻击能力。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值