Skip to main content

FHIR Slicing utilities build on top of Pydantic

Project description

🔪 FHIR Slicing utilities for Pydantic

PyPI version License: MIT Python 3.10+

A Python library that simplifies working with nested elements in FHIR resources using Pydantic models and smart slicing.

🚀 Installation

pip install fhir-slicing

🤔 The Challenge

Working with FHIR resources in Python can be challenging due to their complex structure and extensibility. FHIR resources often contain:

  • Nested elements with cardinality 0..* or 1..*
  • Extension arrays with unknown elements

This leads to verbose and error-prone code when accessing nested data:

# Traditional way to access birth place
birth_place = next(
    (e.valueAddress.city
     for e in patient.extension
     if e.url == "http://hl7.org/fhir/StructureDefinition/patient-birthPlace"),
    None
)
# Traditional way to access systolic blood pressure
bp_reading = observation.component[0].valueQuantity.value  # Fragile! Assumes systolic is first
# or by code but not very readable
systolic = next(
    (c.valueQuantity.value
     for c in observation.component
     if c.code.coding[0].code == "8480-6"),
    None
)

✨ Solution: Smart Slicing

This library introduces a more intuitive way to access nested FHIR data using named slices, inspired by FHIR's slicing mechanism.

Known slices are defined as annotated fields in Pydantic models, which provide:

  • ✅ Validation of slice cardinality
  • 🛡️ Type safety for slice elements
  • 📖 Improved readability

Example: Patient with birthPlace extension

# Access known extensions by name, while preserving access to unknown ones
patient.extension.birthPlace.valueAddress.city
patient.extension[0]  # Still works for accessing any extension

Example: Blood Pressure Observation with systolic and diastolic components

# Access components naturally
bp = BloodPressureObservation.model_validate(data)
systolic = bp.component.systolic.valueQuantity.value
diastolic = bp.component.diastolic.valueQuantity.value

❓ How

Setting up your models for slicing is as simple as subclassing ElementArray and defining a discriminator method.

[!NOTE] Interested in how it works? Check out this blog post for more details. Example: Patient with birthPlace extension

from pydantic_fhir_slicing import Slice, ElementArray
from my_fhir_types import Address, BaseModel

class AddressExtension(BaseModel):
    url: str
    valueAddress: Address

class PatientExtensions(ElementArray):
    birthPlace: AddressExtension = Slice(1, 1)

    def discriminator(self, item) -> str:
        url = item.get("url", None)
        match url:
            case "http://hl7.org/fhir/StructureDefinition/patient-birthPlace":
                return "birthPlace"
            case _:
                return "@default"

class Patient(BaseModel):
    extension: PatientExtensions

Example: Blood Pressure Observation with systolic and diastolic components

from pydantic_fhir_slicing import ElementArray
from my_fhir_types import CodeableConcept, Quantity, BaseModel

class QuantityComponent(BaseModel):
    code: CodeableConcept
    valueQuantity: Quantity

class BPComponents(ElementArray):
    systolic: QuantityComponent = Slice(1, 1)
    diastolic: QuantityComponent = Slice(1, 1)

    def discriminator(self, item: Component) -> str:
        try:
            code = item["code"]["coding"][0]["code"]
            match code:
                case "8480-6":
                    return "systolic"
                case "8462-4":
                    return "diastolic"
                case _:
                    return "@default"
        except (KeyError, IndexError):
            return "@default"

class BloodPressureObservation(BaseModel):
    code: CodeableConcept
    component: BPComponents

👥 Contributing

Contributions are welcome! Please feel free to submit a Pull Request.

📜 License

MIT License

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

fhir_slicing-0.1.1.tar.gz (13.8 kB view details)

Uploaded Source

Built Distribution

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

fhir_slicing-0.1.1-py3-none-any.whl (7.6 kB view details)

Uploaded Python 3

File details

Details for the file fhir_slicing-0.1.1.tar.gz.

File metadata

  • Download URL: fhir_slicing-0.1.1.tar.gz
  • Upload date:
  • Size: 13.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.6.3

File hashes

Hashes for fhir_slicing-0.1.1.tar.gz
Algorithm Hash digest
SHA256 599f2f2ae23a4653727a31b7dffae5e3f04a7d6077b40910772b90ab7db8a21e
MD5 2dece8b8c4603bf872be1f33e70cbe5d
BLAKE2b-256 5f00b43de6ec8d1193bac7530079ddf2c6d725a2ed303b40b7edf469ce2cd7ed

See more details on using hashes here.

File details

Details for the file fhir_slicing-0.1.1-py3-none-any.whl.

File metadata

File hashes

Hashes for fhir_slicing-0.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 f02b0277310412db0bd2e5fc3a95de8bc7508e93e637fd29caf5c6b4b3f56ce6
MD5 ca67d6bbb09071430c60a4824f308dce
BLAKE2b-256 ad0a0f0b3a22bf863ca792c33d9277e250a445613908a63ac8b6ec12c73f24c6

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