Source code for ecoli.library.updaters

"""
============================
Registry of Inverse Updaters
============================

You should interpret words and phrases that appear fully capitalized in
this document as described in :rfc:`2119`. Here is a brief summary of
the RFC:

* "MUST" indicates absolute requirements. Vivarium may not work
  correctly if you don't follow these.
* "SHOULD" indicates strong suggestions. You might have a valid reason
  for deviating from them, but be careful that you understand the
  ramifications.
* "MAY" indicates truly optional features that you can include or
  exclude as you wish.

----------------
Inverse Updaters
----------------

Inverse updaters accept a final and an initial state, and they attempt
to compute an update that could be given to a particular :term:`updater`
to get from the initial state to the final state.

Inverse Updater API
===================

An inverse updater MUST be registered with the same name as its
associated updater. The function MUST accept exactly 2 positional
arguments: the initial state and the final state. The function SHOULD
not accept any other parameters. The function MUST return only the
update needed to get from the initial state to the final state. The
function SHOULD have a name that matches its associated updater
function, only prefixed with ``inverse_``. Inverse updaters MUST return
an empty dictionary only if the update can be ignored.
"""

import numpy as np
from vivarium.core.registry import Registry


inverse_updater_registry = Registry()


[docs] def inverse_update_set(initial_state, final_state): if np.all(initial_state == final_state): return {} return final_state
[docs] def inverse_update_null(initial_state, final_state): """The null updater ignores the update.""" return {}
[docs] def inverse_update_accumulate(initial_state, final_state): if np.all(initial_state == final_state): return {} return final_state - initial_state
[docs] def inverse_update_nonnegative_accumulate(initial_state, final_state): if np.all(initial_state == final_state): return {} return final_state - initial_state
#: Special signal used by _reverse_deep_merge to communicate that no #: updates are needed. Using this signal reduces the update size by #: eliminating subtrees of empty updates. _NO_UPDATE_NEEDED = "No update needed"
[docs] def _reverse_deep_merge(initial, final): if not isinstance(initial, dict): if initial != final: return final return _NO_UPDATE_NEEDED update = {} for key in final: if key not in initial: update[key] = final[key] else: sub_update = _reverse_deep_merge(initial[key], final[key]) if sub_update is not _NO_UPDATE_NEEDED: update[key] = sub_update return update or _NO_UPDATE_NEEDED
[docs] def inverse_update_merge(initial_state, final_state): update = _reverse_deep_merge(initial_state, final_state) return {} if update is _NO_UPDATE_NEEDED else update
[docs] def inverse_update_bulk_numpy(initial_state, final_state): diff = final_state["count"] - initial_state["count"] if np.all(diff == 0): return [] return [ (np.arange(len(initial_state)), diff), ]
[docs] def inverse_update_unique_numpy(initial_state, final_state): return {"set": {field: final_state[field] for field in final_state.dtype.names}}