mirror of
https://github.com/swift-server/swift-aws-lambda-runtime.git
synced 2026-05-03 07:22:27 +00:00
190eb81876
This PR builds on https://github.com/awslabs/swift-aws-lambda-runtime/pull/629 to add convenience structs (Handlers and Adapters) that are `Sendable` **Changes** - **Added Sendable adapter types**: Implemented `ClosureHandlerSendable` - a thread-safe version of existing closure handler that enforces `Sendable` conformance for concurrent execution environments - and added conditional conformance to `Sendable` for other Adapters when the Handler is `Sendable` - **Enhanced handler protocols for concurrency**: Extended handler protocols to support `Sendable` constraints and concurrent response writing through `LambdaResponseStreamWriter & Sendable`, enabling safe multi-threaded invocation processing - **Created comprehensive Lambda Managed Instances examples**: Built three demonstration functions showcasing concurrent execution capabilities, streaming responses, and background processing patterns specific to the new managed instances deployment model **Context** Lambda Managed Instances support multi-concurrent invocations where multiple invocations execute simultaneously within the same execution environment. The runtime now detects the configured concurrency level and launches the appropriate number of RICs to handle concurrent requests efficiently. When `AWS_LAMBDA_MAX_CONCURRENCY` is 1 or unset, the runtime maintains the existing single-threaded behaviour for optimal performance on traditional Lambda deployments. --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Sebastien Stormacq <stormacq@amazon.lu>
248 lines
9.8 KiB
Swift
248 lines
9.8 KiB
Swift
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This source file is part of the SwiftAWSLambdaRuntime open source project
|
|
//
|
|
// Copyright SwiftAWSLambdaRuntime project authors
|
|
// Copyright (c) Amazon.com, Inc. or its affiliates.
|
|
// Licensed under Apache License v2.0
|
|
//
|
|
// See LICENSE.txt for license information
|
|
// See CONTRIBUTORS.txt for the list of SwiftAWSLambdaRuntime project authors
|
|
//
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#if FoundationJSONSupport
|
|
import NIOCore
|
|
|
|
#if canImport(FoundationEssentials)
|
|
import FoundationEssentials
|
|
#else
|
|
import struct Foundation.Data
|
|
import class Foundation.JSONDecoder
|
|
import class Foundation.JSONEncoder
|
|
#endif
|
|
|
|
import Logging
|
|
|
|
public struct LambdaJSONEventDecoder: LambdaEventDecoder, Sendable {
|
|
@usableFromInline let jsonDecoder: JSONDecoder
|
|
|
|
@inlinable
|
|
public init(_ jsonDecoder: JSONDecoder) {
|
|
self.jsonDecoder = jsonDecoder
|
|
}
|
|
|
|
@inlinable
|
|
public func decode<Event>(_ type: Event.Type, from buffer: NIOCore.ByteBuffer) throws -> Event
|
|
where Event: Decodable {
|
|
try buffer.getJSONDecodable(
|
|
Event.self,
|
|
decoder: self.jsonDecoder,
|
|
at: buffer.readerIndex,
|
|
length: buffer.readableBytes
|
|
)! // must work, enough readable bytes
|
|
}
|
|
}
|
|
|
|
public struct LambdaJSONOutputEncoder<Output: Encodable>: LambdaOutputEncoder, Sendable {
|
|
@usableFromInline let jsonEncoder: JSONEncoder
|
|
|
|
@inlinable
|
|
public init(_ jsonEncoder: JSONEncoder) {
|
|
self.jsonEncoder = jsonEncoder
|
|
}
|
|
|
|
@inlinable
|
|
public func encode(_ value: Output, into buffer: inout ByteBuffer) throws {
|
|
try buffer.writeJSONEncodable(value, encoder: self.jsonEncoder)
|
|
}
|
|
}
|
|
|
|
@available(LambdaSwift 2.0, *)
|
|
extension LambdaCodableAdapter {
|
|
/// Initializes an instance given an encoder, decoder, and a handler with a non-`Void` output.
|
|
/// - Parameters:
|
|
/// - encoder: The encoder object that will be used to encode the generic `Output` obtained from the `handler`'s `outputWriter` into a `ByteBuffer`. By default, a JSONEncoder is used.
|
|
/// - decoder: The decoder object that will be used to decode the received `ByteBuffer` event into the generic `Event` type served to the `handler`. By default, a JSONDecoder is used.
|
|
/// - handler: The handler object.
|
|
public init(
|
|
encoder: JSONEncoder = JSONEncoder(),
|
|
decoder: JSONDecoder = JSONDecoder(),
|
|
handler: sending Handler
|
|
)
|
|
where
|
|
Output: Encodable,
|
|
Output == Handler.Output,
|
|
Encoder == LambdaJSONOutputEncoder<Output>,
|
|
Decoder == LambdaJSONEventDecoder
|
|
{
|
|
self.init(
|
|
encoder: LambdaJSONOutputEncoder(encoder),
|
|
decoder: LambdaJSONEventDecoder(decoder),
|
|
handler: handler
|
|
)
|
|
}
|
|
|
|
/// Initializes an instance given a decoder, and a handler with a `Void` output.
|
|
/// - Parameters:
|
|
/// - decoder: The decoder object that will be used to decode the received `ByteBuffer` event into the generic `Event` type served to the `handler`. By default, a JSONDecoder is used.
|
|
/// - handler: The handler object.
|
|
public init(
|
|
decoder: JSONDecoder = JSONDecoder(),
|
|
handler: sending Handler
|
|
)
|
|
where
|
|
Output == Void,
|
|
Handler.Output == Void,
|
|
Decoder == LambdaJSONEventDecoder,
|
|
Encoder == VoidEncoder
|
|
{
|
|
self.init(
|
|
decoder: LambdaJSONEventDecoder(decoder),
|
|
handler: handler
|
|
)
|
|
}
|
|
}
|
|
|
|
@available(LambdaSwift 2.0, *)
|
|
extension LambdaResponseStreamWriter {
|
|
/// Writes the HTTP status code and headers to the response stream.
|
|
///
|
|
/// This method serializes the status and headers as JSON and writes them to the stream,
|
|
/// followed by eight null bytes as a separator before the response body.
|
|
///
|
|
/// - Parameters:
|
|
/// - response: The status and headers response to write
|
|
/// - encoder: The encoder to use for serializing the response, use JSONEncoder by default
|
|
/// - Throws: An error if JSON serialization or writing fails
|
|
public func writeStatusAndHeaders(
|
|
_ response: StreamingLambdaStatusAndHeadersResponse,
|
|
encoder: JSONEncoder = JSONEncoder()
|
|
) async throws {
|
|
encoder.outputFormatting = .withoutEscapingSlashes
|
|
try await self.writeStatusAndHeaders(response, encoder: LambdaJSONOutputEncoder(encoder))
|
|
}
|
|
}
|
|
@available(LambdaSwift 2.0, *)
|
|
extension LambdaRuntime {
|
|
/// Initialize an instance with a `LambdaHandler` defined in the form of a closure **with a non-`Void` return type**.
|
|
/// - Parameters:
|
|
/// - decoder: The decoder object that will be used to decode the incoming `ByteBuffer` event into the generic `Event` type. `JSONDecoder()` used as default.
|
|
/// - encoder: The encoder object that will be used to encode the generic `Output` into a `ByteBuffer`. `JSONEncoder()` used as default.
|
|
/// - logger: The logger to use for the runtime. Defaults to a logger with label "LambdaRuntime".
|
|
/// - body: The handler in the form of a closure.
|
|
public convenience init<Event: Decodable, Output>(
|
|
decoder: JSONDecoder = JSONDecoder(),
|
|
encoder: JSONEncoder = JSONEncoder(),
|
|
logger: Logger = Logger(label: "LambdaRuntime"),
|
|
body: sending @escaping (Event, LambdaContext) async throws -> Output
|
|
)
|
|
where
|
|
Handler == LambdaCodableAdapter<
|
|
LambdaHandlerAdapter<Event, Output, ClosureHandler<Event, Output>>,
|
|
Event,
|
|
Output,
|
|
LambdaJSONEventDecoder,
|
|
LambdaJSONOutputEncoder<Output>
|
|
>
|
|
{
|
|
let handler = LambdaCodableAdapter(
|
|
encoder: encoder,
|
|
decoder: decoder,
|
|
handler: LambdaHandlerAdapter(handler: ClosureHandler(body: body))
|
|
)
|
|
|
|
self.init(handler: handler, logger: logger)
|
|
}
|
|
|
|
/// Initialize an instance with a `LambdaHandler` defined in the form of a closure **with a `Void` return type**.
|
|
/// - Parameter body: The handler in the form of a closure.
|
|
/// - Parameter decoder: The decoder object that will be used to decode the incoming `ByteBuffer` event into the generic `Event` type. `JSONDecoder()` used as default.
|
|
/// - Parameter logger: The logger to use for the runtime. Defaults to a logger with label "LambdaRuntime".
|
|
public convenience init<Event: Decodable>(
|
|
decoder: JSONDecoder = JSONDecoder(),
|
|
logger: Logger = Logger(label: "LambdaRuntime"),
|
|
body: sending @escaping (Event, LambdaContext) async throws -> Void
|
|
)
|
|
where
|
|
Handler == LambdaCodableAdapter<
|
|
LambdaHandlerAdapter<Event, Void, ClosureHandler<Event, Void>>,
|
|
Event,
|
|
Void,
|
|
LambdaJSONEventDecoder,
|
|
VoidEncoder
|
|
>
|
|
{
|
|
let handler = LambdaCodableAdapter(
|
|
decoder: LambdaJSONEventDecoder(decoder),
|
|
handler: LambdaHandlerAdapter(handler: ClosureHandler(body: body))
|
|
)
|
|
|
|
self.init(handler: handler, logger: logger)
|
|
}
|
|
|
|
/// Initialize an instance directly with a `LambdaHandler`, when `Event` is `Decodable` and `Output` is `Void`.
|
|
/// - Parameters:
|
|
/// - decoder: The decoder object that will be used to decode the incoming `ByteBuffer` event into the generic `Event` type. `JSONDecoder()` used as default.
|
|
/// - logger: The logger to use for the runtime. Defaults to a logger with label "LambdaRuntime".
|
|
/// - lambdaHandler: A type that conforms to the `LambdaHandler` protocol, whose `Event` is `Decodable` and `Output` is `Void`
|
|
public convenience init<Event: Decodable, LHandler: LambdaHandler>(
|
|
decoder: JSONDecoder = JSONDecoder(),
|
|
logger: Logger = Logger(label: "LambdaRuntime"),
|
|
lambdaHandler: sending LHandler
|
|
)
|
|
where
|
|
Handler == LambdaCodableAdapter<
|
|
LambdaHandlerAdapter<Event, Void, LHandler>,
|
|
Event,
|
|
Void,
|
|
LambdaJSONEventDecoder,
|
|
VoidEncoder
|
|
>,
|
|
LHandler.Event == Event,
|
|
LHandler.Output == Void
|
|
{
|
|
let handler = LambdaCodableAdapter(
|
|
decoder: LambdaJSONEventDecoder(decoder),
|
|
handler: LambdaHandlerAdapter(handler: lambdaHandler)
|
|
)
|
|
|
|
self.init(handler: handler, logger: logger)
|
|
}
|
|
|
|
/// Initialize an instance directly with a `LambdaHandler`, when `Event` is `Decodable` and `Output` is `Encodable`.
|
|
/// - Parameters:
|
|
/// - decoder: The decoder object that will be used to decode the incoming `ByteBuffer` event into the generic `Event` type. `JSONDecoder()` used as default.
|
|
/// - encoder: The encoder object that will be used to encode the generic `Output` into a `ByteBuffer`. `JSONEncoder()` used as default.
|
|
/// - logger: The logger to use for the runtime. Defaults to a logger with label "LambdaRuntime".
|
|
/// - lambdaHandler: A type that conforms to the `LambdaHandler` protocol, whose `Event` is `Decodable` and `Output` is `Encodable`
|
|
public convenience init<Event: Decodable, Output, LHandler: LambdaHandler>(
|
|
decoder: JSONDecoder = JSONDecoder(),
|
|
encoder: JSONEncoder = JSONEncoder(),
|
|
logger: Logger = Logger(label: "LambdaRuntime"),
|
|
lambdaHandler: sending LHandler
|
|
)
|
|
where
|
|
Handler == LambdaCodableAdapter<
|
|
LambdaHandlerAdapter<Event, Output, LHandler>,
|
|
Event,
|
|
Output,
|
|
LambdaJSONEventDecoder,
|
|
LambdaJSONOutputEncoder<Output>
|
|
>,
|
|
LHandler.Event == Event,
|
|
LHandler.Output == Output
|
|
{
|
|
let handler = LambdaCodableAdapter(
|
|
encoder: encoder,
|
|
decoder: decoder,
|
|
handler: LambdaHandlerAdapter(handler: lambdaHandler)
|
|
)
|
|
|
|
self.init(handler: handler, logger: logger)
|
|
}
|
|
}
|
|
#endif // trait: FoundationJSONSupport
|