前端读取文件夹并通过 SSH 上传:完整实现方案 ✅

在 Web 应用中,除了单文件上传,很多时候我们还需要用户直接选择整个文件夹,并批量上传到远程服务器。典型场景包括:静态资源部署、文档归档、远程备份等。本文整合了 前端文件夹选择方案(webkitdirectory + File System Access API)Node.js + node-ssh 后端上传,实现端到端的完整流程。


前端部分:选择文件夹并上传

前端的目标是让用户选择目录,遍历其中所有文件,并逐一上传到后端。

方案一:webkitdirectory

这是目前兼容度最好的方式,Chrome、Edge、Safari 均支持。

typescript 复制代码
<input type="file" id="folder" webkitdirectory multiple>
<script>
  document.getElementById('folder').addEventListener('change', async (event) => {
    const files = event.target.files;

    for (const file of files) {
      console.log(file.webkitRelativePath, file.name);

      const formData = new FormData();
      formData.append("file", file, file.webkitRelativePath);

      await fetch("http://localhost:3000/upload", {
        method: "POST",
        body: formData
      });
    }
  });
</script>

优点

  • ✅ 兼容度较好,支持主流浏览器。
  • ✅ 能保留相对路径,方便后端还原目录结构。

缺点

  • ❌ Firefox 支持度差。
  • ❌ 必须通过 <input> 触发,不够灵活。

方案二:File System Access API

这是现代浏览器支持的新特性,可以通过 JS 调用目录选择器。

xml 复制代码
<script>
async function readDir() {
  const dirHandle = await window.showDirectoryPicker();

  for await (const entry of dirHandle.values()) {
    if (entry.kind === "file") {
      const file = await entry.getFile();
      console.log(entry.name);

      const formData = new FormData();
      formData.append("file", file, entry.name);

      await fetch("http://localhost:3000/upload", {
        method: "POST",
        body: formData
      });
    }
  }
}
</script>

<button onclick="readDir()">选择文件夹并上传</button>

优点

  • ✅ 灵活,可结合按钮、拖拽交互。
  • ✅ 支持读写权限,可操作文件。

缺点

  • ❌ 兼容性差,仅 Chromium 内核浏览器支持(Chrome、Edge)。
  • ❌ 需要用户手动授权目录访问。

Node.js 后端:接收并通过 SSH 上传

后端主要任务:接收前端传输的文件 → 缓存到本地 → 通过 SSH 上传到远程服务器 → 删除缓存。

安装依赖

复制代码
npm install express multer node-ssh

完整后端代码

ini 复制代码
const express = require('express');
const multer = require('multer');
const { NodeSSH } = require('node-ssh');
const path = require('path');
const fs = require('fs');

const app = express();
const upload = multer({ dest: 'uploads/' });
const ssh = new NodeSSH();

// SSH 配置
const sshConfig = {
  host: 'your-server-ip',
  username: 'your-username',
  privateKey: '/path/to/private/key' // 或 password
};

// 接收文件上传
app.post('/upload', upload.single('file'), async (req, res) => {
  try {
    const localPath = req.file.path;
    const remotePath = path.join('/remote/dir', req.file.originalname);

    await ssh.connect(sshConfig);
    await ssh.putFile(localPath, remotePath);

    fs.unlinkSync(localPath); // 删除本地缓存文件

    res.send('上传成功');
  } catch (err) {
    console.error(err);
    res.status(500).send('上传失败');
  }
});

app.listen(3000, () => {
  console.log('服务已启动: http://localhost:3000');
});

工作流程

  1. 前端选择目录,逐个文件打包成 FormData
  2. 文件 POST 到后端 /upload
  3. 后端存储临时文件,再通过 node-ssh 传输到远程服务器。
  4. 删除临时文件,完成上传。

优缺点对比

前端

  • webkitdirectory:兼容更好,但交互有限。
  • File System Access API:交互灵活,但兼容性差。

后端

  • 采用 express + multer,简单易用。
  • 借助 node-ssh,实现自动化远程部署。

总结

  • 如果项目需要支持更多浏览器,推荐 webkitdirectory
  • 如果项目环境可控,推荐 File System Access API,更灵活。
  • 后端使用 Node.js + node-ssh,能快速实现从前端到服务器的自动化传输。

该方案特别适合 批量文件上传、静态资源部署、远程备份 等场景。

相关推荐
Aniugel9 分钟前
单点登录(SSO)系统
前端
鹏多多13 分钟前
移动端H5项目,还需要react-fastclick解决300ms点击延迟吗?
前端·javascript·react.js
serioyaoyao14 分钟前
上万级文件一起可视化,怎么办?答案是基于 ParaView 的远程可视化
前端
万少20 分钟前
端云一体 一天开发的元服务-奇趣故事匣经验分享
前端·ai编程·harmonyos
WindrunnerMax22 分钟前
从零实现富文本编辑器#11-Immutable状态维护与增量渲染
前端·架构·前端框架
不想秃头的程序员24 分钟前
Vue3 封装 Axios 实战:从基础到生产级,新手也能秒上手
前端·javascript·面试
数研小生38 分钟前
亚马逊商品列表API详解
前端·数据库·python·pandas
你听得到1139 分钟前
我彻底搞懂了 SSE,原来流式响应效果还能这么玩的?(附 JS/Dart 双端实战)
前端·面试·github
不倒翁玩偶40 分钟前
npm : 无法将“npm”项识别为 cmdlet、函数、脚本文件或可运行程序的名称。请检查名称的拼写,如果包括路径,请确保路径正确,然后再试一次。
前端·npm·node.js