Interface to KASH¶
Sage provides an interface to the KASH computer algebra system, which is a free (as in beer!) but closed source program for algebraic number theory that shares much common code with Magma. To use KASH, you must first install it. Visit its web page: http://page.math.tu-berlin.de/~kant/kash.html
Todo
Update the following sentence.
It is not enough to just have KASH installed on your computer.
The KASH interface offers three pieces of functionality:
- kash_console()- A function that dumps you into an interactive command-line KASH session. Alternatively,- type - !kashfrom the Sage prompt.
- kash(expr)- Creation of a Sage object that wraps a KASH object. This provides a Pythonic interface to KASH. For example, if- f=kash.new(10), then- f.Factors()returns the prime factorization of \(10\) computed using KASH.
- kash.function_name(args ...)- Call the indicated KASH function with the given arguments are return the result as a KASH object.
- kash.eval(expr)- Evaluation of arbitrary KASH expressions, with the result returned as a string.
Issues¶
For some reason hitting Control + C to interrupt a calculation does not work correctly. (TODO)
Tutorial¶
The examples in this tutorial require that kash be installed.
Basics¶
Basic arithmetic is straightforward. First, we obtain the result as a string.
sage: kash.eval('(9 - 7) * (5 + 6)')                # optional -- kash
'22'
>>> from sage.all import *
>>> kash.eval('(9 - 7) * (5 + 6)')                # optional -- kash
'22'
Next we obtain the result as a new KASH object.
sage: a = kash('(9 - 7) * (5 + 6)'); a              # optional -- kash
22
sage: a.parent()                                    # optional -- kash
Kash
>>> from sage.all import *
>>> a = kash('(9 - 7) * (5 + 6)'); a              # optional -- kash
22
>>> a.parent()                                    # optional -- kash
Kash
We can do arithmetic and call functions on KASH objects:
sage: a*a                                           # optional -- kash
484
sage: a.Factorial()                                 # optional -- kash
1124000727777607680000
>>> from sage.all import *
>>> a*a                                           # optional -- kash
484
>>> a.Factorial()                                 # optional -- kash
1124000727777607680000
Integrated Help¶
Use the kash.help(name) command to get help about a
given command. This returns a list of help for each of the
definitions of name. Use print
kash.help(name) to nicely print out all signatures.
Arithmetic¶
Using the kash.new command we create Kash objects
on which one can do arithmetic.
sage: # optional - kash
sage: a = kash(12345)
sage: b = kash(25)
sage: a/b
2469/5
sage: a**b
1937659030411463935651167391656422626577614411586152317674869233464019922771432158872187137603759765625
>>> from sage.all import *
>>> # optional - kash
>>> a = kash(Integer(12345))
>>> b = kash(Integer(25))
>>> a/b
2469/5
>>> a**b
1937659030411463935651167391656422626577614411586152317674869233464019922771432158872187137603759765625
Variable assignment¶
Variable assignment using kash is takes place in
Sage.
sage: a = kash('32233')                        # optional -- kash
sage: a                                        # optional -- kash
32233
>>> from sage.all import *
>>> a = kash('32233')                        # optional -- kash
>>> a                                        # optional -- kash
32233
In particular, a is not defined as part of the KASH
session itself.
sage: kash.eval('a')                           # optional -- kash
"Error, the variable 'a' must have a value"
>>> from sage.all import *
>>> kash.eval('a')                           # optional -- kash
"Error, the variable 'a' must have a value"
Use a.name() to get the name of the KASH variable:
sage: a.name()                                 # somewhat random; optional - kash
'sage0'
sage: kash(a.name())                           # optional -- kash
32233
>>> from sage.all import *
>>> a.name()                                 # somewhat random; optional - kash
'sage0'
>>> kash(a.name())                           # optional -- kash
32233
Integers and Rationals¶
We illustrate arithmetic with integers and rationals in KASH.
sage: # optional - kash
sage: F = kash.Factorization(4352)
sage: F[1]
<2, 8>
sage: F[2]
<17, 1>
sage: F
[ <2, 8>, <17, 1> ], extended by:
  ext1 := 1,
  ext2 := Unassign
>>> from sage.all import *
>>> # optional - kash
>>> F = kash.Factorization(Integer(4352))
>>> F[Integer(1)]
<2, 8>
>>> F[Integer(2)]
<17, 1>
>>> F
[ <2, 8>, <17, 1> ], extended by:
  ext1 := 1,
  ext2 := Unassign
