• 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)》

    参考

  • 相关阅读:
    包管理工具--》发布一个自己的npm包
    高并发应用实践——限流手段
    Spring高手之路12——BeanDefinitionRegistry与BeanDefinition合并解析
    vm的生命周期钩子
    【LeetCode】错误的集合&&在排序数组中查找元素的第一个和最后一个位置&&杨氏矩阵&&寻找数组的中心下标&&两个数组的交集
    Pytorch - masked_fill方法参数详解与使用
    代码随想录训练营第28天|LeetCode 93.复原IP地址、78.子集、 90.子集II
    【重温基础算法】内部排序之希尔排序法
    Java 传统的生产者与消费问题以及虚假唤醒
    论坛系统测试用例设计
  • 原文地址:https://www.cnblogs.com/lucky_hu/p/17988315