mirror of
https://github.com/luau-lang/rfcs.git
synced 2025-04-03 01:50:59 +01:00
RFC: Metatable type functions (#69)
* Create metatable type functions RFC * Add reference to metatable-type-syntax * Turn lua into luau Co-authored-by: ariel <aweiss@hey.com> * Turn lua into luau Co-authored-by: ariel <aweiss@hey.com> * Turn lua into luau Co-authored-by: ariel <aweiss@hey.com> * Turn lua into luau Co-authored-by: ariel <aweiss@hey.com> * Turn lua into luau Co-authored-by: ariel <aweiss@hey.com> * Fix typeof() usage * Specify behavior for getmetatable * Specify 2nd argument failure case for setmetatable * Update docs/metatable-type-functions.md Co-authored-by: ariel <aweiss@hey.com> * Update docs/metatable-type-functions.md Co-authored-by: ariel <aweiss@hey.com> * Fix wrongly sectioned failure case for setmetatable * Update docs/metatable-type-functions.md Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com> * Update docs/metatable-type-functions.md Co-authored-by: ariel <aweiss@hey.com> * Specify additional failure case for setmetatable --------- Co-authored-by: ffrostfall <80861876+ffrostflame@users.noreply.github.com> Co-authored-by: ariel <aweiss@hey.com> Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com>
This commit is contained in:
parent
1133124f9c
commit
84f9815c32
1 changed files with 88 additions and 0 deletions
88
docs/metatable-type-functions.md
Normal file
88
docs/metatable-type-functions.md
Normal file
|
@ -0,0 +1,88 @@
|
|||
# Metatable Type Functions
|
||||
|
||||
## Summary
|
||||
|
||||
Implement type functions for `getmetatable` and `setmetatable`.
|
||||
|
||||
## Motivation
|
||||
|
||||
### `setmetatable` Type Function
|
||||
|
||||
There is currently no way for users to apply metatable type information to a type without the usage of `typeof()`. This isn't ideal, as it adds verbosity and boilerplate to common patterns such as object-oriented programming. For example, the following:
|
||||
|
||||
```luau
|
||||
local clock = {}
|
||||
type Identity = typeof(setmetatable({} :: { time: number }, { __index = clock }))
|
||||
```
|
||||
|
||||
could be reduced to:
|
||||
|
||||
```luau
|
||||
local clock = {}
|
||||
type Identity = setmetatable<{ time: number }, { __index: typeof(clock) }>
|
||||
```
|
||||
|
||||
### `getmetatable` Type Function
|
||||
|
||||
[Issue #1435](https://github.com/luau-lang/luau/issues/1435) is caused by a lack of special behavior for `getmetatable` when the type's metatable has a `__metatable` field. This is fixable in a variety of ways, however a type function has multiple benefits and exposes more type-level expression for users. It fits in nicely with the new solver, as type functions are a cleaner alternative to magic functions.
|
||||
|
||||
## Design
|
||||
|
||||
### `setmetatable` Type Function
|
||||
|
||||
In the following code example, `Identity` should evaluate to `{ sound: string, @metatable: { __index: typeof(animal) } }`:
|
||||
|
||||
```luau
|
||||
local animal = {}
|
||||
type Identity = setmetatable<{
|
||||
sound: string
|
||||
}, {
|
||||
__index: typeof(animal)
|
||||
}>
|
||||
```
|
||||
|
||||
When a non-table type (number, boolean) is passed into the type function, it should result in an error. This includes classes.
|
||||
If the type is an intersection or union of tables, then the metatable should be applied to every component type.
|
||||
If the second argument of the type function isn't a table, then it should error.
|
||||
If the first argument of `setmetatable` has a metatable with the `__metatable` field specified, then it should also report an error.
|
||||
|
||||
### `getmetatable` Type Function
|
||||
|
||||
In the following code example, `ReversedIdentity` should evaluate to `{ __index: typeof(animal) }`:
|
||||
|
||||
```luau
|
||||
local animal = {}
|
||||
type Identity = setmetatable<{
|
||||
sound: string
|
||||
}, {
|
||||
__index: typeof(animal)
|
||||
}>
|
||||
|
||||
type ReversedIdentity = getmetatable<Identity>
|
||||
```
|
||||
|
||||
Due to `__metatable`, additional behavior needs to be met. In the following code example, `MetatableType` should evaluate to `"No metatable here!"`:
|
||||
|
||||
```luau
|
||||
type metatable = {
|
||||
__metatable: "No metatable here!"
|
||||
}
|
||||
|
||||
type MetatableType = getmetatable<setmetatable<{}, metatable>>
|
||||
```
|
||||
|
||||
If the type passed in does not support metatables in the type system (i.e. at the time of writing, is not any of a primitive, a singleton, a class, a table, or a union or intersection thereof), it should error when passed to `getmetatable`.
|
||||
If the type passed in is an intersection or union of things that support metatables, then it will return an intersection or union of those metatables respectively.
|
||||
If no metatable is present, but the type system supports metatables for that type, it should return the `nil` type indicating that there is no metatable present.
|
||||
|
||||
## Drawbacks
|
||||
|
||||
Introducing a large library of "global" type functions can clutter naming space for users.
|
||||
|
||||
The name of `setmetatable` has an interpretation of implying a side effect; tables are references, and `setmetatable` is a side-effecting function. It could be argued that `setmetatable` is not a good name for a type function which has no side effects. However, consistency is desirable. A slight rename of a global function could cause confusion, and likely cause mistakes to be made. Type functions also have a precedent of not allowing side effects, so therefore a `setmetatable` type function would not have ambiguity when it comes to side effects.
|
||||
|
||||
## Alternatives
|
||||
|
||||
Do nothing, and rely on unique metatable type syntax to achieve what a `setmetatable` type function would achieve. However, this still has the issue of `getmetatable`. It is flat out incorrect at the moment, and a type function would be an ideal solution to that problem. If a type function is introduced, then any user could reasonably expect a counterpart `setmetatable` type function. Both metatable type functions and metatable syntax can exist at the same time, and that might be ideal.
|
||||
|
||||
Do nothing. `typeof` solves the current issues with accessibility to the current object-oriented programming patterns, and the issue relating to getmetatable can be solved by overloads.
|
Loading…
Add table
Reference in a new issue