Node.js 操作百度网盘实现文件上传(小文件上传,大文件分片上传)

news2025/2/28 15:25:47

Node.js 操作百度网盘实现文件上传(小文件上传,大文件分片上传)

前提准备:获取百度网盘的授权码

https://pan.baidu.com/union/doc/al0rwqzzl

const fs = require('fs');
const crypto = require('crypto');
const path = require('path');
const FormData = require('form-data');
const axios = require('axios');

const access_token = '需要自己创建应用获取授权码'

async function readFile(filePath) {
    return new Promise((resolve, reject) => {
        fs.readFile(filePath, (err, data) => {
            if (err) {
                reject(err);
            } else {
                resolve(data);
            }
        });
    });
}

function getFileInfo(filePath) {
    return new Promise((resolve, reject) => {
        fs.stat(filePath, (err, stats) => {
            if (err) {
                reject(err);
            } else {
                resolve({
                    size: stats.size,
                    isFile: stats.isFile(),
                    name: path.basename(filePath)
                });
            }
        });
    });
}

function getSlice(filePath, start, end) {
    return new Promise((resolve, reject) => {
        const sliceStream = fs.createReadStream(filePath, {
            start,
            end
        });
        let chunk = '';
        sliceStream.on('data', (data) => {
            chunk += data;
        });
        sliceStream.on('end', () => {
            resolve(chunk);
        });
        sliceStream.on('error', (err) => {
            reject(err);
        });
    });
}

async function getFileMd5List(filePath) {
    try {
        const fileInfo = await getFileInfo(filePath);
        const sliceSize = 4 * 1024 * 1024; // 4MB

        if (fileInfo.size <= sliceSize) {
            // 如果文件小于等于4MB,直接计算整个文件的MD5并返回
            const fileData = await readFile(filePath);
            const fileMd5 = crypto.createHash('md5').update(fileData).digest('hex');
            return JSON.stringify([fileMd5]);
        } else {
            // 如果文件大于4MB,分片计算MD5
            const sectionCount = Math.ceil(fileInfo.size / sliceSize);
            const md5List = [];

            for (let i = 0; i < sectionCount; i++) {
                const sliceData = await getSlice(filePath, i * sliceSize, (i + 1) * sliceSize - 1);
                const sliceMd5 = crypto.createHash('md5').update(sliceData).digest('hex');
                md5List.push(sliceMd5);
            }
            return JSON.stringify(md5List);
        }
    } catch (error) {
        console.error('Error in getFileMd5List:', error);
        return null;
    }
}

async function uploadFileToBaidu(filePath, accessToken) {
    try {
        const fileInfo = await getFileInfo(filePath);
        let md5List = await getFileMd5List(filePath);
        console.log(md5List)
        const precreateRes = await axios.post('https://pan.baidu.com/rest/2.0/xpan/file', {
            path: '/apps/后台管理系统/' + fileInfo.name,
            size: fileInfo.size,
            isdir: fileInfo.isFile ? 0 : 1,
            rtype: 1,
            autoinit: 1,
            block_list: md5List
        }, {
            params: {
                method: 'precreate',
                access_token: accessToken
            },
            headers: {
                'Content-Type': 'application/x-www-form-urlencoded'
            }
        });
        const uploadId = precreateRes.data.uploadid;
        if (fileInfo.isFile && fileInfo.size <= 4 * 1024 * 1024) {
            // 小文件上传
            const form = new FormData();
            form.append('file', fs.createReadStream(filePath));
            const uploadSingle = await axios.post('https://d.pcs.baidu.com/rest/2.0/pcs/file', form, {
                    params: {
                        method: 'upload',
                        access_token: accessToken,
                        path: '/apps/后台管理系统/' + fileInfo.name,
                        ondup: 'overwrite'
                    },
                    headers: {
                        ...form.getHeaders(),
                    }
                })
            ;
            return;
        } else {
            const sliceSize = 4 * 1024 * 1024;
            const sectionCount = Math.ceil(fileInfo.size / sliceSize);
            md5List = []
            for (let i = 0; i < sectionCount; i++) {
                const form = new FormData();
                form.append('file', await getSlice(filePath, i * sliceSize, (i + 1) * sliceSize - 1));
                if (i === sectionCount - 1) {
                    form.append('file', await getSlice(filePath, i * sliceSize, fileInfo.size - 1));
                }
                const sectionItem = await axios.put('https://d.pcs.baidu.com/rest/2.0/pcs/superfile2', form, {
                    params: {
                        method: 'upload',
                        access_token: accessToken,
                        type: 'tmpfile',
                        path: '/apps/后台管理系统/' + fileInfo.name,
                        uploadid: uploadId,
                        partseq: i
                    },
                    headers: {
                        ...form.getHeaders(),
                    }
                });
                md5List.push(sectionItem.data.md5)
            }
            console.log(md5List)
            const uploadArray = await axios.post('https://pan.baidu.com/rest/2.0/xpan/file', {
                path: '/apps/后台管理系统/' + fileInfo.name,
                size: fileInfo.size,
                isdir: fileInfo.isFile ? 0 : 1,
                uploadid: uploadId,
                block_list: JSON.stringify(md5List)
            }, {
                params: {
                    method: 'create',
                    access_token: accessToken
                },
                headers: {
                    'User-Agent': 'pan.baidu.com',
                    'Content-Type': 'application/x-www-form-urlencoded',
                },
            });
            console.log(uploadArray.data)
        }
    } catch (e) {
        console.log('error', e)
    }
}

