My understanding is that closures are treated a little differently and can't conditionally be Sendable.
Let's say I have a type that holds onto a closure:
public struct Effect<Input, Output> {
public typealias Closure = (Input) -> Output
public let closure: Closure
public init(closure: @escaping Closure) {
self.closure = closure
}
}
This will work until I need it to be Sendable. It's easy enough to conditionally conform the type to Sendable when Input and Output are Sendable like so:
extension Effect: Sendable where Closure: Input: Sendable, Output: Sendable {}
However, this will correctly result in a warning because Closure isn't Sendable. What if we tried to conditionally conform Closure to Sendable to see what happens...
This just isn't how function types naturally interact with Sendable. The sendability of a function depends on its captures and is orthogonal to its parameter and result types. There's no reason a function that takes Sendable arguments can't be non-Sendable or vice-versa.
If you could write T: (Foo) -> Bar you’d be able to make this conditional though. So it’s something Swift doesn’t expose in its surface language today (“no, there’s no way to do this”) but it’s not impossible to represent.
I don't think dependently-Sendable function types would be a useful feature. Generic code that just wants to pass function values around doesn't need to know that they're function values, and the sendability of the functions will be preserved by opaque generics in the usual way. Constraining the code to know that the values are functions is only useful if the code wants to call those values or produce new values from (say) a closure expression. In either case, not knowing whether the function type is Sendable would just force the code to be maximally conservative on both sides: you'd have to create Sendable closures but immediately treat them as non-Sendable.
In general, when you're working with a known function, you know how you plan to use it and whether those uses require it to be Sendable, so you should just take either a Sendable or a non-Sendable function as appropriate, with no need to be generic over its sendability.
I'm a bit confused. How would a conditional conformance to Sendable based on whether a type parameter representing a function type is Sendable force the code to be maximally conservative / force you to create @Sendable function values? For example, pretend you could write this:
struct S<Arg, Result, Func> where Func == (Arg) -> Result {
let fn: Func
init(fn: Func) {
self.fn = fn
}
func call(arg: Arg) -> Result {
return fn(arg)
}
}
extension S: Sendable where Func: Sendable {}
IMO this is a reasonable thing to want to do. We already let @Sendable function types be passed as generic arguments to a type parameter T where T: Sendable. That said, I'm not sure if it'd be tricky to teach the constraint system to infer closure types as @Sendable based on captures because today the constraint system doesn't reason about closure captures at all for the purpose of type inference; it only infers a closure type to be @Sendable if it's written contextually.
If Func were actually constrained to “any function with this shape”, and you were in an extension of S, and you tried to turn a closure into a Func, you’d have to use a Sendable closure because Funcmight be a sendable function type. You don’t get to assume that Func isn’t Sendable because Func isn’t statically constrained to Sendable; that’s not sound.
You’re right that there isn’t a similar constraint on calls, though. So yes, if you just want an opaque function value that you can pass around and call, you could reinvent it with this feature. Of course, you could also just use a function type.
just for the "use case" record, I stumbled over this very thing as well in the context of my HTML rendering library.
public struct ForEach<Data, Content>: HTML
where Data: Sequence, Content: HTML
{
var sequence: Data
// TODO: Swift 6 - @Sendable is not ideal here, but currently the response generators for hummingbird/vapor require sendable HTML types
// also, currently there is no good way to conditionally apply Sendable conformance based on closure type
var contentBuilder: @Sendable (Data.Element) -> Content
public init(_ sequence: Data, @HTMLBuilder content contentBuilder: @escaping @Sendable (Data.Element) -> Content) {
self.sequence = sequence
self.contentBuilder = contentBuilder
}
public static func _render<Renderer: _HTMLRendering>(_ html: consuming Self, into renderer: inout Renderer, with context: consuming _RenderingContext) {
// ...
}
}
extension ForEach: Sendable where Data: Sendable {}
For most other DSL types I conform to Sendable conditionally based on their generic stored properties. But for the types that take a closure, I cannot really do that. Additionally, the generic trick Holly mentioned above does not really play well with result builders, and I could not get hand-crafted initializer overloads to work either.
With Swift 6 it's not thaaat dramatic, because I can use sending now and HTML types no longer really need to be sendable. (The TODO comment above predates Swift 6). But for 5.10 + concurrency checking it is not a great place to be in.
I think this would be quite useful for LazySequence/LazyCollection types that wrap a closure. Today they are all non-sendable (e.g. [0, 1, 2].lazy.map { $0 * 2 } is a non-sendable LazyCollection. They could be in theory conditionally Sendable, especially if the closure passed to map or one of the other transformations don't capture anything. This likely can't be changed for LazySequence/LazyCollection but we could think about it once we have the Container protocol that works with Spans.