Note
For some very large numbers KASH’s integer factorization seems much faster than PARI’s (which is the default in Sage).
sage: # optional - kash
sage: kash.GCD(15,25)
5
sage: kash.LCM(15,25)
75
sage: kash.Div(25,15)
1
sage: kash(17) % kash(5)
2
sage: kash.IsPrime(10007)
TRUE
sage: kash.IsPrime(2005)
FALSE
sage: kash.NextPrime(10007)                    # optional -- kash
10009
>>> from sage.all import *
>>> # optional - kash
>>> kash.GCD(Integer(15),Integer(25))
5
>>> kash.LCM(Integer(15),Integer(25))
75
>>> kash.Div(Integer(25),Integer(15))
1
>>> kash(Integer(17)) % kash(Integer(5))
2
>>> kash.IsPrime(Integer(10007))
TRUE
>>> kash.IsPrime(Integer(2005))
FALSE
>>> kash.NextPrime(Integer(10007))                    # optional -- kash
10009
Real and Complex Numbers¶
sage: # optional - kash
sage: kash.Precision()
30
sage: kash('R')
Real field of precision 30
sage: kash.Precision(40)
40
sage: kash('R')
Real field of precision 40
sage: z = kash('1 + 2*I')
sage: z
1.000000000000000000000000000000000000000 + 2.000000000000000000000000000000000000000*I
sage: z*z
-3.000000000000000000000000000000000000000 + 4.000000000000000000000000000000000000000*I
sage: kash.Cos('1.24')                         # optional -- kash
0.3247962844387762365776934156973803996992
sage: kash('1.24').Cos()                       # optional -- kash
0.3247962844387762365776934156973803996992
sage: kash.Exp('1.24')                         # optional -- kash
3.455613464762675598057615494121998175400
sage: kash.Precision(30)                       # optional -- kash
30
sage: kash.Log('3+4*I')                        # optional -- kash
1.60943791243410037460075933323 + 0.927295218001612232428512462922*I
sage: kash.Log('I')                            # optional -- kash
1.57079632679489661923132169164*I
sage: kash.Sqrt(4)                             # optional -- kash
2.00000000000000000000000000000
sage: kash.Sqrt(2)                             # optional -- kash
1.41421356237309504880168872421
sage: kash.Floor('9/5')                        # optional -- kash
1
sage: kash.Floor('3/5')                        # optional -- kash
0
sage: x_c = kash('3+I')                        # optional -- kash
sage: x_c.Argument()                           # optional -- kash
0.321750554396642193401404614359
sage: x_c.Imaginary()                          # optional -- kash
1.00000000000000000000000000000
>>> from sage.all import *
>>> # optional - kash
>>> kash.Precision()
30
>>> kash('R')
Real field of precision 30
>>> kash.Precision(Integer(40))
40
>>> kash('R')
Real field of precision 40
>>> z = kash('1 + 2*I')
>>> z
1.000000000000000000000000000000000000000 + 2.000000000000000000000000000000000000000*I
>>> z*z
-3.000000000000000000000000000000000000000 + 4.000000000000000000000000000000000000000*I
>>> kash.Cos('1.24')                         # optional -- kash
0.3247962844387762365776934156973803996992
>>> kash('1.24').Cos()                       # optional -- kash
0.3247962844387762365776934156973803996992
>>> kash.Exp('1.24')                         # optional -- kash
3.455613464762675598057615494121998175400
>>> kash.Precision(Integer(30))                       # optional -- kash
30
>>> kash.Log('3+4*I')                        # optional -- kash
1.60943791243410037460075933323 + 0.927295218001612232428512462922*I
>>> kash.Log('I')                            # optional -- kash
1.57079632679489661923132169164*I
>>> kash.Sqrt(Integer(4))                             # optional -- kash
2.00000000000000000000000000000
>>> kash.Sqrt(Integer(2))                             # optional -- kash
1.41421356237309504880168872421
>>> kash.Floor('9/5')                        # optional -- kash
1
>>> kash.Floor('3/5')                        # optional -- kash
0
>>> x_c = kash('3+I')                        # optional -- kash
>>> x_c.Argument()                           # optional -- kash
0.321750554396642193401404614359
>>> x_c.Imaginary()                          # optional -- kash
1.00000000000000000000000000000
Lists¶
Note that list appends are completely different in KASH than in Python. Use underscore after the function name for the mutation version.
sage: # optional - kash
sage: v = kash([1,2,3]); v
[ 1, 2, 3 ]
sage: v[1]
1
sage: v[3]
3
sage: v.Append([5])
[ 1, 2, 3, 5 ]
sage: v
[ 1, 2, 3 ]
sage: v.Append_([5, 6])
SUCCESS
sage: v
[ 1, 2, 3, 5, 6 ]
sage: v.Add(5)
[ 1, 2, 3, 5, 6, 5 ]
sage: v
[ 1, 2, 3, 5, 6 ]
sage: v.Add_(5)
SUCCESS
sage: v
[ 1, 2, 3, 5, 6, 5 ]
>>> from sage.all import *
>>> # optional - kash
>>> v = kash([Integer(1),Integer(2),Integer(3)]); v
[ 1, 2, 3 ]
>>> v[Integer(1)]
1
>>> v[Integer(3)]
3
>>> v.Append([Integer(5)])
[ 1, 2, 3, 5 ]
>>> v
[ 1, 2, 3 ]
>>> v.Append_([Integer(5), Integer(6)])
SUCCESS
>>> v
[ 1, 2, 3, 5, 6 ]
>>> v.Add(Integer(5))
[ 1, 2, 3, 5, 6, 5 ]
>>> v
[ 1, 2, 3, 5, 6 ]
>>> v.Add_(Integer(5))
SUCCESS
>>> v
[ 1, 2, 3, 5, 6, 5 ]
The Apply command applies a function to each
element of a list:
sage: # optional - kash
sage: L = kash([1,2,3,4])
sage: L.Apply('i -> 3*i')
[ 3, 6, 9, 12 ]
sage: L
[ 1, 2, 3, 4 ]
sage: L.Apply('IsEven')
[ FALSE, TRUE, FALSE, TRUE ]
sage: L
[ 1, 2, 3, 4 ]
>>> from sage.all import *
>>> # optional - kash
>>> L = kash([Integer(1),Integer(2),Integer(3),Integer(4)])
>>> L.Apply('i -> 3*i')
[ 3, 6, 9, 12 ]
>>> L
[ 1, 2, 3, 4 ]
>>> L.Apply('IsEven')
[ FALSE, TRUE, FALSE, TRUE ]
>>> L
[ 1, 2, 3, 4 ]
Ranges¶
the following are examples of ranges.
sage: # optional - kash
sage: L = kash('[1..10]')
sage: L
[ 1 .. 10 ]
sage: L = kash('[2,4..100]')
sage: L
[ 2, 4 .. 100 ]
>>> from sage.all import *
>>> # optional - kash
>>> L = kash('[1..10]')
>>> L
[ 1 .. 10 ]
>>> L = kash('[2,4..100]')
>>> L
[ 2, 4 .. 100 ]
Sequences¶
Tuples¶
Polynomials¶
sage: # optional - kash
sage: f = kash('X^3 + X + 1')
sage: f + f
2*X^3 + 2*X + 2
sage: f * f
X^6 + 2*X^4 + 2*X^3 + X^2 + 2*X + 1
sage: f.Evaluate(10)
1011
sage: Qx = kash.PolynomialAlgebra('Q')
sage: Qx.gen(1)**5 + kash('7/3')
sage1.1^5 + 7/3
>>> from sage.all import *
>>> # optional - kash
>>> f = kash('X^3 + X + 1')
>>> f + f
2*X^3 + 2*X + 2
>>> f * f
X^6 + 2*X^4 + 2*X^3 + X^2 + 2*X + 1
>>> f.Evaluate(Integer(10))
1011
>>> Qx = kash.PolynomialAlgebra('Q')
>>> Qx.gen(Integer(1))**Integer(5) + kash('7/3')
sage1.1^5 + 7/3
Number Fields¶
We create an equation order.
sage: f = kash('X^5 + 4*X^4 - 56*X^2 -16*X + 192')    # optional -- kash
sage: OK = f.EquationOrder()                          # optional -- kash
sage: OK                                              # optional -- kash
Equation Order with defining polynomial X^5 + 4*X^4 - 56*X^2 - 16*X + 192 over Z
>>> from sage.all import *
>>> f = kash('X^5 + 4*X^4 - 56*X^2 -16*X + 192')    # optional -- kash
>>> OK = f.EquationOrder()                          # optional -- kash
>>> OK                                              # optional -- kash
Equation Order with defining polynomial X^5 + 4*X^4 - 56*X^2 - 16*X + 192 over Z
sage: # optional - kash
sage: f = kash('X^5 + 4*X^4 - 56*X^2 -16*X + 192')
sage: O = f.EquationOrder()
sage: a = O.gen(2)
sage: a
[0, 1, 0, 0, 0]
sage: O.Basis()
[
_NG.1,
_NG.2,
_NG.3,
_NG.4,
_NG.5
]
sage: O.Discriminant()
1364202618880
sage: O.MaximalOrder()
Maximal Order of sage2
sage: O = kash.MaximalOrder('X^3 - 77')                  # optional -- kash
sage: I = O.Ideal(5,[2, 1, 0])                           # optional -- kash
sage: I                    # name sage14 below random; optional -- kash
Ideal of sage14
Two element generators:
[5, 0, 0]
[2, 1, 0]
sage: F = I.Factorisation()                  # optional -- kash
sage: F                    # name sage14 random; optional -- kash
[
<Prime Ideal of sage14
Two element generators:
[5, 0, 0]
[2, 1, 0], 1>
]
>>> from sage.all import *
>>> # optional - kash
>>> f = kash('X^5 + 4*X^4 - 56*X^2 -16*X + 192')
>>> O = f.EquationOrder()
>>> a = O.gen(Integer(2))
>>> a
[0, 1, 0, 0, 0]
>>> O.Basis()
[
_NG.1,
_NG.2,
_NG.3,
_NG.4,
_NG.5
]
>>> O.Discriminant()
1364202618880
>>> O.MaximalOrder()
Maximal Order of sage2
>>> O = kash.MaximalOrder('X^3 - 77')                  # optional -- kash
>>> I = O.Ideal(Integer(5),[Integer(2), Integer(1), Integer(0)])                           # optional -- kash
>>> I                    # name sage14 below random; optional -- kash
Ideal of sage14
Two element generators:
[5, 0, 0]
[2, 1, 0]
>>> F = I.Factorisation()                  # optional -- kash
>>> F                    # name sage14 random; optional -- kash
[
<Prime Ideal of sage14
Two element generators:
[5, 0, 0]
[2, 1, 0], 1>
]
Determining whether an ideal is principal.
sage: I.IsPrincipal()                      # optional -- kash
FALSE, extended by:
ext1 := Unassign
>>> from sage.all import *
>>> I.IsPrincipal()                      # optional -- kash
FALSE, extended by:
ext1 := Unassign
Computation of class groups and unit groups:
sage: # optional - kash
sage: f = kash('X^5 + 4*X^4 - 56*X^2 -16*X + 192')
sage: O = kash.EquationOrder(f)
sage: OK = O.MaximalOrder()
sage: OK.ClassGroup()
Abelian Group isomorphic to Z/6
  Defined on 1 generator
  Relations:
  6*sage32.1 = 0, extended by:
  ext1 := Mapping from: grp^abl: sage32 to ids/ord^num: _AA
