首页 > 其他分享 >vue + djangorestframework實現文件下載功能

vue + djangorestframework實現文件下載功能

时间:2023-03-15 18:14:38浏览次数:50  
标签:vue 下載 實現 django self file path data response

1.安裝模塊及配置及配置

先安裝django-cors-headers包

pip3 install djangorestframework django-cors-headers

 

在setting文件中註冊app

INSTALLED_APPS = [
 'django.contrib.admin',
 'django.contrib.auth',
 'django.contrib.contenttypes',
 'django.contrib.sessions',
 'django.contrib.messages',
 'django.contrib.staticfiles',
 'api.apps.ApiConfig',
 'corsheaders', # 注册应用cors
]

 

在setting文件中註冊中間件

MIDDLEWARE = [
 'django.middleware.security.SecurityMiddleware',
 'django.contrib.sessions.middleware.SessionMiddleware',
 'django.middleware.common.CommonMiddleware',
 'django.middleware.csrf.CsrfViewMiddleware',
 'django.contrib.auth.middleware.AuthenticationMiddleware',
 'django.contrib.messages.middleware.MessageMiddleware',
 'django.middleware.clickjacking.XFrameOptionsMiddleware',
 'corsheaders.middleware.CorsMiddleware', # 注册组件cors
]

 

配置置請求方法與請求頭部信息

CORS_ALLOW_METHODS = (
 'GET',
 'OPTIONS',
 'PATCH',
 'POST',
 'VIEW',
)

CORS_ALLOW_HEADERS = (
 'XMLHttpRequest',
 'X_FILENAME',
 'accept-encoding',
 'authorization',
 'content-type',
 'dnt',
 'origin',
 'user-agent',
 'x-csrftoken',
 'x-requested-with',
 'Pragma',
)

 

2.添加路由與方法

路由

from django.contrib import admin
from django.urls import path
from api import views

urlpatterns = [
 path('admin/', admin.site.urls),
 path('downloadFile/', views.Download.as_view()),
]

 

視圖

