I have encountered a snippet of code where call contains another call. For example:
a <- 1
b <- 2
# First call
foo <- quote(a + a)
# Second call (call contains another call)
bar <- quote(foo ^ b)
We can evaluate calls with eval
(eval(foo)
), however eval(bar)
won't work. This is expected as R tries to run "foo" ^ 2
(sees foo
as non-numeric object).
How to evaluate such callception?
I think you might want :
The call before evaluation :
This also works and might be easier to understand:
and going backwards :
And some more
Not completely the same but you could use
bquote
here too if you can afford to definebar
differently :And in that case the close equivalent using
rlang
will be :And a minor thing, you say :
It doesn't, it tries to run
quote(foo)^b
, which will return this same error if you run it directly in the console.Addendum on recursion
Borrowing Oliver's example you can deal with recursion by looping on my solution until you've evaluated all you can, we just have to slightly modifiy our
substitute
call to provide all the environment and not explicit substitutions :To answer this question it might be helpful to split it up in 3 sub problems
For the answer to be complete, we need to locate any subsequently nested call within the call. In addition we would need to avoid the endless loop of
bar <- quote(bar + 3)
.As any call might have nested called eg:
we will have to make sure each stack is evaluated before evaluating the final call.
Following this line of thought, the following function will evaluate even complicated calls.
Showcase
So lets try a few examples. Initially I'll use the example in the question, with one additional slightly more complicated call.
Evaluating each of these gives the desired result:
This is not restricted to simple calls however. Lets extend it to a more interesting call.
Suprisingly enough this also works out just fine.
as when we try to evaluate only the segment that is actually necessary, we get the same result:
Note that this is likely not the most efficient evaluating scheme. Initially the envir variable should be NULL, unless calls like
dat <- x
should be evaluated and saved in a specific environment.Edit: Summary of currently provided answers and performance overview
This question have been given quite some attention since the additional reward was given, and many different answers have been proposed. In this section I'll give a short overview of the answers, their limitations and some of their benefits as well. Note all the answers currently provided are good options, but solve the problem to a differing degree, with different upsides and downsides. This section is thus not meant as a negative review for any of the answers, but a trial to leave an overview of the different methods. The examples presented in above in my answer have been adopted by some of the other answers, while a few have been suggested in the comments of this answer which represented different aspects of the problem. I will use the examples in my answer as well as a few below, to try and illustrate the usefulness of the different methods suggested throughout this post. For completion the different examples are shown in code below. Thanks to @Moody_Mudskipper for the additional examples suggested in the comments below!
Solution versatility
The solutions provided in the answers to the question, solve the problem to various extends. One question might be to which extend these solve the various tasks of evaluating the quoted expressions. To test the versatility of the solutions, example 1 to 5 was evaluated using the raw function provided in each answer. Example 6 & 7 present different kind of problems, and will be treated seperately in a section below (Safety of Implementation). Note the
oshka::expand
returns an unevaluated expression, which was evaluated for after running the function call. In the table below I've visualized the results from the versatility test. Each row is a seperate function in an answer to the question while each column marks an example. For each test the succes is marked as sucess, ERROR and failed for a succesfuly, early interrupted and failed evaluation respectively. (Codes are availible at the end of the answer for reproducability.)Interestingly the simpler calls
bar
,foo
andzz
are mostly handled by all but one answer. Onlyoshka::expand
succesfuly evaluates every method. Only two methods succeed themassive_call
andquz
examples, while onlyoshka::expand
craetes a succesfuly evaluating expression for the particularly nasty conditional statement. One may however note that by design the any intermediate results are saved using theoshka::expand
method, which should be kept in mind while used. This could however be simply fixed by evaluating the expression within function or child-environment to the global environment. Another important note is the 5'th example represents a special problem with most of the answers. As each expression is evaluated individually in 3 out of 5 answers, the call to thestop
function, simply breaks the call. Thus any quoted expression containing a call tostop
shows a simply and especially devious example.Efficiency comparison:
An alternative performance meassure often of concern is pure efficiency or speed. Even if certain methods failed, being aware of the methods limitations, can yield situations where a simpler method is better, due to the speed performance. To compare the methods we need to assume that it is the case that we know the method is sufficient for our problems. For this reason and in order to compare the different methods a benchmarking test was performed using
zz
as the standard. This cuts out one method, for which no benchmarking has been performed. The results are shown below.For the purposes of comparison, the median is a better estimate, as the garbage cleaner might taint certain results and thus the mean. From the output a clear pattern is visible. The more advanced functions takes longer to evaluate. Of the four functions
oshka::expand
is the slowest competitor, being a factor 12 slower than the closest competitor (1835.8 / 152.9 = 12), whileevalception
is the fastest being about twice as fast asfun
(98.7 / 49.5 = 2) and three times faster thaneval_throughout
(damn!) As such if speed is required, it seems the simplest method that will evaluate succesfuly is the way to go.Safety of implementation An important aspect of good implementations is their ability identify and handle devious input. For this aspect example 6 & 7 represent different problems, that could break implementations. Example 6 represents an endless recursion, which might break the R session. Example 7 represents the missing value problem.
Example 6 was run under the same condition. The results are shown below.
Of the four answer, only
evalception(bar)
fails to detect the endless recursion, and crashes the R session, while the remaining succesfuly stops.Note: i do not suggest running the latter example.
Example 7 was run under the same condition. The results are shown below.
An important note is that any evaluation of example 7 will fail. Only
oshka::expand
succeeds, as it is designed to impute any existing value into the expression using the underlying environment. This especially useful feature lets one create complex calls and imputing any quoted expression to expand the expression, while the remaining answers (including my own) fail by design, as they evaluate the expression.Final comments
So there you go. I hope the summary of the answers proves useful, showing the positives and possible negatives of each implementation. Each have their possible scenarios where they would outperform the remaining, while only one could be successfully used in all of the represented circumstances. For versatility the
oshka::expand
is the clear winner, while if speed is preferred one would have to evaluate if the answers could be used for the situation at hand. Great speed improvements is achievable by going with the simpler answers, while they represent different risks possibly crashing the R session. Unlike my earlier summary, the reader is left to decide for themselves which implementation would work best for their specific problem.Code for reproducing the summary
Note this code is not cleaned, simply put together for the summary. In addition it does not contain the examples or function, only their evaluations.
I came up with a simple solution to this, but it seems a little improper and I hope that a more canonical method exists to cope with this situation. Nevertheless, this should hopefully get the job done.
The basic idea is to iterate through your expression and replace the un-evaluated first call with its evaluated value. Code below:
So far this is pretty easy. Of course if your expressions are more complicated this becomes more complicated quickly. For instance, if your expression has
foo^2 + a
then we need to be sure to replace the termfoo^2
witheval(foo)^2
and noteval(foo)
and so on. We can write a little helper function, but it would need a good deal of work to robustly generalize to complexly nested cases:I thought I should somehow be able to do this with
substitute()
but couldn't figure it out. I'm hopeful a more authoritative solution emerges but in the meantime this may work.Here's something that (at least partially) works:
It supports arbitrary nesting but will probably fail with objects of mode
expression
.I found a CRAN package that can do this - oshka: Recursive Quoted Language Expansion.
It recursively replaces quoted language calls by objects in environment.
So call
oshka::expand(bar)
gives(a + a)^b
andeval(oshka::expand(bar))
returns4
. It also works with more complicated calls that @Oliver suggested: