mirror of
https://github.com/luau-lang/rfcs.git
synced 2025-04-03 18:10:56 +01:00
Relax the recursive type restriction. (#92)
* Relax the recursive type restriction. * more alternatives * Update description of behavior pre-restriction.
This commit is contained in:
parent
e78e855dc6
commit
12a39c3fdb
1 changed files with 80 additions and 0 deletions
80
docs/relax-recursive-type-restriction.md
Normal file
80
docs/relax-recursive-type-restriction.md
Normal file
|
@ -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<T> = { data: T }
|
||||||
|
```
|
||||||
|
|
||||||
|
These type aliases can also be recursive:
|
||||||
|
```luau
|
||||||
|
type Tree<T> = { data: T, children: {Tree<T>} }
|
||||||
|
```
|
||||||
|
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<U>` in the signature of `andThen`:
|
||||||
|
```luau
|
||||||
|
type Promise<T> = {
|
||||||
|
create: (T) -> Promise<T>,
|
||||||
|
andThen: <U>(self: Promise<T>, callback: (T) -> Promise<U>) -> Promise<U>
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 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.
|
Loading…
Add table
Reference in a new issue