The purposes of this package are to be able to deal with large R data (rda) files created via a call to save() without loading the R objects but
- providing a description of the top-level variables it contains without restoring/instantiating the objects,
- allowing deserializing of one or a subset of the variables without restoring the others,
- know where (the offset in the file) each top-level variable starts in the file,
- know what is in the file, including any code that might be evaluated.
This can help finding a particular data.frame or variable generally in a collection of rda files when we know identifying characteristics, e.g., a data.frame with columns named "x" and "y", or with a column named "d" with class "Date" or "POSIXct".
Note: If the table of offsets was placed at the end of the RDA file, we would be able provide this "random"/direct access to variables without having to first construct the table of offsets. We explore this elsewhere.
The following is an example of its use:
rda.file = system.file("sampleRDA/class_named_integer_logical_character_uncompress.rda", package = "RDAInfo")
tc = toc(rda.file)
tc
file: path/to/class_named_integer_logical_character_uncompress.rda
encoding: UTF-8
type length class offset
i INTSXP 4 Example 32
l LGLSXP 5 <NA> 195
let STRSXP 7 <NA> 240
The data.frame above shows the
- names of the variables,
- the type of each variable
- length
- class and
- offset/location.
Only i
has a class attribute.
We can extract an individual variable with, e.g.,
tc[[ "let" ]]
[1] "a" "b" "c" "d" "e" "f" "g"
And similarly, we can extract multiple elements, e.g.,
tc[ c("l", "let") ]
tc[ c(2, 3) ]
We (currently) use $
to get the description of an object from the RDAToc
object, e.g.,
tc$i
type length class names name offset
1 INTSXP 4 Example FALSE jane, bob, mary, xi 45
This also shows the names.
For a function object, we have information about the
- number of parameters
- the parameter names
- whether it has a
...
parameter - whether each parameter has a default value
When constructing the data.frame for a collection of variables in an RDA file, by default, we use the union of all fields in the descriptions of the objects. The as.data.frame method also supports using the intersection of the fields, i.e., only the fields common to all object descriptions in the RDA file.
A more recent example that takes the union of the fields in the description of each top-level object in the RDA:
f = system.file("sampleRDA", "test.rda", package = "RDAInfo")
info = toc(f)
info
file: /Users/duncan/Rpackages4/RDAInfo/sampleRDA/test.rda
encoding: UTF-8
type length class names name offset numParams srcref dim hasRowNames colInfo
a REALSXP 10 <NA> FALSE a, b, c, d, e, f, g, h, i, j 32 NA NA NA NA NA
b REALSXP 10 <NA> FALSE NA 260 NA NA NA NA NA
m INTSXP 1 <NA> FALSE NA 365 NA NA NA NA NA
f CLOSXP NA <NA> NA NA 394 1 1, 5, 1, 33, 5, 33, 1, 1 NA NA NA
mx REALSXP 10 <NA> FALSE NA 901 NA NA 5, 2 NA NA
d VECSXP 2 data.frame a, group NA 1046 NA NA 10, 2 TRUE REALSXP, INTSXP
From this, we see
- the class on the data.frame
d
- whether it has row names
- the column names and types for the data.frame
- the dimensions for the data.frame and the matrix
- the length of each object, except the function
- the number of parameters in the function signature
- the srcref is not very relevant.
We have the offset in the file at which each of the objects starts.
load with an option to only assign a subset is a minor help in not overwriting existing variables
Load('filename.rda', vars = c('a', 'b'))
This restores all the variables in the Rda file into a new environment and then
assign those in vars to the target envir
.
This just avoids the caller having to create the new environment and then assigning this subset of
variables. It does allow us to map the variable names to something different, e.g.,
Load('filename.rda', vars = c(x = 'a', y = 'b'))
would create variables x and y, copying a and b to those variable names in the target environment.
And of course, if vars is empty, this would be the same as calling load()
.
This is the start of an R-code implementation of reading an XDR-formatted RDA (rdata) archive. Since this is already implemented fully and efficiently in C (serialize.c), this version will not be competitive. There are however several motivations: 1.see how to implement a version that does not read the values or create the R objects, but provides a summary of the top-level objects:
+ names of variables
+ length and/or dimensions
+ types & classes
+ whether they have names or not
Perfecting this in R code will make it easier to implement correctly in the C code.
-
If we can make this sufficiently competitive with the C code because, while it is slower in R it does less (not allocating memory, not interpreting certain types of bytes), then this can be faster for large but simple RDA files. (This basically means vectors and lists of vectors that don't contain large character vectors, either as vectors or attributes.)
-
A slow R implementation may serve as an interesting example for compiling R code to native instructions, e.g., the Rllvm approach I am experimenting with (RLLVCompile, etc.)
-
This would allow us to experiment with a slight enhancement/extension to the RDA format to provide
- a built-in table of contents/summary of its contents
- direct access to individual objects in the RDA file by name without having to read or restore the preceding objects, but instead jumping directly to their starting position.
-
This package does provide 2 simple routines to deal with XDR int and double values which is functionality that doesn't appear to be in any CRAN package.
In other words, this is an R prototype of something we might consider implementing in C, and specifically, adapting the code in serialize.c (and saveload.c) in the R source. It is not intended to be a production-level solution that will work best in all cases.
√ We did may well implement the most significant bottleneck in C, that is reading large
character vectors. This nows makes the R implementation faster than load()'ing all the objects, at
least for cases of a collection of large vectors.
While I originally thought this table-of-contents functionality would be useful about 20 years ago, the current motivation grew from something I am writing as a case study in "Hacking the R engine" with the hope of helping people learn to experiment with and contribute to the R source code.
While we wrote this only today and have now updated it a little, and we haven't tested it much, this currently
-
Reads
- all vectors, matrices, lists, data.frames
- attributes
- functions
- language objects generally
- environments
-
Does not YET handle
- S4 objects
- Byte code (BCODE)
Contents of the RDA file are 4 vectors, each of length 1e7
- numeric
- numeric
- numeric
- factor
268M uncompressed
On a Macbook Pro (32Gb RAM)
- load: .775 seconds
- Table of Contents: .007 seconds
- Factor: 110.7
(Medians over 5 runs)
If we change the factor to a character vector, the results are not even close. The original R code in this package read each string in a character vector with three R calls;
- consume the CHARSXP integer
- read the number of characters/bytes
- skip that number of bytes So we read the entire character vector with
nc = replicate(len, { readInteger(con); nchar = readInteger(con); seek(con, nchar, "current"); nchar})
The R interpreter overhead is enormous. We could change the first call to readInteger() to seek(, 4L)
as we discard the value. However, this won't change the overall timing sufficiently to make
this approach competitive.
However, switching to a C routine that does the same thing (in src/eatCharacterVectorEelements.c
)
does.
On a Macbook Pro (32Gb RAM)
- load: 3.6 seconds
- Table of Contents: 1.8 seconds
- Speedup factor: 2.0
(Medians over 5 runs, elapsed time.)
This doesn't take into account that the load()
version has used a significant amount of memory
and will have to be garbage collected. (Nor has it summarized the variable types.)
These two variations show the efficiency of factors in representing a large character vector with a small number of unique values.
I (programmatically) scanned my drive for all .rda files. The largest of these contains a single object which is a list with 785 elements.
Loading the file takes 3.92 seconds. Reading the table of contents takes .054.
This should originally have been an RDS file. The name did tell me what it contained - SmallSampleTrainDist.
(A good point is: if this is the largest file you have, just load it to see what it contains; you get more information. Well, I used to have much larger files!)
The code in R/readRDA.R reads the XDR stream. It identifies the different SEXP objects and processes their contents.
A tool that prints the SEXP type and the flags for the given SEXP element is useful for seeing the (hierarchical) structure of the stream. It uses the LLDB debugger to collect the information from the R serialize.c code and display it on the console (or collect it in a Python object.)