cpc

calculation + conversion

cpc parses and evaluates strings of math, with support for units and conversion. 128-bit decimal floating points are used for high accuracy.

It also lets you mix units, so for example 1 km - 1m results in Number { value: 999, unit: Meter }.

Crates.io Documentation

List of all supported units

CLI Installation

Install using cargo: cargo install cpc

To install it manually, grab the appropriate binary from the GitHub Releases page and place it wherever you normally place binaries on your OS.

CLI Usage

cpc '2h/3 to min'

API Installation

Add cpc as a dependency in Cargo.toml.

API Usage

```rust use cpc::{eval}; use cpc::units::Unit;

match eval("3m + 1cm", true, Unit::Celsius, false) { Ok(answer) => { // answer: Number { value: 301, unit: Unit::Centimeter } println!("Evaluated value: {} {:?}", answer.value, answer.unit) }, Err(e) => { println!("{}", e) } } ```

Examples

``` 3 + 4 * 2

8 % 3

(4 + 1)km to light years

10m/2s * 5 trillion s

1 lightyear * 0.001mm in km2

1m/s + 1mi/h in kilometers per h

round(sqrt(2)^4)! liters

10% of abs(sin(pi)) horsepower to watts ```

Supported unit types

Accuracy

cpc uses 128-bit Decimal Floating Point (d128) numbers instead of Binary Coded Decimals for better accuracy. The result cpc gives will still not always be 100% accurate. I would recommend rounding the result to 20 decimals or less.

Performance

It's pretty fast and scales well. In my case, it usually runs in under 0.1ms. The biggest performance hit is functions like log(). log(12345) evaluates in 0.12ms, and log(e) in 0.25ms.

To see how fast it is, you can pass the --verbose flag in CLI, or the verbose argument to eval().

Dev Instructions

Get started

Install Rust.

Run cpc with a CLI argument as input: cargo run -- '100ms to s'

Run in verbose mode, which shows some extra logs: cargo run -- '100ms to s' --verbose

Run tests: cargo test

Build: cargo build

Adding a unit

Nice resources for adding units: - https://github.com/ryantenney/gnu-units/blob/master/units.dat - https://support.google.com/websearch/answer/3284611 (unit list) - https://translatorscafe.com/unit-converter (unit conversion) - https://calculateme.com (unit conversion) - https://wikipedia.org

1. Add the unit

In src/units.rs, units are specified like this: ```rs pub enum UnitType { Time, // etc }

// ...

create_units!( Nanosecond: (Time, d128!(1)), Microsecond: (Time, d128!(1000)), // etc ) ```

The number associated with a unit is it's "weight". For example, if a second's weight is 1, then a minute's weight is 60.

2. Add a test for the unit

Make sure to also add a test for each unit. The tests look like this: rs assert_eq!(convert_test(1000.0, Meter, Kilometer), 1.0); Basically, 1000 Meter == 1 Kilometer.

3. Add the unit to the lexer

Text is turned into tokens (some of which are units) in lexer.rs. Here's one example: rs // ... match string { "h" | "hr" | "hrs" | "hour" | "hours" => tokens.push(Token::Unit(Hour)), // etc } // ...

Potential Improvements

Cross-compiling (from x86_64 macOS)

  1. Install Docker
  2. Install cross. cross works by installing toolchains for whatever target you're building for, then using those toolchains to compile in Docker containers cargo install cross
  3. Set rustup profile to minimal. This means things like rustdoc won't be included in new toolchain installations. You can run rustup set profile default to reset this afterwards. rustup set profile minimal
  4. Build for x86_64 macOS, Linux and Windows. For more targets, check out the targets cross supports cargo build --release --target x86_64-apple-darwin && cross build --release --target x86_64-unknown-linux-musl && cross build --release --target x86_64-pc-windows-gnu
  5. The compiled binaries will now be available inside target/<target>/release/. The filename will be either cpc or cpc.exe.

Releasing a new version

  1. Update CHANGELOG.md
  2. Bump the version number in Cargo.toml
  3. Run cargo test
  4. Cross-compile cpc by following the steps above
  5. Commit and tag in format v#.#.#
  6. Publish on crates.io:
    1. Login by running cargo login and following the instructions
    2. Test publish to ensure there are no issues cargo publish --dry-run
    3. Publish cargo publish
  7. Publish on GitHub
    1. Zip the binaries and rename them like cpc-v1.0.0-macos-x64
    2. Create GitHub release with release notes and attach the zipped binaries