-
Streamlined the behavior of
ggtrace(use_names)
for assigning names to the tracedump, where:-
If a list of expression is provided in
trace_exprs
, uses the names of that list ifTRUE
, and uses deparsed expressions from the method steps as names ifFALSE
. -
If
trace_exprs
is empty, uses deparsed expressions from the method steps as names ifTRUE
, and returns unnamed output ifFALSE
.
-
-
New special value
ggtrace(trace_steps = "all")
which evaluates to all steps in the method body. -
Bug fix in the default behavior of
ggtrace(trace_exprs)
doubly-evaluating an existing step from the method body, which sometimes caused unexpected side effects. This behavior is now prevented by cloning the method environment in such cases.
-
New helper function
layer_is()
, primarily for internal use by sublayer data inspection functions (e.g.,layer_before_stat()
). These functions now become more robust to changes in the internals, by specifically targetting the downstream ofby_layer()
. -
Improved formatting of long-form calls generated by sublayer data functions when
verbose = TRUE
. -
More visible documentation of shorthand aliases for workflow functions introduced in last release.
-
Exported shorthand aliases for all workflow functions. E.g.,
inspect_n()
complement toggtrace_inspect_n()
-
Cleaned up examples and tests to align with
{ggplot2}
version 3.5.0
Upkeep release for the JSM 2023 talk "Sub-layer modularity in the Grammar of Graphics".
-
layer_*()
snapshot functions gain averbose
argument. -
getOption("ggtrace.rethrow_error")
To control re-printing of the error by workflow functions whenerror = TRUE
. Set toFALSE
on load.
get_method_inheritance()
now works correctly for top-level ggprotos - ex:get_method_inheritance(Geom)
(#99; thanks @yjing14 for bug report).
Significant usability improvements, including {cli}
integration.
-
Convenience functions
layer_before_stat()
,layer_after_stat()
,layer_before_geom()
, andlayer_after_scale()
that returns a snapshot of layer data in the internals. Inspired byggplot2::layer_data()
with a similar interface. (#97; thanks @JoFrhwld for suggestion) -
Interactive debugging functions
last_layer_errorcontext()
andlast_sublayer_errorcontext()
which return the internal context of layer errors at the level of theLayer
and sub-Layer
(e.g.,Stat
orGeom
) ggproto methods, respectively.last_sublayer_errorcontext()
is still in experimental phase (error-prone, may be removed in future). -
New catch-all Inspect workflow function
ggtrace_inspect_on_error()
which dumps information about themethod
that errors while renderingx
.
- Inspect workflow functions gain a
error = TRUE
argument, which allows inspection of an earlier intermediate step even when the ggplot rendering process fails down the line (#89)
- Internal
is.ggtrace_placeholder()
returns scalar logical (#94)
ggtrace_inspect_vars()
simplifies the output whenat
is length 1.
ggtrace_capture_env()
removes ggtrace-internal variables before snapshotting the environment (#88)
cond
argument of workflow functions now support integer shorthand for conditioning on the counter. E.g.,cond = 1L
is converted tocond = quote(._counter_ == 1L)
. Multi-length integer vector is supported for highjack workflows. (#84)
-
Added complements to the
base::debug()
family of functions that are compatible with ggproto methods -ggdebug()
,ggdebugonce()
,ggundebug()
-
get_method_inheritance()
to get the list of methods from self and parent ggprotos -
ggtrace_inspect_n()
to get the number of times a method was called in the evaluation of a ggplot -
ggtrace_inspect_which()
to get the indices when a conditioncond
evaluated to true inside a method -
ggtrace_inspect_vars()
to get the value of variables at specified steps of a method's execution -
ggtrace_inspect_args()
to get the value of arguments passed to a method -
ggtrace_highjack_args()
to modify formals of a method at its execution -
with_ggtrace()
gets aout
argument which can take 1 of three options:- "t" or "tracedump" (default): returns the local tracedump from triggering traces on the
method
as the ggplotx
is evaluated - "g" or "gtable": Invisibly returns the
<gtable>
grob after evaluatingx
with injected expressions inmethod
. - "b" or "both": returns the tracedump while rendering the gtable (with
grid::grid.draw()
) as a side effect.
- "t" or "tracedump" (default): returns the local tracedump from triggering traces on the
-
Low-level functions
ggtrace()
/gguntrace()
and wrapperwith_ggtrace()
can now take quosures in themethod
argument, which allows them to be used more programmatically.
ggtrace_capture_env()
default value ofat
is changed to-1L
, which captures a snapshot of the runtime environment right before the method returns. Only the first element is used ifat
is length > 1- "modify" workflows are renamed to "highjack" (starting with
ggtrace_highjack_return()
) to reflect the fact that they always return the graphical output (gtable grob) (#78)
- Fixed an issue where a trace would fail to remove itself with
ggtrace(once = TRUE)
if it was triggered by a copy of the traced function (#59)
- Added
with_ggtrace()
for a functional interface toggtrace()
- Added
ggtrace_capture_fn()
andggtrace_capture_env()
, which return a snapshot of the function/environment of the ggproto method at execution time - Added
ggtrace_inspect_return()
andggtrace_modify_return()
to grab and swap return values at method's execution
- Added workflows section to docs/references
- Added
ggformals()
which returns theformals()
of functions and ggproto methods ggbody()
gains aas.list
argument to control whether output ofbody()
is turned into a list- Exported
set_last_ggtrace()
andset_global_ggtrace()
for the tracedumps
- Fixed compatibility issues with {rlang} v1.0.0 new changes to the call and expression API
- Added
get_method()
which returns ggproto methods as functions - Added wrappers for one-off workflows
ggdebugonce()
andggtraceback()
- The default value for the
verbose
argument ofggtrace()
is changed toFALSE
.
- Deparsed expressions printed as messages are now wrapped in backticks (#57)
ggedit()
now only works whenisTRUE(interactive())
(#62)
- Added
global_ggtrace_on/off()
which are aliases forglobal_ggtrace_state(TRUE/FALSE)
(#63)
clear_(last|global)_ggtrace()
functions now print a message saying that the trace dump has been cleared.- Better error handling for closures that aren't searchable from environments
- Fixed but where methods with class were being treated as bare functions. Now closures and methods outside of ggproto objects (e.g., R6 classes) can be traced too.
- Added FAQ vignette
- Global tracedump is turned off by default (
global_ggtrace_state()
isFALSE
on load) and must be explicitly activated withglobal_ggtrace_state(TRUE)
.
- Improved messages for
global_ggtrace_state()
- Fixed bug where
print_output = TRUE
would evaluate the expression twice (problematic for Inject workflows and causing general slowdowns)
- Global collection of tracedumps can be turned on/off with
global_ggtrace_state()
. It is still active by default but should memory become a concern, it can be turned off withglobal_ggtrace_state(state = FALSE)
.
- All functions in the package now support the tracing/untracing of any arbitrary functions (exported, unexported, user-defined, etc.). This also includes S3/S4 methods like
ggplo2:::ggplot_build.ggplot
andggplot2:::ggplot_add.Layer
. Some other examples of what's now possible.
ggtrace()
gains a...
for extensibility in future updates.ggedit()
gains aremove_trace
argument. WhenTRUE
, it untraces first before editing.- Improved error messages for invalid expressions passed to
method
argument ofggbody()
- Fix bug where
ggtrace()
fails to evaluate the first line when it's the only reachable line (#44)
- Fixed bug where incomplete traces due to early returns would not be logged without any special warning.
ggtrace()
now throws a warning when the actual number of traced steps does not match the expected number of traced steps (i.e., length oftrace_steps
) and logs incomplete tracedumps (#44)
ggtrace(..., once = TRUE)
is less noisy about there being a persistent trace. It now only sends a line of message saying so when the persistent trace is created. When it's triggered, it'll tell you that there is a persistent trace on the methodbut will not remind you to untrace it later nor print the correspondinggguntrace()
code to do so.
- Fixed bug where
trace_exprs
evaluating toNULL
would be removed from the tracedump (#38)
- New function
is_traced()
checks whether a method is currently being traced
-
ggtrace()
now breaks early with a more informative message if the method is not a function (ex:Stat$extra_params
, which is a 1-length character vector"na.rm"
) -
ggbody()
now warns if it's returning the body of a method that's currently being traced (#35) -
ggedit()
now informs you if you're editing on top of an existing trace/edit.
-
Fixed bug where
trace_exprs
would fail to be recycled if it is a list of length-1. Now a 1-length list of an expression as well as an expression by itself will get recycled to match the number of steps passed totrace_steps
. -
Fixed bug where
ggtrace()
wouldn't loop overtrace_exprs
after the first time it's triggered with persistent tracing on (once = TRUE
). Issue was due to failing to reset the internal counter after each time the trace is triggered.
- The
.print
argument toggtrace()
is renamed toprint_output
to make its functionality more transparent.
Several options for finer control over printing and formatting of output from ggtrace()
, in addition to the existing .print
argument:
-
ggtrace()
gets ause_names
argument. WhenTRUE
, it uses the names of the list of expressions passed totrace_exprs
as the names for the tracedump set tolast_ggtrace()
and added toglobal_ggtrace()
. -
ggtrace()
gets averbose
argument. WhenFALSE
, it suppresses the display of all non-message()
information, including information about which expression is evaluated where, as well as the output of those expressions when evaluated (which can be selectively suppressed with.print
for finer control).verbose
isTRUE
by default. -
Setting
options(ggtrace.suppressMessages = TRUE)
will also suppressmessages()
s about what method is being traced, whether a trace has been triggered on a method, whether there exists a persistent trace, etc. This information is very important so using this option is not recommended, but it has been made available. This option is set toFALSE
on package load. -
Setting
options(ggtrace.as_tibble = TRUE)
will return evaluated expressions as tibbles if the output is a data frame. Using this option may be convenient for interactive inspections but it is not recommended for testing or debugging (see related {ggplot2} Github issue). This option is set toFALSE
on package load.
-
Tracedumps accumulated in
global_ggtrace()
are named after the method (+ hexadecimal ID) for ease of searching. (#31) -
Triggering of a trace is now informed via
message()
instead ofcat()
(#29)
ggtrace()
correctly throws an error whentrace_steps
is not ordered. This is checked after the negative index conversion, so something liketrace_steps = c(1, -1)
still works fine).
-
New function
global_ggtrace()
which returns accumulated tracedumps fromggtrace()
. This is useful in conjunction withggtrace(once = FALSE)
for tracing a method that you expect will be called multiple times (ex:Stat$compute_group
gets triggered the same number of times as the number of groups in a panel). -
New function
clear_global_ggtrace()
which clearsglobal_ggtrace()
by setting it toNULL
.
-
Functions now pass around quosures instead of expressions (#29)
-
Testing setup (#28)
- Fixed a bug where the number of expressions passed to
trace_exprs
were allowed to be different from the number oftrace_steps
, causingggtrace()
to silently fail. This now throws an error.
-
trace_exprs
argument can now take a named list of expressions (#21) -
Improved documentation for
ggbody()
-
More informative messages for
ggtrace()
andgguntrace()
- No longer errors on exit when creating a persistence trace with
once = FALSE
-
The unary functions
ggedit()
andgguntrace()
gain dynamic dots...
as the second argument, which gets ignored. This now makes it easy to call these function by modifying the call to an earlier{ggtrace}
function from the console or in other interactive contexts. -
Significant re-write of
ggbody()
for better error handling (#23)-
Aborts if method is not a call or is not of the accepted form, with specific error messages for each.
-
If a method doesn't exist in a parent, directs users to call
ggbody(, inherit = TRUE)
-
If the recursive search with
inherit = TRUE
fails, directs users to load all relevant packages -
Notifies if
inherit = TRUE
but method is defined for the object, not inherited
-
-
Better error handling for
gguntrace()
when the method is no longer being traced (#24)-
Uses the re-written
ggbody()
to validate the method -
Unlike
base::untrace()
, no longer errors if given a method not currently being traced. It now prints a message saying so instead
-
-
Standardization of messages printed by all
{ggtrace}
functions. Messages are now more informative and refer to the ggproto method in the callable formatggproto$method
.
-
The
obj
argument is completely removed from all functions in the package. The constraint on supplying methods as expressions forces users to be intentional about tracing ggproto methods by having to provide them as code. This also allows the functions to return more informative messages, which was the main motivation for this breaking change.-
Because the `obj` argument was only designed for compatibility with the `get("method", ggproto)` syntax for retrieving the function body of ggproto methods, it should not affect interactive workflows. In fact, the shortform `method = ggproto$method` is more convenient and has always been recommended for passing a ggproto method to `{ggtrace}` functions.
-
-
As a reminder, all functions that take the ggproto method in the
method
argument expects an expression in the following forms (this part hasn't changed):ggproto$method
namespace::ggproto$method
namespace:::ggproto$method
-
Accurate string conversion for ggproto objects (#9), made possible by the breaking change.
-
trace_steps
argument can now take negative indices (#22) and has better error handling for out of range indices. -
Better deparsing for the
split_ggproto_method()
internal. -
Added a Tips & Tricks section to the documentation for
ggtrace()
.
- Internal variable
.store
renamed to.ggtrace_storage
to prevent overridingggplot2::.store
(#18)
-
The
~line
keyword forggtrace()
is renamed to~step
for consistency with the argument nametrace_steps
(#14) -
For safety reasons, the
~step
keyword will now only be substituted for the expression at the current step only if~step
is by itself (i.e., is an exact match) (#16, #11).-
For example, `~step` will not be substituted if `quote(head(~step))` is passed to `trace_exprs`. Users are encouraged to return the method's environment with `quote(environment())` or interactively debug with `ggedit()` if they want to manipulate the expression.
-
-
The position of the
obj
argument ofggtrace()
has been moved from second to fourth, to allow for shortcuts likeggtrace(method = ..., 2:3, quote(data))
, which will evaluate and store the output of thedata
variable at the second and third steps of the method body. (#15)
-
trace_exprs
argument ofggtrace()
is now optional. If not provided, defaults to~step
(#13) -
You can now tell
ggbody()
to (recursively) search for the method from its parents withinherit = TRUE
(#12)
- Documentation for some of the functions now contain a Gotchas section for explanations of / solutions to common problems (#10)
-
New function
gguntrace()
with the same syntax for specifying the ggproto method. -
ggtrace()
gains aonce = TRUE
argument, which can be set toFALSE
for persistent tracing
- Fix bug in
ggtrace()
wherestep_deparsed
was being returned as a multi-length vector
- New function
ggedit()
for interactive debugging via directly editing the source code.
-
Refactored
ggtrace()
. The package now only depends on{rlang}
. -
Significant re-write to the readme / documentation
pkgdown site: https://yjunechoe.github.io/ggtrace
Functions: