Source code for mira.metamodel.units

__all__ = [
    'Unit',
    'person_units',
    'day_units',
    'per_day_units',
    'dimensionless_units',
    'per_day_per_person_units',
    'UNIT_SYMBOLS'
]

import os
from typing import Dict, Any

import sympy
from pydantic import BaseModel, Field
from .utils import SympyExprStr


def load_units():
    path = os.path.join(os.path.dirname(os.path.abspath(__file__)),
                        os.pardir, 'dkg', 'resources', 'unit_names.tsv')
    with open(path, 'r') as fh:
        units = {}
        for line in fh.readlines():
            symbol = line.strip()
            units[symbol] = sympy.Symbol(symbol)
    return units


UNIT_SYMBOLS = load_units()


[docs]class Unit(BaseModel): """A unit of measurement.""" class Config: arbitrary_types_allowed = True json_encoders = { SympyExprStr: lambda e: str(e), } json_decoders = { SympyExprStr: lambda e: sympy.parse_expr(e) } expression: SympyExprStr = Field( description="The expression for the unit." )
[docs] @classmethod def from_json(cls, data: Dict[str, Any]) -> "Unit": # Use get_sympy from amr.petrinet, but avoid circular import from mira.sources.amr.petrinet import get_sympy data["expression"] = get_sympy(data, local_dict=UNIT_SYMBOLS) assert data.get('expression') is None or not isinstance( data['expression'], str ) return cls(**data)
person_units = Unit(expression=sympy.Symbol('person')) day_units = Unit(expression=sympy.Symbol('day')) per_day_units = Unit(expression=1/sympy.Symbol('day')) dimensionless_units = Unit(expression=sympy.Integer('1')) per_day_per_person_units = Unit(expression=1/(sympy.Symbol('day')*sympy.Symbol('person')))