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

Game of Sticks Normal and Hard Mode #3

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
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
316 changes: 3 additions & 313 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,315 +1,5 @@
# Game of Sticks

## Description

Create a version of the Game of Sticks where human and AI players can play
against each other.

In the Game of Sticks there is a heap of sticks on a board. On their turn,
each player picks up 1 to 3 sticks. The one who has to pick the final stick
will be the loser.

The following is an example of the game of sticks.

* The game starts with 20 sticks on the board.
* Marvin takes 3 sticks, there are 17 sticks remaining.
* Hal takes 2 sticks, there are 15 sticks remaining.
* Marvin takes 1 stick, there are 14 sticks remaining.
* Hal takes 3 sticks, there are 11 sticks remaining.
* Marvin takes 2 sticks, there are 9 sticks remaining.
* Hal takes 2 sticks, there are 7 sticks remaining.
* Marvin takes 3 sticks, there are 4 sticks remaining.
* Hal takes 1 stick, there are 3 sticks remaining.
* Marvin takes 2 sticks, there is 1 stick remaining.
* Hal has to take the final stick and loses.

This assignment is split into four parts:

1. Implementing the game as a two-player game.
2. Adding an AI that can be played against.
3. Adding an option for training the AI against another AI.
4. Performing mathematical analysis of the problem based on information
gathered from the AI.

## Objectives

### Learning Objectives

After completing this assignment, you should understand:

* Lists and tuples

### Performance Objectives

After completing this assignment, you should be able to:

* Use functions effectively
* Build a simple AI

## Details

### Deliverables

* A Git repo called game-of-sticks containing at least:
* `README.md` file explaining how to run your project
* a `requirements.txt` file
* a suite of tests for your project

### Requirements

* Passing unit tests
* No PEP8 or Pyflakes warnings or errors

## Normal Mode

### Player vs Player

Create a game where two players can play against each other. The two examples
below demonstrate how the game should behave.

Example 1:

```
Welcome to the Game of Sticks!
How many sticks are there on the table initially (10-100)? 10

There are 10 sticks on the board.
Player 1: How many sticks do you take (1-3)? 3

There are 7 sticks on the board.
Player 2: How many sticks do you take (1-3)? 3

There are 4 sticks on the board.
Player 1: How many sticks do you take (1-3)? 3

There is 1 stick on the board.
Player 2: How many sticks do you take (1-3)? 1
Player 2, you lose.
```

Example 2:

```
Welcome to the Game of Sticks!
How many sticks are there on the table initially (10-100)? 500
Please enter a number between 10 and 100.
How many sticks are there on the table initially (10-100)? 3
Please enter a number between 10 and 100.
How many sticks are there on the table initially (10-100)? 50

There are 50 sticks on the board.
Player 1: How many sticks do you take (1-3)? 3

There are 47 sticks on the board.
Player 2: How many sticks do you take (1-3)? 55
Please enter a number between 1 and 3
Player 2: How many sticks do you take (1-3)? 3

There are 44 sticks on the board.
Player 1: How many sticks do you take (1-3)? 3
...

There is 1 stick on the board.
Player 1: How many sticks do you take (1-3)? 1
Player 1, you lose.
```

### Player vs AI

Let's create an artificial intelligence player for the Game of Sticks. Instead
of creating an AI based off knowledge of the optimal strategy, we'll create
an AI that can learn from games it wins and loses, and then we can look at it
to figure out the best strategy.

Consider the functionality of the AI using the following description:

* An AI has a number of hats, one hat for each possible amount of sticks on the
table. Initially, each hat contains three balls that are numbered from 1 to 3.

* At every step of the game that the AI plays, the AI takes a random ball out of
the hat that matches the amount of sticks currently on the board. When the AI
takes a ball out of a hat, it places it next to the hat for waiting, reads the
number on the ball, and takes the amount of sticks that the ball indicates.

* If the AI wins the game, it puts two balls of the type to each hat that has a
ball next to it. Both balls have the same number. If the AI loses, it will
throw away the balls that are next to the hats (note: A hat must always have at
least one ball of each number, hence the last ball of a specific number cannot
be thrown away and must be put back to the hat).

* As more and more games are played, there will be more balls that indicate a
good number of sticks to take. This means that as balls are taken at random, it
becomes more likely that the AI is able to play well.

