COMP343 - Search Based Agents
Github Classroom Assignment Link
Missionaries and Cannibals Solver
For this assignment you’ll implement and compare two search agents for the missionaries and cannibals problem: a breadth first search agent and an iterative-deepening search agent. Below you’ll find a brief look at the code provided to you followed by some discussion of what you need to complete. We’ll spend some class time discussing this assignment as well.
Starter Code
The following has been provided to get you started:
- The file
frontiers.py
provides aStack
andQueue
class. All operations operate in constant time. You do not need to modify any code in this file. - The file
Node.py
provides a partial definition for search treeNode
objects. The__init__
,__str__
,__eq__
, and__ne__
methods are all complete. You’ve also been given a method namedgetPathStateAction
which can be used to get the complete plan developed by your agent. This method will not work properly until you complete the stubbed out methods in the class (more on that below). - The file
problem.py
contains a partial definition for the classMCState
which encapsulates the Missionaries and Cannibals problem logic. You’ve been given the methods__init__
,__str__
,__repr__
,__eq__
,__ne__
, and__hash__
. The remaining methods are stubbed out; you’ll be completing them (more on that below). - The file
agent.py
has stubs for the two agent programs:solve_bfs
andsolve_ids
. It also has a function for reporting on a solution as returned by the agent functions. Finally, a main conditional block has been written that will, when all the code is complete, run both agent programs and report on the results.
You should not modify any of the completed methods or functions. You’re welcome too modify and tweak the main program; you’ll almost certainly want to do so for incremental testing purposes. Just ensure that what is there to start will run as expected when all the code is complete. Main-style conditionals are also in Node.py
and problem.py
if you want to do unit-test style testing on the code in those files.
What You Need To Complete
Your tasks are presented here in a bottom-up fashion. That means, if you complete them in the order listed below, then you’ll avoid any dependency issues between functions. You are not required to implement things in this order, and you are welcome to create extra methods/functions as auxiliary functions/methods for these functions. However, you must complete these methods as specified. No big changes to the overall design of the program. The docstring for each method/function provides more details about their specifications and class time will be dedicated to discussing each method as well.
- In
problem.py
isGoal
A method to check if the state is the/a goal state.isValid
A method that determines if the state is a valid puzzle state.getNext
Given an action, determine the state that follows by applying the action in the state in question. For example,st.getNext((1,1))
returns the state that follows from moving one missionary and one cannibal when in statest
.getAllNext
A Generator that yields (action,state) tuples for all actions that lead to valid stats from the state in question. For example,st.getAllNext()
will generate all the actions one can take fromst
along with the state reached by taking that action. (This method should make use of the previous two methods:getNext
andisValid
.)
- In
Node.py
checkCycle
checks to see if the state associated with the node in question already exists along the node’s parent path. For example,nd.checkCycle()
returns true if any ofnd
’s ancestors has the same state asnd
.getPathActions
traverses the path starting at the node in question and collects all the actions along that path. For example,nd.getPathActions()
returns a list of actions where the first is the action associated withnd
followed by its parent node action, and so on.getPathStates
likegetPathActions
but for puzzle states rather than actions.
- In
agent.py
solve_bfs
uses a breadth-first search to solve the missionaries and cannibals puzzle while also counting the number of nodes checked/expanded as well as the maximum size reached by the frontier/queue. It returns the node that captures the solution along with the two previously mentioned performance metrics.solve_ids
uses iterative-deepening search and returns the same performance metrics assolve_bfs
. Note here that we want the total nodes checked across all of the depth-limited searches as well as the max frontier/stack size reached by any depth-limited search.
Once again, you can create any additional methods or functions you deem necessary, but your finished program must contain implementations for each of the above methods and functions and the code at the bottom of agent.py
in the starter file should run as is.