2023-04-26 23:45:14 +02:00
|
|
|
"""
|
|
|
|
model.py - Part of ants project
|
|
|
|
|
|
|
|
This file implements the mesa model on which our ActiveRandomWalkerAnts
|
|
|
|
will act
|
|
|
|
|
|
|
|
License: AGPL 3 (see end of file)
|
2023-05-17 15:57:23 +02:00
|
|
|
(C) Alexander Bocken, Viviane Fahrni, Grace Kagho
|
2023-04-26 23:45:14 +02:00
|
|
|
"""
|
|
|
|
|
|
|
|
import numpy as np
|
|
|
|
from mesa.model import Model
|
|
|
|
from mesa.space import Coordinate, HexGrid, Iterable
|
2023-04-28 19:10:33 +02:00
|
|
|
from multihex import MultiHexGridScalarFields
|
2023-04-26 23:45:14 +02:00
|
|
|
from mesa.time import SimultaneousActivation
|
|
|
|
from mesa.datacollection import DataCollector
|
|
|
|
from agent import RandomWalkerAnt
|
|
|
|
|
|
|
|
class ActiveWalkerModel(Model):
|
|
|
|
def __init__(self, width : int, height : int , num_max_agents : int,
|
|
|
|
num_initial_roamers : int,
|
|
|
|
nest_position : Coordinate,
|
2023-05-17 18:09:26 +02:00
|
|
|
num_food_sources=5,
|
|
|
|
food_size=10,
|
2023-04-26 23:45:14 +02:00
|
|
|
max_steps:int=1000) -> None:
|
|
|
|
super().__init__()
|
2023-04-28 19:10:33 +02:00
|
|
|
fields=["A", "B", "nests", "food"]
|
2023-04-26 23:45:14 +02:00
|
|
|
self.schedule = SimultaneousActivation(self)
|
2023-04-28 14:45:56 +02:00
|
|
|
self.grid = MultiHexGridScalarFields(width=width, height=height, torus=True, fields=fields)
|
|
|
|
self._unique_id_counter = -1
|
2023-04-26 23:45:14 +02:00
|
|
|
|
|
|
|
self.max_steps = max_steps
|
2023-05-07 15:24:27 +02:00
|
|
|
self.grid.add_nest(nest_position)
|
2023-04-26 23:45:14 +02:00
|
|
|
self.num_max_agents = num_max_agents
|
2023-05-07 15:24:27 +02:00
|
|
|
self.num_new_recruits = 5
|
2023-04-26 23:45:14 +02:00
|
|
|
|
2023-05-07 15:24:27 +02:00
|
|
|
self.decay_rates : dict[str, float] = {"A" :0.01,
|
|
|
|
"B": 0.01,
|
2023-04-29 11:00:42 +02:00
|
|
|
}
|
2023-04-28 14:45:56 +02:00
|
|
|
|
2023-04-26 23:45:14 +02:00
|
|
|
for agent_id in self.get_unique_ids(num_initial_roamers):
|
2023-05-18 12:46:48 +02:00
|
|
|
if self.schedule.get_agent_count() < self.num_max_agents:
|
|
|
|
agent = RandomWalkerAnt(unique_id=agent_id, model=self, look_for_pheromone="A", drop_pheromone="A")
|
|
|
|
self.schedule.add(agent)
|
|
|
|
self.grid.place_agent(agent, pos=nest_position)
|
2023-04-26 23:45:14 +02:00
|
|
|
|
2023-05-17 18:09:26 +02:00
|
|
|
for _ in range(num_food_sources):
|
|
|
|
self.grid.add_food(food_size)
|
2023-05-07 15:24:27 +02:00
|
|
|
|
2023-04-26 23:45:14 +02:00
|
|
|
self.datacollector = DataCollector(
|
|
|
|
model_reporters={},
|
|
|
|
agent_reporters={}
|
|
|
|
)
|
|
|
|
self.datacollector.collect(self) # keep at end of __init___
|
|
|
|
|
2023-04-28 19:10:33 +02:00
|
|
|
def agent_density(self):
|
|
|
|
a = np.zeros((self.grid.width, self.grid.height))
|
|
|
|
for i in range(self.grid.width):
|
|
|
|
for j in range(self.grid.height):
|
|
|
|
a[i,j] = len(self.grid[(i,j)])
|
|
|
|
return a
|
2023-04-28 14:45:56 +02:00
|
|
|
|
|
|
|
|
2023-04-26 23:45:14 +02:00
|
|
|
def step(self):
|
2023-04-28 14:45:56 +02:00
|
|
|
self.schedule.step() # step() and advance() all agents
|
|
|
|
|
|
|
|
# apply decay rate on pheromone levels
|
|
|
|
for key in ("A", "B"):
|
|
|
|
field = self.grid.fields[key]
|
|
|
|
self.grid.fields[key] = field - self.decay_rates[key]*field
|
2023-05-17 15:57:23 +02:00
|
|
|
# TODO: plot to check whether exponential
|
2023-04-28 14:45:56 +02:00
|
|
|
|
2023-04-26 23:45:14 +02:00
|
|
|
self.datacollector.collect(self)
|
|
|
|
|
|
|
|
if self.schedule.steps >= self.max_steps:
|
|
|
|
self.running = False
|
|
|
|
|
|
|
|
def get_unique_id(self) -> int:
|
|
|
|
self._unique_id_counter += 1
|
|
|
|
return self._unique_id_counter
|
|
|
|
|
|
|
|
def get_unique_ids(self, num_ids : int):
|
|
|
|
for _ in range(num_ids):
|
|
|
|
yield self.get_unique_id()
|
|
|
|
|
|
|
|
"""
|
|
|
|
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/>
|
|
|
|
"""
|