>>> from sage.all import *
>>> # optional - kash
>>> f = kash('X^5 + 4*X^4 - 56*X^2 -16*X + 192')
>>> O = kash.EquationOrder(f)
>>> OK = O.MaximalOrder()
>>> OK.ClassGroup()
Abelian Group isomorphic to Z/6
  Defined on 1 generator
  Relations:
  6*sage32.1 = 0, extended by:
  ext1 := Mapping from: grp^abl: sage32 to ids/ord^num: _AA
sage: U = OK.UnitGroup()                                  # optional -- kash
sage: U        # name sage34 below random; optional -- kash
Abelian Group isomorphic to Z/2 + Z + Z
  Defined on 3 generators
  Relations:
  2*sage34.1 = 0, extended by:
  ext1 := Mapping from: grp^abl: sage34 to ord^num: sage30
sage: kash.Apply('x->%s.ext1(x)'%U.name(), U.Generators().List())     # optional -- kash
[ [1, -1, 0, 0, 0], [1, 1, 0, 0, 0], [-1, 0, 0, 0, 0] ]
>>> from sage.all import *
>>> U = OK.UnitGroup()                                  # optional -- kash
>>> U        # name sage34 below random; optional -- kash
Abelian Group isomorphic to Z/2 + Z + Z
  Defined on 3 generators
  Relations:
  2*sage34.1 = 0, extended by:
  ext1 := Mapping from: grp^abl: sage34 to ord^num: sage30
