Hope you've all been enjoying your holidays.
I've used a bit of the free time to start implementing a feature that has been bugging me for a while as something sorely missing from Swift. It's come up a few times in various discussions over the years, so I'd like to gather some feedback and put it up for review.
Sealed Protocols
Motivation
Protocols in Swift are a powerful tool to enable type-erasure and generic programming. Libraries often provide public
protocols for clients to use for these purposes. As an example, the standard library provides StringProtocol
for generic algorithms over both String
and Substring
, with the following caveat:
Do not declare new conformances to
StringProtocol
. Only theString
andSubstring
types in the standard library are valid conforming types.
Similarly, protocols can be useful when designing some more complex types, such as Foundation's recently-redesigned Data
type and accompanying DataProtocol
type. It is not clear whether external conformances to DataProtocol
should be allowed or not.
The ability for clients to add arbitrary conformances to any public protocol limits the ability of library authors to evolve those protocols in certain ways (such as adding new requirements/customisation points), unless those new members come with default implementations, which might not be possible in all cases.
Moreover, enabling the compiler to reason about the complete set of types which conform to a protocol can enable certain optimisations when using existentials involving those protocols. Current advice is to mark protocols which are only conformed-to by classes as refining AnyObject
, so the compiler can omit logic related to handling non-trivial structures.
Unfortunately, this AnyObject
constraint becomes part of the protocol's ABI. If a later version of the library wants to add conformance to a non-class type, removing the AnyObject
refinement would be a breaking change. There are also a number of other patterns (such as trivial types, or enums with single, class-type payloads) which do not have AnyObject
-style marker protocols and so miss out on potential optimisations. If we were instead to mark the protocol as 'sealed', the compiler could perform these optimisations as an implementation detail within the declaring module.
Detailed Design
I propose to add a new attribute called sealed
. A sealed
protocol may be made public, but may only be conformed-to from the module it was declared in.
Attempting to conform to a sealed protocol from outside of its declaring module will produce a compiler error.
// Module A.
sealed public protocol ASealedProtocol { /* ... */ }
// Okay.
extension String: ASealedProtocol { /* ... */ }
// --------------
// Module B.
// Error: cannot conform to sealed protocol 'ASealedProtocol' outside of its declaring module.
struct MyType: ASealedProtocol { /* ... */ }
// Error: cannot conform to sealed protocol 'ASealedProtocol' outside of its declaring module.
extension Int: ASealedProtocol { /* ... */ }
Refinements of sealed protocols must also be sealed, which means they too are only possible within the parent's declaring module.
// Module B.
// Error: cannot inherit from sealed protocol 'ASealedProtocol' outside of its declaring module.
protocol SpecialProtocol: ASealedProtocol {}
Other modules may still add extensions on sealed protocols and use them in protocol-compositions.
// Module B.
typealias SpecialThing = ASealedProtocol & SomeOtherProtocol // Okay.
extension ASealedProtocol { /* ... */ } // Okay.
open
classes may conform to sealed
protocols, which gives libraries a powerful way to provide selective customisation of a particular conformance.
Source and ABI Impact
Adding the sealed
attribute to an non-sealed protocol is potentially source- and binary-breaking, as clients may have written conformances which will no longer compile, and the declaring module may be optimised in a way which is not compatible with those external conformances.
It is a source- and binary-compatible change to 'unseal' a sealed protocol and open it up to external conformances. This mirrors how the open
attribute works for classes.
As previously discussed, there are a number of protocols in the standard library and Foundation which may choose to take advantage of this attribute to enforce expectations which are currently only documented in comments. There is the potential to break some projects which previously ignored the documentation, but it is probably an acceptable risk. They were warned.
Future Directions
Restricting external conformances opens the door for protocols to include non-public requirements, involving non-public types. Taking StringProtocol
as an example, a requirement could be added to access the "internal" _StringGuts
without ugly hacks. As for other protocol requirements, the compiler would check that all conforming types implement it, and it would be usable from any publically-visible function constrained to StringProtocol
, without downcasting to a concrete type or internal
refined protocol.
Alternatives Considered
-
Making
sealed
the default, and adding anopen
attribute (as we do for classes).In Swift, we want to make library evolution easy by making as few commitments as possible by default, and requiring library authors to opt-in to anything which might restrict them in the future. That would indicate that making
public
protocols besealed
by default is the right thing to do. Unfortunately, such a change would be massively source-breaking at this point, so it's pretty much a non-starter. -
Ban
open
members from witnessingsealed
protocol requirements.On the one hand, allowing conformances to be partially/totally overridden might be seen as breaking the seal. On the other hand, allowing it does not limit the ability of library authors to evolve the protocol in question (which is the primary motivation behind this feature), and expands the classes which may participate to include classes from 3rd-party/system libraries.
-
Bikeshedding.
closed
would also be okay, and more closely mirror theopen
keyword for classes. Previous discussions seemed to settle around the namesealed
, but it would be interesting to see which name the community prefers. -
Do nothing.
Always an option
, but undesirable because it means library authors cannot publish a protocol to be used for type-erasure or generic programming without committing to only ever add requirements that can be given default implementations.