Assignment 1: Counting in MapReduce due 4:00pm Jan. 29

By now, you should already be familiar with the Hadoop execution environment (e.g., submitting jobs) and using Maven to organize your assignments. You will be working in the same repo as before, except that everything should go into the package namespace ca.uwaterloo.cs451.a1.

Note that the point of assignment 0 was to familiarize you with GitLab and the Hadoop development environment. Starting this assignment, excuses along the lines of "I couldn't get my repo set up properly", "I couldn't figure out how to push my assignment to Git", etc. will not be accepted. It is your responsibility to sort through any mechanics issue you have.

Before starting this assignment, it is highly recommended that you look at the implementations of bigram relative frequency and co-occurrence matrix computation in Bespin.

In this assignment you'll be computing pointwise mutual information, which is a function of two events x and y:

PMI equation

The larger the magnitude of PMI for x and y is, the more information you know about the probability of seeing y having just seen x (and vice-versa, since PMI is symmetrical). If seeing x gives you no information about seeing y, then x and y are independent and the PMI is zero.

Write two separate implementations (more details below) that computes the PMI of words in the data/Shakespeare.txt collection that's used in the Bespin demos and the previous assignment. Your implementation should be in Java. To be more specific, the event we're after is x occurring on a line in the file (the denominator in the above equation) or x and y co-occurring on a line (the numerator in the above equation). That is, if a line contains "A B C", then the co-occurring pairs are:

If the line contains "A A B C", the co-occurring pairs are still the same as above; same if the line contains "A B C A B C"; or any combinations of A, B, and C in any order.

A few additional specifications:

To ensure consistent tokenization, use the Tokenizer class provided in Bespin that you've used for assignment 0.

You will build two versions of the program (put both in package ca.uwaterloo.cs451.a1):

  1. A "pairs" implementation. The implementation must use combiners. Name this implementation PairsPMI. In the final output, the key should be a co-occurring pair, e.g., (A, B), and the value should be a pair (PMI, count) denoting its PMI and co-occurrence count.
  2. A "stripes" implementation. The implementation must use combiners. Name this implementation StripesPMI. In the final output, the key should be a word (e.g., A), and the value should be a map, where each of the keys is a co-occurring word (e.g., B), and the value is a pair (PMI, count) denoting its PMI and co-occurrence count.

Since PMI is symmetrical, PMI(x, y) = PMI(y, x). However, it's actually easier in your implementation to compute both values, so don't worry about duplicates. Also, use TextOutputFormat so the results of your program are human readable.

Make sure that the pairs implementation and the stripes implementation give the same answers! Make sure that your code runs in the Linux Student CS environment (even if you do development on your own machine), which is where we will be doing the grading. "But it runs on my laptop!" will not be accepted as an excuse if we can't get your code to run.

Running on the Datasci cluster

Now, on the Datasci cluster, run your pairs and stripes implementation on the sample Wikipedia collection stored on HDFS at /data/cs451/simplewiki-20180901-sentences.txt. This is a dump of "simple" Wikipedia that has been tokenized into sentences, one per line. As with the Shakespeare collection, we care about co-occurrence on a line (which is a sentence in this case).

Make sure your code runs on this larger dataset. Assuming that there aren't many competing jobs on the cluster, your programs should not take more than 10 minutes to run. For reference, on an idle cluster, the reference pairs implementation takes around 6 minutes to run and the reference stripes implementation takes around 3 minutes to run.

If your job is taking much longer than that or if it doesn't look your job is making obvious progress, then please kill it so it doesn't waste resources and slow other people's jobs down. To kill your job, first find its application id in the RM webapp, then issue:

$ yarn application -kill application_xxxxxxxxxxxxx_xxxx

Obviously, if the cluster is really busy or if there's a long list of queued jobs, your job will take longer, so use your judgement here. The only point is: be nice. It's a shared resource, and let's not let runaway jobs slow everyone down.

One final detail, set your MapReduce job parameters as follows:

