-
Notifications
You must be signed in to change notification settings - Fork 2.4k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
1 changed file
with
231 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,231 @@ | ||
# Task Naming API | ||
|
||
* Proposal: [SE-NNNN](NNNN-filename.md) | ||
* Authors: [Konrad Malawski](https://github.com/ktoso), [Harjas Monga](https://github.com/Harjas12) | ||
* Review Manager: TBD | ||
* Status: **Awaiting implementation** or **Awaiting review** | ||
* Implementation: Work In Progress [PR](https://github.com/swiftlang/swift/pull/77609) | ||
* Review: ([pitch](https://forums.swift.org/...)) | ||
|
||
## Introduction | ||
|
||
In this proposal, we introduce several new APIs to allow developers to name their Swift Tasks for the purposes of identifying tasks in a human readable way manually and by various developer tools. | ||
|
||
## Motivation | ||
|
||
In previous generations of concurrency technologies, developer tools, such as debuggers, have had access to some kind of label to help describe a process’s concurrent work. Ex: Pthread names or GCD queue names. These names are very helpful to provide extra context to developers when using debugging and profiling tools. | ||
|
||
Currently, Swift Concurrency has no affordances to allow developers to label a Task, which can be troublesome for developers trying to identify "which task" is taking a long time to process or similar questions when observing the system externally. In order to ease the debugging and profiling of Swift concurrency code, developers should be able to annotate their Swift Tasks to describe an asynchronous workload. | ||
|
||
## Proposed solution | ||
|
||
In order to allow developers to provide helpful names for Swift Tasks, the Swift Task creation APIs should be modified to *optionally* allow developers to provide a name for that task. | ||
|
||
Consider the example: | ||
```swift | ||
let getUsers = Task { | ||
await users.get(accountID)) | ||
} | ||
``` | ||
|
||
In order to ease debugging, a developer could create this unstructured task by passing in a name instead: | ||
|
||
``` | ||
let getUsers = Task(name: "Get Users") { | ||
await users.get(accountID) | ||
} | ||
``` | ||
|
||
Or, if a developer has a lot of similar Tasks, they can provide more contextual information using string interpolation. | ||
|
||
``` | ||
let getUsers = Task("Get Users for \(accountID)") { | ||
await users.get(accountID) | ||
} | ||
``` | ||
|
||
By introducing this in a structured, coherent way in Swift itself, rather than developers each inventing their own task-local with a name, runtime inspection tools and debuggers can become aware of task names and show you exactly which accountID was causing the crash or a profiling tool could tell you which accountID request was slow to load. | ||
|
||
## Detailed design | ||
|
||
|
||
Naming tasks is only allowed during their creation, and modifying names is not allowed. Names are arbitrary user-defined strings, which may be computed at runtime because they often contain identifying information such as the request ID or similar runtime information. | ||
|
||
In order to allow naming tasks, the following APIs will be provided on `Task`: | ||
|
||
```swift | ||
init( | ||
name: String? = nil, | ||
priority: TaskPriority? = nil, | ||
operation: sending @escaping @isolated(any) () async -> Success | ||
|
||
static func detached( | ||
name: String? = nil, | ||
priority: TaskPriority? = nil, | ||
operation: sending @escaping @isolated(any) () async -> Success | ||
``` | ||
|
||
In addition to these APIs to name unstructured Tasks, the following API will be added to name Tasks created as a part of a task group task group: | ||
|
||
```swift | ||
mutating func addTask( | ||
name: String? = nil, | ||
executorPreference taskExecutor: (any TaskExecutor)? = nil, | ||
priority: TaskPriority? = nil, | ||
operation: sending @escaping @isolated(any) () async -> ChildTaskResult | ||
) | ||
|
||
mutating func addTaskUnlessCancelled( | ||
name: String? = nil, | ||
executorPreference taskExecutor: (any TaskExecutor)? = nil, | ||
priority: TaskPriority? = nil, | ||
operation: sending @escaping @isolated(any) () async -> ChildTaskResult | ||
) | ||
|
||
``` | ||
|
||
These APIs would be added to all kinds of task groups, including throwing, discarding ones. With the signature being appropriately matching the existing addTask signatures of those groups | ||
|
||
In addition to that, it will be possible to read a name off a task, similar to how the current task's priority is possible to be read: | ||
|
||
```swift | ||
extension Task { | ||
var currentName: String? { get } | ||
} | ||
``` | ||
|
||
### Actor Identity | ||
|
||
This proposal focuses on task names, however, another important part of Swift Concurrency is actors, so in this section we’d like to discuss how there isn’t an actual need for new API to address *actor naming* because of how actors can already conform to protocols. | ||
|
||
An actor can conform e.g. to the `Identifiable` protocol. This works well with constant identifiers, as an actor can have a constant let property implement the `id` requirement from this protocol: | ||
|
||
```swift | ||
actor Worker: Identifiable { | ||
let id: String | ||
|
||
init(id: String) { | ||
self.id = id | ||
} | ||
} | ||
``` | ||
|
||
It is also likely that such identity is how a developer might want to look up and identify such actor in traces or logs, so making use of `Identifiable` seems like a good pattern to follow. | ||
|
||
It is also worth reminding that thread-safety of an actor is ensured even if the `id` were to be implemented using a computed property, because it will be forced to be `nonisolated` because of Swift’s conformance and actor isolation rules: | ||
|
||
```swift | ||
actor Worker: Identifiable { | ||
let workCategory: String = "fetching" // "building" etc... | ||
let workID: Int | ||
|
||
nonisolated var id: String { | ||
"\(workCategory)-\(workID)" | ||
} | ||
} | ||
``` | ||
|
||
#### Distributed Actor Identity | ||
|
||
Distributed actors already implicitly conform to `Identifiable` protocol and have a very useful `id` representation that is always assigned by the actor system by which an actor is managed. | ||
|
||
This id is the natural visual representation of such actor, and tools which want to print an “actor identity” should rely on this. In other words, this simply follows the same general pattern that makes sense for other objects and actors of using Identifiable when available to identify things. | ||
|
||
```swift | ||
distributed actor Worker { // implicitly Identifiable | ||
// nonisolated var id: Self.ActorSystem.ActorID { get } | ||
} | ||
``` | ||
|
||
## Source compatibility | ||
|
||
This proposal only contains additive changes to the API surface. | ||
|
||
Since Swift Tasks names will be optional, there will be no source compatibility issues. | ||
|
||
## ABI compatibility | ||
|
||
This proposal is ABI additive and does not change any existing ABI. | ||
|
||
## Implications on adoption | ||
|
||
Because runtime changes are required, these new APIs will only be available on newer OSes. | ||
|
||
## Future directions | ||
|
||
This proposal does not contain a method to name Swift Tasks created using the `async let` syntax. Unlike the other methods of creating Tasks, the `async let` syntax didn’t have an obvious way to allow a developer to provide a string. A suggestion of how we may provide automatic names to Tasks created via this method will be shown below in the [Alternatives Considered section](##Alternatives-considered). | ||
|
||
## Alternatives considered | ||
|
||
### AsyncLet Task Naming | ||
|
||
While there is no clear way on how to name Swift Task using `async let`, the following were considered. | ||
|
||
#### Approach 1: | ||
|
||
Since we effectively want to express that “the task” is some specific task, we had considered introducing some special casing where if the right hand side of an async let we want to say at creation time that this task is something specific, thus we arrive at the following: | ||
|
||
```swift | ||
async let example: String = Task(name: "get-example") { "example" } | ||
``` | ||
|
||
In order to make this syntax work, we need to avoid double creating tasks. When the compiler sees the `async let` syntax and the `Task {}` initializer, it would need to not create a Task to immediately create another Task inside it, but instead use that Task initializer that we explicitly wrote. | ||
|
||
While, this approach could in theory allow us to name Tasks created using `async let`. It has at least one major issue: | ||
|
||
It can cause surprising behavior and it can be unclear that this would only work when the Task initializer is visible from the async let declaration... I.e. moving the initialization into a method like this: | ||
|
||
```swift | ||
async let example: String = getTask() // error String != Task<String> | ||
|
||
func getTask() -> Task<String> { Task(name: "get-example") { "example" } } | ||
``` | ||
|
||
This would not only break refactoring, as the types are not the same; but also execution semantics, as this refactoring has now caused the task to become an unstructured task “by accident”. Therefore this approach is not viable because it introduces too many easy to make mistakes. | ||
|
||
#### Approach 2: | ||
|
||
Instead of attempting to adding a naming API to the `async let` syntax, we could instead take a different where if developers really want to name a structured Task they can use a `TaskGroup` and the compiler would generate a good default name for the Tasks created using the `async let` syntax. Drawing inspiration from how closures and dispatch blocks are named, we count the declaration in the scope and use that to name it. For example: | ||
|
||
```swift | ||
func getUserImages() async -> [Image] { | ||
|
||
async let profileImg = getProfilePicture() // <- Named "getUserImages.asyncLet-1" | ||
async let headerImg = getHeaderPicture() // <- Named "getUserImages.asyncLet-2" | ||
|
||
. | ||
. | ||
. | ||
} | ||
``` | ||
|
||
These names at the very least give some indication of what the task was created to do, and the developer can opt to use the `TaskGroup` API if more control is desired. | ||
|
||
A slight alternative to this suggestion, is instead of using the name of the surrounding scope, use the name of the parent task instead. For example: | ||
|
||
```swift | ||
Task(name: "get user images for \(userID)") { | ||
async let profileImg = getProfilePicture() // <- Named "getUserImages.asyncLet-1" | ||
async let headerImg = getHeaderPicture() // <- Named "getUserImages.asyncLet-2" | ||
|
||
. | ||
. | ||
. | ||
} | ||
``` | ||
|
||
This approach doesn’t allow developers full control over naming tasks, but it is in same spirit of allowing developer tools to provide more context for a task. | ||
|
||
## Structured Names | ||
|
||
There was some thought given to the idea of allowing developers to group similar tasks (in name only). Consider programs that create hundreds of tasks for network requests; by allowing grouping a runtime analysis tool could surface that in a textual or graphical UI. The API needed would be similar to the one proposed, but with an additional optional `category` argument for the Task initializer. For example: | ||
|
||
```swift | ||
Task(category: "Networking", name: "download profile image for \(userID)) { ... } | ||
``` | ||
Then a debugger than wanted to print all the tasks running when a break point is hit, it could group them by this optional “Networking” category. | ||
This is not in the actual proposal in order to keep the API simple and doesn’t add much additional value over a simple name. | ||
## Acknowledgments |