/unitility

Unitility: The Physics Units of Measure Library for Java with Spring Boot and Quarkus support for web applications development. Immutable, thread-safe, content rich, easy to use.

Primary LanguageJavaMIT LicenseMIT

UNITILITY - The Physics Unit Conversion Solution for Java

Introducing UNITILITY - the Java library designed to simplify physics units conversion. With a wide range of value objects that represent commonly used physical quantities, this solution is built using plain Java for optimal speed and lightweight functionality. Whether you're looking to convert units within the same type or customize to meet your specific needs, UNITILITY offers quick and easy usage in your project, without any heavy frameworks, or external libraries.

- Thread-Safe Architecture: Developed to ensure thread safety, allowing for concurrent access without compromising data integrity through the utilization of immutable objects.
- Kotlin and Groovy friendly: Developed to take some advantages of Kotlin and Groovy features, such as overloaded operators.

Unitility   Maven Central   Build And Test
Vulnerabilities   Security Rating   Quality Gate Status   Coverage  

AUTHOR: Piotr Jazdzyk, MSc Eng
LINKEDIN: https://www.linkedin.com/in/pjazdzyk

TABLE OF CONTENTS

  1. Installation
  2. Tech & dependencies
  3. Supported physical quantities and units
  4. Usage and functionality
    4.1 Working with physical quantities
    4.2 Parsing quantities from string
    4.3 Logic operations
    4.4 Arithmetic operations
    4.5 Equality and sorting
  5. Unitility extension modules
    5.1 Jackson serializers / deserializers
    5.2 SpringBoot web extension
    5.3 Quarkus web extension
    5.4 Jakarta Validation
  6. Creating custom units and quantities
    6.1 Custom unit
    6.2 Registering custom quantity in Spring
    6.3 Registering custom quantity in Quarkus
  7. Compatibility with other JVM languages
    7.1 Groovy
    7.2 Kotlin
  8. Special types: geographic
    8.1 Latitude, Longitude, GeoCoordinate
    8.2 GeoDistance (Haversine equations)
    8.3 Parsing geographic quantities and JSON structures
  9. Collaboration, attribution and citation
  10. Acknowledgments

1. INSTALLATION

Copy the Maven dependency provided below to your pom.xml file, and you are ready to go. For other package managers, check maven central repository: UNITILITY.

<dependency>
    <groupId>com.synerset</groupId>
    <artifactId>unitility-core</artifactId>
    <version>2.4.2</version>
</dependency>

If you use frameworks to develop web applications, it is recommended to use Unitility extension modules, to provide PhysicalQuantity instances serialization and deserialization including path and query parameters.

Extension for the Spring Boot framework:

<dependency>
    <groupId>com.synerset</groupId>
    <artifactId>unitility-spring</artifactId>
    <version>2.4.2</version>
</dependency>

Extension for the Quarkus framework:

<dependency>
    <groupId>com.synerset</groupId>
    <artifactId>unitility-quarkus</artifactId>
    <version>2.4.2</version>
</dependency>

Extensions include CORE module, so you don't have to put it separate in your pom. See section on Spring or Quarkus for better explanation. Unitility has also provided BOM which can be used for dependency management.

2. TECH AND DEPENDENCIES

Unitility is developed using the following technologies:

Core:
image   image   image  

Extension modules:
image   image  

CI/CD:
image   image  

The Key concept of the core module is to use plain Java, to minimize any issues with compatibility, conflicting transitive dependencies or excessive dependency version management.

3. SUPPORTED PHYSICAL QUANTITIES AND UNITS

At the current level of development, the following units are listed below. Each quantity includes the most popular SI units and at least one Imperial unit.

COMMON:

  • Distance: meter [m], centimetre [cm], millimetre [mm], kilometre [km], mile [mi], nautical mile [nmi], feet [ft], inch [in]
  • Area: square meter [m²], square kilometre [km²], square centimetre [cm²], square millimetre [mm²], are [a], hectare [ha], square inch [in²], square foot [ft²], square yard [yd²], acre [ac], square mile [mi²]
  • Volume: cubic meter [m³], cubic centimetre [L], liter [L], hectolitre [hL], millilitre [mL], ounce [fl.oz], pint [pt], gallon [gal]
  • Mass: kilogram [kg], gram [g], milligram [mg], tonne [t], ounce [oz], pound [lb]
  • Angle: degrees [°], radians [rad]

MECHANICAL:

  • Force: newton [N], kilonewton [kN], kilopond [kp], dyne [dyn], pound force [lbf], poundal [pdl]
  • Momentum: Kilogram meter per second [kg·m/s], Gram centimeter per second [g·cm/s], Pound feet per second [lb·ft/s]
  • Torque: Newton meter [N·m], Millinewton meter [mN·m], Kilopond meter [kp·m], Foot pound [ft·lb], Inch pound [in·lb]

THERMODYNAMIC:

  • Temperature: Kelvin [K], Celsius [°C], Fahrenheit [°F]
  • Pressure: Pascal [Pa], Hectopascal [hPa], Megapascal [MPa], Bar [bar], Milli Bar [mbar] PSI [psi], Torr [Torr]
  • Energy: Joule [J], Milli joule [mJ], Kilojoule [kJ], Megajoule [MJ], BTU [BTU], Calorie [cal], Kilocalorie [kcal], Watt hour [Wh], Kilowatt hour [kWh]
  • Power: Watt [W], Kilowatt [kW], Megawatt [MW], BTU/hour [BTU/h], Horse Power [HP]
  • Specific heat: Joules per kilogram Kelvin [J/(kg·K)], Kilojoules per kilogram Kelvin [kJ/(kg·K)], BTU per pound Fahrenheit [BTU/(lb·°F)]
  • Density: Kilogram per cubic meter [kg/m³], Pound per cubic foot [lb/ft³], Pounds per cubic inch [lb/in³]
  • Dynamic viscosity: Kilogram per meter second [kg/(m·s)], Pascal second [Pa·s], Poise [P]
  • Kinematic viscosity: Square meter per second [m²/s], Square foot per second [ft²/s]
  • Specific enthalpy: Joule per kilogram [J/kg], Kilojoule per kilogram [kJ/kg], BTU per pound [BTU/lb]
  • Thermal conductivity: Watts per meter Kelvin [W/(m·K)], Kilowatts per meter Kelvin [kW/(m·K)], BTU per hour foot Fahrenheit [BTU/(h·ft·°F)]
  • Thermal diffusivity: Square meter per second [m²/s], Square feet per second [ft²/s]

