Fall Semester 2021
DUE: Nov 8 (Monday), 2021 09:30 AM
GitHub Classroom Link: https://classroom.github.com/a/O_WUEwwr
For this assignment, your group will profile a Conway's Game of Life simulation, and improve its performance by refactoring several methods (to be determined by the results of the profiling). The program is assumed to be functionally correct. It's only problem is that certain features are too slow. This will consist of the following steps for each method:
- Profiling to determine the most CPU-intensive method that is suboptimal.
- Adding method pinning tests to guard against unintended changes to functionality.
- Refactoring the method to make it more performant.
- Verifying that the pinning tests you added still pass.
- Profiling again to confirm that your refactored method is now performant.
The code is available under the src/ directory.
- Running GameOfLife. For Windows do (for running SlowLifeGUI with argument 5):
For Mac / Linux do (for running SlowLifeGUI with argument 5):
run.bat 5
bash run.sh 5
- Running the TestRunner on GameOfLifePinningTest. For Windows do:
For Mac / Linux do:
runTest.bat
bash runTest.sh
Alternatively, I've created an Eclipse project for you so you can use Eclipse to import the existing project.
The program is an implementation of Conway's Game of Life (https://en.wikipedia.org/wiki/Conway%27s_Game_of_Life). You can change the state of a cell (from living to dead) by clicking on one of the buttons. Cells which are currently alive have an X and a red background; cells that are dead now, but were at any point alive during the current run, will have a green background.
There are several other buttons which invoke different features:
- Run - this will run one iteration of the Game of Life
- Run Continuous - This will run iterations until you press the Stop button.
- Stop - This will stop the current "Run Continuous" run. It will have no effect if the program is not running continuously already.
- Write - This will write the state of the system to a backup file, to be loaded later.
- Undo - This will undo the previous iteration. It will only work for one iteration (that is, you cannot do multiple undos to get back multiple iterations).
- Load - This will load a previously-saved backup file (created using the Write button) to the current world.
- Clear - This will clear the current world.
The application accepts one command line argument, specifying the size of the world (e.g., if you enter 10, then you will create a 10 x 10 world).
For the purposes of performance testing, we will focus on a 5 X 5 world. For the initial pattern, we will use the "blinker" pattern shown in:
https://en.wikipedia.org/wiki/Conway%27s_Game_of_Life#Examples_of_patterns
The actual pattern GIF is at:
https://en.wikipedia.org/wiki/Conway%27s_Game_of_Life#/media/File:Game_of_life_blinker.gif
We will start from the vertical bar on a 5X5 matrix as shown in the GIF.
For an actual full performance test suite, we would have to try multiple world sizes and multiple patterns but for the purposes of this deliverable, we will focus on performance debugging only the above scenario. As it happens, once we debug the above scenario, the application will start running quickly for all scenarios.
There are exactly THREE major performance issues with THREE methods in the code. They could be in any feature of the program! I recommend you try exploratory testing to try out different features to determine which features may have performance problems before profiling the application. There are TWO features that have problems (that is, two buttons). The three performance problems are dispersed in those two features.
In order to determine the "hot spots" of the application, you will need to run a profiler such as VisualVM (download at https://visualvm.github.io/). Using a profiler, determine the THREE methods you can modify to measurably increase the speed of the application without modifying behavior. Refer to Exercise 4 for a detailed explanation of how to use VisualVM to profile an application.
Now there is one more step that you have to do on VisualVM you did not have to do in Exercise 4: you need to replace "GameOfLife" with "*" in the "Profile classes:" window on the right before pressing on the "CPU" button to start profiling. This instructs VisualVM to instrument not only the GameOfLife class (the class with the main method), but all classes in the application. You did not need to do this for the Exercise 4 MonkeySim application because it was single-threaded application. All code in a single-threaded application execute starting from the main method, so the default behavior of VisualVM to instrument starting from the main method class was just fine. GameOfLife is a GUI application and in a Java GUI application, there are multiple event handler threads running in the background to handle events like button presses concurrently with the application. In this case, the code for these threads do not start from the main method.
Before doing refactoring any method, you should create "pinning tests" (as described in the section on legacy code earlier - please review the slides on Writing Testable Code if you need a refresher). These pinning tests should check that the behavior of a modified method was not changed by your refactor. The methods should work EXACTLY the same as before, except they should be faster and take up less CPU time. There should be at least one pinning test per method refactored.
In general, a pinning test doesn't necessarily have to be a unit test; it can be an end-to-end test that you slap on quickly for the purposes of refactoring (without spending the effort to localize tests by mocking external objects). The end-to-end pinning test is then gradually refined into more high quality unit tests. Sometimes this 2-step process is necessary because sometimes you cannot write high quality unit tests before refactoring to make the code more testable (e.g. via dependency injection). So you need a temporary end-to-end pinning test to protect the code base meanwhile. For this deliverable, there is no reason you cannot write unit tests from the get-go for pinning tests as the dependency injection(s) has already been done for you. An example is the setCells method in MainPanel.
Here are some requirements for your pinning tests:
-
You will use the 5 X 5 blinker pattern that I described above when a pattern is required:
https://en.wikipedia.org/wiki/Conway%27s_Game_of_Life#/media/File:Game_of_life_blinker.gif
The vertical bar pattern should be your precondition and the next horizontal bar pattern should be your postcondition. For the postcondition, make sure you check all 25 cells in the 5 X 5 pattern.
-
You are required to localize each pinning unit test within the tested class as we did for Deliverable 2 (meaning it should not exercise any code from external classes). You will have to use Mockito mock objects to achieve this.
-
Also, you may have to use behavior verification instead of state verification to test some methods because the state change happens within a mocked external object. Remember that you can use behavior verification only on mocked objects (technically, you can use Mockito.verify on real objects too using something called a Spy, but you won't need it for this deliverable). You will get point deductions if you don't use mock objects and behavior verification appropriately.
-
Note that even though the class is named GameOfLifePinningTest, the methods you test will not necessarily come from the GameOfLife class. You will create whatever objects from whatever classes are necessary to test the three refactored methods. Hint: there is no reason for you to create a GameOfLife object as there are no methods that you need to refactor there.
You will write all your pinning tests in the class GameOfLifePinningTest by completing the TODOs. Please heed the comments. Just like for Deliverable 2, you can add a Java stack trace to the error message to get information about why your tests are failing by inserting the following line below TestRunner.java line 24:
System.out.println(f.getTrace());
Now refactor the three methods so that they are no longer performance problems. If you look carefully, the three methods do a lot of wasted work for no reason. It should be easy to refactor my removing that work. Make sure that your pinning tests pass after refactoring.
Please use the ReportTemplate.docx file provided in this directory to write your report. If you don't have a .docx compatible word processor, that's perfectly fine as long as you follow the same organization. A PDF version of the file is at ReportTemplate.pdf. Please keep the page separation.
The report should have a title page with:
- Your name(s)
ON YOUR FIRST PAGE, please write a brief introduction on any issues you may have had with VisualVM and also a division of work between partners.
ON A SEPARATE PAGE, write a brief report on the first feature you optimized. Write the name of the feature, the methods you refactored, and a VisualVM screenshot of method "Hot spots" after the refactoring. Please only include the "Hot spots" window in the interest of space. Please refer to Exercise 4 on how the Hot spots window looks like.
ON A SEPARATE PAGE, write a brief report on the second feature you optimized. Write the name of the feature, the methods you refactored, and a VisualVM screenshot of method "Hot spots" after the refactoring. Please take care not to click on any other feature while profiling (including the first feature).
There are two features with performance problems as I mentioned (as in two buttons). Make sure you include both of them. One feature has two problematic methods, another feature has one problematic method.
- Report - 10%
- My performance tests pass (autograder) - 45%
- My pinning tests pass (autograder) - 15%
- Your pinning tests pass (autograder) - 15%
- Your pinning tests are written correctly - 15%
Please read grading_rubric.txt before submitting!
Note that 75% of your grade (besides the report) will be graded by GradeScope autograding. However, adjustments to your autograded score may follow if you make a bad faith attempt at tricking the autograder (e.g. write a pinning test that does not properly test the method you are refactoring).
Each pairwise group will do one submission to GradeScope as usual. The submitting member must use the "View or edit group" link at the top-right corner of the assignment page after submission to add his/her partner.
The submission is divided into two parts:
-
Submit the repository created by GitHub Classroom for your team to GradeScope at the Deliverable 4 GitHub link. Once you submit, GradeScope will run the autograder to grade you and give feedback. If you get deductions, fix your code based on the feedback and resubmit. Repeat until you don't get deductions.
-
Submit your report to GradeScope at the "Deliverable 4 Report" link.
It is encouraged that you submit to GradeScope early and often. Please use the feedback you get on each submission to improve your code!
The GradeScope autograder works in 3 phases:
-
GameOfLife method performance tests (45.0):
I run performance tests on each of the three methods you should optimize. If any of those methods time out after 10 ms, you get a -15 deduction.
-
GameOfLife method pinning tests (15.0):
I run my own pinning tests on each of the three methods you should optimize. These pinning tests pass without you having to do anything (obviously because they are meant to test existing behavior of legacy code). And they should stay that way. If any of the pinning tests fail, you get a -5 deduction.
-
GameOfLifePinningTest method tests (15.0):
I run the pinning tests you wrote (GameOfLifePinningTest) on your implementation. If any of the pinning tests fail, you get a -5 deduction.
-
GameOfLifePinningTest Mocking and Behavior Verification (0.0):
This section gives you feedback on whether you did proper mocking and behavior verification. It does three test runs using your GameOfLifePinningTest:
-
Output on correct program: GameOfLifePinningTest tested on a defect-free implementation. The expected output is:
-------------------------------------------------------------------- ALL TESTS PASSED --------------------------------------------------------------------
-
Output after injecting bug into real object: GameOfLifePinningTest tested on GameOfLifeBuggy.jar with a bug injected into a real object. Since a real object used in your test becomes buggy, the test case that uses that real object should fail.
-
Output after injecting bug into mocked object: GameOfLifePinningTest tested on GameOfLifeBuggy.jar with a bug injected into a mocked object. Since the code in a mocked object is not exercised, the injected bug should have no effect and again, the expected output is:
-------------------------------------------------------------------- ALL TESTS PASSED --------------------------------------------------------------------
-
Output after injecting bug in method checked by behavior verification: GameOfLifePinningTest tested on GameOfLifeBuggy.jar with a bug injected into a method that can only be tested using behavior verification. The method that does behavior verification should fail.
If all goes well, you should see the followimg lines at the end of this section:
PASSED (5/5): Bug injected into real object caused test failures (as it should). PASSED (5/5): Bug injected into mocked object did not cause test failures. PASSED (5/5): Behavior verification correctly detected change in behavior.
If you see FAILED instead, you need to fix your tests. GameOfLifeBuggy.jar is included in the repository if you want to do further testing. Try running runBuggyReal.bat, runBuggyMock.bat, and runBuggyBehavior.bat to execute the JAR file with real object, mocked object, and behavior verification method bug injection respectively (Mac/Linux users can run the *.sh versions). If you try the vertical bar blinker patter specified on GameOfLifePinningTest.java, you will see that each version is defective in its own way. You can also try running your GameOfLifePinningTest on the buggy implementation using runTestBuggyReal.bat, runTestBuggyMock.bat and runTestBuggyBehavior.bat for the three bug injections. You will see the same output given on the GradeScope feedback.
Just because you got PASSED on all three, it does not mean that you are guaranteed to get points for that rubric item. You may have passed simply because you did not yet write the relevant test! So in the end, points will be assigned through manual grading (hence the 0 points assigned in the autograder). But if you wrote the tests and you see FAILED, then you most definitely have a problem.
-
Just like for Exercise 4, each of you should use VisualVM to profile the application and come up with the three methods to refactor. Note that unlike Exercise 4, there are many more features for this program so it is going to be slightly harder to find the methods. After you are done, compare the three methods, discuss, and agree upon the final three.
Next, split the refactoring and testing of the three methods between the two of you. The partner that focused more on testing for Deliverable 2 should now focus more on refactoring, and vice versa, the partner that focused more on implementation for Deliverable 2 should now focus more on the pinning tests. The goal is for both of you to have a balanced set of experiences.
-
VisualVM Download: https://visualvm.github.io/download.html
-
VisualVM Documentation: https://visualvm.github.io/documentation.html
Method profiling is not the only thing that VisualVM knows how to do. It can also profile overall CPU usage, heap memory usage, thread creation/termination, class loading/unloading, Java just-in-time compiler activity, etc. It can also profile heap memory in a detailed way to show which types of objects are filling the memory and where their allocation sites were. And needless to say, VisualVM is not the only profiling tool out there.
In the unlikely case you can't find what you are looking for in existing profilers, you can even write your own profiler using the Java Virtual Machine Tool Interface (JVMTI). JVMTI is what was used to build VisualVM.
-
Creating a Debugging and Profiling Agent with JVMTI https://www.oracle.com/technical-resources/articles/javase/jvmti.html
-
JVMTI Reference https://docs.oracle.com/javase/8/docs/platform/jvmti/jvmti.html