diff --git a/python/README.md b/python/README.md index f1011286..b049d30f 100644 --- a/python/README.md +++ b/python/README.md @@ -11,26 +11,23 @@ [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. +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) -This software was developed for use in [LLNL/GridDyn](https://github.com/LLNL/GridDyn), and [HELICS](https://github.com/GMLC-TDC/HELICS) and is currently a work in progress (though getting close). Namespaces, function names, and code organization is subject to change though is fairly stable at this point, input is welcome. A set of [documentation](https://units.readthedocs.io/en/latest/) is available. ## Table of contents -- [Purpose](#purpose) -- [Limitations](#limitations) -- [Alternatives](#alternatives) -- [C++ Types](#types) -- [Unit Representation](#unit-representation) -- [Building The Library](#building-the-library) -- [Try it out](#try-it-out) -- [Usage](#usage) - - [Units](#unit-methods) - - [Measurements](#measurement-operations) - - [String Conversions](#available-library-functions) -- [Contributions](#contributions) -- [Users](projects-using-the-units-library) -- [Release](#release) +- [Units](#units) + - [Table of contents](#table-of-contents) + - [Purpose](#purpose) + - [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) + - [Contributions](#contributions) + - [Project Using the Units Library](#project-using-the-units-library) + - [Release](#release) ## Purpose @@ -42,145 +39,34 @@ A units library was needed to be able to represent units from a wide range of di 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 -It was desired that the unit representation be a compact type(<=8 bytes) that is typically passed by value, that can represent a wide assortment of units and arbitrary combinations of units. The primary use of the conversions is at run-time to convert user input/output to/from internal units, it is not to provide strict type safety or dimensional analysis, though it can provide some of that. This library does **NOT** provide compile time checking of units. The units library supports units and operations on units where many of the units in use are unknown at compile time and conversions and definitions are dealt with at run time, and may be of a wide variety of units. - -This library is an engineering library, created to represent a huge variety of units and measurements in a simple data type instead of a proliferation of templates. It supports conversion of units to and from strings. It supports mathematical operations on units and measurements which are `constexpr` where possible. It supports units used in a wide variety of scientific and non-scientific contexts. Supports conversions between different units of the same type as well as some typical assumptions for supporting conversions of a few dissimilar types. In some cases it also has some notion of commodities, and support for existing unit standards for strings and naming. +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. ### Basic use case 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 -```cpp -#include +```python + from units_llnl import unit + u1 = unit("m") + u2 = unit("cm") + v1 = u1.convert(10, u2) + assert v1 == 10 * 100 -double GetInputValueAs(const std::string &input, precise_units out) -{ - auto meas=measurement_from_string(input); - return meas.value_as(out); -} + v2 = u1.convert(unit_out=u2, value=20) + assert v2 == 2000 ``` -The return value can be checked for validity as an invalid conversion would result in `constants::invalid_conversion` or `Nan` so can be checked by `std::isnan` -or +```python + from units_llnl import measurement + m1 = measurement("10 m") + m2 = measurement("2.5 s") + m3 = m1 / m2 + m4 = measurement("4.0 m/s") + assert m3 == m4 -```cpp -if (!meas.units().is_convertible(out)) -{ - throw(std::invalid_argument); -} ``` -## Limitations - -- The powers represented by units by default are limited see [Unit representation](#unit-representation) and only normal physical units or common operations are supported, this can be modified at compile time to support a much broader range at the expense of size and computation. -- The library uses floating point and double precision for the multipliers which is generally good enough for most engineering contexts, but does come with the limits and associated loss of precision for long series of calculations on floating point numbers. -- Currency is supported as a unit but it is not recommended to use this for anything beyond basic financial calculations. So, if you are doing a lot of financial calculations or accounting, use something more specific for currency manipulations. It also does not maintain any notion of currency conversions since those fluctuate in value. It may at some point recognize different currency names though commodities. -- Fractional unit powers are not supported in general. While some mathematical operations on units are supported any root operations `sqrt` or `cbrt` will only produce valid results if the result is integral powers of the base units. One exception is limited support for √Hz operations in measurements of Amplitude spectral density. A specific definition of a unit representing square root of Hz is available and will work in combination with other units. -- While conversions of various temperature definitions are supported, there is no generalized support for datums and bias shifts. It may be possible to add some specific cases in the future for common uses cases but the space requirement limits such use. Some of the other libraries have general support for this. -- A few [equation](https://units.readthedocs.io/en/latest/user-guide/equation_units.html) like units are supported these include logarithms, nepers, and some things like Saffir-Simpson, Beaufort, and Fujita scales for wind, and Richter scales for earthquakes. There is capacity within the framework to add a few more equation like units if a need arises. -- There are several units where the specific definition differs when used in different domains. The unit `rad` in the nature of radiation absorbed dose is one such unit as it would conflicts with `rad` in terms of radians. So `rad` means radians by default since that is the more common use in electrical engineering. The use of [domains](https://units.readthedocs.io/en/latest/user-guide/unit_domains.html) in the conversion operations can control this to some extent. For example the 'Nuclear' domain specifies that `rad` refers to radiation absorbed dose instead of the angle. - -## Alternatives - -If you are looking for compile time and prevention of unit errors in equations for dimensional analysis one of these libraries might work for you. - -- [boost units](https://www.boost.org/doc/libs/1_79_0/doc/html/boost_units.html) -Zero-overhead dimensional analysis and unit/quantity manipulation and conversion in C++ -- [Units](https://github.com/nholthaus/units) -A compile-time, header-only, dimensional analysis library built on `C++14` with no dependencies. -- [Units](https://github.com/VincentDucharme/Units) -Another compile time library -- [PhysUnits-CT](https://github.com/martinmoene/PhysUnits-CT-Cpp11) A C++ library for compile-time dimensional analysis and unit/quantity manipulation and conversion. -- [PhysUnits-RT](https://github.com/martinmoene/PhysUnits-RT) -A C++ library for run-time dimensional analysis and unit/quantity manipulation and conversion. -- [Libunits](https://sourceforge.net/projects/libunits/) -The ultimate shared library to do calculations(!) and conversions with any units! Includes all SI and pseudo SI units and thousands of US, Imperial and other units. -- [unitscpp](http://code.google.com/p/unitscpp/) -A lightweight C++ library for physical calculation with units. -- [mpusz/units](https://github.com/mpusz/units) -A compile-time enabled Modern C++ library that provides compile-time dimensional analysis and unit/quantity manipulation. This library is in process for standardization in C++23/26. -- [bernedom/SI](https://github.com/bernedom/SI) -A header only C++ library that provides type safety and user defined literals for handling physical values defined in the International System of Units - -These libraries will work well if the number of units being dealt with is known at compile time. Many also produce zero overhead operations and checking. Therefore in situations where this is possible other libraries are a preferred alternative. - -### Reasons to choose this units library over another option - -1. Conversions to and from regular strings are required -2. The number of units in use is large -3. A specific single unit or measurement type is required to handle many different kinds of units or measurements -4. Uncertainties are needed to be included with the measurements -5. Working with per unit values -6. Dealing with commodities in addition to regular units. i.e. differentiate between a gallon of water and a gallon of gasoline -7. Dealing with equation type units -8. Complete C++ type safety is **NOT** a critical design requirement. -9. Support is needed for some funky custom unit with bizarre base units. - -### Reasons to choose something else - -1. Type safety and dimensional analysis **IS** a design requirement -2. Performance is absolutely critical (many other libraries are zero runtime overhead) -3. You are only working with a small number of known units -4. You cannot use C++11 yet. -5. You need to operate on arbitrary or general fractional powers of base units -6. You need support for arbitrary datum shifts in the unit library - -## Types - -There are only a few types in the library - -- `detail::unit_base` is the base representation of physical units and powers. It uses a bitfield to store the base unit representation in a 4-byte representation. It is mostly expected that unit_base will not be used in a standalone context but through one of other types. -- `unit` is the primary type representing a physical unit it consists of a `float` multiplier along with a `unit_base` and contains this within an 8 byte type. The float has an accuracy of around 7 decimal digits. Units within that tolerance will compare equal. -- `precise_unit` is the a more accurate type representing a physical unit it consists of a `double` multiplier along with a `unit_base` and contains this within an 16 byte type. The double has an accuracy of around 13 decimal digits. Units within that tolerance will compare equal. The remaining 4 bytes are used to contain a commodity object code. -- `measurement` is a 16 byte type containing a double value along with a `unit` and mathematical operations can be performed on it usually producing a new measurement. -- `precise_measurement` is similar to measurement except using a double for the quantity and a `precise_unit` as the units. -- `fixed_measurement` is a 16 byte type containing a double value along with a constant `unit` and mathematical operations can be performed on it usually producing a new `measurement`. The distinction between `fixed_measurement` and `measurement` is that the unit definition of `fixed_measurement` is constant and any assignments get automatically converted, `fixed_measurement`'s are implicitly convertible to a `measurement` of the same value type. fixed_measurement also support some operation with pure numbers by assuming a unit that are not allowed on regular measurement types. -- `fixed_precise_measurement` is similar to `fixed_measurement` except it uses `precise_unit` as a base and uses a double for the measurement instead of a template, and it is 24 bytes instead of 16. -- `uncertain_measurement` is similar to `measurement` except it uses a 32-bit float for the value and contains an uncertainty field which is also 32-bits. Mathematical operations on `uncertain_measurement`s will propagate the uncertainty and convert it as necessary. The class also includes functions for simple-uncertainty propagation like `simple_subtract` which just sums the uncertainties. The sum-of-squares methods is used in the overloaded math operators. Mathematical operations are supported on the type and it interoperates with measurements. - -## Unit representation - -The `unit` class consists of a multiplier and a representation of base units. -The seven [SI units](https://www.nist.gov/pml/weights-and-measures/metric-si/si-units) + radians + currency units + count units. In addition a `unit` has 4 flags, per-unit for per unit or ratio units. One flag\[i_flag\] that is a representation of imaginary units, one flags for a variety of purposes and to differentiate otherwise similar units\[e_flag\]. And a flag to indicate an equation unit. Due to the requirement that the base units fit into a 4-byte type the represented powers of the units are limited. The table below shows the bit representation range and observed range of use in equations and observed usage - -| Base Unit | Bits | Representable range | Normal Range | Intermediate Operations | -| --------- | ---- | ------------------- | ------------ | ----------------------- | -| meter | 4 | \[-8,+7\] | \[-4,+4\] | \[-6,+6\] | -| kilogram | 3 | \[-4,+3\] | \[-1,+1\] | \[-2,+2\] | -| second | 4 | \[-8,+7\] | \[-4,+4\] | \[-6,+6\] | -| ampere | 3 | \[-4,+3\] | \[-2,+2\] | | -| kelvin | 3 | \[-4,+3\] | \[-4,+1\] | | -| mole | 2 | \[-2,+1\] | \[-1,+1\] | | -| candela | 2 | \[-2,+1\] | \[-1,+1\] | | -| currency | 2 | \[-2,+1\] | \[-1,+1\] | | -| count | 2 | \[-2,+1\] | \[-1,+1\] | | -| radians | 3 | \[-4,+3\] | \[-2,+2\] | | - -These ranges were chosen to represent nearly all physical quantities that could be found in various disciplines we have encountered. See [Unit Details](https://units.readthedocs.io/en/latest/details/unit_base.html#) for additional details on the unit base representation. - -The CMake variable `UNITS_BASE_TYPE`, if set to a 64-bit type like `uint64_t`, will double the space requirements but also change the ranges to be at least a power of 4 larger than the above table. See [CMake Reference](https://units.readthedocs.io/en/latest/installation/cmake_variables.html) for more details. - -### Discussion points - -- Currency may seem like a unusual choice in units but numbers involving prices are encountered often enough in various disciplines that it is useful to include as part of a unit. -- Technically count and radians are not units, they are representations of real things. A radian is a representation of rotation around a circle and is therefore distinct from a true unitless quantity even though there are no physical measurements associated with either. -- Count and mole are theoretically equivalent though as a practical matter using moles for counts of things is a bit odd for example 1 GB of data is ~1.6605\*10^-15 mol of data. They are used in different contexts and don't mix very often, the `convert` functions do convert between them if necessary. -- This library **CANNOT** represent fractional unit powers( except for sqrt Hz used in noise density units), and the library follows the order of operation in C++ so **IF** you have equations that any portion of the operation may exceed the numerical limits on powers even if the result does not, **BE CAREFUL**. -- The normal rules about floating point operations losing precision also apply to unit representations with non-integral multipliers. -- With string conversions there are many units that can be interpreted in multiple ways. In general, the priority was given to units in more common use in the United States, or in power systems and electrical engineering which was the origin of this library. -- The unit `year` has different meanings in different contexts. SI defines the default year as `yr`=`365*day`=`8760*hr` the specific [domains](https://units.readthedocs.io/en/latest/user-guide/unit_domains.html) define it differently. 'year' means `365.25` days in the UCUM domain and the mean tropical year for the astronomy domain. -- The i_flag functions such that when squared it goes to 0, similar to the imaginary number `i*conj(i)=i^0`. This is useful for directional units such as compass directions and reactive power in power systems. The e_flag functions as an or operation in multiplication and xor operation during division. -- Measurement/unit equality is an interesting topic. The library takes a pragmatic approach vs. a precise mathematical approach. The precision of a float is taken to be roughly 7 decimal digit of precision. A double used in the 'precise' values to be 13 decimal digits of precision. This precision is sufficient to run a few operations without going out of tolerance from floating point operations. It also comes into equality, which is nominally taken to be values and units within this tolerance level. So, numbers are rounded to a certain number of digits then compared to within a tolerance level. Some effort was made to make this uniform, but tolerance around the last digit is not exact. Comparison operators for the units and measurements are provided. Equality and inequality use the rounded comparison; greater and less than are exact, while `>=` and `<=` check first for > or < conditions then check for equality if needed. There are a few situations that are not totally consistent like `1.0000001*m==1.0*m` and `1.0000001*m>1.0*m`, but such is nature of floating point operations. So, from a mathematical purity sense this isn't consistent but does mostly what was needed. If the difference between the two values is a subnormal number the equality comparison also evaluates to true, even if that would otherwise be outside the numerical tolerance. - -## Defined units - -There are 2 sets of defined units, many common units are defined in the `units` namespace, many others are defined in `units::precise` and subnamespaces. -See [Defined Units](https://units.readthedocs.io/en/latest/user-guide/defined_units.html) for details on the available units. - -### Physics constants - -A set of physical and numerical constants are defined in the `units::constants` namespace. More details and a list of available constants are described in [Physical Units](https://units.readthedocs.io/en/latest/user-guide/Physical_constants.html). Some of the available constants that are measured vs. defined have an `uncertain_measurement` version available as well that includes the uncertainty. - -## Building the library - -There are two parts of the library a header only portion that can simply be copied and used. There are 5 headers `units_decl.hpp` declares the underlying classes. `units_util.hpp` defines some additional helper functions, `unit_defintions.hpp` declares constants for many of the units, and `units.hpp` which is the primary public interface to units,`units_math.hpp` is an optional extra header that includes additional mathematical operations. If `units.hpp` is included in another file and the variable `UNITS_HEADER_ONLY` is defined then none of the functions that require the cpp files are defined. These header files can simply be included in your project and used with no additional building required. - -The second part is a few cpp files that can add some additional functionality. The primary additions from the cpp file are an ability to take roots of units and measurements and convert to and from strings. The `units_conversion_maps.hpp` file defines many of string conversions used in the converters to and from strings. These files can be built as a standalone static library or included in the source code of whatever project want to use them. The code should build with an C++11 or greater compiler. It currently defaults to build with C++14. Most of the library is tagged with constexpr so can be run at compile time to link units that are known at compile time. Unit numerical conversions are not at compile time, so will have a run-time cost. A `quick_convert` function is available to do simple conversions. with a requirement that the units have the same base and not be an equation unit. The cpp code also includes some functions for commodities and will eventually have r20 and x12 conversions, though this is not complete yet. - -It builds by default with the static library. Using `UNIT_BUILD_SHARED_LIBRARY` or `BUILD_SHARED_LIBS` will build the shared library instead. Either one can be used with CMake as a `units::units` target. The header only library target is also generated `units::header_only`. The shared/static library has a CMake target `units::units`. ## Try it out @@ -190,26 +76,6 @@ If you want to try out the string conversion components. There is server running For more details see the [documentation](https://units.readthedocs.io/en/latest/web/index.html) -### Converter Application - -A [converter](https://units.readthedocs.io/en/latest/introduction/converter.html) command line application can be built as part the units library by setting -`UNITS_BUILD_CONVERTER_APP=ON` in the CMake build. This is a simple command line script that takes a measurement entered on the command line and a unit to convert to and returns the new value by itself or part of a string output with the units either simplified or in original form. If you want to run your own converter web server, a docker container is available on [dockerhub](https://hub.docker.com/repository/docker/phlptp/units). - -## Usage - -Many units are defined as `constexpr` objects and can be used directly - -```cpp -#include "units/units.hpp" -using namespace units - -measurement length1=45.0*m; -measurement length2=20.0*m; - -measurement area=length1*length2; - -std::cout<<"the area is "< cround()` round the multiplier to an appropriate number of digits -- ` base_units()` get the base units -- `void clear_flags()` clear any flags associated with the units - -For precise_units only - `commodity()` get the commodity of the unit - `commodity(int commodity)` assign a commodity to the precise_unit. @@ -253,29 +113,6 @@ There are also several operator overloads that apply to units and precise_units. precise_units can usually operate with a `precise_unit` or `unit`, `unit` usually can't operate on `precise_unit`. -#### Unit free functions - -These functions are not class methods but operate on units - -- `std::hash()` generate a hash code of a unit, for things like use in std::unordered_map or other purposes. -- ` unit_cast()` convert a unit into , mainly used to convert a precise_unit into a regular unit. -- `bool is_unit_cast_lossless()` returns true if the multiplier in a precise_unit can be converted exactly into a float. -- `bool isnan()` true if the unit multiplier is a NaN. -- `bool isinf()` true if the unit multiplier is infinite. -- `double quick_convert(, )` generate the conversion factor between two units. This function is constexpr. -- `double quick_convert(double factor, , )` convert a specific value from one unit to another, function is constexpr but does not cover all possible conversions. -- `double convert(, )` generate the conversion factor between two units. -- `double convert(double val, , )` convert a value from one unit to another. -- `double convert(double val, , , double baseValue)` do a conversion assuming a particular basevalue for per unit conversions. -- `double convert(double val, , , double basePower, double baseVoltage)` do a conversion using base units, specifically making assumptions about per unit values in power systems. -- `bool is_error()` check if the unit is a special error unit. -- `bool is_default()` check if the unit is a special default unit. -- `bool is_valid()` check to make sure the unit is not an invalid unit( the multiplier is not a NaN) and the unit_data does not match the defined `invalid_unit`. -- `bool is_temperature()` return true if the unit is a temperature unit such as `F` or `C` or one of the other temperature units. -- `bool is_normal()` return true if the multiplier is a normal number, there is some defined unit base, not the identity unit, the multiplier is not negative, and not the default unit. basically a simple way to check if you have some non-special unit that will behave more or less how you expect it to. -- ` root(, int power)` non constexpr, take the root of a unit, produces `error` unit if the root is not well defined. power can be negative. -- ` sqrt()` convenience function for taking the sqrt of a unit. - ### Measurement Operations - `(val, )` construct a unit from a value and unit object. @@ -286,17 +123,6 @@ These functions are not class methods but operate on units - ` 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 \ -#### Uncertain measurement methods - -Uncertain measurements have a few additional functions to support the uncertainty calculations - -- `simple_add`, `simple_subtract`, `simple_product`, `simple_divide` are equivalent to the associated operator but use simple uncertainty propagation. `simple_product` and `simple_divide` are `constexpr` when compiled with C++14 or greater. The regular operators use root sum of squares propagation. -- `double uncertainty()` get the numerical value of the uncertainty. -- `float uncertainty_f()` get the numerical value of the uncertainty as a float. -- `measurement uncertainty_measurement()` get the uncertainty as a separate measurement -- `double uncertainty_as()` get the uncertainty in terms of a particular unit. -- `double fractional_uncertainty()` get the uncertainty as a fraction of the value. - #### Measurement operators There are several operator overloads which work on measurements or units to produce measurements. @@ -313,75 +139,6 @@ Notes: for regular measurements, `+` and `-` are not defined for doubles due to - `=/` - `=/` basically calling a number multiplied or divided by a `` produces a measurement, specifically `unit` produces a measurement and `precise_unit` produces a precise_measurement. -#### Measurement functions - -These free functions work on any of different measurement types. - -- `measurement measurement_cast()` convert a precise_measurement into measurement -- `fixed_measurement measurement_cast()` convert a fixed_precise_measurement or fixed_measurement into a fixed_measurement -- ` pow(, int)` generate a measurement which is a specific power of another measurement -- ` root(, int)` generate a root of a measurement -- ` sqrt()` take the square root of a measurement of any kind, the units need to have a valid root. -- `bool is_valid()` will result in true if the underlying unit is valid and the value is not a nan. -- `bool isnormal()` will result in true if the underlying unit is normal and the value is not a nan or infinity or subnormal - zero is allowed in the measurement value, but not the unit multiplier. - -### Additional math operations - -A few additional math operations are available in the `"unit_math.hpp"` header on all measurement types. This is a header only and is not included by default. It adds math operations including `ceil`,`floor`,`trunc`,`round`,`fmod`,`sin`,`cos`,`tan`. The trigonometric operations are only defined for measurements that are convertible to radians. Additionally, three type traits are defined including `is_measurement`, `is_precise_measurement` and `is_unit`. These traits are only true for defined measurement types and unit types respectively. - -### Available library functions - -#### String Conversions - -The flags argument is optional in all cases. If not specified it uses the default flags, which can be user declared at run or compile time. - -- `precise_unit unit_from_string( string, flags)`: convert a string representation of units into a precise_unit value. -- `unit unit_cast_from_string( string, flags)`: convert a string representation of units into a unit value NOTE: same as previous function except has an included unit cast for convenience. -- `precise_unit default_unit( string)`: get a unit associated with a particular kind of measurement. for example `default_unit("length")` would return `precise::m` -- `precise_measurement measurement_from_string(string,flags)`: convert a string to a precise_measurement. -- `measurement measurement_cast_from_string(string,flags)`: convert a string to a measurement calls measurement_from_string and does a measurement_cast. -- `uncertain_measurement uncertain_measurement_from_string(string,flags)`: convert a string to an uncertain measurement. Typically the string will have some segment with a `±`, `+/-` or the html equivalent in it to signify the uncertainty. The compact notation for uncertainties is also supported for example `3.5235(19)`. -- `std::string to_string([unit|measurement],flags)` : convert a unit or measurement to a string, all defined units or measurements listed above are supported. The eventual plan is to support a couple different standards for the strings through the flags, But for now they don't do much. - -For more description of the possible flags see [flags](https://units.readthedocs.io/en/latest/user-guide/conversion_flags.html). The default flags can be set through `setDefaultFlags(std::uint32_t flags)` and retrieved through `getDefaultFlags()`. The initial default flag is `OU` but can be modified through `UNITS_DEFAULT_MATCH_FLAGS` compile flag. - -#### User Defined Units - -- `addUserDefinedUnit(std::string name, precise_unit un)` add a new unit that can be used in the string operations. -- `clearUserDefinedUnits()` remove all user defined units from the library. -- `disableUserDefinedUnits()` there is a(likely small-an additional unordered map lookup) performance hit in the string conversions functions if custom units are used so they can be disabled completely if desired. -- `enableUserDefinedUnits()` enable the use of UserDefinedUnits. they are enabled by default. -- `addUserDefinedInputUnit(std::string name, precise_unit un)` add a new unit used only for unit input -- `addUserDefinedOutputUnit(std::string name, precise_unit un)` add an output string for a specific unit -- `removeUserDefinedUnit(std::string name)` remove a specific unit string previously added as a userDefinedUnit can be input, output, or both. - -For more details see [User Defined Units](https://units.readthedocs.io/en/latest/user-guide/user_defined_units.html). - -#### Unit Domains - -- `setUnitsDomain(std::uint32_t newDomain)` : set a default domain to use for string translations if not overridden using the `flags` argument. The function returns the previous domain if the domain only needs to be set temporarily. - -For more description of the Unit Domains supported see [Domains](https://units.readthedocs.io/en/latest/user-guide/Unit_domains.html). Use the constants available in `units::domains` as the argument. The numerical value is subject to change in future releases as this gets refined. - -#### Commodities - -The units library has some support for commodities, more might be added in the future. Commodities are supported in precise_units. - -- `std::uint32_t getCommodity(std::string commodity)` get a commodity code from a string. -- `std::string getCommodityName(std::uint32_t)` get the name of a commodity from its code -- `addUserDefinedCommodity(std::string name, std::uint32_t code)` add a new commodity that can be used in the string operations. -- `clearUserDefinedCommodities()` remove all user defined commodities from the library. -- `disableUserDefinedCommodities()` there is a (likely small) performance hit in string conversions if custom commodities are used so they can be disabled completely if desired. -- `enableUserDefinedCommodities()` enable the use of UserDefinedCommodities. User defined commodities are enabled by default. Defining user specified commodities is thread-safe. - -#### Other unit definitions - -These are all only partially implemented, not recommended for use yet - -- `precise_unit x12_unit(string)` get a unit from an X12 string. -- `precise_unit dod_unit(string)` get a unit from a DOD code string. -- `precise_unit r20_unit(string)` get a unit from an r20 code string. - ## Contributions Contributions are welcome. See [Contributing](./CONTRIBUTING.md) for more details and [Contributors](./CONTRIBUTORS.md) for a list of the current and past Contributors to this project.