FLOWS:

  • Mass flow: Kilogram per second [kg/s], Kilogram per hour [kg/h], Tonne per hour [t/h], Pound per second [lb/s],
  • Volumetric flow: Cubic meters per second [m³/s], Cubic meters per minute [m³/min], Cubic meters per hour [m³/h], Litre per second [L/s], Litre per minute [L/min], Litre per hour [L/h],
  • Gallons per second [gal/s], Gallons per minute [gal/min], Gallons per hour [gal/h]

HUMID AIR SPECIFIC:

  • Humidity ratio: Kilogram per kilogram [kg/kg], Pound per pound [lb/lb]
  • Relative humidity: Percent [%], Decimal [-]

DIMENSIONLESS:

  • Grashof number, Prandtl number, Reynolds number, Bypass factor

SPECIAL TYPES:

Geographic:

  • Latitude: degrees [°], radians [rad]
  • Longitude: degrees [°], radians [rad]
  • GeoCoordinate: [latitude, longitude]
  • GeoDistance: meter [m], kilometer [km], mile [mi], nautical mile [nmi]

All Geographic quantities can be constructed from DMS format (degrees-minutes-seconds), for i.e.: 20°7'22.8"S.

4. USAGE AND FUNCTIONALITY

4.1 Working with physical quantities

PhysicalQuantity of specific type is an aggregate of value and unit with embedded converters. Unitility supports all logic operations and arithmetic transformations. It was decided to use a double type for value as for most engineering purposes BigDecimal level of accuracy is not required.

Creating quantities:
Below is a simple example of how to work with units and convert property from one unit to another or to convert as doubles for further calculations:

// Creating temperature instance of specified units
Temperature temperature = Temperature.ofCelsius(20.5);                  // {20.50 °C}
// Getting converted value for calculations
double valueInCelsius = temperature.getInCelsius();                     // 20.50 °C
double valueInKelvins = temperature.getInKelvins();                     // 293.65 K
double valueInFahrenheits = temperature.getInFahrenheits();             // 68.90 °F
// Checking property current unit, value, and value in base unit
TemperatureUnit unit = temperature.getUnitType();                       // CELSIUS
TemperatureUnit baseUnit = unit.getBaseUnit();                          // KELVIN 
double valueInUnit = temperature.getValue();                            // 20.50 °C
double valueInBaseUnits = temperature.getBaseValue();                   // 293.65 K
// Changing property unit and converting back to base unit
Temperature tempInFahrenheits = temperature.toUnit(TemperatureUnits.FAHRENHEIT);   // {68.90 °F}
Temperature tempInKelvins = temperature.toBaseUnit();                              // {293.65 K}

All quantities have smart toEngineeringFormat() method, which will always adjust values decimal precision to capture by default specified relevant digits depending on your unit type and its value. This way you have guaranteed an elegant output without any additional effort of reformatting. Values will be rounded up using HALF_EVEN approach.

Distance bigDistance = Distance.ofMeters(10);
Distance smallDistance = Distance.ofMeters(0.000123678);
String bigOutput = bigDistance.toEngineeringFormat(3);      // outputs: 10 [m]
String smallOutput = smallDistance.toEngineeringFormat(3);  // outputs: 0.000124 [m]

Curious how Unitility integrates with a real project? Explore the hvac-engine project, a comprehensive library tailored for HVAC engineers, offering advanced capabilities for psychrometrics and thermodynamic analysis of humid air. This project served as the driving force behind the development of the library, showcasing its proficiency in addressing the complex needs of HVAC professionals.

For a more specific example illustrating the integration of this library, take a look at the unitility-example-project. This project focuses on a dry air property physics app and demonstrates the library's functionality in a functional programming style using the io.vavr library.

4.2 Parsing quantities from string

The physical quantity can be instantiated from a string representing the commonly used engineering style of writing values with units: "{value}[{unit}]", for example, "20.5 [K]", but it will also accept input without square brackets. To parse a valid string into a PhysicalQuantity, you need to obtain an instance of the parsing factory provided in the core module. The default parsing factory includes parsers for all supported physical quantities and their related units.

// Create default parsing registry
PhysicalQuantityParsingFactory parsingFactory = PhysicalQuantityParsingFactory.getDefaultParsingFactory();
// Examples of string in engineering format with unit in square brackets
String k1 = "15.1 [W p mxK)]";
String k2 = "15.1 [W/(m.K)]";
String k3 = "   1 5 .  1 [   WpmK   ]";
String k4 = "15.1 W/mK";
String k5 = "15.1"; // This will be resolved to hard-coded default unit of W/mK
// All above strings are properly resolved to Thermal Conductivity, even partially malformed k3.
ThermalConductivity thermCond1 = parsingFactory.parse(ThermalConductivity.class, k1); // {15.1 W/(m·K)}
ThermalConductivity thermCond2 = parsingFactory.parse(ThermalConductivity.class, k2); // {15.1 W/(m·K)}
ThermalConductivity thermCond3 = parsingFactory.parse(ThermalConductivity.class, k3); // {15.1 W/(m·K)}
ThermalConductivity thermCond4 = parsingFactory.parse(ThermalConductivity.class, k4); // {15.1 W/(m·K)}
ThermalConductivity thermCond5 = parsingFactory.parse(ThermalConductivity.class, k5); // {15.1 W/(m·K)}

