写点什么

Python 开发篇——基于 React-Dropzone 开发上传组件

用户头像
DisonTangor
关注
发布于: 2 小时前

这次我要讲述的是在 React-Flask 框架上开发上传组件的技巧。我目前主要以 React 开发前端,在这个过程中认识到了许多有趣的前端 UI 框架——React-BootstrapAnt DesignMaterial UIBulma等。而比较流行的上传组件也不少,而目前用户比较多的是 jQuery-File-UploadDropzone,而成长速度快的新晋有Uppyfilepond。比较惋惜的是Fine-Uploader的作者自 2018 年后就决定不再维护了,原因作为后来者的我就不多过问了,但请各位尊重每一位开源作者的劳动成果。


这里我选择React-Dropzone,原因如下:


  1. 基于 React 开发,契合度高

  2. 网上推荐度高,连 Material UI 都用他开发上传组件

  3. 主要以 DragDrop 为主,但是对于传输逻辑可以由开发者自行设计。例如尝试用 socket-io 来传输 file chunks。对于 node 全栈估计可行,但是我这里使用的是 Flask,需要将 Blob 转 ArrayBuffer。但是如何将其在 Python 中读写,我就没进行下去了。

实例演示

1. axios 上传普通文件:

通过 yarn 将 react-dropzone 和引入:


yarn add react-dropzone axios


前端 js 如下(如有缺失,请自行修改):


import React, {     useState,     useCallback,    useEffect,} from 'react';import {useDropzone} from 'react-dropzone';import "./dropzone.styles.css"import InfiniteScroll from 'react-infinite-scroller';import {    List,    message,    // Avatar,    Spin,} from 'antd';import axios from 'axios';
/*** 计算文件大小* @param {*} bytes * @param {*} decimals * @returns */function formatBytes(bytes, decimals = 2) { if (bytes === 0) return '0 Bytes';
const k = 1024; const dm = decimals < 0 ? 0 : decimals; const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i];}
/*** Dropzone 上传文件* @param {*} props * @returns */function DropzoneUpload(props) { const [files, setFiles] = useState([]) const [loading, setLoading] = useState(false); const [hasMore, setHasMore] = useState(true);
const onDrop = useCallback(acceptedFiles => { setLoading(true); const formData = new FormData(); smallFiles.forEach(file => { formData.append("files", file); }); axios({ method: 'POST', url: '/api/files/multiplefiles', data: formData, headers: { "Content-Type": "multipart/form-data", } }) then(resp => { addFiles(acceptedFiles); setLoading(false); }); }, [files]);
// Dropzone setting const { getRootProps, getInputProps } = useDropzone({ multiple:true, onDrop, });
// 删除附件 const removeFile = file => { const newFiles = [...files] newFiles.splice(newFiles.indexOf(file), 1) setFiles(newFiles) }
useEffect(() => { // init uploader files setFiles([]) },[])
return ( <section className="container"> <div {...getRootProps({className: 'dropzone'})}> <input {...getInputProps()} /> <p>拖动文件或点击选择文件😊</p> </div> <div className="demo-infinite-container"> <InfiniteScroll initialLoad={false} pageStart={0} loadMore={handleInfiniteOnLoad} hasMore={!loading && hasMore} useWindow= {false} > <List dataSource={files} renderItem={item=> ( <List.Item actions={[ // <a key="list-loadmore-edit">编辑</a>, <a key="list-loadmore-delete" onClick={removeFile}>删除</a> ]} // extra={ // } key={item.path}> <List.Item.Meta avatar={ <> { !!item.type && ['image/gif', 'image/jpeg', 'image/png'].includes(item.type) && <img width={100} alt='logo' src={item.preview} /> } </> } title={item.path} description={formatBytes(item.size)} /> </List.Item> )} > {loading && hasMore && ( <div className="demo-loading-container"> <Spin /> </div> )} </List> </InfiniteScroll> </div> </section> );}
复制代码


flask 代码:


def multiplefiles():if 'files' not in request.files:    return jsonify({'message': '没有文件!'}), 200files = request.files.getlist('files')
for file in files: if file: # 通过拼音解决secure_filename中文问题 filename = secure_filename(''.join(lazy_pinyin(file.filename)) Path(UPLOAD_FOLDER + '/' + file_info['dir_path']).mkdir(parents=True, exist_ok=True) file.save(os.path.join(UPLOAD_FOLDER + '/' + file_info['dir_path'], filename))
return jsonify({'message': '保存成功!!'})
复制代码

2. 大文件导入:

通过 file.slice()方法生成文件的 chunks。不要用 Promise.all 容易产生非顺序型的请求,导致文件损坏。


js 代码:


const promiseArray = largeFiles.map(file => new Promise((resolve, reject) => {                            const chunkSize = CHUNK_SIZE;    const chunks = Math.ceil(file.size / chunkSize);    let chunk = 0;    let chunkArray = new Array();    while (chunk <= chunks) {        let offset = chunk * chunkSize;        let slice = file.slice(offset, offset+chunkSize)        chunkArray.push([slice, offset])        ++chunk;    }    const chunkUploadPromises = (slice, offset) => {        const largeFileData = new FormData();        largeFileData.append('largeFileData', slice)        return new Promise((resolve, reject) => {            axios({                method: 'POST',                url: '/api/files/largefile',                data: largeFileData,                headers: {                    "Content-Type": "multipart/form-data"                }            })            .then(resp => {                console.log(resp);                resolve(resp);            })            .catch(err => {                reject(err);            })        })    };
chunkArray.reduce( (previousPromise, [nextChunk, nextOffset]) => { return previousPromise.then(() => { return chunkUploadPromises(nextChunk, nextOffset); }); }, Promise.resolve()); resolve();}))
复制代码


flask 代码:


filename = secure_filename(''.join(lazy_pinyin(filename)))Path(UPLOAD_FOLDER + '/' + file_info['dir_path']).mkdir(parents=True, exist_ok=True)save_path = os.path.join(UPLOAD_FOLDER + '/' + file_info['dir_path'], filename)try:    with open(save_path, 'ab') as f:        f.seek(offset)        f.write(file.stream.read())        print("time: "+ str(datetime.now())+" offset: " + str(offset))except  OSError:    return jsonify({'Could not write to file'}), 500
复制代码

结语

文件传输一直都是 HTTP 的痛点,尤其是大文件传输。最好的方式是自己做个 Client,通过 FTP 和 FTPS 的协议进行传输。第二种来自于大厂很中心化的方法,通过文件的 checksum 来确定文件是否已经上传了,来营造秒传的效果。第三种来自去中心化的 Bittorrent 的方法每一个用户做文件种子,提供文件传输的辅助,目前国内并没有普及使用。

发布于: 2 小时前阅读数: 2
用户头像

DisonTangor

关注

怀揣一个武侠梦的男孩 2020.07.29 加入

还未添加个人简介

评论

发布
暂无评论
Python开发篇——基于React-Dropzone开发上传组件