Title: | Generates Expectations for 'testthat' Unit Testing |
---|---|
Description: | Helps systematize and ease the process of building unit tests with the 'testthat' package by providing tools for generating expectations. |
Authors: | Ludvig Renbo Olsen [aut, cre] |
Maintainer: | Ludvig Renbo Olsen <[email protected]> |
License: | MIT + file LICENSE |
Version: | 0.4.4 |
Built: | 2025-03-05 15:27:16 UTC |
Source: | https://github.com/ludvigolsen/xpectr |
testthat
unit testingA set of utilities and RStudio addins for generating tests.
Ludvig Renbo Olsen, [email protected]
Useful links:
RStudio Addin:
Inserts code for initializing and reporting a
checkmate assert collection
.
See `Details`
for how to set a key command.
assertCollectionAddin(add_comments = TRUE, insert = TRUE, indentation = NULL)
assertCollectionAddin(add_comments = TRUE, insert = TRUE, indentation = NULL)
add_comments |
Whether to add comments around. (Logical) This makes it easy for a user to create their own addin without the comments. |
insert |
Whether to insert the code via
N.B. Mainly intended for testing the addin programmatically. |
indentation |
Indentation of the code. (Numeric) N.B. Mainly intended for testing the addin programmatically. |
After installing the package. Go to:
Tools >> Addins >> Browse Addins >> Keyboard Shortcuts
.
Find "Insert checkmate AssertCollection Code"
and press its field under Shortcut
.
Press desired key command, e.g. Alt+C
.
Press Apply
.
Press Execute
.
Inserts the following (excluding the ----
):
----
# Check arguments ####
assert_collection <- checkmate::makeAssertCollection()
# checkmate::assert_ , add = assert_collection)
checkmate::reportAssertions(assert_collection)
# End of argument checks ####
----
Returns NULL
invisibly.
Ludvig Renbo Olsen, [email protected]
Other addins:
dputSelectedAddin()
,
initializeGXSFunctionAddin()
,
initializeTestthatAddin()
,
insertExpectationsAddin()
,
navigateTestFileAddin()
,
wrapStringAddin()
Wraps string in capture_side_effects()
before parsing and evaluating it.
The side effects (error
, warning
s, message
s) are returned in a list
.
When capturing an error
, no other side effects are captured.
capture_parse_eval_side_effects( string, envir = NULL, copy_env = FALSE, reset_seed = FALSE, disable_crayon = TRUE )
capture_parse_eval_side_effects( string, envir = NULL, copy_env = FALSE, reset_seed = FALSE, disable_crayon = TRUE )
string |
String of code that can be parsed and evaluated in |
envir |
Environment to evaluate in. Defaults to
|
copy_env |
Whether to use deep copies of the environment when capturing side effects. (Logical) Disabled by default to save memory but is often preferable to enable, e.g. when the function
alters non-local variables before throwing its |
reset_seed |
Whether to reset the random state on exit. (Logical) |
disable_crayon |
Whether to disable |
named list
with the side effects.
Ludvig Renbo Olsen, [email protected]
Other capturers:
capture_side_effects()
# Attach package library(xpectr) capture_parse_eval_side_effects("stop('hi!')") capture_parse_eval_side_effects("warning('hi!')") capture_parse_eval_side_effects("message('hi!')")
# Attach package library(xpectr) capture_parse_eval_side_effects("stop('hi!')") capture_parse_eval_side_effects("warning('hi!')") capture_parse_eval_side_effects("message('hi!')")
Captures error
s, warning
s, and message
s from an expression.
In case of an error
, no other side effects are captured.
Simple wrapper for testthat
's
capture_error()
,
capture_warnings()
and
capture_messages()
.
Note: Evaluates expr
up to three times.
capture_side_effects( expr, envir = NULL, copy_env = FALSE, reset_seed = FALSE, disable_crayon = TRUE )
capture_side_effects( expr, envir = NULL, copy_env = FALSE, reset_seed = FALSE, disable_crayon = TRUE )
expr |
Expression. |
envir |
Environment to evaluate in. Defaults to
|
copy_env |
Whether to use deep copies of the environment when capturing side effects. (Logical) Disabled by default to save memory but is often preferable to enable, e.g. when the function
alters non-local variables before throwing its |
reset_seed |
Whether to reset the random state on exit. (Logical) |
disable_crayon |
Whether to disable |
named list
with the side effects.
Ludvig Renbo Olsen, [email protected]
Other capturers:
capture_parse_eval_side_effects()
# Attach packages library(xpectr) fn <- function(raise = FALSE){ message("Hi! I'm Kevin, your favorite message!") warning("G'Day Mam! I'm a warning to the world!") message("Kevin is ma name! Yesss!") warning("Hopefully the whole world will see me :o") if (isTRUE(raise)){ stop("Lord Evil Error has arrived! Yeehaaa") } "the output" } capture_side_effects(fn()) capture_side_effects(fn(raise = TRUE)) capture_side_effects(fn(raise = TRUE), copy_env = TRUE)
# Attach packages library(xpectr) fn <- function(raise = FALSE){ message("Hi! I'm Kevin, your favorite message!") warning("G'Day Mam! I'm a warning to the world!") message("Kevin is ma name! Yesss!") warning("Hopefully the whole world will see me :o") if (isTRUE(raise)){ stop("Lord Evil Error has arrived! Yeehaaa") } "the output" } capture_side_effects(fn()) capture_side_effects(fn(raise = TRUE)) capture_side_effects(fn(raise = TRUE), copy_env = TRUE)
RStudio Addin:
Runs dput()
on the selected code and inserts
it instead of the selection.
See `Details`
for how to set a key command.
dputSelectedAddin(selection = NULL, insert = TRUE, indentation = 0)
dputSelectedAddin(selection = NULL, insert = TRUE, indentation = 0)
selection |
String of code. (Character) E.g. N.B. Mainly intended for testing the addin programmatically. |
insert |
Whether to insert the expectations via
N.B. Mainly intended for testing the addin programmatically. |
indentation |
Indentation of the selection. (Numeric) N.B. Mainly intended for testing the addin programmatically. |
Parses and evaluates the selected code string,
applies dput()
and
inserts the output instead of the selection.
After installing the package. Go to:
Tools >> Addins >> Browse Addins >> Keyboard Shortcuts
.
Find "dput() Selected"
and press its field under Shortcut
.
Press desired key command, e.g. Alt+D
.
Press Apply
.
Press Execute
.
Inserts the output of running
dput()
on the selected code.
Does not return anything.
Ludvig Renbo Olsen, [email protected]
Other addins:
assertCollectionAddin()
,
initializeGXSFunctionAddin()
,
initializeTestthatAddin()
,
insertExpectationsAddin()
,
navigateTestFileAddin()
,
wrapStringAddin()
Applies class()
to each element of `x`
(without recursion). When
class()
returns multiple strings, the first class string is returned.
element_classes(x, keep_names = FALSE)
element_classes(x, keep_names = FALSE)
x |
List with elements. |
keep_names |
Whether to keep existing names. (Logical) |
Gets first string in class()
for all elements.
The main class of each element.
Ludvig Renbo Olsen, [email protected]
Other element descriptors:
element_lengths()
,
element_types()
,
num_total_elements()
# Attach packages library(xpectr) l <- list("a" = c(1,2,3), "b" = "a", "c" = NULL) element_classes(l) element_classes(l, keep_names = TRUE)
# Attach packages library(xpectr) l <- list("a" = c(1,2,3), "b" = "a", "c" = NULL) element_classes(l) element_classes(l, keep_names = TRUE)
Applies length()
to each element of `x`
(without recursion).
element_lengths(x, keep_names = FALSE)
element_lengths(x, keep_names = FALSE)
x |
List with elements. |
keep_names |
Whether to keep existing names. (Logical) |
Simple wrapper for unlist(lapply(x, length))
.
The length of each element.
Ludvig Renbo Olsen, [email protected]
Other element descriptors:
element_classes()
,
element_types()
,
num_total_elements()
# Attach packages library(xpectr) l <- list("a" = c(1,2,3), "b" = 1, "c" = NULL) element_lengths(l) element_lengths(l, keep_names = TRUE)
# Attach packages library(xpectr) l <- list("a" = c(1,2,3), "b" = 1, "c" = NULL) element_lengths(l) element_lengths(l, keep_names = TRUE)
Applies typeof()
to each element of `x`
(without recursion).
element_types(x, keep_names = FALSE)
element_types(x, keep_names = FALSE)
x |
List with elements. |
keep_names |
Whether to keep existing names. (Logical) |
Simple wrapper for unlist(lapply(x, typeof))
.
The type of each element.
Ludvig Renbo Olsen, [email protected]
Other element descriptors:
element_classes()
,
element_lengths()
,
num_total_elements()
# Attach packages library(xpectr) l <- list("a" = c(1,2,3), "b" = "a", "c" = NULL) element_types(l) element_types(l, keep_names = TRUE)
# Attach packages library(xpectr) l <- list("a" = c(1,2,3), "b" = "a", "c" = NULL) element_types(l) element_types(l, keep_names = TRUE)
Based on a set of supplied values for each function argument,
a set of testthat
expect_*
statements are generated.
Included tests: The first value supplied for an argument is considered the valid baseline value. For each argument, we create tests for each of the supplied values, where the other arguments have their baseline value.
When testing a function that alters non-local variables, consider enabling `copy_env`
.
See supported objects in details
.
gxs_function( fn, args_values, extra_combinations = NULL, check_nulls = TRUE, indentation = 0, tolerance = "1e-4", round_to_tolerance = TRUE, strip = TRUE, sample_n = 30, envir = NULL, copy_env = FALSE, assign_output = TRUE, seed = 42, add_wrapper_comments = TRUE, add_test_comments = TRUE, start_with_newline = TRUE, end_with_newline = TRUE, out = "insert", parallel = FALSE )
gxs_function( fn, args_values, extra_combinations = NULL, check_nulls = TRUE, indentation = 0, tolerance = "1e-4", round_to_tolerance = TRUE, strip = TRUE, sample_n = 30, envir = NULL, copy_env = FALSE, assign_output = TRUE, seed = 42, add_wrapper_comments = TRUE, add_test_comments = TRUE, start_with_newline = TRUE, end_with_newline = TRUE, out = "insert", parallel = FALSE )
fn |
Function to create tests for. |
args_values |
The arguments and the values to create tests for. Should be supplied as a named list of lists, like the following:
The first value for each argument (referred to as the 'baseline' value) should be valid
(not throw an N.B. This is not checked but should lead to more meaningful tests. N.B. Please define the list directly in the function call. This is currently necessary. |
extra_combinations |
Additional combinations to test. List of lists, where each combination is a named sublist. E.g. the following two combinations:
N.B. Unspecified arguments gets the baseline value. If you find yourself adding many combinations,
an additional |
check_nulls |
Whether to try all arguments with When enabled, you don't need to add |
indentation |
Indentation of the selection. (Numeric) |
tolerance |
The tolerance for numeric tests as a string, like |
round_to_tolerance |
Whether to round numeric elements to the specified tolerance. (Logical) This is currently applied to numeric columns and vectors (excluding some lists). |
strip |
Whether to insert
Sometimes testthat tests have differences in punctuation and newlines on different systems. By stripping both the error message and the expected message of non-alphanumeric symbols, we can avoid such failed tests. |
sample_n |
The number of elements/rows to sample. Set to Inserts The order of the elements/rows is kept intact. No replacement is used, why no oversampling will take place. When testing a big |
envir |
Environment to evaluate in. Defaults to
|
copy_env |
Whether each combination should be tested in a deep copy of the environment. (Logical) Side effects will be captured in copies of the copy, why two copies of the environment will exist at the same time. Disabled by default to save memory but is often preferable to enable, e.g. when the function changes non-local variables. |
assign_output |
Whether to assign the output of a function call or long selection to a variable. This will avoid recalling the function and decrease cluttering. (Logical) Heuristic: when the The tests themselves can be more difficult to interpret, as you will have to look at the assignment to see the object that is being tested. |
seed |
|
add_wrapper_comments |
Whether to add intro and outro comments. (Logical) |
add_test_comments |
Whether to add comments for each test. (Logical) |
start_with_newline , end_with_newline
|
Whether to have a newline in the beginning/end. (Logical) |
out |
Either "insert" (Default)Inserts the expectations via
"return"Returns the expectations in a These can be prepared for insertion with
|
parallel |
Whether to parallelize the generation of expectations. (Logical) Requires a registered parallel backend. Like with |
The following "types" are currently supported or intended to be supported in the future. Please suggest more types and tests in a GitHub issue!
Note: A set of fallback tests will be generated for unsupported objects.
Type | Supported | Notes |
Side effects | Yes | Errors, warnings, and messages. |
Vector | Yes | Lists are treated differently, depending on their structure. |
Factor | Yes | |
Data Frame | Yes | List columns (like nested tibbles) are currently skipped. |
Matrix | Yes | Supported but could be improved. |
Formula | Yes | |
Function | Yes | |
NULL |
Yes | |
Array | No | |
Dates | No | Base and lubridate . |
ggplot2 | No | This may be a challenge, but would be cool! |
Either NULL
or the unprepared expectations as a character vector
.
Ludvig Renbo Olsen, [email protected]
Other expectation generators:
gxs_selection()
,
initializeGXSFunctionAddin()
,
insertExpectationsAddin()
# Attach packages library(xpectr) ## Not run: fn <- function(x, y, z){ if (x>3) stop("'x' > 3") if (y<0) warning("'y'<0") if (z==10) message("'z' was 10!") x + y + z } # Create expectations # Note: define the list in the call gxs_function(fn, args_values = list( "x" = list(2, 4, NA), "y" = list(0, -1), "z" = list(5, 10)) ) # Add additional combinations gxs_function(fn, args_values = list( "x" = list(2, 4, NA), "y" = list(0, -1), "z" = list(5, 10)), extra_combinations = list( list("x" = 4, "z" = 10), list("y" = 1, "z" = 10)) ) ## End(Not run)
# Attach packages library(xpectr) ## Not run: fn <- function(x, y, z){ if (x>3) stop("'x' > 3") if (y<0) warning("'y'<0") if (z==10) message("'z' was 10!") x + y + z } # Create expectations # Note: define the list in the call gxs_function(fn, args_values = list( "x" = list(2, 4, NA), "y" = list(0, -1), "z" = list(5, 10)) ) # Add additional combinations gxs_function(fn, args_values = list( "x" = list(2, 4, NA), "y" = list(0, -1), "z" = list(5, 10)), extra_combinations = list( list("x" = 4, "z" = 10), list("y" = 1, "z" = 10)) ) ## End(Not run)
Based on the selection (string of code), a set of testthat
expect_*
statements are generated.
Example: If the selected code is the name of a data.frame
object,
it will create an expect_equal
test for each column, along with a test of the column names,
types and classes, dimensions, grouping keys, etc.
See supported objects in details
.
When testing a function that alters non-local variables, consider enabling `copy_env`
.
Feel free to suggest useful tests etc. in a GitHub issue!
Addin: insertExpectationsAddin()
gxs_selection( selection, indentation = 0, tolerance = "1e-4", round_to_tolerance = TRUE, strip = TRUE, sample_n = 30, envir = NULL, copy_env = FALSE, assign_output = TRUE, seed = 42, test_id = NULL, add_wrapper_comments = TRUE, add_test_comments = TRUE, start_with_newline = TRUE, end_with_newline = TRUE, out = "insert" )
gxs_selection( selection, indentation = 0, tolerance = "1e-4", round_to_tolerance = TRUE, strip = TRUE, sample_n = 30, envir = NULL, copy_env = FALSE, assign_output = TRUE, seed = 42, test_id = NULL, add_wrapper_comments = TRUE, add_test_comments = TRUE, start_with_newline = TRUE, end_with_newline = TRUE, out = "insert" )
selection |
String of code. (Character) E.g. |
indentation |
Indentation of the selection. (Numeric) |
tolerance |
The tolerance for numeric tests as a string, like |
round_to_tolerance |
Whether to round numeric elements to the specified tolerance. (Logical) This is currently applied to numeric columns and vectors (excluding some lists). |
strip |
Whether to insert
Sometimes testthat tests have differences in punctuation and newlines on different systems. By stripping both the error message and the expected message of non-alphanumeric symbols, we can avoid such failed tests. |
sample_n |
The number of elements/rows to sample. Set to Inserts The order of the elements/rows is kept intact. No replacement is used, why no oversampling will take place. When testing a big |
envir |
Environment to evaluate in. Defaults to
|
copy_env |
Whether to work in a deep copy of the environment. (Logical) Side effects will be captured in copies of the copy, why two copies of the environment will exist at the same time. Disabled by default to save memory but is often preferable to enable, e.g. when the function changes non-local variables. |
assign_output |
Whether to assign the output of a function call or long selection to a variable. This will avoid recalling the function and decrease cluttering. (Logical) Heuristic: when the The tests themselves can be more difficult to interpret, as you will have to look at the assignment to see the object that is being tested. |
seed |
|
test_id |
Number to append to assignment names. (Whole number) For instance used to create the |
add_wrapper_comments |
Whether to add intro and outro comments. (Logical) |
add_test_comments |
Whether to add comments for each test. (Logical) |
start_with_newline , end_with_newline
|
Whether to have a newline in the beginning/end. (Logical) |
out |
Either "insert" (Default)Inserts the expectations via
"return"Returns the expectations in a These can be prepared for insertion with
|
The following "types" are currently supported or intended to be supported in the future. Please suggest more types and tests in a GitHub issue!
Note: A set of fallback tests will be generated for unsupported objects.
Type | Supported | Notes |
Side effects | Yes | Errors, warnings, and messages. |
Vector | Yes | Lists are treated differently, depending on their structure. |
Factor | Yes | |
Data Frame | Yes | List columns (like nested tibbles) are currently skipped. |
Matrix | Yes | Supported but could be improved. |
Formula | Yes | |
Function | Yes | |
NULL |
Yes | |
Array | No | |
Dates | No | Base and lubridate . |
ggplot2 | No | This may be a challenge, but would be cool! |
Either NULL
or the unprepared expectations as a character vector
.
Ludvig Renbo Olsen, [email protected]
Other expectation generators:
gxs_function()
,
initializeGXSFunctionAddin()
,
insertExpectationsAddin()
# Attach packages library(xpectr) ## Not run: df <- data.frame('a' = c(1, 2, 3), 'b' = c('t', 'y', 'u'), stringsAsFactors = FALSE) gxs_selection("stop('This gives an expect_error test!')") gxs_selection("warning('This gives a set of side effect tests!')") gxs_selection("message('This also gives a set of side effect tests!')") gxs_selection("stop('This: tests the -> punctuation!')", strip = FALSE) gxs_selection("sum(1, 2, 3, 4)") gxs_selection("df") tests <- gxs_selection("df", out = "return") for_insertion <- prepare_insertion(tests) rstudioapi::insertText(for_insertion) ## End(Not run)
# Attach packages library(xpectr) ## Not run: df <- data.frame('a' = c(1, 2, 3), 'b' = c('t', 'y', 'u'), stringsAsFactors = FALSE) gxs_selection("stop('This gives an expect_error test!')") gxs_selection("warning('This gives a set of side effect tests!')") gxs_selection("message('This also gives a set of side effect tests!')") gxs_selection("stop('This: tests the -> punctuation!')", strip = FALSE) gxs_selection("sum(1, 2, 3, 4)") gxs_selection("df") tests <- gxs_selection("df", out = "return") for_insertion <- prepare_insertion(tests) rstudioapi::insertText(for_insertion) ## End(Not run)
gxs_function()
callInitializes the gxs_function()
call with the arguments and default values
of the selected function.
See `Details`
for how to set a key command.
initializeGXSFunctionAddin(selection = NULL, insert = TRUE, indentation = 0)
initializeGXSFunctionAddin(selection = NULL, insert = TRUE, indentation = 0)
selection |
Name of function to test with N.B. Mainly intended for testing the addin programmatically. |
insert |
Whether to insert the code via
N.B. Mainly intended for testing the addin programmatically. |
indentation |
Indentation of the selection. (Numeric) N.B. Mainly intended for testing the addin programmatically. |
Parses and evaluates the selected code string
within the parent environment.
When the output is a function, it extracts the formals (arguments and default values)
and creates the initial `args_values`
for gxs_function()
.
When the output is not a function, it throws an error.
After installing the package. Go to:
Tools >> Addins >> Browse Addins >> Keyboard Shortcuts
.
Find "Initialize gxs_function()"
and press its field under Shortcut
.
Press desired key command, e.g. Alt+F
.
Press Apply
.
Press Execute
.
Inserts gxs_function()
call for
the selected function.
Returns NULL
invisibly.
Ludvig Renbo Olsen, [email protected]
Other expectation generators:
gxs_function()
,
gxs_selection()
,
insertExpectationsAddin()
Other addins:
assertCollectionAddin()
,
dputSelectedAddin()
,
initializeTestthatAddin()
,
insertExpectationsAddin()
,
navigateTestFileAddin()
,
wrapStringAddin()
test_that()
callInserts code for calling testthat::test_that()
.
See `Details`
for how to set a key command.
initializeTestthatAddin(insert = TRUE, indentation = NULL)
initializeTestthatAddin(insert = TRUE, indentation = NULL)
insert |
Whether to insert the code via
N.B. Mainly intended for testing the addin programmatically. |
indentation |
Indentation of the code. (Numeric) N.B. Mainly intended for testing the addin programmatically. |
After installing the package. Go to:
Tools >> Addins >> Browse Addins >> Keyboard Shortcuts
.
Find "Initialize test_that()"
and press its field under Shortcut
.
Press desired key command, e.g. Alt+T
.
Press Apply
.
Press Execute
.
Inserts code for calling
testthat::test_that()
.
Returns NULL
invisibly.
Ludvig Renbo Olsen, [email protected]
Other addins:
assertCollectionAddin()
,
dputSelectedAddin()
,
initializeGXSFunctionAddin()
,
insertExpectationsAddin()
,
navigateTestFileAddin()
,
wrapStringAddin()
Inserts relevant expect_*
tests based
on the evaluation of the selected code.
Example: If the selected code is the name of a data.frame
object,
it will create an expect_equal
test for each column,
along with a test of the column names.
Currently supports side effects (error
, warning
s, message
s),
data.frame
s, and vector
s.
List columns in data.frame
s (like nested tibble
s) are currently skipped.
See `Details`
for how to set a key command.
insertExpectationsAddin( selection = NULL, insert = TRUE, indentation = 0, copy_env = FALSE ) insertExpectationsCopyEnvAddin( selection = NULL, insert = TRUE, indentation = 0, copy_env = TRUE )
insertExpectationsAddin( selection = NULL, insert = TRUE, indentation = 0, copy_env = FALSE ) insertExpectationsCopyEnvAddin( selection = NULL, insert = TRUE, indentation = 0, copy_env = TRUE )
selection |
String of code. (Character) E.g. N.B. Mainly intended for testing the addin programmatically. |
insert |
Whether to insert the expectations via
N.B. Mainly intended for testing the addin programmatically. |
indentation |
Indentation of the selection. (Numeric) N.B. Mainly intended for testing the addin programmatically. |
copy_env |
Whether to work in a deep copy of the environment. (Logical) Side effects will be captured in copies of the copy, why two copies of the environment will exist at the same time. Disabled by default to save memory but is often preferable to enable, e.g. when the function changes non-local variables. |
Parses and evaluates the selected code string
within the parent environment (or a deep copy thereof).
Depending on the output, it creates a set of unit tests
(like expect_equal(data[["column"]], c(1,2,3))
),
and inserts them instead of the selection.
After installing the package. Go to:
Tools >> Addins >> Browse Addins >> Keyboard Shortcuts
.
Find "Insert Expectations"
and press its field under Shortcut
.
Press desired key command, e.g. Alt+E
.
Press Apply
.
Press Execute
.
Inserts testthat::expect_*
unit tests for the selected code.
Returns NULL
invisibly.
Ludvig Renbo Olsen, [email protected]
Other expectation generators:
gxs_function()
,
gxs_selection()
,
initializeGXSFunctionAddin()
Other addins:
assertCollectionAddin()
,
dputSelectedAddin()
,
initializeGXSFunctionAddin()
,
initializeTestthatAddin()
,
navigateTestFileAddin()
,
wrapStringAddin()
Unlists `x`
recursively and finds the total number of elements.
num_total_elements(x, deduplicated = FALSE)
num_total_elements(x, deduplicated = FALSE)
x |
List with elements. |
deduplicated |
Whether to only count the unique elements. (Logical) |
Simple wrapper for
length(unlist(x, recursive = TRUE, use.names = FALSE))
.
The total number of elements in `x`
.
Ludvig Renbo Olsen, [email protected]
Other element descriptors:
element_classes()
,
element_lengths()
,
element_types()
# Attach packages library(xpectr) l <- list(list(list(1, 2, 3), list(2, list(3, 2))), list(1, list(list(2, 4), list(7, 1, list(3, 8)))), list(list(2, 7, 8), list(10, 2, list(18, 1, 4)))) num_total_elements(l) num_total_elements(l, deduplicated = TRUE)
# Attach packages library(xpectr) l <- list(list(list(1, 2, 3), list(2, list(3, 2))), list(1, list(list(2, 4), list(7, 1, list(3, 8)))), list(list(2, 7, 8), list(10, 2, list(18, 1, 4)))) num_total_elements(l) num_total_elements(l, deduplicated = TRUE)
Collapses a list
/vector
of expectation strings and adds the specified indentation.
prepare_insertion( strings, indentation = 0, trim_left = FALSE, trim_right = FALSE )
prepare_insertion( strings, indentation = 0, trim_left = FALSE, trim_right = FALSE )
strings |
Expectation strings. (List or Character) As returned with |
indentation |
Indentation to add. (Numeric) |
trim_left |
Whether to trim whitespaces from the beginning of the collapsed string. (Logical) |
trim_right |
Whether to trim whitespaces from the end of the collapsed string. (Logical) |
A string for insertion with rstudioapi::insertText()
.
Ludvig Renbo Olsen, [email protected]
# Attach packages library(xpectr) ## Not run: df <- data.frame('a' = c(1, 2, 3), 'b' = c('t', 'y', 'u'), stringsAsFactors = FALSE) tests <- gxs_selection("df", out = "return") for_insertion <- prepare_insertion(tests) for_insertion rstudioapi::insertText(for_insertion) ## End(Not run)
# Attach packages library(xpectr) ## Not run: df <- data.frame('a' = c(1, 2, 3), 'b' = c('t', 'y', 'u'), stringsAsFactors = FALSE) tests <- gxs_selection("df", out = "return") for_insertion <- prepare_insertion(tests) for_insertion rstudioapi::insertText(for_insertion) ## End(Not run)
In order for tests to be compatible with R
versions < 3.6.0
,
we set the sample.kind
argument in set.seed()
to "Rounding"
when using R
versions >= 3.6.0
.
set_test_seed(seed = 42, ...)
set_test_seed(seed = 42, ...)
seed |
Random |
... |
Named arguments to |
Initially contributed by R. Mark Sharp (github: @rmsharp).
NULL
.
Ludvig Renbo Olsen, [email protected]
R. Mark Sharp
Extracts formals
and
formats them as an easily testable character vector
.
simplified_formals(fn)
simplified_formals(fn)
fn |
|
A character vector
with the simplified formals.
Ludvig Renbo Olsen, [email protected]
# Attach packages library(xpectr) fn1 <- function(a = "x", b = NULL, c = NA, d){ paste0(a, b, c, d) } simplified_formals(fn1)
# Attach packages library(xpectr) fn1 <- function(a = "x", b = NULL, c = NA, d){ paste0(a, b, c, d) } simplified_formals(fn1)
Samples a vector
, factor
or data.frame
. Useful to reduce size of testthat expect_*
tests.
Not intended for other purposes.
Wraps sample.int()
. data.frame
s are sampled row-wise.
The seed
is set within the function with sample.kind
as "Rounding"
for compatibility with R
versions < 3.6.0
. On exit, the random state is restored.
smpl(data, n, keep_order = TRUE, seed = 42)
smpl(data, n, keep_order = TRUE, seed = 42)
data |
|
n |
Number of elements/rows to sample. N.B. No replacement is used, why |
keep_order |
Whether to keep the order of the elements. (Logical) |
seed |
The |
When `data`
has <=`n`
elements, `data`
is returned.
Otherwise, `data`
is sampled and returned.
Ludvig Renbo Olsen, [email protected]
# Attach packages library(xpectr) smpl(c(1,2,3,4,5), n = 3) smpl(data.frame("a" = c(1,2,3,4,5), "b" = c(2,3,4,5,6), stringsAsFactors = FALSE), n = 3)
# Attach packages library(xpectr) smpl(c(1,2,3,4,5), n = 3) smpl(data.frame("a" = c(1,2,3,4,5), "b" = c(2,3,4,5,6), stringsAsFactors = FALSE), n = 3)
If the `condition`
is TRUE
,
generate error
/warning
/message
with the supplied message.
stop_if(condition, message = NULL, sys.parent.n = 0L) warn_if(condition, message = NULL, sys.parent.n = 0L) message_if(condition, message = NULL, sys.parent.n = 0L)
stop_if(condition, message = NULL, sys.parent.n = 0L) warn_if(condition, message = NULL, sys.parent.n = 0L) message_if(condition, message = NULL, sys.parent.n = 0L)
condition |
The condition to check. (Logical) |
message |
Message. (Character) Note: If |
sys.parent.n |
The number of generations to go back when calling the message function. |
When `condition`
is FALSE
, they return NULL
invisibly.
When `condition`
is TRUE
:
Throws error with the supplied message.
Throws warning with the supplied message.
Generates message with the supplied message.
Returns NULL
invisibly.
Ludvig Renbo Olsen, [email protected]
# Attach packages library(xpectr) # Note: The use of `try()` is just for package-testing purposes a <- 0 try(stop_if(a == 0, "'a' cannot be 0.")) try(warn_if(a == 0, "'a' was 0.")) message_if(a == 0, "'a' was so kind to be 0.")
# Attach packages library(xpectr) # Note: The use of `try()` is just for package-testing purposes a <- 0 try(stop_if(a == 0, "'a' cannot be 0.")) try(warn_if(a == 0, "'a' was 0.")) message_if(a == 0, "'a' was so kind to be 0.")
Removes any character that is not alphanumeric or a space.
(Disabled by default): Remove numbers.
Reduces multiple consecutive whitespaces to a single whitespace and trims ends.
Can for instance be used to simplify error messages before checking them.
strip( strings, replacement = "", remove_spaces = FALSE, remove_numbers = FALSE, remove_ansi = TRUE, lowercase = FALSE, allow_na = TRUE )
strip( strings, replacement = "", remove_spaces = FALSE, remove_numbers = FALSE, remove_ansi = TRUE, lowercase = FALSE, allow_na = TRUE )
strings |
|
replacement |
What to replace blocks of punctuation with. (Character) |
remove_spaces |
Whether to remove all whitespaces. (Logical) |
remove_numbers |
Whether to remove all numbers. (Logical) |
remove_ansi |
Whether to remove ANSI control sequences. (Logical) |
lowercase |
Whether to make the strings lowercase. (Logical) |
allow_na |
Whether to allow |
ANSI control sequences are removed with fansi::strip_ctl()
.
gsub("[^[:alnum:][:blank:]]", replacement, strings))
gsub('[0-9]+', '', strings)
(Note: only if specified!)
trimws( gsub("[[:blank:]]+", " ", strings) )
(Or ""
if remove_spaces
is TRUE
)
The stripped strings.
Ludvig Renbo Olsen, [email protected]
Other strippers:
strip_msg()
# Attach packages library(xpectr) strings <- c( "Hello! I am George. \n\rDon't call me Frank! 123", " \tAs that, is, not, my, name!" ) strip(strings) strip(strings, remove_spaces = TRUE) strip(strings, remove_numbers = TRUE)
# Attach packages library(xpectr) strings <- c( "Hello! I am George. \n\rDon't call me Frank! 123", " \tAs that, is, not, my, name!" ) strip(strings) strip(strings, remove_spaces = TRUE) strip(strings, remove_numbers = TRUE)
Catches side effects (error
, warning
s, message
s), strips the message strings of
non-alphanumeric characters with strip()
and regenerates them.
When numbers in error messages vary slightly between systems (and this variation isn't important to catch), we can strip the numbers as well.
Use case: Sometimes testthat
tests have differences in punctuation and newlines on different
systems. By stripping both the error message and the expected message
(with strip()
), we can avoid such failed tests.
strip_msg( x, remove_spaces = FALSE, remove_numbers = FALSE, remove_ansi = TRUE, lowercase = FALSE )
strip_msg( x, remove_spaces = FALSE, remove_numbers = FALSE, remove_ansi = TRUE, lowercase = FALSE )
x |
Code that potentially throws |
remove_spaces |
Whether to remove all whitespaces. (Logical) |
remove_numbers |
Whether to remove all numbers. (Logical) |
remove_ansi |
Whether to remove ANSI control sequences. (Logical) |
lowercase |
Whether to make the strings lowercase. (Logical) |
Returns NULL
invisibly.
Ludvig Renbo Olsen, [email protected]
Other strippers:
strip()
# Attach packages library(xpectr) library(testthat) # Note: The use of `try()` is just for package-testing purposes try(strip_msg(stop("this 'dot' .\n is removed! 123"))) try(strip_msg(warning("this 'dot' .\n is removed! 123"))) strip_msg(message("this 'dot' .\n is removed! 123")) strip_msg(message("this 'dot' .\n is removed! 123"), remove_numbers = TRUE) error_fn <- function(){stop("this 'dot' .\n is removed! 123")} try(strip_msg(error_fn())) # With testthat tests expect_error(strip_msg(error_fn()), strip("this 'dot' .\n is removed! 123")) expect_error(strip_msg(error_fn(), remove_numbers = TRUE), strip("this 'dot' .\n is removed! 123", remove_numbers = TRUE))
# Attach packages library(xpectr) library(testthat) # Note: The use of `try()` is just for package-testing purposes try(strip_msg(stop("this 'dot' .\n is removed! 123"))) try(strip_msg(warning("this 'dot' .\n is removed! 123"))) strip_msg(message("this 'dot' .\n is removed! 123")) strip_msg(message("this 'dot' .\n is removed! 123"), remove_numbers = TRUE) error_fn <- function(){stop("this 'dot' .\n is removed! 123")} try(strip_msg(error_fn())) # With testthat tests expect_error(strip_msg(error_fn()), strip("this 'dot' .\n is removed! 123")) expect_error(strip_msg(error_fn(), remove_numbers = TRUE), strip("this 'dot' .\n is removed! 123", remove_numbers = TRUE))
Run expression wrapped in both
suppressMessages()
and
suppressWarnings()
.
suppress_mw(expr)
suppress_mw(expr)
expr |
Any expression to run within |
suppressWarnings(suppressMessages(expr))
The output of expr
.
Ludvig Renbo Olsen, [email protected]
# Attach packages library(xpectr) fn <- function(a, b){ warning("a warning") message("a message") a + b } suppress_mw(fn(1, 5))
# Attach packages library(xpectr) fn <- function(a, b){ warning("a warning") message("a message") a + b } suppress_mw(fn(1, 5))
Splits the selection every n characters
and inserts it in a paste0()
call.
See `Details`
for how to set a key command.
wrapStringAddin( selection = NULL, indentation = 0, every_n = NULL, tolerance = 10, insert = TRUE )
wrapStringAddin( selection = NULL, indentation = 0, every_n = NULL, tolerance = 10, insert = TRUE )
selection |
String of code. (Character) N.B. Mainly intended for testing the addin programmatically. |
indentation |
Indentation of the selection. (Numeric) N.B. Mainly intended for testing the addin programmatically. |
every_n |
Number of characters per split. If
N.B. Strings shorter than |
tolerance |
Tolerance. Number of characters. We may prefer not to split a string that's only a few
characters too long. Strings shorter than |
insert |
Whether to insert the wrapped text via
N.B. Mainly intended for testing the addin programmatically. |
After installing the package. Go to:
Tools >> Addins >> Browse Addins >> Keyboard Shortcuts
.
Find "Wrap String with paste0"
and press its
field under Shortcut
.
Press desired key command, e.g. Alt+P
.
Press Apply
.
Press Execute
.
Inserts the following (with newlines and correct indentation):
paste0("first n chars", "next n chars")
Returns NULL
invisibly.
Ludvig Renbo Olsen, [email protected]
Other addins:
assertCollectionAddin()
,
dputSelectedAddin()
,
initializeGXSFunctionAddin()
,
initializeTestthatAddin()
,
insertExpectationsAddin()
,
navigateTestFileAddin()