Parsers are designed to interpret the numeric part as the value and the content within square brackets as the unit symbol. Parsers allow for a limited deviation in input style, as illustrated in the example below. The table presents alternative ways of expressing units in an input string:

NAME DEF ALT EXAMPLES
decimal separator . , 20.1 or 20,1
degrees symbol ° o, deg 20°, 20o, 20deg
multiplication · x or . Pa·m, Pa x m, Pa.m
division / p m/s, mps

Please note that this method of creating quantities is designed to be used for deserializers.
In your code, you should create units in a programmatic way, not parsing from strings.

IMPORTANT: When parsing from string values should be provided using dot "." as decimal separator.
DO NOT USE grouping separators.

This will parse properly:

1000000.00 [Pa] -> OK

This also ill parse properly:

1_000_000.00 [Pa] -> OK

But this will not:

1,000,0000.00 [Pa] -> will FAIL

4.3 Logical operations

  • basic logic operations
Temperature smallerTemp = Temperature.ofCelsius(-20.0);             
Temperature greaterTemp = Temperature.ofCelsius(0.0);               
Temperature smallerOrEqualTemp = Temperature.ofCelsius(-20.0);      
Temperature greaterOrEqualTemp = Temperature.ofCelsius(0.0);        
smallerTemp.isLowerThan(greaterTemp);                            // is true   
greaterTemp.isGreaterThan(smallerTemp);                          // is true
smallerTemp.isEqualOrLowerThan(smallerOrEqualTemp);              // is true
greaterTemp.isEqualOrGreaterThan(greaterOrEqualTemp);            // is true

Only quantities of the same type can be compared. When conducting a comparison, the algorithm will harmonize the target unit with the source unit to ensure a valid comparison.

4.4 Arithmetic operations

The developer's choice determines whether calculations are performed using double values or with physical quantity objects through the available transformation methods:

  • adding or subtracting quantities of the same type:
Temperature sourceTemperature = Temperature.ofCelsius(20);
Temperature temperatureToAdd = Temperature.ofKelvins(20 + 273.15);
Temperature actualTemperature = sourceTemperature.add(temperatureToAdd); // results in: 40 °C

Performing addition or subtraction will yield a PhysicalQuantity with the same unit. The algorithm will convert the unit of the addend quantity to match that of the augend, ensuring that the operation is conducted within a consistent unit. The unit of the resulting quantity will be set to match that of the augend.

  • multiply or divide quantities by scalar:
Temperature temperature = Temperature.ofCelsius(20);
Temperature actualTemperature = temperature.multiply(2); // results in 40 °C
  • multiply or divide quantities by different quantity with a result as double:
Temperature sourceTemperature = Temperature.ofCelsius(20);
Pressure pressure = Pressure.ofPascal(2);
double actualDivideResult = sourceTemperature.divide(pressure); // results in 40 °C

In the provided example, division and multiplication both yield a double value. This outcome arises from the current absence of a unit resolver within the developmental stage. The existing unit values are directly employed for multiplication or division operations.

  • natural logarithm, logarithm with base of 10
Distance roadLength = Distance.ofKilometers(100);
Distance reducedDistance1 = roadLength.log();      // 4.60517(..)
Distance reducedDistance2 = roadLength.log10();    // 2.0
  • trigonometric functions: sin(), cos(), tan(), ctg() including hyperbolic and reversed
Angle exampleAngle = Angle.ofDegrees(90);
double sinResult = exampleAngle.sin();       // 1
double cosResult = exampleAngle.cos();       // 0

Angle angle45 = exampleAngle.withValue(45);
double tanResult = angle45.tan();            // 1
double cotResult = angle45.cot();            // 1

Angle angleRad1 = exampleAngle.toRadians().withValue(1);
double aSinResult = angleRad1.asin();        // 1.570796(..)
double aCosResult = angleRad1.acos();        // 0
double aTanResult = angleRad1.atan();        // 0.785398(..)
double aCotResult = angleRad1.acot();        // 0.785398(..)

Please keep in mind that not all values belong to the domain of a given trigonometric function. Some might throw an exception if they hold invalid value for a specific function. Value will be automatically converted to radians before using Java Math trigonometric functions.

  • ceil, floor, roundHalfEven with relevant digits
ThermalConductivity thermCond = ThermalConductivity.ofWattsPerMeterKelvin(0.00366);
ThermalConductivity ceilResult = thermCond.ceil();                  // 1
ThermalConductivity floorResult = thermCond.floor();                // 0
ThermalConductivity roundedResult = thermCond.roundHalfEven(2);     // 0.0037

Please note that rounding function is based on concept of relevant digits. It is NOT a simple truncation to a specified number of decimal places. This function will evaluate where relevant digits starts and will attempt to keep the specified number by user. For an example for roundHalfEven(3) will output:
0.12345 -> roundHalfEven(3) -> 0.123
0.00012345 -> roundHalfEven(3) -> 0.000123

4.5 Equality and sorting

Each PhysicalQuantity class has implemented the equals and hashCode methods and implements the Comparable interface, including additional methods that might be helpful during development.


Equality condition: PhysicalQuantity instances are equal, if they are of the same unit Type (i.e.: TemperatureUnit) and their values converted to the BASE unit are equal.


