Skip to content

Commit 7db7775

Browse files
committed
wip to add support for service lifecycle
1 parent 30bfa4f commit 7db7775

File tree

8 files changed

+231
-130
lines changed

8 files changed

+231
-130
lines changed

Examples/ServiceLifeCycle/Package.swift

+10-7
Original file line numberDiff line numberDiff line change
@@ -7,21 +7,24 @@ import PackageDescription
77
import struct Foundation.URL
88

99
let package = Package(
10-
name: "pg",
10+
name: "LambdaWithServiceLifecycle",
1111
platforms: [
12-
.macOS(.v15),
12+
.macOS(.v15)
1313
],
1414
dependencies: [
1515
.package(url: "https://github.com/vapor/postgres-nio.git", from: "1.23.0"),
1616
.package(url: "https://github.com/swift-server/swift-aws-lambda-runtime.git", branch: "main"),
1717
.package(url: "https://github.com/swift-server/swift-service-lifecycle.git", from: "2.6.3"),
1818
],
1919
targets: [
20-
.executableTarget(name: "pg", dependencies: [
21-
.product(name: "PostgresNIO", package: "postgres-nio"),
22-
.product(name: "AWSLambdaRuntimeService", package: "swift-aws-lambda-runtime"),
23-
.product(name: "ServiceLifecycle", package: "swift-service-lifecycle")
24-
]),
20+
.executableTarget(
21+
name: "LambdaWithServiceLifecycle",
22+
dependencies: [
23+
.product(name: "PostgresNIO", package: "postgres-nio"),
24+
.product(name: "AWSLambdaRuntimeService", package: "swift-aws-lambda-runtime"),
25+
.product(name: "ServiceLifecycle", package: "swift-service-lifecycle"),
26+
]
27+
)
2528
]
2629
)
2730

