Skip to content
/ MASPY Public

A Python Framework for Multi-Agent Programming with Machine Learning

Notifications You must be signed in to change notification settings

laca-is/MASPY

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

MASPY Logo

MASPY is a Python Framework which aims to ease the development of a Multi-Agent Systems based on the BDI paradigm. In this paradigm, an agent will contain Beliefs, its knowledge, Desire, its wants, and Intentions, how it will achieve their wants.

MASPY creates an abstraction layer to model the agents and the environment where the agents will act. An agent may contain any number of Beliefs, Objectives, Plans, these to classes model desires and intentions, respectively. The agent can communicate with others agents, by sending or requesting any of the previous entitys. This Framework uses a Knowledge Query Model Language to model the communication between agents using acts of speech, or perfomatives, as a base model, in this way an agent can ask, tell, command or teach anything it needs and knows.

Table of Contents

Install

To install MASPY you can use package-management system pip:

pip install maspy-ml

To update your already installed version of MASPY to the latest one, you can use:

pip install maspy-ml -U

The minimum version of Python guarateed to work is 3.10, altough earlier versions may work.

Latest changes

maspy-v0.2.3:
- Properly addition of Intention Mechanisms
- Updated usage of environment plans

maspy-v0.2.2:
- Removed Numpy necessity

maspy-v0.2.1:
- New internal function to perceive environments during plans
- Fixed problem of overwriting perceived beliefs of the environment
- Better error explanations for debugging
- More concise system logging

maspy-v0.2.0:
- Better typing for functions
- Fixed perception speed
- Fixed ending multiple agents problem
- Better agent finder

About MASPY

The MASPY Framework is made of four main classes. The Agent Class for the managing beliefs, goals and plans. The Environment Class for providing perception and interaction context to agents The Communication Class for agents to exchange messages and information The Admin Class for configuration and control of the system

MASPY CLASSES

Each Agent follows the BDI reasoning and each cycle occors as presented in the following diagram:

MASPY REASONING

Using MASPY

Import

To use the Framework you need this simple import from maspy import * , nothing more or less.

Everything for MASPY to run correctly in imported this way.

Creating a new Agent

To create a new agent, you only need to extend Agent in your class, this adds all of the necessary logic to execute an agent. the following snippet shows how to create an DummyAgent.

To create an instance of any agent, only the Extension is needed.

Technically, this is a MASPY Agent:

Dummy Agent

from maspy import *

class DummyAgent(Agent):
    pass

my_agent = DummyAgent()
named_agent = DummyAgent("Ag")

When the snippet above is run, this is the printed result:

Starting MASPY Program
# Admin #> Registering Agent DummyAgent:('DummyAgent', 1)
Channel:default> Connecting agent DummyAgent:('DummyAgent', 1)
# Admin #> Registering Agent DummyAgent:('Ag', 1)
Channel:default> Connecting agent DummyAgent:('Ag', 1)

It will execute indeterminably, while doing nothing.

Initial Beliefs and Goals

The agent can start with some inital Beliefs or Goals.

For most of theses explanations, the code from "examples/ex_parking.py" will be used.

from maspy import *

class Driver(Agent):
    def __init__(self, agent_name=None):
        super().__init__(agent_name)
        self.add(Belief("budget",(rnd.randint(6,10),rnd.randint(12,20)),adds_event=False))
        self.add(Goal("park"))

driver = Driver("Drv")

Managing Beliefs and Goals

Here are some info about Beliefs and Goals being created and removed.

  • This function adds a goal to the agent;
  • The first field represents the goal key and must always be a string;
  • The second field represents arguments of the goal and will always be a tuple;
    • Each argument can have any structure, with each position of the tuple representing a different one;
  • The third field represents the goal source. It is "self" by default, or another agent.
  • adding or removing a goal always creates an event for agent, which will try to find a applicable plan.
agent = DummyAgent("Ag")
agent.add( Goal(key, args, source) )
agent.rm( Goal(key, args, source) )
        
