Replies: 3 comments 9 replies
-
Remind me to do a bigger response later but something that might be a nice to know:
You don't need sourcemaps when using teal. Teal does a great job at keeping the line numbers at the same place (and maybe more?) So, if lua complains about an error being at line X you can just open the corresponding teal file and go to the same line. For other tools, see the "awesome-teal" repo. Also, about the |
Beta Was this translation helpful? Give feedback.
-
May have found a bit of extra time, so lets see how far I get :)
Yep, almost everyone wants this but a lot of the time people can't agree on how it would actually look/work. The problem lays within the fact that
Yes, something closer to interfaces would be nice. I would even go as far as being able to make them implemented implicitly but that is opening the "structural typing" can of worms which is probably best kept closed, at least for now. Some strict examples on when one versus the other should be used would need to be made though. ESPECIALLY when it comes to writing .d.tl files for lua or even worse, C/Rust libraries.
I am not quite sure why it needs code in the tlconfig.lua file. Do you want to be able to extend types that aren't your own? (It almost reads like that but not quite).
Yes, removing the error on Of course, type guards and similar are also welcome if you ask me. But one step at a time, shall we? :)
I am not sure about introducing the
Already went over this in my previous message
Giving them their own section in records does make sense if you ask me. This might then also lead to a reasonably way to write the kind of inheritance that works this way, especially if the metamethod table can be its own proper type that you can reuse.
look at https://github.com/teal-language/awesome-teal Note: This is just my look at things as someone who maintains a Rust <-> Teal FFI library and as I am not a maintainer and as it has been a while since I used teal (or lua for that matter) take my take on things with a decent grain of salt ... :) |
Beta Was this translation helpful? Give feedback.
-
Nil checking, inheritance, and interfaces (and probably intersection types) I think are the big elephants in the room for Teal right now. The main issue I would say is implementation. Not to be rude, but everybody and their mother has an opinion on how these should be done, but very few people are actually able/willing to put the time in to implement them (which is totally fair). See the following discussions and issues for more information:
Teal has a limited form of this via the local var: string | number = getSomeValue()
if var is string then
local str = "+" .. var
else
local num = 1 + var
end but due to the limitations of discriminating table types at runtime, there has been discussion of user defined
In Teal, records are nominally typed, and I do think there is a strong argument for having a way to structurally type things as a lot of Lua code only cares about the structure of its tables rather than the names. I think most of this comes down to the fact that this is an open source project that people are graciously donating their free time and effort to. I for example have gotten extremely busy over the past year or so and haven't been able to make any meaningful strides on any of this. |
Beta Was this translation helpful? Give feedback.
-
Hi,
I was hired a few years ago to make some Lua for embedded systems. Since I’m coming from the TypeScript world, I was missing the ability to type things, and so was looking for a project like Teal. Teal looks very promising and I’d like to push my company to use it. But in a production environment, I cannot take it before the syntax is stabilised and complete enough to meet our needs. Anyway, I hope I’ll convince them to give me some time to work on the project.
Here are my first remarks regarding syntax. Maybe some of my remarks are against the policy of the project, or already discussed. If so, please tell me…
Nil checking
In Lua, every variable can be nil. But a nil variable may not be expected. It would be good to indicate that an attribute, function or method parameter can be optional (i.e. is nil-able). In TypeScript, we use question marks and exclamation points.
As the language is not stabilised yet, I think this change can be made directly, but if not, it could be protected with a “tlconfig.lua” option. TypeScript has such an option called StrictNullCheck.
Types and objects
We need 2 different ways to define types. “record” is currently emitting some code, making the type really exist in Lua. We need another kind of definition (if “record” were called “class”, I’d call it “interface”, but there, I think “type” or even “shape” would be a better candidate) which would only define the shape of what is expected and would totally be erased at compilation time. This can be useful for, for example, an “options” parameter to give to a function. We don’t need to emit anything in Lua, only to document the different options that may be given as entries in a table.
It would also be a good thing to be able to define anonymous shapes. We wouldn’t always need to name the expected shape of the “options” table. For example:
This would only be used to make sure the function is called with appropriate options, but the emitted Lua file would simply be:
Inheritage
Regarding records, we should also have a syntax allowing inheritage like this:
By default, this would emit, as it is doing now:
One of the things that I think is important about Teal, is that it can interact with existing code. It is then rather easy to use it in an existing Lua project, migrating files step by step. Defining an entry such as “createObject” in “tlconfig.lua” would allow every existing code to use the record the way they want. The value of this option parameter would be a callable (either a function or an object with a metatable defining “__call”), called with the name of the type and the list of inherited types and returning the created type. A “shape” could define the default (inherited) properties of the created object. For example:
Consequences of inheritage
What would be interesting also would be to have the ability to indicate that a field is private, protected or public. Nothing would change in the emitted code, but the type checker would ensure proper usage.
It would also be interesting to have the ability to indicate that a method returns self. Indeed, if you have a class “A” with a method returning self, you can simply declare it as returning an object of type “A”. But then, if a class “B” inherits from “A”, then this declaration will not be totally right. If it is possible to directly indicate that the method returns self, the problem is solved.
Unions and type guards
In order to keep Teal as a type checker, I think it would be a good idea not to emit unexpected code. I’d rather see the “is” keyword used as in TypeScript. So, the type checker would detect code as:
and would know that “var” is a string after the “if” condition.
If you want to create a union of any kind, it should be possible. And if you want to narrow the type, you either need to use the
type(var)
construction or to write a custom type guard:Enumerations
I saw that you were discussing about emitting code for enumerations. I think it is a good idea, but we need to keep a pure typing enum, not emitting any code. TypeScript defines the “const enum” for that. It would also be very useful to be able to define more restrictive types, maybe using unions which, as a first approach, could replace this “const enum”:
Note in this example the usage of “type”, which could be a way of defining aliases for complex types used in multiple places.
Requires and unknown
Again, what I like with Teal is that it can interact with Lua which allows us to introduce it in a Lua project step by step. But for that, it is needed that we can require Lua files without having errors nor warnings. For that reason, type checker could make a best effort to guess the type of imported object, but by default, it should use any, and not unknown. So, if I require any Lua file, I don’t have any error, any warning.
On the contrary, “unknown” should be a type which can be given by hand to a variable. The difference between “any” and “unknown” is that the former can be assigned to any type while the latter cannot.
Metamethods
A simple note about metamethods : I’m not sure it is a better idea than using metatable (inside record) and defining the content of the metatable there. That’s because a metatable can have more than the defined metamethods, and a metatable can even have a metatable, and this does not seem to be definable in Teal.
Tools
This is more a question than a remark: are there already tools compatible with Teal ? Prettier, Luacheck, source map for debugging, etc. ?
I’d be glad to hear your thoughts about all this. If you’d prefer me to write multiple issues, please tell me also. Anyway, I also think it may be good to have a single issue ticket, which would be a reference for everything needed for version 1.0?
Beta Was this translation helpful? Give feedback.
All reactions