Many modern systems have client components like iOS, macOS or watchOS applications as well as server components that those clients interact with. Serverless functions are often the easiest and most efficient way for client application developers to extend their applications into the cloud.
Serverless functions are increasingly becoming a popular choice for running event-driven or otherwise ad-hoc compute tasks in the cloud. They power mission critical microservices and data intensive workloads. In many cases, serverless functions allow developers to more easily scale and control compute costs given their on-demand nature.
When using serverless functions, attention must be given to resource utilization as it directly impacts the costs of the system. This is where Swift shines! With its low memory footprint, deterministic performance, and quick start time, Swift is a fantastic match for the serverless functions architecture.
Combine this with Swift's developer friendliness, expressiveness, and emphasis on safety, and we have a solution that is great for developers at all skill levels, scalable, and cost effective.
Swift Tencent SCF Runtime is a forked version from Swift AWS Lambda Runtime, designed to make building cloud functions in Swift simple and safe. The library is an implementation of the Tencent SCF Custom Runtime API and uses an embedded asynchronous HTTP Client based on SwiftNIO that is fine-tuned for performance in the SCF Custom Runtime context. The library provides a multi-tier API that allows building a range of cloud functions: From quick and simple Closures to complex, performance-sensitive event handlers.
This is the beginning of an open-source project actively seeking contributions. While the core API is considered stable, the API may still evolve as we get closer to a 1.0
version. There are several areas which need additional attention, including but not limited to:
- Further performance tuning
- Additional trigger events
- Additional documentation and best practices
- Additional examples
If you have used Swift AWS Lambda Runtime, you may find most of the APIs familiar. If you have never used Tencent SCF, AWS Lambda or Docker before, check out this getting started guide which helps you with every step from zero to a running cloud function.
First, create a SwiftPM project and pull Swift Tencent SCF Runtime as dependency into your project:
// swift-tools-version:5.2
import PackageDescription
let package = Package(
name: "my-cloud-function",
platforms: [
.macOS(.v10_15),
],
products: [
.executable(name: "MyCloudFunction", targets: ["MyCloudFunction"]),
],
dependencies: [
.package(url: "https://github.com/stevapple/swift-tencent-scf-runtime", from: "0.2.0"),
],
targets: [
.target(name: "MyCloudFunction", dependencies: [
.product(name: "TencentSCFRuntime", package: "tencent-scf-runtime"),
]),
]
)
Next, create a main.swift
and implement your cloud function.
The simplest way to use TencentSCFRuntime
is to pass in a closure, for example:
// Import the module.
import TencentSCFRuntime
// In this example we are receiving and responding with strings.
SCF.run { (context, name: String, callback: @escaping (Result<String, Error>) -> Void) in
callback(.success("Hello, \(name)"))
}
More commonly, the event would be a JSON, which is modeled using Codable
, for example:
// Import the module.
import TencentSCFRuntime
// Request, uses Decodable for transparent JSON decoding.
private struct Request: Decodable {
let name: String
}
// Response, uses Encodable for transparent JSON encoding.
private struct Response: Encodable {
let message: String
}
// In this example we are receiving and responding with `Codable`.
SCF.run { (context, request: Request, callback: @escaping (Result<Response, Error>) -> Void) in
callback(.success(Response(message: "Hello, \(request.name)")))
}
Since most SCF functions are triggered by events originating in the Tencent Cloud platform like CMQ
, COS
or APIGateway
, the package also includes a TencentSCFEvents
module that provides implementations for most common SCF event types to further simplify writing SCF functions. For example, handling a CMQ
event:
// Import the modules.
import TencentSCFRuntime
import TencentSCFEvents
// In this example we are receiving CMQ Messages from a CMQ Topic, with no response (Void).
SCF.run { (context, event: CMQ.Topic.Event, callback: @escaping (Result<Void, Error>) -> Void) in
for record in event.records {
...
}
callback(.success(Void()))
}
Modeling SCF functions as Closures is both simple and safe. Swift Tencent SCF Runtime will ensure that the user-provided code is offloaded from the network processing thread such that even if the code becomes slow to respond or gets hang, the underlying process can continue to function. This safety comes at a small performance penalty from context switching between threads. In many cases, the simplicity and safety of using the Closure based API is often preferred over the complexity of the performance-oriented API.
Performance sensitive cloud functions may choose to use a more complex API which allows user code to run on the same thread as the networking handlers. Swift Tencent SCF Runtime uses SwiftNIO as its underlying networking engine which means the APIs are based on SwiftNIO concurrency primitives like the EventLoop
and EventLoopFuture
. For example:
// Import the modules.
import NIOCore
import TencentSCFRuntime
import TencentSCFEvents
// Our SCF handler, conforms to EventLoopSCFHandler.
struct Handler: EventLoopSCFHandler {
typealias Event = COS.Event // Request type
typealias Output = Void // Response type
// In this example we are receiving a COS Event, with no response (Void).
func handle(_ event: Event, context: SCF.Context) -> EventLoopFuture<Output> {
...
context.eventLoop.makeSucceededFuture(Void())
}
}
SCF.run(Handler())
Beyond the small cognitive complexity of using the EventLoopFuture
based APIs, note these APIs should be used with extra care. An EventLoopSCFHandler
will execute the user code on the same EventLoop
(thread) as the library, making processing faster but requiring the user code to never call blocking APIs as it might prevent the underlying process from functioning.
To deploy SCF functions to Tencent SCF Platform, you need to compile the code for CentOS 7.6 which is the OS used on SCF microVMs, package it as a Zip file, and upload to Tencent Cloud.
Tencent Cloud offers several tools to interact and deploy cloud functions to SCF including TCCLI and Serverless Framework. The Examples Directory includes complete sample build and deployment scripts that utilize these tools.
Note the examples mentioned above use dynamic linking, therefore bundle the required Swift libraries in the Zip package along side the executable. You may choose to link the SCF function statically (using -static-stdlib
) which could improve performance but requires additional linker flags.
To build the SCF function for CentOS 7.6, use the Docker image published as Swift toolchains for SCF, as demonstrated in the examples.
The library defines three protocols for the implementation of an SCF Handler. From low-level to more convenient:
An EventLoopFuture
based processing protocol for an SCF function that takes a ByteBuffer
and returns a ByteBuffer?
asynchronously.
ByteBufferSCFHandler
is the lowest level protocol designed to power the higher level EventLoopSCFHandler
and SCFHandler
based APIs. Users are not expected to use this protocol, though some performance sensitive applications that operate at the ByteBuffer
level or have special serialization needs may choose to do so.
public protocol ByteBufferSCFHandler {
/// The SCF handling method.
/// Concrete SCF handlers implement this method to provide the SCF functionality.
///
/// - Parameters:
/// - context: Runtime `Context`.
/// - event: The event or input payload encoded as `ByteBuffer`.
///
/// - Returns: An `EventLoopFuture` to report the result of the SCF function back to the runtime engine.
/// The `EventLoopFuture` should be completed with either a response encoded as `ByteBuffer` or an `Error`.
func handle(_ event: ByteBuffer, context: SCF.Context) -> EventLoopFuture<ByteBuffer?>
/// Clean up the SCF resources asynchronously.
/// Concrete SCF handlers implement this method to shutdown resources like `HTTPClient`s and database connections.
///
/// - Note: In case your SCF function fails while creating your `SCFHandler` in the `HandlerFactory`, this method
/// **is not invoked**. In this case you must cleanup the created resources immediately in the `HandlerFactory`.
func shutdown(context: SCF.ShutdownContext) -> EventLoopFuture<Void>
}
EventLoopSCFHandler
is a strongly typed, EventLoopFuture
based asynchronous processing protocol for an SCF function that takes a user defined In and returns a user defined Out.
EventLoopSCFHandler
extends ByteBufferSCFHandler
, providing ByteBuffer
-> Event
decoding and Output
-> ByteBuffer?
encoding for Codable
and String.
EventLoopSCFHandler
executes the user provided cloud function on the same EventLoop
as the core runtime engine, making the processing fast but requires more care from the implementation to never block the EventLoop
. It it designed for performance sensitive applications that use Codable
or String based cloud functions.
public protocol EventLoopSCFHandler: ByteBufferSCFHandler {
associatedtype Event
associatedtype Output
/// The SCF handling method.
/// Concrete SCF handlers implement this method to provide the SCF functionality.
///
/// - Parameters:
/// - context: Runtime `Context`.
/// - event: Event of type `Event` representing the event or request.
///
/// - Returns: An `EventLoopFuture` to report the result of the SCF function back to the runtime engine.
/// The `EventLoopFuture` should be completed with either a response of type `Output` or an `Error`.
func handle(_ event: Event, context: SCF.Context) -> EventLoopFuture<Output>
/// Encode a response of type `Output` to `ByteBuffer`.
/// Concrete SCF handlers implement this method to provide coding functionality.
///
/// - Parameters:
/// - allocator: A `ByteBufferAllocator` to help allocate the `ByteBuffer`.
/// - value: Response of type `Output`.
///
/// - Returns: A `ByteBuffer` with the encoded version of the `value`.
func encode(allocator: ByteBufferAllocator, value: Output) throws -> ByteBuffer?
/// Decode a`ByteBuffer` to a request or event of type `Event`
/// Concrete SCF handlers implement this method to provide coding functionality.
///
/// - Parameters:
/// - buffer: The `ByteBuffer` to decode.
///
/// - Returns: A request or event of type `Event`.
func decode(buffer: ByteBuffer) throws -> Event
}
SCFHandler
is a strongly typed, completion handler based asynchronous processing protocol for an SCF function that takes a user defined Event
and returns a user defined Output
.
SCFHandler
extends ByteBufferSCFHandler
, performing ByteBuffer
-> Event
decoding and Output
-> ByteBuffer
encoding for Codable
and String.
SCFHandler
offloads the user provided SCF execution to a DispatchQueue
making processing safer but slower.
public protocol SCFHandler: EventLoopSCFHandler {
/// Defines to which `DispatchQueue` the SCF execution is offloaded to.
var offloadQueue: DispatchQueue { get }
/// The SCF handling method.
/// Concrete SCF handlers implement this method to provide the SCF functionality.
///
/// - Parameters:
/// - context: Runtime `Context`.
/// - event: Event of type `Event` representing the event or request.
/// - callback: Completion handler to report the result of the SCF function back to the runtime engine.
/// The completion handler expects a `Result` with either a response of type `Output` or an `Error`.
func handle(_ event: Event, callback: @escaping (Result<Out, Error>) -> Void)
}
In addition to protocol-based SCF functions, the library provides support for Closure-based ones, as demonstrated in the overview section above. Closure-based SCF functions are based on the SCFHandler
protocol which mean they are safer. For most use cases, Closure-based cloud function is a great fit and users are encouraged to use them.
The library includes implementations for Codable
and String based SCF functions. Since Tencent Cloud messages are primarily JSON based, this can cover the most common use cases.
public typealias CodableClosure<In: Decodable, Out: Encodable> = (SCF.Context, In, @escaping (Result<Out, Error>) -> Void) -> Void
public typealias StringClosure = (SCF.Context, String, @escaping (Result<String, Error>) -> Void) -> Void
This design allows for additional event types as well, and such SCF implementation can extend one of the above protocols and provided their own ByteBuffer
-> Event
decoding and Output
-> ByteBuffer
encoding.
When calling the user provided SCF function, the library provides a Context
class that provides metadata about the execution context, as well as utilities for logging and allocating buffers.
public final class Context: CustomDebugStringConvertible {
/// The request ID, which identifies the request that triggered the function invocation.
public let requestID: String
/// The memory limit of the cloud function in MB.
public let memoryLimit: UInt
/// The time limit of the cloud function event in ms.
public let timeLimit: DispatchTimeInterval
/// The timestamp that the function times out.
public let deadline: DispatchWallTime
/// The UIN of cloud function actor.
public let uin: String
/// The APPID that the cloud function belongs to.
public let appid: String
/// The Tencent Cloud region that the cloud function is in.
public let region: TencentCloud.Region?
/// The name of the cloud function.
public let name: String
/// The namespace of the cloud function.
public let namespace: String
/// The version of the cloud function.
public let version: Version
/// The role credential from SCF environment.
public let credential: TencentCloud.Credential?
/// `Logger` to log with.
///
/// - Note: The `LogLevel` can be configured using the `LOG_LEVEL` environment variable.
public let logger: Logger
/// The `EventLoop` the SCF function is executed on. Use this to schedule work with.
/// This is useful when implementing the `EventLoopSCFHandler` protocol.
///
/// - Note: The `EventLoop` is shared with the SCF Runtime Engine and should be handled with extra care.
/// Most importantly the `EventLoop` must never be blocked.
public let eventLoop: EventLoop
/// `ByteBufferAllocator` to allocate `ByteBuffer`.
/// This is useful when implementing `EventLoopSCFHandler`.
public let allocator: ByteBufferAllocator
}
The library’s behavior can be fine tuned using environment variables based configuration. The library supported the following environment variables:
LOG_LEVEL
: Define the logging level as defined by SwiftLog. Set to INFO by default.MAX_REQUESTS
: Max cycles the library can handle before exiting. Set to none by default.STOP_SIGNAL
: Signal to capture for termination. Set to TERM by default.REQUEST_TIMEOUT
: Max time to wait for responses to come back from the SCF runtime engine. Set to none by default.
If you'd like to test the cloud function with HTTP requests, we provide a simple local emulator. To enable it, set the environment variable LOCAL_SCF_SERVER_ENABLED
to true
.
By swift run
, you'll see the logger output saying LocalSCFServer started and listening on 127.0.0.1:9001, receiving events on /invoke
which means the local emulator is up. Then you can send HTTP POST
requests to http://127.0.0.1:9001/invoke
endpoint to invoke your cloud function.
We also provide a way to inject environment variables without actually changing it. Since it only works in DEBUG mode, don't forget to wrap such codes with #if DEBUG
and #endif
.
// Import the module.
import TencentSCFRuntime
#if DEBUG
// Start LocalServer by default.
SCF.Env["LOCAL_SCF_SERVER_ENABLED"] = "true"
// Simulate SCF environment.
let variables = [
"TENCENTCLOUD_UIN": "100012345678",
"TENCENTCLOUD_APPID": "123456789",
"TENCENTCLOUD_REGION": "ap-shanghai"
]
SCF.Env.update(with: variables)
#endif
// Our SCF handler.
SCF.run { (context, name: String, callback: @escaping (Result<String, Error>) -> Void) in
...
}
We simulate the SCF environment with some variables set by default. The value set by user code is of the highest priority, while the framework's default has the lowest.
You can read some contextual variables through SCF.Context
. All the environment variables can be accessed through SCF.Env
.
The library is designed to integrate with SCF Runtime Engine via the SCF Custom Runtime API which was introduced as part of SCF Custom Runtime in 2020. The latter is an HTTP server that exposes three main RESTful endpoint:
/runtime/invocation/next
/runtime/invocation/response
/runtime/invocation/error
A single SCF execution workflow is made of the following steps:
- The library calls SCF Runtime Engine
/next
endpoint to retrieve the next invocation request. - The library parses the response HTTP headers and populate the Context object.
- The library reads the
/next
response body and attempt to decode it. Typically it decodes to user providedEvent
type which extendsDecodable
, but users may choose to write SCF functions that receive the input as String orByteBuffer
which require less, or no decoding. - The library hands off the
Context
andEvent
event to the user provided handler. In the case ofSCFHandler
based handler this is done on a dedicatedDispatchQueue
, providing isolation between user's and the library's code. - User provided handler processes the request asynchronously, invoking a callback or returning a future upon completion, which returns a Result type with the Out or Error populated.
- In case of error, the library posts to SCF Runtime Engine
/error
endpoint to provide the error details, which will show up on SCF logs. - In case of success, the library will attempt to encode the response. Typically it encodes from user provided
Output
type which extendsEncodable
, but users may choose to write SCF functions that return a String orByteBuffer
, which require less, or no encoding. The library then posts the response to SCF Runtime Engine/response
endpoint to provide the response to the callee.
The library encapsulates the workflow via the internal SCFRuntimeClient
and SCFRunner
structs respectively.
SCF Runtime Engine controls the Application lifecycle and in the happy case never terminates the application, only suspends it's execution when no work is available.
As such, the library main entry point is designed to run forever in a blocking fashion, performing the workflow described above in an endless loop.
That loop is broken if/when an internal error occurs, such as a failure to communicate with SCF Custom Runtime Engine API, or under other unexpected conditions.
By default, the library also registers a Signal handler that traps INT
and TERM
, which are typical Signals used in modern deployment platforms to communicate shutdown request.
Serverless Cloud Functions can be invoked directly from the SCF console, SCF API, TCCLI and Tencent Cloud toolkit. More commonly, they are invoked as a reaction to an events coming from the Tencent Cloud platform. To make it easier to integrate with Tencent Cloud platform events, the library includes an TencentSCFEvents
target which provides abstractions for many commonly used events. Additional events can be easily modeled when needed following the same patterns set by TencentSCFEvents
. Integration points with the Tencent Cloud platform include:
Note: Each one of the integration points mentioned above includes a set of Decodable
structs that transform Tencent Cloud's data model for these APIs. APIGateway response is wrapped into an Encodable
struct with three different initializers to help you build any valid response.
Cloud functions performance is usually measured across two axes:
-
Cold start times: The time it takes for a cloud function to startup, ask for an invocation and process the first invocation.
-
Warm invocation times: The time it takes for a cloud function to process an invocation after the cloud function has been invoked at least once.
Larger packages size (Zip file uploaded to SCF platform) negatively impact the cold start time, since SCF needs to download and unpack the package before starting the process.
Swift provides great Unicode support via ICU. Therefore, Swift-based SCF functions include the ICU libraries which tend to be large. This impacts the download time mentioned above and an area for further optimization. Some of the alternatives worth exploring are using the system ICU that comes with CentOS 7 (albeit older than the one Swift ships with) or working to remove the ICU dependency altogether. We welcome ideas and contributions to this end.