agent.add( Goal("check_house", {"Area": [50,100], "Rooms": 5}, ("Seller",27) ) )
agent.add( Goal("SendInfo", ("Information",["List","of","Information",42]) ) )
agent.rm( Goal("walk", source=("trainer",2)) )
  • This function adds a belief to the agent;
  • The first an second field work exaclty the same way as the goal's
  • The third field represents the belief source. It is "self" by default, another agent or an environment.
  • The fourth field dictates if the adding or removing the belief will generate a new event.
    • By default it does, but sometimes one does not want a group of beliefs to be considerend new events
agent = DummyAgent("Ag")
agent.add( Belief(Key, Args, Source, Adds_Event) )
agent.rm( Belief(Key, Args, Source, Adds_Event) )

agent.add( Belief("Dirt", (("remaining",3),[(2,2),(3,7),(5,1)])) )
agent.rm( Belief("spot",("A",4,"free"),"Parking",False) )
agent.add( Belief("velocity",57) )

Defining plans

To define plans it is also really simple, it only needs the @pl decoration. This decoration must contain the plan change {gain, lose or test}, the data that changed {Belief(s) or Goal(s)} and optionally a context needed to be true to execute the plan {Belief(s) or Goal(s)}.

    change: TypeVar('gain'|'lose'|'test')
    changed_data: Iterable[Belief | Goal] | Belief | Goal
    context: Iterable[Belief | Goal] | Belief | Goal

    @pl(change, changed_data, context)
    def foo(self,src, *changed_data.args, *context.args):

Resuming the the driver example, you can implement a plan the following way:

from maspy import *

class Driver(Agent):
    def __init__(self, agent_name=None):
        super().__init__(agent_name)
        self.add(Belief("budget",(rnd.randint(6,10),rnd.randint(12,20)),adds_event=False))
        self.add(Goal("park"))

    # This plan will be executed whenever the agent gains the goal "checkPrice"
    # Every plan needs at least self and src, plus the arguments from the trigger and context
    # for this plan, the context is the belief of a budget with wanted and max prices
    @pl(gain,Goal("checkPrice", Any),Belief("budget",(Any, Any)))
    def check_price(self, src, given_price, budget):
        ...

driver = Driver("Drv")

Running the agents

Running the system is simple, given the utilities support we have in place. The Admin module contains a few useful methods to start and manage the system.

Starting all agents

In case you only need to start all agents, the following snippet is enough.

driver1 = Driver("Drv")
driver2 = Driver("Drv")

Admin().start_system()

In this example, both agents have the same name "Drv".

For communication to not be ambiguous, the Admin names them "Drv_1" and "Drv_2".

Communication between Agents

After starting the agents they may use the default channel or be connected to private one.

from maspy import *

class Manager(Agent):
    def __init__(self, agt_name=None):
        super().__init__(agt_name,show_exec=False,show_cycle=False)
        self.add(Belief("spotPrice",rnd.randint(12,20),adds_event=False))

    @pl(gain,Goal("sendPrice"),Belief("spotPrice", Any))
    def send_price(self,src,spot_price):
        # The agent manager sends a goal to the manager via the Parking channel
        self.send(src,achieve,Goal("checkPrice",spot_price),"Parking")

class Driver(Agent):
    def __init__(self, agt_name=None):
        super().__init__(agt_name,show_exec=False,show_cycle=False)
        self.add(Belief("budget",(rnd.randint(6,10),rnd.randint(12,20)),adds_event=False))
        self.add(Goal("park"))
    
    @pl(gain,Goal("park"))
    def ask_price(self,src):
        # The agent driver sends a goal to the manager via the Parking channel
        self.send("Manager", achieve, Goal("sendPrice"),"Parking")

park_ch = Channel("Parking")
manager = Manager()
driver = Driver("Drv")
Admin().connect_to([manager,driver],park_ch)

The following are the different directives to send messages between agents.

self.send(<target>, <directive>, <info>, optional[<channel>])

Directives:
tell 		-> Add Belief on target
untell		-> Remove Belief from target
achieve 	-> Add Goal to target
unachieve	-> Remove Goal from target
askOne		-> Ask for Belief from target
askOneReply	-> Ask for Belief from target and wait for Reply
askAll		-> Ask for all similar Beliefs from target
askAllReply	-> Ask for all similar Beliefs from target and wait for Reply
tellHow		-> Add Plan on target
untellHow	-> Remove Plan from target
askHow 		-> Ask for Plan from target
askHowReply	-> Ask for Plan from target and wait for Reply

Environment

MASPY also gives an abstraction to model the environment.

Here's how you create a parking lot for the manager and driver from before:

Creating an environment

from maspy import *

class Park(Environment):
    def __init__(self, env_name=None):
        super().__init__(env_name)
        # This creates in the environment a percept for connected agents to perceive.
        # This specific percept does not create a event when percieved by an agent 
        self.create(Percept("spot",(1,"free"),adds_event=False))

    def park_spot(self, agt, spot_id):
        # The function get gives you percepts from the environment
        # It has various filters to make this search more precise
        spot = self.get(Percept("spot",(spot_id,"free")))
        if spot:
            # This function is used to modify the arguments of an percept.
            self.change(spot,(spot_id,driver))

            # You could also remove the old and create the new
            self.remove(spot)
            self.create(Percept("spot",(spot_id,driver)))

    def leave_spot(self, agt):
        spot = self.get(Percept("spot",("ID",driver)))
        if spot:
            self.change(spot,(spot.args[0],"free"))

Allowing agents to interact with an environment

from maspy import *

class Park(Environment):
    def __init__(self, env_name=None):
        super().__init__(env_name)
        self.create(Percept("spot",(1,"free"),adds_event=False))

    def park_spot(self, agt, spot_id):
        spot = self.get(Percept("spot",(spot_id,"free")))
        if spot:
            self.change(spot,(spot_id,driver))

class Driver(Agent)
    @pl(gain,Goal("park",("Park_Name", Any)))
    def park_on_spot(self,src,park_name,spot_id):
        # This agent functions makes the connection with an environment or channel
        # Just give it Channel(Name) or Envrionment(Name) to add it to the agent 
        self.connect_to(Environment(park_name))

        # After the connection, the agent can execute the envrionment plan directly
        self.park_spot(spot_id)

Simplest System with Every Class

from maspy import *

class SimpleEnv(Environment):
    def env_act(self, agt, agent2):
        self.print(f"Contact between {agt} and {agent2}")

class SimpleAgent(Agent):
    @pl(gain,Goal("say_hello", Any))
    def send_hello(self,src,agent):
        self.send(agent,tell,Belief("Hello"),"SimpleChannel")

    @pl(gain,Belief("Hello"))
    def recieve_hello(self,src):
        self.print(f"Hello received from {src}")
        self.env_act(src)

if __name__ == "__main__":
    Admin().set_logging(show_exec=True)
    agent1 = SimpleAgent()
    agent2 = SimpleAgent()
    env = SimpleEnv()
    ch = Channel("SimpleChannel")
    Admin().connect_to([agent1,agent2],[env,ch])
    agent1.add(Goal("say_hello",(agent2.my_name,)))
    Admin().start_system()

This code will generate the following prints:

# Admin #> Starting MASPY Program
# Admin #> Registering Agent SimpleAgent:('SimpleAgent', 1)
# Admin #> Registering Channel:default
Channel:default> Connecting agent SimpleAgent:('SimpleAgent', 1)
# Admin #> Registering Agent SimpleAgent:('SimpleAgent', 2)
Channel:default> Connecting agent SimpleAgent:('SimpleAgent', 2)
# Admin #> Registering Environment SimpleEnv:SimpleEnv
# Admin #> Registering Channel:SimpleChannel
Environment:SimpleEnv> Connecting agent SimpleAgent:('SimpleAgent', 1)
Channel:SimpleChannel> Connecting agent SimpleAgent:('SimpleAgent', 1)
Environment:SimpleEnv> Connecting agent SimpleAgent:('SimpleAgent', 2)
Channel:SimpleChannel> Connecting agent SimpleAgent:('SimpleAgent', 2)
Agent:SimpleAgent_1> Adding Goal say_hello(SimpleAgent_2)[self]
Agent:SimpleAgent_1> New Event: gain,Goal say_hello(SimpleAgent_2)[self]
# Admin #> Starting Agents
Agent:SimpleAgent_1> Running gain : Goal say_hello(typing.Any)[self], [], send_hello() )
Channel:SimpleChannel> SimpleAgent_1 sending tell:Belief Hello(())[SimpleAgent_1] to SimpleAgent_2
Agent:SimpleAgent_2> Adding Belief Hello(())[SimpleAgent_1]
Agent:SimpleAgent_2> New Event: gain,Belief Hello(())[SimpleAgent_1]
Agent:SimpleAgent_2> Running gain : Belief Hello(())[self], [], recieve_hello() )
Agent:SimpleAgent_2> Hello received from SimpleAgent_1
Environment:SimpleEnv> Contact between SimpleAgent_2 and SimpleAgent_1
# Admin #> [Closing System]
# Admin #> Still running agent(s):
SimpleAgent_1 | SimpleAgent_2 |
# Admin #> Ending MASPY Program

This program must be terminated using a ctrl+c. Otherwise the system would continue running indeterminately.

Rough edges

