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

unnormalize() with grouped data #415

Merged
merged 30 commits into from
Sep 12, 2023
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
7ef5efd
working example
etiennebacher May 3, 2023
585dd89
fix small bug
etiennebacher May 3, 2023
c9a08a3
some comments
etiennebacher May 3, 2023
da1738e
Merge branch 'main' into unnormalize-grouped-data
etiennebacher May 4, 2023
ee4c1f1
fix order of attributes
etiennebacher May 4, 2023
b274484
refactor unnormalize for groups
etiennebacher May 4, 2023
dfa44b5
test error message
etiennebacher May 4, 2023
8b704a0
fix test
etiennebacher May 4, 2023
cb3a0d5
fix args of unnormalize
etiennebacher May 4, 2023
1cff323
clean some lints
etiennebacher May 5, 2023
b1d08e9
start same for `unstandardize()` [skip ci]
etiennebacher May 15, 2023
9a73be1
Merge branch 'main' into unnormalize-grouped-data
strengejacke May 30, 2023
9cae605
typo
etiennebacher May 31, 2023
81865f0
styler
etiennebacher May 31, 2023
378ee57
Merge branch 'main' into unnormalize-grouped-data
strengejacke Jun 9, 2023
5015d7c
Merge branch 'main' into unnormalize-grouped-data
strengejacke Jun 14, 2023
835d4b5
Merge branch 'main' into unnormalize-grouped-data
etiennebacher Aug 21, 2023
9840bbe
fix unstandardize() for groups
etiennebacher Aug 21, 2023
ab31810
remove former test
etiennebacher Aug 21, 2023
2cb4e64
Merge branch 'unnormalize-grouped-data' of https://github.com/easysta…
etiennebacher Aug 21, 2023
86478c2
ensure that both functions return a grouped dataframe
etiennebacher Aug 21, 2023
e98a562
lintr, styler
etiennebacher Aug 21, 2023
ff912ee
Merge branch 'main' into unnormalize-grouped-data
etiennebacher Aug 22, 2023
5fd51c8
Merge branch 'main' into unnormalize-grouped-data
etiennebacher Aug 30, 2023
957dd35
lintr, styler, minor doc
etiennebacher Aug 30, 2023
04d08b3
Merge branch 'main' into unnormalize-grouped-data
etiennebacher Sep 10, 2023
6e39256
add some extra checks
etiennebacher Sep 11, 2023
cdfb455
fix
etiennebacher Sep 11, 2023
db9f30a
unnormalize: from warning to error
etiennebacher Sep 11, 2023
6b7a42a
bump news
etiennebacher Sep 11, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions NAMESPACE
Original file line number Diff line number Diff line change
Expand Up @@ -175,6 +175,7 @@ S3method(to_numeric,logical)
S3method(to_numeric,numeric)
S3method(unnormalize,data.frame)
S3method(unnormalize,default)
S3method(unnormalize,grouped_df)
S3method(unnormalize,numeric)
S3method(unstandardize,array)
S3method(unstandardize,character)
Expand Down
24 changes: 21 additions & 3 deletions R/normalize.R
Original file line number Diff line number Diff line change
Expand Up @@ -214,9 +214,15 @@ normalize.grouped_df <- function(x,
}

