From 6eedc0f259f4f5795cac41dfef37c96d58ba12a9 Mon Sep 17 00:00:00 2001 From: selkamand Date: Sat, 9 Nov 2024 15:05:50 +1100 Subject: [PATCH 1/3] feat: added assert_length and friends Resolves issue #32 --- R/assert_length.R | 101 ++++++++++++++++++++++++++++ tests/testthat/test-assert_length.R | 88 ++++++++++++++++++++++++ 2 files changed, 189 insertions(+) create mode 100644 R/assert_length.R create mode 100644 tests/testthat/test-assert_length.R diff --git a/R/assert_length.R b/R/assert_length.R new file mode 100644 index 0000000..28d0783 --- /dev/null +++ b/R/assert_length.R @@ -0,0 +1,101 @@ +# Assert Length ----------------------------------------------------------- + +#' Assert Length +#' +#' Assert object has a specific length +#' +#' @param x object to check length of +#' @param length expected length (number) +#' @param msg custom error message +#' @param call (logical) whether to preserve call in error message +#' @param arg_name (character) name of argument being tested +#' +#' @return invisible(TRUE) +#' @export +assert_length <- assert_create( + func = function(x, length) { + if(!is.numeric(length)) return("'length' must be numeric") + if(length(length) != 1) return("'length' must be a single number") + if(!is.whole.number(length)) return("'length' must be a whole number") + if(length < 0) return("'length' must be non-negative") + + length(x) == length + }, + default_error_msg = "'{arg_name}' must have length {length}, not {length(arg_value)}" +) + +#' Assert Length Greater Than +#' +#' Assert object length is greater than a threshold +#' +#' @inheritParams assert_length +#' @return invisible(TRUE) +#' @export +assert_length_greater_than <- assert_create( + func = function(x, length) { + if(!is.numeric(length)) return("'length' must be numeric") + if(length(length) != 1) return("'length' must be a single number") + if(!is.whole.number(length)) return("'length' must be a whole number") + if(length < 0) return("'length' must be non-negative") + + length(x) > length + }, + default_error_msg = "'{arg_name}' must have length greater than {length}, not {length(arg_value)}" +) + +#' Assert Length Greater Than or Equal To +#' +#' Assert object length is greater than or equal to a threshold +#' +#' @inheritParams assert_length +#' @return invisible(TRUE) +#' @export +assert_length_greater_than_or_equal_to <- assert_create( + func = function(x, length) { + if(!is.numeric(length)) return("'length' must be numeric") + if(length(length) != 1) return("'length' must be a single number") + if(!is.whole.number(length)) return("'length' must be a whole number") + if(length < 0) return("'length' must be non-negative") + + length(x) >= length + }, + default_error_msg = "'{arg_name}' must have length greater than or equal to {length}, not {length(arg_value)}" +) + +#' Assert Length Less Than +#' +#' Assert object length is less than a threshold +#' +#' @inheritParams assert_length +#' @return invisible(TRUE) +#' @export +assert_length_less_than <- assert_create( + func = function(x, length) { + if(!is.numeric(length)) return("'length' must be numeric") + if(length(length) != 1) return("'length' must be a single number") + if(!is.whole.number(length)) return("'length' must be a whole number") + if(length < 0) return("'length' must be non-negative") + + length(x) < length + }, + default_error_msg = "'{arg_name}' must have length less than {length}, not {length(arg_value)}" +) + +#' Assert Length Less Than or Equal To +#' +#' Assert object length is less than or equal to a threshold +#' +#' @inheritParams assert_length +#' @return invisible(TRUE) +#' @export +assert_length_less_than_or_equal_to <- assert_create( + func = function(x, length) { + if(!is.numeric(length)) return("'length' must be numeric") + if(length(length) != 1) return("'length' must be a single number") + if(!is.whole.number(length)) return("'length' must be a whole number") + if(length < 0) return("'length' must be non-negative") + + length(x) <= length + }, + default_error_msg = "'{arg_name}' must have length less than or equal to {length}, not {length(arg_value)}" +) diff --git a/tests/testthat/test-assert_length.R b/tests/testthat/test-assert_length.R new file mode 100644 index 0000000..0099d05 --- /dev/null +++ b/tests/testthat/test-assert_length.R @@ -0,0 +1,88 @@ +cli::test_that_cli("assert_length() works", configs = "plain", { + # Works for objects with correct length + expect_true(assert_length(1:3, 3)) + expect_true(assert_length(c("a", "b"), 2)) + expect_true(assert_length(list(1, 2, 3), 3)) + expect_true(assert_length(NULL, 0)) + + # Aborts for objects with incorrect length + expect_error(assert_length(1:3, 2), "'1:3' must have length 2, not 3", fixed = TRUE) + expect_error(assert_length(c("a", "b"), 3), "'c(\"a\", \"b\")' must have length 3, not 2", fixed = TRUE) + + # Aborts for invalid length parameter + expect_error(assert_length(1:3, "3"), "'length' must be numeric", fixed = TRUE) + expect_error(assert_length(1:3, c(2, 3)), "'length' must be a single number", fixed = TRUE) + expect_error(assert_length(1:3, 2.5), "'length' must be a whole number", fixed = TRUE) + expect_error(assert_length(1:3, -1), "'length' must be non-negative", fixed = TRUE) + + # Error messages use variable name + x <- 1:3 + expect_error(assert_length(x, 2), "'x' must have length 2, not 3", fixed = TRUE) + + # Custom error messages work + expect_error(assert_length(1:3, 2, msg = "Custom error"), "Custom error", fixed = TRUE) +}) + +cli::test_that_cli("assert_length_greater_than() works", configs = "plain", { + # Works for objects with length > threshold + expect_true(assert_length_greater_than(1:3, 2)) + expect_true(assert_length_greater_than(c("a", "b", "c"), 1)) + + # Aborts for objects with length <= threshold + expect_error(assert_length_greater_than(1:3, 3), "'1:3' must have length greater than 3, not 3", fixed = TRUE) + expect_error(assert_length_greater_than(1:3, 4), "'1:3' must have length greater than 4, not 3", fixed = TRUE) + + # Aborts for invalid length parameter + expect_error(assert_length_greater_than(1:3, "3"), "'length' must be numeric", fixed = TRUE) + expect_error(assert_length_greater_than(1:3, c(2, 3)), "'length' must be a single number", fixed = TRUE) + expect_error(assert_length_greater_than(1:3, 2.5), "'length' must be a whole number", fixed = TRUE) + expect_error(assert_length_greater_than(1:3, -1), "'length' must be non-negative", fixed = TRUE) +}) + +cli::test_that_cli("assert_length_greater_than_or_equal_to() works", configs = "plain", { + # Works for objects with length >= threshold + expect_true(assert_length_greater_than_or_equal_to(1:3, 2)) + expect_true(assert_length_greater_than_or_equal_to(1:3, 3)) + + # Aborts for objects with length < threshold + expect_error(assert_length_greater_than_or_equal_to(1:3, 4), + "'1:3' must have length greater than or equal to 4, not 3", fixed = TRUE) + + # Aborts for invalid length parameter + expect_error(assert_length_greater_than_or_equal_to(1:3, "3"), "'length' must be numeric", fixed = TRUE) + expect_error(assert_length_greater_than_or_equal_to(1:3, c(2, 3)), "'length' must be a single number", fixed = TRUE) + expect_error(assert_length_greater_than_or_equal_to(1:3, 2.5), "'length' must be a whole number", fixed = TRUE) + expect_error(assert_length_greater_than_or_equal_to(1:3, -1), "'length' must be non-negative", fixed = TRUE) +}) + +cli::test_that_cli("assert_length_less_than() works", configs = "plain", { + # Works for objects with length < threshold + expect_true(assert_length_less_than(1:3, 4)) + expect_true(assert_length_less_than(c("a", "b"), 3)) + + # Aborts for objects with length >= threshold + expect_error(assert_length_less_than(1:3, 3), "'1:3' must have length less than 3, not 3", fixed = TRUE) + expect_error(assert_length_less_than(1:3, 2), "'1:3' must have length less than 2, not 3", fixed = TRUE) + + # Aborts for invalid length parameter + expect_error(assert_length_less_than(1:3, "3"), "'length' must be numeric", fixed = TRUE) + expect_error(assert_length_less_than(1:3, c(2, 3)), "'length' must be a single number", fixed = TRUE) + expect_error(assert_length_less_than(1:3, 2.5), "'length' must be a whole number", fixed = TRUE) + expect_error(assert_length_less_than(1:3, -1), "'length' must be non-negative", fixed = TRUE) +}) + +cli::test_that_cli("assert_length_less_than_or_equal_to() works", configs = "plain", { + # Works for objects with length <= threshold + expect_true(assert_length_less_than_or_equal_to(1:3, 4)) + expect_true(assert_length_less_than_or_equal_to(1:3, 3)) + + # Aborts for objects with length > threshold + expect_error(assert_length_less_than_or_equal_to(1:3, 2), + "'1:3' must have length less than or equal to 2, not 3", fixed = TRUE) + + # Aborts for invalid length parameter + expect_error(assert_length_less_than_or_equal_to(1:3, "3"), "'length' must be numeric", fixed = TRUE) + expect_error(assert_length_less_than_or_equal_to(1:3, c(2, 3)), "'length' must be a single number", fixed = TRUE) + expect_error(assert_length_less_than_or_equal_to(1:3, 2.5), "'length' must be a whole number", fixed = TRUE) + expect_error(assert_length_less_than_or_equal_to(1:3, -1), "'length' must be non-negative", fixed = TRUE) +}) \ No newline at end of file From 49f29e3e3636485c5629b95c9fd48a52ee055a72 Mon Sep 17 00:00:00 2001 From: selkamand Date: Sat, 9 Nov 2024 15:30:01 +1100 Subject: [PATCH 2/3] fix: whole number check in assert_length --- R/assert_length.R | 10 +++++----- tests/testthat/test-assert_length.R | 2 +- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/R/assert_length.R b/R/assert_length.R index 28d0783..dda9909 100644 --- a/R/assert_length.R +++ b/R/assert_length.R @@ -16,7 +16,7 @@ assert_length <- assert_create( func = function(x, length) { if(!is.numeric(length)) return("'length' must be numeric") if(length(length) != 1) return("'length' must be a single number") - if(!is.whole.number(length)) return("'length' must be a whole number") + if(!is_whole_number(length)) return("'length' must be a whole number") if(length < 0) return("'length' must be non-negative") length(x) == length @@ -35,7 +35,7 @@ assert_length_greater_than <- assert_create( func = function(x, length) { if(!is.numeric(length)) return("'length' must be numeric") if(length(length) != 1) return("'length' must be a single number") - if(!is.whole.number(length)) return("'length' must be a whole number") + if(!is_whole_number(length)) return("'length' must be a whole number") if(length < 0) return("'length' must be non-negative") length(x) > length @@ -54,7 +54,7 @@ assert_length_greater_than_or_equal_to <- assert_create( func = function(x, length) { if(!is.numeric(length)) return("'length' must be numeric") if(length(length) != 1) return("'length' must be a single number") - if(!is.whole.number(length)) return("'length' must be a whole number") + if(!is_whole_number(length)) return("'length' must be a whole number") if(length < 0) return("'length' must be non-negative") length(x) >= length @@ -73,7 +73,7 @@ assert_length_less_than <- assert_create( func = function(x, length) { if(!is.numeric(length)) return("'length' must be numeric") if(length(length) != 1) return("'length' must be a single number") - if(!is.whole.number(length)) return("'length' must be a whole number") + if(!is_whole_number(length)) return("'length' must be a whole number") if(length < 0) return("'length' must be non-negative") length(x) < length @@ -92,7 +92,7 @@ assert_length_less_than_or_equal_to <- assert_create( func = function(x, length) { if(!is.numeric(length)) return("'length' must be numeric") if(length(length) != 1) return("'length' must be a single number") - if(!is.whole.number(length)) return("'length' must be a whole number") + if(!is_whole_number(length)) return("'length' must be a whole number") if(length < 0) return("'length' must be non-negative") length(x) <= length diff --git a/tests/testthat/test-assert_length.R b/tests/testthat/test-assert_length.R index 0099d05..d12a435 100644 --- a/tests/testthat/test-assert_length.R +++ b/tests/testthat/test-assert_length.R @@ -85,4 +85,4 @@ cli::test_that_cli("assert_length_less_than_or_equal_to() works", configs = "pla expect_error(assert_length_less_than_or_equal_to(1:3, c(2, 3)), "'length' must be a single number", fixed = TRUE) expect_error(assert_length_less_than_or_equal_to(1:3, 2.5), "'length' must be a whole number", fixed = TRUE) expect_error(assert_length_less_than_or_equal_to(1:3, -1), "'length' must be non-negative", fixed = TRUE) -}) \ No newline at end of file +}) From d2c4321d4a2b4b63e5004af07f7e70ec05e9a067 Mon Sep 17 00:00:00 2001 From: selkamand Date: Sat, 9 Nov 2024 15:32:14 +1100 Subject: [PATCH 3/3] fix: added assert_length docs --- DESCRIPTION | 1 + NAMESPACE | 5 +++ man/assert_length.Rd | 31 +++++++++++++++++++ man/assert_length_greater_than.Rd | 31 +++++++++++++++++++ man/assert_length_greater_than_or_equal_to.Rd | 31 +++++++++++++++++++ man/assert_length_less_than.Rd | 31 +++++++++++++++++++ man/assert_length_less_than_or_equal_to.Rd | 31 +++++++++++++++++++ 7 files changed, 161 insertions(+) create mode 100644 man/assert_length.Rd create mode 100644 man/assert_length_greater_than.Rd create mode 100644 man/assert_length_greater_than_or_equal_to.Rd create mode 100644 man/assert_length_less_than.Rd create mode 100644 man/assert_length_less_than_or_equal_to.Rd diff --git a/DESCRIPTION b/DESCRIPTION index 95a8557..e35ab17 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -40,6 +40,7 @@ Collate: 'assert_functions.R' 'set_operations.R' 'assert_includes.R' + 'assert_length.R' 'assert_names.R' 'assert_numerical.R' 'assert_set.R' diff --git a/NAMESPACE b/NAMESPACE index e770db2..d70824d 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -29,6 +29,11 @@ export(assert_greater_than_or_equal_to) export(assert_identical) export(assert_includes) export(assert_int) +export(assert_length) +export(assert_length_greater_than) +export(assert_length_greater_than_or_equal_to) +export(assert_length_less_than) +export(assert_length_less_than_or_equal_to) export(assert_list) export(assert_logical) export(assert_logical_vector) diff --git a/man/assert_length.Rd b/man/assert_length.Rd new file mode 100644 index 0000000..2cfc00b --- /dev/null +++ b/man/assert_length.Rd @@ -0,0 +1,31 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/assert_length.R +\name{assert_length} +\alias{assert_length} +\title{Assert Length} +\usage{ +assert_length( + x, + length, + msg = NULL, + call = rlang::caller_env(), + arg_name = NULL +) +} +\arguments{ +\item{x}{object to check length of} + +\item{length}{expected length (number)} + +\item{msg}{custom error message} + +\item{call}{(logical) whether to preserve call in error message} + +\item{arg_name}{(character) name of argument being tested} +} +\value{ +invisible(TRUE) +} +\description{ +Assert object has a specific length +} diff --git a/man/assert_length_greater_than.Rd b/man/assert_length_greater_than.Rd new file mode 100644 index 0000000..a8f66ef --- /dev/null +++ b/man/assert_length_greater_than.Rd @@ -0,0 +1,31 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/assert_length.R +\name{assert_length_greater_than} +\alias{assert_length_greater_than} +\title{Assert Length Greater Than} +\usage{ +assert_length_greater_than( + x, + length, + msg = NULL, + call = rlang::caller_env(), + arg_name = NULL +) +} +\arguments{ +\item{x}{object to check length of} + +\item{length}{expected length (number)} + +\item{msg}{custom error message} + +\item{call}{(logical) whether to preserve call in error message} + +\item{arg_name}{(character) name of argument being tested} +} +\value{ +invisible(TRUE) +} +\description{ +Assert object length is greater than a threshold +} diff --git a/man/assert_length_greater_than_or_equal_to.Rd b/man/assert_length_greater_than_or_equal_to.Rd new file mode 100644 index 0000000..31a1560 --- /dev/null +++ b/man/assert_length_greater_than_or_equal_to.Rd @@ -0,0 +1,31 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/assert_length.R +\name{assert_length_greater_than_or_equal_to} +\alias{assert_length_greater_than_or_equal_to} +\title{Assert Length Greater Than or Equal To} +\usage{ +assert_length_greater_than_or_equal_to( + x, + length, + msg = NULL, + call = rlang::caller_env(), + arg_name = NULL +) +} +\arguments{ +\item{x}{object to check length of} + +\item{length}{expected length (number)} + +\item{msg}{custom error message} + +\item{call}{(logical) whether to preserve call in error message} + +\item{arg_name}{(character) name of argument being tested} +} +\value{ +invisible(TRUE) +} +\description{ +Assert object length is greater than or equal to a threshold +} diff --git a/man/assert_length_less_than.Rd b/man/assert_length_less_than.Rd new file mode 100644 index 0000000..af78840 --- /dev/null +++ b/man/assert_length_less_than.Rd @@ -0,0 +1,31 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/assert_length.R +\name{assert_length_less_than} +\alias{assert_length_less_than} +\title{Assert Length Less Than} +\usage{ +assert_length_less_than( + x, + length, + msg = NULL, + call = rlang::caller_env(), + arg_name = NULL +) +} +\arguments{ +\item{x}{object to check length of} + +\item{length}{expected length (number)} + +\item{msg}{custom error message} + +\item{call}{(logical) whether to preserve call in error message} + +\item{arg_name}{(character) name of argument being tested} +} +\value{ +invisible(TRUE) +} +\description{ +Assert object length is less than a threshold +} diff --git a/man/assert_length_less_than_or_equal_to.Rd b/man/assert_length_less_than_or_equal_to.Rd new file mode 100644 index 0000000..98d8ff0 --- /dev/null +++ b/man/assert_length_less_than_or_equal_to.Rd @@ -0,0 +1,31 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/assert_length.R +\name{assert_length_less_than_or_equal_to} +\alias{assert_length_less_than_or_equal_to} +\title{Assert Length Less Than or Equal To} +\usage{ +assert_length_less_than_or_equal_to( + x, + length, + msg = NULL, + call = rlang::caller_env(), + arg_name = NULL +) +} +\arguments{ +\item{x}{object to check length of} + +\item{length}{expected length (number)} + +\item{msg}{custom error message} + +\item{call}{(logical) whether to preserve call in error message} + +\item{arg_name}{(character) name of argument being tested} +} +\value{ +invisible(TRUE) +} +\description{ +Assert object length is less than or equal to a threshold +}