The project still has some rough edges that should be considered.

  • The framework API will probably have a decent amount of breaking changes in the future.
  • There is no support to run a `MASPY`` system in a distributed setting.
  • The system performance still unmeasured, altough running a toy system with over thousands of agents was possible.

Papers published

MASPY: Towards the Creation of BDI Multi-Agent Systems, WESAAC 2023

Internal Functions

___ : Represents either an object instance outside its class, or self inside its class.

Iterable : Represents any data structuture that can be iterated.

Agent

___.print_beliefs #  Print all agent's current beliefs-
___.print_goals #  Print all agent's current goals
___.print_plans #  Print all agent's current plans
___.print_events #  Print all agent's current events

"""
Prints the given arguments with the agent's name as a prefix.
Args:
    *args: The arguments to be printed.
    **kwargs: The keyword arguments to be printed.
Returns:
    None
"""
___.print(*args, **kwargs) 

"""
Connects agent to a target
Args: 
    target - Channel, Environment or str
	when target is str it searches for a file with str for its name
Returns: 
    Connected Channel, Environment or None
"""
___.connect_to(target: Environment | Channel | str)

"""
Disconnects agent from a target
Args: 
    target - Channel or Environment
Returns: 
    None
"""
___.disconnect_from(target: Channel | Environment)

"""
Adds one or more beliefs and(or) goals in agent
Args: 
    data_type - Belief, Goal, Beliefs and(or) Goals
Returns: 
    None
"""
___.add(data_type: Belief | Goal | Iterable[Belief | Goal])

"""
Removes one or more beliefs and(or) goals from agent
Args: 
    data_type - Belief, Goal, Beliefs and(or) Goals
Returns: 
    None
"""
___.rm(data_type: Belief | Goal | Iterable[Belief | Goal])

"""
Adds one or more plans in agent
Args: 
    plan - Plan or Plan List
Returns: 
    None
"""
def add_plan(plan: Plan | List[Plan]):

"""
Removes one or more plans from agent
Args: 
    plan - Plan or Plan List
Returns: 
    None
"""
def rm_plan(plan: Plan | List[Plan]):

"""
Checks if the agent has an belief, goal, plan or event
Args:
    data_type - Belief, Goal, Plan or Event
Returns:
    bool: True if has, False if not
"""
___.has(data_type: Belief | Goal | Plan | Event)

"""
Creates an test event for the given data_type
Args:
	data_type: The Belief or Goal to be tested
Returns:
	None
"""
___.test(self, data_type: Belief | Goal)

"""
Suspends the current plan for a given time or until a certain event is received.
Args:
	timeout: The time in seconds to suspend the plan. Defaults to None.
	event: The event to wait for. Defaults to None.
Returns:
	None
"""
___.wait(timeout: Optional[float] = None, event: Optional[Event] = None)

"""  
Retrieves a specific data from the agent's knowledge on the given data_type and search parameters
Args:
    data_type - Belief, Goal, Plan or Event: The type of data to retrieve.
    search_with - Belief, Goal, Plan or Event: The info to search with. Defaults to None.
    all - bool: Whether to return all matching data or just the first match. Defaults to False.
    ck_chng - bool: Whether to check the changes argument in the data. Defaults to True.
    ck_type - bool: Whether to check the type of the data. Defaults to True.
    ck_args - bool: Whether to check the arguments of the data. Defaults to True.
    ck_src - bool: Whether to check the source of the data. Defaults to True.
Returns:
    List[data] | data: The retrieved data of the specified type.
    If no matches are found, returns None.
"""  
___.get(data_type: Belief | Goal | Plan | Event,
        search_with:  Belief | Goal | Plan | Event = None,
        all = False, ck_chng=True, ck_type=True, ck_args=True, ck_src=True)
"""
ACTS = 	tell | untell |
	tellHow | untellHow |
	achieve | unachieve |
	askOne | askAll | askHow
	askOneReply | askAllReply | askHowReply

MSG = Belief | Ask | Goal | Plan | List[Belief | Ask | Goal | Plan]

Sends a message to target agent or agents, optionally through a channel
Args:
    target - str or List[str]: broadcast, target agent name or agents names to send the message to.
    act - ACTS: The directive of the message.
    msg - MSG: The message to send.
    channel - str: The channel to send the message through. Defaults to DEFAULT_CHANNEL.
Returns:
    None
"""
___.send(target: str | List[str], act: ACTS, msg: MSG, channel: str = DEFAULT_CHANNEL)

"""
Finds another agent's name also connected in an Environment or Channel
Args:
    agent_name - str or list[str]: The class agent or list containing the class name and instance name.
    cls_type - str or None: The type of class to search in. Defaults to None.	
    cls_name - str or None: The name of the class to search in. Defaults to None.
Returns:
    list[str] or None: A list of the names of the agents with the provided specifications.
"""
___.list_agents(agent_class: str | list[str],
  	    cls_type: Optional[str] = None,
  	    cls_name: Optional[str] = None)

"""
Perceives the specified environment(s) and updates the agent's beliefs.
Args:
    env_name str or list[str]: The name of the environment or a list of environment names to perceive.
        env_name can also accepts "all" to perceive all connected environments.
Returns:
    None:
"""
___.perceive(env_name: str | list[str])

"""
# Depreciated Method: Now you should directly use the environment function #
Retrieves the environment instance with the given name to make an action
Args:
    env_name - str: The name of the environment to retrieve.
Returns:
    Environment or None: The retrieved environment or None found.
"""
___.action(env_name: str)

"""
Stops and Removes all intentions and events from the intention and event list that both contains the given data_type.
Args:
	data_type: The data type to remove from the intentions.
Returns:
	None
"""
___.drop_all_desire(data_type: Belief | Goal)

"""
Stops and Removes the oldest intention and event from the intention and event list that both contains the given data_type.
Args:
	data_type: The data type to remove from the intentions.
Returns:
	None
"""
___.drop_desire(data_type: Belief | Goal)

"""
Removes all events from the event list that contains the given data_type.
Args:
	data_type: The data type to remove from the intentions.
Returns:
	None
"""
___.drop_all_event(data_type: Belief | Goal)

"""
Removes the oldest event from the event list that contains the given data_type.
Args:
	data_type: The data type to remove from the intentions.
Returns:
	None
"""
___.drop_event(data_type: Belief | Goal)

"""
Stops and Removes all intentions from the intention list that contains the given data_type.
Args:
	data_type: The data type to remove from the intentions.
Returns:
	None
"""
___.drop_all_intention(data_type: Belief | Goal)

"""
Stops and Removes the oldest intention from the intention list that contains the given data_type.
Args:
	data_type: The data type to remove from the intentions.
Returns:
	None
"""
___.drop_intention(data_type: Belief | Goal)

"""
Start the agent reasoning cycle manually.
Args:
    None
Returns:
    None
"""
___.start()

"""
Stop the cycle of the agent.
    
This method stops the cycle of the agent by setting the `stop_flag` event to True,
indicating that the cycle should be stopped. It also sets the `paused_agent` flag
to True, indicating that the agent has been paused.Finally, it sets the `running`
flag to False, indicating that the agent is no longer running.

Args:
    None
Returns:
    None
"""
___.stop_cycle()

Environment

___.print_percepts # prints all environment's current percepts

"""
Prints the given arguments with the environment's name as a prefix.
Args:
    *args: The arguments to be printed.
    **kwargs: The keyword arguments to be printed.
Returns:
    None
"""
___.print(*args, **kwargs ) 

"""
Creates in environment one or multiple percepts
Args:
    percept - list[Percept] | Percept: The one or multiple Percept to be created.
Returns:
    None
"""
___.create(percept: list[Percept] | Percept)

"""
Retrieves from environment one or multiple percepts that match the given percept
Args:
    percept - Percept: The percept to search for.
    all - bool: If True, returns all matching percepts. Defaults to False.
    ck_group - bool: Whether to check the group of the percept. Defaults to False.
    ck_args - bool: Whether to check the arguments of the percept. Defaults to True.
Returns:
    List[Percept] or Percept: The retrieved percept(s).
    If no matches are found, returns None.
"""
___.get(percept: Percept, all: bool=False, ck_group: bool=False, ck_args: bool=True)

"""
Changes the arguments of a percept
Args:
    old_percept - Percept: The percept to be changed.
    new_args - Percept.args: The new arguments for the percept.
Returns:
    None
"""
___.change(old_percept: Percept, new_args: Percept.args)

"""
Deletes from environment one or multiple percepts
Args:
    percept - Iterable[Percept] | Percept: The one or multiple Percept to be deleted.
Returns:
    None
"""
___.delete(percept: list[Percept] | Percept)

Admin

"""
Starts the reasoning cycle of all created agents
Args:
    None
Returns:
    None
"""
Admin().start_system()

"""
Starts the reasoning cycle of one of multiple agents
Args:
    agents: list[Agent] or Agent: The agent(s) to start their reasoning cycle
Returns:
    None
"""
Admin().start_agents(agents: Union[list[Agent], Agent])

"""
Connects any number agents to any number of Channels and Environments
Args:
    agents: Iterable[Agent] or Agent: The agent(s) to connect
    targets: Iterable[Environment or Channel], Environment or Channel: The target(s) to connect to
Returns:
    None
"""
Admin().connect_to(agents: Iterable | Agent,
		   targets: Iterable[Environment | Channel] | Environment | Channel)

"""
Sets the logging configuration for the whole system
Args:
    show_exec: bool: Whether to show execution logs
    show_cycle: bool: Whether to show reasoning cycle logs
    show_prct: bool: Whether to show perception logs
    show_slct: bool: Whether to show selection of plans logs
    set_<class>: bool: Whether to affect the class to True | False
Returns:
    None
""" 
Admin().set_logging(show_exec: bool, show_cycle: bool=False,
                show_prct: bool=False, show_slct: bool=False, 
                set_admin=True, set_agents=True,
                set_channels=True, set_environments=True)
					
"""
Slows all agents reasoning cycles by x seconds
Args:
    time: int | float: The time to sleep in seconds
Returns:
    None
"""
Admin().slow_cycle_by(time: int | float):

About

A Python Framework for Multi-Agent Programming with Machine Learning

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages