nodejs實(shí)現(xiàn)范圍請(qǐng)求的實(shí)現(xiàn)代碼
使服務(wù)器支持范圍請(qǐng)求,允許客戶端只請(qǐng)求文檔的一部分。其流程是:
- 客戶端向服務(wù)端發(fā)起請(qǐng)求
- 服務(wù)端響應(yīng),附上Accept-Ranges頭(值表示表示范圍的單位,通常是“bytes”),告訴客戶端其接受范圍請(qǐng)求
- 客戶端發(fā)送新的請(qǐng)求,附上Ranges頭,告訴服務(wù)端請(qǐng)求的是一個(gè)范圍
- 服務(wù)端收到范圍請(qǐng)求,分情況響應(yīng):
- 范圍有效,服務(wù)端返回206 Partial Content,發(fā)送指定范圍內(nèi)內(nèi)容,并在Content-Range頭中指定該范圍
- 范圍無(wú)效,服務(wù)端返回416 Requested Range Not Satisfiable,并在Content-Range中指明可接受范圍
請(qǐng)求中的Ranges頭格式為(這里不考慮多范圍請(qǐng)求了):
Ranges: bytes=[start]-[end]
其中 start 和 end 并不是必須同時(shí)具有:
- 如果 end 省略,服務(wù)器應(yīng)返回從 start 位置開(kāi)始之后的所有字節(jié)
- 如果 start 省略,end 值指的就是服務(wù)器該返回最后多少個(gè)字節(jié)
- 如果均未省略,則服務(wù)器返回 start 和 end 之間的字節(jié)
響應(yīng)中的Content-Range頭有兩種格式:
當(dāng)范圍有效返回 206 時(shí):
Content-Range: bytes (start)-(end)/(total)
當(dāng)范圍無(wú)效返回 416 時(shí):
Content-Range: bytes */(total) //代碼實(shí)現(xiàn) res.setHeader('Content-Range', `bytes */${totalSize}`);
添加函數(shù)處理范圍請(qǐng)求:
【注意】這里吧不合理的范圍請(qǐng)求直接設(shè)置為返回全部,如果想設(shè)置范圍錯(cuò)誤就直接不返回可以設(shè)置http狀態(tài)碼為416
/** * 范圍請(qǐng)求 * 不在合理的范圍內(nèi)就直接返回所有 * 在合理規(guī)范的范圍內(nèi)就按照返回對(duì)應(yīng)的部分 * @param {*} totleSize 總的范圍 * @param {*} req * @param {*} res */ module.exports = (totleSize, req, res) => { const range = req.headers['range']; if (!range) { return {code: 200}; } const sizes = range.match(/bytes=(\d*)-(\d*)/); const end = sizes[2] || totleSize - 1; const start = sizes[1] || totleSize - end; //不規(guī)范的范圍,是要直接一起返回,所以設(shè)置狀態(tài)碼200 if(start > end || start < 0 || end > totleSize) { return {code: 200}; } res.setHeader('Accept-Ranges', 'bytes'); res.setHeader('Content_Range', `bytes ${start}-${end}/${totleSize}`); res.setHeader('Content_Length', end-start); return { code: 206, start: parseInt(start), end: parseInt(end) }; };
范圍讀?。?/p>
let rs; const { code, start, end } = range(stats.size, req, res); if (code === 200) { res.statusCode = 200; rs = fs.createReadStream(filePath); } else { res.statusCode = 206;//表示是部分內(nèi)容 //創(chuàng)建一個(gè)流讀取文件, {start, end}表示文件讀取的起始點(diǎn)和終點(diǎn) rs = fs.createReadStream(filePath, { start, end }); } //有這個(gè)pipe就不需要再去調(diào)用end了,這樣直接調(diào)用end會(huì)導(dǎo)致res接受不到東西,因?yàn)閜ipe是異步的 rs.pipe(res);
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
node.js中的fs.statSync方法使用說(shuō)明
這篇文章主要介紹了node.js中的fs.statSync方法使用說(shuō)明,本文介紹了fs.statSync的方法說(shuō)明、語(yǔ)法、接收參數(shù)、使用實(shí)例和實(shí)現(xiàn)源碼,需要的朋友可以參考下2014-12-12Node+Express連接MySQL實(shí)現(xiàn)增刪改查功能
這篇文章主要為大家詳細(xì)介紹了Node如何結(jié)合Express連接MySQL實(shí)現(xiàn)增刪改查功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-05-05NodeJs操作MongoDB教程之分頁(yè)功能以及常見(jiàn)問(wèn)題
這篇文章主要給大家介紹了關(guān)于NodeJs操作MongoDB教程之分頁(yè)功能以及常見(jiàn)問(wèn)題的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用NodeJs具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04nodejs爬蟲(chóng)抓取數(shù)據(jù)亂碼問(wèn)題總結(jié)
這篇文章主要給大家總結(jié)了下nodejs爬蟲(chóng)抓取數(shù)據(jù)亂碼問(wèn)題的相關(guān)資料,需要的朋友可以參考下2015-07-07Nodejs中的計(jì)時(shí)器(setTimeout?setIntervals?etImmediate)使用案例解析
這篇文章主要介紹了Nodejs中的計(jì)時(shí)器(setTimeout?setIntervals?etImmediate)使用案例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06node.js連接mongoDB數(shù)據(jù)庫(kù) 快速搭建自己的web服務(wù)
這篇文章主要為大家詳細(xì)介紹了node.js連接mongoDB數(shù)據(jù)庫(kù),如何快速搭建自己的web服務(wù),感興趣的小伙伴們可以參考一下2016-04-04