In other words, equality is set to represent equality in the physical context. By default, values stored as double types are compared directly with their full decimal digit precision. In some cases, you might need to assume equality for a specified level of accuracy beyond which you will consider quantities as equal in your business application. Please find examples below:

// Basic equality
Temperature tempInC1 = Temperature.ofCelsius(20.0);
Temperature tempInC2 = Temperature.ofCelsius(20.0);
Temperature tempInK = Temperature.ofKelvins(20 + 273.15);
tempInC1.equals(tempInC2);  // true
tempInC1.equals(tempInK);   // true

// Equality with precision
Temperature manyDigits = Temperature.ofFahrenheit(21.1234567);
Temperature fewDigits = Temperature.ofFahrenheit(21.123);
manyDigits.equals(fewDigits);                               // false
manyDigits.equalsWithPrecision(fewDigits, 0.001);           // true

Precision has to be provided as commonly phrased "epsilon" which simply is a double value representing required decimal accuracy.

Sorting is based on the same principle as equality, quantities are compared and sorted based on their physical context, meaning that base unit and value in base unit are used to determine the order of elements.

// Unsorted array of quantities
Temperature[] temperatures = {
    Temperature.ofCelsius(54),     // equiv. of: 327.15 K
    Temperature.ofKelvins(10),     // equiv. of: 10.00 K
    Temperature.ofCelsius(-20),    // equiv. of: 253.15 K
    Temperature.ofCelsius(100)     // equiv. of: 373.15 K
};
// Sorting quantities        
Arrays.sort(temperatures);  // sorted order: 10K, -20oC, 54oC, 100oC

Hashcode follows the same concept and is calculated for quantity base value and base unit.

5. UNITILITY EXTENSION MODULES

In most web applications, especially in microservice architecture, data is frequently exchanged between services. PhysicalQuantity will have to be represented as a JSON structure in request/response objects or even as one string, often used as a request/query parameter or path param/variable. To address this, additional modules have been provided with ready serializers/deserializers and integration with the most popular frameworks.


IMPORTANT:
a) JSON request body of PhysicalQuantity should follow the structure shown in the section below (5.1)
b) Sending request via path param or query param, should be used i.e.: 20.0oC, without square brackets or special characters.


Json request body example:

{
    "airTemperature": {
      "value": 20.5,
      "unit": "°C"
    },
    "airPressure": {
      "value": 101325.0,
      "unit": "Pa"
    },
    "relativeHumidity": {
      "value": 55.0,
      "unit": "%"
    }
}

Using as path param:

/api/v1/temperatures/20.5C

Make sure that quantities used in path variables or query parameters are without square brackets. Parsers are designed to filter them out, but some recent versions of Tomcat server have issues with "[]" so it is better to avoid using them. Other special characters can be easily replaced with simpler equivalents, as presented in the table in section 4.2.

5.1 Jackson serializers and deserializers

The Jackson library is utilized in many frameworks, enabling the serialization of objects into a JSON structure and deserialization back to Java objects. To include this module in your project, use the following dependency:

<dependency>
    <groupId>com.synerset</groupId>
    <artifactId>unitility-jackson</artifactId>
    <version>2.4.2</version>
</dependency>

PhysicalQuantity JSON structure for valid serialization / deserialization has been defined as in the following example:

{
  "value": 10.5,
  "unit": "oC"
}

The Jackson module provides a configured SimpleModule with registered StdSerializer and JsonDeserializer instances. Each PhysicalQuantity class has its own defined deserializer. Deserializers use PhysicalQuantityParsingFactory to obtain the appropriate parser depending on the class type. This module is part of the Spring and Quarkus modules, therefore, it does not need to be added explicitly if framework extensions are included in the project.

5.2 Spring Boot module

Module tested for Spring Boot version: 3.3.0
Spring Boot module includes unitility-jackson and unitility-core modules, and it will automatically create required beans through the autoconfiguration dependency injection mechanism. To use Spring extension module, add the following dependency:

<dependency>
    <groupId>com.synerset</groupId>
    <artifactId>unitility-spring</artifactId>
    <version>2.4.2</version>
</dependency>

Adding Spring module to the project will automatically:

  • register PhysicalQuantityJacksonModule in ObjectMapper bean (JSON serialization/deserialization)
  • register Converter instances in WebMvcConfigurer bean (query/path params serialization/deserialization)

This configuration allows using PhysicalQuantity types in request / response objects and in query parameters or path variables, as in the example below:

@RestController
public class DefaultUnitsController {

    @GetMapping("/temperatures/{temperature}")
    public Temperature getTemperature(@PathVariable("temperature") Temperature temperature) {
        // Some code
        return temperature;
    }

    @PostMapping("/temperatures")
    public Temperature postTemperature(@RequestBody Temperature temperature) {
        // Some code
        return temperature;
    }
    
}

For special cases when custom unit is created, which is not a part of standard Unitiltiy package, additional configuration steps must be carried out to ensure that custom unit is properly resolved from JSON or path/query params. For more details see a section: Registering custom quantity in Spring.

5.3 Quarkus module

Module tested for Quarkus platform version: 3.11.0
Quarkus module includes unitility-jackson and unitility-core modules, and it will be automatically discovered through Jandex index and will create required CDI beans. To use Quarkus extension module, add following dependency:

<dependency>
    <groupId>com.synerset</groupId>
    <artifactId>unitility-quarkus</artifactId>
    <version>2.4.2</version>
</dependency>

Adding Quarkus module to the project will automatically:

  • register PhysicalQuantityJacksonModule in ObjectMapper using ObjectMapperCustomizer bean (JSON serialization/deserialization)
  • register ParamConverter instances in Jakarta ParamConverterProvider bean (query/path params serialization/deserialization)

This configuration allows using PhysicalQuantity types in request / response objects and in query parameters or path variables, as in the example below