**Example**:

Let us consider an example where there are 10 sticks at the beginning. The hat
contents for the AI are as follows:

| | | | | | | | | |
------- | ----- | ----- | ----- | ----- | ----- | ----- | ----- | ----- | ----- | -----
hat | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10
content | 1,2,3 | 1,2,3 | 1,2,3 | 1,2,3 | 1,2,3 | 1,2,3 | 1,2,3 | 1,2,3 | 1,2,3 | 1,2,3

The game may proceed as follows:

1. Player takes 3 sticks, there are 7 sticks remaining.
2. AI randomly picks up ball 2 from the hat 7. This means that the AI takes 2 sticks, and there are 5 sticks remaining.
3. Player takes 1 stick, there are 4 sticks remaining.
4. AI randomly picks up ball 3 from hat 4. This means that AI takes 3 sticks, and there is 1 stick remaining.
5. Player has to take the final stick and loses.

Now, the situation with the AI is as follows:

| | | | | | | | | |
------- | ----- | ----- | ----- | --- | ----- | ----- | --- | ----- | ----- | -----
hat | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10
content | 1,2,3 | 1,2,3 | 1,2,3 | 1,2 | 1,2,3 | 1,2,3 | 1,3 | 1,2,3 | 1,2,3 | 1,2,3
beside | | | | 3 | | | 2 | | |

As the AI wins the game, it will put the balls that are next to the hats back to the hats with extra balls. The situation is now as follows:

| | | | | | | | | |
------- | ----- | ----- | ----- | ------- | ----- | ----- | ------- | ----- | ----- | -----
hat | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10
content | 1,2,3 | 1,2,3 | 1,2,3 | 1,2,3,3 | 1,2,3 | 1,2,3 | 1,2,2,3 | 1,2,3 | 1,2,3 | 1,2,3

Now the AI will more likely take 3 sticks in the case of four sticks remaining on the board, and 2 sticks in case there are 7 sticks remaining on the board.

Your task is to modify the human vs. human version of the game so that the player can choose to play against an AI that works as described above. After each game, the AI will update the contents of its hats. The AI will play relatively randomly at first, but you will notice that it will start to learn a strategy as you play against it.

The following example displays how the program should behave after you have finished this step.

```
Welcome to the Game of Sticks!
How many sticks are there on the table initially (10-100)? 10
Options:
Play against a friend (1)
Play against the computer (2)
Which option do you take (1-2)? 2

There are 10 sticks on the board.
Player 1: How many sticks do you take (1-3)? 3

There are 7 sticks on the board.
AI selects 2.

There are 5 sticks on the board.
Player 1: How many sticks do you take (1-3)? 3

There are 2 sticks on the board.
AI selects 2.
AI loses.
Play again (y/n)? y

There are 10 sticks on the board.
Player 1: How many sticks do you take (1-3)? 1

There are 9 sticks on the board.
AI selects 1.

There are 8 sticks on the board.
Player 1: How many sticks do you take (1-3)? 3

There are 5 sticks on the board.
AI selects 3.

There are 2 sticks on the board.
Player 1: How many sticks do you take (1-3)? 2
You lose.
Play again (y/n)? y

There are 10 sticks on the board.
Player 1: How many sticks do you take (1-3)? 3

There are 7 sticks on the board.
AI selects 2.

There are 5 sticks on the board.
Player 1: How many sticks do you take (1-3)? 3

There are 2 sticks on the board.
AI selects 2.
AI loses.
Play again (y/n)? n
```

## Hard Mode

In addition to the requirements from **Normal Mode**:

### AI vs AI

In the previous part we created an AI that is able to learn from playing
against the player. As we play against it, we notice that it takes a
considerable amount of time before the AI is able to perform against a human
player. In this assignment, you need to modify the program so that the player
can choose to play either against a naive AI or a pre-trained AI.

In order to pre-train an AI, you need to create a program that allows two AIs
to battle against each others -- say a hundred thousand times (after the
training is working, try out different numbers as well!) -- and after that the
player will be set to play against the AI that is ready to battle the player.

The following example shows how the game would work with the trained AI option.

```
Welcome to the Game of Sticks!
How many sticks are there on the table initially (10-100)? 10
Options:
Play against a friend (1)
Play against the computer (2)
Play against the trained computer (3)
Which option do you take (1-3)? 3
Training AI, please wait...

There are 10 sticks on the board.
Player 1: How many sticks do you take (1-3)? 3

There are 7 sticks on the board.
AI selects 2.

There are 5 sticks on the board.
Player 1: How many sticks do you take (1-3)? 1

There are 4 sticks on the board.
AI selects 3.

There is 1 stick on the board.
Player 1: How many sticks do you take (1-3)? 1
You lose.
Play again (y/n)? y

There are 10 sticks on the board.
Player 1: How many sticks do you take (1-3)? 2

There are 8 sticks on the board.
AI selects 3.

There are 5 sticks on the board.
Player 1: How many sticks do you take (1-3)? 2

There are 3 sticks on the board.
AI selects 2.

There is 1 stick on the board.
Player 1: How many sticks do you take (1-3)? 1
You lose.
Play again (y/n)? n
```

### Mathematical analysis

The AI can play very well -- but what is its strategy?

You can find the strategy by looking at the contents of the hats that the AI is
using. What is the strategy and can you show mathematically why it works?

## Nightmare Mode

Use the same strategy to create a game of Tic-Tac-Toe with a trained AI player.
[This paper](http://aitopics.org/sites/default/files/classic/Machine_Intelligence_2/MI2-Ch9-MichieChambers.pdf) may help.

## Credit

This assignment is taken from [Stanford's Nifty Assignments 2014](http://nifty.stanford.edu/2014/laaksonen-vihavainen-game-of-sticks/).
Start the game from the begin_sticks_game.py file.
It will then ask if you'd like to play a 2-player game, 1-player against AI, or 1-player against a trained AI.
Once you've made your choice it will begin the corresponding game and you are able to play as many times as you'd like.
78 changes: 78 additions & 0 deletions begin_sticks_game.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
from sticks_two_player import TwoPlayerGame
from sticks_player_vs_AI import OnePlayerGame, ArtIn

def do_game(ai_1, ai_2):
'''plays through one single 15-stick game'''
number_of_sticks = 15
winner = None
while number_of_sticks > 1:
ai_choice = ai_1.pick_some_sticks(number_of_sticks)
number_of_sticks = number_of_sticks - ai_choice
if number_of_sticks <= 1:
if number_of_sticks == 1:
winner = ai_1
else:
winner = ai_2
break

ai_choice = ai_2.pick_some_sticks(number_of_sticks)
number_of_sticks = number_of_sticks - ai_choice
if number_of_sticks <= 1:
if number_of_sticks == 1:
winner = ai_2
else:
winner = ai_1
break

return winner, winner.game_choices

def create_AI(ai_1, ai_2):
'''Trains AI by pitting two ai's against each other 1000 times. Begins a game with
the winningest AI'''
print("Training AI...")
games = 1000
ai_1_choices = {}
ai_2_choices = {}
ai_1_count = 0
ai_2_count = 0
good_choices = {}
while games != 0:
winner, good_choices = do_game(ai_1, ai_2)

if winner == ai_1:
ai_1_count += 1
for pair in good_choices.items():
if pair[0] in ai_1_choices.keys():
ai_1_choices[pair[0]].extend(pair[1])
else:
ai_1_choices[pair[0]] = pair[1]
else:
ai_2_count += 1
for pair in good_choices.items():
if pair[0] in ai_2_choices.keys():
ai_2_choices[pair[0]].extend(pair[1])
else:
ai_2_choices[pair[0]] = pair[1]

games -= 1

if ai_1_count > ai_2_count:
game = OnePlayerGame(ai_1_choices)
game.start()
else:
game = OnePlayerGame(ai_2_choices)
game.start()


if __name__ == '__main__':
usr_choice = input("Welcome to the Game of Sticks, would you like to play against a (H)uman, (A)I, or a (T)rained AI? ").lower()
if usr_choice == "h":
game = TwoPlayerGame()
game.start()
elif usr_choice == "a":
game = OnePlayerGame()
game.start()
else:
ai_1 = ArtIn(15)
ai_2 = ArtIn(15)
create_AI(ai_1, ai_2)
Loading