mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-04 10:50:54 +01:00
Add more accepted RFCs.
This commit is contained in:
parent
e2176e35e1
commit
da39486e13
3 changed files with 115 additions and 1 deletions
|
@ -35,7 +35,7 @@ When the initial comment period expires, the RFC can be merged if there's consen
|
|||
|
||||
When revisions on the RFC text that affect syntax/semantics are suggested, they need to be incorporated before a RFC is merged; a merged RFC represents a maximally accurate version of the language change that is going to be implemented.
|
||||
|
||||
In some cases RFCs may contain conditional compatibility clauses. E.g. there are cases where a change is potentially not backwards compatible, but is believed to be substantially beneficial that it can be implemented if, in practice, the backwards compatibility implications are minimal. As a strawman example, if we wanted to introduce a non-context-specific keyword `globally_coherent_buffer`, we would be able to do so if our analysis of Luau code (based on the Roblox platform at the moment) informs us that no script in existence uses this keyword. In cases like this an RFC may need to be revised after the initial implementation attempt based on the data that we gather.
|
||||
In some cases RFCs may contain conditional compatibility clauses. E.g. there are cases where a change is potentially not backwards compatible, but is believed to be substantially beneficial that it can be implemented if, in practice, the backwards compatibility implications are minimal. As a strawman example, if we wanted to introduce a non-context-specific keyword `globallycoherent`, we would be able to do so if our analysis of Luau code (based on the Roblox platform at the moment) informs us that no script in existence uses this keyword. In cases like this an RFC may need to be revised after the initial implementation attempt based on the data that we gather.
|
||||
|
||||
In general, RFCs can also be updated after merging to make the language of the RFC more clear, but should not change their meaning. When a new feature is built on top of an existing feature that has an RFC, a new RFC should be created instead of editing an existing RFC.
|
||||
|
||||
|
|
61
rfcs/change-assert-return.md
Normal file
61
rfcs/change-assert-return.md
Normal file
|
@ -0,0 +1,61 @@
|
|||
# Change `assert` to return one value
|
||||
|
||||
> Note: this RFC was adapted from an internal proposal that predates RFC process
|
||||
|
||||
## Summary
|
||||
|
||||
Change `assert` to only return the first argument on success
|
||||
|
||||
## Motivation
|
||||
|
||||
Today `assert()` function has confusing semantics:
|
||||
|
||||
`assert(x)` fails if `x` is falsy and returns `x` otherwise (this is okay!)
|
||||
`assert(x, y)` fails with `y` as the error message if `x` is falsy, and returns `x, y` otherwise (why?)
|
||||
`assert(x, y, z)` fails with `y` as the error message if `x` is falsy, and returns `x, y, z` otherwise (why???)
|
||||
|
||||
It's not clear what purpose is there behind returning more than one argument from `assert`, as it doesn't seem like it can be useful.
|
||||
|
||||
Specifically, when a two-argument form is used, the second argument must be an error message (otherwise if the first argument is falsy, the error behavior will be confusing); if it *is* in fact an error message, it's not clear why it's useful to return that error message when the first argument was truthful.
|
||||
|
||||
When a three-argument form is used, the third argument is just ignored and passed through, again for no apparent benefit.
|
||||
|
||||
## Design
|
||||
|
||||
This proposal argues that long term it's cleaner for us to fix the semantics here to just return the first argument. Any extra arguments will be ignored at runtime as usual, but our type checker can start treating these as invalid.
|
||||
|
||||
To be more precise, we'd be formally switching from
|
||||
|
||||
```
|
||||
declare function assert<T...>(...: T...): T...
|
||||
```
|
||||
|
||||
to
|
||||
|
||||
```
|
||||
declare function assert<T>(value: T, error: string?): T
|
||||
```
|
||||
|
||||
This allows us to align the runtime behavior of `assert` to the type definition and reduces the chance of an accidental error - since all return values are forwarded when the call is used as the last function argument, this code transformation is safe:
|
||||
|
||||
```
|
||||
- foo(bar)
|
||||
+ foo(assert(bar))
|
||||
```
|
||||
|
||||
But this code transformation may be unsafe if `foo` has optional parameters:
|
||||
|
||||
```
|
||||
- foo(assert(bar))
|
||||
+ foo(assert(bar, "bar should not be nil"))
|
||||
```
|
||||
|
||||
After this proposal the transformation becomes safe.
|
||||
|
||||
## Drawbacks
|
||||
|
||||
This may break user code. Unfortunately there is only one way for us to find out if it does, which is to roll this change out. We will use release notes to communicate the change and, if concerns are raised by the community ahead of time, we will investigate them.
|
||||
|
||||
## Alternatives
|
||||
|
||||
Alternatively we could keep `assert` behavior exactly as it is; this would match mainline Lua but we want to make the language better and if we were designing the standard library again we'd never make this choice, hence the proposal.
|
53
rfcs/function-table-freeze.md
Normal file
53
rfcs/function-table-freeze.md
Normal file
|
@ -0,0 +1,53 @@
|
|||
# table.freeze
|
||||
|
||||
> Note: this RFC was adapted from an internal proposal that predates RFC process
|
||||
|
||||
## Summary
|
||||
|
||||
Add `table.freeze` which allows to make a table read-only in a shallow way.
|
||||
|
||||
## Motivation
|
||||
|
||||
Lua tables by default are freely modifiable in every possible way: you can add new fields, change values for existing fields, or set or unset the metatable.
|
||||
|
||||
Today it is possible to customize the behavior for *adding* new fields by setting a metatable that overrides `__newindex` (including setting `__newindex` to a function that always errors to prohibit additions of new fields).
|
||||
|
||||
Today it is also possible to customize the behavior of setmetatable by "locking" the metatable - this can be achieved by setting a meta-index `__metatable` to something, which would block setmetatable from functioning and force metatable to return the provided value. With this it's possible to prohibit customizations of a table's behavior, but existing fields can still be assigned to.
|
||||
|
||||
To make an existing table read-only, one needs to combine these mechanisms, by creating a new table with a locked metatable, which has an `__index` function pointing to the old table. However, this results in iteration and length operator not working on the resulting table, and carries a performance cost - both for creating the table, and for repeated property access.
|
||||
|
||||
## Design
|
||||
|
||||
This proposal proposes formalizing the notion of "read-only" tables by providing a new table functions:
|
||||
|
||||
`table.freeze(t)`: given a non-frozen table t, freezes it; fails when t is not a table or is already frozen. Returns t.
|
||||
`table.isfrozen(t)`: given a table t, returns a boolean indicating the frozen status; fails when t is not a table.
|
||||
|
||||
When a table is frozen, the following is true:
|
||||
|
||||
- Attempts to modify the existing keys of the table fail (regardless of how they are performed - via table assignments, rawset, or any other methods like table.sort)
|
||||
- Attempts to add new keys to the table fail, unless `__newindex` is defined on the metatable (in which case the assignment is routed through `__newindex` as usual)
|
||||
- Attempts to change the metatable of the table fail
|
||||
- Reading the table fields or iterating through the table proceeds as usual
|
||||
|
||||
This feature is useful for two reasons:
|
||||
|
||||
a) It allows an easier way to expose sandboxed objects that aren't possible to monkey-patch for security reasons. We actually already have support for freezing and use it internally on various builtin tables like math, we just don't expose it to Lua.
|
||||
|
||||
b) It allows an easier way to expose immutable objects for consistency/correctness reasons. For example, Cryo library provides an implementation of immutable data structures; with this functionality, it's possible to implement a lighter-weight library by, for example, extending a table with methods to return mutated versions of the table, but retaining the usual table interface
|
||||
|
||||
To limit the use of `table.freeze` to cases when table contents can be freely manipulated, `table.freeze` shall fail when the table has a locked metatable (but will succeed if the metatable isn't locked).
|
||||
|
||||
## Drawbacks
|
||||
|
||||
Exposing the internal "readonly" feature may have an impact on interoperability between scripts - for example, it becomes possible to freeze some tables that scripts may be expecting to have write access to from other scripts. Since we don't provide a way to unfreeze tables and freezing a table with a locked metatable fails, in theory the impact should not be any worse than allowing to change a metatable, but the full extents are unclear.
|
||||
|
||||
There may be existing code in the VM that allows changing frozen tables in ways that are benign to the current sanboxing code, but expose a "gap" in the implementation that becomes significant with this feature; thus we would need to audit all table writes when implementing this.
|
||||
|
||||
## Alternatives
|
||||
|
||||
We've considered exposing a recursive freeze. The correct generic implementation is challenging since it requires supporting infinitely nested tables when working on the C stack (or a stackless implementation that requires heap allocation); also, to handle self-recursive tables requires a separate temporary tracking table since stopping the traversal at frozen sub-tables is insufficient as their children may not have been frozen. As such, we leave recursive implementation to user code.
|
||||
|
||||
We've considered exposing thawing. The problem with this is that freezing is required for sandboxing, and as such we'd need to support "permafrozen" status that is separate from "frozen". This complicates implementation and we didn't find compelling use cases for thawing - if it becomes necessary we can always expose it separately.
|
||||
|
||||
We've considered calling this "locking", but the term has connotations coming from multithreading that aren't applicable here, and in absence of unlocking, "locking" makes a bit less sense.
|
Loading…
Add table
Reference in a new issue