• python批量上传文件到七牛云


    导航

    • 引子
    • 棘手的需求
    • 化繁为简
    • 实战案例
    • 结语
    • 参考

    本文首发于智客工坊-《python批量上传文件到七牛云》,感谢您的阅读,预计阅读时长3min。

    古之立大事者,不惟有超世之才,亦必有坚忍不拔之志。 —— 苏轼

    引子

    七牛云对象存储 Kodo 是七牛云提供的高可靠、强安全、低成本、可扩展的存储服务。您可通过控制台、API、SDK 等方式简单快速地接入七牛存储服务,实现海量数据的存储和管理。通过 Kodo 可以进行文件的上传、下载和管理。

    我们通常将图片、音视频、javascript、css等静态资源上传到七牛云,利用cdn提高访问速度。

    但是,遗憾的是,七牛云没有提供批量上传指定文件目录中所有文件的功能,而只能单个上传。

    棘手的需求

    我们有一个基于NextJs构建的web项目,每次发布都会在本地打包,发布的文件成百上千,且目录是多层结构。

    通常我们是直接将打包后的文件拷贝到web服务器替换。坏就坏在,里面有个wasm类型的文件超过2.8M,首次加载超过10s,导致整个网站卡顿。

    如何解决呢?

    从nextjs官网查阅发现,支持为静态资源添加路径前缀并支持发布到CDN。

    const isProd = process.env.NODE_ENV === 'production'
    
    module.exports = {
      // Use the CDN in production and localhost for development.
      assetPrefix: isProd ? 'https://cdn.mydomain.com' : '',
    }
    

    静态资源上传到 CDN,这是存在目前存在的最大的问题,虽然在next.config.js中可以配置assetPrefix字段,但实际使用起来还是非常困难。

    打包后的 js 和 css,引用路由均为/_next/static开头。

    假设我们的CDN地址是 http://static.qq.com

    
    

    最后拼接后的地址是:

    
    

    中间多出了/_next/static的路径,最后的结果是页面需要加载的资源和上传的资源路径不一致,就会各种 404。

    无独有偶,也有博友遇到了类似的问题-《nextjs:如何将静态资源发布到 CDN》

    化繁为简

    复杂的事情简单化。

    综合各方的信息,最终决定尝试将_next/static及其文件整体上传到七牛云。

    在七牛云的存储目录如下:

    根目录/_next/static/

    这样代码上就不需要做改动。

    唯一要做的就是实现批量上传文件到七牛云。

    实战案例

    废话不多说,我们现在就用代码来实现上面的需求吧!

    
    # -*- coding: utf-8 -*-
    # author: zhikecore
    # https://developer.qiniu.com/kodo/1242/python#rs-stat
    
    import os;
    from qiniu import Auth, put_file, etag
    from qiniu import BucketManager
    
    # 全局变量
    # 需要填写你的 Access Key 和 Secret Key
    ACCESS_KEY='您申请的七牛云Access Key'
    SECRET_KEY='您申请的七牛云Secret Key'
    
    #要上传的空间
    BUCKET_NAME = '您申请的七牛云空间名称bucket'
    
    # 遍历文件
    def scan_dir(path):
        for root, dirs, files in os.walk(path):
            for file in files:
                filePath=os.path.join(root, file)
                fileDir=os.path.dirname(filePath).replace('\\','/')
                fileKey=fileDir.replace(path,'_next/static')+'/'+file
                #print('fileDir:'+fileDir+',filePath:'+filePath+',fileKey:'+fileKey)
                #delfileKey(fileKey)
                #print("del fileKey:"+fileKey+",success!")
                upload(filePath,fileKey)
                print("upload fileKey:"+fileKey+",success!")
            for dir in dirs:
                dir=os.path.join(root, dir)
                print(dir)
    
    
    def upload(filepath,fileKey):
        #构建鉴权对象
        q = Auth(ACCESS_KEY, SECRET_KEY)
        #要上传的空间
        bucket_name = BUCKET_NAME
        #上传后保存的文件名
        key=fileKey
        #生成上传 Token,可以指定过期时间等
        token = q.upload_token(bucket_name, key, 3600)
        #要上传文件的本地路径
        localfile = filepath
        print('key:'+key,"localfile:"+localfile)
        ret, info = put_file(token, key, localfile, version='v2') 
        print(info)
        assert ret['key'] == key
        assert ret['hash'] == etag(localfile)
    
    def delfileKey(fileKey):
        #初始化Auth状态
        q = Auth(ACCESS_KEY, SECRET_KEY)
        #初始化BucketManager
        bucket = BucketManager(q)
        #你要测试的空间, 并且这个key在你空间中存在
        bucket_name = BUCKET_NAME
        key = fileKey
        #删除bucket_name 中的文件 key
        ret, info = bucket.delete(bucket_name, key)
        print(info)
    
    
    # 如果该文件不是被import,则执行下面代码。
    if __name__ == '__main__':
        scan_dir('./_next/static')    
    
    

    代码中几乎每行都有注释,这里不再赘述。

    至此,代码已经编写完成,我们保存为next_package_upload.py文件。

    接下来,就是让代码run起来。



    我们再次访问一下网站,查看wasm文件资源的加载速度。


    很显然从10s+提升到10ms内,性能大幅提升。

    这里需要注意的是,每次发版我会先删除旧的文件,再上传新的文件。delfileKey方法可以根据您的实际选择使用或者不用。

    没有安装python环境的新手同学可以参考《Python安装和使用教程(windows)》

    参考

  • 相关阅读:
    解决安卓中 ARouter There is no route match the path in group问题
    如何化解从数据到数据资源入表的难题
    Django 的国际化与本地化详解
    【Elasticsearch索引】索引基础操作
    SAP gui 登录条目不让修改
    数据结构之单链表
    kubernetes部署dashboard
    AN动画基础——缓动动画
    安卓项目结构分析
    Java学习笔记---多态综合练习
  • 原文地址:https://www.cnblogs.com/lucky_hu/p/17988315