r"""
Interface to Sage
This is an expect interface to *another* copy of the Sage
interpreter.
"""
import cPickle, os
from expect import Expect, ExpectElement, FunctionElement
import sage.misc.preparser
from sage.structure.sage_object import dumps, load
class Sage(Expect):
r"""
Expect interface to the Sage interpreter itself.
INPUT:
- ``server`` - (optional); if specified runs Sage on a
remote machine with address. You must have ssh keys setup so you
can login to the remote machine by typing "ssh remote_machine" and
no password, call _install_hints_ssh() for hints on how to do
that.
The version of Sage should be the same as on the local machine,
since pickling is used to move data between the two Sage process.
EXAMPLES: We create an interface to a copy of Sage. This copy of
Sage runs as an external process with its own memory space, etc.
::
sage: s = Sage()
Create the element 2 in our new copy of Sage, and cube it.
::
sage: a = s(2)
sage: a^3
8
Create a vector space of dimension `4`, and compute its
generators::
sage: V = s('QQ^4')
sage: V.gens()
((1, 0, 0, 0), (0, 1, 0, 0), (0, 0, 1, 0), (0, 0, 0, 1))
Note that V is a not a vector space, it's a wrapper around an
object (which happens to be a vector space), in another running
instance of Sage.
::
sage: type(V)
<class 'sage.interfaces.sage0.SageElement'>
sage: V.parent()
Sage
sage: g = V.0; g
(1, 0, 0, 0)
sage: g.parent()
Sage
We can still get the actual parent by using the name attribute of
g, which is the variable name of the object in the child process.
::
sage: s('%s.parent()'%g.name())
Vector space of dimension 4 over Rational Field
Note that the memory space is completely different.
::
sage: x = 10
sage: s('x = 5')
5
sage: x
10
sage: s('x')
5
We can have the child interpreter itself make another child Sage
process, so now three copies of Sage are running::
sage: s3 = s('Sage()')
sage: a = s3(10)
sage: a
10
This `a=10` is in a subprocess of a subprocesses of your
original Sage.
::
sage: _ = s.eval('%s.eval("x=8")'%s3.name())
sage: s3('"x"')
8
sage: s('x')
5
sage: x
10
The double quotes are needed because the call to s3 first evaluates
its arguments using the s interpreter, so the call to s3 is passed
``s('"x"')``, which is the string ``"x"``
in the s interpreter.
"""
def __init__(self, logfile = None,
preparse = True,
python = False,
init_code = None,
server = None,
server_tmpdir = None,
remote_cleaner = True,
**kwds):
"""
EXAMPLES::
sage: sage0 == loads(dumps(sage0))
True
"""
if python:
if server:
command = "sage -python -u"
else:
command = "sage -python -u"
prompt = ">>>"
if init_code is None:
init_code = ['from sage.all import *', 'import cPickle']
else:
command = "sage"
prompt = "sage: "
if init_code is None:
init_code = ['import cPickle', '%colors NoColor']
Expect.__init__(self,
name = 'sage',
prompt = prompt,
command = command,
restart_on_ctrlc = False,
logfile = logfile,
init_code = init_code,
server = server,
server_tmpdir = server_tmpdir,
remote_cleaner = remote_cleaner,
**kwds
)
self._preparse = preparse
def cputime(self, t=None):
"""
Return cputime since this Sage subprocess was started.
EXAMPLES::
sage: sage0.cputime() # random output
1.3530439999999999
sage: sage0('factor(2^157-1)')
852133201 * 60726444167 * 1654058017289 * 2134387368610417
sage: sage0.cputime() # random output
1.6462939999999999
"""
s = self.eval('cputime(%s)'%t)
i = s.rfind('m')
if i != -1:
s = s[i+1:-1]
return float(s)
def trait_names(self):
"""
EXAMPLES::
sage: t = sage0.trait_names()
sage: len(t) > 100
True
sage: 'gcd' in t
True
"""
return eval(self.eval('print repr(globals().keys())'))
def quit(self, verbose=False):
"""
EXAMPLES::
sage: s = Sage()
sage: s.eval('2+2')
'4'
sage: s.quit()
"""
import signal
if not self._expect is None:
pid = self._expect.pid
if verbose:
if self.is_remote():
print "Exiting spawned %s process (local pid=%s, running on %s)"%(self,pid,self._server)
else:
print "Exiting spawned %s process (pid=%s)."%(self, pid)
try:
for i in range(10):
self._expect.sendline(chr(3))
self._expect.sendline('quit_sage(verbose=%s)'%verbose)
self._so_far(wait=0.2)
os.killpg(pid, 9)
os.kill(pid, 9)
except (RuntimeError, OSError), msg:
pass
try:
os.killpg(pid, 9)
os.kill(pid, 9)
except OSError:
pass
try:
self._expect.close(signal.SIGQUIT)
except Exception:
pass
self._expect = None
def __call__(self, x):
"""
EXAMPLES::
sage: a = sage0(4)
sage: a.parent()
Sage
sage: a is sage0(a)
True
TESTS::
sage: sage0(axiom(x^2+1)) #optional - axiom
x^2 + 1
"""
if isinstance(x, ExpectElement):
if x.parent() is self:
return x
else:
return self(x.sage())
if isinstance(x, str):
return SageElement(self, x)
if self.is_local():
open(self._local_tmpfile(),'w').write(cPickle.dumps(x,2))
return SageElement(self, 'cPickle.load(open("%s"))'%self._local_tmpfile())
else:
open(self._local_tmpfile(),'w').write(dumps(x))
self._send_tmpfile_to_server()
return SageElement(self, 'loads(open("%s").read())'%self._remote_tmpfile())
def __reduce__(self):
"""
EXAMPLES::
sage: sage0.__reduce__()
(<function reduce_load_Sage at 0x...>, ())
"""
return reduce_load_Sage, tuple([])
def _quit_string(self):
"""
EXAMPLES::
sage: sage0._quit_string()
'from sage.misc.misc import delete_tmpfiles; delete_tmpfiles()'
"""
return 'from sage.misc.misc import delete_tmpfiles; delete_tmpfiles()'
def preparse(self, x):
"""
Returns the preparsed version of the string s.
EXAMPLES::
sage: sage0.preparse('2+2')
'Integer(2)+Integer(2)'
"""
return sage.misc.preparser.preparse(x)
def eval(self, line, strip=True, **kwds):
"""
Send the code x to a second instance of the Sage interpreter and
return the output as a string.
This allows you to run two completely independent copies of Sage at
the same time in a unified way.
INPUT:
- ``line`` - input line of code
- ``strip`` - ignored
EXAMPLES::
sage: sage0.eval('2+2')
'4'
"""
if self._preparse:
line = self.preparse(line)
return Expect.eval(self, line, **kwds).strip()
def set(self, var, value):
"""
Set the variable var to the given value.
EXAMPLES::
sage: sage0.set('x', '2')
sage: sage0.get('x')
'2'
"""
cmd = '%s=%s'%(var,value)
out = self.eval(cmd)
if 'Traceback' in out:
raise TypeError, "Error executing code in Sage\nCODE:\n\t%s\nSage ERROR:\n\t%s"%(cmd, out)
def get(self, var):
"""
Get the value of the variable var.
EXAMPLES::
sage: sage0.set('x', '2')
sage: sage0.get('x')
'2'
"""
return self.eval('print %s'%var).strip()
def clear(self, var):
"""
Clear the variable named var.
Note that the exact format of the NameError for a cleared variable
is slightly platform dependent, see trac #10539.
EXAMPLES::
sage: sage0.set('x', '2')
sage: sage0.get('x')
'2'
sage: sage0.clear('x')
sage: 'NameError' in sage0.get('x')
True
"""
self.eval('del %s'%var)
def _contains(self, v1, v2):
"""
EXAMPLES::
sage: sage0._contains('2', 'QQ')
True
"""
return self.eval('%s in %s'%(v1,v2)) == "True"
def console(self):
"""
Spawn a new Sage command-line session.
EXAMPLES::
sage: sage0.console() #not tested
----------------------------------------------------------------------
| Sage Version ..., Release Date: ... |
| Type notebook() for the GUI, and license() for information. |
----------------------------------------------------------------------
...
"""
sage0_console()
def version(self):
"""
EXAMPLES::
sage: sage0.version()
'Sage Version ..., Release Date: ...'
sage: sage0.version() == version()
True
"""
return sage0_version()
def _object_class(self):
"""
EXAMPLES::
sage: sage0._object_class()
<class 'sage.interfaces.sage0.SageElement'>
"""
return SageElement
def new(self, x):
"""
EXAMPLES::
sage: sage0.new(2)
2
sage: _.parent()
Sage
"""
return SageElement(self, x)
class SageElement(ExpectElement):
def __getattr__(self, attrname):
"""
EXAMPLES::
sage: m = sage0(4)
sage: four_gcd = m.gcd
sage: type(four_gcd)
<class 'sage.interfaces.sage0.SageFunction'>
"""
self._check_valid()
return SageFunction(self, attrname)
def _sage_(self):
"""
Return local copy of self.
EXAMPLE::
sage: sr = mq.SR(allow_zero_inversions=True)
sage: F,s = sr.polynomial_system()
sage: F == sage0(F)._sage_()
True
"""
P = self.parent()
if P.is_remote():
P.eval('save(%s, "%s")'%(self.name(), P._remote_tmpfile()))
P._get_tmpfile_from_server(self)
return load(P._local_tmp_file())
else:
P.eval('save(%s, "%s")'%(self.name(), P._local_tmpfile()))
return load(P._local_tmpfile())
class SageFunction(FunctionElement):
def __call__(self, *args, **kwds):
"""
EXAMPLES::
sage: four_gcd = sage0(4).gcd
sage: four_gcd(6)
2
"""
P = self._obj.parent()
args = [P(x) for x in args]
args = ','.join([x.name() for x in args])
kwds = ",".join(["%s=%s"%(k,P(v).name()) for k,v in kwds.iteritems()])
if args != "" and kwds != "":
callstr = '%s.%s(%s,%s)'%(self._obj._name, self._name, args, kwds)
elif kwds != "":
callstr = '%s.%s(%s)'%(self._obj._name, self._name, kwds)
elif args != "":
callstr = '%s.%s(%s)'%(self._obj._name, self._name, args)
else:
callstr = '%s.%s()'%(self._obj._name, self._name)
z = SageElement(P, callstr)
return z
def __repr__(self):
"""
EXAMPLES::
sage: sage0(4).gcd
<built-in method gcd of sage.rings.integer.Integer object at 0x...>
"""
return str(self._obj.parent().eval('%s.%s'%(self._obj._name, self._name)))
sage0 = Sage()
def reduce_load_Sage():
"""
EXAMPLES::
sage: from sage.interfaces.sage0 import reduce_load_Sage
sage: reduce_load_Sage()
Sage
"""
return sage0
def reduce_load_element(s):
"""
EXAMPLES::
sage: from sage.interfaces.sage0 import reduce_load_element
sage: s = dumps(1/2)
sage: half = reduce_load_element(s); half
1/2
sage: half.parent()
Sage
"""
import base64
s = base64.b32encode(s)
sage0.eval('import base64')
return sage0('loads(base64.b32decode("%s"))'%s)
import os
def sage0_console():
"""
Spawn a new Sage command-line session.
EXAMPLES::
sage: sage0_console() #not tested
----------------------------------------------------------------------
| Sage Version ..., Release Date: ... |
| Type notebook() for the GUI, and license() for information. |
----------------------------------------------------------------------
...
"""
os.system('sage')
def sage0_version():
"""
EXAMPLES::
sage: from sage.interfaces.sage0 import sage0_version
sage: sage0_version() == version()
True
"""
return str(sage0('version()'))