Files
Sébastien Stormacq e0f064a93e Refactor project directories (#621)
This PR refactors the project's directories.
As the number of source files grows, I created subdirectories to
separate the runtime itself, from its HTTP Client (`RuntimeClient`) and
local HTTP Server (`Lambda+LocalServer`).

The new layout looks like this:

```text
Sources
├── AWSLambdaRuntime
│   ├── FoundationSupport
│   │   ├── Context+Foundation.swift
│   │   ├── Lambda+JSON.swift
│   │   └── Vendored
│   │       ├── ByteBuffer-foundation.swift
│   │       └── JSON+ByteBuffer.swift
│   ├── HTTPClient
│   │   ├── ControlPlaneRequest.swift
│   │   ├── ControlPlaneRequestEncoder.swift
│   │   ├── LambdaRuntimeClient+ChannelHandler.swift
│   │   ├── LambdaRuntimeClient.swift
│   │   └── LambdaRuntimeClientProtocol.swift
│   ├── HTTPServer
│   │   ├── Lambda+LocalServer+Pool.swift
│   │   └── Lambda+LocalServer.swift
│   ├── Lambda.swift
│   ├── LambdaClock.swift
│   ├── LambdaContext.swift
│   ├── LambdaRequestID.swift
│   ├── LambdaResponseStreamWriter+Headers.swift
│   ├── LambdaRuntimeError.swift
│   ├── Runtime
│   │   ├── LambdaHandlers.swift
│   │   ├── LambdaRuntime+Codable.swift
│   │   ├── LambdaRuntime+Handler.swift
│   │   ├── LambdaRuntime+ServiceLifecycle.swift
│   │   └── LambdaRuntime.swift
│   ├── SendableMetatype.swift
│   ├── Utils.swift
│   └── Version.swift
└── MockServer
    └── MockHTTPServer.swift
```
2026-01-01 11:19:13 -05:00

134 lines
5.2 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
//
//===----------------------------------------------------------------------===//
import NIOCore
@available(LambdaSwift 2.0, *)
struct ControlPlaneRequestEncoder: _EmittingChannelHandler {
typealias OutboundOut = ByteBuffer
private var host: String
private var byteBuffer: ByteBuffer!
init(host: String) {
self.host = host
}
mutating func writeRequest(
_ request: ControlPlaneRequest,
context: ChannelHandlerContext,
promise: EventLoopPromise<Void>?
) {
self.byteBuffer.clear(minimumCapacity: self.byteBuffer.storageCapacity)
switch request {
case .next:
self.byteBuffer.writeString(.nextInvocationRequestLine)
self.byteBuffer.writeHostHeader(host: self.host)
self.byteBuffer.writeString(.userAgentHeader)
self.byteBuffer.writeString(.CRLF) // end of head
context.write(self.wrapOutboundOut(self.byteBuffer), promise: promise)
context.flush()
case .invocationResponse(let requestID, let payload):
let contentLength = payload?.readableBytes ?? 0
self.byteBuffer.writeInvocationResultRequestLine(requestID)
self.byteBuffer.writeHostHeader(host: self.host)
self.byteBuffer.writeString(.userAgentHeader)
self.byteBuffer.writeContentLengthHeader(length: contentLength)
self.byteBuffer.writeString(.CRLF) // end of head
if let payload = payload, contentLength > 0 {
context.write(self.wrapOutboundOut(self.byteBuffer), promise: nil)
context.write(self.wrapOutboundOut(payload), promise: promise)
} else {
context.write(self.wrapOutboundOut(self.byteBuffer), promise: promise)
}
context.flush()
case .invocationError(let requestID, let errorMessage):
let payload = errorMessage.toJSONBytes()
self.byteBuffer.writeInvocationErrorRequestLine(requestID)
self.byteBuffer.writeContentLengthHeader(length: payload.count)
self.byteBuffer.writeHostHeader(host: self.host)
self.byteBuffer.writeString(.userAgentHeader)
self.byteBuffer.writeString(.unhandledErrorHeader)
self.byteBuffer.writeString(.CRLF) // end of head
self.byteBuffer.writeBytes(payload)
context.write(self.wrapOutboundOut(self.byteBuffer), promise: promise)
context.flush()
case .initializationError(let errorMessage):
let payload = errorMessage.toJSONBytes()
self.byteBuffer.writeString(.runtimeInitErrorRequestLine)
self.byteBuffer.writeContentLengthHeader(length: payload.count)
self.byteBuffer.writeHostHeader(host: self.host)
self.byteBuffer.writeString(.userAgentHeader)
self.byteBuffer.writeString(.unhandledErrorHeader)
self.byteBuffer.writeString(.CRLF) // end of head
self.byteBuffer.writeBytes(payload)
context.write(self.wrapOutboundOut(self.byteBuffer), promise: promise)
context.flush()
}
}
mutating func writerAdded(context: ChannelHandlerContext) {
self.byteBuffer = context.channel.allocator.buffer(capacity: 256)
}
mutating func writerRemoved(context: ChannelHandlerContext) {
self.byteBuffer = nil
}
}
extension String {
static let CRLF: String = "\r\n"
static let userAgent = "Swift-Lambda/\(Version.current)"
static let userAgentHeader: String = "user-agent: \(userAgent)\r\n"
static let unhandledErrorHeader: String = "lambda-runtime-function-error-type: Unhandled\r\n"
static let nextInvocationRequestLine: String =
"GET /2018-06-01/runtime/invocation/next HTTP/1.1\r\n"
static let runtimeInitErrorRequestLine: String =
"POST /2018-06-01/runtime/init/error HTTP/1.1\r\n"
}
extension ByteBuffer {
fileprivate mutating func writeInvocationResultRequestLine(_ requestID: String) {
self.writeString("POST /2018-06-01/runtime/invocation/")
self.writeString(requestID)
self.writeString("/response HTTP/1.1\r\n")
}
fileprivate mutating func writeInvocationErrorRequestLine(_ requestID: String) {
self.writeString("POST /2018-06-01/runtime/invocation/")
self.writeString(requestID)
self.writeString("/error HTTP/1.1\r\n")
}
fileprivate mutating func writeHostHeader(host: String) {
self.writeString("host: ")
self.writeString(host)
self.writeString(.CRLF)
}
fileprivate mutating func writeContentLengthHeader(length: Int) {
self.writeString("content-length: ")
self.writeString("\(length)")
self.writeString(.CRLF)
}
}