对于那些想要将此限制为一次请求的人,我有一个新颖的方法来解决这个问题。事件流请求像任何AJAX请求一样向服务器传递cookies。我的方法是在发出请求之前设置包含数据的cookies,然后在第一个响应后立即丢弃这些cookies。
现在需要注意一些关于cookies的限制。首先,一个cookie的最大长度是4096字节,包括名称以及可能的3字节开销。其次,每个浏览器可以安全存储的cookie数量为50个。不同浏览器的最大数量不同,谷歌浏览器是180个,安卓浏览器是50个。
我将让您确定如何在这些限制周围实现逻辑,但我会提供我的实现示例。请注意,此实现使用js-cookies来操作cookies。我们还在服务器端使用了cookie-parser中间件。
浏览器JS
const productIDs = ["b0708d2c-fe46-4251-96e0-1cfb3cd05eb0", "244d1e73-b5b4-4c59-8d4e-006fc1b190fe"];
const size = new TextEncoder().encode(JSON.stringify(productIDs)).length;
const chunkSize = Math.floor(1024 * 3.5);
const segments = Math.ceil(size / chunkSize);
const currentCookies = Object.keys(Cookies.get()).length;
if (segments + currentCookies <= 50){
let payloads = [];
for (let segment = 0; segment < segments; segment++){
const payloadLength = Math.ceil(productIDs.length / segments);
const startPosition = segment * payloadLength;
payloads[segment] = productIDs.slice(startPosition, startPosition + payloadLength);
}
for (const [index, payload] of Object.entries(payloads)){
Cookies.set(`qp[${index}]`, JSON.stringify(payload));
}
const source = new EventSource("start-processing");
source.addEventListener("connected", () => {
for (const [index, payload] of Object.entries(payloads)){
Cookies.remove(`qp[${index}]`);
}
});
}else{
console.log(`ERROR: Cookie limit was reached.`);
}
如你所见,我们将
chunkSize
设置为3584字节。这似乎是一个相当安全的填充量,但你可以根据自己的需求进行调整。请记住,这种方法会均匀分配数组值到所有有效负载数组中。这意味着你不太可能完全达到chunkSize的值。
你可以将最大cookie数从50调整为任何你想要的值。如果你不关心移动设备或旧版浏览器,只想支持现代桌面浏览器,那么150是支持Firefox的最低要求。请查看
this page以获取更多有关浏览器限制的信息。
在我们将产品ID拆分为各自的数组之后,我们为每个块生成cookie。在这种情况下,我将其标记为
qp
,并在后面加上一个索引到块的
[0]
标识符。
最后,我们启动事件流。你需要有一个立即响应的事件,以便浏览器知道要清除新创建的cookie。此外,最好在某个地方有一个独立的函数来清除这些cookie,以防响应没有返回,或者在响应到达之前刷新了浏览器。
我建议在上述代码块的开头添加以下代码,以清除可能残留的任何先前数据。
for (const cookie of Object.keys(Cookies.get()).filter(x => x.startsWith("qp"))){
Cookies.remove(cookie);
}
服务器端JS(Express)
async (request, response) => {
response.writeHead(200, {
'Content-Type': 'text/event-stream',
'Connection': 'keep-alive',
'Cache-Control': 'no-cache'
});
const productIDs = Object.entries(request.cookies).filter(([name, value]) => name.startsWith("qp")).map(([index, value]) => JSON.parse(value)).flat();
const sendEvent = (event, data) => response.write(`event: ${event}\ndata: ${JSON.stringify(data)}\n\n`);
sendEvent("connected", {});
}
相当简单的东西。那个相当长的一行代码是一种非常简单的方法,可以获取所有的数组并将它们展平为一个单一的数组。只需确保
startsWith("qp")
部分设置为您的cookie名称,并且显然要确保您网站上没有其他以该字符串开头的cookie。
sendEvent()
是一个非常简单的函数,用于响应客户端。只需立即使用
connected
事件进行响应,以确保清理cookies。
这就是全部了!这确实是一个非常简单的方法,虽然在cookie限制方面有一些缺陷,但我觉得一个有预警意识的开发者将能够判断是否使用这种技术。一旦转化为一个易于重复使用的函数,你可以轻松地在任何需要的地方实现它。