"""
===============
Diffusion Field
===============
"""
import sys
import os
import argparse
import copy
import numpy as np
from scipy import constants
from scipy.ndimage import convolve
from vivarium.core.process import Process, assoc_path
from vivarium.core.engine import Engine
from vivarium.core.composition import PROCESS_OUT_DIR
from vivarium.library.units import units, remove_units
from vivarium.library.topology import get_in
from vivarium.library.dict_utils import deep_merge
from ecoli.library.lattice_utils import (
get_bin_site,
get_bin_volume,
make_gradient,
apply_exchanges,
ExchangeAgent,
make_diffusion_schema,
)
from ecoli.analysis.colony.snapshots import plot_snapshots
NAME = "diffusion_field"
# 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 DiffusionField(Process):
"""
Diffusion in 2-dimensional fields of molecules with agent exchange
Agent uptake and secretion occurs at agent locations.
Notes:
* Diffusion constant of glucose in 0.5 and 1.5 percent agarose gel
is around :math:`6 * 10^{-10} \\frac{m^2}{s}` (Weng et al. 2005.
Transport of glucose and poly(ethylene glycol)s in agarose gels).
* Conversion to micrometers:
:math:`6 * 10^{-10} \\frac{m^2}{s}=600 \\frac{micrometers^2}{s}`.
"""
name = NAME
defaults = {
"time_step": 1,
"molecules": ["glc"],
"initial_state": {},
"n_bins": [10, 10],
"bounds": [10 * units.um, 10 * units.um],
"depth": 3000.0 * units.um, # um
"diffusion": 5e-1 * units.um**2 / units.sec,
"gradient": {},
"exchanges_path": ("boundary", "exchanges"),
"external_path": ("boundary", "external"),
"location_path": ("boundary", "location"),
}
def __init__(self, parameters=None):
super().__init__(parameters)
# initial state
self.molecule_ids = self.parameters["molecules"]
self.initial = self.parameters["initial_state"]
# parameters
self.n_bins = self.parameters["n_bins"]
self.bounds = self.parameters["bounds"]
depth = self.parameters["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))
# volume, to convert between counts and concentration
self.bin_volume = get_bin_volume(self.n_bins, self.bounds, depth)
# initialize gradient fields
gradient = self.parameters["gradient"]
if gradient:
unitless_bounds = [bound.to(units.um).magnitude for bound in self.bounds]
gradient_fields = make_gradient(gradient, self.n_bins, unitless_bounds)
self.initial.update(gradient_fields)
self.exchanges_path = tuple(self.parameters["exchanges_path"])
self.external_path = tuple(self.parameters["external_path"])
self.location_path = tuple(self.parameters["location_path"])
[docs]
def initial_state(self, config=None):
return {
"fields": {
field: self.initial.get(field, self.ones_field())
for field in self.molecule_ids
},
}
[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"]
# 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,
)
# diffuse field
new_fields = self.diffuse(new_fields, timestep)
# 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": agent_updates,
}
deep_merge(update["agents"], local_environments)
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 ones_field(self):
return np.ones((self.n_bins[0], self.n_bins[1]), dtype=np.float64)
# diffusion functions
[docs]
def diffusion_delta(self, field, timestep):
"""calculate concentration changes cause by 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 - field, 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
# testing
[docs]
def get_random_field_config():
n_bins = (20, 20)
return {
"molecules": ["glc"],
"initial_state": {"glc": 1.0 * np.random.rand(n_bins[0], n_bins[1])},
"n_bins": n_bins,
"bounds": (20, 20) * units.um,
"depth": 1e-2 * units.um,
"diffusion": 1e-2 * units.um**2 / units.sec, # slow diffusion
}
[docs]
def get_gaussian_config():
return {
"molecules": ["glc"],
"n_bins": (20, 20),
"bounds": (20, 20) * units.um,
"depth": 100 * units.um,
"gradient": {
"type": "gaussian",
"molecules": {"glc": {"center": [0.5, 0.5], "deviation": 1}},
},
}
[docs]
def get_exponential_config():
return {
"molecules": ["glc"],
"n_bins": (20, 20),
"bounds": (20, 20) * units.um,
"depth": 100 * units.um,
"gradient": {
"type": "exponential",
"molecules": {
"glc": {"center": [1.0, 1.0], "base": 1 + 1e-3, "scale": 10.0}
},
},
}
def test_all():
run_diffusion_field(
config=get_random_field_config(), total_time=60, filename="random"
)
run_diffusion_field(
config=get_gaussian_config(), total_time=60, filename="gaussian"
)
run_diffusion_field(
config=get_exponential_config(), total_time=60, filename="exponential"
)
[docs]
def plot_fields(data, config, out_dir="out", filename="fields"):
unitless_config = remove_units(config)
fields = {time: time_data["fields"] for time, time_data in data.items()}
plot_snapshots(
unitless_config["bounds"], fields=fields, out_dir=out_dir, filename=filename
)
[docs]
def run_diffusion_field(config=None, total_time=100, filename="snapshots"):
config = config or {}
diff_process = DiffusionField(config)
# make the toy exchange agent
agent_id = "0"
agent_params = {
"mol_ids": ["glc"],
"default_exchange": 100,
"max_move": 1.0,
}
agent_process = ExchangeAgent(agent_params)
# get initial fields
initial_fields = diff_process.initial_state()
# put them together in a simulation
sim = Engine(
processes={
"diff": diff_process,
"agents": {agent_id: {"exchange": agent_process}},
},
topology={
"diff": {port: (port,) for port in 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=filename,
)
# uv run ecoli/processes/environment/diffusion_field.py
if __name__ == "__main__":
# test_all()
parser = argparse.ArgumentParser(description="diffusion_field")
parser.add_argument("--random", "-r", action="store_true", default=False)
parser.add_argument("--gaussian", "-g", action="store_true", default=False)
parser.add_argument("--exponential", "-e", action="store_true", default=False)
args = parser.parse_args()
no_args = len(sys.argv) == 1
if no_args:
test_all()
if args.random:
run_diffusion_field(
config=get_random_field_config(), total_time=60, filename="random"
)
if args.gaussian:
run_diffusion_field(
config=get_gaussian_config(), total_time=60, filename="gaussian"
)
if args.exponential:
run_diffusion_field(
config=get_exponential_config(), total_time=60, filename="exponential"
)