UNPKG

faastjs

Version:

Serverless batch computing made simple.

177 lines 26.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.receiveMessages = exports.processAwsErrorMessage = exports.createSQSQueue = exports.publishFunctionCallMessage = exports.sendResponseQueueMessage = exports.createSNSTopic = void 0; const abort_controller_1 = require("@aws-sdk/abort-controller"); const error_1 = require("../error"); const log_1 = require("../log"); const serialize_1 = require("../serialize"); const shared_1 = require("../shared"); const throttle_1 = require("../throttle"); const wrapper_1 = require("../wrapper"); async function createSNSTopic(sns, Name) { const topic = await sns.createTopic({ Name }); return topic.TopicArn; } exports.createSNSTopic = createSNSTopic; function countRequests(bytes) { return Math.ceil(bytes / (64 * 1024)); } async function sendResponseQueueMessage(sqs, QueueUrl, message, cc) { try { const request = { QueueUrl, MessageBody: (0, serialize_1.serialize)(message) }; const len = request.MessageBody.length; if (len > 262144) { throw new Error(`Response length (${len} bytes) exceeds limit of 256KiB`); } await sqs.sendMessage(request); } catch (err) { log_1.log.warn(`sendResponseQueueMessage failed: ${err}`); const errorResponse = (0, wrapper_1.createErrorResponse)(err, cc); const errorRequest = { QueueUrl, MessageBody: (0, serialize_1.serialize)(errorResponse) }; try { await sqs.sendMessage(errorRequest); } catch (errorResponseError) { log_1.log.warn(`Error sending error response to response queue: ${errorResponseError}`); } } } exports.sendResponseQueueMessage = sendResponseQueueMessage; function publishFunctionCallMessage(sns, TopicArn, message, metrics) { const serialized = (0, serialize_1.serialize)(message); metrics.sns64kRequests += countRequests(serialized.length); return (0, throttle_1.retryOp)((err, n) => n < 6 && err?.message?.match(/does not exist/), () => sns.publish({ TopicArn, Message: serialized })); } exports.publishFunctionCallMessage = publishFunctionCallMessage; async function createSQSQueue(QueueName, VTimeout, sqs) { try { const createQueueRequest = { QueueName, Attributes: { VisibilityTimeout: `${VTimeout}` } }; const response = await sqs.createQueue(createQueueRequest); const QueueUrl = response.QueueUrl; const arnResponse = await sqs.getQueueAttributes({ QueueUrl, AttributeNames: ["QueueArn"] }); const QueueArn = arnResponse.Attributes?.QueueArn; return { QueueUrl, QueueArn }; } catch (err) { throw new error_1.FaastError(err, "create sqs queue"); } } exports.createSQSQueue = createSQSQueue; /* c8 ignore next */ function processAwsErrorMessage(message) { let err = new error_1.FaastError(message); if (message?.match(/Process exited before completing/) || message?.match(/signal: killed/) || message?.match(/Runtime exited/)) { err = new error_1.FaastError({ cause: err, name: error_1.FaastErrorNames.EMEMORY }, "possibly out of memory"); } else if (message?.match(/time/)) { err = new error_1.FaastError({ cause: err, name: error_1.FaastErrorNames.ETIMEOUT }, "timeout"); } else if (message?.match(/EventAgeExceeded/)) { err = new error_1.FaastError({ cause: err, name: error_1.FaastErrorNames.ECONCURRENCY }, "concurrency limit exceeded"); } return err; } exports.processAwsErrorMessage = processAwsErrorMessage; async function receiveMessages(sqs, ResponseQueueUrl, metrics, cancel) { try { const MaxNumberOfMessages = 10; const abortController = new abort_controller_1.AbortController(); const request = sqs.receiveMessage({ QueueUrl: ResponseQueueUrl, WaitTimeSeconds: 20, MaxNumberOfMessages, MessageAttributeNames: ["All"], AttributeNames: ["SentTimestamp"] }, { abortSignal: abortController.signal }); const response = await Promise.race([request, cancel]); if (!response) { abortController.abort(); return { Messages: [] }; } const { Messages = [] } = response; const receivedBytes = Messages.reduce((sum, m) => sum + (m.Body?.length ?? 0), 0); metrics.outboundBytes += receivedBytes; const inferredSqsRequestsReceived = countRequests(receivedBytes); // Each request is counted as both sent and received. metrics.sqs64kRequests += inferredSqsRequestsReceived * 2; if (Messages.length > 0) { sqs.deleteMessageBatch({ QueueUrl: ResponseQueueUrl, Entries: Messages.map(m => ({ Id: m.MessageId, ReceiptHandle: m.ReceiptHandle })) }).catch(_ => { }); metrics.sqs64kRequests++; } return { Messages: Messages.map(processIncomingQueueMessage).filter(shared_1.defined), isFullMessageBatch: Messages.length === MaxNumberOfMessages }; } catch (err) { throw new error_1.FaastError(err, "receiveMessages"); } } exports.receiveMessages = receiveMessages; function processIncomingQueueMessage(m) { // AWS Lambda Destinations // (https://aws.amazon.com/blogs/compute/introducing-aws-lambda-destinations/) // are used to route failures to the response queue. These // messages are generated by AWS Lambda and are constrained to the format it // provides. const raw = (0, serialize_1.deserialize)(m.Body); if (raw.responseContext) { const message = raw; const snsMessage = message.requestPayload; const record = snsMessage.Records[0]; const sCall = (0, serialize_1.deserialize)(record.Sns.Message); let error; const destinationError = message.responsePayload; if (destinationError) { error = processAwsErrorMessage(destinationError.errorMessage); error.stack = destinationError.stackTrace?.join("\n"); } else { error = processAwsErrorMessage(message.requestContext.condition); } const executionId = message.requestContext.requestId; return { ...(0, wrapper_1.createErrorResponse)(error, { call: sCall, startTime: new Date(record.Sns.Timestamp).getTime(), executionId }), timestamp: new Date(message.timestamp).getTime() }; } else { const message = raw; switch (message.kind) { case "promise": case "iterator": message.timestamp = Number(m.Attributes.SentTimestamp); break; case "cpumetrics": break; case "functionstarted": break; default: { console.warn(`Unknown message received from response queue`); } } return raw; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws-queue.js","sourceRoot":"","sources":["../../../src/aws/aws-queue.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbortController } from \"@aws-sdk/abort-controller\";\nimport { SNS } from \"@aws-sdk/client-sns\";\nimport { CreateQueueRequest, SQS, Message as SQSMessage } from \"@aws-sdk/client-sqs\";\nimport { SNSEvent } from \"aws-lambda/trigger/sns\";\nimport { FaastError, FaastErrorNames } from \"../error\";\nimport { log } from \"../log\";\nimport { Message, PollResult } from \"../provider\";\nimport { deserialize, serialize } from \"../serialize\";\nimport { defined } from \"../shared\";\nimport { retryOp } from \"../throttle\";\nimport { CallingContext, FunctionCall, createErrorResponse } from \"../wrapper\";\nimport { AwsMetrics } from \"./aws-faast\";\n\nexport async function createSNSTopic(sns: SNS, Name: string) {\n    const topic = await sns.createTopic({ Name });\n    return topic.TopicArn!;\n}\n\nfunction countRequests(bytes: number) {\n    return Math.ceil(bytes / (64 * 1024));\n}\n\nexport async function sendResponseQueueMessage(\n    sqs: SQS,\n    QueueUrl: string,\n    message: Message,\n    cc: CallingContext\n) {\n    try {\n        const request = { QueueUrl, MessageBody: serialize(message) };\n        const len = request.MessageBody.length;\n        if (len > 262144) {\n            throw new Error(`Response length (${len} bytes) exceeds limit of 256KiB`);\n        }\n        await sqs.sendMessage(request);\n    } catch (err) {\n        log.warn(`sendResponseQueueMessage failed: ${err}`);\n        const errorResponse = createErrorResponse(err, cc);\n        const errorRequest = { QueueUrl, MessageBody: serialize(errorResponse) };\n        try {\n            await sqs.sendMessage(errorRequest);\n        } catch (errorResponseError: any) {\n            log.warn(\n                `Error sending error response to response queue: ${errorResponseError}`\n            );\n        }\n    }\n}\n\nexport function publishFunctionCallMessage(\n    sns: SNS,\n    TopicArn: string,\n    message: FunctionCall,\n    metrics: AwsMetrics\n) {\n    const serialized = serialize(message);\n    metrics.sns64kRequests += countRequests(serialized.length);\n    return retryOp(\n        (err, n) => n < 6 && err?.message?.match(/does not exist/),\n        () => sns.publish({ TopicArn, Message: serialized })\n    );\n}\n\nexport async function createSQSQueue(QueueName: string, VTimeout: number, sqs: SQS) {\n    try {\n        const createQueueRequest: CreateQueueRequest = {\n            QueueName,\n            Attributes: {\n                VisibilityTimeout: `${VTimeout}`\n            }\n        };\n        const response = await sqs.createQueue(createQueueRequest);\n        const QueueUrl = response.QueueUrl!;\n        const arnResponse = await sqs.getQueueAttributes({\n            QueueUrl,\n            AttributeNames: [\"QueueArn\"]\n        });\n        const QueueArn = arnResponse.Attributes?.QueueArn;\n        return { QueueUrl, QueueArn };\n    } catch (err: any) {\n        throw new FaastError(err, \"create sqs queue\");\n    }\n}\n\n/* c8 ignore next  */\nexport function processAwsErrorMessage(message?: string): Error {\n    let err = new FaastError(message);\n    if (\n        message?.match(/Process exited before completing/) ||\n        message?.match(/signal: killed/) ||\n        message?.match(/Runtime exited/)\n    ) {\n        err = new FaastError(\n            { cause: err, name: FaastErrorNames.EMEMORY },\n            \"possibly out of memory\"\n        );\n    } else if (message?.match(/time/)) {\n        err = new FaastError({ cause: err, name: FaastErrorNames.ETIMEOUT }, \"timeout\");\n    } else if (message?.match(/EventAgeExceeded/)) {\n        err = new FaastError(\n            { cause: err, name: FaastErrorNames.ECONCURRENCY },\n            \"concurrency limit exceeded\"\n        );\n    }\n    return err;\n}\n\nexport async function receiveMessages(\n    sqs: SQS,\n    ResponseQueueUrl: string,\n    metrics: AwsMetrics,\n    cancel: Promise<void>\n): Promise<PollResult> {\n    try {\n        const MaxNumberOfMessages = 10;\n        const abortController = new AbortController();\n        const request = sqs.receiveMessage(\n            {\n                QueueUrl: ResponseQueueUrl!,\n                WaitTimeSeconds: 20,\n                MaxNumberOfMessages,\n                MessageAttributeNames: [\"All\"],\n                AttributeNames: [\"SentTimestamp\"]\n            },\n            { abortSignal: abortController.signal }\n        );\n        const response = await Promise.race([request, cancel]);\n        if (!response) {\n            abortController.abort();\n            return { Messages: [] };\n        }\n\n        const { Messages = [] } = response;\n        const receivedBytes = Messages.reduce((sum, m) => sum + (m.Body?.length ?? 0), 0);\n        metrics.outboundBytes += receivedBytes;\n        const inferredSqsRequestsReceived = countRequests(receivedBytes);\n        // Each request is counted as both sent and received.\n        metrics.sqs64kRequests += inferredSqsRequestsReceived * 2;\n\n        if (Messages.length > 0) {\n            sqs.deleteMessageBatch({\n                QueueUrl: ResponseQueueUrl!,\n                Entries: Messages.map(m => ({\n                    Id: m.MessageId!,\n                    ReceiptHandle: m.ReceiptHandle!\n                }))\n            }).catch(_ => {});\n            metrics.sqs64kRequests++;\n        }\n        return {\n            Messages: Messages.map(processIncomingQueueMessage).filter(defined),\n            isFullMessageBatch: Messages.length === MaxNumberOfMessages\n        };\n    } catch (err: any) {\n        throw new FaastError(err, \"receiveMessages\");\n    }\n}\n\ninterface LambdaDestinationError {\n    errorMessage: string;\n    errorType?: string;\n    stackTrace?: string[];\n}\n\ninterface LambdaDestinationMessage {\n    version: string;\n    timestamp: string;\n    requestContext: {\n        requestId: string;\n        functionArn: string;\n        condition: \"RetriesExhausted\" | \"Success\";\n        approximateInvokeCount: number;\n    };\n    requestPayload: object;\n    responseContext: {\n        statusCode: number;\n        executedVersion: string;\n        functionError?: string;\n    };\n    responsePayload: LambdaDestinationError | object;\n}\n\nfunction processIncomingQueueMessage(m: SQSMessage): Message | void {\n    // AWS Lambda Destinations\n    // (https://aws.amazon.com/blogs/compute/introducing-aws-lambda-destinations/)\n    // are used to route failures to the response queue. These\n    // messages are generated by AWS Lambda and are constrained to the format it\n    // provides.\n    const raw = deserialize(m.Body!);\n    if (raw.responseContext) {\n        const message = raw as LambdaDestinationMessage;\n        const snsMessage = message.requestPayload as SNSEvent;\n        const record = snsMessage.Records[0];\n        const sCall: FunctionCall = deserialize(record.Sns.Message);\n        let error: Error | undefined;\n        const destinationError = message.responsePayload as LambdaDestinationError;\n        if (destinationError) {\n            error = processAwsErrorMessage(destinationError.errorMessage);\n            error.stack = destinationError.stackTrace?.join(\"\\n\");\n        } else {\n            error = processAwsErrorMessage(message.requestContext.condition);\n        }\n        const executionId = message.requestContext.requestId;\n        return {\n            ...createErrorResponse(error, {\n                call: sCall,\n                startTime: new Date(record.Sns.Timestamp).getTime(),\n                executionId\n            }),\n            timestamp: new Date(message.timestamp).getTime()\n        };\n    } else {\n        const message = raw as Message;\n        switch (message.kind) {\n            case \"promise\":\n            case \"iterator\":\n                message.timestamp = Number(m.Attributes!.SentTimestamp);\n                break;\n            case \"cpumetrics\":\n                break;\n            case \"functionstarted\":\n                break;\n            default: {\n                console.warn(`Unknown message received from response queue`);\n            }\n        }\n        return raw;\n    }\n}\n"]}