public class DefaultUnitsResource {

    @GET
    @Path("/temperatures/{temperature}")
    @Produces(MediaType.APPLICATION_JSON)
    public Temperature getTemperature(@PathParam("temperature") Temperature temperature) {
        // Some code
        return temperature;
    }

    @POST
    @Path("/temperatures")
    @Produces(MediaType.APPLICATION_JSON)
    public Temperature postTemperature(Temperature temperature) {
        // Some code
        return temperature;
    }
}

For special cases when custom unit is created, which is not a part of standard Unitiltiy package, additional configuration steps must be carried out to ensure that custom unit is properly resolved from JSON or path/query params. For more details see a section: Registering custom quantity in Quarkus.

5.4 Jakarta validation

The validation module encompasses preconfigured validator classes along with corresponding annotations designed for bean validation purposes. The available annotations are as follows:

  • @PhysicalMin
  • @PhysicalMax
  • @PhysicalRange

These annotations are designed to accept a string containing a value and its corresponding unit, representing a physical quantity. The validator will utilize the default parsing factory utility to generate a PhysicalQuantity instance reflecting the provided limiting value. Subsequently, it will compare this instance to the validated quantity. Both quantities will undergo comparison based on their base units, allowing users to conveniently specify limits in any of the supported units.

By default, the provided limits are inclusive. To designate a value as exclusive, employ the parameter "inclusive = false".

Validators will be automatically discovered via Spring Boot or Quarkus dependency injection mechanisms once the dependency is added to the pom.xml file or any other dependency management tool.

Example of specifying minimum value limit, inclusive:

    @GetMapping("/dry-air")
    DryAirResponse getDryAir(@RequestParam @PhysicalMin("-150oC") Temperature temperature);

Example of specifying maximum value as exclusive limit:

    @GetMapping("/dry-air")
    DryAirResponse getDryAir(@RequestParam @PhysicalMax(value = "373.15K", inclusive = false) Temperature temperature);

Example of specifying range, where both limits are exclusive:

    @GetMapping("/dry-air")
    DryAirResponse getDryAir(@RequestParam @PhysicalRange(min = "-150oC", minIncl = false, max = "1000oC", maxIncl = false) Temperature temperature);

Keep in mind that validated class must be annotated with: @Validated annotation.

Following dependencies MUST be added to the project in order to unitility-validation to work:

6. CREATING CUSTOM QUANTITIES

The Unitility includes a set of the most commonly used quantities and related units with an emphasis on thermodynamics. However, the framework foundation can be successfully used to define almost any unit from economy, biology, electronics, and even for logistics to represent the quantity of bottles in different sized packages. Sooner or later, a developer might face a case where he would like to add a new unit or quantity to the library. I will be including requested units on a regular basis. If this is not urgent, please go to the ISSUES page and let me know what is needed. If you can't wait, below are instructions on how to create a custom unit and also how to ensure that all your custom units/quantities are registered correctly in Spring or Quarkus.

6.1 Custom unit

If you need to extend standard unit definitions for a given quantity, the simplest way is to create a new unit class extending the interface of the desired unit family. To present this, let's add a new unit of Rankine degree to Temperature family. Create class or enum (my preference) and extend TemperatureUnits interface and implement required methods to convert from base unit to Rankine and vice versa. To ensure valid conversion with standard Temperature unit family, set the same base unit type as in the library default TemperatureUnits:

@Override
public TemperatureUnit getBaseUnit() {
   return TemperatureUnits.KELVIN;
}

After this is done, you can use your own custom unit in Temperature class and convert between standard library units and your own custom units:

Temperature tempInC = Temperature.ofKelvins(100);                    // 100.00 K
Temperature tempInR = Temperature.of(200, CustomTempUnits.RANKINE);  // eq. of 111.111 K
Temperature totalTemp = temperatureInC.plus(temperatureInR);         // 211.111 K

6.2 Custom physical quantity

In this section, we create new CustomAngle quantity together with new CustomAngleUnit of Revolutions [rev]. To do this, we need to create a new class, CustomAngle, and extend the CalculableQuantity<AngleUnit, CustomAngle> interface. CalculableQuantity is PhysicalQuantity with arithmetic operations handling. In typical case, most of the required methods are defined as default in the interface, which should be sufficient for most cases. However, some needs to be implemented for the new unit, as shown in the example here: CustomAngle.

In the next step, create your implementation of AngleUnit interface, which should include the way, how to convert one unit from another. Here is an example based on enum to introduce revolution unit: CustomAngleUnits.

After this is done, we can create and use our custom angles:

// Creating instance of custom unit
CustomAngle revolutions = CustomAngle.ofRevolutions(1);         // CustomAngle{1.0 rev}
// Converting to any unit implementing AngleUnit type
CustomAngle degrees = revolutions.toUnit(AngleUnits.DEGREES);   // CustomAngle{360.0°}
// All transformations work withing the units implementing AngleUnit
CustomAngle resultingRevolutions = revolutions.plus(degrees);   // CustomAngle{2.0 rev}

6.3 Registering custom quantities in SPRING

After creating a custom unit, to ensure that it is properly resolved from JSON or path/query params, the following steps must be taken to make it work. A complete example of a new custom unit properly registered in a Spring framework can be found here: unitility-spring-example.

As a first step, a new parsing factory has to be created, which must include currently supported quantities and custom user quantities: CustomParsingFactory.

After a new parsing factory is created and all standard and new custom quantities parsers are properly registered, you can now create a configuration and register new JacksonModule and new Converter in FormatterRegistry: CustomAngleConfiguration.

After this step is done, you can freely use your CustomAngle unit in your web application:

@RestController
public class CustomAngleController {

