Skip to main content

Unit Conversion fields for Django and Postgres

Project description

django-pint-field

Published on Django Packages PyPI Python versions Django versions Documentation License: MIT

Store, validate, and convert physical quantities in Django using Pint.

Why Django Pint Field?

Django Pint Field enables you to:

  • Store quantities (like 1 gram, 3 miles, 8.120391 angstroms) in your Django models
  • Edit quantities in forms with automatic unit conversion
  • Compare quantities in different units (e.g., compare weights in pounds vs. kilograms)
  • Display quantities in user-preferred units while maintaining accurate comparisons
  • Perform aggregations and lookups across different units of measurement

The package uses a PostgreSQL composite field to store both the magnitude and units, along with a base-unit comparator value for accurate comparisons. This approach ensures that users can work with their preferred units while maintaining data integrity and comparability. For this reason, the project only works with PostgreSQL databases.

Requirements

  • Python 3.11+
  • Django 4.2+
  • PostgreSQL database
  • Pint 0.24+

Installation

pip install django-pint-field

Add to your INSTALLED_APPS:

INSTALLED_APPS = [
    # ...
    "django_pint_field",
    # ...
]

Run migrations:

python manage.py migrate django_pint_field
Failure to run django-pint-field migrations before running migrations for models using PintFields will result in errors. The migration creates a required composite type in your PostgreSQL database.

Previous versions of the package added three composite types to the database. The newest migration modifies the columns with these types to use a single composite type.

Tips for Upgrading from Legacy django-pint-field

