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

Don't inline classes in constructor if possible #481

Merged
merged 14 commits into from
Nov 4, 2024
Merged
Show file tree
Hide file tree
Changes from 13 commits
Commits
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
4 changes: 2 additions & 2 deletions R/aaa.R
Original file line number Diff line number Diff line change
Expand Up @@ -26,10 +26,10 @@ new_function <- function(args = NULL,
topNamespaceName <- function(env = parent.frame()) {
env <- topenv(env)
if (!isNamespace(env)) {
return()
return() # print visible
}

getNamespaceName(env)
as.character(getNamespaceName(env)) # unname
}

is_string <- function(x) {
Expand Down
2 changes: 1 addition & 1 deletion R/base.R
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@
name = quote(quote(x)),
call = quote(quote({})),

`function` = quote(function() {}),
`function` = quote(function() NULL),

Check warning on line 47 in R/base.R

View check run for this annotation

Codecov / codecov/patch

R/base.R#L47

Added line #L47 was not covered by tests
environment = quote(new.env(parent = emptyenv()))
)}

Expand Down
48 changes: 35 additions & 13 deletions R/class-spec.R
Original file line number Diff line number Diff line change
Expand Up @@ -81,27 +81,49 @@
}

class_construct <- function(.x, ...) {
eval(class_construct_expr(.x, ...))
class_constructor(.x)(...)
}


