Due: 9/9 (Fri) 18:00 (This is a hard deadline)
This assignment is for you to become familiar with Python, which will be used to build a backend system in hw3. This is an individual assignment.
ANY KIND OF CHEATING IS NOT ALLOWED FOR THE HOMEWORK. Please be aware that we run software plagiarism detection tools.
From the beginning to the end of the development, you must use GitHub for version control. This will be used for your submission as well, so please keep it in mind.
In this assignment, you will practice fetching HTML files from the Internet website, parsing the HTML files, and creating a csv file for the parsed data. Based on the provided skeleton code, you will implement the functions needed to do the tasks above.
You don't have to understand the implemented methods provided in the skeleton code, but studying them can be useful for your future term project. This is a modification of google's Baby Names Python Exercise.
The goal of this assignment is to practice using various functions of the Python language. In this assignment, you will get familiar with:
- Built-in functions
- Python provides many useful built-in functions. You can write code more efficiently and effectively if you are fluent with such functions.
- Class
- You will need to modularize and encapsulate your code to neatly manage your project. Class is a core concept to achieve such objectives. It is an important component in design patterns that you will learn in future classes. Since you will define your own models with classes in homework 3, you should be familiar with it.
- Decorator
- Decorator is a special syntax in python for transforming functions using closure. It is also an important feature for applying design patterns in python, such as defining static methods or class methods. Many functions from Django are provided as decorators as well, especially with user authentication, which you will be using in homework 3.
- Lambda function
- Lambda function helps you write your python code in a functional programming style. In certain situations, functional style makes your code more concise and easy to modularize. For more information, read Functional Programming HowTo.
You need to implement the fetch.py
file that sends a request to fetch the HTML files from the Social Security Administration website, which provides neat data on each year's most popular names for newborn babies in the US. We will extract HTML files ranging from 2001 to 2018. After fetching the files, you should save them under babydata
directory. The expected HTML files are given for reference under babydata
directory. HTML files generated from your script should be identical to example HTML files under babydata
. To complete the script, you should implement the following functions:
1) fetch_top_1000
This method receives the year of interest and the target webpage's url and returns the contents of the html file fetched from the given webpage. Use urllib
to fetch the html file. Refer to python documentation for more information on urllib
. (Hint: You may explore the official babynames website and try sending requests. Send a request with Chrome network tab open, and then by inspecting the request with the Chrome network tab, you will be able to find which items are required in the body field.)
2) safe_internet_fetch
You need to implement a decorator that checks whether the request was successful. This decorator will decorate the fetch_top_1000
function to handle the situations where fetching fails due to an invalid url or unstable internet connection by throwing BabyFetchException
.
You can catch the above error using [urllib.error.URLError] (https://docs.python.org/3/library/urllib.error.html)
This method should be implemented as a decorator in order to be easily applied to other methods. Please check out this document to understand how to implement python decorator.
For debuggability, defining your own exception and intentionally raising the exception is useful because you can identify the root cause of the exception and distinguish the exception from others. We provide BabyFetchException
for this purpose. If the decorated function fails to fetch the data, this decorator has to raise the custom BabyFetchException
with custom error message: Request Failed
. We will not grade your decorator with its stack trace but just test the class of the raised exception and its message.
You need to implement the babyname parser class that parses the popular names and their ranks from the html files that your script generated.
In the skeleton code, BabyParser
class is defined in babyname_parser.py
. To complete the parser, you should implement the following functions:
1) __init__
This method is the constructor of the parser. In this method, you need to extract a list of the tuples of (rank, male-name, female-name) from the given html file.
2) check_filename_existence
You need to implement a decorator that checks whether the file exists or not.
This decorator will decorate the __init__
of the parser to prevent constructing a parser that parses non-existing file names.
Just like the safe_internet_fetch
above, we provide BabynameFileNotFoundException
for this purpose.
If there is no such file with the filename given by the function to decorate, this decorator has to raise the custom BabynameFileNotFoundException
with custom error message: No such file: {filename}
.
You can assume that the dirname
and year
are given as first two arguments for the function to decorate.
3) parse
This method applies a given lambda function to all of the extracted tuples and returns the results.
Usage
When you are done, you can parse any information about the popluar name ranks in a year by using the implemented parser.
In the python console, you can test your code with the provided 2001.html
file like below.
>>> from babyname_parser import BabynameParser
>>> parser = BabynameParser("/{YOUR_HW_PATH}/babydata", 2001)
>>> parser.year
'2001'
>>> parser.rank_to_names_tuples
[('1', 'Jacob', 'Emily'), ('2', 'Michael', 'Madison'), ('3', 'Matthew', 'Hannah'), ...]
>>> parser.parse(lambda rank_to_names_tuple: rank_to_names_tuple[0])
['1', '2', '3', '4', '5', '6', '7', '8', '9', '10', ...]
>>> BabynameParser("/{YOUR_HW_PATH}/babydata", 1900)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "babyname_parser.py", line 61, in decorated
raise BabynameFileNotFoundException("No such file: {}".format(filename))
babyname_parser.BabynameFileNotFoundException: No such file: /{YOUR_HW_PATH}/babydata/1900.html
You should NOT edit parts other than those marked as TODO
s in your skeleton code when you submit your assignment, because we will grade your BabynameParser
implementation by not only running run.py
file but also importing the class and test the methods like above (with different files and lambdas, of course).
Also, the error message that comes out when the entered file does not exist must be
No such file: {filename}
and the result of parse
method must be a list of all processed tuples.
The same applies to the fetch.py
.
In run.py
file, a skeleton code using the parser is provided.
Your task is to complete the code to parse all the html files that contain the popular baby names BabynameParser
, and save all the data in a single csv file.
When you run python run.py 2001 2018
,the script must generate a csv file named babyname.report.csv
under babydata
directory.
The content of the csv file should be as follows:
$ tail -10 /{YOUR_HW_PATH}/babydata/babyname.report.csv
2018,991,Valery,F,
2018,992,Aiyana,F,147
2018,993,Azaria,F,
2018,994,Evalyn,F,42
2018,995,Kailyn,F,92
2018,996,Keilani,F,
2018,997,Whitley,F,
2018,998,Austyn,F,
2018,999,Danica,F,128
2018,1000,Elina,F,
$ head -10 /{YOUR_HW_PATH}/babydata/babyname.report.csv
year,rank,name,gender,rank_change
2001,1,Jacob,M,
2001,2,Michael,M,
2001,3,Matthew,M,
2001,4,Joshua,M,
2001,5,Christopher,M,
2001,6,Nicholas,M,
2001,7,Andrew,M,
2001,8,Joseph,M,
2001,9,Daniel,M,
We will test your code under Python 3.9.
fetch.py
(total 3 points)safe_internet_fetch
: 1 pointfetch_top_1000
: 2 points
babyname_parser.py
(total 6 points)check_filename_existence
: 1 pointBabynameParser > __init__
: 3 pointsBabynameParser > parse
: 2 points
run.py
(total 6 points)BabyRecord > to_csv_record
: 1 pointmain
: 5 points
- Partial points for minor errors
Due: 9/9 (Fri) 18:00 (This is a hard deadline)
You must create your own private repository under your account. You don't need to submit your homework by email. Make sure to name your repository properly and to add the TAs as collaborators in your repository settings! (Detailed instructions below). We will take the snapshot of the main branch of your Github repository at the deadline and grade it. Since there are many students in this class, we need to automatize the grading process, so your homework may not be graded properly if your directory hierarchy doesn't look like this:
repository_root
├── LICENSE.txt
├── NOTICE.txt
├── babydata
│ ├── 2001.html-2018.html
│ └── babyname.report.csv
├── babyname_parser.py
├── fetch.py
└── run.py
Also, make sure to push your work on Github on time.
- Go to
https://education.github.com
. - Follow instructions to
Request a discount
.
- Go to your github profile:
https://github.com/USERNAME
. - Under
Repositories
, click onNew
. - Fill in
swpp-hw1-USERNAME
as your repository name, and mark it asPrivate
. (e.g.,swpp-hw1-jaewooMaeng
)- You will get points deducted by setting your repository format wrong (e.g., swpp_hw1_jaewooMaeng (X)).
- If you make your repository public, you will get 0 point for the HW1.
- Click
Create repository
. - In terminal, go to the directory that you will be working in (e.g.,
~/workspace/swpp-hw1-USERNAME
or~/swpp-hw1-USERNAME
) - Type in and run the following commands. You can also find the same instructions on Github after taking step 4.
echo "# asdf" > README.md
git init
git add README.md
git commit -m "first commit"
git branch -M main
git remote add origin https://github.com/USERNAME/swpp-hw1-USERNAME
git push -u origin main
Alternatively you can start by copying skeleton codes :)
cp -r ${swppfall2022-home}/hw1 ~/workspace/swpp-hw1-USERNAME
git init
git add .
git commit -m "first commit"
git branch -M main
git remote add origin https://github.com/USERNAME/swpp-hw1-USERNAME
git push -u origin main
- Under
Settings
thenManage access
tab, invite TAs as your collaborator:swpp-tas
. - You're all set! After finishing your homework, push your contents to your repository on time!