Vue 通過 minio 上傳文件

在Vue中minio上傳文件(重點(diǎn)是:stream流)
1、minio的安裝

npm i minio --save
//或者
cnpm i minio --save

2襟锐、minio.js文件
封裝了連接minio、文件上傳,文件刪除等方法
注:此文件中的url都是在minio設(shè)置了永久鏈接的基礎(chǔ)上
2.1連接minio

const minioClient = new Minio.Client({
  endPoint: 'xxx.xxx.x.xxx', // 地址
  port: xx, // 端口號(hào)注竿,若地址為類似test.minio.com,就不必寫端口號(hào)
  useSSL: false, // 是否使用ssl
  accessKey: '', // 登錄的accessKey
  secretKey: '' // secretKey
});

2.2通過stream流上傳(不可以直接上傳,需要轉(zhuǎn)流)

const stream = require('stream')
const Minio = require('minio')
import {
  getNowTime
} from "@common/publicmethods"
/**
 *
 * @export 上傳文件(stream流方法)
 * @param {*} backetName String 存儲(chǔ)桶名字
 * @param {*} fileObj Object 文件對(duì)象
 * @param {*} path String 文件存儲(chǔ)路徑
 * @param {*} vm Object 調(diào)用該方法的頁(yè)面的this
 * @return {*} null
 */
export function uploadMinIo(backetName,fileObj, path, vm) {
  if (
    fileObj
  ) {
    let file = fileObj;
    console.log("file", file);
    //判斷是否選擇了文件
    if (file == undefined) {
      //未選擇
    } else {
      //選擇
      // 獲取文件類型及大小
      // 給文件名加上當(dāng)前時(shí)間
      const fileName = getNowTime("time") + file.name;
      const fileDate = getNowTime("fileDate") // 生成以日為分類的文件夾
      const mineType = file.type;
      const fileSize = file.size;
      console.log("fileName", fileName);
      //參數(shù)
      let metadata = {
        "content-type": mineType,
        "content-length": fileSize,
      };
      //判斷儲(chǔ)存桶是否存在
      minioClient.bucketExists(backetName, function (err) {
        console.log("判斷儲(chǔ)存桶是否存在");
        if (err) {
          if (err.code == "NoSuchBucket")
            return console.log("bucket does not exist.");
          return console.log(err);
        }
        //準(zhǔn)備上傳
        let reader = new FileReader();
        console.log(reader);
        reader.readAsDataURL(file);
        reader.onloadend = function (e) {
          //讀取完成觸發(fā)魂贬,無(wú)論成功或失敗
          console.log("ee", e);
          const dataurl = e.target.result;
          //base64轉(zhuǎn)blob
          const blob = toBlob(dataurl);
          //blob轉(zhuǎn)arrayBuffer
          let reader2 = new FileReader();
          reader2.readAsArrayBuffer(blob);
          reader2.onload = function (ex) {
            //定義流
            let bufferStream = new stream.PassThrough();
            //將buffer寫入
 
            bufferStream.end(Buffer.from(ex.target.result));
            //上傳
            minioClient.putObject(
              backetName,
              `${path}/${fileDate}/${fileName}`,
              bufferStream,
              fileSize,
              metadata,
              function (err, etag) {
                // console.log("dddddd");
                if (err == null) { // 為空則代表上傳成功
                  let res = {
                    path: `http://192.168.0.226:30014/${backetName}/${path}/${fileDate}/${fileName}`,
                    result: 1,
                  };
                  // 成功生成url后調(diào)用
                  // 調(diào)用傳進(jìn)來(lái)的this的的方法巩割,然后通過該方法把成功事件發(fā)送出去
                  vm.handleAvatarSuccess(res, vm.filedname);
                  vm.fileName = fileName;
                  vm.$message({
                    message: "上傳成功!",
                    type: "success",
                  });
                  // 由于minio設(shè)置了永久鏈接付燥,該生成臨時(shí)url的方法就不再使用
                  // minioClient.presignedGetObject(
                  //   "medialibrary",
                  //   `archive${a}${fileName}`,
                  //   24 * 60 * 60,
                  //   function (err, presignedUrl) {
                  //     if (err) return console.log(err);
                  //     let res = {
                  //       path: presignedUrl,
                  //       result: 1,
                  //     };
                  //     // 成功生成url后調(diào)用
                  //     vm.handleAvatarSuccess(res, vm.filedname);
                  //     vm.fileName = fileName;
                  //     vm.$message({
                  //       message: "上傳成功喂分!",
                  //       type: "success",
                  //     });
                  //     console.log("鏈接:",presignedUrl);
                  //   }
                  // );
                }
              }
            );
          };
        };
      });
    }
  } else {
    this.$message({
      message: "文件類型錯(cuò)誤!",
      type: "error",
    });
  }
}

2.3通過帶預(yù)簽名的url上傳(最好是minio設(shè)置了鏈接永久訪問)
1.先拿到預(yù)簽名鏈接
2.再通過預(yù)簽名上傳文件

// 兩個(gè)參數(shù)机蔗,存儲(chǔ)桶的名字蒲祈,要上傳文件的名字。例如test.txt
/**
 *
 * @export 獲取上傳鏈接(url方法)獲取上傳的url;會(huì)生成一個(gè)帶預(yù)簽名的鏈接
 * @param {*} bucket String 存儲(chǔ)桶的名字
 * @param {*} totalFolderName String 總文件夾名字 例如:imagelibrary
 * @param {*} fileName String 文件名字
 * @return {*} Promise
 */
export function getUploadUrl(bucket, totalFolderName,fileName) {
  let defaultPath = getNowTime("fileDate"); // 添加默認(rèn)的以日為分類的文件夾
  return minioClient.presignedPutObject(bucket, `${totalFolderName}/${defaultPath}/${fileName}`)
}
 
// 通過url上傳
/**
 *
 * @export 上傳文件(url方法)通過獲取帶預(yù)簽名的鏈接上傳
 * @param {*} url String 預(yù)簽名鏈接
 * @param {*} data Object 文件對(duì)象
 * @return {*} Promise
 */
export function uploadByUrl(url, data) {
  return fetch(url, {
    mode: "cors", // 解決跨域
    headers: {
      Accept: "application/json,text/plain,/",
    },
    method: "PUT",
    body: data,
  });
}

2.4刪除對(duì)象

/**
 * @export 從存儲(chǔ)桶中刪除一個(gè)對(duì)象
 * @param {*} bucketName 存儲(chǔ)桶的名字
 * @param {*} objectPathAndName 要?jiǎng)h除對(duì)象的路徑(注意:要寫對(duì)路徑萝嘁,緊跟存儲(chǔ)桶后面的路徑)
 * @description 
 * @author 
 * @version V1.0.0
 * @return {*} Promise
*/
export function removeObject(bucketName,objectPathAndName){
  return minioClient.removeObject(bucketName, objectPathAndName)
}

補(bǔ)充:base64轉(zhuǎn)blob

/**
 *
 * @export base64轉(zhuǎn)blob
 * @param {*} base64Data Object base64數(shù)據(jù)
 * @return {*} blob
 */
//base64轉(zhuǎn)blob
export function toBlob(base64Data) {
  let byteString = base64Data
  if (base64Data.split(',')[0].indexOf('base64') >= 0) {
    byteString = window.atob(base64Data.split(',')[1]) // base64 解碼
  } else {
    byteString = unescape(base64Data.split(',')[1])
  }
  // 獲取文件類型
  let mimeString = base64Data.split(';')[0].split(":")[1] // mime類型
 
  // ArrayBuffer 對(duì)象用來(lái)表示通用的梆掸、固定長(zhǎng)度的原始二進(jìn)制數(shù)據(jù)緩沖區(qū)
  // let arrayBuffer = new ArrayBuffer(byteString.length) // 創(chuàng)建緩沖數(shù)組
  // let uintArr = new Uint8Array(arrayBuffer) // 創(chuàng)建視圖
 
  let uintArr = new Uint8Array(byteString.length) // 創(chuàng)建視圖
 
  for (let i = 0; i < byteString.length; i++) {
    uintArr[i] = byteString.charCodeAt(i)
  }
  // 生成blob
  const blob = new Blob([uintArr], {
    type: mimeString
  })
  // 使用 Blob 創(chuàng)建一個(gè)指向類型化數(shù)組的URL, URL.createObjectURL是new Blob文件的方法,可以生成一個(gè)普通的url,可以直接使用,比如用在img.src上
  return blob
}

3、獲取時(shí)間方法
publicmethods.js

/**
 *
 * @export 獲取當(dāng)前時(shí)間并格式化
 * @param {*} isDate Boolean 為真返回時(shí)分秒牙言; 為"time"返回不帶連接符的時(shí)分秒酸钦;為"fileDate"返回文件夾日期
 * @return {*} yyyy-MM-dd hh:mm:ss
 */
export function getNowTime(isDate = false) {
    var now = new Date();
    var year = now.getFullYear(); //得到年份
    var month = now.getMonth(); //得到月份
    var date = now.getDate(); //得到日期
    var hh = now.getHours();
    var mm = now.getMinutes();
    var ss = now.getSeconds();
    var hour = " " + hh + ":" + mm + ":" + ss + ""; //默認(rèn)時(shí)分秒 如果傳給后臺(tái)的格式為年月日時(shí)分秒,就需要加這個(gè)咱枉,如若不需要卑硫,此行可忽略
    var hours = hh + '' + mm + '' + ss + "-";
    month = month + 1;
    month = month.toString().padStart(2, "0");
    date = date.toString().padStart(2, "0");
    var defaultDate = `${year}-${month}-${date}${hour}`;
    if (isDate) {
        var defaultDate = `${year}-${month}-${date}`;
    }
    if (isDate == 'time') { // 返回給文件前添加的時(shí)間
        var defaultDate = `${year}${month}${date}${hours}`;
    }
    if (isDate == 'fileDate') { // 返回默認(rèn)創(chuàng)建文件夾的時(shí)間
        var defaultDate = `${year}${month}${date}`;
    }
    return defaultDate;
}

