"""Data structures that describe a three-body decay."""
from __future__ import annotations
import sys
from typing import TYPE_CHECKING, Dict
from attrs import field, frozen
from attrs.validators import instance_of
from polarimetry._attrs import assert_spin_value, to_ls, to_rational
if TYPE_CHECKING:
import sympy as sp
if sys.version_info < (3, 8):
from typing_extensions import Literal
else:
from typing import Literal
[docs]@frozen
class Particle:
name: str
latex: str
spin: sp.Rational = field(converter=to_rational, validator=assert_spin_value)
parity: Literal[-1, 1]
mass: float
width: float
[docs]@frozen
class IsobarNode:
parent: Particle
child1: Particle | IsobarNode
child2: Particle | IsobarNode
interaction: LSCoupling | None = field(default=None, converter=to_ls)
@property
def children(self) -> tuple[Particle, Particle]:
return self.child1, self.child2
[docs]@frozen
class ThreeBodyDecay:
states: OuterStates
chains: tuple[ThreeBodyDecayChain, ...]
def __attrs_post_init__(self) -> None:
expected_initial_state = self.initial_state
expected_final_state = set(self.final_state.values())
for i, chain in enumerate(self.chains):
if chain.parent != expected_initial_state:
msg = (
f"Chain {i} has initial state {chain.parent.name}, but should have"
f" {expected_initial_state.name}"
)
raise ValueError(msg)
final_state = {chain.spectator, *chain.decay_products}
if final_state != expected_final_state:
to_str = lambda s: ", ".join(p.name for p in s) # noqa: E731
msg = (
f"Chain {i} has final state {to_str(final_state)}, but should have"
f" {to_str(expected_final_state)}"
)
raise ValueError(msg)
@property
def initial_state(self) -> Particle:
return self.states[0]
@property
def final_state(self) -> dict[Literal[1, 2, 3], Particle]:
return {k: v for k, v in self.states.items() if k != 0}
[docs] def find_chain(self, resonance_name: str) -> ThreeBodyDecayChain:
for chain in self.chains:
if chain.resonance.name == resonance_name:
return chain
msg = f"No decay chain found for resonance {resonance_name}"
raise KeyError(msg)
[docs] def get_subsystem(self, subsystem_id: Literal[1, 2, 3]) -> ThreeBodyDecay:
child1_id, child2_id = get_decay_product_ids(subsystem_id)
child1 = self.final_state[child1_id]
child2 = self.final_state[child2_id]
filtered_chains = [
chain
for chain in self.chains
if chain.decay_products in {(child1, child2), (child2, child1)}
]
return ThreeBodyDecay(self.states, filtered_chains)
[docs]def get_decay_product_ids(spectator_id: Literal[1, 2, 3]) -> tuple[int, int]:
if spectator_id == 1:
return 2, 3
if spectator_id == 2:
return 3, 1
if spectator_id == 3:
return 1, 2
msg = f"Spectator ID has to be one of 1, 2, 3, not {spectator_id}"
raise ValueError(msg)
OuterStates = Dict[Literal[0, 1, 2, 3], Particle]
"""Mapping of the initial and final state IDs to their `.Particle` definition."""
[docs]@frozen
class ThreeBodyDecayChain:
decay: IsobarNode = field(validator=instance_of(IsobarNode))
def __attrs_post_init__(self) -> None:
if not isinstance(self.decay.child1, IsobarNode):
msg = f"Child 1 has of type {IsobarNode.__name__} (the decay)"
raise TypeError(msg)
if not isinstance(self.decay.child1.child1, Particle):
msg = f"Child 1 of child 1 has of type {Particle.__name__}"
raise TypeError(msg)
if not isinstance(self.decay.child1.child1, Particle):
msg = f"Child 1 of child 1 has of type {Particle.__name__}"
raise TypeError(msg)
if not isinstance(self.decay.child2, Particle):
msg = f"Child 2 has of type {Particle.__name__} (spectator)"
raise TypeError(msg)
if self.incoming_ls is None: # pyright: ignore[reportUnnecessaryComparison]
msg = "LS-coupling for production node required"
raise ValueError(msg)
if self.outgoing_ls is None: # pyright: ignore[reportUnnecessaryComparison]
msg = "LS-coupling for decay node required"
raise ValueError(msg)
@property
def parent(self) -> Particle:
return self.decay.parent
@property
def resonance(self) -> Particle:
return self.decay.child1.parent
@property
def decay_products(self) -> tuple[Particle, Particle]:
return (
self.decay.child1.child1,
self.decay.child1.child2,
)
@property
def spectator(self) -> Particle:
return self.decay.child2
@property
def incoming_ls(self) -> LSCoupling:
return self.decay.interaction
@property
def outgoing_ls(self) -> LSCoupling:
return self.decay.child1.interaction
[docs]@frozen
class LSCoupling:
L: int
S: sp.Rational = field(converter=to_rational, validator=assert_spin_value)