class Download(DispatchMethod,APIView):

    def post(self, request):
        file_name = request.data.get('file_name')
        file_path = request.data.get('file_path')
        connKey = request.data.get("connKey")

        hostname = request.data.get("hostname")
        login_key = request.data.get("login_key")
        program_id = request.data.get("program_id")
        stime = str(datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S"))

        language = getLanguage(request)
        data = checkToken(request)
        if data.get("status") > 0:
            bu = UmbResourceBll(connKey)
            data = bu.getFileObj(file_name,file_path,language)
            WriteApiLog(stime, program_id, login_key, hostname, "[method:post]", self)
            if data.get("status"):
                return JsonResponse(data, status=200)
            return data
        else:
            return JsonResponse(data, status=401)

class UmbResourceBll(object):

    def __init__(self, connkey=k_ov.local_db_id):
        self.dbConnKey = baseUtils.checkConnkey(connkey)
        self.ud = UmbResourceDal()
        self.ulang = langUtils()

    def response_param(self, response_code, response_msg, language=None, error_log=None):
        if language:
            if response_code == -99:
                sMessage = self.ulang.getMSG(response_msg, language) + str(error_log)
            else:
                sMessage = self.ulang.getMSG(response_msg, language)
        else:
            sMessage = response_msg
        return response_code, sMessagedef getFileObj(self, file_name, file_path, language):
        status = 0
        sMessage = ""
        rows = []
        try:
            # StreamingHttpResponse 流式響應,内容的迭代器形式,以内容流的方式响应
            file_obj = StreamingHttpResponse(self.file_iterator(file_path))
            file_obj['Content-Type'] = 'application/octet-stream'
            file_obj['Access-Control-Expose-Headers'] = "Content-Disposition, Content-Type"
            # escape_uri_path防止文件是中文名出現亂碼
            file_obj['Content-Disposition'] = "attachment; filename={}".format(escape_uri_path(file_name))

            # status, sMessage = self.response_param(2, 'UW_0120', language)
            return file_obj

        except Exception as e:
            status, sMessage = self.response_param(-99, 'AM_0008', language, error_log=e)
            return {"rows": rows, "status": status, "message": sMessage}

    @staticmethod
    def file_iterator(file_path, chunk_size=512):
        """
        讀出文件 以rb的形式
        文件生成器,防止文件過大,導致內存溢出
        :param file_path: 文件絕對路勁
        :param chunk_size: 每次讀出的大小
        :return:
        """
        with open(file_path, 'rb') as f:
            while True:
                content = f.read(chunk_size)
                if content:
                    yield content
                else:
                    break

 

3.前端展示及獲取邏輯

template
<template>
    <div class="right-button">
        <div class="right-button">
            <el-row style="margin-left:10px">
                 <el-button size="mini" icon="el-icon-view" type="primary" @click="showFiles">{{$t("視圖")}}</el-button>
                 <el-button size="mini" icon="el-icon-folder-opened"  @click="downloadFile" type="success">{{$t("下載")}}</el-button>
            </el-row>
        </div>
    </div>  
</template>

 

script

     // 下載指定的文件
        downloadFile(){
            // 從session中獲取被點擊的主table行數據
            let current_row = window.sessionStorage.getItem("nowRow")

            current_row = eval('('+current_row+')')
            if(!current_row.file_name){
                return
            }

            DownloadFileParames.file_name = current_row.file_name
            DownloadFileParames.file_path = current_row.file_path
            DownloadFileParames.connKey = this.$store.state.now_conn_key
            let file_type

            return new Promise((resolve, reject) =>{

                this.$http.post(apiDownloadFile,DownloadFileParames,{
                    "headers": { 'Authorization': "jwt " + window.localStorage.getItem("token") },
                    "responseType": "blob"
                }).then(

                    response => {
                        resolve(response.data)

                        // 根據文件後綴設置文件流的文件類型
                        if(DownloadFileParames.file_name.endsWith("pdf")){
                            file_type = "application/pdf"
                        }else if(DownloadFileParames.file_name.endsWith("mp4")){
                            file_type = "video/mpeg4"
                        }

                        let blob = new Blob([response.data], {
                            type:file_type
                        })

                        // 文件名
                        let fileNameEncode = response.headers['content-disposition'].split("filename=")[1]

                        // 解码 函数可把字符串作为 URI 组件进行解码
                        let fileName = decodeURIComponent(fileNameEncode)

                        //  msSaveBlob和msSaveOrOpenBlob 方法允许用户在客户端上保存文件
                        // window.navigator 对象包含有关访问者浏览器的信息
                        if (window.navigator.msSaveOrOpenBlob) {
                            navigator.msSaveBlob(blob, fileName)
                        } else {
                            // 創建a標籤
                            var link = document.createElement('a')

                            // 設置a標籤鏈接
                            link.href = window.URL.createObjectURL(blob)

                            // 將文件名賦值給a標籤的下載屬性并觸發點擊
                            link.download = fileName
                            link.click()

                            //释放内存
                            window.URL.revokeObjectURL(link.href)
                        }
                    }, err => {
                        reject(this.$message({
                            showClose: false,
                            message: res.data.message,
                            type: 'error',
                            duration: 2000    
                        }))
                    }
                )
            })
        }

 

 

标签:vue,下載,實現,django,self,file,path,data,response
From: https://www.cnblogs.com/wuzhengzheng/p/17219462.html

相关文章

  • vue入门篇之Vue.js 组件
    Vue.js组件是该框架最强大的功能之一,能够扩展HTML元素并封装可重用的代码。通过组件系统,我们可以使用独立可复用的小组件来构建大型应用,几乎任何类型的应用的界面都可以......
  • vuex的应用
    需求:系统顶部添加项目下拉框,顶部下拉框选项改变时其他模块下拉框同时改变。1.开始进入系统的时候获取项目列表页面调用接口//获取项目列表store.dispatch("getProj......
  • 初始Vue
    Vue简介介绍与描述Vue是一套用来动态构建用户界面的渐进式JavaScript框架-构建用户界面:把数据通过某种办法变成用户界面-渐进式:Vue可以自底向上逐层地应用,简单应用只......
  • vue3+vite+vant文件上传
    1:文件上传下载1//上传文件2constafterRead=()=>{3for(letfileoffileList.value){4//1:加载状......
  • MVVM模式和第一个Vue程序
    Vue:MVVM模式和第一个Vue程序什么是MVVM​ MVVM(Model-View-ViewModel)是一种软件架构设计模式,由微软WPF(用于替代WinForm,以前就是用这个技术开发桌面应用程序的)和Silv......
  • VUE基础:组件CSS样式、props属性
    VUE基础部分组件css样式例子<stylelang="less">//这里记得改名,然后给div也加名字.comName-container{padding:1020px20px;background-color:lightskyblue......
  • 前端常考vue面试题(必备)
    computed的实现原理computed本质是一个惰性求值的观察者。computed内部实现了一个惰性的watcher,也就是computedwatcher,computedwatcher不会立刻求值,同时持有......
  • vue源码分析-挂载流程和模板编译
    前面几节我们从newVue创建实例开始,介绍了创建实例时执行初始化流程中的重要两步,配置选项的资源合并,以及响应式系统的核心思想,数据代理。在合并章节,我们对Vue丰富的选项......
  • 校招前端vue面试题(边面边更)
    Vue模版编译原理知道吗,能简单说一下吗?简单说,Vue的编译过程就是将template转化为render函数的过程。会经历以下阶段:生成AST树优化codegen首先解析模版,生成AST语法树(......
  • 总结vue的13种传值方法
    1,父子传值,自定义属性传递参数,子组件使用props接收数据2,子传父,父组件提供自定义事件,子组件通过$emit触发事件进行传值3,v-mode传递属性4,.sync传值,提供一个具体的属性,提供......