>>> kash.Apply('x->%s.ext1(x)'%U.name(), U.Generators().List())     # optional -- kash
[ [1, -1, 0, 0, 0], [1, 1, 0, 0, 0], [-1, 0, 0, 0, 0] ]
Function Fields¶
sage: # optional - kash
sage: k = kash.FiniteField(25)
sage: kT = k.RationalFunctionField()
sage: kTy = kT.PolynomialAlgebra()
sage: T = kT.gen(1)
sage: y = kTy.gen(1)
sage: f = y**3 + T**4 + 1
>>> from sage.all import *
>>> # optional - kash
>>> k = kash.FiniteField(Integer(25))
>>> kT = k.RationalFunctionField()
>>> kTy = kT.PolynomialAlgebra()
>>> T = kT.gen(Integer(1))
>>> y = kTy.gen(Integer(1))
>>> f = y**Integer(3) + T**Integer(4) + Integer(1)
Long Input¶
The KASH interface reads in even very long input (using files) in a robust manner, as long as you are creating a new object.
Note
Using kash.eval for long input is much less robust, and is not
recommended.
sage: a = kash(range(10000))                                  # optional -- kash
>>> from sage.all import *
>>> a = kash(range(Integer(10000)))                                  # optional -- kash
Note that KASH seems to not support string or integer literals with more than 1024 digits, which is why the above example uses a list unlike for the other interfaces.
- class sage.interfaces.kash.Kash(max_workspace_size=None, maxread=None, script_subdirectory=None, restart_on_ctrlc=True, logfile=None, server=None, server_tmpdir=None)[source]¶
- Bases: - Expect- Interface to the Kash interpreter. - AUTHORS: - William Stein and David Joyner 
 - clear(var)[source]¶
