Source code for vivarium.framework.time

"""
====================
The Simulation Clock
====================

The components here provide implementations of different kinds of simulation
clocks for use in ``vivarium``.

For more information about time in the simulation, see the associated
:ref:`concept note <time_concept>`.

"""
from datetime import datetime, timedelta
from functools import partial
from numbers import Number
from typing import TYPE_CHECKING, Callable, List, Union

import numpy as np
import pandas as pd

if TYPE_CHECKING:
    from vivarium.framework.engine import Builder
    from vivarium.framework.population.population_view import PopulationView
    from vivarium.framework.event import Event
    from vivarium.framework.population import SimulantData

from vivarium.framework.values import list_combiner
from vivarium.manager import Manager

Time = Union[pd.Timestamp, datetime, Number]
Timedelta = Union[pd.Timedelta, timedelta, Number]
NumberLike = Union[np.ndarray, pd.Series, pd.DataFrame, Number]


[docs] class SimulationClock(Manager): """A base clock that includes global clock and a pandas series of clocks for each simulant""" @property def name(self): return "simulation_clock" @property def columns_created(self) -> List[str]: return ["next_event_time", "step_size"] @property def columns_required(self) -> List[str]: return ["tracked"] @property def time(self) -> Time: """The current simulation time.""" if not self._clock_time: raise ValueError("No start time provided") return self._clock_time @property def stop_time(self) -> Time: """The time at which the simulation will stop.""" if not self._stop_time: raise ValueError("No stop time provided") return self._stop_time @property def minimum_step_size(self) -> Timedelta: """The minimum step size.""" if not self._minimum_step_size: raise ValueError("No minimum step size provided") return self._minimum_step_size @property def standard_step_size(self) -> Timedelta: """The standard varied step size.""" if not self._standard_step_size: raise ValueError("No standard step size provided") return self._standard_step_size @property def step_size(self) -> Timedelta: """The size of the next time step.""" if not self._clock_step_size: raise ValueError("No step size provided") return self._clock_step_size @property def event_time(self) -> Time: "Convenience method for event time, or clock + step" return self.time + self.step_size def __init__(self): self._clock_time: Time = None self._stop_time: Time = None self._minimum_step_size: Timedelta = None self._standard_step_size: Timedelta = None self._clock_step_size: Timedelta = None self._individual_clocks: PopulationView = None self._pipeline_name = "simulant_step_size" # TODO: Delegate this functionality to "tracked" or similar when appropriate self._simulants_to_snooze = pd.Index([])
[docs] def setup(self, builder: "Builder"): self._step_size_pipeline = builder.value.register_value_producer( self._pipeline_name, source=lambda idx: [pd.Series(np.nan, index=idx).astype("timedelta64[ns]")], preferred_combiner=list_combiner, preferred_post_processor=self.step_size_post_processor, ) self.register_step_modifier = partial( builder.value.register_value_modifier, self._pipeline_name ) builder.population.initializes_simulants( self.on_initialize_simulants, creates_columns=self.columns_created ) builder.event.register_listener("post_setup", self.on_post_setup) self._individual_clocks = builder.population.get_view( columns=self.columns_created + self.columns_required )
[docs] def on_post_setup(self, event: "Event") -> None: if not self._step_size_pipeline.mutators: ## No components modify the step size, so we use the default ## and remove the population view self._individual_clocks = None
[docs] def on_initialize_simulants(self, pop_data: "SimulantData") -> None: """Sets the next_event_time and step_size columns for each simulant""" if self._individual_clocks: clocks_to_initialize = pd.DataFrame( { "next_event_time": [self.event_time] * len(pop_data.index), "step_size": [self.step_size] * len(pop_data.index), }, index=pop_data.index, ) self._individual_clocks.update(clocks_to_initialize)
[docs] def simulant_next_event_times(self, index: pd.Index) -> pd.Series: """The next time each simulant will be updated.""" if not self._individual_clocks: return pd.Series(self.event_time, index=index) return self._individual_clocks.subview(["next_event_time", "tracked"]).get(index)[ "next_event_time" ]
[docs] def simulant_step_sizes(self, index: pd.Index) -> pd.Series: """The step size for each simulant.""" if not self._individual_clocks: return pd.Series(self.step_size, index=index) return self._individual_clocks.subview(["step_size", "tracked"]).get(index)[ "step_size" ]
[docs] def step_backward(self) -> None: """Rewinds the clock by the current step size.""" self._clock_time -= self.step_size
[docs] def step_forward(self, index: pd.Index) -> None: """Advances the clock by the current step size, and updates aligned simulant clocks.""" self._clock_time += self.step_size if self._individual_clocks and index.any(): update_index = self.get_active_simulants(index, self.time) clocks_to_update = self._individual_clocks.get(update_index) if not clocks_to_update.empty: clocks_to_update["step_size"] = self._step_size_pipeline(update_index) # Simulants that were flagged to get moved to the end should have a next event time # of stop time + 1 minimum timestep clocks_to_update.loc[self._simulants_to_snooze, "step_size"] = ( self.stop_time + self.minimum_step_size - self.time ) # TODO: Delegate this functionality to "tracked" or similar when appropriate self._simulants_to_snooze = pd.Index([]) clocks_to_update["next_event_time"] = ( self.time + clocks_to_update["step_size"] ) self._individual_clocks.update(clocks_to_update) self._clock_step_size = self.simulant_next_event_times(index).min() - self.time
[docs] def get_active_simulants(self, index: pd.Index, time: Time) -> pd.Index: """Gets population that is aligned with global clock""" if index.empty or not self._individual_clocks: return index next_event_times = self.simulant_next_event_times(index) return next_event_times[next_event_times <= time].index
[docs] def move_simulants_to_end(self, index: pd.Index) -> None: if self._individual_clocks and index.any(): self._simulants_to_snooze = self._simulants_to_snooze.union(index)
[docs] def step_size_post_processor(self, values: List[NumberLike], _) -> pd.Series: """Computes the largest feasible step size for each simulant. This is the smallest component-modified step size (rounded down to increments of the minimum step size), or the global step size, whichever is larger. If no components modify the step size, we default to the global step size. Parameters ---------- values A list of step sizes Returns ------- pandas.Series The largest feasible step size for each simulant """ min_modified = pd.DataFrame(values).min(axis=0).fillna(self.standard_step_size) ## Rescale pipeline values to global minimum step size discretized_step_sizes = ( np.floor(min_modified / self.minimum_step_size).replace(0, 1) * self.minimum_step_size ) ## Make sure we don't get zero return discretized_step_sizes
[docs] class SimpleClock(SimulationClock): """A unitless step-count based simulation clock.""" CONFIGURATION_DEFAULTS = { "time": { "start": 0, "end": 100, "step_size": 1, "standard_step_size": None, } } @property def name(self): return "simple_clock"
[docs] def setup(self, builder): super().setup(builder) time = builder.configuration.time self._clock_time = time.start self._stop_time = time.end self._minimum_step_size = time.step_size self._standard_step_size = ( time.standard_step_size if time.standard_step_size else self._minimum_step_size ) self._clock_step_size = self._standard_step_size
def __repr__(self): return "SimpleClock()"
[docs] def get_time_stamp(time): return pd.Timestamp(time["year"], time["month"], time["day"])
[docs] class DateTimeClock(SimulationClock): """A date-time based simulation clock.""" CONFIGURATION_DEFAULTS = { "time": { "start": {"year": 2005, "month": 7, "day": 2}, "end": { "year": 2010, "month": 7, "day": 2, }, "step_size": 1, "standard_step_size": None, # Days } } @property def name(self): return "datetime_clock"
[docs] def setup(self, builder): super().setup(builder) time = builder.configuration.time self._clock_time = get_time_stamp(time.start) self._stop_time = get_time_stamp(time.end) self._minimum_step_size = pd.Timedelta( days=time.step_size // 1, hours=(time.step_size % 1) * 24 ) self._standard_step_size = ( pd.Timedelta( days=time.standard_step_size // 1, hours=(time.standard_step_size % 1) * 24 ) if time.standard_step_size else self._minimum_step_size ) self._clock_step_size = self._minimum_step_size
def __repr__(self): return "DateTimeClock()"
[docs] class TimeInterface: def __init__(self, manager: SimulationClock): self._manager = manager
[docs] def clock(self) -> Callable[[], Time]: """Gets a callable that returns the current simulation time.""" return lambda: self._manager.time
[docs] def step_size(self) -> Callable[[], Timedelta]: """Gets a callable that returns the current simulation step size.""" return lambda: self._manager.step_size
[docs] def simulant_next_event_times(self) -> Callable[[pd.Index], pd.Series]: """Gets a callable that returns the next event times for simulants.""" return self._manager.simulant_next_event_times
[docs] def simulant_step_sizes(self) -> Callable[[pd.Index], pd.Series]: """Gets a callable that returns the simulant step sizes.""" return self._manager.simulant_step_sizes
[docs] def move_simulants_to_end(self) -> Callable[[pd.Index], None]: """Gets a callable that moves simulants to the end of the simulation""" return self._manager.move_simulants_to_end
[docs] def register_step_size_modifier( self, modifier: Callable[[pd.Index], pd.Series], requires_columns: List[str] = (), requires_values: List[str] = (), requires_streams: List[str] = (), ) -> None: """Registers a step size modifier. Parameters ---------- modifier Modifier of the step size pipeline. Modifiers can take an index and should return a series of step sizes. requires_columns A list of the state table columns that already need to be present and populated in the state table before the modifier is called. requires_values A list of the value pipelines that need to be properly sourced before the modifier is called. requires_streams A list of the randomness streams that need to be properly sourced before the modifier is called.""" return self._manager.register_step_modifier( modifier, requires_columns, requires_values, requires_streams )