关于 koa-formidable 一些问题

这两天在研究node + koa + koa-formiable 上传图片和文件

最简单的使用方法

const Koa = require('koa')
const app = new Koa()
const formidable = require('koa-formidable')

app.use(formidable({
     uploadDir: config.uploadDir, // 上传目录
     keepExtensions: true,  // 保持原有后缀名
 }))


然后在文件中使用

let filePath = ctx.request.files.file.path;
let net_filePath = '/public/files/' + path.parse(filePath).base;

文件已经上传到指定的目录,非常方便。 但是我们不可能所有文件都上传到一个文件夹吧。 这时候需要按需自定义上传路径

但是koa-formable 的文档说明少之又少,没办法只能硬着头皮研究, 网上找到一个例子


    // 这个是一个异步,在路由里面前面是有async的
const upkoafile = async (ctx, next) => {
    var fs = require('fs');
    var formidable = require('koa-formidable'); // 这里用koa下的formidable模块,如果直接套用会出现关于req.on 的报错
    // 直接new formidable.IncomingForm() 也会报错它不能new,或者说下面这句已经帮我们new好了
    var form = formidable.parse(ctx.request);
    /*
    这里的form是一个函数,具体如下
    function (done) {
        var form = opts instanceof formidable.IncomingForm
            ? opts
            : new formidable.IncomingForm(opts)
        form.parse(ctx.req, function (err, fields, files) {
            if (err) return done(err)
            done(null, { fields: fields, files: files })
        })
    }
    看到这里,我想有人应该明白了,koa-formidable只不过在原来的模块基础上加了一点东西,甚至可以不用这个模块
    var formidable = require('formidable');
    var form = new formidable.IncomingForm();
    form.parse(ctx.req,function(){...})
    */
    var p = new Promise((resolve, reject) => {
        form((opt, obj) => {
            var file = obj.files.file;
            var filename = file.name;
            var buffer = fs.readFileSync(file.path); // 读取文件,此时就可以上传了
            //... 到这里就已经回到之前的步骤了,后面的http只需要在合适的地方返回promise就行了
            var opts = {
                //...上面一样
            }
            http.request(opts, function (resp) {
                resp.setEncoding('utf8');
                resp.on('data', function (chunk) {
                    body += chunk;   // 这个是上传的服务器返回的结果
                })
                resp.on('end', function () {
                    try {
                        var result = JSON.parse(body);
                        return resolve(result); //上传结束客户端返回
                    } catch (e) {
                        return reject(e);
                    }
                })
            }).on('error', function (e) {
                return reject(e); //出错后返回
            }).write(buffer)   // http上次
                .end();
        })
    })
    var body = await p; //上面的promise返回
    return ctx.body = body;
}

看到这里思路清晰了很多,但是我觉得我的需求不需要用到 像上面代码中的 http 模块,
于是我尝试直接使用 formidable 模块,

    var formidable = require('formidable')
    var form = new formidable.IncomingForm()
    form.uploadDir = uploadPath
    form.keepExtensions = true

    form.parse(ctx.req, function (err, fields, files) {
        console.log('files', files)

    })

这时出现恶心的找不到文件路劲


image.png

但是如果配置修改了的话:


    var formidable = require('formidable')
    var form = new formidable.IncomingForm()
    // form.uploadDir = uploadPath
    // form.keepExtensions = true

    form.parse(ctx.req, function (err, fields, files) {
        console.log('files', files)

    })

文件就会自动上传到一个莫名其妙的目录

image.png

结合上面两个情况想了很久,发现一个规律。 有可能是我指定的上传的目录文件没有新建。 所以就会报没有找到路径的错误, 但如果没有设置 uploadDir 属性,文件就上传到默认的路径了。 最后用fs模块, 每次上传图片之后就新建目录,然后问题就解决了

    fs.mkdir(uploadPath, function(err) {
        if(err) {
            return console.log(err)
        }
        console.log('创建文件夹成功')
    })

接下来就是异步问题了, 这里详细记录我的思考过程,日后反思。

参考文章 https://blog.csdn.net/u011782108/article/details/68062603
但是连接的作者并没有上传设置路径,所以我跟他的还不太一样

操作步骤

