Source code for rpy2.robjects.robject
import os
import sys
import tempfile
import weakref
import rpy2.rinterface
import rpy2.rinterface_lib.callbacks
from . import conversion
rpy2.rinterface.initr()
class RSlots(object):
""" Attributes of an R object as a Python mapping.
The parent proxy to the underlying R object is held as a
weak reference. The attributes are therefore not protected
from garbage collection unless bound to a Python symbol or
in an other container.
"""
__slots__ = ['_robj', ]
def __init__(self, robj):
self._robj = weakref.proxy(robj)
def __getitem__(self, key):
value = self._robj.do_slot(key)
return conversion.rpy2py(value)
def __setitem__(self, key, value):
rpy2_value = conversion.py2rpy(value)
self._robj.do_slot_assign(key, rpy2_value)
def __len__(self):
return len(self._robj.list_attrs())
def keys(self):
for k in self._robj.list_attrs():
yield k
__iter__ = keys
def items(self):
for k in self._robj.list_attrs():
v = self[k]
yield (k, v)
def values(self):
for k in self._robj.list_attrs():
v = self[k]
yield v
_get_exported_value = rpy2.rinterface.baseenv['::']
[docs]class RObjectMixin(object):
""" Class to provide methods common to all RObject instances. """
__rname__ = None
__tempfile = rpy2.rinterface.baseenv.find("tempfile")
__file = rpy2.rinterface.baseenv.find("file")
__fifo = rpy2.rinterface.baseenv.find("fifo")
__sink = rpy2.rinterface.baseenv.find("sink")
__close = rpy2.rinterface.baseenv.find("close")
__readlines = rpy2.rinterface.baseenv.find("readLines")
__unlink = rpy2.rinterface.baseenv.find("unlink")
__show = _get_exported_value('methods', 'show')
__slots = None
@property
def slots(self):
""" Attributes of the underlying R object as a Python mapping.
The attributes can accessed and assigned by name (as if they
were in a Python `dict`)."""
if self.__slots is None:
self.__slots = RSlots(self)
return self.__slots
def __repr__(self):
try:
rclasses = ('R object with classes: {} mapped to:'
.format(tuple(self.rclass)))
except Exception:
rclasses = 'Unable to fetch R classes.' + os.linesep
os.linesep.join((rclasses,
repr(super())))
return rclasses
def __str__(self):
s = []
with (rpy2.rinterface_lib
.callbacks.obj_in_module(rpy2.rinterface_lib.callbacks,
'consolewrite_print', s.append)):
self.__show(self)
s = str.join('', s)
return s
def __getstate__(self, ):
return (super().__getstate__(), self.__dict__.copy())
def __setstate__(self, state):
rds, __dict__ = state
super().__setstate__(rds)
self.__dict__.update(__dict__)
[docs] def r_repr(self):
""" String representation for an object that can be
directly evaluated as R code.
"""
return repr_robject(self, linesep='\n')
@property
def rclass(self):
"""
R class for the object, stored as an R string vector.
When setting the rclass, the new value will be:
- wrapped in a Python tuple if a string (the R class
is a vector of strings, and this is made for convenience)
- wrapped in a StrSexpVector
Note that when setting the class R may make a copy of
the whole object (R is mostly a functional language).
If this must be avoided, and if the number of parent
classes before and after the change are compatible,
the class name can be changed in-place by replacing
vector elements."""
try:
res = super(RObjectMixin, self).rclass
res = rpy2.rinterface.sexp.rclass_get(self.__sexp__)
return res
except rpy2.rinterface._rinterface.embedded.RRuntimeError as rre:
if self.typeof == rpy2.rinterface.RTYPES.SYMSXP:
# Unevaluated expression: has no class.
return (None, )
else:
raise rre
@rclass.setter
def rclass(self, value):
if isinstance(value, str):
value = (value, )
new_cls = rpy2.rinterface.StrSexpVector(value)
rpy2.rinterface.sexp.rclass_set(self.__sexp__, new_cls)
def repr_robject(o, linesep=os.linesep):
s = rpy2.rinterface.baseenv.find("deparse")(o)
s = str.join(linesep, s)
return s
[docs]class RObject(RObjectMixin, rpy2.rinterface.Sexp):
""" Base class for all non-vector R objects. """
def __setattr__(self, name, value):
if name == '_sexp':
if not isinstance(value, rpy2.rinterface.Sexp):
raise ValueError(
'_attr must contain an object '
'that inherits from rpy2.rinterface.Sexp '
'(not from %s)' % type(value)
)
super(RObject, self).__setattr__(name, value)