Ars*_*min 6 zip amazon-web-services node.js aws-lambda aws-api-gateway
我有 Lambda,它对 DynamoDB 执行多次调用,创建一个大型字符串化 JSON 对象作为响应,并通过 API 网关传递到客户端应用程序。当然,API Gateway 具有“启用内容编码”选项集,并且所有数据都以压缩形式通过互联网传递。
问题在于 Lambda 响应本身未压缩,并且达到了 6MB 响应限制。是否可以压缩 Lambda 响应,然后以某种自然的方式在客户端解压缩?
我检查了 JSZip 和 ADM Zip 等 Node.js 库,令我惊讶的是,尽管它们允许在内存中输出解压数据,但它们不允许在内存中输入,如字符串、缓冲区或其他内容,而仅允许文件。Lambda 已经有一些与处理文件相关的限制和意外,因此我想避免以下冗余工作流程:
有没有更自然的方法来处理这个问题?
小智 10
尽管我同意使用分页或 S3 更适合大量数据,但您可以压缩从 Lambda 函数返回的响应,以避免在需要时达到 6MB 限制。
不幸的是,尽管 API Gateway 现在支持 gzip 等内容编码,但我的理解是,在 API Gateway 对其进行压缩之前,Lambda 函数的响应仍然需要低于 6MB 限制。我对此不是100%确定,所以如果我错了,请有人纠正我。
好消息是,node.js 标准库现在通过 zlib 模块内置了对 gzip 的支持。我用它来压缩刚刚超过 6MB 的响应,并将大小减少了大约 78%。这样做的缺点是我必须在 Angular 客户端应用程序中手动解压缩数据(为此我使用了 pako npm 库)。
像下面这样的东西对我来说效果很好(Typescript 4 + node.js 12):
import { APIGatewayEvent, APIGatewayProxyResult } from 'aws-lambda';
import { gzip } from 'zlib';
export const handler = async (event: APIGatewayEvent): Promise<APIGatewayProxyResult> => {
const response = { some: 'data' };
const gzippedResponse = await gzipString(JSON.stringify(response));
return {
statusCode: 200,
body: JSON.stringify({ data: gzippedResponse.toString('base64') }),
};
};
const gzipString = async (input: string): Promise<Buffer> => {
const buffer = Buffer.from(input);
return new Promise((resolve, reject) => gzip(buffer, (err, data) => {
if (err) {
reject(err);
}
resolve(data);
}));
};
Run Code Online (Sandbox Code Playgroud)
Ars*_*min -1
我遵循了下一个方法。对于后端:
const { deflate, unzip } = require("zlib");
const { promisify } = require("util");
const asyncDeflate = promisify(deflate);
async zip(object) {
return (await asyncDeflate(JSON.stringify(object))).toString("base64");`
}
Run Code Online (Sandbox Code Playgroud)
对于前端:
import * as pako from "pako";
export function unzip(base64str: string) {
const strData = atob(base64str);
// Convert binary string to character-number array
const charData = strData.split("").map((x) => { return x.charCodeAt(0); });
// Turn number array into byte-array
const binData = new Uint8Array(charData);
return JSON.parse(pako.inflate(binData, { to: "string" }));
}
Run Code Online (Sandbox Code Playgroud)
所以它与最近的答案非常相似。
| 归档时间: |
|
| 查看次数: |
7410 次 |
| 最近记录: |