// uploadFileToBaidu(path.resolve(__dirname, './WeChatProjects.zip'), access_token)
// uploadFileToBaidu(path.resolve(__dirname, './auth.js'), access_token)

在这里插入图片描述
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/990119.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

解某麦数据请求参数analysis加密

意外发现一个可以查询app下载量得网站&#xff0c; 想筛选一下哪些下载量在1w-10w之间&#xff0c;大概需要5k个.。 感觉应该没啥加密&#xff0c;好把&#xff0c;是我小看了&#xff0c;有个参数是加密得&#xff0c;如图。 analysis 扣js开始&#xff0c; f12 去资源文件…

AP5125 外置MOS LED降压恒流驱动器 过EMC认证 车灯驱动线路图

产品描述 AP5125 是一款外围电路简单的 Buck 型平均电流检测模式的 LED 恒流驱动器&#xff0c;适用于 8-100V 电压范围的非隔离式大功率恒流 LED 驱动领域。芯片采用固定频率 140kHz 的 PWM 工作模式&#xff0c; 利用平均电流检测模式&#xff0c;因此具有优异的负载调整 率…

算法通关村第十八关:青铜挑战-回溯是怎么回事

青铜挑战-回溯是怎么回事 回溯&#xff0c;最重要的算法之一 主要解决一些暴力枚举也搞不定的问题&#xff0c;例如组合、分割、子集、排列、棋盘等 从性能角度来看回溯算法的效率并不高&#xff0c;但对于这些暴力都搞不定的算法能出结果就很好了&#xff0c;效率低点没关系…

Unity入门教程||创建项目(上)

一、介绍 目的&#xff1a;通过尝试制作一款使用玩家角色把小球弹飞的简单小游戏&#xff0c;熟悉使用Unity进行游戏开发的基本流程。 软件环境&#xff1a;Unity 2017.3.0f3&#xff0c;Visual Studio 2013 二、创建新项目 1&#xff0c;启动Unity后将出现一个并列显示Pro…

喜报丨迪捷软件入选浙江省2023年省级产业数字化服务商

近日&#xff0c;根据《关于组织开展2023年度省级产业数字化服务商申报工作的通知》要求&#xff0c;省经信厅公布2023年省级产业数字化服务商名单&#xff0c;浙江迪捷软件科技有限公司榜上有名。 省级产业数字化服务商上榜名单的评选在企业申报、地方推荐、专家评审、综合评估…

OpenCV_CUDA_VS编译安装

一、OpenCV 我这里是下载的OpenCV4.5.4&#xff0c;但是不知道到在vs里面build时一直报错&#xff0c;后面换了4.7.0的版本测试&#xff0c;安装成功。 Release OpenCV 4.5.4 opencv/opencv GitHub 这个里面有官方预编译好的OpenCV库&#xff0c;可以直接食用。 扩展包&am…

分享一款月销五千万的即拼七人拼团系统开发模式!

社交裂变能带来巨大流量是众所周知的&#xff0c;下面就给大家分享一款月销五千万的商业模式——即拼七人拼团。这款模式可以做到用互联网思维引流&#xff0c;让终端用户自主裂变新用户&#xff0c;实现团队持续长久发展。 即拼七人拼团模式的玩法很简单&#xff1a; 用户可以…

科技云报道:生成式AI已成为企业新兴风险,但我们不应该因噎废食

科技云报道原创。 2023年&#xff0c;生成式AI技术破茧成蝶&#xff0c;引发了一场全球范围的数字革命。 从最初的聊天、下棋开始&#xff0c;到医疗、金融、制造、教育、科研等&#xff0c;生成式AI表现出了强大的创造力和无限潜力。据不完全统计&#xff0c;截至今年8月底&…

首款带屏幕3000MbpsFTTR全光网关产品,中兴通讯推出全屋光纤组网

中兴通讯推出全新产品RoomPON 5.0&#xff0c;这是市场上首款带屏幕的3000Mbps FTTR系列产品。据官方消息&#xff0c;该产品于九月八日在深圳光博会期间发布&#xff0c;备受期待的RoomPON 5.0全光系列产品拥有以下技术亮点&#xff1a; 超高接入&#xff0c;超广覆盖&#xf…

