Dataclassish Interoperability#

This guide demonstrates how unxt types work with the dataclassish library for introspection and manipulation of unitful quantities and related types.

unxt types (dimensions, units, unit systems, and quantities) are all Equinox modules, which are Python dataclasses. This means they work seamlessly with dataclassish functions for field manipulation and introspection.

Summary#

dataclassish provides convenient tools for introspecting and manipulating unxt types:

  • fields(obj): Get all field information for a quantity or related type

  • field_keys(obj): Iterate over field names

  • field_values(obj): Iterate over field values

  • field_items(obj): Iterate over (name, value) pairs

  • asdict(obj): Convert to a dictionary representation

  • astuple(obj): Convert to a tuple of values

  • get_field(obj, name): Access a specific field by name

  • **replace(obj, **kwargs)**: Create a copy with modified fields

All of these work seamlessly across:

  • Dimensions

  • Units

  • Unit Systems

  • Quantity types (Quantity, BareQuantity, Angle, Distance, etc.)

This makes it easy to build generic code that works with unxt types without needing to know their internal structure.

Setup#

import unxt as u
import dataclassish as dc

print(f"unxt version: {u.__version__}")
print(f"dataclassish version: {dc.__version__}")

Dimensions#

Let’s start by exploring how dataclassish works with unxt dimensions.

# Create a dimension
length_dim = u.dimension("length")
print(f"Dimension: {length_dim}")
print(f"Type: {type(length_dim)}")

# Get fields
fields = dc.fields(length_dim)
print(f"\nFields: {fields}")

# Get field keys, values, and items
print(f"\nField keys: {list(dc.field_keys(length_dim))}")
print(f"Field values: {list(dc.field_values(length_dim))}")
print(f"Field items: {list(dc.field_items(length_dim))}")

Units#

Now let’s explore units with dataclassish.

# Create a unit
meter = u.unit("m")
print(f"Unit: {meter}")
print(f"Type: {type(meter)}")

# Get fields
fields = dc.fields(meter)
print(f"\nFields: {fields}")

# Get field keys, values, items
print(f"\nField keys: {list(dc.field_keys(meter))}")
print(f"Field values: {list(dc.field_values(meter))}")
print(f"Field items: {list(dc.field_items(meter))}")

Unit Systems#

Let’s explore unit systems with dataclassish.

# Create a unit system
si = u.unitsystem("si")
print(f"Unit System: {si}")
print(f"Type: {type(si)}")

# Get fields
fields = dc.fields(si)
print(f"\nFields: {fields}")

# Get field keys, values, items
print(f"\nField keys: {list(dc.field_keys(si))}")
print(f"Field values: {list(dc.field_values(si))}")
print(f"Field items: {list(dc.field_items(si))}")
# Convert to dict and tuple
ussystem_dict = dc.asdict(si)
print(f"asdict(unit_system): {ussystem_dict}")

ussystem_tuple = dc.astuple(si)
print(f"astuple(unit_system): {ussystem_tuple}")

Quantities#

Now let’s explore how dataclassish works with quantities. We’ll examine different quantity types.

Basic Quantity (with dimension checking)#

The Quantity type includes dimension parametrization and dimension checking.

# Create a quantity with dimension checking
distance = u.Quantity(10.0, "m")
print(f"Quantity: {distance}")
print(f"Type: {type(distance)}")

# Get fields
fields = dc.fields(distance)
print(f"\nFields: {fields}")

# Get field keys, values, items
print(f"\nField keys: {list(dc.field_keys(distance))}")
print(f"Field values: {list(dc.field_values(distance))}")
print(f"Field items: {list(dc.field_items(distance))}")
# Convert to dict and tuple
qty_dict = dc.asdict(distance)
print(f"asdict(quantity): {qty_dict}")

qty_tuple = dc.astuple(distance)
print(f"astuple(quantity): {qty_tuple}")

# Use replace() to modify a quantity
new_distance = dc.replace(distance, value=20.0)
print(f"\nOriginal: {distance}")
print(f"After replace(value=20.0): {new_distance}")
# Use get_field() to access individual fields
value_field = dc.get_field(distance, "value")
print(f"get_field(distance, 'value'): {value_field}")

unit_field = dc.get_field(distance, "unit")
print(f"get_field(distance, 'unit'): {unit_field}")

BareQuantity (lightweight, no dimension checking)#

# BareQuantity is a lightweight alternative without dimension checking
bare_qty = u.quantity.BareQuantity(5.0, "km")
print(f"BareQuantity: {bare_qty}")
print(f"Type: {type(bare_qty)}")

# All dataclassish functions work the same
print(f"\nFields: {dc.fields(bare_qty)}")
print(f"Field keys: {list(dc.field_keys(bare_qty))}")
print(f"Field values: {list(dc.field_values(bare_qty))}")
print(f"asdict: {dc.asdict(bare_qty)}")
print(f"astuple: {dc.astuple(bare_qty)}")

# Replace works here too
new_bare_qty = dc.replace(bare_qty, value=10.0)
print(f"\nOriginal: {bare_qty}")
print(f"After replace(value=10.0): {new_bare_qty}")

Angle (specialized quantity with wrapping)#

Angle is a specialized quantity type for angular measurements with automatic wrapping.

# Create an Angle quantity
theta = u.Angle(45.0, "deg")
print(f"Angle: {theta}")
print(f"Type: {type(theta)}")

# Dataclassish introspection
print(f"\nFields: {dc.fields(theta)}")
print(f"Field keys: {list(dc.field_keys(theta))}")
print(f"asdict: {dc.asdict(theta)}")

# Replace the angle value
new_theta = dc.replace(theta, value=90.0)
print(f"\nOriginal: {theta}")
print(f"After replace(value=90.0): {new_theta}")

# Get individual fields
angle_value = dc.get_field(theta, "value")
angle_unit = dc.get_field(theta, "unit")
print(f"\nAngle value: {angle_value}")
print(f"Angle unit: {angle_unit}")