If using [django-pgtrigger](https://django-pgtrigger.readthedocs.io/en/latest/commands/) or other packages that depend on it (e.g.: django-pghistory), we highly recommend that you temporarily uninstall all triggers before running the django-pint-field migrations. It is also a good practice to make a backup of your database before running the migration. Users freshly installing `django-pint-field` do not need to worry about this warning.
python manage.py pgtrigger uninstall

Then run the migrations:

python manage.py migrate django_pint_field

Reinstall the triggers after the migrations are complete:

python manage.py pgtrigger install

Quick Start

  1. Define your model:
from decimal import Decimal
from django.db import models
from django_pint_field.models import DecimalPintField


class Product(models.Model):
    name = models.CharField(max_length=100)
    weight = DecimalPintField(
        default_unit="gram",
        unit_choices=["gram", "kilogram", "pound", "ounce"],
    )
  1. Use it in your code:
from django_pint_field.units import ureg

# Create objects
product = Product.objects.create(
    name="Coffee Bag",
    weight=ureg.Quantity(Decimal("340"), "gram"),
)

# Query using different units
products = Product.objects.filter(
    weight__gte=ureg.Quantity(Decimal("0.5"), "kilogram"),
)

# Access values
print(product.weight)  # 340 gram
print(product.weight.quantity)  # 340 gram (accessing the Pint Quantity object)
# Convert to different units
print(product.weight.quantity.to("kilogram"))  # 0.340 kilogram
print(product.weight.kilogram)  # 0.340 kilogram
print(product.weight.kilogram__2)  # 0.34 kilogram  (rounded to 2 decimal places)

Features

Field Types

  • IntegerPintField: For whole number quantities
  • DecimalPintField: For precise decimal quantities

Form Fields and Widgets

  • Built-in form fields with unit conversion
  • TabledPintFieldWidget for displaying unit conversion tables
  • Customizable validation and unit choices

Form Fields

  • IntegerPintFormField: Used in forms with IntegerPintField.
  • DecimalPintFormField: Used in forms with DecimalPintField.

Widgets

  • PintFieldWidget: Default widget for all django pint field types.
  • TabledPintFieldWidget: Provides a table showing conversion to each of the unit_choices.

TabledPintFieldWidget

Django REST Framework Integration

from django_pint_field.rest import DecimalPintRestField


class ProductSerializer(serializers.ModelSerializer):
    weight = DecimalPintRestField()

    class Meta:
        model = Product
        fields = ["name", "weight"]
The package is tested to work with both Django REST Framework and Django Ninja.

Supported Lookups

  • exact
  • gt, gte
  • lt, lte
  • range
  • isnull

Aggregation Support

from django_pint_field.aggregates import PintAvg, PintSum

Product.objects.aggregate(
    avg_weight=PintAvg("weight"),
    total_weight=PintSum("weight"),
)

Supported Aggregates

  • PintAvg
  • PintCount
  • PintMax
  • PintMin
  • PintSum
  • PintStdDev
  • PintVariance
  • PintPercentile - continuous percentile (PERCENTILE_CONT)
  • PintMedian - 50th percentile
  • pint_histogram() - equi-width histogram helper

Running Totals and Partitioned Aggregates

Use PintWindow for unit-aware window functions; a plain django.db.models.Window around a Pint aggregate is rejected (it cannot carry the unit conversion):

from django.db.models import F
from django_pint_field.aggregates import PintSum, PintWindow

Product.objects.annotate(
    running_total=PintWindow(PintSum("weight"), order_by=F("pk").asc()),
)

The result is a PintFieldProxy in the field's base unit; pass output_unit= on the wrapped aggregate (or call .quantity.to(...)) to convert. Ordered-set aggregates (PintPercentile, PintMedian) cannot be used in a window.

Advanced Usage

Custom Units

Create your own unit registry:

from pint import UnitRegistry

custom_ureg = UnitRegistry(non_int_type=Decimal)
custom_ureg.define("custom_unit = [custom]")

# In settings.py
DJANGO_PINT_FIELD_UNIT_REGISTER = custom_ureg

Indexing

Django Pint Field supports creating indexes on the comparator components of Pint fields. Indexes can improve query performance when filtering, ordering, or joining on Pint field values.

Single Field Index

from django_pint_field.indexes import PintFieldComparatorIndex


class Package(models.Model):
    weight = DecimalPintField("gram")

    class Meta:
        indexes = [PintFieldComparatorIndex(fields=["weight"])]

Multi-Field Index

from django_pint_field.indexes import PintFieldComparatorIndex


class Package(models.Model):
    weight = DecimalPintField("gram")
    volume = DecimalPintField("liter")

    class Meta:
        indexes = [PintFieldComparatorIndex(fields=["weight", "volume"])]

You can also use additional index options, as usual. e.g.:

  • name: Custom index name
  • condition: Partial index condition
  • include: Additional columns to include in the index
  • db_tablespace: Custom tablespace for the index

Settings

# settings.py

# Set decimal precision for the entire project
DJANGO_PINT_FIELD_DECIMAL_PRECISION = 40

# Configure custom unit registry
DJANGO_PINT_FIELD_UNIT_REGISTER = custom_ureg

# Set default format for quantity display
DJANGO_PINT_FIELD_DEFAULT_FORMAT = "D"  # Options: D, P, ~P, etc.

Credits

Modified from django-pint with a focus on composite field storage and enhanced comparison capabilities.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

django_pint_field-2026.6.1.tar.gz (51.7 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

django_pint_field-2026.6.1-py3-none-any.whl (57.2 kB view details)

Uploaded Python 3

File details

Details for the file django_pint_field-2026.6.1.tar.gz.

File metadata

  • Download URL: django_pint_field-2026.6.1.tar.gz
  • Upload date:
  • Size: 51.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.3 {"installer":{"name":"uv","version":"0.11.3","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for django_pint_field-2026.6.1.tar.gz
Algorithm Hash digest
SHA256 64d9f7fc4afcd987bb57c982df2d5a891414d291b84af54b8d6e47c4c3f08b3b
MD5 356f24d04b25285ca63b3d3860082d16
BLAKE2b-256 0c877da1a66bb70ab336d9bd9bced8cb306589c22fa449c6b55f1771887f54f4

See more details on using hashes here.

File details

Details for the file django_pint_field-2026.6.1-py3-none-any.whl.

File metadata

  • Download URL: django_pint_field-2026.6.1-py3-none-any.whl
  • Upload date:
  • Size: 57.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.3 {"installer":{"name":"uv","version":"0.11.3","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for django_pint_field-2026.6.1-py3-none-any.whl
Algorithm Hash digest
SHA256 c6a4b6dfeee8ee1998d1bf5962d82c2d609c68a7124731f21c3f18968ed0fe5b
MD5 041178951f7c6087fd1d054ce65088c3
BLAKE2b-256 09ed7eabe58f5d374be74bc79e01f31aeaee8a04673b89b882ce76dbb4151663

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page