springboot实现文件防盗链设计

简介: `shigen`,一位专注于Java、Python、Vue和Shell的博主,分享成长和技术。近期将探讨SpringBoot实现图片防盗链,通过限制`Referer`防止资源被盗用。基础版通过`WebMvcConfigurer`配置静态资源,升级版添加拦截器检查`Referer`,确保请求来源合法性。详细代码实现和案例可在文中链接找到。一起学习,每天进步!

shigen坚持更新文章的博客写手,擅长Java、python、vue、shell等编程语言和各种应用程序、脚本的开发。记录成长,分享认知,留住感动。
个人IP:shigen

👋👋👋hello,伙伴们好久不见,最近shigen遇到了很多的事情,现在算是安顿好了,也可以开始技术分享了。但是分享的频率的话,肯定会比之前低的很多,因为我也要正式的学习新的技术了。

来今天的正题:springboot实现图片防盗链。可能看起来场景比较抽象,这里shigen给出之前的一个例子:对象存储服务的流量被盗刷了,当时官方给的解决方案包括我后来采用的方式就是referer的限制。

referer的限制

后来我的对象存储服务的流量就正常了。那今天我也是好奇这个用springboot怎么实现。在接下来的内容中,我将会着重分享我的设计。

首先了解一下Referer是什么吧。

什么是Referer

这里告别充满广告和垃圾网站的搜索引擎,直接GPT查询:

Referer(来源)是HTTP头部字段之一,用于指示客户端是从哪个页面跳转或发起请求的。当客户端(通常是浏览器)向服务器发送请求时,它会在HTTP头部中包含 Referer 字段,告诉服务器请求的来源页面的URL。这个字段可以帮助服务器了解请求的上下文和用户行为,有助于进行数据分析、安全检查等操作。

也就是说请求一个资源的时候,浏览器的请求头信息中会带上Referer字段标示出当下的请求的上一个请求是什么地方来的。

那基于这个原理,我们就可以设计出自己的防盗链。

java代码的设计

基础版

假设我们的springboot项目中可以直接通过http请求访问到某个路径下的资源。我们先这样的尝试吧。我们的配置肯定要实现WebMvcConfigurer这个接口,实现资源的映射。那我就直接展示我的代码:

@Configuration
@EnableWebMvc
@Slf4j
public class MvcConfig implements WebMvcConfigurer {
   
   

    /**
     * 静态资源保存目录
     */
    public static final String FILE_RESOURCE_PATH = "file:" + System.getProperty("user.dir") + "/files/";

    @Override
    public void addResourceHandlers(ResourceHandlerRegistry registry) {
   
   
        log.error("静态资源保存目录:{}", FILE_RESOURCE_PATH);
        registry.addResourceHandler("/files/**")
                .addResourceLocations(FILE_RESOURCE_PATH);
    }
}

其实我们最终实现的文件路径就是项目根路径/files/文件夹的全部文件。

这样我们就可以通过http请求访问了。

http访问静态资源

但是,明显的我们的资源不是很安全。因为任意来源、任何人都可以访问到它。那我们限制来源的话,这个时候Referer就可以派上用场了。

升级版

升级版本,我们就需要统一拦截一下请求,看看请求头中是否包含Referer信息,且是我们约定的Referer。这样才能判定是正常的请求,进行流量的放行,否则的话就是要去拦截。

接下来先去设计一个拦截器:

@Slf4j
@Component
public class ResourceInterceptor extends HandlerInterceptorAdapter {
   
   

    @Resource
    private ReferConfig referConfig;

    /**
     * 匹配的文件种类
     */
    private static final String FILE_REGEX = "\\.(html|css|js|jpg|jpeg|png|gif|bmp|svg|pdf|doc|docx|xls|xlsx|ppt|pptx|mp4|mov)$";
    private static final Pattern FILE_REGEX_PATTERN = Pattern.compile(FILE_REGEX, Pattern.CASE_INSENSITIVE);


    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
   
   
        // 获取请求的 URL
        String requestUrl = request.getRequestURL().toString();
        log.info("requestUrl:{}", requestUrl);
        // 检查是否是静态资源请求
        if (referConfig.isEnabled() && isStaticResource(requestUrl)) {
   
   
            // 检查防盗链策略
            if (!isValidReferer(request)) {
   
   
                response.setStatus(HttpServletResponse.SC_FORBIDDEN);
                return false;
            }
        }
        return true;
    }

    /**
     * 正则验证请求的资源
     *
     * @param url 请求资源
     * @return 是否匹配
     */
    private boolean isStaticResource(String url) {
   
   
        return FILE_REGEX_PATTERN.matcher(url).find();
    }

    /**
     * 检查 Referer 头,判断请求是否合法
     *
     * @param request 请求
     * @return 是否是合法请求
     */
    private boolean isValidReferer(HttpServletRequest request) {
   
   
        String referer = request.getHeader("Referer");
        return CollectionUtil.contains(referConfig.getAllowedOrigins(), referer);
    }
}

