"""
========================
Reaction Diffusion Field
========================
"""
import copy
import os
import numpy as np
from pint import Quantity
from scipy import constants
from scipy.ndimage import convolve
from vivarium.core.process import Process, assoc_path
from vivarium.core.composition import PROCESS_OUT_DIR
from vivarium.core.engine import Engine
from vivarium.library.units import units
from vivarium.library.dict_utils import deep_merge
from ecoli.library.lattice_utils import (
get_bin_site,
get_bin_volume,
apply_exchanges,
ExchangeAgent,
make_gradient,
make_diffusion_schema,
)
from vivarium.library.topology import get_in
from ecoli.analysis.colony.snapshots import plot_snapshots
NAME = "reaction_diffusion"
# laplacian kernel for diffusion
LAPLACIAN_2D = np.array([[0.0, 1.0, 0.0], [1.0, -4.0, 1.0], [0.0, 1.0, 0.0]])
AVOGADRO = constants.N_A
[docs]
class ReactionDiffusion(Process):
"""
Reaction and Diffusion in 2-dimensional fields of molecules, with agent
uptake and secretion at agent locations.
Parameters: TODO
"""
name = NAME
defaults = {
"time_step": 1,
"molecules": [],
"n_bins": [10, 10],
"bounds": [10 * units.um, 10 * units.um],
"depth": 3000.0 * units.um, # um
# Diffusion coefficient: 10.1016/j.cej.2015.01.130
"diffusion": 600 * units.um**2 / units.sec,
"exchanges_path": ("boundary", "exchanges"),
"external_path": ("boundary", "external"),
"location_path": ("boundary", "location"),
# these parameters are not in diffusion_field
"reactions": {},
"kinetic_parameters": {},
"internal_time_step": 1,
# these parameters are in diffusion_field
# 'initial_state': {},
"gradient": None,
}
def __init__(self, parameters=None):
super().__init__(parameters)
# initial state
self.molecule_ids = self.parameters["molecules"]
# parameters
self.n_bins = self.parameters["n_bins"]
self.bounds = self.parameters["bounds"]
depth = self.parameters["depth"]
self.bin_volume = get_bin_volume(self.n_bins, self.bounds, depth)
# diffusion
diffusion = self.parameters["diffusion"]
bins_x = self.n_bins[0]
bins_y = self.n_bins[1]
length_x = self.bounds[0]
length_y = self.bounds[1]
dx = length_x / bins_x
dy = length_y / bins_y
dx2 = dx * dy
self.diffusion = diffusion / dx2
self.diffusion_dt = 0.01 * units.sec
# self.diffusion_dt = 0.5 * dx ** 2 * dy ** 2 / (2 * self.diffusion * (dx ** 2 + dy ** 2))
self.exchanges_path = tuple(self.parameters["exchanges_path"])
self.external_path = tuple(self.parameters["external_path"])
self.location_path = tuple(self.parameters["location_path"])
# check reactions
molecule_ids = set(self.molecule_ids)
for rxn_id, rxn_config in self.parameters["reactions"].items():
# check that 'stoichiometry' and 'catalyzed by' parameters
# are the only parameters in 'reactions', and that all declared
# reaction molecules are also in the declared molecule_ids
stoichiometry = rxn_config.get("stoichiometry")
catalyst = rxn_config.get("catalyzed by")
assert stoichiometry, f"reaction {rxn_id} is missing a stoichiometry"
assert catalyst, f"reaction {rxn_id} is missing a catalyst"
factors = set(stoichiometry.keys())
assert factors.issubset(molecule_ids), (
f"reaction {rxn_id} molecules {factors} "
f"are not in declared fields {molecule_ids}"
)
assert catalyst in molecule_ids, (
f"reaction {rxn_id} catalyst {catalyst} "
f"is not in declared fields {molecule_ids}"
)
# check corresponding kinetics. Each reaction must also have an entry
# under 'kinetic_parameters, with the catalyst id mapping to a kcat_f
# parameter and a km for one molecule in the environment
rxn_kinetics = self.parameters["kinetic_parameters"].get(rxn_id)
# TODO: Need better way to handle units
for rxn_kinetic_params in rxn_kinetics.values():
for param_name, rxn_kinetic_param in rxn_kinetic_params.items():
if isinstance(rxn_kinetic_param, Quantity):
rxn_kinetic_params[param_name] = rxn_kinetic_param.magnitude
assert (
rxn_kinetics
), "each reaction must have corresponding kinetic_parameters"
kinetics_catalyst = rxn_kinetics.keys()
assert (
len(kinetics_catalyst) == 1
), "each kinetic_parameters reaction must have only one enzyme"
assert set(kinetics_catalyst).issubset(molecule_ids), (
f"kinetic_parameters reaction {rxn_id} catalyst {catalyst} "
f"is not in declared fields {molecule_ids}"
)
kinetics_params = list(rxn_kinetics[list(kinetics_catalyst)[0]].keys())
assert "kcat_f" in kinetics_params, (
f"kinetic_parameters reaction {rxn_id} " f"needs a kcat_f parameters"
)
kinetics_params.remove("kcat_f")
assert len(kinetics_params) == 1, "only one km allowed"
assert set(kinetics_params).issubset(molecule_ids), (
f"kinetic_parameters reaction {rxn_id} substrate {kinetics_params} "
f"is not in declared fields {molecule_ids}"
)
[docs]
def initial_state(self, config=None):
"""
sets uniform initial state at the concentration provided for each the molecule_id in `config`
"""
config = config or {}
initial = {
"fields": {},
"dimensions": {
"bounds": self.parameters["bounds"],
"n_bins": self.parameters["n_bins"],
"depth": self.parameters["depth"],
},
}
# initialize gradient fields
gradient_molecules = []
gradient = self.parameters.get("gradient")
if gradient:
gradient_molecules = list(gradient["molecules"].keys())
unitless_bounds = [bound.to(units.um).magnitude for bound in self.bounds]
gradient_fields = make_gradient(gradient, self.n_bins, unitless_bounds)
initial["fields"].update(gradient_fields)
# uniform field for all molecules not in gradient
for mol_id in self.molecule_ids:
if mol_id not in gradient_molecules:
initial["fields"][mol_id] = config.get(mol_id, 0.0) * self.ones_field()
return initial
[docs]
def ports_schema(self):
return make_diffusion_schema(
self.exchanges_path,
self.external_path,
self.location_path,
self.parameters["bounds"],
self.parameters["n_bins"],
self.parameters["depth"],
self.molecule_ids,
self.ones_field(),
)
[docs]
def next_update(self, timestep, states):
fields = states["fields"]
agents = states["agents"]
dimensions = states["dimensions"]
# volume, to convert between counts and concentration
self.n_bins = dimensions["n_bins"]
self.bounds = dimensions["bounds"]
self.bin_volume = get_bin_volume(self.n_bins, self.bounds, dimensions["depth"])
# make new fields for the updated state
new_fields = copy.deepcopy(fields)
###################
# apply exchanges #
###################
new_fields, agent_updates = apply_exchanges(
agents,
new_fields,
self.exchanges_path,
self.location_path,
self.n_bins,
self.bounds,
self.bin_volume,
)
#####################
# react and diffuse #
#####################
t = 0
while t < timestep:
new_fields = self.react(new_fields, timestep)
new_fields = self.diffuse(new_fields, timestep)
t += self.parameters["internal_time_step"]
# get total delta from exchange, diffusion, reaction
delta_fields = {
mol_id: new_fields[mol_id] - field for mol_id, field in fields.items()
}
# get each agent's new local environment
local_environments = self.get_local_environments(agents, new_fields)
update = {"fields": delta_fields, "agents": local_environments}
deep_merge(update["agents"], agent_updates)
return update
[docs]
def get_bin_site(self, location):
return get_bin_site(location, self.n_bins, self.bounds)
[docs]
def get_single_local_environments(self, location, fields):
bin_site = self.get_bin_site(location)
local_environment = {}
for mol_id, field in fields.items():
local_environment[mol_id] = {
"_value": field[bin_site] * units.mM,
"_updater": "set",
}
return local_environment
[docs]
def get_local_environments(self, agents, fields):
local_environments = {}
if agents:
for agent_id, specs in agents.items():
assoc_path(
local_environments,
(agent_id,) + self.external_path,
self.get_single_local_environments(
get_in(specs, self.location_path),
fields,
),
)
return local_environments
[docs]
def zeros_field(self):
return np.zeros((self.n_bins[0], self.n_bins[1]), dtype=np.float64)
[docs]
def ones_field(self):
return np.ones((self.n_bins[0], self.n_bins[1]), dtype=np.float64)
[docs]
def diffusion_delta(self, field, timestep):
"""calculate new concentrations resulting from diffusion"""
field_new = field.copy()
t = 0.0
dt = min(timestep, self.diffusion_dt.to(units.sec).magnitude)
diffusion = self.diffusion.to(1 / units.sec).magnitude
while t < timestep:
field_new += (
diffusion * dt * convolve(field_new, LAPLACIAN_2D, mode="reflect")
)
t += dt
return field_new
[docs]
def diffuse(self, fields, timestep):
new_fields = {}
for mol_id, field in fields.items():
# run diffusion if molecule field is not uniform
if len(set(field.flatten())) != 1:
new_field = self.diffusion_delta(field, timestep)
else:
new_field = field
new_fields[mol_id] = new_field
return new_fields
[docs]
def react(self, fields, timestep):
new_fields = fields.copy()
reactions = self.parameters["reactions"]
kinetic_parameters = self.parameters["kinetic_parameters"]
for rxn_id, rxn in reactions.items():
# get the parameters
kinetics = kinetic_parameters[rxn_id]
stoich = rxn["stoichiometry"]
substrate_ids = list(stoich.keys())
# catalyst
catalyst_id = rxn["catalyzed by"]
kcat = kinetics[catalyst_id]["kcat_f"]
catalyst_field = fields[catalyst_id]
for substrate_id in substrate_ids:
substrate_km = kinetics[catalyst_id].get(substrate_id)
substrate_field = fields[substrate_id]
if np.sum(catalyst_field) > 0.0 and np.sum(substrate_field) > 0.0:
# calculate flux and delta
flux = kcat * catalyst_field * substrate_field
# add km term if declared
if substrate_km:
denominator = substrate_field + substrate_km
flux /= denominator
delta = stoich[substrate_id] * flux * timestep
# updates
new_fields[substrate_id] += delta
return new_fields
[docs]
def main():
total_time = 100
depth = 2 * units.um
n_bins = [50, 50]
bounds = [50 * units.um, 50 * units.um]
# make the reaction diffusion process
params = {
# 'boundary_path': ('a', 'b', 'c'), # needs to be 'boundary' for snapshots plot
"molecules": [
"beta-lactam",
"beta-lactamase",
],
"n_bins": n_bins,
"bounds": bounds,
"depth": depth,
"reactions": {
"antibiotic_hydrolysis": {
"stoichiometry": {
"beta-lactam": -1,
},
"catalyzed by": "beta-lactamase",
}
},
"kinetic_parameters": {
"antibiotic_hydrolysis": {
"beta-lactamase": {
"kcat_f": 10.0, # kcat for forward reaction
"beta-lactam": 0.1, # Km
},
},
},
}
rxn_diff_process = ReactionDiffusion(params)
# make the toy exchange agent
agent_id = "0"
agent_params = {"mol_ids": ["beta-lactamase"]}
agent_process = ExchangeAgent(agent_params)
# get initial fields
initial_fields = rxn_diff_process.initial_state({"beta-lactam": 1.0})
# initial_agents = {'agents': {agent_id: {'boundary': {'angle': 0.0}}}}
# put them together in a simulation
sim = Engine(
processes={
"rxn_diff": rxn_diff_process,
"agents": {agent_id: {"exchange": agent_process}},
},
topology={
"rxn_diff": {
port: (port,) for port in rxn_diff_process.ports_schema().keys()
},
"agents": {agent_id: {"exchange": {"boundary": ("boundary",)}}},
},
initial_state=initial_fields,
)
sim.update(total_time)
# plot
data = sim.emitter.get_data_unitless()
# add empty angle back in for the plot (this is undesirable)
for t in data.keys():
data[t]["agents"][agent_id]["boundary"]["angle"] = 0.0
data[t]["agents"][agent_id]["boundary"]["length"] = 1.0
data[t]["agents"][agent_id]["boundary"]["width"] = 1.0
out_dir = os.path.join(PROCESS_OUT_DIR, "environment", NAME)
if not os.path.exists(out_dir):
os.makedirs(out_dir)
plot_snapshots(
n_snapshots=6,
bounds=get_in(data, (max(data), "dimensions", "bounds")),
agents={time: d["agents"] for time, d in data.items()},
fields={time: d["fields"] for time, d in data.items()},
out_dir=out_dir,
filename="snapshots",
)
# uv run ecoli/processes/environment/reaction_diffusion_field.py
if __name__ == "__main__":
main()