* Initial commit for UDTF * library names to luacase * table runtime representation still depends on VM * addressing comment: boolean is equivalent to true | false * getname() should return string not table * Added a section about type function type checking as one of future works * type functions can call other type functions but not regular functions * user configurable timeout added * Type functions can not refer to outer scopes * lType -> typelib * print to warn; error/return nothing to fail reduction * removed getname() as there aren't real usecases * fix typos in method names * Starlark approach without execution limit * big revision on library methods, reason for using userdata, illegal constructs in type functions; need to add support for generic types and list of supported libraries/globals * colon to invoke methods * comment: ~ to not * Added indexer methods for Class types * using colons instead of periods to invoke method calls * added globals/libraries to a list of illegal constructs * revising typelib library and method descriptions * renaming methods, overloading __eq __le __newindex, metatable as representation of tables * small fix to reflect table method name change to class method names * make nil, unknown, never, any an instance attribute of typelib; merge getindextype/getindexresulttype into one function * Removed overloading metamethods for now * added tbl:getprop(key) to obtain a single value * addprop(key, ty.nil) != delprop(key) * merge addprop() and delprop() into one * typelib.nil clashes with the reserved keyword, so changed to typelib.niltype * == operator overridden with __eq * Fixed rawget example * Added parameters to newtable function * Added parameters to newfunction function * Revised typelib.function * fix typelib.newfunction * Clarification to typelib.newfunction methods * fix arguments of Function.setparameters and Function.setreturns * Error for returning non-typelib instance * added parameters to typelib.newmetatable * Delete String.getmetatable() * information about pass by reference and typelib.copy() method * moved stringle, boolean, number to be an instance attribute of typelib * Update docs/user-defined-type-functions.md Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com> * Apply suggestions from code review Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com> * remove print() for first iteration * removed typelib.is*() and added typelib.type() instead * Update docs/user-defined-type-functions.md Co-authored-by: James <jamesramsauer@gmail.com> * Update docs/user-defined-type-functions.md Co-authored-by: Alexander McCord <11488393+alexmccord@users.noreply.github.com> * remove metatable from type * merge newmetatable with newfunction * some revisions + typelib->type * change typelib.type() to :is(...) * big revision with new information, features, supported methods, list of available globals * better example for long running type functions * combine getstringsingleton, getbooleansingleton * optional arguments does not mean you don't need to write it * forgot to finish my sentence Co-authored-by: aaron <aweiss@hey.com> * add information about type functions not being able to be exported * Forgot string and utf8 library * Add a note about increasing the list of available globals/libraries: * Make a big edit pass over the RFC * Update user-defined-type-functions.md * We'll call the library `types`, not `type`. * Opinionatedly clean up the types API. * Take into account read/write properties * Update examples to match revised API * Update docs/user-defined-type-functions.md Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com> * Update docs/user-defined-type-functions.md Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com> * Update docs/user-defined-type-functions.md Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com> * Update docs/user-defined-type-functions.md Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com> * Cleaning up the API a bit more. * Restructuring the API reference a bit. * Fix typo with type signature * Add missing code blocks * Update docs/user-defined-type-functions.md Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com> * Add a small paragraph clarifying the scoping and shadowing rules being the same as type aliases --------- Co-authored-by: vegorov-rbx <75688451+vegorov-rbx@users.noreply.github.com> Co-authored-by: James <jamesramsauer@gmail.com> Co-authored-by: Alexander McCord <11488393+alexmccord@users.noreply.github.com> Co-authored-by: aaron <aweiss@hey.com> Co-authored-by: aaron <aaronweiss@roblox.com> |
||
---|---|---|
.github/workflows | ||
docs | ||
mascot.png | ||
README.md | ||
TEMPLATE.md |
Background
Whenever Luau language changes its syntax or semantics (including behavior of builtin libraries), we need to consider many implications of the changes.
Whenever new syntax is introduced, we need to ask:
- Is it backwards compatible?
- Is it easy for machines and humans to parse?
- Does it create grammar ambiguities for current and future syntax?
- Is it stylistically coherent with the rest of the language?
- Does it present challenges with editor integration like autocomplete?
For changes in semantics, we should be asking:
- Is behavior easy to understand and non-surprising?
- Can it be implemented performantly today?
- Can it be sandboxed assuming malicious usage?
- Is it compatible with type checking and other forms of static analysis?
For new standard library functions, we should be asking:
- Is the new functionality used/useful often enough in existing code?
- Does the standard library implementation carry important performance benefits that can't be achieved in user code?
- Is the behavior general and unambiguous, as opposed to solving a problem / providing an interface that's too specific?
- Is the function interface amenable to type checking / linting?
In addition to these questions, we also need to consider that every addition carries a cost, and too many features will result in a language that is harder to learn, harder to implement and ensure consistent implementation quality throughout, slower, etc. In addition, any language is greater than the sum of its parts and features often have non-intuitive interactions with each other.
Since reversing these decisions is incredibly costly and can be impossible due to backwards compatibility implications, all user facing changes to Luau language and core libraries must go through an RFC process.
Process
To open an RFC, a Pull Request must be opened which creates a new Markdown file in docs/
folder. The RFCs should follow the template TEMPLATE.md
, and should have a file name that is a short human readable description of the feature (using lowercase alphanumeric characters and dashes only). Try using the general area of the RFC as a prefix, e.g. syntax-generic-functions.md
or function-debug-info.md
.
Every open RFC will be open for at least two calendar weeks. This is to make sure that there is sufficient time to review the proposal and raise concerns or suggest improvements. The discussion points should be reflected on the PR comments; when discussion happens outside of the comment stream, the points salient to the RFC should be summarized as a followup.
When the initial comment period expires, the RFC can be merged if there's consensus that the change is important and that the details of the syntax/semantics presented are workable. The decision to merge the RFC is made by the Luau team.
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.
Each RFC will be assigned a shepherd. This person will support you in getting feedback, request necessary changes, and ultimately either accept (and merge) the RFC or turn it away.
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.
When there's no consensus that the feature is broadly beneficial and can be implemented, an RFC will be closed. The decision to close the RFC is made by the Luau team.
Note that in some cases an RFC may be closed because we don't have sufficient data or believe that at this point in time, the stars do not line up sufficiently for this change to be worthwhile, but this doesn't mean that it may never be considered again; an RFC PR may be reopened if new data is available since the original discussion, or if the PR has changed substantially to address the core problems raised in the prior round.
Implementation
When an RFC gets merged, the feature can be implemented; however, there's no set timeline for that implementation. In some cases implementation may land in a matter of days after an RFC is merged, in some it may take months.
To avoid having permanently stale RFCs, in rare cases Luau team can remove a previously merged RFC when the landscape is believed to change enough for a feature like this to warrant further discussion.
When an RFC is implemented and the implementation is enabled via feature flags, RFC should be updated to include "Status: Implemented" at the top level (before Summary section).