exports.addCover = async (ctx) => {

    let uploadPath = `${config.uploadDir}/public/files/server/coverimg/${moment().format('YYYYMMDD')}`
    let timeStamp = `${moment().format('YYYY-MM-DD')} ${moment().hours()}:${moment().minutes()}:${moment().seconds()}`

    function isExistFile() {
        return new Promise(function(resolve, reject) {
            fs.exists(uploadPath, function (exists) {
                if (exists) return resolve(true)

                if (!exists) {
                    fs.mkdir(uploadPath, function (err) {
                        if (err) {
                            return reject(false)
                        }
                        resolve(true)
                    })
                }
            })
        })
    }

    // function createFile() {
    //     return new Promise(function(resolve, reject) {
    //         console.log('uploadPath', uploadPath)
    //         fs.mkdir(uploadPath, function (err) {
    //             if (err) {
    //                 reject(err)
    //             }
    //             resolve()
    //         })
    //     }).catch(error => console.log('caught2', error))
    // }

    function cb(err, fields, files) {
        return new Promise((resolve, reject) => {
        
            console.log('files', files)
            let file = files.file;
            let filename = file.name;
            resolve(filename)
        })
    }

    function opeFile() {
        return new Promise(function(resolve, reject) {
            let form = new formidable.IncomingForm()
            form.uploadDir = uploadPath  //上传路径
            form.keepExtensions = true   //是否保持拓展名
            form.parse(ctx.req, function (err, fields, files) {
                // resolve('sssss')
                // if(err) return reject(err)

                // console.log('ssssssss')
                // let file = files.file
                // filePath = file.path
                // fileName = file.name
                // console.log(fileName)
                // console.log(filePath)
                // resolve({
                //     code: '001',
                //     msg: '上传成功',
                //     filePath,
                //     fileName
                // })
            })
        })

    }

    // 如果promise函数没有添加 .catch 捕捉错误,会弹出提示
    // (node:42811) UnhandledPromiseRejectionWarning: undefined
    // (node: 42811) UnhandledPromiseRejectionWarning: Unhandled promise rejection.This error originated either by throwing inside of an async function without a catch block, or by rejecting a promise which was not handled with .catch(). (rejection id: 1)
    // (node: 42811)[DEP0018] DeprecationWarning: Unhandled promise rejections are deprecated.In the future, promise rejections that are not handled will terminate the Node.js process with a non - zero exit code.
    
    async function handleAsync() {
        try {
            const isExist = await isExistFile()
            console.log('isExist', isExist)


            return isExist
            // if (isExist) {

            //     let form = new formidable.IncomingForm()
            //     form.uploadDir = uploadPath  //上传路径
            //     form.keepExtensions = true   //是否保持拓展名

            //     console.log('form', form)
            //     // console.log('uploadPath', uploadPath)
            //     // const v = form.parse(ctx.req, function (err, fields, files) {
            //     //     return 'sssss'
            //     // })

            //     // const v = await cb(formCb)
            //     // console.log('v', v)            
            // } 

        } catch (e) {
            console.log(e)
        }
    }
    const p = await handleAsync()
    
    console.log('p', p)
    ctx.body = p




    // let result = checkFs.then(function(data) {

    //     return opeFile

    // })
    
    // let { filePath, fileName } = result
    // await user_db.sqlQuery('INSERT INTO server_article_addcover (filepath, filename, created_at) values (?,?,?)', [filePath, fileName, timeStamp])
    // ctx.body = result

    // 将数据保存进数据库
    // let result = await user_db.sqlQuery('INSERT INTO server_article_addcover (filepath, filename, created_at) values (?,?,?)', [filePath, fileName, timeStamp])

    // return ctx.body = {
    //     code: '001',
    //     msg: '上传成功',
    //     filepath: filePath.replace(/^\/public/, ''),
    //     filename: fileName
    // }



    // fs.exists(uploadPath, function(exists) {
    //     if (!exists) {
    //         fs.mkdir(uploadPath, function (err) {
    //             if (err) {
    //                 return console.log(err)
    //             }
    //             console.log('创建文件夹成功')
    //         })
    //     }




    //     const formidable = require('formidable')
    //     let filePath, fileName
    //     let form = new formidable.IncomingForm()
    //     form.uploadDir = uploadPath  //上传路径
    //     form.keepExtensions = true   //是否保持拓展名


    //     form.parse(ctx.req, async function (err, fields, files) {
    //         let file = files.file
    //         filePath = file.path
    //         fileName = file.name

    //         // 将数据保存进数据库
    //         let result = await user_db.sqlQuery('INSERT INTO server_article_addcover (filepath, filename, created_at) values (?,?,?)', [filePath, fileName, timeStamp])

    //         return ctx.body = {
    //             code: '001',
    //             msg: '上传成功',
    //             filepath: filePath.replace(/^\/public/, ''),
    //             filename: fileName
    //         }
    //     })


    // })


}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容