class_construct_expr <- function(.x, ...) {
class_construct_expr <- function(.x, envir = NULL, package = NULL) {
f <- class_constructor(.x)

# For S7 class constructors with a non-NULL @package property
# Instead of inlining the full class definition, use either
# `pkgname::classname()` or `classname()`
if (is_class(f) && !is.null(f@package)) {
# Check if the class can be resolved as a bare symbol without pkgname::
# Note: During package build, using pkg::class for a package's own symbols
# will raise an error from `::`.
if (identical(package, f@package)) {
t-kalinowski marked this conversation as resolved.
Show resolved Hide resolved
return(call(f@name))
} else {
# namespace the pkgname::classname() call
cl <- as.call(list(quote(`::`), as.name(f@package), as.name(f@name)))
t-kalinowski marked this conversation as resolved.
Show resolved Hide resolved

# check the call evaluates to f.
# This will error if package is not installed or object is not exported.
f2 <- eval(cl, baseenv())
if (!identical(f, f2)) {
msg <- sprintf(
"`%s::%s` is not identical to the class with the same @package and @name properties",
f@package, f@name
)
stop(msg, call. = FALSE)
}
return(as.call(list(cl)))
}
}

# If the constructor is a closure wrapping a simple expression, try
# to extract the expression
# (mostly for nicer printing and introspection.)

## early return if not safe to unwrap
# can't unwrap if we're passing on ...
if(...length()) {
return(as.call(list(f, ...)))
}

# can't unwrap if the closure is potentially important
# (this can probably be relaxed to allow additional environments)
fe <- environment(f)
if(!identical(fe, baseenv())) {
return(as.call(list(f, ...)))
if (!identical(fe, baseenv())) {
return(as.call(list(f)))
}

# special case for `class_missing`
Expand All @@ -111,8 +133,8 @@

# `new_object()` must be called from the class constructor, can't
# be safely unwrapped
if("new_object" %in% all.names(fb)) {
return(as.call(list(f, ...)))
if ("new_object" %in% all.names(fb)) {
return(as.call(list(f)))

Check warning on line 137 in R/class-spec.R

View check run for this annotation

Codecov / codecov/patch

R/class-spec.R#L137

Added line #L137 was not covered by tests
}

# maybe unwrap body if it is a single expression wrapped in `{`
Expand All @@ -133,7 +155,7 @@
}

#else, return a call to the constructor
as.call(list(f, ...))
as.call(list(f))

Check warning on line 158 in R/class-spec.R

View check run for this annotation

Codecov / codecov/patch

R/class-spec.R#L158

Added line #L158 was not covered by tests
}

class_constructor <- function(.x) {
Expand Down
15 changes: 7 additions & 8 deletions R/class.R
Original file line number Diff line number Diff line change
Expand Up @@ -16,14 +16,11 @@
#' * An S7 class, like [S7_object].
#' * An S3 class wrapped by [new_S3_class()].
#' * A base type, like [class_logical], [class_integer], etc.
#' @param package Package name. It is good practice to set the package
#' name when exporting an S7 class from a package because it prevents
#' clashes if two packages happen to export a class with the same
#' name.
#' @param package Package name. This is automatically resolved if the class is
#' defined in a package, and `NULL` otherwise.
#'
#' Setting `package` implies that the class is available for external use,
#' so should be accompanied by exporting the constructor. Learn more
#' in `vignette("packages")`.
#' Note, if the class is intended for external use, the constructor should be
#' exported. Learn more in `vignette("packages")`.
#' @param abstract Is this an abstract class? An abstract class can not be
#' instantiated.
#' @param constructor The constructor function. In most cases, you can rely
Expand Down Expand Up @@ -134,7 +131,9 @@ new_class <- function(
all_props[names(new_props)] <- new_props

if (is.null(constructor)) {
constructor <- new_constructor(parent, all_props)
constructor <- new_constructor(parent, all_props,
envir = parent.frame(),
package = package)
}

object <- constructor
Expand Down
47 changes: 38 additions & 9 deletions R/constructor.R
Original file line number Diff line number Diff line change
@@ -1,18 +1,26 @@
new_constructor <- function(parent, properties) {
new_constructor <- function(parent, properties,
envir = asNamespace("S7"), package = NULL) {
properties <- as_properties(properties)
arg_info <- constructor_args(parent, properties)
arg_info <- constructor_args(parent, properties, envir, package)
self_args <- as_names(names(arg_info$self), named = TRUE)

if (identical(parent, S7_object) || (is_class(parent) && parent@abstract)) {
new_object_call <-
if (has_S7_symbols(envir, "new_object", "S7_object")) {
bquote(new_object(S7_object(), ..(self_args)), splice = TRUE)
} else {
bquote(S7::new_object(S7::S7_object(), ..(self_args)), splice = TRUE)
}

return(new_function(
args = arg_info$self,
body = as.call(c(quote(`{`),
# Force all promises here so that any errors are signaled from
# the constructor() call instead of the new_object() call.
unname(self_args),
new_call("new_object", c(list(quote(S7_object())), self_args))
new_object_call
)),
env = asNamespace("S7")
env = envir
))
}

Expand Down Expand Up @@ -42,15 +50,19 @@ new_constructor <- function(parent, properties) {
parent_args <- as_names(names(arg_info$parent), named = TRUE)
names(parent_args)[names(parent_args) == "..."] <- ""
parent_call <- new_call(parent_name, parent_args)
body <- new_call("new_object", c(parent_call, self_args))
body <- new_call(
if (has_S7_symbols(envir, "new_object")) "new_object" else c("S7", "new_object"),
c(parent_call, self_args)
)

env <- new.env(parent = asNamespace("S7"))
env <- new.env(parent = envir)
env[[parent_name]] <- parent_fun

new_function(args, body, env)
}

constructor_args <- function(parent, properties = list()) {
constructor_args <- function(parent, properties = list(),
envir = asNamespace("S7"), package = NULL) {
parent_args <- formals(class_constructor(parent))

# Remove read-only properties
Expand All @@ -66,7 +78,7 @@ constructor_args <- function(parent, properties = list()) {

self_args <- as.pairlist(lapply(
setNames(, self_arg_nms),
function(name) prop_default(properties[[name]]))
function(name) prop_default(properties[[name]], envir, package))
)

list(parent = parent_args,
Expand All @@ -81,8 +93,14 @@ is_property_dynamic <- function(x) is.function(x$getter)
missing_args <- function(names) {
lapply(setNames(, names), function(i) quote(class_missing))
}

new_call <- function(call, args) {
as.call(c(list(as.name(call)), args))
if (is.character(call)) {
call <- switch(length(call),
as.name(call),
as.call(c(quote(`::`), lapply(call, as.name))))
}
as.call(c(list(call), args))
}

as_names <- function(x, named = FALSE) {
Expand All @@ -91,3 +109,14 @@ as_names <- function(x, named = FALSE) {
}
lapply(x, as.name)
}

has_S7_symbols <- function(env, ...) {
env <- topenv(env)
hadley marked this conversation as resolved.
Show resolved Hide resolved
if (identical(env, asNamespace("S7")))
return (TRUE)
if (!isNamespace(env))
return (FALSE)
imports <- getNamespaceImports(env)[["S7"]]
symbols <- c(...) %||% getNamespaceExports("S7")
all(symbols %in% imports)
hadley marked this conversation as resolved.
Show resolved Hide resolved
}
4 changes: 2 additions & 2 deletions R/property.R
Original file line number Diff line number Diff line change
Expand Up @@ -155,8 +155,8 @@ str.S7_property <- function(object, ..., nest.lev = 0) {
print(object, ..., nest.lev = nest.lev)
}

prop_default <- function(prop) {
prop$default %||% class_construct_expr(prop$class)
prop_default <- function(prop, envir, package) {
prop$default %||% class_construct_expr(prop$class, envir, package)
}

#' Get/set a property
Expand Down
13 changes: 5 additions & 8 deletions man/new_class.Rd

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

2 changes: 1 addition & 1 deletion tests/testthat/_snaps/class.md
Original file line number Diff line number Diff line change
Expand Up @@ -112,7 +112,7 @@
foo <- new_class("foo", abstract = TRUE)
foo()
Condition
Error in `new_object()`:
Error in `S7::new_object()`:
! Can't construct an object from abstract class <foo>

# abstract classes: can't inherit from concrete class
Expand Down
38 changes: 38 additions & 0 deletions tests/testthat/_snaps/external-generic.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,3 +5,41 @@
Output
<S7_external_generic> foo::bar(x)

# new_method works with both hard and soft dependencies

Code
args(Foo)
Output
function (bar = t0::AnS7Class())
NULL
Code
args(t2::AnS7Class2)
Output
function (bar = t0::AnS7Class())
t-kalinowski marked this conversation as resolved.
Show resolved Hide resolved
NULL
Code
args(t2:::AnInternalClass)
Output
function (foo = t0::AnS7Class(), bar = AnS7Class2())
NULL

---

Code
new_class("Foo", properties = list(bar = new_class("MadeUpClass", package = "t0")))
Condition
Error:
! 'MadeUpClass' is not an exported object from 'namespace:t0'
Code
new_class("Foo", properties = list(bar = new_class("MadeUpClass", package = "MadeUpPackage")))
Condition
Error in `loadNamespace()`:
! there is no package called 'MadeUpPackage'
Code
modified_class <- t0::AnS7Class
attr(modified_class, "xyz") <- "abc"
new_class("Foo", properties = list(bar = modified_class))
Condition
Error:
! `t0::AnS7Class` is not identical to the class with the same @package and @name properties

1 change: 1 addition & 0 deletions tests/testthat/t0/NAMESPACE
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
# Generated by roxygen2: do not edit by hand

export(AnS7Class)
export(an_s3_generic)
export(an_s7_generic)
3 changes: 3 additions & 0 deletions tests/testthat/t0/R/t0.R
Original file line number Diff line number Diff line change
Expand Up @@ -3,3 +3,6 @@ an_s7_generic <- S7::new_generic("an_s7_generic", "x")

#' @export
an_s3_generic <- function(x) UseMethod("an_s3_generic")

#' @export
AnS7Class <- S7::new_class("AnS7Class")
2 changes: 2 additions & 0 deletions tests/testthat/t2/NAMESPACE
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
# Generated by roxygen2: do not edit by hand

export(AnS7Class2)
export(an_s7_class)
importFrom(t0,AnS7Class)
importFrom(t0,an_s3_generic)
importFrom(t0,an_s7_generic)
11 changes: 11 additions & 0 deletions tests/testthat/t2/R/t2.R
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,24 @@ S7::method(an_s7_generic, an_s7_class) <- function(x) "foo"
S7::method(an_s3_generic, an_s7_class) <- function(x) "foo"


#' @importFrom t0 AnS7Class
#' @export
AnS7Class2 <- S7::new_class("AnS7Class2", properties = list(bar = AnS7Class))

AnInternalClass <- S7::new_class("AnInternalClass", properties = list(
foo = AnS7Class,
bar = AnS7Class2
))


another_s7_generic <- S7::new_external_generic("t1", "another_s7_generic", "x")
S7::method(another_s7_generic, S7::class_character) <- function(x) "foo"
S7::method(another_s7_generic, an_s7_class) <- function(x) "foo"

another_s3_generic <- S7::new_external_generic("t1", "another_s3_generic", "x")
S7::method(another_s3_generic, an_s7_class) <- function(x) "foo"


.onLoad <- function(libname, pkgname) {
S7::methods_register()
}
13 changes: 8 additions & 5 deletions tests/testthat/test-class.R
Original file line number Diff line number Diff line change
Expand Up @@ -232,16 +232,19 @@ test_that("c(<S7_class>, ...) gives error", {
})

test_that("can round trip to disk and back", {
foo1 <- new_class("foo1", properties = list(y = class_integer))
foo2 <- new_class("foo2", properties = list(x = foo1))

f <- foo2(x = foo1(y = 1L))
eval(quote({
foo1 <- new_class("foo1", properties = list(y = class_integer))
foo2 <- new_class("foo2", properties = list(x = foo1))
f <- foo2(x = foo1(y = 1L))
}), globalenv())

f <- globalenv()[["f"]]
path <- tempfile()
saveRDS(f, path)
f2 <- readRDS(path)

expect_equal(f2, f)
expect_equal(f, f2)
rm(foo1, foo2, f, envir = globalenv())
})


Expand Down
2 changes: 0 additions & 2 deletions tests/testthat/test-constructor.R
Original file line number Diff line number Diff line change
Expand Up @@ -160,8 +160,6 @@ test_that("can create constructors with missing or lazy defaults", {
"Can\'t set read-only property Person@birthdate")
})



test_that("Dynamic settable properties are included in constructor", {
Foo <- new_class(
name = "Foo", package = NULL,
Expand Down
Loading
Loading