5、解決跨域問題(修改如下文件中的配置)

image.png
ingress:
  enabled: true
  labels: {}
    # node-role.kubernetes.io/ingress: platform
 
  annotations:
    kubernetes.io/ingress.class: nginx
    # kubernetes.io/tls-acme: "true"
    # kubernetes.io/ingress.allow-http: "false"
    # kubernetes.io/ingress.global-static-ip-name: ""
    # nginx.ingress.kubernetes.io/secure-backends: "true"
    # nginx.ingress.kubernetes.io/backend-protocol: "HTTPS"
    # nginx.ingress.kubernetes.io/whitelist-source-range: 0.0.0.0/0
    nginx.ingress.kubernetes.io/cors-allow-headers: DNT,web-token,app-token,Authorization,Accept,Origin,Keep-Alive,User-Agent,X-Mx-ReqToken,X-Data-Type,X-Auth-Token,X-Requested-With,If-Modified-Since,Cache-Control,Content-Type,Range,token,Cookie,x-amz-content-sha256,x-amz-date
    nginx.ingress.kubernetes.io/cors-allow-methods: PUT, GET, POST, OPTIONS, DELETE
    nginx.ingress.kubernetes.io/cors-allow-origin: '*'
    nginx.ingress.kubernetes.io/enable-cors: 'true'

6蚕断、存儲(chǔ)大小限制問題(加上下面一行就可以了)


image.png

7欢伏、minio官方文檔

官方鏈接

8、minio設(shè)置鏈接永久訪問(其他方法)

minio永久訪問

總結(jié)

轉(zhuǎn)載過來(lái)亿乳,哪天需要了方便查找硝拧【斗ぃ看到2.2通過stream流上傳(不可以直接上傳,需要轉(zhuǎn)流)這步解決了我的上傳后找不到文件問題障陶。原文鏈接:https://www.jb51.net/article/255316.htm

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末滋恬,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子抱究,更是在濱河造成了極大的恐慌恢氯,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,548評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件鼓寺,死亡現(xiàn)場(chǎng)離奇詭異酿雪,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)侄刽,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,497評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門指黎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人州丹,你說(shuō)我怎么就攤上這事醋安。” “怎么了墓毒?”我有些...
    開封第一講書人閱讀 167,990評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵吓揪,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我所计,道長(zhǎng)柠辞,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,618評(píng)論 1 296
  • 正文 為了忘掉前任主胧,我火速辦了婚禮叭首,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘踪栋。我一直安慰自己焙格,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,618評(píng)論 6 397
  • 文/花漫 我一把揭開白布夷都。 她就那樣靜靜地躺著眷唉,像睡著了一般。 火紅的嫁衣襯著肌膚如雪囤官。 梳的紋絲不亂的頭發(fā)上冬阳,一...
    開封第一講書人閱讀 52,246評(píng)論 1 308
  • 那天,我揣著相機(jī)與錄音党饮,去河邊找鬼肝陪。 笑死,一個(gè)胖子當(dāng)著我的面吹牛劫谅,可吹牛的內(nèi)容都是我干的见坑。 我是一名探鬼主播嚷掠,決...
    沈念sama閱讀 40,819評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼捏检,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼荞驴!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起贯城,我...
    開封第一講書人閱讀 39,725評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤熊楼,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后能犯,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體鲫骗,經(jīng)...
    沈念sama閱讀 46,268評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,356評(píng)論 3 340
  • 正文 我和宋清朗相戀三年踩晶,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了执泰。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,488評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡渡蜻,死狀恐怖术吝,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情茸苇,我是刑警寧澤排苍,帶...
    沈念sama閱讀 36,181評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站学密,受9級(jí)特大地震影響淘衙,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜腻暮,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,862評(píng)論 3 333
  • 文/蒙蒙 一彤守、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧哭靖,春花似錦遗增、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,331評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至抡草,卻和暖如春饰及,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背康震。 一陣腳步聲響...
    開封第一講書人閱讀 33,445評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工燎含, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人腿短。 一個(gè)月前我還...
    沈念sama閱讀 48,897評(píng)論 3 376
  • 正文 我出身青樓屏箍,卻偏偏與公主長(zhǎng)得像绘梦,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子赴魁,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,500評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容