2025-01-23
Writing this changelog from the train, as the ICE approaches Berlin Südkreuz. More fixes to the generics system, tightening the type system in the presence of interfaces. I'd like to especially thank Steve Vermeulen, Michael Dowling and Corey Williamson for their bug reports, test cases and feedback on these issues, helping to get Teal 0.24 to a more stable shape.
This release features commits by Pierre Chapuis, André L. Alvares and Hisham Muhammad.
- Fixed the resolution of type arguments in generics, especially for function calls (#905, #908, #909, #910)
- Fixed the resolution of the
__call
metamethod for generics. (#904) - Fixed the subtype checking of generic records against generic interfaces. (#859)
- Fixed an error when matching against an undefined type.
- Bumped the pinned compat53 dependency, which had a build issue on Windows. (#916)
2025-01-12
A minor bugfix release, thanks to feedback received upon the release of 0.24.2.
This release features commits by Hisham Muhammad.
- Fixes code generation of nested generics that should not be elided (#902).
- Fixes a regression in generic records with a __call metamethod (#901).
- Record fields declared with self type were not being properly type checked in table literals (#846).
2025-01-10
Another bugfix release for Teal 0.24. We got more feedback, bug report, test cases.
The good news is that the arity changes went very smoothly: no bug reports or complaints about that big change, really. The lack of feedback means that people must have adapted well to the new behavior, and that the pragma support and documentation on how to migrate was sufficient. A big win!
The bad news is that the added complexity to the type system caused by the
introduction of interfaces and the self
type, have produced various bugs.
Those came up especially due to the interactions between interfaces, the
self
type and generics. Most bugs were fixed for this release, but not all.
This release features commits by Pierre Chapuis, JR Mitchell, Corey Williamson, @FourierTransformer, @FractalU and Hisham Muhammad.
record
declarations nested insideinterface
declarations are no longer allowed; allowing a concrete object inside an abstract object was an oversight.- Relational operators on aliases to comparable types now correctly return boolean
- Nominal function types can now be used as type variable constraints.
- Improved standard library definitions
io.input
andio.output
typesstring.find
- Compiler now produces warnings for unread variables.
- Record fields inherited from interfaces now resolve
self
on declaration (#877). - Fixes parsing of
.lua
files, respecting Lua heuristics for function calls across multiple lines. - Fixes a syntax error with
return;
is not the last statement of a block (#893). - Eager resolution of unused type arguments (#881).
- Nested generic type aliases work with eager resolution (#888).
- Fixes support for generics on array types (#880).
- Fixes support for generic unions (#787).
- Check if number of types in declaration exceeds number of variables (#868).
- API fixes:
tl.get_types
reports all inherited interface fields- reports
self
symbol in methods
- Internal Compiler Error when declaring an invalid union in a function arg in a field (#856)
- Crash on a
for
loop that explicitly usesnext
and an unknown variable. - Stack overflow when a nested interface extends its parent interface.
- Error messages for
>
no longer have its operands flipped - Fix: do not get file names and locations out-of-sync when compiling multiple files.
- CLI: compatibility fix for Windows.
2024-10-25
A bugfix release for Teal 0.24. Several fixes that bring the implementation closer to the vision of what the 2024 edition is intended to be, thanks to the invaluable feedback of the user community!
This release features commits by François Perrad, Steve Vermeulen and Hisham Muhammad.
- Standard library
- Improved signatures for
math.frexp
andmath.ldexp
- Improved signatures for
As expected, fixes are mainly related to the new features introduced in Teal 0.24:
- Better handling of abstract vs. concrete types
- Fixed subtyping and casting checks for records vs. interfaces (#830).
- Disallow passing non-record types as arguments (#813).
- Allow interface instances to be required concretely
- Always elide abstract types from generated code
- Fixes for improved type inference
- Fixed type inference for
is x == nil or ...
(#823). - Resolve type parameter in
setmetatable
using nested records (#772). - Ignore non-literal booleans in table key declarations (#816).
- Type variables from function results also resolve arguments (#838).
- Do not mistake missing optional argument for a vararg.
- Fixed type inference for
- Improved code generation
- Generate
__is
-aware code foris
on unions (#742). - Handles metamethod for
__eq
(#814).
- Generate
- Fixes for
self
type- Fixed a nil reference exception (#824).
- Corrected resolution of
self
in methods (#812).
2024-10-07
This a big release! Teal 0.24.0, codenamed "Teal Spring '24" as a nod to
the coolest language release name ever ("Clipper Summer '87"), is a
culmination of work that has spanned multiple years, and it would not be
possible without the amazing feedback given by the community on the next
branch, where this was developed. This release does include that "lot of
new code" that I wanted to merge in the "near future", which I referred
to in the changelog entry for the previous release. It is nice to see it
all coming together!
The main feature is the addition of interfaces, which introduces a model for subtyping table types in the language. That should allow for representing the various kinds of object models that are used across the Lua ecosystem, without promoting any particular class/object system over the others. Most of the other changes, such as macro expressions, came along as related features built to support this. The support for optional argument markers is a stop towards stricter type safety, and the addition of pragma markers to control this feature is a statement of intent for allowing gradual transitions as the language evolves. All in all, the goal is to make the 2024 edition of Teal a safer and more pleasant language to use.
And yes, an October release is called Spring because it is now spring in the southern hemisphere. :)
This release features commits by François Perrad, Victor Ilchev and Hisham Muhammad.
- Interfaces: you can now declare abstract interfaces, and record types
can implement them. This allows you to declare subtyping relations, and
better support inheritance models (e.g.
local record Circle is Shape
).- Records and interfaces may declare a
where
clause with an expression overself
, which allows theis
operator to discriminate the type. - With discriminated record types, it is now possible to declare unions over multiple record types.
- "Arrayrecords" are no longer a distinct type: they are just records
that implement an array interface (e.g.
local record R is {T}
) - Type variables in functions can now have constraints on interfaces
(e.g.
function my_func<F is MyInterface>(...)
) self
is now a valid type that can be used when declaring arguments in functions declared in interfaces and records- When a record is declared to be a subtype of an interface using
is
, any function arguments usingself
in the parent interface type will then resolve to the child record's type.
- When a record is declared to be a subtype of an interface using
- Records and interfaces may declare a
- Optional/required arguments in function calls - functions may declare
arguments as optional, affecting the required arity of function calls.
The rightmost arguments of a function can have their variable names
annotated with a
?
sign, indicating that the argument does not need to be passed in a function call. This refers only to the presence of the argument in a call, and not to its type or nullability: a required argument may still be called with an explicitnull
, but an optional argument may be elided.- In previous versions of Teal, all function arguments were effectively
optional. This means that Teal is now stricter which checking for
function calls. You can use
--feat-arity=off
in the command line orfeat_arity = "off"
intlconfig.lua
to obtain the previous behavior. - To convert code that uses the old arity checking rules to the new
behavior, you can also use compiler pragmas in the code,
--#pragma arity off
and--#pragma arity off
to disable or enable stricter arity checks.
- In previous versions of Teal, all function arguments were effectively
optional. This means that Teal is now stricter which checking for
function calls. You can use
- Macro expressions: you can declare a restricted form of function called
a
macroexp
, which is always expanded inline. These can also be used to declare compile-time metamethods, which expand without requiring a metatable at runtime. Thewhere
clauses used in interfaces and records are syntax sugar for macro expressions that implement a pseudo-metamethod__is
. - Dynamic
require
calls that do not take a module name as a literal string are now allowed, and returnany
; you can load a static type definition usinglocal type MyType = require("...")
and then cast your dynamic require like so:local my_mod = require(var) as MyType
- The type system is more nominal: all named types are now treated nominally,
except for unions, which are always type aliases. Previously record types
were nominal, but a named typed that resolved to a primitive such as
integer
was structural. There are still subleties on the rules for when alocal type
produces a new distinct type or an alias; they are explained in the docs. - The
<total>
attribute for variables can now only be applied when initializing variables with literal tables. This is a minor breaking change, but the usefulness of this attribute in other cases was very limited, and it also produced misleading results. - Improved type signatures in the standard library
select
produces variadic returns
- Simplified API in the
tl
module.- The 0.15 API is still supported for backwards compatibility.
tl build
was removed. Cyan should be used instead as the build tool for Teal projects.
- Fix commits included in this release refer both to bugs present
in the
master
branch as well as fixes specific to the 0.24 series, reported by the community during the beta testing of this release. The Git history and the GitHub issues list contain a more detailed accounting of the bugfixes that went into this release. Some of the fixes include:tl check
now reports if the input file does not exist.- Reporting location of the end of an
if
block correctly. - No longer crash if a
require()
epression fails to resolve (#778). tl types
now reports the types of variables infor
loops.- Improved error message when calling functions with insufficient arguments.
- Disallowing using a base type as a type variable name.
- Type arguments resolving correctly in recursive functions.
- Type arguments resolving correctly in nested records (#754).
or
type inference between types with a subtyping relation resolves to whichever is the larger type.- Reporting an error if an iterator used in a generic
for
does not declare a return value type (#736). - When checking
<total>
values, no longer reporting record function and metamethods as missing fields (#749). - Localizing a record no longer makes the local a type (#759).
- Bad assignments of record tables are reported (#752).
- Nominal type alias declarations work as expected (#238).
- Nominals with generics can be resolved correcty (#777).
- Nested types are not closed too early (#775).
- Not failing when resolving nested empty tables.
- Exporting generics and type aliases from modules correctly
using
return
at the toplevel (#804).
2023-11-05
It's been a long time since the last release! Lots of fixes got merged in, and there's a lot of new code that I want to merge in the near future. So, let's get a more fix-oriented release out before more significant changes.
This release features commits by @JLPLabs, François Perrad, JR Mitchell and Hisham Muhammad.
- Accept return-like syntax for vararg inputs in function type signatures
(e.g.
function(string...): string...
) - Parser no longer accepts annotations (e.g.
<const>
) infor-in
variables. It used to accept but ignore them before. - New warning for
or
between incompatible types in a context where no union is expected. - Record functions now implicitly inherit type variables from its parent record's definition. (#669)
- Standard library improvements:
- improved return values for
debug.getlocal
- improvements to
utf8.codes
signature io.write
andfile:write
signatures also accept numbersfile:close
signature now produces Lua 5.2+ returns (and matches Lua 5.2+ behavior thanks to update in Compat-5.3 library)- file reader functions (
io.read
,io.lines
,FILE:read
,FILE:lines
) now support the"n"
/"*n"
pattern
- improved return values for
- No longer produces spurious warnings when using methods with
pcall
. (#668) - Better acceptance of
record
andenum
as soft keywords. is
inference now takes into account that unions also imply nil, therefore accepting anelse
case even if all cases of an union are tested inif
/else if
branches. (#699):
-style function calls also check for the__index
metamethod. Also avoids mistaking__index
for an incorrect method use.- Resolves type arguments when both types in an
or
expression match the expected type. (#709) - Checks records nominally when resolving metamethods for operators. (#710)
- Avoids confusing a localized global for an automatic type narrowing when reporting unused variables.
- When generating compatibility bit32 code, always do it, even for invalid variables, instead of emitting incompatible operators.
- Compiler no longer crashes if type variable resolution fails when setting an inference location.
- Compiler no longer crashes when checking for a reassigned const global.
- Unused local types no longer emit dummy variables.
tl
for many files is much faster as it avoids running the GC on every file.- Package loader now populates the
...
tuple
2023-04-27
This is primarily a bugfix release, but it does include some significant quality-of-life improvements when using record methods.
This release features commits by JR Mitchell, @IcyLava and Hisham Muhammad.
- Improved checks over uses of
self
in forward-declared methods:- In a
record
declaration, when fields of typefunction
are declared within a and they have a matchingself
first argument, those are now treated as methods (similar to explicitfunction R:method
declarations using colon-notation), so misuses of:
versus.
can be detected in those as well.
- In a
- Warn when invoking a method as a function, even if the type of the first non-self argument matches.
- Standard library improvements:
- Add optional third argument for
string.rep
- Add optional third argument for
- Public signature of
tl.gen
features second return value of typeResult
- Acept record method declaration for nested records. (#648)
- Do not consider an inner type to be a missing key when declaring a total table. (#647)
- Change inference order in function calls to better match programmer
intent: infer
self
(which is usually explicitly given) before return values (which are implied by context). - Preserve narrowed union in a loop if the variable is not assigned. (#617)
- Don't miss a "unused variable" warning when it is a narrowed union.
- Fixes the detection of record method redeclaration. (#620)
- Fixes the error message when number of return arguments is mismatched. (#618)
2023-01-23
A minor bugfix release, thanks to feedback received upon the release of 0.15.0.
This release features commits by Hisham Muhammad.
- Fixes resolution of nominals across scope levels.
- Fixes a crash when resolving nested records in a record function. (#615)
- Avoid spurious inference warnings in local declarations.
2023-01-20
Two new language features, a bunch of inference improvements, some more stricter checks for correctness, and a bunch of fixes! There's a lot to unpack in this release, but all changes should be backwards-compatible for correct code, so migration should be smooth for end-users of the compiler.
However, if you're using tl
module API directly, please note that there are
some API changes, listed below.
This release features commits by Li Jin, Carl Lei, Yang Li, Pierre Chapuis, @lenscas, Stéphane Veyret, and Hisham Muhammad.
- Type-only
require
:- Adds new syntax for requiring a module's exported type
without generating a
require()
call in the output Lua:local type MyType = require("mytype")
-- circular definitions oflocal type
-required types are allowed, as long as the type is only referenced but its contents are not dereferenced.
- Adds new syntax for requiring a module's exported type
without generating a
- New variable attribute
<total>
:- It declares a const variable with a table where the domain of its keys are totally declared.
- This check can only be applied when a literal table is given at the time of variable initialization, and only for table types with well-known finite domains: maps with enum keys, maps with boolean keys, records.
- Note that the requirement is that keys are declared: they may still be explicitly declared to be nil.
- Type inference improvements:
- Improved flow-typing in
if
blocks - if a block ends withreturn
, then the remainder of the function can infer the negation of its condition. - In contexts where the expected type of an expression
is known (e.g. assignments, declarations with explicit
type signatures), the return type a function call can
now determine the type argument of a function.
- In particular,
local x: T = setmetatable({}, mt)
can now infer that{}
isT
without a cast.
- In particular,
- When initializing a union with a value, that value is now used by the flow-typing engine to narrow the variable's concrete known type.
- Improved flow-typing in
- Local functions can be forward-declared with a
local
variable declaration and then implemented with barefunction
notation, more similarly to how record functions (and Lua functions) work. - Handling of
.
and index notations is now more consistent for both maps and records:- Map keys can now use
.
-notation like records, and get the same checks (e.g. if a key type is a valid enum) __index
metamethod works for.
-notation as well.
- Map keys can now use
- Some stricter checks:
- Unused type arguments in function signatures are now flagged as an error.
- Using the
#
operator on a numeric-keyed map now produces a warning, and on a non-numeric-keyed map it produces an error. - Redeclaration of boolean keys in a map literal are now an error.
- Cannot declare a union between multiple tuple types as these can't be discriminated.
- Cannot redeclare a record method with different type signatures; produces a warning when redeclaring with the same type signature.
- Standard library improvements:
pcall
andxpcall
have vararg return types
- Fixed check between
or
expression and nominal unions. - Fixed function call return values when returning unions with type arguments. (#604)
- Fixed an error when exporting type aliases. (#586)
__call
metamethods that are declared using a type alias now resolve correctly. (#605)- Generic return types are more consistenly checked, potentially reporting errors that went undetected before.
- Fixed type variable name conflicts in record functions, ensuring nested uses of generic record functions using type variables don't cause conflicts. (#560)
- Fixed the inference of type arguments in return values. (#512)
- Fixed the error message for redefined functions. (#566)
- Fixed a case where a userdata record type is not resolved properly. (#585)
- Fixed some issues with the
<close>
attribute. tl warnings
no longer misses some warning types.
- Report error on unterminated long comments instead of generating invalid code.
- Cleaner code is produced for
is nil
checks.
API changes for more consistent processing of syntax errors:
tl.lex
changed: it now returns an{Error}
array instead of a list of error Tokens.- from
function tl.lex(input: string): {Token}, {Token}
- to
function tl.lex(input: string, filename: string): {Token}, {Error}
- from
tl.parse_program
changed: it no longer returns the first integer argument, which as always ignored by every caller.- from
function tl.parse_program(tokens: {Token}, errs: {Error}, filename: string): integer, Node, {string}
- to
function tl.parse_program(tokens: {Token}, errs: {Error}, filename: string): Node, {string}
- from
tl.parse
is a new function that combinestl.lex
andtl.parse_program
the right way, avoiding previous mistakes; this is the preferred function to use moving forward.- signature:
function tl.parse(input: string, filename: string): Node, {Error}, {string}
- signature:
- Better handling of newlines on Windows.
tl types
: in function calls of polymorphic functions, now the return type of the resolved function is reported.- More informative pretty-print of polymorphic function signatures in error messages.
- More informative error message for field assignment failures.
- Improved error message for inconsistent fields (#576)
- The parser now does some lookahead heuristics to provide nicer error messages.
- The compiler now detects mis-aligned
end
keywords when reporting about unmatched blocks, which helps pointing to the place where a mistake actually occurred, instead of the end of the file. - For debugging the compiler itself:
- new environment variable mode
TL_DEBUG=1
inspects AST nodes and types as program is compiled. TL_DEBUG
with a negative value-x
will run likeTL_DEBUG=1
but halt the compiler after processing input linex
.
- new environment variable mode
2022-08-23
And a week after 0.14.0, here's 0.14.1, with bugfixes to a few regressions identified by users. This is essentially a bugfix release, with a few improvements triggered via regressions or confusing error messages.
This release features commits by Hisham Muhammad.
- Improved error reporting on declarations with missing
local
/global
- Ensured
exp or {}
with empty tables continues to work as it used to in 0.13.x, skipped more advanced flow-typing as simply returning the type of the left-hand sideexp
. - Improved propagation of invalid types in unions
- Avoids weird messages such as
got X | Y, expected X | Y
when "Y" is invalid in the scope, but was valid on declaration of a function argument, for example.
- Avoids weird messages such as
- Fix missing bidirectional flow of expected types in
return
expressions and expand it on string constants. (#553)- This is almost a feature as flow-checking was made smarter for
string constants, but it takes the shape of a regression test because
return
expression checks were incomplete and caused some correct code to pass type checking before which stopped passing on 0.14.0 when some checks were extended. We further extend them here to make more valid code pass.
- This is almost a feature as flow-checking was made smarter for
string constants, but it takes the shape of a regression test because
- Fixes a crash when resolving a "flattened" union. (#551)
2022-08-16
It's been a year! It was good to let the language sit for a while and see what kind of feedback we get from real world usage. This has also been an opportunity to get bug reports and fixes in, to help further stabilize the compiler.
This release features commits by Li Jin, Enrique García Cota, Patrick Desaulniers, Mark Tulley, Corey Williamson, @Koeng101 and Hisham Muhammad.
- Global functions now require
global function
, you can no longer declare them with a barefunction
; this makes it more consistent with other global variables. (#500) - Forward references for global types:
global type Name
can be used to forward-declare a type and use it, for example, in record fields, even if it is declared in another file. (#534)
- Polymorphic functions (as declared in record methods) can be used
as iterators for a generic
for
. (#525) - You can no longer define a global with the same name as a local which is already in scope. (#545)
- Initial support for the
__index
metamethod: you can declare it as a function. (#541) - Dropped support for the old backtick syntax for type variables.
- Type checker checks validity of Lua 5.4
<close>
annotations:- the output target must be set to "5.4"
- there can be only one
<close>
variable per declaration (to ensure correct order of__close
metamethods being called) - the type of the variable is closable
- globals can't be
<close>
<close>
variables can't be assigned to
- Nested type arguments are now valid syntax without requiring a
disambiguating space. The parser used to reject constructs of the
type
X<Y<T>>
due to ambiguity with>>
; it is now accepted (old time C++ coders will recall this same issue with C++98!) - Standard library improvements:
math.maxinteger
andmax.mininteger
are always available (in any Lua version, with or without lua-compat-5.3 installed)
- Local types are now elided from generated code if they are never
used in a concrete value context: if you declare a
local type T
such as a function alias, it no longer produces alocal T = {}
in the Lua output.
tl.pretty_print_ast
now takes aTargetMode
argument to specify flavor of Lua to produce in output:- new
TargetMode
for outputting Lua 5.4-specific code including<const>
and<close>
annotations.
- new
tl.load
now always runs the type checker, which is needed to produce type information required to output correct code in some cases. The default behavior, however, remains backwards compatible and allows code to run in the presence of type checking errors. You can override this behavior and maketl.load
returnnil, err
in case of type checking errors by using a "c" prefix in the mode flag (e.g. "ct", "cb", "cbt").tl.type_check
returns two values.
- Project CI now runs on GitHub Actions
- Windows portability improvements
- Windows and Mac now get tested regularly in the GitHub Actions CI!
- New hint warning when the value of a function with multiple return values is being returned as part of an expression, causing the additional returns to be lost.
- Improved error messages for map errors.
tl check
avoids re-checking files already loaded in the environment: this allows you to check yourglobal_env_def
module without getting redeclaration errors.- Build script for stand-alone binary is now included in the repository.
- Fixed resolution of type variables to avoid clashes when nested uses of generic functions used the same type variable name. (#442)
- Map index keys are checked nominally, not structurally. (#533)
- Fixed flow checking of enums in
or
. (#487) - Properly return type errors for fields when comparing records. (#456)
- Fixed a stack overflow when comparing unions with type variables. (#507)
- Fixed a stack overflow in record comparisons. (#501)
- It now reports the location of a redeclared record correctly. (#542)
- Fixed resolution of nested type aliases (#527) and nested nominals (#499).
- Fixed check of inconsistent record function declarations. (#517)
tl.loader
reports correct filename in debug information. (#508)- Fixed error message when
tlconfig.lua
contains errors. - Tables can't be assigned to record types declared to be
userdata
. (#460) - Fix check of arrays of enums in overloaded record functions.
- Fixed crash when attempting to declare a record method on an unknown variable. (#470)
2021-07-30
A long while since the last release! This one contains a bunch of fixes, but no significant language changes. Trying to keep the language stable for a while so that we better understand what's needed and what the priorities should be, as more people use it.
This release features commits by Corey Williamson, Patrick Desaulniers, Enrique García Cota, Francisco Castro, Jason Dogariu, @factubsio @JLPLabs, @sovietKitsune, @DragonDePlatino and Hisham Muhammad.
- Generic
for
now accepts records that implement a__call
metamethod - Allows
#
to be used on tuple tables - Metamethods for
__tostring
,__pairs
,__gc
can be declared - Standard library definition improvements:
collectgarbage
has smarter argumentserror
: first argument can be anythingos.time
: accepts a table as inputtable.concat
accepts numbers in the array argument
tl.load
now behaves more like Lua'sload
, using the default environment if the fourth argument is not given.
- Report warning for excess lvalues correctly
tl
finds?/init.lua
in itsinclude_dir
entries as well- Updates to make test suite run on Windows
tl --quiet
silences configuration warnings- Documentation: various fixes and improvements
- New guide on
.d.tl
declaration files
- New guide on
- Fixes for compat code when using
--gen-target=5.1
- Various crash fixes for record method declarations:
- Fixed crash when inferring type for self
- Fixed crash when declaring methods on an untyped self
- Fixed crash when redeclaring a method already declared in the record definition
- Fixed a crash when traversing a
.
on something that's not a record - Error messages now report
...
in variadic function types
2021-03-19
A quick bugfix release, to correct the behavior of global_env_def
, which
wasn't being properly loaded from tlconfig.lua. Getting a quick release out
of the door so that people who were using preload_modules
don't stay on
an old version just because of that!
This release features commits by Kim Alvefur.
- Fixed loading
global_env_def
from tlconfig.lua.
2021-03-18
A long awaited feature arrives: integers!
Teal now has a separate type integer
for integers, and number
continues to
be the general floating-point value.
The integer type works for any Lua version, but from Teal's point of view it has unknown precision. The actual precision depends on your Lua VM: if you're running on Lua 5.3 or above, you get 64-bit integers; if you're running Lua 5.2 or below or LuaJIT, the implementation still generates regular numbers, but it checks that operations declared to work on integers will only use values declared as integers, which is useful even if the VM does not have a native integer value type.
See more details about the behavior below, as well as a quick guide to help you in the conversion process!
Apart from integers, this release features your usual assortment of tooling tweaks and bugfixes!
This release features commits by Kim Alvefur and Hisham Muhammad.
- Integer type!
integer
is a new type of undefined precision, deferring to the Lua VM- integer constants without a dot (e.g. 123) are of type integer
- operations preserve the integer type according to Lua 5.4 semantics
(e.g.
integer + integer = integer
,integer + number = number
, etc.) integer
is a subtype ofnumber
- this generally means that an
integer
is accepted everywhere anumber
is expected - note that this does not extend to type variables in generics,
which are invariant:
Foo<integer>
is not the same asFoo<number>
- if a variable is of type
integer
, then an arbitrarynumber
is not acceptedlocal x: integer = 1.0
doesn't work, even though the floating-point number has a valid integer representation
- this generally means that an
- variables (and type variables) are inferred as integer when initialized with integers
- this can happen directly (
local x = 1
) or indirectly (local x = math.min(i, j)
) - to infer an integral value as a
number
, use .0:local x = 1.0
infers anumber
- this can happen directly (
- the standard library was annotated for integers
- it only returns integers, but it always accepts floats in arguments that
must be integral — this matches the behavior of the Lua standard library
implementation: wherever you need to pass an integral value, you can pass
an equivalent float with an integer representation, e.g.
table.insert(t, 1.0, "hello")
— therefore, this is also valid in Teal, but note that the validity of the float is not checked at compile time, this is a Lua standard library API check.
- it only returns integers, but it always accepts floats in arguments that
must be integral — this matches the behavior of the Lua standard library
implementation: wherever you need to pass an integral value, you can pass
an equivalent float with an integer representation, e.g.
- Redeclared keys in table literals are now an error.
- More improvements thanks to bidirectional inference:
- You can now make an
or
between values of two different types if the context expects a union.
- You can now make an
Quick guide for existing converting code to use integers - I tried to be as careful as possible in making the transition easier, but this will undoubtedly break some of your code — don't worry, fixing it is easy. Any new error may fall into one of two categories:
- Your value is meant to be an integer, but it is being detected as a
number
- if your variables are meant to be integers (indices in an array, counters,
line-column coordinates, etc.), most likely your code will already be using
them as
integer
becauselocal x = 0
now infers tointeger
. - If you explictly declared it as
local x: number
, just change it tolocal x: integer
. - Your functions will often feature arguments and return types declared
as
number
where they should now beinteger
, if they take or produce integer values.- This will cause, especially for return types, a line like
my_integer = func()
to produce an errorgot number, expected integer
;func
needs to be changed fromfunction func(): number
tofunction func(): integer
.
- This will cause, especially for return types, a line like
- To convert floating-point numbers to integers you can use
math
library functions such asmath.floor()
.
- Your value is meant to be a floating-point number, but it is being detected as an
integer
- if your variables are meant to be floating-point and Teal is inferring them
as integer because of their initialization value, the easiest thing is to
change them to floating-point constants, changing from
local dx = 0
tolocal dx = 0.0
- Due to subtyping, integers generally work wherever numbers are required,
but if you need to convert an integer to a number in an invariant context such
as a type variable in a generic, you can force it with a cast (
i as number
).
- new API function
tl.version()
, which returns the version of the compiler. - make unknowns a kind of warning, not a separate array
- only ever return Result values, do not take them as inputs
- each Result only contains errors for the given file
- use
env.loaded
(andenv.loaded_order
) to traverse all results and get all errors
- use
- renamed
preload_modules
(an array) toglobal_env_def
(a single entry). This option caused some confusion before, so we reworked the documentation to better explain its intent.- For example, to load global definitions for Love2D from a file
love.d.tl
, useglobal_env_def = "love"
in your tlconfig.lua file.
- For example, to load global definitions for Love2D from a file
tl run -l <module> <script>
(long form--require <module>
)- works like
lua -l
; type checks the module and requires it prior to running the script.
- works like
tl gen --check
now reports warnings- When processing
.lua
files, unknown variables are now a type of warning, so you can enable and disable them using the usual warning features from the CLI and tlconfig.lua - Some performance improvements in the compiler itself, thanks to some
profile-directed optimization (in other words, putting the most-used
branches first in long
if
/elseif
chains). tl types
now produces its JSON output in a stable ordertl types
now returns more information- types in function names,
for
variables
- types in function names,
- Nested type alias now work as expected
- Fix a crash when doing flow inference on inexistant variables (#409)
- Fix partial resolution of type arguments
- Some fixes related to arrayrecords
- Nested types now work correctly with arrayrecords
- Fixed use of self in functions declared in arrayrecords
2021-02-27
Time for a new release, with some significant updates to make the behavior of the language more smooth.
Language-wise, the big change is bidirectional inference for table literals,
meaning that the types of literal tables (that is, the ones you write with { ... }
in your code) now get more precise types when they are defined in
contexts that have type declarations, such as variables with explicit types,
or function arguments where the types are known.
There have been also various improvements in the API and tooling departments, all aiming to produce better development experience -- do keep an eye for exciting things thet are in the works in this area!
This release features commits by Patrick Desaulniers, Corey Williamson, and Hisham Muhammad.
- Bidirectional inference for table literals
- Type information propagates outside-in when a table literal is declared as the rhs value of a variable declaration, assignment or as a function argument.
- This fixed various outstanding bugs where unintuitive behavior was observed.
local type T1 = T2
now makes nominalT1
an alias for nominal typeT2
, and not a separate type (same applies toglobal type
).
preload_modules
functionality was moved fromtl.process
intotl.init_env
, so that API clients that use the lower-level operations can preload modules.tl.get_token_at
can return the token given a token list and a position.
tl gen
now has a--check
flag, which makes it reject code that does not typecheck and not generate the output.- Warnings output is now presented in a stable order.
tl types
has had various improvements generating better data for editor integration:- The symbols list is much shorter as empty or redundant scopes are eliminated.
- Incomplete expressions now output type information for the fragment that could be parsed.
- Function types look nicer in error messages and tooling.
- Unions of nominal types now reduce considering aliasing.
- Various fixes for the behavior of
tl types
.- The top level scope is now correctly closed at the end of file.
- Symbols list is now properly sorted.
- Function types point to the correct filenames.
2021-02-14
More bugfixes! This includes some behavior fixes in the flow inference, so some code that wasn't accepted before may be accepted now and vice-versa, but the overall behavior is more correct now. Apart from this, no language changes other than fixes.
This release features commits by Corey Williamson, @JLPLabs and Hisham Muhammad.
- Some more propagation of "truthiness" when flow-typing expressions:
assert
is now known by the compiler to be truthy when performing flow propagation of type information. This means that, for example, givenx
of typeA | B
, in an expression of the formx is A and assert(exp1(x)) or exp2(x)
,x
will be known to be of typeB
inexp2(x)
.
t or {}
is now only valid for table types- Standard library improvements
math.randomseed
accepts Lua 5.4 second argument
get_types
now returns numeric type codes, and more complete type data for IDE integration consumption- The parser collects the list of required modules, for use by later use by build tooling
- The type checker now collects the list of filenames resolved by
require
, for use by build tooling
- Performance improvements for the lexer
- Performance improvements for the JSON encoder used by
tl types
tl types
: proper escaping of JSON strings- some parser fixes to avoid crashes on incorrect input
2021-02-07
Having great users means having great feedback! Here's a quick bugfix release addressing a few issues that were reported by the community right after the release of Teal 0.11.0.
This release features commits by Hisham Muhammad.
- Fixed inference in cases that combine
is
and==
(#382) - Fixed tuple resolution in literal tables
- Fixed incorrect
<const>
in table inference (#383)
2021-02-07
The FOSDEM 2021 edition!
This new release does not include big language changes, but includes a lot of
new stuff! The new tl types
infrastructure for IDE tooling, build.tl
support in tl build
, code generation options, and tons of bugfixes.
This release features commits by Corey Williamson, @lenscas, Patrick Desaulniers and Hisham Muhammad.
- The module return type can be inferred anywhere at the top-level, not only at the outermost scope (#334)
- Records are compared nominally and all other types are compared structurally
- Flow inference added to the
==
operator- Testing for a value in an
if
test propagates it into the block, which is especially useful for constraining a string into an enum, for example
- Testing for a value in an
- Standard library improvements
debug
library
- New experimental command:
tl types
,- It produces a JSON report designed to be consumed by IDE tooling: for each input file, it lists all types used in the source code by line and column location, all symbols declared by scope, all types and globals
- Also includes experimental a Teal API, which is used to implement
tl types
:tl.get_types
, which produces the above report as a recordtl.symbols_in_scope
, which given a report, a line and column, reports all symbols visible at that location.
- This feature is still experimental because the compiler was extended to perform type checking on syntactically-incorrect inputs (because IDEs want to perform analysis on unfinished programs). Some incorrect inputs may cause the type checker it to stumble!
- Support for
build.tl
file intl build
, which allows you to run a build script written in Teal before your files are compiled (for example, to produce generated code) tl build
now looks fortlconfig.lua
in parent directories, so you can call it from anywhere in your project tree- Improved error messages and reduced the number of cascading syntax errors
- The compat53 dependency is now optional (pcall-loaded) in generated Lua
code. You can make it required or disable it completely using
gen_compat
intlconfig.lua
and--gen-compat={required|optional|off}
- You can now target the Lua version of the generated code with
gen_target
and--gen-target
. Supported modes are5.1
(for LuaJIT, Lua 5.1 and 5.2) and5.3
for (Lua 5.3 and Lua 5.4). - Friendly warnings on misuse of
pairs
/ipairs
- Type checker now runs in the package loader
- Preserves explicit semicolons in the input, to deal correctly with Lua grammar ambiguity
- New inference engine for
is
operator- Does negation correctly implementing DeMorgan's laws
- Fixed
is
for type arguments - Fixed resolution of metamethod fields in nested types (#326)
- Fixed type comparison for tuple elements
- Fixed invariant comparison between empty tables and nominals (#332)
- Fixed precedence of
+
relative to..
- A large number of parser fixes on bad input corner cases,
thanks to fuzz testing and user feedback, prompted by the
tl types
work!
2021-01-07
Fixes, lots of fixes! We shipped a bunch of stuff in 0.10, so it's time to clean some rough edges, thanks to the great feedback we've received in that release. No big changes in the language this time (other than some things that should have worked but didn't now do!). In the tooling department, we now have configurable warnings, so you can enable or disable them by category. Also, stay tuned for the next Teal meetup later this month!
This release features commits by Darren Jennings, Corey Williamson and Hisham Muhammad.
- Allow using metamethods such as
__call
and operators on therecord
prototype itself (#313) - Maps are now invariant on key and value types (#318)
- It is now an error to declare too many iteration variables in a
for-in
construct - Some standard library definitions were changed from using polymorphic function to using union types (#317)
- Warning categories
- You can disable specific warning categories such as redeclared
variables with the
--wdisable
flag or with thedisable_warnings
entry intlconfig.lua
- You can list existing warning categories with
tl warnings
- You can disable specific warning categories such as redeclared
variables with the
- You can turn warnings into errors with
--werror
- Nicer error message when
type
is declared without alocal
orglobal
qualifier
- Fixed a parser crash on incomplete expressions
- Nested record prototypes can be used everywhere record prototypes can
- Fixed a leak of type variables when defining generic functions (#322)
- Fixed resolution of type arguments in recursive type definitions to avoid crashing the compiler
- Ensured that n-ary function returns don't leak into variable declaration types (#309)
- Fix
is
code generation for userdata
2020-12-31
We've had a lot of activity in Teal this month, so let's wrap it with a new release! We had our very first Teal meetup (recording), a Twitch stream session, lots of activity in our Gitter chat with folks from the community building various projects, and providing great feedback which helped us prioritize and shape the evolution of the language. So let's celebrate with Teal 0.10.0, which packs a lot of new stuff! Happy New Year! 🎉
This release features commits by Enrique García Cota, Darren Jennings, Corey Williamson and Hisham Muhammad.
- Metamethods!
- Records can now declare metamethods as part of their definitions, which informs the Teal type checker about supported operations for that record type. (#299)
- Metatables are not automatically attached, you still need to use
setmetatable
: check the documentation for an example on using records with metamethods and metatables. - Operator metamethods for
//
and bitwise ops are supported even when running Teal on top of Lua 5.1.
- Userdata records
- The practical difference, as far as type checking goes, is that they no longer count as a "table" type in unions.
or
expressions now accept subtypes: for example, anor
between a union and one of its elements is now accepted.- Some breaking changes for cleaning things up:
- The language now rejects unknown variable annotations. That includes
<close>
, which is not currently supported. Note that Teal does support the<const>
annotation (for all Lua versions) and performs the check at compile time. - Dropped. support for record and enum definitions inside table literals.
This was a remnant from when variable and type definitions were mixed. Now
all record and enum definitions need to happen inside a
local type
orglobal type
definition (and may be nested).
- The language now rejects unknown variable annotations. That includes
- Standard library definition improvements:
math.log
accepts a second argument
- Compiler warnings!
- The Teal compiler now features a warning system, and reports on unused variables.
- Teal Playground integration in the Teal repository
- Now, every PR triggers a Github Action that produces a link to a Teal Playground which allows you to test the compiler changes introduced by the PR right from your browser!
tl build
now returns a non-zero error code on type errors
- Detects a bad use of
:
without a proper()
call (#282) - Fixed type inference for variadic return in
for
loops (#293) - Always check for union validity after type arguments are resolved when declaring a record (#290)
- No longer suggest "consider using an enum" when indexing non-records
- Fixes for control-flow based inference
- Fix expression inference in
elseif
- Propagate facts across parentheses
- Fix expression inference in
- Standard library definition fixes:
os.date
second argument was fixed
2020-12-16
Three months after 0.8, it's time to release Teal 0.9! It features language improvements, bugfixes, new contributors.
This release features commits by Domingo Alvarez Duarte, Corey Williamson, Pierre Chapuis and Hisham Muhammad.
- New tuple type!
- You can declare a type such as
local t: {string, number} = {"hi", 1}
and thent[1]
gets correctly resolved asstring
andt[2]
as number.t[x]
with an unknownx
gets resolved asstring | number
. - Inference between arrays and tuples allows for some automatic conversions.
- You can declare a type such as
record
declarations now accept arbitrary strings as field names, using square brackets and string notations, similar to Lua tables:[ "end" ] : string
- Support for the
//
integer division operator- This is also supported for target VMs that do not support it natively!
For those environments, it outputs
math.floor(x / y)
in the Lua code.
- This is also supported for target VMs that do not support it natively!
For those environments, it outputs
- Additions and tweaks to the standard library definitions:
string.gsub
function argument return values
- Error messages of the type
X is not a X
(when two different nominal types happen to have the same name) are now disambiguated by presenting the filenames and locations of the two different declarations, as inX (declared in foo.tl:9:2) is not a X (declared in bar.tl:8:2)
- The
tl
compiler module now exposes some of its external types for use by tooling (such as thetl
CLI script) - Compiler now generates compatibility code using
bit32
library (part ofcompat53
) for target VMs that do not support bitwise operators natively. - Performance improvements in the compiler!
./tl check tl.tl
went from avg 612ms in 0.8.2 to 315ms
- CI now tests Teal with Lua 5.4 as well.
- Fixed the support for bitwise xor operator
~
- Fixed the support for escapes in string literals, including Unicode escapes.
- Various fixes to the inference of unions.
- Fixed invariant type comparisons for union types.
- It now skips
nil
correctly when expanding union types: For example,{ 5, nil }
is now correctly inferred as{ number }
instead of{ number | nil }
, because nil is valid for all types.
- Cleaned up shadowed variable declarations in the compiler.
- Cleaned up and added more extensive test cases for subtyping rules
for
nil
,any
, union and intersection types, and caught some edge cases in the process.
2020-11-06
This is another bugfix release.
This release features commits by Hisham Muhammad, Corey Williamson and fang.
tl
now caches required modules to not load code more than once (#245)- fixed a compiler crash using
math.atan
tl
now usesloadstring
instead ofrequire
to load config, to avoid issues with LUA_PATH- do not shadow argument types with function arguments
- expand tuples but not nominals in return types (#249)
- fix
is
inference onelse
block when using nominal types (#250) for
accepts a function call that returns a nominal which resolves to an iterator. (#183)- fix invariant type checking of records
- various fixes for the lexer pretty-printer
2020-10-24
This is a small bugfix release, to get some of the recent fixes out in the default package.
This release features commits by Hisham Muhammad, Corey Williamson and Casper.
- Standard library definition improvements:
debug.traceback
3-argument form and__name
metatable field
- Added a simple type checking for config entries in
tlconfig.lua
- Fixed a crash when detecting wrong use of
self
on a method with no arguments. (#228) - Fixed the declaration of type aliases for nominals, which was causing a stack overflow. (#238)
- Only infer an array if we can infer its elements: When resolving the type of a table literal, a function with no returns produces an empty tuple, and that does not give us enough data to resolve a type. (#234)
- Improved cleanup for the test suite.
2020-09-18
Some big language changes in this release! Type declarations were made more
consistent, with the introduction of local type T = D
for declaring a type
T
given a definition D
(likewise for global type
), instead of pretending
that type declarations were assignments. This allowed us to remove the kludgy
functiontype
word (the definition for function types is the same as used for
function arguments: local type F = function()···
This release also includes major tooling improvements! The tl
CLI received
major attention, with a new command tl build
and many new options in
tlconfig.lua
, making tl
more comfortable to use for building projects. The
tl
module also received some API improvements for programmatic use of the
compiler.
This release features commits by Darren Jennings, Patrick Desaulniers, Corey Williamson, Chris West and Hisham Muhammad.
local type
andglobal type
syntax for type declarations- Dropped support for
functiontype
: usefunction
instead - Shorthand syntax forms for declarations, similar to
local function
in Lua:local record R
as a synonym forlocal type R = record
(same forglobal
)local enum E
as a synonym forlocal type E = enum
(same forglobal
)- Shorthand forms are also accepted nested inside records
- Extended flow-based inference for empty tables: now when declaring a
variable with an empty table (
local V = {}
), if its first use (lexically) is as an argument to a function, the compiler will infer the argument type to the variable - Enums can be indexed with string methods
- Lots of additions and tweaks to the standard library definitions:
collectgarbage
coroutine.close
math.atan
overload- the many metamethods in metatable definitions
- Fixed declaration of
select()
- New command
tl build
, which is able to compile multiple files at once, following configuration intlconfig.lua
tl run
passes arguments to script, populating...
tl gen
: new flag-o, --output
- Many newly supported options in
tlconfig.lua
:build_dir
source_dir
files
include
exclude
include_dir
(new name of the previousinclude
)skip_compat53
- New functions in the
tl
module API:tl.gen
, a high-level function for generating Lua code from Teal code, akin totl gen
on the CLItl.process_string
, a lower-level driver function which produces the result data structure including the AST for later processing
- Fixed a file handle leak in
tl.process
- Initial newlines from input file are preserved in the generated output file, so that line numbers match in stack traces produced at runtime
- It now reports error when
...
is used in non-vararg functions - Stricter type checks:
- no longer accepts indexing a record with an arbitrary string
- no longer accepts matching a map as a record
- Fixed resolution of multiple labels
- Better handling of varargs for
for in
iterators - Parser fixes:
- Accept
()
as a return type in a function declaration - Require separators in key/value table constructors
- Fix acceptance of a stray separator in table constructors (accepted) and function calls (not accepted)
- Accept
2020-06-14
A lot of important bugfixes driven by community reports!
Starting with this release, version numbers in the 0.x series will indicate the presence of breaking changes when bumping the second number (0.x.0), and both bugfixes and compatible additions when bumping the third number (0.x.y). This should make it easier to see when a release breaks something.
This release adds new features but introduces no breaking changes, so we're calling it 0.7.1.
- support tuples in
as
operator for casting vararg returns: you can now cast the return of variadic functions with a parenthesized list of types - semantics of scope of
until
now matches that of Lua - Standard library:
- Fixed return type of
load
- Fixed return type of
tl
CLI:- New flag
--version
- New flag
-q
,--quiet
- New flag
tl
module:- New function
tl.load
which is like Lua'sload()
but enables loading Teal code
- New function
- Various fixes in the grammar definition
- Fixes the resolution of generic types by delaying the resolution of type arguments
- Fixes parsing of paranthesized function return with varargs
- Does not stop parsing the input toplevel when a stray
end
is found - Avoids a compiler crash when type checking call returns in
a
return
statement - Fixes an error when parsing invalid function arguments
- Fixes a crash in flow analysis when a variable does not exist
2020-06-08
This release had a bunch of community contributions!
thread
is a new primitive type for representing coroutines- Disallow unions of multiple function types, applying a similar
restriction that was already in place for table types.
These can't be efficiently destructured at runtime with
is
. - Lots of additions to the standard library:
- more overloads for the
load
function loadfile
anddofile
functionscoroutine
library- More math functions:
tointeger
,type
andult
- math variables:
maxinteger
andmininteger
- improvement the type definition of
pcall
xpcall
- more overloads for the
tl
now reports when there is a syntax error loadingtlconfig.lua
- A functiontype declaration can now refer to itself.
- Nominals declared in nested types now resolve correctly.
2020-06-03
- Completed definitions of the
math
table to the standard library
- Resolve arrays of unions, and flatten unions of unions
2020-05-31
More bugfixes!
- An important fix for the code generation of table literals:
it was causing incorrect Lua output for arrays with expanded
contents, such as
{...}
- Resolution of pcall now recurses: the compiler can now handle a
pcalled
require
correctly - Better error messages for errors using
require
: do not report "module not found" when a module was found but does not report type information.
2020-05-29
Since I was on a roll doing bugfixes, here's a quick version bump!
- Detect and report unresolved nested nominal types
- Fix scope restriction for record functions
- Standard library tables can now be required with
require()
2020-05-29
This release does not include a lot of changes, but it does change the rules for exporting types across modules, so it is worth the middle-number bump.
- Types declared as
local
can be exported in modules - Types that are
local
to a module and which are used in function signatures (e.g. as return types) can be destructured from requiring modules. This already partially worked before, but it required the variable to be used in the original module to force a type resolution.- Generics still require more work, but non-generic types should be propagating as described above. Generic types export correctly if made part of the module record definition.
- Fixed a parsing error when a colon in a variable declaration was not followed by a type list
2020-05-18
Another collection of bugfixes!
- Report error when a required module is not found
- Detects unresolved nominal types
- Fixes when using record type object as a record instance itself
- Fixes in the expansion of the type of a table literal
- Fixes the indentation in generated code
- More standard library additions
- Improved/fixed error messages
2020-05-05
Fixes a regression in record functions.
2020-05-03
- Record definitions can be nested: this should make declarations of types in modules more natural
- Improvement in the position of error messages
2020-04-27
This is the first release with the simplified syntax for generics, without requiring backticks!
Besides the usual bugfixes (see the Git history for details), here are the main changes:
- Type variables no longer need backticks: you can
declare
function<T, U>(x: T): U
and useT
andU
as types inside the function - Base type declarations can contain parentheses for
clarity, e.g.
{string: (string|number)}
- More standard library typing improvements
--include
/-I
flag for including directories to the module search pathtl run
now auto-invokes the Teal package loader, so.tl
files running with it canrequire()
other.tl
files without needing to usetl gen
to generate.lua
files first.tl run
can also pass arguments to the running program again
2020-04-08
This is the first release where the language is officially named Teal!
Lots of bugfixes, as well as language changes, tooling improvements and beginnings of documentation!
- Union types, still restricted (can only union between
at most one table type and a non-table type -- in other
words, type resolution at runtime happens using the
Lua
type()
)is
operator for specializing union types with a test
goto
and labels- Type variables need to be declared in functions with
angle brackets:
function<...>
(this fixes scoping rules) global function
is now valid syntax- Accepts any element type on a table typed
{any:any}
. - Booleans are stricter
- Disallow extending records with functions outside of original scope
- Standard library typing improvements
- New command-line parser using argparse,
typecheck using
tl check
-l
command line option for requiring libraries--skip-compat53
flag- Detects
lua
hashbang to enter lax mode
2020-02-02
First tagged release! Announced at FOSDEM 2020.
- Everything!