job.getConfiguration().setInt("mapred.max.split.size", 1024 * 1024 * 32);
job.getConfiguration().set("mapreduce.map.memory.mb", "3072");
job.getConfiguration().set("mapreduce.map.java.opts", "-Xmx3072m");
job.getConfiguration().set("mapreduce.reduce.memory.mb", "3072");
job.getConfiguration().set("mapreduce.reduce.java.opts", "-Xmx3072m");

What the last four options do is fairly obvious. The first sets the maximum split size to be 32 MB. What effect does that have? (Hint, consider the physical execution of MapReduce programs we discussed in class)

It's worth noting again: the Datasci cluster is a shared resource, and how fast your jobs complete will depend on how busy it is. You're advised to begin the assignment early as to avoid long job queues. "I wasn't able to complete the assignment because there were too many jobs running on the cluster" will not be accepted as an excuse if your assignment is late.

Turning in the Assignment

Please follow these instructions carefully!

Make sure your repo has the following items:

When grading, we will pull your repo and build your code:

$ mvn clean package

Your code should build successfully. We are then going to check your code (both the pairs and stripes implementations).

We're going to run your code on the Linux student CS environment as follows (we will make sure the collection is there):

$ hadoop jar target/assignments-1.0.jar ca.uwaterloo.cs451.a1.PairsPMI \
   -input data/Shakespeare.txt -output cs451-bigdatateach-a1-shakespeare-pairs \
   -reducers 5 -threshold 10

$ hadoop jar target/assignments-1.0.jar ca.uwaterloo.cs451.a1.StripesPMI \
   -input data/Shakespeare.txt -output cs451-bigdatateach-a1-shakespeare-stripes \
   -reducers 5 -threshold 10

Make sure that your code runs in the Linux Student CS environment (even if you do development on your own machine), which is where we will be doing the grading. "But it runs on my laptop!" will not be accepted as an excuse if we can't get your code to run.

You can run the above instructions using check_assignment1_public_linux.py, with something like:

$ wget https://student.cs.uwaterloo.ca/~cs451/assignments/check_assignment1_public_linux.py
$ chmod +x check_assignment1_public_linux.py
$ ./check_assignment1_public_linux.py bigdatateach

We're going to run your code on the Datasci cluster as follows:

$ hadoop jar target/assignments-1.0.jar ca.uwaterloo.cs451.a1.PairsPMI \
   -input /data/cs451/simplewiki-20180901-sentences.txt -output cs451-bigdatateach-a1-wiki-pairs \
   -reducers 5 -threshold 50

$ hadoop jar target/assignments-1.0.jar ca.uwaterloo.cs451.a1.StripesPMI \
   -input /data/cs451/simplewiki-20180901-sentences.txt -output cs451-bigdatateach-a1-wiki-stripes \
   -reducers 5 -threshold 50

You can run the above instructions using check_assignment1_public_datasci.py, with something like:

$ wget https://student.cs.uwaterloo.ca/~cs451/assignments/check_assignment1_public_datasci.py
$ chmod +x check_assignment1_public_datasci.py
$ ./check_assignment1_public_datasci.py bigdatateach

Important: Make sure that your code accepts the command-line parameters above! That is, make sure the check scripts work!

When you've done everything, commit to your repo and remember to push back to origin. You should be able to see your edits in the web interface. Before you consider the assignment "complete", verify everything above works by performing a clean clone of your repo and going through the steps above.

That's it! There's no need to send us anything—we already know your username from assignment 0. Note that everything should be committed and pushed to origin before the deadline.

Hints

Update 2024-09-06 4:40p: Note that floating point numbers are inexact. Your results and the expected results are both rounded to 4 decimal places.

Grading

This assignment is worth a total of 35 points, broken down as follows:

Reference Running Times

To help you gauge the efficiency of your solution, we are giving you the running times of our reference implementations. Keep in mind that these are machine dependent and can vary depending on the server/cluster load.

Class name Running time Linux Running time Datasci
PairsPMI 80 seconds 5 minutes 45 seconds
StripesPMI 50 seconds 3 minutes

Back to top