531 lines
18 KiB
R
Raw Normal View History

2025-01-12 00:52:51 +08:00
## simple call, only field names
fg <- setRefClass("foo", c("bar", "flag"))
f0 <- new("foo") # deprecated, but should still work
f1 <- fg(flag = "testing")
f1$bar <- 1
stopifnot(identical(f1$bar, 1))
## add method
fg$methods(showAll = function() c(bar, flag))
stopifnot(all.equal(f1$showAll(), c(1, "testing")))
str(f1)
fg <- setRefClass("foo", list(bar = "numeric", flag = "character",
tag = "ANY"),
methods = list(addToBar = function(incr) {
b <- bar + incr
bar <<- b
b
} )
)
fg$lock("flag")
stopifnot(identical(fg$lock(), "flag"))
ff <- new("foo", bar = 1.5)
stopifnot(identical(ff$bar, 1.5))
ff$bar <- pi
stopifnot(identical(ff$bar, pi))
## flag has not yet been set
ff$flag <- "flag test"
stopifnot(identical(ff$flag, "flag test"))
## but no second assign
stopifnot(is(tryCatch(ff$flag <- "new", error = function(e)e), "error"))
## test against generator
f2 <- fg(bar = pi, flag = "flag test")
## identical does not return TRUE if *contents* of env are identical
stopifnot(identical(ff$bar, f2$bar), identical(ff$flag, f2$flag))
## but flag was now assigned once
stopifnot(is(tryCatch(f2$flag <- "new", error = function(e)e), "error"))
str(f2)
## add some accessor methods
fg$accessors("bar")
ff$setBar(1:3)
stopifnot(identical(ff$getBar(), 1:3))
ff$getBar()
stopifnot(all.equal(ff$addToBar(1), 2:4))
## Add a method
fg$methods(barTimes = function(x) {
"This method multiples field bar by argument x
and this string is self-documentation"
setBar(getBar() * x)})
ffbar <- ff$getBar()
ff$barTimes(10)
stopifnot(all.equal(ffbar * 10, ff$getBar()))
ff$barTimes(.1)
## inheritance. redefines flag so should fail:
stopifnot(is(tryCatch(setRefClass("foo2", list(b2 = "numeric",
flag = "complex"),
contains = "foo",
refMethods = list(addBoth = function(incr) {
addToBar(incr) #uses inherited class method
setB2(getB2() + incr)
})),
error = function(e)e), "error"))
## but with flag as a subclass of "characters", should work
## Also subclasses "tag" which had class "ANY before
setClass("ratedChar", contains = "character",
representation(score = "numeric"))
foo2 <- setRefClass("foo2", list(b2 = "numeric", flag = "ratedChar",
tag = "numeric"),
contains = "foo",
methods = list(addBoth = function(incr) {
addToBar(incr) #uses inherited class method
b2 <<- b2 + incr
}))
## now lock the flag field; should still allow one write
foo2$lock("flag")
f2 <- foo2(bar = -3, flag = as("ANY", "ratedChar"),
b2 = ff$bar, tag = 1.5)
## but not a second one
stopifnot(is(tryCatch(f2$flag <- "Try again",
error = function(e)e), "error"))
str(f2)
f22 <- foo2(bar = f2$bar)
## same story if assignment follows the initialization
f22$flag <- f2$flag
stopifnot(is(tryCatch(f22$flag <- "Try again",
error = function(e)e), "error"))
## Exporting superclass object
f22 <- fg(bar = f2$bar, flag = f2$flag)
f2e <- f2$export("foo")
stopifnot(identical(f2e$bar, f22$bar), identical(f2e$flag, f22$flag),
identical(class(f2e), class(f22)))
stopifnot(identical(f2$flag, as("ANY", "ratedChar")),
identical(f2$bar, -3),
all.equal(f2$b2, 2:4+0))
f2$addBoth(-1)
stopifnot(all.equal(f2$bar, -4), all.equal(f2$b2, 1:3+0))
## test callSuper()
foo3 <- setRefClass("foo3", fields = list(flag2 = "ratedChar"),
contains = "foo2",
methods = list(addBoth = function(incr) {
callSuper(incr)
flag2 <<- as(paste(flag, paste(incr, collapse = ", "),
sep = "; "),
"ratedChar")
incr
}))
f2 <- foo2(bar = -3, flag = as("ANY", "ratedChar"), b2 = 1:3)
f3 <- foo3()
f3$import(f2)
stopifnot(all.equal(f3$b2, f2$b2), all.equal(f3$bar, f2$bar),
all.equal(f3$flag, f2$flag))
f3$addBoth(1)
stopifnot(all.equal(f3$bar, -2), all.equal(f3$b2, 2:4+0),
all.equal(f3$flag2, as("ANY; 1", "ratedChar")))
## but the import should have used up the one write for $flag
stopifnot(is(tryCatch(f3$flag <- "Try again",
error = function(e)e), "error"))
str(f3)
## importing the same class (not very useful but documented to work)
f3 <- foo3()
f4 <- foo3(bar = -3, flag = as("More", "ratedChar"), b2 = 1:3, flag2 = f2$flag)
f3$import(f4)
stopifnot(identical(f3$bar, f4$bar),
identical(f3$flag, f4$flag),
identical(f3$b2, f4$b2),
identical(f3$flag2, f4$flag2))
## similar to $import() but using superclass object in the generator call
## The explicitly supplied flag= should override and be allowed
## by the default $initialize()
f3b <- foo3(f2, flag = as("Other", "ratedChar"),
flag2 = as("More", "ratedChar"))
## check that inherited and direct field assignments worked
stopifnot(identical(f3b$tag, f2$tag),
identical(f3b$flag, as("Other", "ratedChar")),
identical(f3b$flag2, as("More", "ratedChar")))
## the $new() method should match the generator function
f3b <- foo3$new(f2, flag = as("Other", "ratedChar"),
flag2 = as("More", "ratedChar"))
stopifnot(identical(f3b$tag, f2$tag),
identical(f3b$flag, as("Other", "ratedChar")),
identical(f3b$flag2, as("More", "ratedChar")))
## a class with an initialize method, and an extra slot (legal, not a good idea)
setOldClass(c("simple.list", "list"))
fg4 <- setRefClass("foo4",
contains = "foo2",
methods = list(
initialize = function(...) {
.self$initFields(...)
.self@made <<- R.version
.self
}),
representation = list(made = "simple.list")
)
f4 <- new("foo4", flag = as("another test", "ratedChar"), bar = 1:3)
stopifnot(identical(f4@made, R.version))
## a trivial class with no fields, using fields = list(), failed up to rev 56035
foo5 <- setRefClass("foo5", fields = list(),
methods = list(bar = function(test)
paste("*",test,"*")))
f5 <- foo5()
stopifnot(identical( f5$bar("xxx"), paste("*","xxx", "*")))
## simple active binding test
abGen <- setRefClass("ab",
fields = list(a = "ANY",
b = function(x) if(missing(x)) a else {a <<- x; x}))
ab1 <- abGen(a = 1)
stopifnot(identical(ab1$a, 1), identical(ab1$b, 1))
ab1$b <- 2
stopifnot(identical(ab1$a, 2), identical(ab1$b, 2))
## a simple editor for matrix objects. Method $edit() changes some
## range of values; method $undo() undoes the last edit.
mEditor <- setRefClass("matrixEditor",
fields = list(data = "matrix",
edits = "list"),
methods = list(
edit = function(i, j, value) {
## the following string documents the edit method
'Replaces the range [i, j] of the
object by value.
'
backup <-
list(i, j, data[i,j])
data[i,j] <<- value
edits <<- c(list(backup),
edits)
invisible(value)
},
undo = function() {
'Undoes the last edit() operation
and update the edits field accordingly.
'
prev <- edits
if(length(prev)) prev <- prev[[1]]
else stop("No more edits to undo")
edit(prev[[1]], prev[[2]], prev[[3]])
## trim the edits list
length(edits) <<- length(edits) - 2
invisible(prev)
}
))
xMat <- matrix(1:12,4,3)
xx <- mEditor(data = xMat)
xx$edit(2, 2, 0)
xx$data
xx$undo()
mEditor$help("undo")
stopifnot(all.equal(xx$data, xMat))
## add a method to save the object
mEditor$methods(
save = function(file) {
'Save the current object on the file
in R external object format.
'
base::save(.self, file = file)
},
counter = function(i) {
'The number of items in the i-th edit.
(Used to test usingMethods())
'
if(i > 0 && i <= length(edits))
length(edits[[i]][[3]])
else
0L
}
)
tf <- tempfile()
xx$save(tf) #$
load(tf)
unlink(tf)
stopifnot(identical(xx$data, .self$data))
## tests of $trace() methods
## debugging an object
xx$trace(edit, quote(xxTrace <<- TRUE))
## debugging all objects from class mEditor in method $undo()
mEditor$trace(undo, quote(mETrace <<- TRUE))
xxTrace <- mETrace <- FALSE
xx$edit(2,3,100)
xx$undo()
## will not have changed the xx$undo() method (already used)
stopifnot(identical(xxTrace, TRUE), identical(mETrace, FALSE))
## but a new object works the other way around
xxTrace <- mETrace <- FALSE
xx <- mEditor(data = xMat)
xx$edit(2,3,100)
xx$undo()
stopifnot(identical(xxTrace, FALSE), identical(mETrace, TRUE))
markViewer <- ""
setMarkViewer <- function(what)
markViewer <<- what
## Inheriting a reference class: a matrix viewer
mv <- setRefClass("matrixViewer",
fields = c("viewerDevice", "viewerFile"),
contains = "matrixEditor",
methods = list( view = function() {
dd <- dev.cur(); dev.set(viewerDevice)
devAskNewPage(FALSE)
matplot(data, main = paste("After",length(edits),"edits"))
dev.set(dd)},
edit = # invoke previous method, then replot
function(i, j, value) {
callSuper(i, j, value)
view()
}))
## initialize and finalize methods
mv$methods( initialize = function(file = "./matrixView.pdf", ...) {
viewerFile <<- file
pdf(viewerFile)
viewerDevice <<- dev.cur()
message("Plotting to ", viewerFile)
dev.set(dev.prev())
setMarkViewer("ON")
initFields(...)
},
finalize = function() {
dev.off(viewerDevice)
setMarkViewer("OFF")
})
## a counts method to test usingMethods()
mv$methods( counts = function() {
usingMethods("counter")
sapply(seq_along(edits), "counter")
})
ff <- mv( data = xMat)
stopifnot(identical(markViewer, "ON")) # check initialize
ff$edit(2,2,0)
ff$data
if(methods:::.hasCodeTools()) # otherwise 'counter' is not visible
stopifnot(identical(ff$counts(), length(ff$edits[[1]][[3]])))
ff$undo()
stopifnot(all.equal(ff$data, xMat))
rm(ff)
gc()
stopifnot(identical(markViewer, "OFF")) #check finalize
## tests of copying
viewerPlus <- setRefClass("viewerPlus",
fields = list( text = "character",
viewer = "matrixViewer"))
ff <- mv( data = xMat)
v1 <- viewerPlus(text = letters, viewer = ff)
v2 <- v1$copy()
v3 <- v1$copy(TRUE)
v2$text <- "Hello, world"
v2$viewer$data <- t(xMat) # change a field in v2$viewer
v3$text <- LETTERS
v3$viewer <- mv( data = matrix(nrow=1,ncol=1))
## with a deep copy all is protected, with a shallow copy
## the environment of a copied field remains the same,
## but replacing the whole field should be local
stopifnot(identical(v1$text, letters),
identical(v1$viewer, ff),
identical(v2$text, "Hello, world"))
v3 <- v1$copy(TRUE)
v3$viewer$data <- t(xMat) # should modify v1$viewer as well
stopifnot(identical(v1$viewer$data, t(xMat)))
## the field() method
stopifnot(identical(v1$text, v1$field("text")))
v1$field("text", "Now is the time")
stopifnot(identical(v1$field("text"), "Now is the time"))
## setting a non-existent field, or a method, should throw an error
stopifnot(is(tryCatch(v1$field("foobar", 0), error = function(e)e), "error"),
is(tryCatch(v1$field("copy", 0), error = function(e)e), "error") )
## the methods to extract class definition and generator
stopifnot(identical(v3$getRefClass()$def, getRefClass("viewerPlus")$def),
identical(v3$getClass(), getClass("viewerPlus")))
## deal correctly with inherited methods and overriding existing
## methods from $methods(...)
refClassA <- setRefClass("refClassA", methods=list(foo=function() "A"))
refClassB <- setRefClass("refClassB", contains="refClassA")
mnames <- objects(getClass("refClassB")@refMethods)
refClassB$methods(foo=function() callSuper())
stopifnot(identical(refClassB()$foo(), "A"))
mnames2 <- objects(getClass("refClassB")@refMethods)
stopifnot(identical(mnames2[is.na(match(mnames2,mnames))], "foo#refClassA"))
refClassB$methods(foo=function() paste(callSuper(), "Version 2"))
stopifnot(identical(refClassB()$foo(), "A Version 2"))
stopifnot(identical(mnames2, objects(getClass("refClassB")@refMethods)))
if(methods:::.hasCodeTools()) {
## code warnings assigning locally to field names
stopifnot(is(tryCatch(mv$methods(test = function(x)
{ data <- x[!is.na(x)]; mean(data)}),
warning = function(e)e), "warning"))
## warnings for nonlocal assignment that is not a field
stopifnot(is(tryCatch(mv$methods(test2 = function(x) {something <<- data[!is.na(x)]}), warning = function(e)e), "warning"))
## error for trying to assign to a method name
stopifnot(is(tryCatch(mv$methods(test3 = function(x) {edit <<- data[!is.na(x)]}), error = function(e)e), "error"))
} else
warning("Can't run some tests: recommended package codetools is not available")
## tests (fragmentary by necessity) of promptClass for reference class
ccon <- textConnection("ctxt", "w")
suppressMessages(promptClass("refClassB", filename = ccon))
## look for a method, inheritance, inherited method
stopifnot(length(c(grep("foo.*refClassA", ctxt),
grep("code{foo()}", ctxt, fixed = TRUE),
grep("linkS4class{refClassA", ctxt, fixed = TRUE))) >= 3)
close(ccon)
rm(ctxt)
## tests related to subclassing environments. These really test code in the core, viz. builtin.c
a <- refClassA()
ev <- new.env(parent = a) # parent= arg
stopifnot(is.environment(ev))
foo <- function()"A"; environment(foo) <- a # environment of function
stopifnot(identical(as.environment(a), environment(foo)))
xx <- 1:10; environment(xx) <- a # environment attribute
stopifnot(identical(as.environment(a), environment(xx)))
## tests of [[<- and $<- for subclasses of environment. At one point
## methods for these assignments were defined and caused
## inf. recursion when the arguments to the [[<- case were changed in base.
setClass("myEnv", contains = "environment")
m <- new("myEnv", a="test")
m2 <- new("myEnv"); m3 <- new("myEnv")
## test that new.env() is called for each new object
stopifnot(!identical(as.environment(m), as.environment(m2)),
!identical(as.environment(m3), as.environment(m2)))
m[["x"]] <- 1; m$y <- 2
stopifnot(identical(c(m[["x"]], m$y), c(1,2)), is(m, "myEnv"))
rm(x, envir = m) # check rm() works, does not clobber class
stopifnot(identical(sort(objects(m)), sort(c("a", "y"))),
is(m, "myEnv"))
## tests of binding & environment tools with subclases of environment
lockBinding("y", m)
stopifnot(bindingIsLocked("y", m))
unlockBinding("y", m)
stopifnot(!bindingIsLocked("y", m))
makeActiveBinding("z", function(value) {
if(missing(value))
"dummy"
else
"dummy assignment"
}, m)
stopifnot(identical(get("z", m),"dummy"))
## assignment will return the value but do nothing
stopifnot(identical(assign("z","other", m), "other"),
identical(get("z", m),"dummy"))
## this has to be last--Seems no way to unlock an environment!
lockEnvironment(m)
stopifnot(environmentIsLocked(m))
rm(m)
m <- new("myEnv")
stopifnot(length(ls(m)) == 0)
## used to contain the previous content
## test of callSuper() to a hidden default method for initialize() (== initFields)
TestClass <- setRefClass ("TestClass",
fields = list (text = "character"),
methods = list(
print = function () {cat(text)},
initialize = function(text = "", ...) callSuper(text = paste(text, ":", sep=""),...)
))
tt <- TestClass("hello world")
stopifnot(identical(tt$text, "hello world:"))
## now a subclass with another field & another layer of callSuper()
TestClass2 <- setRefClass("TestClass2",
contains = "TestClass",
fields = list( version = "integer"),
methods = list(
initialize = function(..., version = 0L)
callSuper(..., version = version+1L))
)
tt <- TestClass2("test", version = 1L)
stopifnot(identical(tt$text, "test:"), identical(tt$version, as.integer(2)))
tt <- TestClass2(version=3L) # default text
stopifnot(identical(tt$text, ":"), identical(tt$version, as.integer(4)))
## test some capabilities but read-only for .self
.changeAllFields <- function(replacement) {
fields <- names(.refClassDef@fieldClasses)
for(field in fields)
eval(substitute(.self$FIELD <- replacement$FIELD,
list(FIELD = field)))
}
mEditor$methods(change = .changeAllFields)
xx <- mEditor(data = xMat)
xx$edit(2, 2, 0)
yy <- mEditor(data = xMat+1)
yy$change(xx)
stopifnot(identical(yy$data, xx$data), identical(yy$edits, xx$edits))
## but don't allow assigment
if(methods:::.hasCodeTools())
stopifnot(is(tryCatch(yy$.self$data <- xMat, error = function(e)e), "error"))
## the locked binding of refGeneratorSlot class should prevent modifying
## methods, locking fields or setting accessor methods
## Nothing special about refGeneratorSlot in this test -- the point is just
## to use a standard reference class known to be defined in a package
evr <- getRefClass("refGeneratorSlot") # in methods
stopifnot(is(tryCatch(evr$methods(foo = function()"..."), error = function(e)e), "error"),
is(tryCatch(evr$lock("def"), error = function(e)e), "error"),
is(tryCatch(evr$accessors("def"), error = function(e)e), "error"))
##getRefClass() method and function should work with either
## a class name or a class representation (bug report 14600)
tg <- setRefClass("tg", fields = "a")
t1 <- tg(a=1)
tgg <- t1$getRefClass()
tggg <- getRefClass("tg")
stopifnot(identical(tgg$def, tggg$def),
identical(tg$def, tgg$def))
## this used to fail in initFieldArgs() from partial matching "self"
selfClass <- setRefClass("selfClass",
fields=list(
self="character", super="character", sub="character"
)
)
stopifnot(identical(selfClass(self="B", super="A", sub="C")$self, "B"))