Skip to content
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

feat(ecmascript): Callable #539

Closed
wants to merge 4 commits into from
Closed

Conversation

yossydev
Copy link
Contributor

I have created Callable as a new implementation. I copied the contents of function.rs and added Proxy to each function.

Now, I’m thinking of rewriting is_callable as follows:

pub(crate) fn is_callable<'a, 'b>(
    argument: impl TryInto<Function<'b>>,
    gc: NoGcScope<'a, '_>,
) -> Option<Callable<'a>> {
    // 1. If argument is not an Object, return false.
    let Ok(argument) = argument.try_into() else {
        return None;
    };
    // 2. If argument has a [[Call]] internal method, return true.
    is_callable(argument, gc)
}

However, if I do this, it will cause compilation errors in all the places where is_callable is currently being used. Since the existing functions mostly accept Function, this behavior seems natural.

At this stage, does this implementation approach for Callable align with the intended design?

Copy link
Collaborator

@aapoalas aapoalas left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is exactly what I was thinking, yes.

... But now that I see it and read the spec a little more, it might be that we should just rename Function to Callable and add Proxy there. The spec says this about callable objects:

an object is a member of the built-in type Object; and a function is a callable object.

and

In addition, some objects are callable; these are referred to as functions or function objects

ie. The spec is saying that "whenever I say 'function', what I mean is any kind of actual function or a callable Proxy". And this is validated by how the spec is written. So the Function enum doesn't actually correspond to anything in the spec... It does make sense from a human standpoint but it might not make actual sense from the code standpoint...

is_callable definitely should return Callable, and basically every place where we have Function as a parameter or saved in a struct should indeed have Callable... The only question is if there will remain any places whatsoever where the Function enum would remain at all?

}
Callable::BuiltinPromiseCollectorFunction => todo!(),
Callable::BuiltinProxyRevokerFunction => todo!(),
Callable::Proxy(d) => write!(f, "PROXY_DISCRIMINANT({:?})", d),
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nitpick: Should be Proxy({:?).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

thank you! fixed to 2e8bba3

Callable::BuiltinPromiseResolvingFunction(d) => agent[d].object_index,
Callable::BuiltinPromiseCollectorFunction => todo!(),
Callable::BuiltinProxyRevokerFunction => todo!(),
Callable::Proxy(_) => todo!(),
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nitpick: This should be unreachable as well.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

thank you! return None 91f9ae7

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just call unreachable!()

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Got it, I’ve made the changes! d491dbb

@yossydev
Copy link
Contributor Author

… But now that I see it and read the spec a little more, it might be that we should just rename Function to Callable and add Proxy there. The spec says this about callable objects:

Yeah, I was also wondering if that might be the right approach while trying to refactor this. I was a bit unsure about when we would actually need to treat Function and Callable separately.

Should we go ahead and rewrite function.rs as callable.rs, or should we merge this PR and gradually replace the places using Function with Callable? If we decide to go with the former, I think we can go ahead and close this PR. What do you think? 👀

@aapoalas
Copy link
Collaborator

… But now that I see it and read the spec a little more, it might be that we should just rename Function to Callable and add Proxy there. The spec says this about callable objects:

Yeah, I was also wondering if that might be the right approach while trying to refactor this. I was a bit unsure about when we would actually need to treat Function and Callable separately.

Should we go ahead and rewrite function.rs as callable.rs, or should we merge this PR and gradually replace the places using Function with Callable? If we decide to go with the former, I think we can go ahead and close this PR. What do you think? 👀

I'm a bit unsure. If we add Proxy to Function and rename it to Callable, then we'll get basically all we want the easy way. But! It might still be that there's a place somewhere where the spec does actually rule out callable Proxies, and if that is the case then we'll have removed the enum that would've been used to represent that. The worst thing is that we won't see that case in our code if we just rename Function to Callable.

But, if we do go ahead with a separate Callable then we're going to do a lot of work that might end up being entirely unnecessary :/

@aapoalas
Copy link
Collaborator

A quick search through the spec suggests that Proxies are never categorically separated from callable objects, so a closing this PR and just renaming Function to Callable seems proper.

@aapoalas aapoalas closed this Jan 18, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants