import os
import argparse
import numpy as np
from vivarium.core.composer import Composer
from vivarium.core.engine import Engine
from vivarium.core.composition import (
COMPOSITE_OUT_DIR,
)
from vivarium.library.dict_utils import deep_merge
from vivarium.library.units import units, remove_units
# processes
from ecoli.processes.environment.multibody_physics import (
Multibody,
make_random_position,
)
from ecoli.processes.environment.reaction_diffusion_field import ReactionDiffusion
from ecoli.composites.environment.grow_divide import GrowDivideExchange, GrowDivide
# plots
from ecoli.analysis.colony.snapshots import (
format_snapshot_data,
get_agent_ids,
plot_snapshots,
DEFAULT_SV,
)
NAME = "lattice_environment"
# make a configuration dictionary for the Lattice compartment
[docs]
def make_lattice_config(
time_step=None,
jitter_force=None,
bounds=None,
n_bins=None,
depth=None,
concentrations=None,
random_fields=None,
molecules=None,
diffusion=None,
keep_fields_emit=None,
set_config=None,
parallel=None,
):
config = {"multibody": {}, "reaction_diffusion": {}}
if time_step:
config["multibody"]["time_step"] = time_step
config["reaction_diffusion"]["time_step"] = time_step
if bounds is not None:
config["multibody"]["bounds"] = bounds
config["reaction_diffusion"]["bounds"] = bounds
config["reaction_diffusion"]["n_bins"] = remove_units(bounds)
if n_bins is not None:
config["reaction_diffusion"]["n_bins"] = n_bins
if jitter_force:
config["multibody"]["jitter_force"] = jitter_force
if depth:
config["reaction_diffusion"]["depth"] = depth
if diffusion:
config["reaction_diffusion"]["diffusion"] = diffusion
if concentrations:
config["reaction_diffusion"]["gradient"] = {
"type": "uniform",
"molecules": concentrations,
}
if random_fields:
config["reaction_diffusion"]["gradient"]["type"] = "random"
molecules = list(concentrations.keys())
config["reaction_diffusion"]["molecules"] = molecules
elif molecules:
# molecules are a list, assume uniform concentrations of 1
config["reaction_diffusion"]["molecules"] = molecules
if keep_fields_emit:
# by default no fields are emitted
config["reaction_diffusion"]["_schema"] = {
"fields": {
field_id: {"_emit": False}
for field_id in molecules
if field_id not in keep_fields_emit
}
}
if parallel:
config["reaction_diffusion"]["_parallel"] = True
config["multibody"]["_parallel"] = True
if set_config:
config = deep_merge(config, set_config)
return config
[docs]
class Lattice(Composer):
"""
Lattice: A two-dimensional lattice environmental model with multibody physics and diffusing molecular fields.
"""
name = NAME
defaults = {
# To exclude a process, from the compartment, set its
# configuration dictionary to None, e.g. colony_mass_deriver
"multibody": {
"bounds": [10, 10] * units.um,
},
"reaction_diffusion": {
"molecules": [],
"n_bins": [10, 10],
"bounds": [10, 10] * units.um,
"depth": 3000.0 * units.um,
"diffusion": 1e-2 * units.um**2 / units.sec,
},
}
def __init__(self, config=None):
super().__init__(config)
[docs]
def generate_processes(self, config):
processes = {
"multibody": Multibody(config["multibody"]),
"reaction_diffusion": ReactionDiffusion(config["reaction_diffusion"]),
}
return processes
[docs]
def generate_topology(self, config):
topology = {
"multibody": {"agents": ("agents",)},
"reaction_diffusion": {
"agents": ("agents",),
"fields": ("fields",),
"dimensions": ("dimensions",),
},
}
return topology
def test_lattice(
n_agents=1,
total_time=1000,
exchange=False,
external_molecule="X",
bounds=[25, 25] * units.um,
n_bins=None,
initial_field=None,
growth_rate=0.05, # fast growth
growth_noise=5e-4,
return_data=False,
):
# lattice configuration
lattice_config_kwargs = {
"bounds": bounds,
"n_bins": n_bins or remove_units(bounds),
"depth": 2 * units.um,
"diffusion": 1e-3 * units.um**2 / units.sec,
# 'time_step': 60,
"jitter_force": 1e-5,
"concentrations": {external_molecule: 1.0},
}
if initial_field is not None:
lattice_config_kwargs["concentrations"] = {external_molecule: initial_field}
lattice_config = make_lattice_config(**lattice_config_kwargs)
# agent configuration
agent_config = {
"growth": {"growth_rate": growth_rate, "default_growth_noise": growth_noise},
"divide_condition": {"threshold": 2500 * units.fg},
}
exchange_config = {"exchange": {"molecules": [external_molecule]}}
# lattice composer
lattice_composer = Lattice(lattice_config)
# agent composer
if exchange:
agent_composer = GrowDivideExchange({**agent_config, **exchange_config})
else:
agent_composer = GrowDivide(agent_config)
# make the composite
lattice_agent_composite = lattice_composer.generate()
# add agents
agent_ids = [str(agent_id) for agent_id in range(n_agents)]
for agent_id in agent_ids:
agent = agent_composer.generate({"agent_id": agent_id})
lattice_agent_composite.merge(composite=agent, path=("agents", agent_id))
# initial state
initial_state = {
# 'fields': {
# external_molecule: initial_field if (initial_field is not None) else np.ones((n_bins[0], n_bins[1]))},
"agents": {
agent_id: {
"boundary": {
"location": make_random_position(bounds),
"mass": 1500 * units.fg,
}
}
for agent_id in agent_ids
}
}
# make the experiment
experiment_config = {
"processes": lattice_agent_composite.processes,
"topology": lattice_agent_composite.topology,
"initial_state": initial_state,
"progress_bar": True,
}
spatial_experiment = Engine(**experiment_config)
# run the simulation
spatial_experiment.update(total_time)
data = spatial_experiment.emitter.get_data_unitless()
if return_data:
return data
[docs]
def main():
out_dir = os.path.join(COMPOSITE_OUT_DIR, NAME)
os.makedirs(out_dir, exist_ok=True)
parser = argparse.ArgumentParser(description="lattice composite")
parser.add_argument(
"-exchange",
"-e",
action="store_true",
default=False,
help="simulate agents with exchange",
)
args = parser.parse_args()
bounds = [25, 25] * units.um
total_time = 4000
n_agents = 3
if args.exchange:
# GrowDivide agents with Exchange
data = test_lattice(
exchange=True,
n_agents=n_agents,
total_time=total_time,
bounds=bounds,
return_data=True,
)
else:
# GrowDivide agents
n_bins = [20, 20]
initial_field = np.zeros((n_bins[0], n_bins[1]))
initial_field[:, -1] = 100
data = test_lattice(
n_agents=n_agents,
total_time=total_time,
bounds=bounds,
n_bins=n_bins,
initial_field=initial_field,
return_data=True,
)
# format the data for plot_snapshots
agents, fields = format_snapshot_data(data)
initial_ids = list(data[0]["agents"].keys())
agent_ids = get_agent_ids(agents)
# make colors based on initial agents
agent_colors = {}
hues = [n / 360 for n in [120, 270, 300, 240, 360, 30, 60]]
for idx, initial_id in enumerate(initial_ids):
hue = hues[idx]
color = [hue] + DEFAULT_SV
for agent_id in agent_ids:
if agent_id.startswith(initial_id, 0, len(initial_id)):
agent_colors[agent_id] = color
plot_snapshots(
bounds,
agents=agents,
fields=fields,
n_snapshots=4,
agent_colors=agent_colors,
out_dir=out_dir,
filename=f"lattice_snapshots{'_exchange' if args.exchange else ''}",
)
# uv run ecoli/composites/environment/lattice.py [-e if exchanges on]
if __name__ == "__main__":
main()