    private final Logger logger = LoggerFactory.getLogger(this.getClass().getSimpleName());

    @GetMapping("/angles/{angle}")
    public CustomAngle getCustomAnglePath(@PathVariable("angle") CustomAngle customAngle) {
        processCustomAngle(customAngle);
        return customAngle;
    }

    @PostMapping("/angles")
    public CustomAngle getCustomAngleBody(@RequestBody CustomAngle customAngle) {
        processCustomAngle(customAngle);
        return customAngle;
    }
}

6.4 Registering custom quantities in Quarkus

Registering custom unit in Quarkus is a bit different compared to Spring. A Complete example of new custom unit properly registered in a Quarkus framework can be found here: unitility-quarkus-example. In this case, a first step is the same, new parsing factory must be created to include currently supported and new custom quantities created by user: CustomParsingFactory.

After a new parsing factory is created and all standard and new custom quantities parsers are properly registered, you can now create a new ObjectMapperCustomizer and register JacksonModule with new parsing factory: CustomObjectMapperCustomizer.

The Last step is to define new PathParamConverterProvider and register ParamConverters for all custom quantities: CustomAngleParamProvider.

After this step is done, you can freely use your CustomAngle unit in your web application:

@RestController
public class CustomAngleController {

    private final Logger logger = LoggerFactory.getLogger(this.getClass().getSimpleName());

    @GetMapping("/angles/{angle}")
    public CustomAngle getCustomAnglePath(@PathVariable("angle") CustomAngle customAngle) {
        processCustomAngle(customAngle);
        return customAngle;
    }

    @PostMapping("/angles")
    public CustomAngle getCustomAngleBody(@RequestBody CustomAngle customAngle) {
        processCustomAngle(customAngle);
        return customAngle;
    }
}

In Quarkus, IntellijJ might highlight PhysicalQuantity types when used as path or query parameters, but when you run the application, they will be correctly resolved.

7. COMPATIBILITY WITH OTHER JVM LANGUAGES

Unitility can be easily used with other JVM-based programming languages. There are some features of these languages that make using Unitility easier and more elegant, for example, through the use of overloaded operators. I am more than happy to provide even greater integration with other JVM languages, please use ISSUES to leave there any suggestion in that regard.

7.1 Groovy - using overloaded operators

Please find the below example of Unitility usage in Groovy, taking advantage of overloaded operators:

// Temperature examples
def t1 = Temperature.ofCelsius(20)
def t2 = Temperature.ofCelsius(10)
def t3 = Temperature.ofKelvins(303.15)  // =30 oC
// Adding & subtracting: The same unit
def t4 = t1 + t2                // Temperature{30.0°C}
def t5 = t1 - t2                // Temperature{10.0°C}
def t6 = t1 + 15.5              // Temperature{35.5°C}    
// Adding & subtracting: Different units of the same quantity,
// resolving to a first addend unit type.
def t7 = t1 + t3                // Temperature{50.0°C}
def t8 = t1 - t3                // Temperature{-10.0°C}
// Multiply
def t9 = t1 * t2                // will resolve to double = 200.0
def t10 = t1 * 2                // Temperature{40.0°C}
// Divide
def t11 = t1 / t2               // will resolve to double = 2.0
def t12 = t1 / 2                // Temperature{10.0°C}
// Logical operations
def isGreater = t1 > t2         // true
def isLower = t1 < t2           // false
def isGreaterOrEq = t1 >= t2    // true
def isEqual = t1 == t1          // true

7.2 Kotlin - using overloaded operators

Usage in Kotlin is analogous, please find the below some examples:

// Temperature examples
val t1 = Temperature.ofCelsius(20.0)
val t2 = Temperature.ofCelsius(10.0)
val t3 = Temperature.ofKelvins(303.15) // =30 oC

// Adding & subtracting: The same unit
val t4 = t1 + t2
val t5 = t1 - t2
val t6 = t1 + 15.5
println(t4) // Temperature{30.0°C}
println(t5) // Temperature{10.0°C}
println(t6) // Temperature{35.5°C}

// Adding & subtracting: Different units of the same quantity,
// resolving to a first addend unit type.
val t7 = t1 + t3
val t8 = t1 - t3
println(t7) // Temperature{50.0°C}
println(t8) // Temperature{-10.0°C}

// Multiply
val t9 = t1 * t2
val t10 = t1 * 2.0
println(t9) // will resolve to double = 200.0
println(t10) // Temperature{40.0°C}

// Divide
val t11 = t1 / t2
val t12 = t1 / 2.0
println(t11) // will resolve to double = 2.0
println(t12) // Temperature{10.0°C}

// Logical operations
val isGreater = t1 > t2         // true
val isLower = t1 < t2           // false
val isGreaterOrEq = t1 >= t2    // true
val isEqual = t1 == t1          // true

8. SPECIAL TYPES: GEOGRAPHIC

A set of dedicated types has been provided to cover spatial types used in expressing geographic measures: Latitude, Longitude, GeoCoordinate and GeoDistance. These classes allow representing coordinates on Earth and real curvature distance between these coordinates.

8.1 Geographic Latitude, Longitude and GeoCoordinate

The Latitude class includes methods that allow for easy conversion to the Degrees-Minutes-Seconds (DMS) format. This format provides a more popular representation of geographic coordinates, making it convenient for various applications where DMS notation is preferred. Latitude range is: -90 to 90 degrees.
The Longitude class, analogous to the Latitude class, represents a geographic longitude coordinate. It adheres to the standard range of -180 to +180 degrees, covering the westernmost point at -180 degrees and the easternmost point at +180 degrees.

