0

0

Crystal语言的Kemal框架如何接收文件

月夜之吻

月夜之吻

发布时间:2026-01-18 11:51:03

|

563人浏览过

|

来源于php中文网

原创

Kemal中文件上传必须用env.params.files而非body或json,因multipart/form-data需特殊解析;文件对象含filename、content_type、tempfile等属性,须校验类型、净化路径、限制大小并安全处理临时文件。

crystal语言的kemal框架如何接收文件

Kemal 框架中接收文件必须用 env.params.files,不是 env.params.bodyenv.params.json —— 这是最常踩的坑。

为什么 env.params.body 拿不到上传的文件?

因为 HTTP 文件上传(multipart/form-data)的数据结构和普通表单完全不同:文件内容是二进制块 + 边界分隔符,不能被当成普通键值对解析。Kemal 会把这类请求自动解析为 HTTP::FormData 对象,并将文件单独挂载到 env.params.files 下,而 env.params.body 只保留纯文本字段(如 namedescription 等)。

  • 错误写法:env.params.body["avatar"] → 返回 nil,永远拿不到文件
  • 正确入口:env.params.files["avatar"] → 返回 HTTP::FormData::File 实例
  • 必须确保 HTML 表单设置了 enctype="multipart/form-data",否则浏览器根本不会发送文件

HTTP::FormData::File 的关键属性和安全处理

拿到文件对象后,别急着 .save 或直接读取内容。它包含原始字节流,但不带路径校验、类型检查或大小限制 —— 直接使用有风险。

AI at Meta
AI at Meta

Facebook 旗下的AI研究平台

下载
  • file.filename:客户端提交的原始文件名(不可信,可能含 ../ 路径遍历
  • file.content_type:MIME 类型(如 "image/png"),可做基础白名单校验
  • file.tempfile:指向系统临时文件的 Tempfile 对象,已写入磁盘,可安全读取或移动
  • file.size:文件字节数,建议在接收前通过 env.request.headers["Content-Length"] 做前置拦截(防超大上传)
post "/upload" do |env|
  # ✅ 正确获取上传的文件
  avatar_file = env.params.files["avatar"]
  return env.response.status_code = 400 unless avatar_file

✅ 安全提取文件名(去路径、加哈希)

orig_name = File.basename(avatar_file.filename) safename = "#{SecureRandom.hex(8)}#{orig_name}"

✅ 白名单校验 MIME 类型

unless ["image/jpeg", "image/png", "image/gif"].includes?(avatar_file.content_type) env.response.status_code = 415 next "Unsupported file type" end

✅ 移动临时文件到 public/uploads/

upload_dir = File.join(Dir.current, "public", "uploads") Dir.mkdir_p(upload_dir) dest_path = File.join(upload_dir, safe_name) File.rename(avatar_file.tempfile.path, dest_path)

"Uploaded: #{safe_name}" end

常见错误现象与修复

上传失败时,通常不是代码报错,而是静默 400/404 或文件为空 —— 因为 Kemal 默认不启用 multipart 解析,或中间件顺序不对。

  • 现象env.params.files 总是空 Hash → 原因:没调用 require "kemal" 后的默认 multipart 中间件未生效;修复:确认没手动覆盖 Kemal.config.middleware,且没禁用 Kemal::Multipart
  • 现象:上传大文件时连接中断 → 原因:Kemal 默认无上传大小限制,但底层 HTTP::Server 有缓冲区上限;修复:启动前设置 Kemal.config.max_request_size = 20_000_000(单位字节)
  • 现象file.tempfile.path 报错 “No such file” → 原因:多次调用 file.tempfile 会导致临时文件被清理;修复:只调用一次并缓存结果,或直接用 file.io 流式读取

文件上传看似简单,但绕不开路径净化、类型校验、大小控制和临时文件生命周期管理 —— 这四点漏掉任一,上线后都可能变成安全漏洞或服务雪崩点。

相关专题

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

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

178

2024.05.11

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

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

212

2025.12.18

json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

412

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

533

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

309

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

74

2025.09.10

html版权符号
html版权符号

html版权符号是“©”,可以在html源文件中直接输入或者从word中复制粘贴过来,php中文网还为大家带来html的相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

613

2023.06.14

html在线编辑器
html在线编辑器

html在线编辑器是用于在线编辑的工具,编辑的内容是基于HTML的文档。它经常被应用于留言板留言、论坛发贴、Blog编写日志或等需要用户输入普通HTML的地方,是Web应用的常用模块之一。php中文网为大家带来了html在线编辑器的相关教程、以及相关文章等内容,供大家免费下载使用。

653

2023.06.21

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

65

2026.01.16

热门下载

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

精品课程

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

共46课时 | 2.9万人学习

AngularJS教程
AngularJS教程

共24课时 | 2.7万人学习

CSS教程
CSS教程

共754课时 | 20.1万人学习

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

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