-
Notifications
You must be signed in to change notification settings - Fork 307
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Feature: Added Methods #320
base: master
Are you sure you want to change the base?
Conversation
Thanks for the PR. I'm gonna give this some thought. At first glance I am comfortable with a feature like this, but I'm not entirely sure about the syntax for declaring the methods. @d5 thoughts? |
m := { f1: func(r)() { return r == undefined}} | ||
out = m.f1()`, | ||
nil, false) | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Add tests for array as well
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for pointing that out - it completely slipped my mind while writing the other cases.
I added a few that I think test the broadest categories of array operations. If there are any other unit tests needed, please let me know.
Thanks for the PR. I really like the documentation on this change. I thought a lot about adding "methods" in the past, and, I've explored Python-like syntax as well. I will definitely think more about this suggested syntax. I'm curious, what would be the problems if we introduce a new language construct such as "this" or even "context" (maybe we can find an interesting approach that is similar to Go's context)? Just a quick thought. EDIT: I totally misunderstood the suggested syntax. Lol. Still I will think more about it. 😄 |
Thanks for getting to this so quickly, both of you. I hemmed and hawed over what to do in regard to the a keyword (Obviously it would be "this" I think, just due to convention). Several of my prototype versions used a keyword like that, but it just never felt go-like. The context idea is neat - it could perhaps even be an array of context, with each context on top of the other? (That would be more difficult to implement, but I did consider it. There was always the question in my mind about how useful that would end up being.) A ctx object would be interesting though... especially if the most "root" object contained the context from golang. I'm not certain either way. I landed on this syntax as a kind of call-back to how golang tends to do things. EDIT: Oh and thanks for the compliment on the documentation changes - I spent a lot of time trying to make sure it was of the same calibre as the rest of Tengo's docs. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM overall. Just two comments. An array example in the documentation would be nice. That said, should we have receivers on arrays at all. I suppose it simplifies things a little on the OpIndex
, but it seems counterintuitive.
@geseq Yeah, I considered omitting them from the specification, but because Tengo has no defined classes nor definitions, it made more sense to me to not artificially limit which collection scopes allowed receiver passing. |
SyntaxI've been thinking about the method declaration syntax, and these are a few thoughts I've had on the subject. This is all just speculative brainstorming. I based my ideas off of this python snippet: class Obj:
def __init__(self, y):
self.x = y
def getX(self):
return self.x
//...
o = Obj(23)
n = o.getX() //n = 23 Translated to Tengo-ish syntax: newObj := func() {
return {
x: 23
getX: func(self) {
return self.x
}
}
}
n := newObj().getX() //n = 23 Pros:
Cons:
If we went with a standard keyword, To avoid that, we could set up a new definition keyword to append to newObj := func() {
return {
x: 23
getX: func method () {
return self.x
}
}
}
n := newObj().getX() //n = 23 Here I am using Perhaps changing This format would likely open itself to having a However, this idea does make it less intuitive to capture a receiver via a closure. Current Branchm := {
grab: func(outer)() {
return func(inner)() {
return outer
}
}
}
f := m.grab()
f() // returns m Possible Change with 'Self'm := {
grab: func*() {
outer := self
return func*() {
return outer
}
}
}
f := m.grab()
f() // returns m Personally, I think this is relatively niche, but I thought I'd mention it, due to how many hours have been lost figuring out what Overall though, If we don't go with the syntax proposed in the PR comment, I think that the above snippet is the second-best candidate. |
I've spent some time to try this, but, I'm still not quite convinced if we should make this change. This is a significant change and naturally increases the maintenance costs by a lot (more bugs and edge cases) and makes the syntax a bit more alienating to Go coders in my opinion. But my main concern is maintainability to be honest. I'm actually more interested in Go's context-like approach: is immutable, can be safely passed around, and, (like @DawDavis mentioned earlier) can be potentially useful if we use it to pass from actual Go context value. |
@d5 , I have had half a mind to create a module that includes ctx stuff for Tengo, due to the project I upstreamed this PR from. Is that a direction that you're willing to explore with Tengo moving forward? |
I'm sure there will be more details we will have to figure out, but, I think Go-like "context" construct is better because it's more easily understood and very commonly used by most Go coders. One obvious question is should that be explicitly declared? Or should we make it some kind of intrinsic value (or keyword)? |
How about this approach?
|
OVERVIEW
This PR aims to bring a receiver-alike functionality to Tengo, allowing for getters, setters, mutators, and more. Overall, the change makes it much easier to use some OOP principles in Tengo, much as the 'this' keyword does in certain JS applications.
A method declaration is syntactically defined as follows:
This is done to alleviate two major concerns I had with this feature:
Specifics
These proposed Tengo receivers do not carry a guarantee about their underlying type: It could be any index-able Tengo type, or undefined, should the method be called in a non-parent context. For example:
You may notice that unlike JS, where 'this' contains some scope information no matter the context, Tengo's receivers very often carry
undefined
as their value.Another important question is how this is different from a self-capturing closure. e.g:
at first glance, this serves a similar purpose, however the given relationship breaks down if you attempt to copy
m
.This is because closures capture at definition time, whereas receivers capture at call-time. The same operation, when done with a receiver, acts as a method call:
Implementation
The implementation of receivers is handled by altering the way free-vars are stored when a method is indexed out of a map or array. At parsing, a look-ahead is performed on any selector or index expressions, in order to see if they directly feed into a call operation. If they do, they mark themselves as being a method call.
At compile time, a select or index expression is then compiled to the
OpIndex
opcode, with a new flag set afterwards, akin to the spread flag, that indicates whether the given indexing operation should include its context info in the indexed object.Then finally, at run time, when OpIndex is called with a method flag, it behaves as per usual, except that it attempts to cast the retrieved object as a compiled function, and then if it is successful, copies the function, sets freevar[0] inside of the function as itself (the indexed map/array), and then continues normally.
This copy prevents the given map context from escaping this particular function call.
It is also useful to note that the freevars in compiled functions are now always +1 in length, as the 0th item is reserved for the receiver implementation. If no receiver exists, a placeholder "" freevar is created in order to reserve the space. This is done to streamline the function calls at runtime.
Caveats
This feature does not extend to non-compiled methods, mainly because that would require a huge API change that I do not feel comfortable suggesting. Additionally, I questioned the usefulness of being able to move context out into Go-space.
Accessing methods in any way that is not an "index" or "select" operation results in an
undefined
receiver. To my mind, this is fine, mainly because those are the places where parent-access is most frequently required.Continued extensions of this feature
I considered adding a
bind
keyword to the specification, which would allow a receiver to be permanently bound to a specific map/array.And while I believe this may be a useful feature, I do think its use is extremely niche.
Benchmarks:
Master:
This Branch: