0

0

使用 Go 的 FileServer 安全地提供静态文件

聖光之護

聖光之護

发布时间:2025-11-02 20:30:01

|

1080人浏览过

|

来源于php中文网

原创

使用 go 的 fileserver 安全地提供静态文件

本文介绍了如何使用 Go 语言的 net/http 包中的 FileServer 函数来安全地提供静态文件,特别是如何在只允许客户端访问特定文件(例如 index.html)的同时,保护其他文件(例如 JavaScript 文件)不被直接访问。文章提供了两种实现方法:使用中间件包装 FileServer 和自定义 http.FileSystem 接口。

在使用 Go 构建 Web 应用时,经常需要提供静态文件,例如 HTML、CSS、JavaScript 和图片。net/http 包中的 FileServer 函数提供了一种简单的方式来实现这个功能。然而,默认情况下,FileServer 会暴露指定目录下的所有文件,这可能带来安全风险。例如,你可能希望只允许用户访问 index.html,而阻止直接访问 JavaScript 文件。本文将介绍两种方法来解决这个问题。

方法一:使用中间件过滤文件

第一种方法是使用中间件来包装 FileServer,通过中间件来过滤客户端的请求,只允许访问特定类型的文件。下面是一个示例代码:

package main

import (
    "log"
    "net/http"
    "path/filepath"
)

// GlobFilterHandler 是一个中间件,用于过滤文件,只允许访问符合指定模式的文件。
func GlobFilterHandler(h http.Handler, pattern string) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        path := r.URL.Path
        fileName := filepath.Base(path)

        ok, err := filepath.Match(pattern, fileName)
        if err != nil {
            log.Println("Error in pattern match:", err)
            http.Error(w, "Internal Server Error", http.StatusInternalServerError) // 更加友好的错误提示
            return
        }
        if !ok {
            http.NotFound(w, r)
            return
        }

        h.ServeHTTP(w, r)
    })
}

func main() {
    // 假设静态文件位于 /tmp/dtest 目录
    fileHandler := http.FileServer(http.Dir("/tmp/dtest"))

    // 只允许访问 *.js 文件
    wrappedHandler := GlobFilterHandler(fileHandler, "*.js")

    // 将处理程序注册到服务器
    http.Handle("/", wrappedHandler)  // 注意这里需要指定路由

    // 启动服务器
    log.Fatal(http.ListenAndServe(":8080", nil))
}

代码解释:

  1. GlobFilterHandler 函数接收一个 http.Handler 和一个模式字符串作为参数,返回一个新的 http.Handler。
  2. 新的 http.Handler 会检查请求的 URL 路径,提取文件名,并使用 filepath.Match 函数来判断文件名是否符合指定的模式。
  3. 如果文件名不符合模式,或者匹配过程中发生错误,则返回 404 Not Found 错误。
  4. 如果文件名符合模式,则调用原始的 http.Handler 来处理请求。

使用方法:

  1. 将上述代码保存为 main.go 文件。
  2. 创建一个名为 /tmp/dtest 的目录,并在该目录下创建一些文件,例如 index.html 和 script.js
  3. 运行 go run main.go 命令启动服务器。
  4. 浏览器中访问 http://localhost:8080/script.js 可以正常访问,访问 http://localhost:8080/index.html 会返回 404 Not Found 错误。

注意事项:

AssemblyAI
AssemblyAI

转录和理解语音的AI模型

下载
  • 这个示例只允许访问 .js 文件。你可以根据需要修改模式字符串来允许访问其他类型的文件。
  • 需要注意路由的设置,确保中间件能够正确地拦截请求。

方法二:自定义 http.FileSystem 接口

第二种方法是自定义 http.FileSystem 接口,并实现自己的文件访问控制逻辑。下面是一个示例代码:

package main

import (
    "fmt"
    "log"
    "net/http"
    "path/filepath"
)

// GlobDir 是一个自定义的 http.FileSystem 实现,用于过滤文件,只允许访问符合指定模式的文件。
type GlobDir struct {
    Dir     http.Dir
    Pattern string
}

// Open 方法实现了 http.FileSystem 接口,用于打开文件。
func (d GlobDir) Open(name string) (http.File, error) {
    baseName := filepath.Base(name)

    ok, err := filepath.Match(d.Pattern, baseName)
    if err != nil {
        return nil, err
    }
    if !ok {
        return nil, fmt.Errorf("%s not match GlobDir pattern.", baseName)
    }

    return d.Dir.Open(name)
}

func main() {
    // 假设静态文件位于 /tmp/dtest 目录
    fileHandler := http.FileServer(GlobDir{
        Dir:     http.Dir("/tmp/dtest"),
        Pattern: "*.js",
    })

    // 将处理程序注册到服务器
    http.Handle("/", fileHandler) // 注意这里需要指定路由

    // 启动服务器
    log.Fatal(http.ListenAndServe(":8080", nil))
}

代码解释:

  1. GlobDir 结构体实现了 http.FileSystem 接口,包含一个 http.Dir 类型的成员和一个模式字符串。
  2. Open 方法是 http.FileSystem 接口的必须实现的方法,用于打开文件。
  3. Open 方法首先提取文件名,并使用 filepath.Match 函数来判断文件名是否符合指定的模式。
  4. 如果文件名不符合模式,则返回一个错误。
  5. 如果文件名符合模式,则调用原始的 http.Dir 的 Open 方法来打开文件。

使用方法:

  1. 将上述代码保存为 main.go 文件。
  2. 创建一个名为 /tmp/dtest 的目录,并在该目录下创建一些文件,例如 index.html 和 script.js。
  3. 运行 go run main.go 命令启动服务器。
  4. 在浏览器中访问 http://localhost:8080/script.js 可以正常访问,访问 http://localhost:8080/index.html 会返回一个错误。

注意事项:

  • 这个示例只允许访问 .js 文件。你可以根据需要修改模式字符串来允许访问其他类型的文件。
  • 自定义 http.FileSystem 接口可以提供更灵活的文件访问控制逻辑。

总结

本文介绍了两种使用 Go 的 FileServer 安全地提供静态文件的方法:使用中间件过滤文件和自定义 http.FileSystem 接口。你可以根据实际需求选择合适的方法。使用中间件的方法更加简单,但灵活性稍差。自定义 http.FileSystem 接口的方法更加灵活,但实现起来稍微复杂一些。无论选择哪种方法,都需要注意文件访问控制,确保Web应用的安全。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

184

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

226

2025.12.18

什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

184

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

226

2025.12.18

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

761

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1570

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

651

2023.11.24

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

49

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Sass 教程
Sass 教程

共14课时 | 0.9万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.6万人学习

CSS教程
CSS教程

共754课时 | 43.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号