2022-10-06 13:49:10 +00:00
|
|
|
from unittest import TestCase
|
|
|
|
|
|
|
|
import io
|
|
|
|
import os
|
|
|
|
import networkx as nx
|
|
|
|
|
|
|
|
from os.path import join
|
|
|
|
|
2022-10-13 20:43:16 +00:00
|
|
|
from soil import config, network, environment, agents, simulation
|
|
|
|
from test_main import CustomAgent
|
2022-10-06 13:49:10 +00:00
|
|
|
|
|
|
|
ROOT = os.path.abspath(os.path.dirname(__file__))
|
2022-10-16 15:54:03 +00:00
|
|
|
EXAMPLES = join(ROOT, "..", "examples")
|
2022-10-06 13:49:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TestNetwork(TestCase):
|
|
|
|
def test_load_graph(self):
|
|
|
|
"""
|
|
|
|
Load a graph from file if the extension is known.
|
|
|
|
Raise an exception otherwise.
|
|
|
|
"""
|
2022-10-16 15:54:03 +00:00
|
|
|
config = {"network_params": {"path": join(ROOT, "test.gexf")}}
|
|
|
|
G = network.from_config(config["network_params"])
|
2022-10-06 13:49:10 +00:00
|
|
|
assert G
|
|
|
|
assert len(G) == 2
|
|
|
|
with self.assertRaises(AttributeError):
|
2022-10-16 15:54:03 +00:00
|
|
|
config = {"network_params": {"path": join(ROOT, "unknown.extension")}}
|
|
|
|
G = network.from_config(config["network_params"])
|
2022-10-06 13:49:10 +00:00
|
|
|
print(G)
|
|
|
|
|
|
|
|
def test_generate_barabasi(self):
|
|
|
|
"""
|
|
|
|
If no path is given, a generator and network parameters
|
|
|
|
should be used to generate a network
|
|
|
|
"""
|
2022-10-16 15:54:03 +00:00
|
|
|
cfg = {"params": {"generator": "barabasi_albert_graph"}}
|
2022-10-06 13:49:10 +00:00
|
|
|
with self.assertRaises(Exception):
|
|
|
|
G = network.from_config(cfg)
|
2022-10-16 15:54:03 +00:00
|
|
|
cfg["params"]["n"] = 100
|
|
|
|
cfg["params"]["m"] = 10
|
2022-10-06 13:49:10 +00:00
|
|
|
G = network.from_config(cfg)
|
|
|
|
assert len(G) == 100
|
|
|
|
|
|
|
|
def test_save_geometric(self):
|
|
|
|
"""
|
2022-10-16 15:54:03 +00:00
|
|
|
There is a bug in networkx that prevents it from creating a GEXF file
|
2022-10-06 13:49:10 +00:00
|
|
|
from geometric models. We should work around it.
|
|
|
|
"""
|
|
|
|
G = nx.random_geometric_graph(20, 0.1)
|
|
|
|
env = environment.NetworkEnvironment(topology=G)
|
|
|
|
f = io.BytesIO()
|
2022-10-16 15:54:03 +00:00
|
|
|
assert env.G
|
|
|
|
network.dump_gexf(env.G, f)
|
2022-10-13 20:43:16 +00:00
|
|
|
|
|
|
|
def test_networkenvironment_creation(self):
|
|
|
|
"""Networkenvironment should accept netconfig as parameters"""
|
|
|
|
model_params = {
|
2022-10-16 15:54:03 +00:00
|
|
|
"topology": {"path": join(ROOT, "test.gexf")},
|
|
|
|
"agents": {
|
|
|
|
"topology": True,
|
|
|
|
"distribution": [
|
|
|
|
{
|
|
|
|
"agent_class": CustomAgent,
|
|
|
|
}
|
|
|
|
],
|
2022-10-13 20:43:16 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
env = environment.Environment(**model_params)
|
2022-10-16 15:54:03 +00:00
|
|
|
assert env.G
|
2022-10-13 20:43:16 +00:00
|
|
|
env.step()
|
2022-10-16 15:54:03 +00:00
|
|
|
assert len(env.G) == 2
|
2022-10-13 20:43:16 +00:00
|
|
|
assert len(env.agents) == 2
|
2022-10-16 15:54:03 +00:00
|
|
|
assert env.agents[1].count_agents(state_id="normal") == 2
|
|
|
|
assert env.agents[1].count_agents(state_id="normal", limit_neighbors=True) == 1
|
2022-10-13 20:43:16 +00:00
|
|
|
assert env.agents[0].neighbors == 1
|
2022-10-06 13:49:10 +00:00
|
|
|
|
|
|
|
def test_custom_agent_neighbors(self):
|
|
|
|
"""Allow for search of neighbors with a certain state_id"""
|
|
|
|
config = {
|
2022-10-16 15:54:03 +00:00
|
|
|
"model_params": {
|
|
|
|
"topology": {"path": join(ROOT, "test.gexf")},
|
|
|
|
"agents": {
|
|
|
|
"topology": True,
|
|
|
|
"distribution": [{"weight": 1, "agent_class": CustomAgent}],
|
|
|
|
},
|
2022-10-06 13:49:10 +00:00
|
|
|
},
|
2022-10-16 15:54:03 +00:00
|
|
|
"max_time": 10,
|
2022-10-06 13:49:10 +00:00
|
|
|
}
|
|
|
|
s = simulation.from_config(config)
|
|
|
|
env = s.run_simulation(dry_run=True)[0]
|
2022-10-16 15:54:03 +00:00
|
|
|
assert env.agents[1].count_agents(state_id="normal") == 2
|
|
|
|
assert env.agents[1].count_agents(state_id="normal", limit_neighbors=True) == 1
|
2022-10-06 13:49:10 +00:00
|
|
|
assert env.agents[0].neighbors == 1
|
|
|
|
|
2022-10-13 20:43:16 +00:00
|
|
|
def test_subgraph(self):
|
2022-10-16 15:54:03 +00:00
|
|
|
"""An agent should be able to subgraph the global topology"""
|
2022-10-13 20:43:16 +00:00
|
|
|
G = nx.Graph()
|
|
|
|
G.add_node(3)
|
|
|
|
G.add_edge(1, 2)
|
|
|
|
distro = agents.calculate_distribution(agent_class=agents.NetworkAgent)
|
2022-10-16 15:54:03 +00:00
|
|
|
aconfig = config.AgentConfig(distribution=distro, topology=True)
|
|
|
|
env = environment.Environment(name="Test", topology=G, agents=aconfig)
|
2022-10-13 20:43:16 +00:00
|
|
|
lst = list(env.network_agents)
|
|
|
|
|
|
|
|
a2 = env.find_one(node_id=2)
|
|
|
|
a3 = env.find_one(node_id=3)
|
|
|
|
assert len(a2.subgraph(limit_neighbors=True)) == 2
|
|
|
|
assert len(a3.subgraph(limit_neighbors=True)) == 1
|
|
|
|
assert len(a3.subgraph(limit_neighbors=True, center=False)) == 0
|
|
|
|
assert len(a3.subgraph(agent_class=agents.NetworkAgent)) == 3
|