Assignment 2: State-Space Search
CSE 415: Introduction to Artificial Intelligence
The University of Washington, Seattle, Winter 2025
Part A is Due Wednesday, January 22, at 23:59 PM. Parts B and C are due Monday, January 27. You may do this assignment either individually or in a partnership of two. If you work in a partnership, you'll be turning in one set of files for each of the three parts, rather than two sets of files for each part.
 
Introduction

This assignment covers state-space search in single-agent, deterministic environments. Later we will consider multi-agent search in environments with stochastic dynamics. We start by introducing a framework for the classical theory of problem solving. Then we examine heuristic search. This assignment is quite substantial, and that's one reason it is divided into two parts.

After doing the coding and executions involved in Part A and Part B, you'll put it into a document in Part C. For this part, you'll submit one report file called SearchReport.pdf. Half of the report will be about your work in Part A, and the other half will be about Part B.

Part A (Blind Search)

This part is about basic problem formulation and basic search algorithms.

1. First, download the starter code for this assignment. Unzip it with a utility such as Windows Compressed Folders, etc.

2. Examine each of the two problem formulation files provided. One is the Humans, Robots and Ferry problem. The other is the Towers of Hanoi problem. An additional file is provided, Farmer_Fox.py, and you will be writing one new problem formulation in that file.

In the given formulation files, pay particular attention to how the State class is defined and how the operators are established. We will be talking about this code in class.

3. An interactive solving client is provided. This is implemented in the file Int_Solv_Client.py. Optional but recommended: Try running the client with Towers of Hanoi by using the following command in a Linux, Darwin, or Cygwin command shell. To do this, issue the command:
python Int_Solv_Client.py TowersOfHanoi 3
If you are not comfortable working in Linux, Darwin, or Cygwin, you may wish to simply create a working folder with the Interactive Solving Client in it, and all the problem formulation files in it, and edit the importing code of the Interactive Solving Client so that it simply imports the problem formulation of your choice, and then run the program from IDLE, PyCharm, or other IDE you might be using.
4. Gain familiarity with how the HumansRobotsFerry.py formulation works.

(a) Run the interactive solving client and by trying different operators solve the problem of getting all the humans and robots across the river. (b) Examine the formulation code, and focus most of your attention on the can_move method, which implements the preconditions for all the operators of this formulation. (c) In this method, there are five lines of code containing "return False". Explain what is going on in the 4th one of these lines. Put your answer into your Part C Report file ("SearchReport.pdf") under the heading "Part A Step 4 (c)." To create your SearchReport.pdf, you may rename the starter-code file, which has the name "SearchReport_template.tex" to "SearchReport.tex" and then edit that in Overleaf. If you are not inclined to use Overleaf, you may create a comparable file in Word or Google Docs and then convert it to pdf.

5. Create your own problem formulation for the "Farmer, Fox, Chicken, and Grain" problem covered in class. Your code should follow the same structure used in HumansRobotsFerry.py. For your operators, implement the set of operators given in the solution to the worksheet (posted in ED) on the Farmer, Fox, Chicken and Grain problem. Define the operators in the same order that they appear in the presentation of the set of operators in the given solution. That will help ensure that solutions found by algorithms will match the expected solutions. Test your formulation using the Interactive solving client and debug your formulation as needed.
6. Then produce a session transcript for your formulation based on an automated solver. This transcript will be in the form of a log file. You can produce it by running the starter-code program ItrDFS.py and capturing the output and directing it to a file. The file should be named FFCG-log.txt.
7. Using parts of the starter-code file ItrDFS.py, which implements a loop-based depth-first search method ("DFS" in the following), implement BFS (Breadth-First Search) as it is specified in the lecture slides. Put your implementation of BFS into the starter-template file ItrBFS.py. Make sure that these implementations keep track of "predecessor" links (also known as back links), and they can report a shortest path from start to goal, for whatever problem they are applied to. (You can implement these links using a hash table, i.e., dictionary, that maps each state other than the initial state to its parent state. The initial state should be mapped to a special value such as None, or -1, which is up to you.)
8. Compare DFS and BFS on the following problems: (i) Humans, Robots and Ferry, (ii) Farmer, Fox, Chicken, and Grain, and (iii) 4-Disk Towers of Hanoi. For each combination of algorithm and problem, report the following: (a) the path found from start to goal, (b) the length of the path, (c) the number of nodes expanded (i.e., removed from OPEN and had successors generated even if there were 0 successors). Report this information by entering it into your report document using the table provided in the its template file. Finally, just for the Towers-of-Hanoi problem with 4 disks, explain (i) why the maximum length of the OPEN list is more for one algorithm than the other, and (ii) why the solution PATH length is different for one algorithm from that of the other. Include the table in your Part C report, under the section heading "Part A, Step 8".
Part B (Informed Search)

9. The starter-code file UCS.py, provides an implemention of the Uniform-Cost Search algorithm. This file also provides you with an implementation of a special kind of priority queue which supports not only the standard insert and delete_min operations, but also a method called contains, which returns True if a given element is in the priority queue. It also allows accessing and updating the priority value of any element in the priority queue. See the documentation for more information. This priority queue is not just adequate for implementing Uniform-Cost Search, but is particularly helpful in implementing the A* Search algorithm.
The main difference between Breadth-First Search and Uniform-Cost Search (UCS) is that UCS works with problem formulations that have different costs associated with different moves. In terms of graph search, there are edge costs, and they are not necessarily assumed to be all equal to 1 as they essentially are in Breadth-First Search.

The starter code includes a new problem formulation for you to use with UCS; it's the French Cities network in which the problem is to find a shortest route from the initial-state city to the goal-state city. These are, by default, RENNES and AVIGNON. (These could easily be changed by editing the formulation file's line defining STARTING_CITY, etc.) This formulation is in the file FranceWithCosts.py.
The State class in this formulation file has a method edge_distance(s2), which is used to get the edge cost from the current state to s2. You'll be setting up the UCS.py program to use these cost values to determine the cost of the lowest-cost path from the starting state to any newly reached state.
When the UCS algorithm reaches a goal state, it produces a backtrace of "backlinks" (sometimes called predecessor links) to represent the actual lowest-cost path from the start state to the goal state found. The file UCS.py includes a method runUCS.

For the default initial state and goal state, the results are

['Rennes', 'Nantes', 'Limoges', 'Lyon', 'Avignon']
for the self.PATH variable and
1041.0
for the self.TOTAL_COST variable.

Page 184 in the reading shows the state space for this problem, with the distances on the graph edges. When you run the code you should get a solution path of 4 edges (5 cities) and total distance 1041: [’Rennes’, ’Nantes’, ’Limoges’, ’Lyon’, ’Avignon’]. It might be formatted differently in the printout. The map is also shown below:

Map showing French cities (which serve as the states of this problem) and distance values on roads (represented as graph edges) between them.

Implement an A* search program, putting your code into the starting template file AStar.py. You may wish to cut and paste from the UCS.py file, since most of the implementation will be identical.

Develop your A* program to work with the heuristic function defined in the file FranceWithDXHeuristic.py. Your program will be operated as follows:

python AStar.py FranceWithDXHeuristic
When it is working, it should find the same path that UCS does, but it should have only expanded 12 states instead of 15.

Here are some hints about how to modify UCS.py to implement A*. Change the names of the file and the algorithm in the code and the comments. After the code that imports the Problem, assign self.h = Problem.h, so that you can call the heuristic function easily in your algorithm. Within the search algorithm, any time UCS uses gs or new_g as a priority value, you should compute f (or call it fs or new_f, depending on the context), and use the f value as the priority.

When checking the CLOSED list, don't necessarily delete the new state if the same city is already on CLOSED, but look at the priority, and if the new state has a lowever priority value, delete the state on CLOSED and put the new one on OPEN.

Make sure that the first five CAPITALIZED variables in class AStar get assigned the values they need (as described in the corresponding comments), so that you can receive credit on autograder tests. These include self.COUNT through self.TOTAL_COST. The solution path should be set as the value of self.PATH, and it must be a list of strings, such as ['Rennes', 'Nantes', 'Limoges', 'Lyon', 'Avignon']. Also, be sure to assign proper values in the dictionary for self.g.

10. The Eight Puzzle is a simplified version of the popular Fifteen Puzzle. A photo of one commercial model of the Fifteen Puzzle is given below on the left, and a picture of the Eight Puzzle is on the right. (For image credits, see the acknowledgments at the bottom of this page.)

       

Try to solve some instances of the Eight Puzzle using UCS.py. There is a default instance you should try first. You can do it with the command line:

python UCS.py EightPuzzle
There is an easy way to try more instances. Simply provide a string on the command line that represents the initial state in list form.
python UCS.py EightPuzzle '[[3, 1, 2], [4, 5, 0], [6, 7, 8]]'
Another way is to create a file that first imports EightPuzzle's definitions and then overwrites the definition of CREATE_INITIAL_STATE. This is illustrated in the file puzzle_rotate_2.py. To run UCS.py on such a puzzle, type this:
python UCS.py puzzle_rotate_2
This instance of the Eight Puzzle is named this way, because the solution requires moving all the outside pieces "around" the middle square a distance of two steps each. The lowest-cost path for this instance should have total cost 14.0, since there are seven tiles that move, and they each move twice. Note, if you prefer to do your testing by creating extra files like puzzle_rotate_2, DO NOT turn those files in with your solutions. Just report the results you get as part of your Part C report file under "Part B: UCS 8-puzzle tests".

Example puzzles to try:

# Puzzle A. (should be very fast)
python UCS.py EightPuzzle '[[3,0,1],[6,4,2],[7,8,5]]'

# Puzzle B. (should not take long)
python UCS.py EightPuzzle '[[3,1,2],[6,8,7],[5,4,0]]'

# Puzzle C. (May take a few minutes)
python UCS.py EightPuzzle '[[4,5,0],[1,2,8],[3,7,6]]'

# Puzzle D. (May take several minutes)
python UCS.py EightPuzzle '[[0,8,2],[1,7,4],[3,6,5]]'
11. Implement the following two pre-defined (i.e., standard) heuristics for the Eight Puzzle. (a) Hamming Distance (count the number of tiles out of place, but not the blank, in order to maintain admissibility), and (b) Total of Manhattan distances for the 8 tiles. Each of these should be in a separate file. If you look at the starter code file FranceWithDXHeuristic.py, it shows you how your file should import the basic problem formulation module (EightPuzzle) and then define the heuristic function h. When you are ready to test your Hamming distance heuristic, for example, you will type something like this to test it out.
# Test with Puzzle A:
python AStar.py EightPuzzleWithHamming '[[3,0,1],[6,4,2],[7,8,5]]'
If your A* algorithm and heuristics are working correctly, then you should see some rather noticeable improvements in the search speed and statistics.
Part C (Written Report)

12. Comparing Heuristics and Writing the Report: After testing these heuristics on the four previously given instances of the Eight Puzzle, record the results in your report. Also, compare them with "no heuristic" (straight Uniform Cost Search). We recommend that you write the report by starting with the latex template file SearchReport_template.tex that comes with the starter code. If you prefer to use Word, please use the same report style and structure. You can see the "empty" PDF here.

In your Part C report, you'll have, in the next section about Part B, a section of your report called "Heuristics for the Eight Puzzle". Compare the performance of these heuristics on each of the example puzzles (given above). For each puzzle-heuristic pair, report (a) whether the puzzle was successfully solved, (b) length of the solution path found, in number of edges, (c) total cost of the path found, (d) number of states expanded, and (d) MAX_OPEN_LENGTH. Put this information into the table (given with the second table template in the same file as the table template described in Part A), described below.

The table should have a row for each puzzle instance and heuristic (e.g., "(Puzzle A, Hamming)", and columns for the following: puzzle instance permutation (e.g., [3,0,1,6,4,2,7,8,5]; this flattened list is OK here), success (yes/no), count of expanded nodes, aborted (yes/no). If it takes more than 5 minutes to solve a particular problem with a particular heuristic then note that; you may abort such runs.

13. (Optional, for 5 points of extra credit). Find or create one more Eight-Puzzle heuristic that you believe may be interesting in terms of either how informed it is, how easy it is to compute, or how it relates to other known heuristics. (It must not be so trivial that it has a constant or mostly constant value! It should either be admissible or have some compensating quality that you can describe.) Implement and apply your heuristic. Compare the results from this with the results you got for the other heuristics and describe them in the last section of your report.
Keep in Mind

Here are some ideas to keep in mind for this assignment. The problem format we are using not only permits the "manual solving" that you do with the Interactive Solving Client. It also permits automatic solving, such as with DFS, and BFS. Thus it is important that you follow the problem structure correctly. Certain methods are required in the State class, including __init__, __eq__, __hash__, __str__, and copy. The operators must have the correct structure, as well. The nice thing about the Interactive Solving Client is that it provides a pretty good means to test whether your formulation will be OK.

Even if you are familiar with Python, there may be some constructs in the formulation files that you are not familiar with. You can read up on list comprehensions (p.31 of the first reading) and lambda expressions (pp. 47-52 of that reading).

Parts of your code will be graded by an autograder. To do this, your files will be imported as modules by the autograder. Your search should not automatically start running when the module is imported. The starter template file AStar.py comes with some code at the end of the file that takes care of this. This code is: "if __name__ == '__main__' " which avoids running the search automatically if it is being imported, but still runs it automatically if it is the main program. The autograder will be comparing the solutions and statistics found by your code with expected solutions, by calling functions in your code and examining the values of certain variables in your code. These variables are generally already in place for you in the starter code UCS.py. So you should not rename variables like self.PATH or any of the other variables written with all CAPITAL letters.

Commenting your Code

Each of your Python files should begin with a multiline string (more details are shown below.)

Follow reasonable commenting practice in your code. The comments in the starter code can serve as examples of the commenting style that we consider appropriate.

What to Turn In

For Part A, turn in the following files to GradeScope:

Farmer_Fox.py
FFCG-log.txt
ItrBFS.py
Each of the Python files should start with a multiline comment in the following format with your own information rather than the sample information given here.
'''Farmer_Fox.py
by John Nathan Smith and Emily Doe
UWNetIDs: jnsmith25, emilydoe24
Student numbers: 2576543, 2483119

Assignment 2, Part A, in CSE 415, Winter 2025
 
This file contains my (or our) problem formulation for the problem of
the Farmer, Fox, Chicken, and Grain.
'''

For Part B, turn in the following files to Gradescope:

AStar.py
EightPuzzleWithHamming.py
EightPuzzleWithManhattan.py

For Part C, turn in your file "SearchReport.pdf". This will start with some specific information shown below. Then it will contain your results mentioned above, for each of Part A and Part B. At the end, it will contain your "partnership retrospective."

Here, the partnership retrospective starts with whether or not you worked in a partnership. Then, if so, it will contain (a) the names of the people in your team, (b) how you divided up the work, and (c) what if anything in this collaboration was a new experience for either partner.

The turn-in method is via file uploads to GradeScope. Be sure to upload your Part A files to the A2-Part-A assignment, and upload your Part B files to the A2-Part-B assignment. Finally, upload your SearchReport.pdf file to the A2-Part-C assignment.

Updates and Corrections
 

If needed, updates and corrections will be posted here, and/or in ED.