Source code for rpy2.robjects.robject
import os, sys
import tempfile
import weakref
import rpy2.rinterface
rpy2.rinterface.initr()
from . import conversion
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.ri2ro(value)
def __setitem__(self, key, value):
rpy2_value = conversion.py2ri(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
[docs]class RObjectMixin(object):
""" Class to provide methods common to all RObject instances. """
__rname__ = None
__tempfile = rpy2.rinterface.baseenv.get("tempfile")
__file = rpy2.rinterface.baseenv.get("file")
__fifo = rpy2.rinterface.baseenv.get("fifo")
__sink = rpy2.rinterface.baseenv.get("sink")
__close = rpy2.rinterface.baseenv.get("close")
__readlines = rpy2.rinterface.baseenv.get("readLines")
__unlink = rpy2.rinterface.baseenv.get("unlink")
__rclass = rpy2.rinterface.baseenv.get("class")
__rclass_set = rpy2.rinterface.baseenv.get("class<-")
__show = rpy2.rinterface.baseenv.get("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:
rclasses = 'Unable to fetch R classes.' + os.linesep
res = os.linesep.join((rclasses,
super(RObjectMixin, self).__repr__()))
return res
def __str__(self):
if sys.platform == 'win32':
tmpf = tempfile.NamedTemporaryFile(mode="w+", delete=False)
tfname = tmpf.name
tmp = self.__file(rpy2.rinterface.StrSexpVector([tfname,]),
open=rpy2.rinterface.StrSexpVector(["r+", ]))
self.__sink(tmp)
else:
writeconsole = rpy2.rinterface.get_writeconsole_regular()
s = []
def f(x):
s.append(x)
rpy2.rinterface.set_writeconsole_regular(f)
self.__show(self)
if sys.platform == 'win32':
self.__sink()
s = tmpf.readlines()
tmpf.close()
self.__close(tmp)
try:
del tmpf
os.unlink(tfname)
except WindowsError:
if os.path.exists(tfname):
print('Unable to unlink tempfile %s' % tfname)
s = str.join(os.linesep, s)
else:
rpy2.rinterface.set_writeconsole_regular(writeconsole)
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')
def _rclass_get(self):
try:
res = self.__rclass(self)
#res = conversion.ri2py(res)
return res
except rpy2.rinterface.RRuntimeError as rre:
if self.typeof == rpy2.rinterface.SYMSXP:
#unevaluated expression: has no class
return (None, )
else:
raise rre
def _rclass_set(self, value):
if isinstance(value, str):
value = (value, )
new_cls = rpy2.rinterface.StrSexpVector(value)
res = self.__rclass_set(self, new_cls)
self.__sexp__ = res.__sexp__
rclass = property(_rclass_get, _rclass_set, None,
"""
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.""")
def repr_robject(o, linesep=os.linesep):
s = rpy2.rinterface.baseenv.get("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)