// Latitude and Longitude types are based on Angular units
Latitude latitude = Latitude.ofDegrees(-20.123);
Longitude longitude = Longitude.ofDegrees(20.123);
// Both can be reduced to a string in DMS format or in ENG format:
String latInDMS = latitude.toDMSFormat(2);          // Outputs: 20°7'22.8"S
String latInENG = latitude.toEngineeringFormat();   // Outputs: -20.123 [°]

You can also create Latitude or Longitude instance providing degrees, minutes and seconds:

// Instance from degrees, minutes, seconds
Latitude latFromDMS = Latitude.ofDegMinSec(20, 7, 22.8, CardinalDirection.SOUTH);   // Latitude{-20.123°}
Longitude longFromDMS = Longitude.ofDegMinSec(20, 7, 22.8, CardinalDirection.EAST); // Longitude{20.123°}

The GeoCoordinate class combines both Latitude and Longitude to form a complete geographic coordinate. It facilitates easy management and manipulation of spatial data, allowing seamless integration into various applications requiring precise location information.

// GeoCoordinate class represents a coordinate of specific point in the globe, using Latitude and Longitude and optional name
GeoCoordinate coordinateExample = GeoCoordinate.of(latitude, longitude, "my location");
// GeoCoordinate can be reduced to DMS format, ENG format, or decimal degrees format
// Decimal degrees format with coma separating latitude from longitude is for ie: how Google Maps output cords
String geoCoordDMS = coordinateExample.toDMSFormat();             // 20°7'22.8"S, 20°7'22.8"E
String geoCoordEND = coordinateExample.toEngineeringFormat();     // -20.12 [°], 20.12 [°]
String geoCoordDEC = coordinateExample.toDecimalDegrees();        // -20.12, 20.12

Latitude and Longitude do not enforce any angular value limit, but GeoCoordinate will do. Make sure that your latitude and longitude values fall withing planet Earth's acceptable limits.

8.2 GeoDistance - spherical distance between two coordinates

The GeoDistance class represents the spherical distance between two coordinates on Earth, considering the curvature of the Earth. It incorporates calculations involving the start and target coordinates, true bearing, and distance in specified units. The underlying Haversine equations serve as the basis for the curved distance calculation. To create a GeoDistance object, you can initialize it with start and target coordinates along with the desired unit type for representing the distance. See the example below:

GeoCoordinate wroclaw = GeoCoordinate.of(Latitude.ofDegrees(51.102772), Longitude.ofDegrees(16.885802));
GeoCoordinate newYork = GeoCoordinate.of(Latitude.ofDegrees(40.712671), Longitude.ofDegrees(-74.004655));

GeoDistance geoDistance = GeoDistance.ofKilometers(wroclaw, newYork);

String distanceInEng = geoDistance.toEngineeringFormat();   // 6669.896095258197 [km]
Angle trueBearing = geoDistance.getTrueBearing();           // Angle{-61.07915625042435°}

Please note that true bearing is provided in range <-180,+180> degrees, and it is absolute value, to Earth's true north. Alternatively, GeoDistance can be initialized with starting coordinate, bearing and distance in that case, target coordinate will be calculated. Provided distance must be true, curved distance.

GeoDistance toNewYork = GeoDistance.of(wroclaw, trueBearing, Distance.ofKilometers(6669.896095258197));
String targetCoordinate = toNewYork.getTargetCoordinate().toDecimalDegrees();   // 40.712671, -74.004655

As you can see the results, we have set heading towards New York from Wrocław with previously calculated curved distance between these two cities, and we have reached to exactly the same spot.

Progression and translation:
The GeoDistance class supports two methods to simulate distance change:
a) methods named: with(), which accepts new target coordinate or bearing and distance, retaining current starting point coordinate and calculates distance to the new coordinates (specified or calculated):

GeoCoordinate wellington = GeoCoordinate.of(Latitude.ofDegrees(-41.289463), Longitude.ofDegrees(174.774913));

GeoDistance toWellington = toNewYork.with(wellington);

Distance distance = toWellington.getDistance();                                      // 18005.6198226 km
String wroclawCoordinate = toWellington.getStartCoordinate().toDecimalDegrees();     // 51.102772, 16.885802
String wellingtonCoordinate = toWellington.getTargetCoordinate().toDecimalDegrees(); // -41.289463, 174.774913

Previously, toNewYork represented the distance from Wrocław to New York. Following this change, it now represents the distance from Wrocław to Wellington, along with the updated bearing and calculated distance.

b) Methods named translate() will now accept a new target coordinate or bearing and distance. They set the previous target coordinate as the current start coordinate and calculate the distance to the new coordinates, whether specified or calculated.

GeoDistance toWellingtonBis = toNewYork.translate(wellington);

Distance distanceBis = toWellingtonBis.getDistance();                                      // 14403.6934729 km
String wroclawCoordinateBis = toWellingtonBis.getStartCoordinate().toDecimalDegrees();     // 40.712671, -74.004655
String wellingtonCoordinateBis = toWellingtonBis.getTargetCoordinate().toDecimalDegrees(); // -41.289463, 174.774913

Previously, toNewYork represented the distance from Wrocław to New York. After this change, it now represents the distance from New York to Wellington, along with the updated bearing and calculated distance.

8.3 Parsing geographic quantities and JSON structures

Parsers have been provided mostly for the purpose of deserialization in Spring Boot or Quarkus, but they can also be used directly in the code:

hysicalQuantityParsingFactory geoParsingFactory = PhysicalQuantityParsingFactory.getDefaultParsingFactory();
Latitude parsedLat1 = geoParsingFactory.parse(Latitude.class, "20°7'22.8\"S");
Latitude parsedLat2 = geoParsingFactory.parse(Latitude.class, "20deg 7min 22.8sec");

Latitude, Longitude can be used as JSON request body or as value in path variable / query param. Path param usage example:

