2023-04-26 23:45:14 +02:00
|
|
|
"""
|
|
|
|
agent.py - Part of ants project
|
|
|
|
|
|
|
|
This model implements the actual agents on the grid (a.k.a. the ants)
|
|
|
|
|
|
|
|
License: AGPL 3 (see end of file)
|
|
|
|
(C) Alexander Bocken, Viviane Fahrni, Grace Kragho
|
|
|
|
"""
|
|
|
|
import numpy as np
|
2023-05-07 15:24:27 +02:00
|
|
|
import numpy.typing as npt
|
2023-04-26 23:45:14 +02:00
|
|
|
from mesa.agent import Agent
|
|
|
|
from mesa.space import Coordinate
|
|
|
|
|
|
|
|
class RandomWalkerAnt(Agent):
|
2023-04-28 19:10:33 +02:00
|
|
|
def __init__(self, unique_id, model, look_for_chemical=None,
|
2023-05-07 15:24:27 +02:00
|
|
|
energy_0=1, chemical_drop_rate_0=1, sensitvity_0=0.1,
|
|
|
|
alpha=0.5, drop_chemical=None,
|
|
|
|
) -> None:
|
2023-04-26 23:45:14 +02:00
|
|
|
super().__init__(unique_id=unique_id, model=model)
|
|
|
|
|
|
|
|
self._next_pos : None | Coordinate = None
|
|
|
|
|
2023-04-28 14:45:56 +02:00
|
|
|
self.prev_pos : None | Coordinate = None
|
2023-04-28 19:10:33 +02:00
|
|
|
self.look_for_chemical = look_for_chemical
|
2023-04-29 11:00:42 +02:00
|
|
|
self.drop_chemical = drop_chemical
|
2023-04-26 23:45:14 +02:00
|
|
|
self.energy : float = energy_0
|
|
|
|
self.sensitvity : float = sensitvity_0
|
|
|
|
self.chemical_drop_rate : float = chemical_drop_rate_0 #TODO: check whether needs to be separated into A and B
|
|
|
|
self.alpha = alpha
|
|
|
|
|
2023-04-28 14:45:56 +02:00
|
|
|
|
2023-05-07 15:24:27 +02:00
|
|
|
def sens_adj(self, props) -> npt.NDArray[np.float_] | float:
|
|
|
|
# if props iterable create array, otherwise return single value
|
|
|
|
try:
|
|
|
|
iter(props)
|
|
|
|
except TypeError:
|
|
|
|
if props > self.sensitvity:
|
|
|
|
# TODO: nonlinear response
|
|
|
|
return props
|
|
|
|
else:
|
|
|
|
return 0
|
|
|
|
|
|
|
|
arr : list[float] = []
|
|
|
|
for prop in props:
|
|
|
|
arr.append(self.sens_adj(prop))
|
|
|
|
return np.array(arr)
|
|
|
|
|
2023-04-28 14:45:56 +02:00
|
|
|
|
2023-04-26 23:45:14 +02:00
|
|
|
def step(self):
|
2023-05-07 15:24:27 +02:00
|
|
|
# TODO: sensitvity decay
|
|
|
|
|
2023-04-29 11:00:42 +02:00
|
|
|
if self.prev_pos is None:
|
|
|
|
i = np.random.choice(range(6))
|
|
|
|
self._next_pos = self.neighbors()[i]
|
|
|
|
return
|
2023-05-07 15:24:27 +02:00
|
|
|
|
|
|
|
# Ants dropping A look for food
|
|
|
|
if self.drop_chemical == "A":
|
|
|
|
for neighbor in self.front_neighbors:
|
|
|
|
if self.model.grid.is_food(neighbor):
|
|
|
|
self.drop_chemical = "B"
|
|
|
|
self.prev_pos = neighbor
|
|
|
|
self._next_pos = self.pos
|
|
|
|
|
|
|
|
# Ants dropping B look for nest
|
|
|
|
elif self.drop_chemical == "B":
|
|
|
|
for neighbor in self.front_neighbors:
|
|
|
|
if self.model.grid.is_nest(neighbor):
|
|
|
|
self.look_for_chemical = "A" # Is this a correct interpretation?
|
|
|
|
self.drop_chemical = "A"
|
|
|
|
#TODO: Do we flip the ant here or reset prev pos?
|
|
|
|
# For now, flip ant just like at food
|
|
|
|
self.prev_pos = neighbor
|
|
|
|
self._next_pos = self.pos
|
|
|
|
|
|
|
|
for agent_id in self.model.get_unique_ids(self.model.num_new_recruits):
|
|
|
|
agent = RandomWalkerAnt(unique_id=agent_id, model=self.model, look_for_chemical="B", drop_chemical="A")
|
|
|
|
agent._next_pos = self.pos
|
|
|
|
self.model.schedule.add(agent)
|
|
|
|
self.model.grid.place_agent(agent, pos=neighbor)
|
|
|
|
|
|
|
|
# follow positive gradient
|
2023-04-28 19:10:33 +02:00
|
|
|
if self.look_for_chemical is not None:
|
|
|
|
front_concentration = [self.model.grid.fields[self.look_for_chemical][cell] for cell in self.front_neighbors ]
|
2023-05-07 15:24:27 +02:00
|
|
|
front_concentration = self.sens_adj(front_concentration)
|
|
|
|
current_pos_concentration = self.sens_adj(self.model.grid.fields[self.look_for_chemical][self.pos])
|
|
|
|
gradient = front_concentration - np.repeat(current_pos_concentration, 3)
|
2023-04-28 19:10:33 +02:00
|
|
|
index = np.argmax(gradient)
|
|
|
|
if gradient[index] > 0:
|
|
|
|
self._next_pos = self.front_neighbors[index]
|
|
|
|
return
|
|
|
|
|
|
|
|
# do biased random walk
|
|
|
|
p = np.random.uniform()
|
|
|
|
if p < self.alpha:
|
|
|
|
self._next_pos = self.front_neighbor
|
|
|
|
else:
|
|
|
|
# need copy() as we would otherwise remove the tuple from all possible lists (aka python "magic")
|
|
|
|
other_neighbors = self.neighbors().copy()
|
|
|
|
other_neighbors.remove(self.front_neighbor)
|
|
|
|
random_index = np.random.choice(range(len(other_neighbors)))
|
|
|
|
self._next_pos = other_neighbors[random_index]
|
2023-04-28 14:45:56 +02:00
|
|
|
|
2023-04-28 19:10:33 +02:00
|
|
|
def drop_chemicals(self) -> None:
|
|
|
|
# should only be called in advance() as we do not use hidden fields
|
|
|
|
if self.drop_chemical is not None:
|
|
|
|
self.model.grid.fields[self.drop_chemical][self.pos] += self.chemical_drop_rate
|
2023-04-26 23:45:14 +02:00
|
|
|
|
|
|
|
def advance(self) -> None:
|
2023-04-28 14:45:56 +02:00
|
|
|
self.drop_chemicals()
|
2023-04-28 19:10:33 +02:00
|
|
|
self.prev_pos = self.pos
|
2023-04-29 11:00:42 +02:00
|
|
|
self.model.grid.move_agent(self, self._next_pos)
|
2023-04-28 14:45:56 +02:00
|
|
|
|
2023-04-28 19:10:33 +02:00
|
|
|
# TODO: find out how to decorate with property properly
|
|
|
|
def neighbors(self, pos=None, include_center=False):
|
|
|
|
if pos is None:
|
|
|
|
pos = self.pos
|
|
|
|
return self.model.grid.get_neighborhood(pos, include_center=include_center)
|
|
|
|
|
2023-04-26 23:45:14 +02:00
|
|
|
@property
|
|
|
|
def front_neighbors(self):
|
2023-04-28 19:10:33 +02:00
|
|
|
"""
|
|
|
|
returns all three neighbors which the ant can see
|
|
|
|
"""
|
|
|
|
assert(self.prev_pos is not None)
|
|
|
|
all_neighbors = self.neighbors()
|
|
|
|
neighbors_at_the_back = self.neighbors(pos=self.prev_pos, include_center=True)
|
|
|
|
return list(filter(lambda i: i not in neighbors_at_the_back, all_neighbors))
|
|
|
|
|
|
|
|
@property
|
|
|
|
def front_neighbor(self):
|
|
|
|
"""
|
|
|
|
returns neighbor of current pos
|
|
|
|
which is towards the front of the ant
|
|
|
|
"""
|
|
|
|
neighbors_prev_pos = self.neighbors(self.prev_pos)
|
|
|
|
for candidate in self.front_neighbors:
|
|
|
|
# neighbor in front direction only shares current pos as neighborhood with prev_pos
|
|
|
|
candidate_neighbors = self.model.grid.get_neighborhood(candidate)
|
|
|
|
overlap = [x for x in candidate_neighbors if x in neighbors_prev_pos]
|
|
|
|
if len(overlap) == 1:
|
|
|
|
return candidate
|
|
|
|
|
2023-04-26 23:45:14 +02:00
|
|
|
|
|
|
|
"""
|
|
|
|
This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, version 3.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Affero General Public License along with this program. If not, see <https://www.gnu.org/licenses/>
|
|
|
|
"""
|