当前位置: 首页 > news >正文

前端安全直传MinIO方案

目的:前端直接上传文件到Minio,不通过服务器中转文件。密钥不能在前端明文传输。

## 一、架构设计

```mermaid

sequenceDiagram

    前端->>后端: 1.请求上传凭证

    后端->>MinIO: 2.生成预签名URL

    后端-->>前端: 3.返回预签名URL

    前端->>MinIO: 4.使用URL直传文件

    MinIO-->>前端: 5.返回上传结果

```

---

## 二、服务端实现

### 1. 环境配置

```bash

# 安装依赖

npm install minio express dotenv cors

```

### 2. 安全凭证管理

```javascript

// .env文件

MINIO_ENDPOINT=your-minio.example.com

MINIO_PORT=9000

MINIO_USE_SSL=true

MINIO_ACCESS_KEY=your_access_key

MINIO_SECRET_KEY=your_secret_key

```

### 3. 预签名URL生成接口

```javascript

const Minio = require('minio')

const express = require('express')

const app = express()

require('dotenv').config()

const minioClient = new Minio.Client({

  endPoint: process.env.MINIO_ENDPOINT,

  port: parseInt(process.env.MINIO_PORT),

  useSSL: process.env.MINIO_USE_SSL === 'true',

  accessKey: process.env.MINIO_ACCESS_KEY,

  secretKey: process.env.MINIO_SECRET_KEY

})

// 生成预签名上传URL

app.get('/api/upload-token', (req, res) => {

  const fileId = `${Date.now()}-${Math.random().toString(36).substr(2, 9)}`

  const objectName = `uploads/${fileId}/${req.query.fileName}`

  const expiry = 15 * 60 // 15分钟有效期

  minioClient.presignedPutObject(

    process.env.MINIO_BUCKET,

    objectName,

    expiry,

    (err, presignedUrl) => {

      if (err) return res.status(500).json({ error: err.message })

      res.json({

        presignedUrl,

        objectUrl: `https://${process.env.MINIO_ENDPOINT}/${process.env.MINIO_BUCKET}/${objectName}`

      })

    }

  )

})

app.listen(3000, () => console.log('Server running on port 3000'))

```

---

## 三、前端实现

### 1. 获取上传凭证

```javascript

async function getUploadToken(fileName) {

  const response = await fetch(`/api/upload-token?fileName=${encodeURIComponent(fileName)}`)

  return response.json()

}

```

### 2. 文件上传组件

```html

<input type="file" id="fileInput" />

<button οnclick="uploadFile()">上传</button>

<script>

async function uploadFile() {

  const fileInput = document.getElementById('fileInput')

  const file = fileInput.files[0]

  // 获取上传凭证

  const { presignedUrl, objectUrl } = await getUploadToken(file.name)

  // 直传MinIO

  const result = await fetch(presignedUrl, {

    method: 'PUT',

    body: file,

    headers: {

      'Content-Type': file.type

    }

  })

  if (result.ok) {

    console.log('文件地址:', objectUrl)

  } else {

    console.error('上传失败')

  }

}

</script>

```

---

## 四、安全增强措施

### 1. MinIO存储桶策略

```json

{

  "Version": "2012-10-17",

  "Statement": [

    {

      "Effect": "Deny",

      "Principal": "*",

      "Action": "s3:*",

      "Resource": "arn:aws:s3:::your-bucket/*",

      "Condition": {

        "NumericGreaterThan": {"s3:signatureAge": "900"}

      }

    },

    {

      "Effect": "Allow",

      "Principal": "*",

      "Action": "s3:PutObject",

      "Resource": "arn:aws:s3:::your-bucket/uploads/*"

    }

  ]

}

```

### 2. 服务端安全中间件

```javascript

// 添加CORS限制

const cors = require('cors')

app.use(cors({

  origin: ['https://your-domain.com'],

  methods: ['GET', 'POST'],

  maxAge: 300

}))

// 请求频率限制

const rateLimit = require('express-rate-limit')

const limiter = rateLimit({

  windowMs: 15 * 60 * 1000,

  max: 100

})

app.use('/api/upload-token', limiter)

```

---

## 五、大文件分片上传

### 1. 前端分片处理

```javascript

async function uploadLargeFile(file) {

  const CHUNK_SIZE = 5 * 1024 * 1024 // 5MB

  const chunks = Math.ceil(file.size / CHUNK_SIZE)

  // 初始化分片上传

  const { uploadId } = await initMultipartUpload(file.name)

  // 并发上传分片

  const uploadPromises = []

  for (let i = 0; i < chunks; i++) {

    const chunk = file.slice(i * CHUNK_SIZE, (i + 1) * CHUNK_SIZE)

    uploadPromises.push(uploadChunk(uploadId, i + 1, chunk))

  }

  await Promise.all(uploadPromises)

  // 完成上传

  return completeUpload(uploadId)

}

```

### 2. 分片上传服务端支持

```javascript

// 初始化分片上传

app.post('/api/multipart-init', (req, res) => {

  const objectName = `uploads/${uuidv4()}/${req.query.fileName}`

  minioClient.initiateNewMultipartUpload(

    process.env.MINIO_BUCKET,

    objectName,

    (err, uploadId) => {

      if (err) return res.status(500).send(err)

      res.json({ uploadId, objectName })

    }

  )

})

