Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add cell-centric discrete spaces (experimental) #1994

Merged
merged 74 commits into from
Feb 27, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
74 commits
Select commit Hold shift + click to select a range
dd54f21
Add draft implementation for CellSpace
Corvince Jan 19, 2024
7b58d8e
add features to cell
Corvince Jan 23, 2024
aef4a8b
remove create_neighborhood_getter
Corvince Jan 23, 2024
3240a2b
update benchmark models
Corvince Jan 23, 2024
d3df44e
try to import Grid directly from experimental
Corvince Jan 23, 2024
c80a0bd
adds a pythonic WolfSheep implementation
quaquel Jan 24, 2024
93d7db5
replace radius check with value error in neighborhood
quaquel Jan 24, 2024
be7b843
minor further code cleanup of wolfsheep
quaquel Jan 24, 2024
bd788ec
add HexGrid
quaquel Jan 24, 2024
cb0ef18
typo fix
quaquel Jan 24, 2024
94f06b1
various updates
quaquel Jan 27, 2024
be6c1af
Merge remote-tracking branch 'upstream/main' into experimental/cell-s…
quaquel Jan 27, 2024
875d0ed
add NetworkGrid
quaquel Jan 27, 2024
e4d121d
add NetworkGrid
quaquel Jan 27, 2024
5769a05
change how empties is handled
quaquel Jan 27, 2024
82a64d9
further cleanup of handling of empties
quaquel Jan 27, 2024
5fdc787
correct handling of radius in Schelling large
quaquel Jan 27, 2024
152995c
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Jan 27, 2024
c524034
Merge remote-tracking branch 'upstream/main' into experimental/cell-s…
quaquel Jan 28, 2024
972ecc4
Update cell_space.py
quaquel Jan 28, 2024
a656d93
change from Random to random
quaquel Jan 28, 2024
8df9276
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Jan 28, 2024
f4a1089
small fixes and default capacity=None
Corvince Jan 28, 2024
1ebcf06
initial tests for cell_space
quaquel Jan 29, 2024
d362bae
some additional tests
quaquel Jan 29, 2024
c7d82d0
additional unit tests
quaquel Jan 30, 2024
b2842c0
restructure files and folders
Corvince Jan 31, 2024
740f003
various updates
quaquel Feb 2, 2024
2d8fcb5
additional tests and temporary fix for select_random_empty_cell
quaquel Feb 9, 2024
99ff668
Merge branch 'main' into experimental/cell-space
quaquel Feb 9, 2024
565ae0a
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 9, 2024
ab87b70
improved annotations
quaquel Feb 9, 2024
ab13bbc
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 9, 2024
8f2e3dd
Update discrete_space.py
quaquel Feb 9, 2024
ac8ff31
Merge branch 'experimental/cell-space' of https://github.com/Corvince…
quaquel Feb 15, 2024
aa03962
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 15, 2024
9b28f9d
correct handling of seeds when running examples in issolation
quaquel Feb 16, 2024
bbca3ac
Merge branch 'experimental/cell-space' of https://github.com/Corvince…
quaquel Feb 16, 2024
fe5a93f
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 16, 2024
2c80330
added docstrings
quaquel Feb 17, 2024
c79aaf0
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 17, 2024
8384cf1
reformating and minor update to tests
quaquel Feb 17, 2024
253531c
Merge branch 'experimental/cell-space' of https://github.com/Corvince…
quaquel Feb 17, 2024
ebdee8e
Add optional neighborhood_func to Grid class
Corvince Feb 17, 2024
bb2fc52
allow n-dimensional grids
Corvince Feb 17, 2024
b3efdee
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 17, 2024
8c7ad05
add validation, remove neighborhood_func, add tests
Corvince Feb 18, 2024
819e903
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 18, 2024
0f49dc7
Make Cell* generic
Corvince Feb 19, 2024
da94fb1
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 19, 2024
b210686
Make Cell* generic
Corvince Feb 19, 2024
c35591c
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 19, 2024
065e5ea
fixes to tests
quaquel Feb 19, 2024
1bfa009
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 19, 2024
6b1c454
type hint 3.9 fix in network.py
quaquel Feb 19, 2024
e0d5673
Merge branch 'experimental/cell-space' of https://github.com/Corvince…
quaquel Feb 19, 2024
cf789f2
fix type checking
Corvince Feb 19, 2024
50666c1
update capacity, add annotations import
Corvince Feb 19, 2024
e520d3c
fix wolf_sheep
Corvince Feb 20, 2024
85251f0
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 20, 2024
3a247cb
fix wolf_sheep
Corvince Feb 20, 2024
86127f2
Merge branch 'experimental/cell-space' of https://github.com/Corvince…
quaquel Feb 20, 2024
fb19879
seperate code path for 2d and nd grids when connecting cells
quaquel Feb 21, 2024
5ea5d35
seed as kwarg
quaquel Feb 21, 2024
b7ac86c
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 21, 2024
76ca4a5
code formatting fix
quaquel Feb 21, 2024
45e3e9f
testing
quaquel Feb 22, 2024
3d8cdfd
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 22, 2024
ce8ca0b
Merge branch 'main' into experimental/cell-space
quaquel Feb 22, 2024
18bd2a3
Merge branch 'experimental/cell-space' of https://github.com/Corvince…
quaquel Feb 22, 2024
62c844e
minor docstring update
quaquel Feb 23, 2024
adff7aa
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 23, 2024
7d5e6b0
Merge branch 'main' into experimental/cell-space
EwoutH Feb 27, 2024
b32af9f
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Feb 27, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
51 changes: 29 additions & 22 deletions benchmarks/Schelling/schelling.py
Original file line number Diff line number Diff line change
@@ -1,12 +1,14 @@
import mesa
from mesa import Model
from mesa.experimental.cell_space import CellAgent, OrthogonalMooreGrid
from mesa.time import RandomActivation


