Problem Set 4: Memory Scramble
- Iteration 1 due
- Monday, November 20, 2017, 10:00 pm
- Code reviews due
- Monday, November 27, 2017, 10:00 pm
- Iteration 2 due
- Thursday, November 30, 2017, 10:00 pm
The purpose of this problem set is to explore multithreaded programming with a shared mutable data type, which you will protect using different thread safety strategies; and to practice writing a client/server system.
Design Freedom and Restrictions
On this problem set, you have substantial design freedom.
Pay attention to the PS4 instructions in the provided code.
You must satisfy the provided specifications for certain methods in Board
, TextServer
, and WebServer
, and you should not change the code in ServerMain
.
You may rewrite or delete any of the other provided code, add new classes and methods, etc.
Take care that the coordinate system of your game board matches the specifications: (column, row) coordinates start at (1, 1) in the top left corner, increasing horizontally to the right and vertically downwards.
Your code will be tested against the text and HTTP protocols specified below. It is your responsibility to examine Didit feedback and make sure your code compiles and runs for grading, but you must do your own testing to ensure correctness.
Get the code
To get started,
- Ask Didit to create a remote
psets/ps4
repository for you on Athena. - Pull the repo from Athena using Git:
git clone ssh://[username]@athena.dialup.mit.edu/mit/6.031/git/fa17/psets/ps4/[username].git ps4
If you need a refresher on how to create, clone, and import your repository, see Problem Set 0.
Overview
On this problem set, you will build a networked multiplayer version of Memory, a.k.a. Concentration, the game in which you turn over face-down cards and try to find matching pairs. Our version will have players turning over cards simultaneously, rather than taking turns. We’ll call it Memory Scramble.
Problem 1: you will design and implement one or more ADTs for this game.
Problem 2: you will implement a server that communicates using a text protocol over network sockets.
Problems 3 & 4: you will design and implement watching for changes to the board, and implement a HTTP server. When you’re done, you can play Memory Scramble in your browser.
Iterative Development Recommendation
First, do problems 1 & 2 without considering concurrency: don’t make your Board
or other ADTs threadsafe, and don’t handle multiple concurrent connections in your text server.
Implement the gameplay rules specified below by writing tests, choosing data structures, and writing code.
Consider the abstraction functions and rep invariants of all your data types.
At this point, you should pass Didit’s public TextProtocolTest
, which uses only a single client.
Both automatic and manual grading for iter1 will focus on this part of the problem set.
Then, revise your work to make your system safe for concurrency and handle multiple concurrent clients in your text server. Revise your choice of ADT operations based on the need for atomic actions, revise your implementation using threadsafe data structures and a technique for blocking calls, and write your thread safety arguments.
Finally, go on to complete problems 3 & 4.
Game board · Gameplay rules · Example game transcript · Playing over Telnet · Playing on the web
Game board
The Memory Scramble game board has a grid of spaces. Each space starts with a card. As cards are matched and removed, spaces become empty.
A card in our game is a non-empty case-sensitive string of non-whitespace non-newline characters.
This allows “pictures” like Hello
and pictures like 🌈 by using emoji characters.
For example:
👁 | 💖 | M | √-1 | ☕️ |
All cards start face down. Players will turn them face up, and the cards will either turn face down again (if the player turned over cards that don’t match) or be removed from the board (if they do match).
A game may start with either…
A randomly-generated board, where the user specifies dimensions and a set of cards, and those cards are distributed randomly and as evenly as possible. For example:
A random 3×3 board with { 🌈, 🦄 }
| → |
|
Or, a board loaded from a file. Here is the formal grammar for board files:
BOARD_FILE ::= COLUMN NEWLINE ROW NEWLINE (CARD NEWLINE)+
CARD ::= [^\s\n\r]+
COLUMN ::= INT
ROW ::= INT
INT ::= [0-9]+
NEWLINE ::= "\n" | "\r" "\n"?
In BOARD_FILE
, COLUMN
is the number of columns, ROW
is the number of rows, and the cards are listed reading across each row, starting with the top row.
For example, the board above would be specified in a file as:
3↵ 3↵ 🦄↵ 🦄↵ 🌈↵ 🌈↵ 🌈↵ 🦄↵ 🌈↵ 🦄↵ 🌈↵ | In this example, A valid board file must have exactly |
Gameplay rules
Multiple players manipulate the cards on the board concurrently. They play the game by trying to turn over pairs of identical cards. Here’s an informal summary of the rules:
- 1
- When a player turns over a first card in a pair, they control that card. If someone else already controls it, they block until they can take control.
- 2
- When the player turns over a second card, if the cards match, the player keeps control of them; otherwise, the player gives up control. The cards stay face up for now.
- 3
- When the player makes their next move, if their previous cards matched, those cards are removed from the board; otherwise, if no one controls them, they turn face down.
Complete rules
First card: a player tries to turn over a first card by identifying a space on the board…
- 1-A
- If there is no card there (the player identified an empty space, perhaps because the card was just removed by another player), the operation fails.
- 1-B
- If the card is face down, it turns face up (all players can now see it) and the player controls that card.
- 1-C
- If the card is already face up, but not controlled by another player, then it remains face up, and the player controls the card.
- 1-D
- And if the card is face up and controlled by another player, the operation blocks. The player will contend with other players to take control of the card at the next opportunity.
Second card: once a player has turned over a first card, they can try to turn over a second card…
- 2-A
- If there is no card there, the operation fails. The player also relinquishes control of their first card (but it remains face up for now).
- 2-B
- If the card is face up and controlled by a player (another player or themselves), the operation fails. To avoid deadlocks, the operation does not block. The player also relinquishes control of their first card (but it remains face up for now).
- —
- If the card is face down, or if the card is face up but not controlled by a player, then:
- 2-C
- If it is face down, it turns face up.
- 2-D
- If the two cards are the same, that’s a successful match! The player keeps control of both cards (and they remain face up on the board for now).
- 2-E
- If they are not the same, the player relinquishes control of both cards (again, they remain face up for now).
While one player is blocked turning over a first card, other players continue to play normally. They are not blocked, unless they also try to turn over a first card controlled by another player.
After trying to turn over a second card, successfully or not, the player will try again to turn over a first card. When they do that, before following the rules above, they finish their previous play:
- 3-A
- If they had turned over a matching pair, they control both cards. Now, those cards are removed from the board, and they relinquish control of them. Score-keeping is not specified as part of the game.
- 3-B
- Otherwise, they had turned over one or two non-matching cards, and relinquished control but left them face up on the board. Now, for each of those card(s), if the card is still on the board, currently face up, and currently not controlled by another player, the card is turned face down.
If the player never tries to turn over a new first card, then then the steps of 3-A/B never occur.
Example game transcript
| We start with a 3×3 grid of face-down cards and 3 players: Alice, Bob, and Charlie. | ||||||||||
Alice turns over the top left card. She controls that card for the moment (rule 1-B). |
| ||||||||||
| Bob and Charlie also try to turn over that card at the same time. Both are now blocked waiting for the chance to control it (1-D). | ||||||||||
Alice turns over the bottom right card (2-C). It doesn’t match. Alice no longer controls any cards, but they stay face up for now (2-E). Either Bob or Charlie will now get to control the top left card. |
| ||||||||||
| Bob becomes the controller of that red heart card (1-C). Alice hasn’t made another move, so the purple heart is still face up. Charlie is still waiting. | ||||||||||
Alice goes ahead and turns over a new first card, a yellow heart at the center of the board. The red heart is controlled by Bob, so it stays face up (3-B). But the purple heart isn’t controlled, so it turns face down (same 3-B). |
| ||||||||||
| While Alice is thinking about her second card, Bob turns over the top right card. His first and second cards are a match! He keeps control of them for now (2-D). Charlie is still blocked. | ||||||||||
Bob turns over a new first card, a green heart on the left side. His matched red hearts are removed from the board (3-A). |
| ||||||||||
| Charlie finally gets a chance at the top left card — but it’s gone (1-A). Charlie is now ready to turn over a new first card. Alice and Bob each control one card and will try to turn over a matching second. |
Playing over Telnet
To support multiple players using the same memory game server, we’ll define two protocols for client/server memory games: a text protocol (played over Telnet) and a HTTP protocol (played in the web browser).
First, a text protocol to use over a plain socket connection.
Each connection to the server is considered a single player. If the client disconnects from the server and reconnects, they are now playing as a different player.
Here is the formal grammar for requests and responses:
REQUEST ::= LOOK_REQUEST | FLIP_REQUEST | QUIT_REQUEST
LOOK_REQUEST ::= "look" NEWLINE
FLIP_REQUEST ::= "flip" SPACE COLUMN SPACE ROW NEWLINE
QUIT_REQUEST ::= "quit" NEWLINE
RESPONSE ::= ((MINE HERE)+ NEWLINE)+
MINE ::= SPACE | ">"
HERE ::= "*" | CARD | SPACE
CARD ::= [^\s\n\r]+
COLUMN ::= INT
ROW ::= INT
INT ::= [0-9]+
SPACE ::= " "
NEWLINE ::= "\n" | "\r" "\n"?
Columns and rows on the board are indexed starting from 1 at the left and top.
For both LOOK_REQUEST
and FLIP_REQUEST
requests, the server RESPONSE
is an ASCII-art picture of the game board.
To the left of each card is a >
if the player who sent the request controls that card, otherwise a space.
Face-down cards are represented by *
, and missing cards by a single space.
This presentation works best when cards have single-character strings, so that the columns line up; and none of the cards are *
, since that’s the symbol for face-down cards.
Depending on your terminal and Telnet client, emoji may not work either ☹️.
For example:
␣A␣*␣*↵ ␣␣>B␣␣↵ ␣*␣*␣C↵ |
In this example, There are no cards at (1,2) or (3,2). The cards at (1,1), (2,2) and (3,3) are face up, and the player receiving this message controls the |
look
The server responds with the current board.
flip column row
The server tries to flip over the card at (column, row)
for the player, following the rules above, and responds with the current board after trying to flip.
Since the response is sent after the flip, this request may block waiting for another player to relinquish control of a card. When the blocking operation completes, the response is sent, and subsequent requests from the client are processed.
quit
The server takes no action, sends no response, and closes the connection with the client.
Any cards that player controls will, unfortunately, never be relinquished.
Playing on the web
The second protocol will work over standard HTTP, which means every communication is a client request with a server response.
All of our client requests will be GET
requests for a particular path, and all server responses will be plain text.
In this protocol, requests include a player ID chosen by the client. All requests with the same player ID are the actions of a single player.
Here is the formal grammar for requests and responses:
REQUEST ::= "/look/" PLAYER
| "/watch/" PLAYER
| "/flip/" PLAYER "/" COLUMN "," ROW
RESPONSE ::= COLUMN NEWLINE ROW NEWLINE (SPOT NEWLINE)+
PLAYER ::= [\w]+
SPOT ::= "none" | "down" | "up " CARD | "my " CARD
CARD ::= [^\s\n\r]+
COLUMN ::= INT
ROW ::= INT
INT ::= [0-9]+
NEWLINE ::= "\n" | "\r" "\n"?
Columns and rows on the board are indexed starting from 1 at the left and top.
For all requests, the server responds with the current board.
In the response, COLUMN
is the number of columns, ROW
is the number of rows, and the cards are listed reading across each row, starting with the top row.
none
indicates no card in that location, down
is a face-down card, up
indicates a face-up card controlled by another player (or by no one), and my
is a face-up card controlled by the player who sent the request.
For example, the same board above for the same player:
3↵ 3↵ up␣A↵ down↵ down↵ none↵ my␣B↵ none↵ down↵ down↵ up␣C↵ |
Again, Unlike the ASCII-art, which is intended to be readable for humans, this format is intended for machines, and there is no ambiguity for a card |
GET /look/player
The server responds with the current board.
GET /watch/player
The server does not immediately respond. Instead, this requests blocks, and the server waits until the next time the board changes. At that point, it responds with the current board.
A change is defined as any cards turning face up or face down or being removed from the board.
A GET /watch/player
request must block until such a change occurs.
When there is no change in the state of the cards — for example, if a player tries to turn over an empty space, or if a player takes control or relinquishes control of a card but it does not turn face up or down — pending watch requests must continue to block.
While a GET /watch/player
request is blocked, the player may send other requests, and they will be processed normally.
For example, a concurrent GET /look/player
request must not block.
GET /flip/player/column,row
The server tries to flip over the card at (column, row)
for the player, following the rules above, and responds with the current board after trying to flip.
Since the response is sent after the flip, this request may block waiting for another player to relinquish control of a card.
Sending concurrent GET /flip/...
requests while the player is waiting for a blocked flip to complete is not allowed.
Problem 1: Game board
Specify, test, and implement a mutable threadsafe Board
ADT to represent the Memory Scramble game board.
It may be useful to define other ADTs as well.
Board.java
defines two static factory methods for creating new boards that you must support:
parseFromFile(filename)
: creates a board by parsing a file in the format described above
generateRandom(columns,rows,cards)
: generates a new columns
-×-rows
—size board filled randomly with the given cards in as equal numbers as possible
Remember to include: a specification for every class and every method you write; abstraction functions and rep invariants, and checkRep
and toString
; safety from rep exposure; and clear, complete thread safety arguments.
Iterative development recommendation: implement a single-threaded board first, and connect it to a single-threaded text server in Problem 2.1. The iter1 grading will focus on your text server without concurrency. Then come back and revise for thread safety and correct behavior with concurrent players.
For parsing board files, a parser generator (like ParserLib) is more complexity than you need. Reminder: Problem Set 2 suggested several ways to read in files.
To implement player identity:
- Each player might use a different identifier, e.g. a unique number.
- Each player might be represented by a different object, e.g. a
Player
ADT. Some of the player’s state might be stored in thePlayer
object, or theBoard
might store all the state.
You decide which actions are atomic, since the rules do not specify. In rule 3-B, for example, turning the two cards back over might be a single action — where no other player can see or manipulate the intermediate board — or two separate actions. But the board must never reach an inconsistent state, such as two players controlling the same card, or having a face-down card that a player still controls.
To implement blocking according to the gameplay rules, synchronized
blocks alone may not be enough:
- Threads might wait for a message from a blocking queue.
- You might use
Object
methodswait()
(to block) andnotifyAll()
(to wake up blocked threads). These calls must be inside asynchronized
block. When waiting threads wake up, only one will win the race for the lock, similar to how only one waiting consumer can take a single message from a blocking queue. - Or you might use another tool from the
java.util.concurrent
package.
When a player is waiting, the thread must be blocked waiting for an event, it must not be busy-waiting (see the Reordering example in Concurrency).
For keeping track of which players control cards, or matched (or mismatched) pairs that need to be removed (or turned face down), threadsafe data structures may be useful:
ConcurrentHashMap
offers atomic operations likeputIfAbsent
.- The
java.util.concurrent.atomic
package provides mutable references with atomic operations likecompareAndSet
.
Tips for reading and writing
BufferedReader.readLine()
reads a line of text from an input stream.
Its definition of a line (terminated by \n
, \r
, or \r\n
) matches the NEWLINE productions in our grammars.
You can wrap a BufferedReader
around both InputStreamReader
and FileReader
objects.
PrintWriter.println()
prints formatted text to an output stream, and terminates the current line by writing the line separator string.
It always writes a line separator, even if the input string already ends with one.
The line separator is \n
on *nix systems and \r\n
on Windows, both of which are allowed by the NEWLINE productions in our grammars.
You can wrap a PrintWriter
around a Writer
or OutputStream
objects.
Simulating a game
In ConsoleMain.java
we have provided skeleton code for a program that simulates multiple players interacting with a board, randomly flipping over cards.
At this point, you should be able to fill in that main()
method and run simulated Memory Scramble games.
By instrumenting your code, for example with println
statements or by keeping track of cards that different players flip over, you should observe correct control of cards, correct blocking when players try to flip over a first card, and correct removal of matching cards when players make a next move.
Notice that your simulation should not terminate if one player turns over matching cards in their last move (leaving two cards face up under that player’s control) and another player tries to flip one of them as their first card. In Eclipse, click the stop button to terminate the program.
Problem 2: Text server
In TextServer.java
we have provided skeleton code for a single-threaded server that accepts connections from one client at a time:
- The constructor makes a new
ServerSocket
for thisTextServer
. - The
serve()
method waits for new client, callshandleConnection()
to exchange messages with that client, then waits for a new client again. - The
handleRequest()
method is called inhandleConnection()
for each request. It has example code to handle requests that begin with"hello"
. We’ll say that a validhello
request has a single word (matching the regex\w+
) that the server should greet. "hello"
requests are not part of our protocol, and you should replace the example code once you understand it.
In the provided TextServerTest.java
file:
testHelloValid
tests the behavior of the provided server on the valid request"hello world"
.testHelloInvalid
tests its behavior on the invalid request"hello 🌍"
.
Erratum
Windows users, your system’s default character encoding might not be UTF-8, in which case testHelloInvalid
will fail, and you may have problems reading the provided board files with emoji.
😱
In Eclipse, right-click on the ps4 project, go to Properties, and on the Resource pane, under Text file encoding, select Other: UTF-8. Launching from Eclipse will now work. To launch on the command line, you need to specify the encoding there, too:
java -Dfile.encoding=UTF-8 -cp bin memory.ServerMain ... etc ...
To see the example code running, use ServerMain.java
.
This provided class parses command-line arguments, calls one of the static Board
creators based on those arguments, and creates and starts a server.
You should not need to change any of this code.
To run ServerMain
on the command line, first cd
to your ps4
directory.
Then start a text server on port 4444 with a 3×3 board of rainbows and unicorns:
java -cp bin memory.ServerMain text 4444 3 3 1F308 1F984
To run it in Eclipse, first right click ServerMain
and select Run As → Java Application.
You should see an error in the console.
Then go to Run → Run Configurations…, select the “Arguments” tab, and enter in the “Program arguments” box:
text 4444 3 3 1F308 1F984
See Unicode’s Full Emoji List to find character codes for other emoji. To start the server with a board from a file, give the filename instead of size and cards:
text 4444 boards/perfect.txt
Once the server is running, connect using telnet localhost 4444
in another terminal.
See the reading on Sockets & Networking for an introduction to telnet
and advice for running it on Windows.
The TextServer
doesn’t use the game board yet, but we can try out the handler for hello
requests.
Here is an example Telnet session.
User input is shown in green.
For input to the telnet connection, newlines (pressing enter) are also shown with ↵:
$ telnet localhost 4444
Trying ::1...
Connected to localhost.
Escape character is '^]'.
hello world↵
Hello,
world!
hello (world)↵
Go away,
(world).
Ctrl-]
telnet> quit↵
Connection closed.
Our server doesn’t support the quit
request yet, so instead we type the escape character Ctrl
-]
and quit the Telnet program to close the connection.
Remember that the server keeps running until you terminate its process, and if you try to run a second server on the same port, it will fail with “address already in use”.
If you run the server on the command line, use Ctrl
-C
in that terminal.
If you run the server in Eclipse, click the double-gray-X button to close consoles for terminated programs, then click the stop button.
Specify, test, and implement a complete server: first single-threaded, then multithreaded.
2.1 Single-player game
In TextServer
, complete the constructor and the handleRequest
method so the server implements the text protocol and gameplay rules above, using a single instance of your game board ADT.
The specifications of the constructor, port()
, and serve()
are required, but you are free to change any of the code in this class.
You can and should change the spec of handleRequest()
, for example to keep track of the current player, or to handle QUIT_REQUEST
inputs that have no response.
See tips for reading and writing in Problem 1. Also:
- The example code in
handleConnection()
creates aPrintWriter
with automatic flushing: callingprintln()
will flush the stream automatically. If you use another method, likeprint()
, remember to callflush()
. - It is easy to end up with an extra newline at the end of a response, by calling
println()
with a string that already has a newline at the end. Make sure you follow the protocol to the letter.
When you write tests in TextServerTest
, you may write tests similar to the examples we’ve provided, and you may also write tests using ByteArrayInput
/OutputStream
stubs as described in Sockets & Networking.
Emoji may not work in your terminal or Telnet client 😭 — use letter cards instead.
In the arguments to ServerMain
for a random board, capital A is 41, B is 42, etc.
For example, to start a 3×3 board of A
and B
:
java -cp bin memory.ServerMain text 4444 3 3 41 42
2.2 Multi-player game
Modify your TextServer
so it can maintain multiple client connections simultaneously, and clients can play together on the same game board.
Each client connection should be maintained by its own thread.
When one client is blocked waiting to control a card, other clients must be able to continue playing normally.
Make sure your TextServer
has: complete specifications; abstraction function, rep invariant, and checkRep
; safety from rep exposure; and a clear, complete thread safety argument.
While TextServer
does not need to be a threadsafe type (it may only be safe to use a TextServer
instance from one thread), it must document why its use of multiple internal threads is safe.
Problem 3: Board listeners
Specify, test, and implement support for listeners to your game board that are informed when the board changes.
You may choose to implement either listeners that are informed of only a single change, or listeners that are informed of all future changes.
In the next problem, you will use your board listeners to implement “watch” requests, notifying web players when the board changes. In that specification:
- A change is defined as any cards turning face up or face down or being removed from the board.
- When there is no change in the state of the cards — for example, if a player tries to turn over an empty space, or if a player takes control or relinquishes control of a card but it does not turn face up or down — it is not considered a change.
You decide the specification for Board
listeners:
- If the listener receives only a single change, it might be a blocking method that returns when the board changes.
- Listeners might implement a callback interface specified by
Board
, and call a method to add themselves as a listener. The board will call them back on each change. - Listeners might be consumers of a threadsafe queue of events provided by the board. The board puts an event object on the queue for each change.
You decide which changes are atomic. The “watch” request specification does not say whether, when a pair of matched (or mismatched) cards is removed (or turned face down), that is reported as a single change or two separate changes.
At this point, you should be able to use your listener support in ConsoleMain.java
to watch the board during a simulated game.
Problem 4: Web server
To implement the web server, we will use the com.sun.net.httpserver
HTTP server library provided as part of the official implementation of Java 8.
In WebServer.java
we have provided skeleton code for a web game server:
- The
WebServer
constructor creates aHttpServer
and sets it to handle concurrent requests by automatically creating and reusing multiple threads (Executors.newCachedThreadPool
). - The constructor has an example of attaching a handler for requests to paths that begin with the prefix
/hello/
. The handler receives aHttpExchange
object that represents a single client/server request/response. The handler callback will run on a background thread, not the main thread. Since the server handles concurrent requests on multiple threads, multiple threads may be running various callbacks concurrently. - The example handler calls the private method
handleHello()
, which demonstrates how to use theHttpExchange
to examine the request and send a response. IfhandleHello
were to block, the client would be blocked waiting to read the response. - Requests for
/hello/
are not part of our protocol, and you should replace the example code once you understand it.
In the provided WebServerTest.java
file:
testHelloValid
tests the behavior of the provided server on the valid requestGET /hello/w0rld
.testHelloInvalid
tests its behavior on invalid requestGET /hello/world!
.
To see the example code running, use ServerMain.java
to start a web server on port 8080.
For example, on the command line:
java -cp bin memory.ServerMain web 8080 3 3 1F308 1F984
The WebServer
doesn’t use the game board yet, but we can try out the handler for /hello/
.
Browse to:
http://localhost:8080/hello/world
→ valid request,Hello, world!
http://localhost:8080/hello/%F0%9F%8C%8D
→ invalid request,Go away, 🌍.
Specify, test, and implement support for /look/player
, /watch/player
, and /flip/player/column,row
requests according to the HTTP game protocol.
In WebServer
, the specifications of the constructor, port()
, start()
, and stop()
are required, but you are free to change any of the code in this class.
While one request is blocked, other requests must be handled normally.
In the provided code,
server.setExecutor(Executors.newCachedThreadPool());
handles each request on a separate thread.
Make sure your WebServer
has: complete specifications; abstraction function, rep invariant, and checkRep
; safety from rep exposure; and a clear, complete thread safety argument.
While WebServer
does not need to be a threadsafe type (it may only be safe to use a WebServer
instance from one thread), it must document why it is safe to call back its own request handlers from multiple internal threads.
Once your server is complete, you can play the game online! We’ve built a client web page that speaks the game protocol with your server:
Play Memory Scramble
Enter the address of your running server (the default is localhost:8080
), or connect to a friend’s server by IP address.
There is no code running on the web.mit.edu
server here: the web page is running JavaScript in your browser that talks directly with your game server.
Before you’re done
Make sure you have documented specifications, in the form of properly-formatted Javadoc comments, for all your types and operations.
The spec of every type should include whether it is threadsafe.
Make sure you have documented abstraction functions and representation invariants, in the form of a comment near the field declarations, for all your implementations.
With the rep invariant, also say how the type prevents rep exposure.
Make sure every threadsafe type has a documented thread safety argument that explains why that type is threadsafe.
For every type that creates or uses multiple threads, its thread safety argument must explain why its use of multiple threads is safe.
Make sure you specify, test, and implement
equals()
andhashCode()
for all immutable types.Use
@Override
when you overridetoString()
,equals()
, andhashCode()
.Also use
@Override
when a class implements an interface method, to remind readers where they can find the spec.Make sure you have a thorough, principled test suite for every type.
Submitting
Make sure you commit AND push your work by 10:00pm on the deadline date.
This problem set can take much longer for Didit to process than previous psets, so push early. If you see, for example, 7 builds pending in Didit’s SYSTEM STATUS, that means there are seven people whose builds much finish before yours can start.
Remember that Didit feedback is provided on a best-effort basis:
- There is no guarantee that Didit tests will run within any particular timeframe, or at all. If you push code close to the deadline, the large number of submissions will slow the turnaround time before your code is examined.
- If you commit and push right before the deadline, the Didit build does not have to complete in order for that commit to be graded.
- Passing some or all of the public tests on Didit is no guarantee that you will pass the full battery of autograding tests, but:
- Failing the
TextProtocolTest
is sure to mean many lost points on the problem set. - Failing the
WebProtocolTest
is sure to mean many lost points on iter2.
- Failing the
Grading
Your overall ps4 grade will be computed as approximately:
35% iter1 autograde + 5% iter1 manual grade + 45% iter2 autograde + 15% iter2 manual grade
The autograder test cases will not change from iter1 to iter2, but their point values will. In order to encourage incremental development, iter1 autograding will focus on the text protocol, not HTTP; and on situations where clients make moves in series, not concurrently. On iter2, autograding will look at both servers, and situations with more concurrency.
Manual grading of iter1 will only examine internal documentation (AF, RI, etc.) and implementation that does not relate to concurrency. Manual grading of iter2 may examine any part, including re-examining those docs or code, and how you addressed code review feedback.