x <- as.data.frame(x)
for (rows in grps) {
x[rows, ] <- normalize(
x[rows, , drop = FALSE],

# create column(s) to store dw_transformer attributes
for (i in select) {
info$groups[[paste0("attr_", i)]] <- rep(NA, length(grps))
}

for (rows in seq_along(grps)) {
tmp <- normalize(
x[grps[[rows]], , drop = FALSE],
select = select,
exclude = exclude,
include_bounds = include_bounds,
Expand All @@ -225,9 +231,21 @@ normalize.grouped_df <- function(x,
add_transform_class = FALSE,
...
)

# store dw_transformer_attributes
for (i in select) {
info$groups[rows, paste0("attr_", i)][[1]] <- list(unlist(attributes(tmp[[i]])))
}

x[grps[[rows]], ] <- tmp
}

# last column of "groups" attributes must be called ".rows"
info$groups <- data_relocate(info$groups, ".rows", after = -1)

# set back class, so data frame still works with dplyr
attributes(x) <- utils::modifyList(info, attributes(x))
class(x) <- c("grouped_df", class(x))
x
}

Expand Down
22 changes: 19 additions & 3 deletions R/standardize.R
Original file line number Diff line number Diff line change
Expand Up @@ -341,9 +341,14 @@ standardize.grouped_df <- function(x,
reference, weights, keep_factors = force
)

for (rows in args$grps) {
args$x[rows, ] <- standardize(
args$x[rows, , drop = FALSE],
# create column(s) to store dw_transformer attributes
for (i in select) {
args$info$groups[[paste0("attr_", i)]] <- rep(NA, length(args$grps))
}

for (rows in seq_along(args$grps)) {
tmp <- standardize(
args$x[args$grps[[rows]], , drop = FALSE],
select = args$select,
exclude = NULL,
robust = robust,
Expand All @@ -358,7 +363,18 @@ standardize.grouped_df <- function(x,
add_transform_class = FALSE,
...
)

# store dw_transformer_attributes
for (i in select) {
args$info$groups[rows, paste0("attr_", i)][[1]] <- list(unlist(attributes(tmp[[i]])))
}

args$x[args$grps[[rows]], ] <- tmp
}

# last column of "groups" attributes must be called ".rows"
args$info$groups <- data_relocate(args$info$groups, ".rows", after = -1)

# set back class, so data frame still works with dplyr
attributes(args$x) <- args$info
args$x
Expand Down
108 changes: 102 additions & 6 deletions R/unnormalize.R
Original file line number Diff line number Diff line change
Expand Up @@ -16,11 +16,33 @@ unnormalize.default <- function(x, ...) {
#' @rdname normalize
#' @export
unnormalize.numeric <- function(x, verbose = TRUE, ...) {
## TODO implement algorithm include_bounds = FALSE
include_bounds <- attr(x, "include_bounds")
min_value <- attr(x, "min_value")
range_difference <- attr(x, "range_difference")
to_range <- attr(x, "to_range")

# if function called from the "grouped_df" method, we use the dw_transformer
# attributes that were recovered in the "grouped_df" method

dots <- match.call(expand.dots = FALSE)[["..."]]
grp_attr_dw <- eval(dots$grp_attr_dw, envir = parent.frame(1L))

if (!is.null(grp_attr_dw)) {

names(grp_attr_dw) <- gsub(".*\\.", "", names(grp_attr_dw))
etiennebacher marked this conversation as resolved.
Show resolved Hide resolved

include_bounds <- unname(grp_attr_dw["include_bounds"])
min_value <- unname(grp_attr_dw["min_value"])
range_difference <- unname(grp_attr_dw["range_difference"])
to_range <- unname(grp_attr_dw["to_range"])

if (is.na(to_range)) to_range <- NULL

} else {

## TODO implement algorithm include_bounds = FALSE
include_bounds <- attr(x, "include_bounds")
min_value <- attr(x, "min_value")
range_difference <- attr(x, "range_difference")
to_range <- attr(x, "to_range")

}

if (is.null(min_value) || is.null(range_difference)) {
if (verbose) {
Expand Down Expand Up @@ -54,7 +76,81 @@ unnormalize.data.frame <- function(x,
regex = regex,
verbose = verbose
)
x[select] <- lapply(x[select], unnormalize, verbose = verbose)

# if function called from the "grouped_df" method, we use the dw_transformer
# attributes that were recovered in the "grouped_df" method

dots <- match.call(expand.dots = FALSE)[["..."]]

if (!is.null(dots$grp_attr_dw)) {
grp_attr_dw <- eval(dots$grp_attr_dw, envir = parent.frame(1L))
} else {
grp_attr_dw <- NULL
}

for (i in select) {
var_attr <- which(grepl(paste0("^attr\\_", i, "\\."), names(grp_attr_dw)))
attrs <- grp_attr_dw[var_attr]
x[[i]] <- unnormalize(x[[i]], verbose = verbose, grp_attr_dw = attrs)
}

x
}

#' @rdname normalize
#' @export
unnormalize.grouped_df <- function(x,
select = NULL,
exclude = NULL,
ignore_case = FALSE,
regex = FALSE,
verbose = TRUE,
...) {

# evaluate select/exclude, may be select-helpers
select <- .select_nse(select,
x,
exclude,
ignore_case,
regex = regex,
remove_group_var = TRUE,
verbose = verbose
)

info <- attributes(x)
# works only for dplyr >= 0.8.0
grps <- attr(x, "groups", exact = TRUE)[[".rows"]]

x <- as.data.frame(x)

for (i in select) {
if (is.null(info$groups[[paste0("attr_", i)]])) {
insight::format_error(
paste("Couldn't retrieve the necessary information to unnormalize",
text_concatenate(i, enclose = "`"))
)
}
}
for (rows in seq_along(grps)) {

# get the dw_transformer attributes for this group
raw_attrs <- unlist(info$groups[rows, startsWith(names(info$groups), "attr")])
if (length(select) == 1L) {
names(raw_attrs) <- paste0("attr_", select, ".", names(raw_attrs))
}

tmp <- unnormalize(
x[grps[[rows]], , drop = FALSE],
select = select,
exclude = exclude,
ignore_case = ignore_case,
regex = regex,
verbose = verbose,
grp_attr_dw = raw_attrs
)
x[grps[[rows]], ] <- tmp
}
# set back class, so data frame still works with dplyr
attributes(x) <- utils::modifyList(info, attributes(x))
x
}
63 changes: 62 additions & 1 deletion R/unstandardize.R
Original file line number Diff line number Diff line change
Expand Up @@ -143,8 +143,69 @@ unstandardize.grouped_df <- function(x,
reference = NULL,
robust = FALSE,
two_sd = FALSE,
select = NULL,
exclude = NULL,
ignore_case = FALSE,
regex = FALSE,
verbose = TRUE,
...) {
insight::format_error("Cannot (yet) unstandardize a `grouped_df`.")

# evaluate select/exclude, may be select-helpers
select <- .select_nse(select,
x,
exclude,
ignore_case,
regex = regex,
remove_group_var = TRUE,
verbose = verbose
)

info <- attributes(x)

# works only for dplyr >= 0.8.0
grps <- attr(x, "groups", exact = TRUE)[[".rows"]]

x <- as.data.frame(x)

for (i in select) {
if (is.null(info$groups[[paste0("attr_", i)]])) {
insight::format_error(
paste("Couldn't retrieve the necessary information to unnormalize",
text_concatenate(i, enclose = "`"))
etiennebacher marked this conversation as resolved.
Show resolved Hide resolved
)
}
}

for (rows in seq_along(grps)) {

# get the dw_transformer attributes for this group
raw_attrs <- unlist(info$groups[rows, startsWith(names(info$groups), "attr")])
if (length(select) == 1L) {
names(raw_attrs) <- paste0("attr_", select, ".", names(raw_attrs))
}

tmp <- unstandardise(
x[grps[[rows]], , drop = FALSE],
center = center,
scale = scale,
reference = reference,
robust = robust,
two_sd = two_sd,
select = select,
exclude = exclude,
ignore_case = ignore_case,
regex = regex,
verbose = verbose,
grp_attr_dw = raw_attrs
)
x[grps[[rows]], ] <- tmp
}
# set back class, so data frame still works with dplyr
attributes(x) <- utils::modifyList(info, attributes(x))
x



}

#' @export
Expand Down
11 changes: 11 additions & 0 deletions man/normalize.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

47 changes: 46 additions & 1 deletion tests/testthat/test-unnormalize.R
Original file line number Diff line number Diff line change
Expand Up @@ -54,8 +54,53 @@ test_that("unnormalize and unstandardized x 4", {
# select helpers ------------------------------
test_that("unnormalize regex", {
x <- normalize(mtcars, select = "mpg")
expect_equal(
expect_identical(
unnormalize(x, select = "pg", regex = TRUE),
unnormalize(x, select = "mpg")
)
})


test_that("unnormalize: grouped data", {
skip_if_not_installed("poorman")

# 1 group, 1 normalized var
norm <- poorman::group_by(iris, Species)
norm <- normalize(norm, "Sepal.Length")
expect_identical(
poorman::ungroup(unnormalize(norm, "Sepal.Length")),
iris
)

# 2 groups, 1 normalized var
set.seed(123)
test <- iris
test$grp <- sample(c("A", "B"), nrow(test), replace = TRUE)
norm <- poorman::group_by(test, Species, grp)
norm <- normalize(norm, "Sepal.Length")
expect_identical(
poorman::ungroup(unnormalize(norm, "Sepal.Length")),
test
)

# 2 groups, 2 normalized vars
set.seed(123)
test <- iris
test$grp <- sample(c("A", "B"), nrow(test), replace = TRUE)
norm <- poorman::group_by(test, Species, grp)
norm <- normalize(norm, c("Sepal.Length", "Petal.Length"))
expect_identical(
poorman::ungroup(unnormalize(norm, c("Sepal.Length", "Petal.Length"))),
test
)

# can't recover attributes
norm <- poorman::group_by(iris, Species)
norm <- normalize(norm, "Sepal.Length")
attr(norm, "groups") <- NULL

expect_error(
unnormalize(norm, "Sepal.Length"),
regexp = "Couldn't retrieve the necessary information"
)
})