class SchellingAgent(mesa.Agent):
class SchellingAgent(CellAgent):
"""
Schelling segregation agent
"""

def __init__(self, unique_id, model, agent_type):
def __init__(self, unique_id, model, agent_type, radius, homophily):
"""
Create a new Schelling agent.
Args:
Expand All @@ -16,31 +18,32 @@ def __init__(self, unique_id, model, agent_type):
"""
super().__init__(unique_id, model)
self.type = agent_type
self.radius = radius
EwoutH marked this conversation as resolved.
Show resolved Hide resolved
self.homophily = homophily

def step(self):
similar = 0
for neighbor in self.model.grid.iter_neighbors(
self.pos, moore=True, radius=self.model.radius
):
neighborhood = self.cell.neighborhood(radius=self.radius)
for neighbor in neighborhood.agents:
if neighbor.type == self.type:
similar += 1

# If unhappy, move:
if similar < self.model.homophily:
self.model.grid.move_to_empty(self)
if similar < self.homophily:
self.move_to(self.model.grid.select_random_empty_cell())
else:
self.model.happy += 1


class Schelling(mesa.Model):
class Schelling(Model):
"""
Model class for the Schelling segregation model.
"""

def __init__(
self,
width=40,
height=40,
width=40,
homophily=3,
radius=1,
density=0.8,
Expand All @@ -51,35 +54,40 @@ def __init__(
Create a new Schelling model.

Args:
width, height: Size of the space.
height, width: Size of the space.
density: Initial Chance for a cell to populated
minority_pc: Chances for an agent to be in minority class
homophily: Minimum number of agents of same class needed to be happy
radius: Search radius for checking similarity
seed: Seed for Reproducibility
"""
super().__init__(seed=seed)
self.width = width
self.height = height
self.width = width
self.density = density
self.minority_pc = minority_pc
self.homophily = homophily
self.radius = radius

self.schedule = mesa.time.RandomActivation(self)
self.grid = mesa.space.SingleGrid(width, height, torus=True)
self.schedule = RandomActivation(self)
self.grid = OrthogonalMooreGrid(
[height, width],
torus=True,
capacity=1,
random=self.random,
)

self.happy = 0

# Set up agents
# We use a grid iterator that returns
# the coordinates of a cell as well as
# its contents. (coord_iter)
for _, pos in self.grid.coord_iter():
for cell in self.grid:
if self.random.random() < self.density:
agent_type = 1 if self.random.random() < self.minority_pc else 0
agent = SchellingAgent(self.next_id(), self, agent_type)
self.grid.place_agent(agent, pos)
agent = SchellingAgent(
self.next_id(), self, agent_type, radius, homophily
)
agent.move_to(cell)
self.schedule.add(agent)

def step(self):
Expand All @@ -93,13 +101,12 @@ def step(self):
if __name__ == "__main__":
import time

# model = Schelling(seed=15, width=40, height=40, homophily=3, radius=1, density=0.625)
# model = Schelling(seed=15, height=40, width=40, homophily=3, radius=1, density=0.625)
model = Schelling(
seed=15, width=100, height=100, homophily=8, radius=2, density=0.8
seed=15, height=100, width=100, homophily=8, radius=2, density=0.8
)

start_time = time.perf_counter()
for _ in range(100):
model.step()

print(time.perf_counter() - start_time)
68 changes: 34 additions & 34 deletions benchmarks/WolfSheep/wolf_sheep.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,41 +9,40 @@
Northwestern University, Evanston, IL.
"""

import mesa
import math

from mesa import Model
from mesa.experimental.cell_space import CellAgent, OrthogonalVonNeumannGrid
from mesa.time import RandomActivationByType

class Animal(mesa.Agent):
def __init__(self, unique_id, model, moore, energy, p_reproduce, energy_from_food):

class Animal(CellAgent):
def __init__(self, unique_id, model, energy, p_reproduce, energy_from_food):
super().__init__(unique_id, model)
self.energy = energy
self.p_reproduce = p_reproduce
self.energy_from_food = energy_from_food
self.moore = moore

def random_move(self):
next_moves = self.model.grid.get_neighborhood(self.pos, self.moore, True)
next_move = self.random.choice(next_moves)
# Now move:
self.model.grid.move_agent(self, next_move)
self.move_to(self.cell.neighborhood().select_random_cell())

def spawn_offspring(self):
self.energy /= 2
offspring = self.__class__(
self.model.next_id(),
self.model,
self.moore,
self.energy,
self.p_reproduce,
self.energy_from_food,
)
self.model.grid.place_agent(offspring, self.pos)
offspring.move_to(self.cell)
self.model.schedule.add(offspring)

def feed(self):
...

def die(self):
self.model.grid.remove_agent(self)
self.cell.remove_agent(self)
self.remove()

def step(self):
Expand All @@ -67,8 +66,9 @@ class Sheep(Animal):

def feed(self):
# If there is grass available, eat it
agents = self.model.grid.get_cell_list_contents(self.pos)
grass_patch = next(obj for obj in agents if isinstance(obj, GrassPatch))
grass_patch = next(
obj for obj in self.cell.agents if isinstance(obj, GrassPatch)
)
if grass_patch.fully_grown:
self.energy += self.energy_from_food
grass_patch.fully_grown = False
Expand All @@ -80,8 +80,7 @@ class Wolf(Animal):
"""

def feed(self):
agents = self.model.grid.get_cell_list_contents(self.pos)
sheep = [obj for obj in agents if isinstance(obj, Sheep)]
sheep = [obj for obj in self.cell.agents if isinstance(obj, Sheep)]
if len(sheep) > 0:
sheep_to_eat = self.random.choice(sheep)
self.energy += self.energy_from_food
Expand All @@ -90,7 +89,7 @@ def feed(self):
sheep_to_eat.die()


class GrassPatch(mesa.Agent):
class GrassPatch(CellAgent):
"""
A patch of grass that grows at a fixed rate and it is eaten by sheep
"""
Expand All @@ -100,7 +99,7 @@ def __init__(self, unique_id, model, fully_grown, countdown):
Creates a new patch of grass

Args:
grown: (boolean) Whether the patch of grass is fully grown or not
fully_grown: (boolean) Whether the patch of grass is fully grown or not
countdown: Time for the patch of grass to be fully grown again
"""
super().__init__(unique_id, model)
Expand All @@ -117,7 +116,7 @@ def step(self):
self.countdown -= 1


class WolfSheep(mesa.Model):
class WolfSheep(Model):
"""
Wolf-Sheep Predation Model

Expand All @@ -126,7 +125,6 @@ class WolfSheep(mesa.Model):

def __init__(
self,
seed,
height,
width,
initial_sheep,
Expand All @@ -136,7 +134,7 @@ def __init__(
grass_regrowth_time,
wolf_gain_from_food=13,
sheep_gain_from_food=5,
moore=False,
seed=None,
):
"""
Create a new Wolf-Sheep model with the given parameters.
Expand All @@ -152,6 +150,7 @@ def __init__(
once it is eaten
sheep_gain_from_food: Energy sheep gain from grass, if enabled.
moore:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Missing documentation.

seed
"""
super().__init__(seed=seed)
# Set parameters
Expand All @@ -161,24 +160,25 @@ def __init__(
self.initial_wolves = initial_wolves
self.grass_regrowth_time = grass_regrowth_time

self.schedule = mesa.time.RandomActivationByType(self)
self.grid = mesa.space.MultiGrid(self.height, self.width, torus=False)
self.schedule = RandomActivationByType(self)
self.grid = OrthogonalVonNeumannGrid(
[self.height, self.width],
torus=False,
capacity=math.inf,
random=self.random,
)

# Create sheep:
for _ in range(self.initial_sheep):
pos = (
self.random.randrange(self.width),
self.random.randrange(self.height),
)
energy = self.random.randrange(2 * sheep_gain_from_food)
sheep = Sheep(
self.next_id(),
self,
moore,
energy,
sheep_reproduce,
sheep_gain_from_food,
self.next_id(), self, energy, sheep_reproduce, sheep_gain_from_food
)
self.grid.place_agent(sheep, pos)
sheep.move_to(self.grid[pos])
self.schedule.add(sheep)

# Create wolves
Expand All @@ -189,21 +189,21 @@ def __init__(
)
energy = self.random.randrange(2 * wolf_gain_from_food)
wolf = Wolf(
self.next_id(), self, moore, energy, wolf_reproduce, wolf_gain_from_food
self.next_id(), self, energy, wolf_reproduce, wolf_gain_from_food
)
self.grid.place_agent(wolf, pos)
wolf.move_to(self.grid[pos])
self.schedule.add(wolf)

# Create grass patches
possibly_fully_grown = [True, False]
for _agent, pos in self.grid.coord_iter():
for cell in self.grid:
fully_grown = self.random.choice(possibly_fully_grown)
if fully_grown:
countdown = self.grass_regrowth_time
else:
countdown = self.random.randrange(self.grass_regrowth_time)
patch = GrassPatch(self.next_id(), self, fully_grown, countdown)
self.grid.place_agent(patch, pos)
patch.move_to(cell)
self.schedule.add(patch)

def step(self):
Expand All @@ -213,7 +213,7 @@ def step(self):
if __name__ == "__main__":
import time

model = WolfSheep(15, 25, 25, 60, 40, 0.2, 0.1, 20)
model = WolfSheep(25, 25, 60, 40, 0.2, 0.1, 20, seed=15)

start_time = time.perf_counter()
for _ in range(100):
Expand Down
4 changes: 4 additions & 0 deletions mesa/experimental/__init__.py
Original file line number Diff line number Diff line change
@@ -1 +1,5 @@
from .jupyter_viz import JupyterViz, make_text, Slider # noqa
from mesa.experimental import cell_space


__all__ = ["JupyterViz", "make_text", "Slider", "cell_space"]
23 changes: 23 additions & 0 deletions mesa/experimental/cell_space/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
from mesa.experimental.cell_space.cell import Cell
from mesa.experimental.cell_space.cell_agent import CellAgent
from mesa.experimental.cell_space.cell_collection import CellCollection
from mesa.experimental.cell_space.discrete_space import DiscreteSpace
from mesa.experimental.cell_space.grid import (
Grid,
HexGrid,
OrthogonalMooreGrid,
OrthogonalVonNeumannGrid,
)
from mesa.experimental.cell_space.network import Network

__all__ = [
"CellCollection",
"Cell",
"CellAgent",
"DiscreteSpace",
"Grid",
"HexGrid",
"OrthogonalMooreGrid",
"OrthogonalVonNeumannGrid",
"Network",
]
Loading