+61-24
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,17 @@
1+
//===----------------------------------------------------------------------===//
2+
//
3+
// This source file is part of the SwiftAWSLambdaRuntime open source project
4+
//
5+
// Copyright (c) 2025 Apple Inc. and the SwiftAWSLambdaRuntime project authors
6+
// Licensed under Apache License v2.0
7+
//
8+
// See LICENSE.txt for license information
9+
// See CONTRIBUTORS.txt for the list of SwiftAWSLambdaRuntime project authors
10+
//
11+
// SPDX-License-Identifier: Apache-2.0
12+
//
13+
//===----------------------------------------------------------------------===//
14+
115
import AWSLambdaRuntimeService
216
import Logging
317
import PostgresNIO
@@ -8,9 +22,50 @@ struct LambdaFunction {
822

923
static func main() async throws {
1024

11-
var logger = Logger(label: "Example")
25+
var logger = Logger(label: "ServiceLifecycleExample")
1226
logger.logLevel = .trace
1327

28+
let pgClient = try preparePostgresClient(
29+
host: Lambda.env("DB_HOST") ?? "localhost",
30+
user: Lambda.env("DB_USER") ?? "postgres",
31+
password: Lambda.env("DB_PASSWORD") ?? "secret",
32+
dbName: Lambda.env("DB_NAME") ?? "test"
33+
)
34+
35+
/// Instantiate LambdaRuntime with a closure handler implementing the business logic of the Lambda function
36+
let runtime = LambdaRuntimeService(logger: logger) { (event: String, context: LambdaContext) in
37+
38+
do {
39+
// Use initialized service within the handler
40+
// IMPORTANT - CURRENTLY WHEN THERE IS AN ERROR, THIS CALL HANGS !!!
41+
let rows = try await pgClient.query("SELECT id, username FROM users")
42+
for try await (id, username) in rows.decode((Int, String).self) {
43+
logger.debug("\(id) : \(username)")
44+
}
45+
} catch {
46+
logger.error("PG Error: \(error)")
47+
}
48+
}
49+
50+
/// Use ServiceLifecycle to manage the initialization and termination
51+
/// of the PGClient together with the LambdaRuntime
52+
let serviceGroup = ServiceGroup(
53+
services: [pgClient, runtime],
54+
gracefulShutdownSignals: [.sigterm], // add SIGINT for CTRL+C in local testing
55+
logger: logger
56+
)
57+
try await serviceGroup.run()
58+
59+
// perform any cleanup here
60+
}
61+
62+
private static func preparePostgresClient(
63+
host: String,
64+
user: String,
65+
password: String,
66+
dbName: String
67+
) throws -> PostgresClient {
68+
1469
var tlsConfig = TLSConfiguration.makeClientConfiguration()
1570
// Load the root certificate
1671
let rootCert = try NIOSSLCertificate.fromPEMBytes(Array(eu_central_1_bundle_pem.utf8))
@@ -22,32 +77,14 @@ struct LambdaFunction {
2277
tlsConfig.certificateVerification = .fullVerification
2378

2479
let config = PostgresClient.Configuration(
25-
host: Lambda.env("DB_HOST") ?? "localhost",
80+
host: host,
2681
port: 5432,
27-
username: Lambda.env("DB_USER") ?? "postgres",
28-
password: Lambda.env("DB_PASSWORD") ?? "",
29-
database: Lambda.env("DB_NAME") ?? "test",
82+
username: user,
83+
password: password,
84+
database: dbName,
3085
tls: .prefer(tlsConfig)
3186
)
3287

33-
let postgresClient = PostgresClient(configuration: config)
34-
35-
/// Instantiate LambdaRuntime with a closure handler implementing the business logic of the Lambda function
36-
let runtime = LambdaRuntimeService { (event: String, context: LambdaContext) in
37-
/// Use initialized service within the handler
38-
let rows = try await postgresClient.query("SELECT id, username FROM users")
39-
for try await (id, username) in rows.decode((Int, String).self) {
40-
logger.trace("\(id) : \(username)")
41-
}
42-
}
43-
44-
/// Use ServiceLifecycle to manage the initialization and termination
45-
/// of the services as well as the LambdaRuntime
46-
let serviceGroup = ServiceGroup(
47-
services: [postgresClient, runtime],
48-
gracefulShutdownSignals: [.sigterm],
49-
logger: logger
50-
)
51-
try await serviceGroup.run()
88+
return PostgresClient(configuration: config)
5289
}
5390
}
Original file line numberDiff line numberDiff line change
@@ -1,78 +1,95 @@
1+
//===----------------------------------------------------------------------===//
2+
//
3+
// This source file is part of the SwiftAWSLambdaRuntime open source project
4+
//
5+
// Copyright (c) 2025 Apple Inc. and the SwiftAWSLambdaRuntime project authors
6+
// Licensed under Apache License v2.0
7+
//
8+
// See LICENSE.txt for license information
9+
// See CONTRIBUTORS.txt for the list of SwiftAWSLambdaRuntime project authors
10+
//
11+
// SPDX-License-Identifier: Apache-2.0
12+
//
13+
//===----------------------------------------------------------------------===//
14+
15+
// you can download the root certificate for your RDS instance region from the following link:
16+
// https://docs.aws.amazon.com/AmazonRDS/latest/UserGuide/UsingWithRDS.SSL.html
17+
118
let eu_central_1_bundle_pem = """
2-
-----BEGIN CERTIFICATE-----
3-
MIICtDCCAjmgAwIBAgIQenQbcP/Zbj9JxvZ+jXbRnTAKBggqhkjOPQQDAzCBmTEL
4-
MAkGA1UEBhMCVVMxIjAgBgNVBAoMGUFtYXpvbiBXZWIgU2VydmljZXMsIEluYy4x
5-
EzARBgNVBAsMCkFtYXpvbiBSRFMxCzAJBgNVBAgMAldBMTIwMAYDVQQDDClBbWF6
6-
b24gUkRTIGV1LWNlbnRyYWwtMSBSb290IENBIEVDQzM4NCBHMTEQMA4GA1UEBwwH
7-
U2VhdHRsZTAgFw0yMTA1MjEyMjMzMjRaGA8yMTIxMDUyMTIzMzMyNFowgZkxCzAJ
8-
BgNVBAYTAlVTMSIwIAYDVQQKDBlBbWF6b24gV2ViIFNlcnZpY2VzLCBJbmMuMRMw
9-
EQYDVQQLDApBbWF6b24gUkRTMQswCQYDVQQIDAJXQTEyMDAGA1UEAwwpQW1hem9u
10-
IFJEUyBldS1jZW50cmFsLTEgUm9vdCBDQSBFQ0MzODQgRzExEDAOBgNVBAcMB1Nl
11-
YXR0bGUwdjAQBgcqhkjOPQIBBgUrgQQAIgNiAATlBHiEM9LoEb1Hdnd5j2VpCDOU
12-
5nGuFoBD8ROUCkFLFh5mHrHfPXwBc63heW9WrP3qnDEm+UZEUvW7ROvtWCTPZdLz
13-
Z4XaqgAlSqeE2VfUyZOZzBSgUUJk7OlznXfkCMOjQjBAMA8GA1UdEwEB/wQFMAMB
14-
Af8wHQYDVR0OBBYEFDT/ThjQZl42Nv/4Z/7JYaPNMly2MA4GA1UdDwEB/wQEAwIB
15-
hjAKBggqhkjOPQQDAwNpADBmAjEAnZWmSgpEbmq+oiCa13l5aGmxSlfp9h12Orvw
16-
Dq/W5cENJz891QD0ufOsic5oGq1JAjEAp5kSJj0MxJBTHQze1Aa9gG4sjHBxXn98
17-
4MP1VGsQuhfndNHQb4V0Au7OWnOeiobq
18-
-----END CERTIFICATE-----
19-
-----BEGIN CERTIFICATE-----
20-
MIIEBTCCAu2gAwIBAgIRAO8bekN7rUReuNPG8pSTKtEwDQYJKoZIhvcNAQELBQAw
21-
gZoxCzAJBgNVBAYTAlVTMSIwIAYDVQQKDBlBbWF6b24gV2ViIFNlcnZpY2VzLCBJ
22-
bmMuMRMwEQYDVQQLDApBbWF6b24gUkRTMQswCQYDVQQIDAJXQTEzMDEGA1UEAwwq
23-
QW1hem9uIFJEUyBldS1jZW50cmFsLTEgUm9vdCBDQSBSU0EyMDQ4IEcxMRAwDgYD
24-
VQQHDAdTZWF0dGxlMCAXDTIxMDUyMTIyMjM0N1oYDzIwNjEwNTIxMjMyMzQ3WjCB
25-
mjELMAkGA1UEBhMCVVMxIjAgBgNVBAoMGUFtYXpvbiBXZWIgU2VydmljZXMsIElu
26-
Yy4xEzARBgNVBAsMCkFtYXpvbiBSRFMxCzAJBgNVBAgMAldBMTMwMQYDVQQDDCpB
27-
bWF6b24gUkRTIGV1LWNlbnRyYWwtMSBSb290IENBIFJTQTIwNDggRzExEDAOBgNV
28-
BAcMB1NlYXR0bGUwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCTTYds
29-
Tray+Q9VA5j5jTh5TunHKFQzn68ZbOzdqaoi/Rq4ohfC0xdLrxCpfqn2TGDHN6Zi
30-
2qGK1tWJZEd1H0trhzd9d1CtGK+3cjabUmz/TjSW/qBar7e9MA67/iJ74Gc+Ww43
31-
A0xPNIWcL4aLrHaLm7sHgAO2UCKsrBUpxErOAACERScVYwPAfu79xeFcX7DmcX+e
32-
lIqY16pQAvK2RIzrekSYfLFxwFq2hnlgKHaVgZ3keKP+nmXcXmRSHQYUUr72oYNZ
33-
HcNYl2+gxCc9ccPEHM7xncVEKmb5cWEWvVoaysgQ+osi5f5aQdzgC2X2g2daKbyA
34-
XL/z5FM9GHpS5BJjAgMBAAGjQjBAMA8GA1UdEwEB/wQFMAMBAf8wHQYDVR0OBBYE
35-
FBDAiJ7Py9/A9etNa/ebOnx5l5MGMA4GA1UdDwEB/wQEAwIBhjANBgkqhkiG9w0B
36-
AQsFAAOCAQEALMh/+81fFPdJV/RrJUeoUvFCGMp8iaANu97NpeJyKitNOv7RoeVP
37-
WjivS0KcCqZaDBs+p6IZ0sLI5ZH098LDzzytcfZg0PsGqUAb8a0MiU/LfgDCI9Ee
38-
jsOiwaFB8k0tfUJK32NPcIoQYApTMT2e26lPzYORSkfuntme2PTHUnuC7ikiQrZk
39-
P+SZjWgRuMcp09JfRXyAYWIuix4Gy0eZ4rpRuaTK6mjAb1/LYoNK/iZ/gTeIqrNt
40-
l70OWRsWW8jEmSyNTIubGK/gGGyfuZGSyqoRX6OKHESkP6SSulbIZHyJ5VZkgtXo
41-
2XvyRyJ7w5pFyoofrL3Wv0UF8yt/GDszmg==
42-
-----END CERTIFICATE-----
43-
-----BEGIN CERTIFICATE-----
44-
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
76-
89cEIGdbjsA=
77-
-----END CERTIFICATE-----
78-
"""
19+
-----BEGIN CERTIFICATE-----
20+
MIICtDCCAjmgAwIBAgIQenQbcP/Zbj9JxvZ+jXbRnTAKBggqhkjOPQQDAzCBmTEL
21+
MAkGA1UEBhMCVVMxIjAgBgNVBAoMGUFtYXpvbiBXZWIgU2VydmljZXMsIEluYy4x
22+
EzARBgNVBAsMCkFtYXpvbiBSRFMxCzAJBgNVBAgMAldBMTIwMAYDVQQDDClBbWF6
23+
b24gUkRTIGV1LWNlbnRyYWwtMSBSb290IENBIEVDQzM4NCBHMTEQMA4GA1UEBwwH
24+
U2VhdHRsZTAgFw0yMTA1MjEyMjMzMjRaGA8yMTIxMDUyMTIzMzMyNFowgZkxCzAJ
25+
BgNVBAYTAlVTMSIwIAYDVQQKDBlBbWF6b24gV2ViIFNlcnZpY2VzLCBJbmMuMRMw
26+
EQYDVQQLDApBbWF6b24gUkRTMQswCQYDVQQIDAJXQTEyMDAGA1UEAwwpQW1hem9u
27+
IFJEUyBldS1jZW50cmFsLTEgUm9vdCBDQSBFQ0MzODQgRzExEDAOBgNVBAcMB1Nl
28+
YXR0bGUwdjAQBgcqhkjOPQIBBgUrgQQAIgNiAATlBHiEM9LoEb1Hdnd5j2VpCDOU
29+
5nGuFoBD8ROUCkFLFh5mHrHfPXwBc63heW9WrP3qnDEm+UZEUvW7ROvtWCTPZdLz
30+
Z4XaqgAlSqeE2VfUyZOZzBSgUUJk7OlznXfkCMOjQjBAMA8GA1UdEwEB/wQFMAMB
31+
Af8wHQYDVR0OBBYEFDT/ThjQZl42Nv/4Z/7JYaPNMly2MA4GA1UdDwEB/wQEAwIB
32+
hjAKBggqhkjOPQQDAwNpADBmAjEAnZWmSgpEbmq+oiCa13l5aGmxSlfp9h12Orvw
33+
Dq/W5cENJz891QD0ufOsic5oGq1JAjEAp5kSJj0MxJBTHQze1Aa9gG4sjHBxXn98
34+
4MP1VGsQuhfndNHQb4V0Au7OWnOeiobq
35+
-----END CERTIFICATE-----
36+
-----BEGIN CERTIFICATE-----
37+
MIIEBTCCAu2gAwIBAgIRAO8bekN7rUReuNPG8pSTKtEwDQYJKoZIhvcNAQELBQAw
38+
gZoxCzAJBgNVBAYTAlVTMSIwIAYDVQQKDBlBbWF6b24gV2ViIFNlcnZpY2VzLCBJ
39+
bmMuMRMwEQYDVQQLDApBbWF6b24gUkRTMQswCQYDVQQIDAJXQTEzMDEGA1UEAwwq
40+
QW1hem9uIFJEUyBldS1jZW50cmFsLTEgUm9vdCBDQSBSU0EyMDQ4IEcxMRAwDgYD
41+
VQQHDAdTZWF0dGxlMCAXDTIxMDUyMTIyMjM0N1oYDzIwNjEwNTIxMjMyMzQ3WjCB
42+
mjELMAkGA1UEBhMCVVMxIjAgBgNVBAoMGUFtYXpvbiBXZWIgU2VydmljZXMsIElu
43+
Yy4xEzARBgNVBAsMCkFtYXpvbiBSRFMxCzAJBgNVBAgMAldBMTMwMQYDVQQDDCpB
44+
bWF6b24gUkRTIGV1LWNlbnRyYWwtMSBSb290IENBIFJTQTIwNDggRzExEDAOBgNV
45+
BAcMB1NlYXR0bGUwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCTTYds
46+
Tray+Q9VA5j5jTh5TunHKFQzn68ZbOzdqaoi/Rq4ohfC0xdLrxCpfqn2TGDHN6Zi
47+
2qGK1tWJZEd1H0trhzd9d1CtGK+3cjabUmz/TjSW/qBar7e9MA67/iJ74Gc+Ww43
48+
A0xPNIWcL4aLrHaLm7sHgAO2UCKsrBUpxErOAACERScVYwPAfu79xeFcX7DmcX+e
49+
lIqY16pQAvK2RIzrekSYfLFxwFq2hnlgKHaVgZ3keKP+nmXcXmRSHQYUUr72oYNZ
50+
HcNYl2+gxCc9ccPEHM7xncVEKmb5cWEWvVoaysgQ+osi5f5aQdzgC2X2g2daKbyA
51+
XL/z5FM9GHpS5BJjAgMBAAGjQjBAMA8GA1UdEwEB/wQFMAMBAf8wHQYDVR0OBBYE
52+
FBDAiJ7Py9/A9etNa/ebOnx5l5MGMA4GA1UdDwEB/wQEAwIBhjANBgkqhkiG9w0B
53+
AQsFAAOCAQEALMh/+81fFPdJV/RrJUeoUvFCGMp8iaANu97NpeJyKitNOv7RoeVP
54+
WjivS0KcCqZaDBs+p6IZ0sLI5ZH098LDzzytcfZg0PsGqUAb8a0MiU/LfgDCI9Ee
55+
jsOiwaFB8k0tfUJK32NPcIoQYApTMT2e26lPzYORSkfuntme2PTHUnuC7ikiQrZk
56+
P+SZjWgRuMcp09JfRXyAYWIuix4Gy0eZ4rpRuaTK6mjAb1/LYoNK/iZ/gTeIqrNt
57+
l70OWRsWW8jEmSyNTIubGK/gGGyfuZGSyqoRX6OKHESkP6SSulbIZHyJ5VZkgtXo
58+
2XvyRyJ7w5pFyoofrL3Wv0UF8yt/GDszmg==
59+
-----END CERTIFICATE-----
60+
-----BEGIN CERTIFICATE-----
61+
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
93+
89cEIGdbjsA=
94+
-----END CERTIFICATE-----
95+
"""

Package.swift

+4-4
Original file line numberDiff line numberDiff line change
@@ -8,17 +8,17 @@ let package = Package(
88
products: [
99
// this library exports `AWSLambdaRuntimeCore` and adds Foundation convenience methods
1010
.library(name: "AWSLambdaRuntime", targets: ["AWSLambdaRuntime"]),
11-
11+
1212
// this library exports `AWSLambdaRuntime` and adds conformances to `Service` from Swift Service Lifecycle
1313
.library(name: "AWSLambdaRuntimeService", targets: ["AWSLambdaRuntimeService"]),
14-
14+
1515
// this has all the main functionality for lambda and it does not link Foundation
1616
.library(name: "AWSLambdaRuntimeCore", targets: ["AWSLambdaRuntimeCore"]),
17-
17+
1818
// plugin to package the lambda, creating an archive that can be uploaded to AWS
1919
// requires Linux or at least macOS v15
2020
.plugin(name: "AWSLambdaPackager", targets: ["AWSLambdaPackager"]),
21-
21+
2222
// for testing only
2323
.library(name: "AWSLambdaTesting", targets: ["AWSLambdaTesting"]),
2424
],

Sources/AWSLambdaRuntimeCore/Lambda.swift

+18-1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ import Dispatch
1616
import Logging
1717
import NIOCore
1818
import NIOPosix
19+
import Synchronization
1920

2021
#if os(macOS)
2122
import Darwin.C
@@ -30,16 +31,28 @@ import ucrt
3031
#endif
3132

3233
public enum Lambda {
34+
35+
// allow to gracefully shitdown the runtime client loop
36+
// this supports gracefull shutdown of the Lambda runtime when integarted with Swift ServiceLifeCycle
37+
private static let cancelled: Mutex<Bool> = Mutex(false)
38+
public static func cancel() {
39+
Lambda.cancelled.withLock {
40+
$0 = true
41+
}
42+
}
3343
package static func runLoop<RuntimeClient: LambdaRuntimeClientProtocol, Handler>(
3444
runtimeClient: RuntimeClient,
3545
handler: Handler,
3646
logger: Logger
3747
) async throws where Handler: StreamingLambdaHandler {
3848
var handler = handler
3949

40-
while !Task.isCancelled {
50+
var cancelled: Bool = Lambda.cancelled.withLock { $0 }
51+
while !Task.isCancelled && !cancelled {
52+
logger.trace("Waiting for next invocation")
4153
let (invocation, writer) = try await runtimeClient.nextInvocation()
4254

55+
logger.trace("Received invocation : \(invocation.metadata.requestID)")
4356
do {
4457
try await handler.handle(
4558
invocation.event,
@@ -56,7 +69,11 @@ public enum Lambda {
5669
try await writer.reportError(error)
5770
continue
5871
}
72+
73+
logger.trace("Completed invocation : \(invocation.metadata.requestID)")
74+
cancelled = Lambda.cancelled.withLock { $0 }
5975
}
76+
logger.trace("Lambda runLoop() \(cancelled ? "cancelled" : "completed")")
6077
}
6178

6279
/// The default EventLoop the Lambda is scheduled on.

Sources/AWSLambdaRuntimeCore/LambdaRuntime.swift

+5
Original file line numberDiff line numberDiff line change
@@ -107,4 +107,9 @@ public final class LambdaRuntime<Handler>: @unchecked Sendable where Handler: St
107107
#endif
108108
}
109109
}
110+
111+
/// Cancels the runtime client loop.
112+
public func cancel() {
113+
Lambda.cancel()
114+
}
110115
}

0 commit comments

Comments
 (0)