diff --git a/CHANGELOG.md b/CHANGELOG.md index e4dc852b..3c2aadcb 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,6 +5,49 @@ All notable changes to this project after the 0.2.0 release will be documented i The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/). This project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). +## [0.11.0][] -2024-12-25 + +Python package release, and other updates and fixes + +### Changed + +- Updated cmake and other dependencies [#342][] +- Updated copyright date to 2024 [#351][] +- updated the r20 units to be mostly operational [#314][] +- Updated third party libraries, and some new CI builders [#335][],[#336][] + +### Fixed + +- fixed some issues with windows.h header conflicts [#345][] +- fixed an issue using CMAKE_CXX_STANDARD [#339][] +- fixed some missing test coverage to 100% [#316][],[#317][],[#318][] + +### Added + +- Added Python Library [#349][],[#350][] +- Added some additional equation units and tests [#315][] +- Added a number of additional units string [#322][],[#331][] +- added unary operation to measurement classes [#327][] + +[#314]: https://github.com/LLNL/units/pull/314 +[#315]: https://github.com/LLNL/units/pull/315 +[#316]: https://github.com/LLNL/units/pull/316 +[#317]: https://github.com/LLNL/units/pull/317 +[#318]: https://github.com/LLNL/units/pull/318 +[#322]: https://github.com/LLNL/units/pull/322 +[#331]: https://github.com/LLNL/units/pull/331 +[#335]: https://github.com/LLNL/units/pull/335 +[#336]: https://github.com/LLNL/units/pull/336 +[#339]: https://github.com/LLNL/units/pull/339 +[#342]: https://github.com/LLNL/units/pull/342 +[#349]: https://github.com/LLNL/units/pull/349 +[#350]: https://github.com/LLNL/units/pull/350 +[#351]: https://github.com/LLNL/units/pull/351 + +## [0.10.2][] -2024-12-22 + +Prototype release for python see all changes in 0.11.0 + ## [0.9.2][] - 2024-05-10 update some dependencies and other minor fixes @@ -83,6 +126,13 @@ A few user suggested tweaks, and support additional unit string conversions supp [#299]: https://github.com/LLNL/units/pull/299 [#303]: https://github.com/LLNL/units/pull/303 + +[0.9.0]: https://github.com/LLNL/units/releases/tag/v0.9.0 +[0.9.1]: https://github.com/LLNL/units/releases/tag/v0.9.1 +[0.9.2]: https://github.com/LLNL/units/releases/tag/v0.9.2 +[0.10.2]: https://github.com/LLNL/units/releases/tag/v0.10.2 +[0.11.0]: https://github.com/LLNL/units/releases/tag/v0.11.0 + ## [0.7.0][] - 2022-12-17 Added several math operations for units, restored coverage to 100%, added uncertain constants and cleaner string generation and interpretation around the use of '.' and uncertain measurements, and added support for new SI prefixes. diff --git a/pyproject.toml b/pyproject.toml index 059de66c..8617500a 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -4,7 +4,7 @@ build-backend = "scikit_build_core.build" [project] name = "units_llnl" -version = "0.10.2.post2" +version = "0.11.0" description = "Python bindings for the LLNL units library" readme = "python/README.md" requires-python = ">=3.10" diff --git a/python/CMakeLists.txt b/python/CMakeLists.txt deleted file mode 100644 index 39e8129a..00000000 --- a/python/CMakeLists.txt +++ /dev/null @@ -1,2 +0,0 @@ -cmake_minimum_required(VERSION 3.22...3.31) -project(llnl-units LANGUAGES CXX) diff --git a/python/README.md b/python/README.md index b8d6770e..5a9cfc54 100644 --- a/python/README.md +++ b/python/README.md @@ -11,7 +11,7 @@ [Documentation](https://units.readthedocs.io/en/latest/) -The Units library provides a means of working with units of measurement at runtime, including conversion to and from strings. It provides a small number of types for working with units and measurements and operations necessary for user input and output with units. For additional description and discussion see [Readme](https://github.com/LLNL/units/blob/main/README.md) +The Units library provides a means of working with units of measurement at runtime, including conversion to and from strings. It provides a small number of types for working with units and measurements and operations necessary for user input and output with units. For additional description and discussion see [Readme](https://github.com/LLNL/units/blob/main/README.md). The Python library is a wrapper around the C++ library using [nanobind](https://github.com/wjakob/nanobind). ## Table of contents @@ -21,9 +21,15 @@ The Units library provides a means of working with units of measurement at runti - [Basic use case](#basic-use-case) - [Try it out](#try-it-out) - [Unit methods](#unit-methods) - - [Unit Operators](#unit-operators) - - [Measurement Operations](#measurement-operations) - - [Measurement operators](#measurement-operators) + - [Constructors](#constructors) + - [Methods](#methods) + - [Operators](#operators) + - [Measurements](#measurements) + - [Constructors](#constructors-1) + - [Methods](#methods-1) + - [Operators](#operators-1) + - [Other library methods](#other-library-methods) + - [Future plans](#future-plans) - [Contributions](#contributions) - [Project Using the Units Library](#project-using-the-units-library) - [Release](#release) @@ -36,7 +42,6 @@ A units library was needed to be able to represent units from a wide range of di 2. Being able to use the unit as a singular type that could contain any unit, and not introduce a huge number of types to represent all possible units. 3. Being able to associate a completely arbitrary unit given by users with a generic interface and support conversions between those user defined units and other units. 4. The library has its origins in power systems so support for per-unit operations was also lacking in the alternatives. -5. Capture uncertainty and uncertainty calculations directly with a measurement The python wrapper around the library is mainly intended to be able to handle various string representations and easily handle conversions, along with some support for commodities and packaging. @@ -45,10 +50,10 @@ The python wrapper around the library is mainly intended to be able to handle va The primary use case for the library is string operations and conversion. For example if you have a library that does some computations with physical units. In the library code itself the units are standardized and well defined. For example take a velocity, internally everything is in meters per second, but there is a configuration file that takes in the initial data and you would like to broadly support different units on the input ```python -from units_llnl import unit +from units_llnl import Unit -u1 = unit("m") -u2 = unit("cm") +u1 = Unit("m") +u2 = Unit("cm") v1 = u1.convert(10, u2) assert v1 == 10 * 100 @@ -57,12 +62,12 @@ assert v2 == 2000 ``` ```python -from units_llnl import measurement +from units_llnl import Measurement -m1 = measurement("10 m") -m2 = measurement("2.5 s") +m1 = Measurement("10 m") +m2 = Measurement("2.5 s") m3 = m1 / m2 -m4 = measurement("4.0 m/s") +m4 = Measurement("4.0 m/s") assert m3 == m4 ``` @@ -76,65 +81,88 @@ For more details see the [documentation](https://units.readthedocs.io/en/latest/ ### Unit methods -These operations apply to units and precise_units - -- `()` construct from a base unit_data -- `(, double multiplier)` construct a unit from a base data and a multiplier -- `(double multiplier, )` construct from a multiplier and another unit -- also available are copy constructor and copy assignments -- ` inv()` generate a new unit containing the inverse unit `m.inv()= 1/m` -- ` pow(int power)` take a unit to power(NOTE: beware of limits on power representations of some units, things will always wrap so it is defined but may not produce what you expect). `power` can be negative. -- `bool is_exactly_the_same()` compare two units and check for exact equivalence in both the unit_data and the multiplier, NOTE: this uses double equality -- `bool has_same_base(|)` check if the is the same -- `equivalent_non_counting(|)` check if the units are equivalent ignoring the counting bases -- `bool is_convertible()` check if the units are convertible to each other, currently checks `equivalent_non_counting()`, but some additional conditions might be allowed in the future to better match convert. -- `int unit_type_count()` count the number of unit bases used, (does not take into consideration powers, just if the dimension is used or not. -- `bool is_per_unit()` true if the unit has the per_unit flag active -- `bool is_equation()` true if the unit has the equation flag active -- `bool has_i_flag()` true if the i_flag is marked active -- `bool has_e_flag()` true if the e_flag is marked active -- `double multiplier()` return the unit multiplier as a double - -- `commodity()` get the commodity of the unit -- `commodity(int commodity)` assign a commodity to the precise_unit. - -#### Unit Operators - -There are also several operator overloads that apply to units and precise_units. - -- `=*` generate a new unit with the units multiplied ie `m*m` does what you might expect and produces a new unit with `m^2` -- `=/` generate a new unit with the units divided ie `m/s` does what you might expect and produces a new unit with meters per second. NOTE: `m/m` will produce `1` it will not automatically produce a `pu` though we are looking at how to make a 'pu_m\*m=m' so units like strain might work smoothly. - -- `bool ==` compare two units. this does a rounding compare so there is some tolerance to roughly 7 significant digits for \ and 13 significant digits for . -- `bool !=` the opposite of `==` - -precise_units can usually operate with a `precise_unit` or `unit`, `unit` usually can't operate on `precise_unit`. - -### Measurement Operations - -- `(val, )` construct a unit from a value and unit object. -- `double value() const` get the measurement value as a double. -- ` convert_to() const` convert the value in the measurement to another unit base -- ` convert_to_base() const` convert to a base unit, i.e. a unit whose multiplier is 1.0 -- ` units() const` get the units used as a basis for the measurement -- ` as_unit() const` take the measurement as is and convert it into a single unit. For Examples say a was 10 m. calling as_unit() on that measurement would produce a unit with a multiplier of 10 and a base of meters. -- `double value_as()` get the value of a measurement as if it were measured in \ - -#### Measurement operators - -There are several operator overloads which work on measurements or units to produce measurements. - -- `'*', '/', '+','-'` are all defined for mathematical operations on a measurement and produce another measurement. -- `%` `*`, and `/` are defined for \\\ -- `*`, and `/` are defined for \\\ - -Notes: for regular measurements, `+` and `-` are not defined for doubles due to ambiguity of what that operation means. For `fixed_measurement` types this is defined as the units are known at construction and cannot change. For `fixed_measurement` types if the operator would produce a new measurement with the same units it will be a fixed measurement, if not it reverts to a regular measurement. - -- `==`, `!=`, `>`, `<`, `>=`, `<=` are defined for all measurement comparisons -- `=*` -- `=*` -- `=/` -- `=/` basically calling a number multiplied or divided by a `` produces a measurement, specifically `unit` produces a measurement and `precise_unit` produces a precise_measurement. +These operations apply the `Units` object in Python. It maps to a `precise_unit` in C++. The Unit object is immutable like a python string so a new one is created for methods that modify the unit in some way. + +#### Constructors +- `Unit(unit_str:str)` construct from a string +- `Unit(unit_str:str,commodity_str:str)` construct a unit from a unit string and commodity string + +#### Methods +- `inv()->Unit` generate a new unit containing the inverse unit `Unit('m').inv()== Unit('1/m')` +- `pow(int power)->Unit` take a unit to power(NOTE: beware of limits on power representations of some units, things will always wrap so it is defined but may not produce what you expect). `power` can be negative. +- `is_exactly_the_same(other:Unit)->bool` compare two units and check for exact equivalence in both the unit_data and the multiplier +- `has_same_base(other:Unit)->bool` check if the units have the same base units +- `equivalent_non_counting(other:Unit)->bool` check if the units are equivalent ignoring the counting bases +- `is_convertible_to(other:Unit)->bool` check if the units are convertible to each other, currently checks `equivalent_non_counting()`, but some additional conditions might be allowed in the future to better match convert. +- `convert(value:float,unit_out:Unit|str)->float` convert a value from the existing unit to another, can also be a string +- `is_per_unit()->bool` true if the unit has the per_unit flag active +- `is_equation()->bool` true if the unit has the equation flag active +- `is_valid()->bool` true if the unit is a valid unit +- `is_normal()->bool` true if the unit is a normal unit (not error, nan, or subnormal) +- `is_error()->bool` true if the unit is an error unit (e.g invalid conversion) +- `isfinite()->bool` true if the unit does not have an infinite multiplier +- `isinf()->bool` true if the unit does have an infinite multiplier +- `root(power:int)->Unit` return a new unit taken to the root power +- `sqrt()->Unit` returns a new unit which is the square root of the current unit +- `to_string()->str` returns the string representation of the unit. This string is guaranteed to produce the same unit as the current unit, but may not be the same string as was used to create it. +- `multiplier()->float` return the unit multiplier as a floating point number +- `set_multiplier(mult:float)->Unit` generate a new Unit with the set multiplier +- `commodity()->str` get the commodity of the unit +- `set_commodity(int commodity)` generate a new unit with the assigned commodity. + +#### Operators + +- `*`,`/` with other units produces a new unit +- `**` is an exponentiation operator and produces a new unit +- `*`, `/` with a floating point generates a `Measurement` +- `==` and `!=` produce the appropriate comparison operators +- f string formatting also works with units + + +### Measurements + +#### Constructors +- `Measurement(measurement_str:str)` construct from a string +- `Measurement(value:float, unit:Unit|str)` construct a `Measurement` from a value and a `Unit` or string representing a `Unit` + +#### Methods +- `inv()->Unit` generate a new unit containing the inverse unit `Unit('m').inv()== Unit('1/m')` +- `pow(int power)->Unit` take a unit to power(NOTE: beware of limits on power representations of some units, things will always wrap so it is defined but may not produce what you expect). `power` can be negative. +- `is_normal()->bool` true if the unit is a normal unit (not error, nan, or subnormal) +- `is_valid()->bool` true if the `Measurement` is a valid Measurement (not error) +- `root(power:int)->Measurement` return a new unit taken to the root power +- `sqrt()->Unit` returns a new unit which is the square root of the current unit +- `to_string()->str` returns the string representation of the `Measurement`. This string is guaranteed to produce the equivalent `Measurement` as the current `Measurement`, but may not be the same string as was used to create it. +- `value()->float` return the numerical portion of a `Measurement` +- `set_value(value:float)->Measurement` generate a new `Measurement` with the new Value +- `units()->Unit` get the `Unit` associated with a `Measurement` +- `set_units(unit:Unit|str)` generate a new `Measurment` with the new units +- `value_as(unit:Unit|str)->float` convert the value of the `Measurement` to a new `Unit` +- `convert_to(unit:Unit|str)->Measurement` create a new `Measurement` with the new units and the value converted to those units +- `convert_to_base()->Measurement` create a new `Measurement` with the units as the base measurement units +- `is_close(other:Measurement)->bool` return true if the two measurements are close (both converted to non precise measurement and compared) + +#### Operators + +- `*`,`/` with other `Measurements` produces a new Measurement +- `+`,`-` with other `Measurements` ensures the units are in the same base unit and performs the appropriate action +- `**` is an exponentiation operator and produces a new `Measurement` +- `*`, `/` with a floating point generates a `Measurement` +- `==`,`!=`,`>`,`<`,`>=`,`<=` produce the appropriate comparison operators +- f string formatting also works with units + + +### Other library methods + +- `convert(value:float,unit_in:Unit|str,unit_out:Unit|str)->float` generate a value represented by one unit in terms of another +- `convert_pu(value:float,unit_in:Unit|str,unit_out:Unit|str, base:float)->float` "generate a value represented by one unit in terms of another if one of the units is in per-unit, the base_value is used in part of the conversion" +- `default_unit(unit_type:str)->Unit` generate a unit used for a particular type of measurement +- `add_user_defined_unit(unit_name|str,unit_definition:str|Unit)` add a custom string representing a particular unit to use in future string translations +- `add_units_from_file(file|str)` inject a list of user defined units from a file + +### Future plans + +Uncertain measurements will likely be added, along with some math operations on measurements (floor, ceil, round, etc). Also some more commodity operations, and x12 and r20 unit types. ## Contributions diff --git a/python/units_python.cpp b/python/units_python.cpp index e21742ae..a0f9bbee 100644 --- a/python/units_python.cpp +++ b/python/units_python.cpp @@ -132,6 +132,7 @@ NB_MODULE(units_llnl_ext, mod) "unit_out"_a, "value represented by one unit in terms of another") .def("is_per_unit", &units::precise_unit::is_per_unit) + .def("is_equation", &units::precise_unit::is_equation) .def( "is_valid", [](const units::precise_unit& type) { @@ -239,14 +240,14 @@ NB_MODULE(units_llnl_ext, mod) [](const units::precise_measurement& measurement, const units::precise_unit& unit) { return measurement.convert_to(unit); - }) + },"create a new `Measurement` with the new units and the value converted to those units") .def( "convert_to", [](const units::precise_measurement& measurement, const char* units) { return measurement.convert_to( units::unit_from_string(std::string(units))); - }) + },"create a new `Measurement` with the new units and the value converted to those units") .def( "convert_to_base", &units::precise_measurement::convert_to_base, @@ -276,12 +277,12 @@ NB_MODULE(units_llnl_ext, mod) "is_valid", [](const units::precise_measurement& measurement) { return units::is_valid(measurement); - }) + },"true if the `Measurement` is a valid Measurement (not error)") .def( "is_normal", [](const units::precise_measurement& measurement) { return units::isnormal(measurement); - }) + },"true if the unit is a normal unit(not error, nan, or subnormal)") .def( "root", [](const units::precise_measurement& measurement, int root) { @@ -298,7 +299,7 @@ NB_MODULE(units_llnl_ext, mod) const units::precise_measurement& measurement2) { return units::measurement_cast(measurement1) == units::measurement_cast(measurement2); - }) + },"return true if the two measurements are close (both converted to non precise measurement and compared)") .def( "__repr__", [](const units::precise_measurement& measurement) { @@ -318,7 +319,7 @@ NB_MODULE(units_llnl_ext, mod) "value"_a, "unit_in"_a, "unit_out"_a, - "value represented by one unit in terms of another"); + "generate a value represented by one unit in terms of another"); mod.def( "convert", [](double val, const char* unitIn, const char* unitOut) { @@ -367,7 +368,12 @@ NB_MODULE(units_llnl_ext, mod) "get the default unit to use for a particular type of measurement"); mod.def( "add_user_defined_unit", - &units::addUserDefinedUnit, + [](const char* unit_name, const units::precise_unit &unit_definition) { + units::addUserDefinedUnit( + std::string(unit_name), + unit_definition); + }, "unit_name"_a, + "unit_definition"_a, "add a custom string to represent a user defined unit"); mod.def( "add_user_defined_unit", @@ -375,7 +381,8 @@ NB_MODULE(units_llnl_ext, mod) units::addUserDefinedUnit( std::string(unit_name), units::unit_from_string(std::string(unit_definition))); - }, + }, "unit_name"_a, + "unit_definition"_a, "add a custom string to represent a user defined unit"); mod.def( "defined_units_from_file",