Latitude path param usage example:
/routes/latitude/20°7'22.8"S/longitude/-14°7'12.4"W
/routes/latitude/20o7min22.8secS/longitude/-14o7min12.4secW
/routes/latitude/20.123deg/longitude/-14.123deg
/routes/latitude/20.123/longitude/-14.123

As you can observe, the deserializer behaves liberally, accepting input in various formats to accommodate users' preferences. Latitude, Longitude, GeoCoordinate, and GeoDistance each have their own JSON deserializers, enabling their use in request bodies. A couple of simple examples are provided below.

Example: Latitude in DMS format as a JSON request body:

{
  "value" : "20°7'22.80000000000399\"S"
}

Example: Latitude as a pair of value and unit symbol:

{
  "value" : -20.123456,
  "unit": "deg"
}

Example: Latitude as single value, which will always be resolved to decimal degrees:

{
   "value" : -20.123456
}

The GeoCoordinate JSON structure follows the same pattern.
Example: GeoCoordinate defined by Latitude in DMS format and longitude as a value-unit pair:

{
  "latitude" : {
      "value" : "20°7'22.80\"S"
    },
  "longitude" : {
      "value" : -14.1234,
      "unit": "deg"
    }
}

GeoDistance is serialized or deserialized from the exemplary structures as presented below.

Example: GeoDistance - the base case, where both start and target coordinates are known:

{
    "startCoordinate" : {
        "latitude": {
            "value": -20.123,
            "unit": "deg"
        },
        "longitude": {
            "value": 20.123,
            "unit": "°"
        },
        "name": "MyStartLoc"
    },
    "targetCoordinate" : {
        "latitude": {
            "value": -20.123,
            "unit": "°"
    },
    "longitude": {
        "value": 20.123,
        "unit": "°"
      },
      "name": "MyTargetLoc"
    }
}

Example: GeoDistance in a case where the start coordinate, true bearing, and distance to the target are known:

{
    "startCoordinate" : {
        "latitude": {
            "value": -20.123,
            "unit": "deg"
        },
        "longitude": {
            "value": 20.123,
            "unit": "°"
        },
        "name": "MyStartLoc"
    }, 
    "trueBearing": {
        "value": -20.123,
        "unit": "deg"
    }, 
    "distance": {
        "value": 1000.0,
        "unit": "km"
    }
}

General note:

  • Latitude & Longitude: field "value" for Latitude and Longitude is mandatory, it can accept double value or a string in DMS or ENG format, field "unit" is optional, it is omitted when value is provided as DMS or ENG format. In case of value provided as a double, unit will be automatically resolved to decimal degrees.
  • GeoCoordinate: fields "latitude" and "longitude" are mandatory, "name" is optional.
  • GeoDistance: field "startCoordinate" is mandatory. For variant with two coordinates "targetCoordinate" must be provided, if not available the "trueBearing" and "distance" must be provided.

Arithmetic transformations:
For Latitude, Longitude, and GeoDistance, arithmetic operations function in the same manner as for other PhysicalQuantities. GeoCoordinate does not support arithmetic operations, as it is just a composite data objects used to construct GeoDistance. It's important to note that the natural behavior is such that the result of arithmetic operations will be based on the with() progression. This means that the added value will maintain the starting point, adjust the distance, and recalculate the target coordinate in the process.

// Sum of two GeoQuantities
GeoCoordinate start = GeoCoordinate.of(Latitude.ofDegrees(51.1), Longitude.ofDegrees(16.9));
GeoCoordinate target = GeoCoordinate.of(Latitude.ofDegrees(40.8), Longitude.ofDegrees(-74.1));

GeoDistance firstGeoDistance = GeoDistance.ofKilometers(start, target);         // 6670.048729447209 km
GeoDistance secondGeoDistance = firstGeoDistance.translate(Distance.ofKilometers(1000)); // 1000 km

GeoDistance sumOfDistances = firstGeoDistance.plus(secondGeoDistance);          // 7670.048729447209 km and new target
GeoDistance greaterDistance = sumOfDistances.plus(Distance.ofKilometers(1000)); // 8670.048729447209 km and new target
GeoDistance evenGreaterDistance = greaterDistance.plus(1000);                   // 9670.048729447209 km and new target

9. COLLABORATION, ATTRIBUTION, AND CITATION

I welcome other developers who are interested in physics and engineering to collaborate on this project. Any contributions or suggestions would be greatly appreciated.
Feel free to contact me if yoy have any questions or comments.


If there are particular units relevant to your scientific field that you'd like me to include, please inform me.
I aim for this library to be helpful and simplify your life.


This work is licensed under the terms of the MIT License with the additional requirement that proper attribution be given to the Piotr Jażdżyk as the original author in all derivative works and publications.

For citation and attribution, I have provided badges that you can include in your project to showcase your usage of the library:

Small shield with referenced most recent version tag:
Unitility

[![Unitility](https://img.shields.io/github/v/release/pjazdzyk/Unitility?label=Unitility&color=13ADF3&logo=)](https://github.com/pjazdzyk/Unitility)

Tech shield with version tag for manual adjustment (you can indicate which version you actually use):
Unitility

[![Unitility](https://img.shields.io/badge/UNITILITY-v2.1.1-13ADF3?style=for-the-badge&logo=)](https://github.com/pjazdzyk/Unitility)

10. ACKNOWLEDGMENTS

Special thanks to Kret11, VeloxDigits, Olin44, and others for all discussions on architecture we had.
I extend my heartfelt gratitude to the Silesian University of Technology for imparting invaluable knowledge to me.
Thanks to Mathieu Soysal for his Maven central publisher.
Badges used in readme: Shields.io and Badges 4 README.md.

May the force be with you.