具体的业务逻辑的验证都在注释里,这里需要注意:

  1. 关于Referer的配置最好写成动态的,便于后期的拓展
  2. 对于url的请求判断最好使用正则表达式,因为url本身请求的就是静态资源,但是后边带了其他的参数可能导致直接绕过

对于自定的配置类,shigen是这样的设计:

@Configuration
@ConfigurationProperties(prefix = "refer")
@Data
public class ReferConfig {
   
   
    /**
     * 是否开启防盗链拦截
     */
    private boolean enabled;
    /**
     * 允许的Referer请求
     */
    private List<String> allowedOrigins;

}

接下来就是配置到拦截器上。

@Configuration
@EnableWebMvc
@Slf4j
public class MvcConfig implements WebMvcConfigurer {
   
   

    @Resource
    private UserArgumentResolver userArgumentResolver;

    @Resource
    private ResourceInterceptor resourceInterceptor;

    /**
     * 静态资源保存目录
     */
    public static final String FILE_RESOURCE_PATH = "file:" + System.getProperty("user.dir") + "/files/";

    @Override
    public void addArgumentResolvers(List<HandlerMethodArgumentResolver> resolvers) {
   
   
        resolvers.add(userArgumentResolver);
    }

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
   
   
        registry.addInterceptor(resourceInterceptor).addPathPatterns("/**");
    }

    @Override
    public void addResourceHandlers(ResourceHandlerRegistry registry) {
   
   
        log.error("静态资源保存目录:{}", FILE_RESOURCE_PATH);
        registry.addResourceHandler("/files/**")
                .addResourceLocations(FILE_RESOURCE_PATH);
    }
}

待一切完成,我们这里再检查一下配置文件就可以正常的启动服务进行测试了。

refer:
  enabled: true
  allowed-origins:
    - http://www.shigen.com

此时,我们再次在浏览器中直接访问:

浏览器中403

这时我们想要正常的访问,就得借助于接口测试工具了。

手动加上Referer

以上就是关于文件防盗链的设计分享,相关代码可参考:新增文件防盗链案例

与shigen一起,每天不一样!

目录
相关文章
|
3月前
|
XML 前端开发 Java
基于SpringBoot 3.3实现任意文件在线预览功能的技术分享
【8月更文挑战第30天】在当今的数字化办公环境中,文件在线预览已成为提升工作效率、优化用户体验的重要功能之一。无论是文档、图片、PDF还是代码文件,用户都期望能够直接在浏览器中快速查看而无需下载。本文将围绕如何在Spring Boot 3.3框架下实现这一功能,分享一系列技术干货,助力开发者高效构建文件预览服务。
345 2
|
14天前
|
Java 应用服务中间件
SpringBoot获取项目文件的绝对路径和相对路径
SpringBoot获取项目文件的绝对路径和相对路径
53 1
SpringBoot获取项目文件的绝对路径和相对路径
|
8天前
|
网络协议 Java
springboot配置hosts文件
springboot配置hosts文件
36 11
|
25天前
|
XML Java Kotlin
springboot + minio + kkfile实现文件预览
本文介绍了如何在容器中安装和启动kkfileviewer,并通过Spring Boot集成MinIO实现文件上传与预览功能。首先,通过下载kkfileviewer源码并构建Docker镜像来部署文件预览服务。接着,在Spring Boot项目中添加MinIO依赖,配置MinIO客户端,并实现文件上传与获取预览链接的接口。最后,通过测试验证文件上传和预览功能的正确性。
springboot + minio + kkfile实现文件预览
|
14天前
|
存储 前端开发 JavaScript
|
14天前
|
存储 Java API
|
1月前
|
easyexcel Java UED
SpringBoot中大量数据导出方案:使用EasyExcel并行导出多个excel文件并压缩zip后下载
在SpringBoot环境中,为了优化大量数据的Excel导出体验,可采用异步方式处理。具体做法是将数据拆分后利用`CompletableFuture`与`ThreadPoolTaskExecutor`并行导出,并使用EasyExcel生成多个Excel文件,最终将其压缩成ZIP文件供下载。此方案提升了导出效率,改善了用户体验。代码示例展示了如何实现这一过程,包括多线程处理、模板导出及资源清理等关键步骤。
|
15天前
|
Java
SpringBoot获取文件将要上传的IP地址
SpringBoot获取文件将要上传的IP地址
29 0
|
1月前
|
消息中间件 Java 大数据
大数据-56 Kafka SpringBoot与Kafka 基础简单配置和使用 Java代码 POM文件
大数据-56 Kafka SpringBoot与Kafka 基础简单配置和使用 Java代码 POM文件
65 2
|
1月前
|
Java 关系型数据库 数据库连接
SpringBoot项目使用yml文件链接数据库异常
【10月更文挑战第3天】Spring Boot项目中数据库连接问题可能源于配置错误或依赖缺失。YAML配置文件的格式不正确,如缩进错误,会导致解析失败;而数据库驱动不匹配、连接字符串或认证信息错误同样引发连接异常。解决方法包括检查并修正YAML格式,确认配置属性无误,以及添加正确的数据库驱动依赖。利用日志记录和异常信息分析可辅助问题排查。
179 10