- Clear the variable named - var.- Kash variables have a record structure, so if sage1 is a polynomial ring, sage1.1 will be its indeterminate. This prevents us from easily reusing variables, since sage1.1 might still have references even if sage1 does not. - For now, we don’t implement variable clearing to avoid these problems, and instead implement this method with a noop. 
 - eval(x, newlines=False, strip=True, **kwds)[source]¶
- Send the code in the string s to the Kash interpreter and return the output as a string. - INPUT: - s– string containing Kash code
- newlines– boolean (default:- True); if- False, remove all backslash-newlines inserted by the Kash output formatter
- strip– ignored
 
 - function_call(function, args=None, kwds=None)[source]¶
- EXAMPLES: - sage: kash.function_call('ComplexToPolar', [1+I], {'Results' : 1}) # optional -- kash 1.41421356237309504880168872421 - >>> from sage.all import * >>> kash.function_call('ComplexToPolar', [Integer(1)+I], {'Results' : Integer(1)}) # optional -- kash 1.41421356237309504880168872421 
 - help(name=None)[source]¶
- Return help on KASH commands. - This returns help on all commands with a given name. If name is - None, return the location of the installed Kash HTML documentation.- EXAMPLES: - sage: X = kash.help('IntegerRing') # random; optional -- kash 1439: IntegerRing() -> <ord^rat> 1440: IntegerRing(<elt-ord^rat> m) -> <res^rat> 1441: IntegerRing(<seq()> Q) -> <res^rat> 1442: IntegerRing(<fld^rat> K) -> <ord^rat> 1443: IntegerRing(<fld^fra> K) -> <ord^num> 1444: IntegerRing(<rng> K) -> <rng> 1445: IntegerRing(<fld^pad> L) -> <ord^pad> - >>> from sage.all import * >>> X = kash.help('IntegerRing') # random; optional -- kash 1439: IntegerRing() -> <ord^rat> 1440: IntegerRing(<elt-ord^rat> m) -> <res^rat> 1441: IntegerRing(<seq()> Q) -> <res^rat> 1442: IntegerRing(<fld^rat> K) -> <ord^rat> 1443: IntegerRing(<fld^fra> K) -> <ord^num> 1444: IntegerRing(<rng> K) -> <rng> 1445: IntegerRing(<fld^pad> L) -> <ord^pad> - There is one entry in X for each item found in the documentation for this function: If you type - print(X[0])you will get help on about the first one, printed nicely to the screen.- AUTHORS: - Sebastion Pauli (2006-02-04): during Sage coding sprint 
 
 
- class sage.interfaces.kash.KashElement(parent, value, is_name=False, name=None)[source]¶
- Bases: - ExpectElement
- sage.interfaces.kash.is_KashElement(x)[source]¶
- Return - Trueif- xis of type- KashElement.- EXAMPLES: - sage: from sage.interfaces.kash import is_KashElement sage: is_KashElement(2) doctest:...: DeprecationWarning: the function is_KashElement is deprecated; use isinstance(x, sage.interfaces.abc.KashElement) instead See https://github.com/sagemath/sage/issues/34804 for details. False sage: is_KashElement(kash(2)) # optional - kash True - >>> from sage.all import * >>> from sage.interfaces.kash import is_KashElement >>> is_KashElement(Integer(2)) doctest:...: DeprecationWarning: the function is_KashElement is deprecated; use isinstance(x, sage.interfaces.abc.KashElement) instead See https://github.com/sagemath/sage/issues/34804 for details. False >>> is_KashElement(kash(Integer(2))) # optional - kash True