Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[core] Only one LambdaRuntime.run() can be called at a time (fix #507) #508

Open
wants to merge 26 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
456569a
make LambdaRuntime a singleton without breaking the API
sebsto Mar 14, 2025
a63a639
fix license header
sebsto Mar 14, 2025
40b6127
convert Mutex to NIOLockedValueBox
sebsto Mar 14, 2025
d474eba
Replace NIOLockedValueBox with Mutex
sebsto Mar 15, 2025
46e76f3
revert replacing NIOLockedValueBox by Mutex
sebsto Mar 15, 2025
299b9bc
remove typed throw (workaround for https://github.com/swiftlang/swift…
sebsto Mar 15, 2025
e3a3851
fix integration tests
sebsto Mar 15, 2025
4ebf24f
Replace NIOLockedValueBox with Mutex
sebsto Mar 15, 2025
f3e65ac
Merge branch 'main' into sebsto/fix_507
sebsto Mar 18, 2025
3aee427
use Atomic instead of Mutex.
sebsto Mar 19, 2025
e445f90
Merge branch 'sebsto/fix_507' of github.com:sebsto/swift-aws-lambda-r…
sebsto Mar 19, 2025
0509eaa
revert `try` on `runtime.init()` in doc
sebsto Mar 19, 2025
aa6395f
revert unwanted change
sebsto Mar 19, 2025
066ab76
revert unwanted change
sebsto Mar 19, 2025
e01b25d
swift-format
sebsto Mar 19, 2025
23c4b64
Merge branch 'main' into sebsto/fix_507
sebsto Mar 20, 2025
57e2396
Update Sources/AWSLambdaRuntime/LambdaRuntime.swift
sebsto Mar 21, 2025
85c988d
Update Sources/AWSLambdaRuntime/LambdaRuntime.swift
sebsto Mar 21, 2025
ab80580
Update Sources/AWSLambdaRuntime/LambdaRuntimeError.swift
sebsto Mar 21, 2025
10304ce
Update Sources/AWSLambdaRuntime/LambdaRuntimeError.swift
sebsto Mar 21, 2025
c1e68fc
Update Tests/AWSLambdaRuntimeTests/LambdaRuntimeTests.swift
sebsto Mar 21, 2025
918492a
Update Tests/AWSLambdaRuntimeTests/LambdaRuntimeTests.swift
sebsto Mar 21, 2025
6ada041
Update Tests/AWSLambdaRuntimeTests/LambdaRuntimeTests.swift
sebsto Mar 21, 2025
06df831
add package visibility to Code
sebsto Mar 21, 2025
50b9df8
swift format
sebsto Mar 24, 2025
fa0fd88
remove print statement
sebsto Mar 24, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
25 changes: 24 additions & 1 deletion Sources/AWSLambdaRuntime/LambdaRuntime.swift
Original file line number Diff line number Diff line change
Expand Up @@ -15,16 +15,21 @@
import Logging
import NIOConcurrencyHelpers
import NIOCore
import Synchronization

#if canImport(FoundationEssentials)
import FoundationEssentials
#else
import Foundation
#endif

// This is our gardian to ensure only one LambdaRuntime is running at the time
// We use an Atomic here to ensure thread safety
private let _isRunning = Atomic<Bool>(false)

// We need `@unchecked` Sendable here, as `NIOLockedValueBox` does not understand `sending` today.
// We don't want to use `NIOLockedValueBox` here anyway. We would love to use Mutex here, but this
// sadly crashes the compiler today.
// sadly crashes the compiler today (on Linux).
public final class LambdaRuntime<Handler>: @unchecked Sendable where Handler: StreamingLambdaHandler {
// TODO: We want to change this to Mutex as soon as this doesn't crash the Swift compiler on Linux anymore
let handlerMutex: NIOLockedValueBox<Handler?>
Expand All @@ -48,7 +53,25 @@ public final class LambdaRuntime<Handler>: @unchecked Sendable where Handler: St
self.logger.debug("LambdaRuntime initialized")
}

/// Make sure only one run() is called at a time
public func run() async throws {

// we use an atomic global variable to ensure only one LambdaRuntime is running at the time
let (_, original) = _isRunning.compareExchange(expected: false, desired: true, ordering: .acquiringAndReleasing)

// if the original value was already true, run() is already running
if original {
throw LambdaRuntimeError(code: .moreThanOneLambdaRuntimeInstance)
}

defer {
_isRunning.store(false, ordering: .releasing)
}

try await self._run()
}

private func _run() async throws {
let handler = self.handlerMutex.withLockedValue { handler in
let result = handler
handler = nil
Expand Down
13 changes: 9 additions & 4 deletions Sources/AWSLambdaRuntime/LambdaRuntimeError.swift
Original file line number Diff line number Diff line change
Expand Up @@ -12,8 +12,10 @@
//
//===----------------------------------------------------------------------===//

package struct LambdaRuntimeError: Error {
package enum Code {
public struct LambdaRuntimeError: Error {
package enum Code: Sendable {

/// internal error codes for LambdaRuntimeClient
case closingRuntimeClient

case connectionToControlPlaneLost
Expand All @@ -32,14 +34,17 @@ package struct LambdaRuntimeError: Error {
case missingLambdaRuntimeAPIEnvironmentVariable
case runtimeCanOnlyBeStartedOnce
case invalidPort

/// public error codes for LambdaRuntime
case moreThanOneLambdaRuntimeInstance
}

package init(code: Code, underlying: (any Error)? = nil) {
self.code = code
self.underlying = underlying
}

package var code: Code
package var underlying: (any Error)?
var code: Code
public var underlying: (any Error)?

}
84 changes: 84 additions & 0 deletions Tests/AWSLambdaRuntimeTests/LambdaRuntimeTests.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
//===----------------------------------------------------------------------===//
//
// This source file is part of the SwiftAWSLambdaRuntime open source project
//
// Copyright (c) 2025 Apple Inc. and the SwiftAWSLambdaRuntime project authors
// 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 Foundation
import Logging
import NIOCore
import Synchronization
import Testing

@testable import AWSLambdaRuntime

@Suite("LambdaRuntimeTests")
struct LambdaRuntimeTests {

@Test("LambdaRuntime can only be run once")
func testLambdaRuntimerunOnce() async throws {

// First runtime
let runtime1 = LambdaRuntime(
handler: MockHandler(),
eventLoop: Lambda.defaultEventLoop,
logger: Logger(label: "Runtime1")
)

// Second runtime
let runtime2 = LambdaRuntime(
handler: MockHandler(),
eventLoop: Lambda.defaultEventLoop,
logger: Logger(label: "Runtime1")
)

try await withThrowingTaskGroup(of: Void.self) { taskGroup in
// start the first runtime
taskGroup.addTask {
await #expect(throws: Never.self) {
try await runtime1.run()
}
}

// wait a small amount to ensure runtime1 task is started
try await Task.sleep(for: .seconds(1))

// Running the second runtime should trigger LambdaRuntimeError
await #expect(throws: LambdaRuntimeError.self) {
try await runtime2.run()
}

// cancel runtime 1 / task 1
taskGroup.cancelAll()
}

// Running the second runtime should work now
try await withThrowingTaskGroup(of: Void.self) { taskGroup in
taskGroup.addTask {
await #expect(throws: Never.self) { try await runtime2.run() }
}

// Set timeout and cancel the runtime 2
try await Task.sleep(for: .seconds(2))
taskGroup.cancelAll()
}
}
}

struct MockHandler: StreamingLambdaHandler {
mutating func handle(
_ event: NIOCore.ByteBuffer,
responseWriter: some AWSLambdaRuntime.LambdaResponseStreamWriter,
context: AWSLambdaRuntime.LambdaContext
) async throws {

}
}