// 获取分片上传URL

app.get('/api/multipart-url', (req, res) => {

  const presignedUrl = minioClient.presignedPutObject(

    process.env.MINIO_BUCKET,

    req.query.objectName,

    15 * 60,

    { partNumber: req.query.partNumber, uploadId: req.query.uploadId }

  )

  res.json({ presignedUrl })

})

```

---

## 六、监控与日志

### 1. MinIO操作日志

```bash

# 启用访问日志

mc admin config set myminio audit_webhook endpoint=http://log-server:3000/logs

```

### 2. 服务端监控指标

```javascript

const prometheus = require('prom-client')

const uploadCounter = new prometheus.Counter({

  name: 'file_uploads_total',

  help: 'Total number of file uploads',

  labelNames: ['status']

})

app.post('/api/upload', (req, res) => {

  uploadCounter.inc({ status: 'started' })

  // ...上传逻辑

})

```

---

## 七、方案优势

1. **零密钥暴露**:前端仅使用临时预签名URL

2. **高安全性**:15分钟有效期+IP限制+HTTPS

3. **高性能**:支持5GB+大文件分片上传

4. **可扩展**:轻松集成CDN和自动转码

5. **合规性**:满足GDPR和等保要求

---

## 八、部署注意事项

1. 启用MinIO的HTTPS访问

2. 定期轮换MinIO根密钥

3. 设置存储桶生命周期策略

4. 监控存储桶使用情况

5. 启用防病毒扫描

6. 配置自动告警规则

```bash

# MinIO客户端配置示例

mc alias set myminio https://minio.example.com your_access_key your_secret_key

mc mb myminio/secure-uploads

mc lifecycle set myminio/secure-uploads lifecycle.json

```

---

通过该方案,前端可直接安全上传文件到MinIO,无需在后端保存中转文件,同时确保敏感凭证不会暴露在前端代码中。实际部署时建议结合具体业务需求调整安全策略和性能参数。


文章转载自:

http://DicETjcr.rnytd.cn
http://jI6ampwW.rnytd.cn
http://Hh2jGQU6.rnytd.cn
http://Zioz8kRY.rnytd.cn
http://NDFtf3LS.rnytd.cn
http://H8HwIdzg.rnytd.cn
http://0tHtmsKf.rnytd.cn
http://gYm4zjxu.rnytd.cn
http://84jPTz1Y.rnytd.cn
http://Ebd98H7Q.rnytd.cn
http://by88SJCr.rnytd.cn
http://5F7pq4O1.rnytd.cn
http://MLVn3oF4.rnytd.cn
http://FJIKKtgh.rnytd.cn
http://FZA8oobQ.rnytd.cn
http://MW0Kn8gd.rnytd.cn
http://YroltHSa.rnytd.cn
http://Q0wdPNGG.rnytd.cn
http://87ftzOcF.rnytd.cn
http://t8fvTlDK.rnytd.cn
http://NRIeHVPC.rnytd.cn
http://fRGlJ4kW.rnytd.cn
http://laifQg5X.rnytd.cn
http://PQqgWN2S.rnytd.cn
http://axNVwxL9.rnytd.cn
http://42COiuPX.rnytd.cn
http://6NKYAEeb.rnytd.cn
http://l41U5Rkp.rnytd.cn
http://hGuTxvq0.rnytd.cn
http://eE2BR00o.rnytd.cn
http://www.dtcms.com/a/214254.html

相关文章:

  • Spring Cloud Gateway 限流实践:基于 Redis 令牌桶算法的网关层流量治理
  • Visual Studio 调试中 PDB 与图像不匹配
  • springcloud---gateway
  • [攻防世界] easyphp writeup
  • 北京大学肖臻老师《区块链技术与应用》公开课:02-BTC-密码学原理
  • 【React】- React-RND 深度使用指南:实现自由拖拽、避坑受控陷阱!
  • Java—— 多线程 第一期
  • cursor/vscode连接低版本的系统(glibc<2.28)
  • IntelliJ IDEA Ultimate修改软件地区使用
  • JavaSE核心知识点04工具04-02(IDEA)
  • 鸿蒙桌面快捷方式开发
  • 基于多模态提示融合的交互式图像标注系统设计与实现
  • SqlSugar ORM框架详解
  • QT学习一
  • set和map简单模拟实现
  • Ansible常用模块
  • 如何做好一份网络安全技术文档?
  • java中的线程安全的集合
  • 航空航天领域对滚珠丝杆的精度要求有多高?
  • 汉诺集团CDN+富氢水机全球发布:科技赋能健康,革新饮水革命
  • Java大师成长计划之第31天:Docker与Java应用容器化
  • 消防营区管理升级:豪越科技智能仓储与装备管理的力量
  • 解锁webpack:处理跨域devserver、摇树treeshaking、图片压缩sharp
  • xhr、fetch和axios
  • 第Y1周打卡——调用官方权重进行检测
  • O2OA服务器配置与管理-自定义消息提醒
  • 【Java学习笔记】单例设计模式
  • Netty应用:从零搭建Java游戏服务器网络框架
  • 进程通信(管道,共享内存实现)
  • Java设计模式之责任链模式:从基础到高级的全面解析