Talk | ICCV‘23清华大学博士生诸子钰:3D-VisTA通用统一的3D视觉语言预训练模型

​​​​​​ 本期为TechBeat人工智能社区第529期线上Talk&#xff01; 北京时间9月7日(周四)20:00&#xff0c; 清华大学博士生—诸子钰的Talk已准时在TechBeat人工智能社区开播&#xff01; 他与大家分享的主题是: “3D-VisTA通用统一的3D视觉语言预训练模型”&#xff0c;他…

FreeBASIC通过Delphi7 DLL调用MS SOAP使用VB6 Webservice

前几篇笔记习练了IIS soapis30配置、VB6 webservice创建、Delphi7和VB6 webservice访问&#xff1a; VB6 COM webservice发布&#xff0c;VB.NET和Delphi 7 对webservice访问&#xff0c;及MS Soap Toolkit 3.0在IIS上的ISAPI配置_Mongnewer的博客-CSDN博客 本篇笔记重点编写…

IIS短文件名泄露漏洞复现

IIS短文件名泄露漏洞复现 前言一、漏洞描述二、漏洞原理1.什么是短文件2.短文件特征 三、漏洞验证三、漏洞防御总结 前言 IIS短文件名泄露漏洞比较老了&#xff0c;而且只适合于windowsiisasp的网络结构&#xff0c;所有如下的复现步骤看下就行了&#xff0c;关键是要弄懂原理…

钉钉(自建应用)无需代码连接畅捷通T+Cloud的方法

1 使用场景 企业日常工作中&#xff0c;经常会涉及到各种各样的订单审批流程&#xff0c;为了提高工作效率&#xff0c;大多数企业内部会选择畅捷通TCloud作为财务ERP系统&#xff0c;钉钉作为OA审批系统。为了保证流程的顺畅和高效&#xff0c;需要将畅捷通TCloud创建的销售单…

opencv基础: 视频,摄像头读取与保存的常用方法

当然还可以从视频中抓取截图&#xff0c;所以现在聊一下常用的抓取视频截图的的方法。 VideoCapture 方法 cv2.VideoCapture();cv2.VideoCapture( device);cv2.VideoCapture(filename);上面有三种构造方法&#xff0c; 第一种是无法构造方法。 第二种参数device是一个数字。 …

F#奇妙游(30):计算表达式与ADT

Computation Expression More F#中自定义的 Computation Expression 一共有8个语法构造&#xff0c;其中match!是let!的语法糖。 在前面的一个帖子里CE初探我们已经介绍了 computation expression 中的绑定和返回&#xff0c;语法是let!和return。通过使用这两个语法&#xff…

浅谈 Spring AOP

AOP&#xff08;Aspect Oriented Programming&#xff09;&#xff1a;⾯向切⾯编程&#xff0c;它是⼀种思想&#xff0c;它是对某⼀类事情的集中处理。⽐如⽤户登录权限的效验&#xff0c;没学 AOP 之前&#xff0c;我们所有需要判断⽤户登录的⻚⾯&#xff08;中的⽅法&…

ipad必须要配原装的笔么?电容笔性价比高的品牌

众所周知&#xff0c;由于Apple pencil的出现&#xff0c;现在网上越来越多平替触控笔的出现&#xff0c;无论是价格和功能&#xff0c;几乎都很接近。很多小伙伴不知如何下手&#xff0c;不知道如何从众多品牌中挑选出适合自己的&#xff0c;今天我为大家总结几款好用平价电容…

Baklib:2023年企业知识库的新最好选择!

传统的企业知识管理方式主要是通过文件档案、会议记录、员工手册等方式来进行知识管理。这种方式的缺点是效率低下&#xff0c;信息不够及时、准确、全面&#xff0c;而且很难达到知识共享的效果。随着信息技术的发展&#xff0c;现代化的企业知识管理方式越来越受到企业的青睐…

骨传导耳机佩戴舒适吗?盘点骨传导耳机舒适度比较好的几款耳机!

相信很多年轻人和我一样&#xff0c;佩戴耳机成了日常的习惯&#xff0c;蓝牙耳机已经融入了我们的日常生活和工作。但长期戴耳机也有很多的问题存在&#xff0c;比如长时间佩戴导致耳道疼痛、甚至头痛&#xff0c;或是耳机隔音效果太好&#xff0c;导致错过身边的重要信息&…

记录aardio和Pythonl联动,为python做界面、做单exe文件的几个知识点

关于aardio的几个目录的说明 如果要aardio创建python关联程序,强烈建议新建工程向导中选择“窗口程序”+python来生成,会自动建立好目录,十分方便。 写好的python脚本或工程,直接放到py目录中,aardio中可以直接import导入该目录中的python模块,无需带目录,类似: aar…