# Savage Computer Algebra System
Savage is a new computer algebra system written from scratch in pure Rust.
Its goals are correctness, simplicity, and usability, in that order.
The entire system compiles to a single, dependency-free executable just
2.5 MB in size. While that executable will of course grow as Savage matures,
the plan is to eventually deliver a useful computer algebra system in 5 MB
or less.
![Screenshot](https://user-images.githubusercontent.com/2702526/158006796-7d3aad2a-217f-421a-b3f8-20498d32b0f0.png)
The name "Savage" is a reference/homage to [Sage](https://www.sagemath.org/),
the leading open-source computer algebra system. Since Sage already exists
and works very well, it would make no sense to attempt to create a clone of it.
Instead, Savage aims to be something of an antithesis to Sage: Where Sage is
a unified frontend to dozens of mathematics packages, Savage is a tightly-integrated,
monolithic system. Where Sage covers many areas of mathematics, including cutting-edge
research topics, Savage will focus on the "bread and butter" math employed by
engineers and other people who *use*, rather than develop, mathematical concepts.
Where Sage features amazingly sophisticated implementations of countless functions,
Savage has code that is savagely primitive, getting the job done naively but correctly,
without worrying whether the performance is still optimal when the input is
a million-digit number.
**Savage is in early development and is not yet ready to be used for serious work.**
It is, however, ready to play around with, and is happily accepting contributions
to move the project forward.
## Features
This is what Savage offers **today:**
* Arbitrary-precision integer, rational, and complex arithmetic
* Input, simplification, and evaluation of symbolic expressions
* First-class support for vectors and matrices, with coefficients being arbitrary expressions
* REPL with syntax and bracket highlighting, persistent history, and automatic multi-line input
* Macro-based system for defining functions with metadata and automatic type checking
* [Usable as a library](#savage-as-a-library) from any Rust program
The following features are **planned,** with some of the groundwork already done:
* User-defined variables and functions
* Built-in help system
* Many more functions from various areas of math
* More powerful expression simplification
* Jupyter kernel
By contrast, the following are considered **non-features** for Savage,
and there are no plans to add them either now or in the future:
* *Advanced/research-level mathematics:* As a rule of thumb, if it doesn't belong
in a typical undergraduate course, it probably doesn't belong in Savage.
* *Physics/finance/machine learning/other areas adjacent to math:* The scope would
grow without bounds and that is exactly what Savage aims to avoid.
* *Formal verification of implementations:* The required technologies aren't mature yet
and Savage is not a research project.
* *Performance at the expense of simplicity:* Yes, I know that multiplication
can be done faster using some fancy Fourier tricks. No, I won't implement that.
* *General-purpose programming:* Too complex, and not the focus of this project.
* *File/network I/O:* Savage performs computations, nothing more and nothing less.
Functions have no side effects.
* *Modules/packages/extensions/plugins:* The world is complicated enough.
Either something is built in, or Savage doesn't have it at all.
* *GUI:* Although it's possible to create a GUI frontend backed by the `savage_core`
crate, there are no plans to do so within the Savage project itself.
## Installation
Building Savage from source requires [Rust](https://www.rust-lang.org/) **1.56 or later.**
Once a supported version of Rust is installed on your system, you only need to run
```
cargo install savage
```
to install the Savage REPL to your Cargo binary directory (usually `$HOME/.cargo/bin`).
Of course, you can also just clone this repository and `cargo run` the REPL from the
repository root.
In the future, there will be pre-built executables for major platforms
available with every Savage release.
## Tour
### Arithmetic
Arithmetic operations in Savage have no precision limits (other than the amount
of memory available in your system):
```
in: 1 + 1
out: 2
in: 1.1 ^ 100
out: 13780.612339822270184118337172089636776264331200038466433146477552154985209
5523076769401159497458526446001
in: 3 ^ 4 ^ 5
out: 373391848741020043532959754184866588225409776783734007750636931722079040617
26525122999368893880397722046876506543147515810872705459216085858135133698280918
73141917485942625809388070199519564042855718180410466812887974029255176680123406
17298396574731619152386723046235125934896058590588284654793540505936202376547807
44273058214452705898875625145281779341335214192074462302751872918543286237573706
39854853194764169262638199728870069070138992565242971985276987492741962768110607
02333710356481
```
Results are automatically printed in either fractional or decimal form,
depending on whether the input contained fractions or decimal numbers:
```
in: 6/5 * 3
out: 18/5
in: 1.2 * 3
out: 3.6
```
The variable `i` is predefined to represent the imaginary unit, allowing for
complex numbers to be entered using standard notation:
```
in: (1 + i) ^ 12
out: -64
```
### Linear algebra
Vectors and matrices are first-class citizens in Savage and support the standard
addition, subtraction, multiplication, and exponentiation operators. Coefficients
can be arbitrary expressions:
```
in: [a, b] - [a, c]
out: [0, b - c]
in: [a, b, c] * 3
out: [a * 3, b * 3, c * 3]
in: [[1, 2], [3, 4]] * [5, 6]
out: [17, 39]
```
Determinants are evaluated symbolically:
```
in: det([[a, 2], [3, a]])
out: a ^ 2 - 6
```
### Logic
The standard `&&`, `||`, `!`, and comparison operators are available. Savage
automatically evaluates many tautologies and contradictions, even in the presence
of undefined variables:
```
in: a && true
out: a
in: a || true
out: true
in: a || !a
out: true
in: a < a
out: false
```
### Number theory
Verify that the Mersenne number *M31* is a prime number:
```
in: is_prime(2^31 - 1)
out: true
```
Compute the ten millionth prime number:
```
in: nth_prime(10^7)
out: 179424673
```
Compute the number of primes up to ten million:
```
in: prime_pi(10^7)
out: 664579
```
These functions for dealing with prime numbers are powered by the ultra-fast
[`primal`](https://crates.io/crates/primal) crate. Many more functions from
number theory will be added to Savage in the future.
## Savage as a library
All of Savage's actual computer algebra functionality is contained in the
[`savage_core`](https://crates.io/crates/savage_core) crate. That crate exposes
everything necessary to build software that leverages symbolic math capabilities.
Assuming `savage_core` has been added as a dependency to a crate's `Cargo.toml`,
it can be used like this:
```rust
use std::collections::HashMap;
use savage_core::{expression::Expression, helpers::*};
fn main() {
// Expressions can be constructed by parsing a string literal...
let lhs = "det([[a, 2], [3, a]])".parse::().unwrap();
// ... or directly from code using helper functions.
let rhs = pow(var("a"), int(2)) - int(6);
let mut context = HashMap::new();
// The context can be used to set the values of variables during evaluation.
// Change "b" to "a" to see this in action!
context.insert("b".to_owned(), int(3));
assert_eq!(lhs.evaluate(context), Ok(rhs));
}
```
Please note that at this point, the primary purpose of the `savage_core` crate is
to power the Savage REPL, so any use by third-party crates should be considered
somewhat experimental. Note also that like the rest of Savage, `savage_core` is
licensed under the terms of the [AGPL](LICENSE), which imposes conditions on any
dependent software that go beyond what is required by the more common permissive
licenses. Make sure you understand the AGPL and its implications before adding
`savage_core` as a dependency to your crate.
## Acknowledgments
Savage stands on the shoulders of the giant that is the Rust ecosystem. Among the
many third-party crates that Savage relies on, I want to highlight two that play
a particularly important role:
* [`num`](https://crates.io/crates/num) is the fundamental crate for all numeric
computations in Savage. It provides the crucial `BigInt` type that enables
standard arithmetic operations to be performed with arbitrary precision.
`num`'s code is of high quality and extremely well tested.
* [`chumsky`](https://crates.io/crates/chumsky) is the magic behind Savage's expression
parser. I have looked at every parser crate currently available and found Chumsky's
API to be by far the most intuitive. Furthermore, Chumsky's author is highly
responsive on the issue tracker, and has personally helped me understand and resolve
two major issues that arose during the development of Savage's parser.
## License
Copyright © 2021-2022 Philipp Emanuel Weidmann ()
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see .
**By contributing to this project, you agree to release your
contributions under the same license.**