diff --git a/docs/relax-recursive-type-restriction.md b/docs/relax-recursive-type-restriction.md new file mode 100644 index 0000000..3dd3da5 --- /dev/null +++ b/docs/relax-recursive-type-restriction.md @@ -0,0 +1,80 @@ +# Relax the recursive type restriction + +## Summary + +Relax the [recursive type restriction](recursive-type-restriction.md) by implementing isorecursive +types properly for parameterized type aliases. + +## Motivation + +Luau supports parameterized type aliases as follows: +```luau +type Box = { data: T } +``` + +These type aliases can also be recursive: +```luau +type Tree = { data: T, children: {Tree} } +``` +but the [recursive type restriction](recursive-type-restriction.md) limits us to supporting only +direct recursion as above, where the type aliases' parameters remain unchanged. + +At the time the restriction was implemented, Luau featured a greedy type inference engine that would +forward declare type aliases using a free type. Recursive uses of that type alias in the definition +would attempt to instantiate them before the alias is resolved (effectively a noop). The result was +often nonsensical, incomplete types with unresolved free types present in them. + +This limitation means that Luau's type system is unable to express common recursive type patterns +for data structures, such as `andThen` for a monadic data type like `Promise`. For instance, the +following definition is rejected because of the use of `Promise` in the signature of `andThen`: +```luau +type Promise = { + create: (T) -> Promise, + andThen: (self: Promise, callback: (T) -> Promise) -> Promise +} +``` + +## Design + +Today, Luau has a new type inference engine that is able to deal with dynamic constraint resolution +and has already allowed us to build a system for arbitrary type functions, both builtin and +provided with the language and [user-defined type functions](user-defined-type-functions.md). We can +relax the recursive type restriction today in this new type inference engine by treating type +aliases as proper type functions that are lazily expanded when their contents are needed. + +## Drawbacks + +Eliminating this restriction means committing to an intrinsically more complicated architecture for +type inference in Luau in general. The Luau team, however, has already made the decision to do this, +and so there seems to be little drawback beyond the obvious implementation work required in making +type alias expansion completely lazy. This pays for itself in the considerable gain in expressivity +gained for users of the type system. + +## Alternatives + + +### Equirecursive Types + +The main alternative besides doing nothing at all would be to implement equirecursive types. +Equirecursive types have considerably greater complexity, both for humans and for computers. They +pose considerable algorithmic challenges for both type checking and type inference, and require +an implementation of a canonicalization of recursive types that is O(n log n). Isorecursive types +are simpler, map closely to what is already implemented in Luau today, and match the intuitions +users likely have from interacting with recursive types in nominal object-oriented languages. + +### Strict type aliases with Caching + +[A previous RFC](https://github.com/asajeffrey/rfcs/blob/recursive-type-unrestriction/docs/recursive-type-unrestriction.md) +proposed loosening the recursive type restriction, but still leaving it partially in tact, by implementing a strategy of +mapping distinct recursive uses of a type alias to a free type during the definition portion of constructing the type alias +and then tying the knot using that mapping as a cache. This feels in the spirit of Landin's Knot, and was a meaningful +proposal to improve the status quo without us implementing a new type inference engine, but as mentioned in the drawbacks +section, this work is already well underway for other reasons and so the appeal of taking this route today is lessened. + +### Lazy recursive type instantiations with Caching + +The previous RFC also mentioned briefly both the approach proposed by this RFC and an approach that uses a cache as well +to limit the size of the graph. This may be useful, and may in fact be what we implement, but the author of this RFC does +not feel that it is necessary to make a commitment one way or the other to the implementation using caching. We will do +whatever gets us the correct semantics for recursive types, and if there is any caching, it should not be visible to the +user.