This is the main repository for the Lowe Institute's Automation of data processing and plot creation. The goal is to be able to pull data automatically, process it, and create plots at the beginning of each report-writing cycle. This will make our lives a lot easier and free up time to do fun stuff!
- Late model Python 3
- Late model R and RStudio
- Git
If you are unfamiliar with either Python or R (or both), you are welcome to learn it (resources are listed at the bottom of the README). However, if you do not want to learn them, that's okay as well. You are still required to learn how to use Git and GitHub, since all files (including non-code files) will be placed here. You will need the following languages for the following tasks:
- Python: Pulling data from APIs, preliminary processing of data (getting rid of stuff we obviously don't want), preliminary data cleaning, creating CSV files that can be pulled and processed in R.
- R: Plot-specific data processing (i.e., rearranging a dataframe to get what you want), plotting in
ggplot2
.
If you don't learn how to use one of these languages, you won't be able to contribute for those particular tasks. But, there will still be plenty more you can do!
We define some basic conventions to streamline our workflow and make things easier to organize. If you take issue with any of these, please let us know! We're more than willing to change things that don't work for you all :)
The main branch for this repository is called master
. This is where we will do most of our work. However, if you are making changes to an existing code file, then create a new branch for it. More details on branches blow.
We will be using GitHub Issues (the issues tab on the repository) to track work and give assignments. If the assignment requires you to heavily modify existing code, make a new branch for working on this issue. If not, you can push directly to the master
branch.
We will keep track of branches with a systematic way of titling them. They should be titled as
<IssueNumber>-<Short-Description>
You can create a branch using
git checkout -b branchName
For example, if my assignment were to fix this README file, and this assignment was given to me as ticket number 5, I would create a new branch since I am editing an existing file. To do this, I would type
git checkout -b 5-readme-fix
When you are done working on this branch, create a pull request and we will review it before merging it to the master
branch.
Remember, it is only required that you create a branch when you are instructed to make major changes to an existing file, like README.md
. If not, you are still free to make your own branch, but it is not required and you can just push to master
. If all you are doing is adding stuff (like additional helper functions), and not modifying what already exists, there is no need to create a new branch. Just to minimize merge conflicts :)
If you're working with someone concurrently on the same script, it's sometimes helpful to work together on it. Luckily, Visual Studio Code (the text editor / IDE you should use) has a Live Share extension pack. TO install it, search for extensions (there should be an icon on the left-had-side bar). Search "liveshare" and install the Live Share Extension Pack.
Here we will define coding conventions to ensure proper documentation of our code. By following these conventions, it helps others (and maybe even your future self) understand the code you've written.
These conventions are highly detailed and specific. If some of these conventions do not work for the code you are writing feel free to use conventions that suit your code and coding style. However, we do ask that your code to be readable and consistent.
Function names must be descriptive, i.e. names such as "function" or "do_something" can NOT be used. For example, if you want to write a function that gets a city name then your function can be named get_city_name()
.
To guarantee that we pass in the correct argument type to a function, function parameters must include type hinting. With this method, if an argument with the wrong type is passed in, a type error will be given.
An example of a function definition is
def get_city_name(geoid: str):
You can also use default arguments for your function parameters to guarantee that an argument is passed into the function even if no argument is given during the function call. For example
def geoid_from_city(city: str = "Rancho Mirage, CA"):
will still run even if you call the function geoid_from_city()
.
Docstrings are a convenient way of documenting the functions we write. Docstrings inclue a description of your function, the arguments it needs, and the output of the function. Docstrings appear right after the function definition and you can create one by using triple quotes """ [docstring here] """
or ''' [docstring here] '''
.
Here is an example docstring from the acs_detail_table.py
file.
def get_population_estimate(year: str, city: str):
"""
Pulls the population estimate data from ACS for a city and year.
NOTE: Returns "Connection refused by the server.." if no connection to server
Argument(s)
-----------------------
year: year of data needed to pull in string format. Example:
"2021"
city: city id in string format. Example is Rancho Mirage:
"59500"
Output
-----------------------
Population estimate in List of List of strings format. Example:
[['NAME', 'B01001_001E', 'state', 'place'], ['Palm Springs city, California', '47897', '06', '55254']]
"""
For consistency, all arguments have a one tab indent and its description should align with the description of the other arguments. For the output, output descriptions are NOT indented but the example should have a one tab indent. Output examples should be given. However, if your output is too long or too difficult to type, ex. a dataframe, it can be omitted.
If there is something important about the function that you want to convey to readers, use NOTE:
to highlight the important information.
We've provided a sample docstring and function definition for you to copy paste and fill in below:
def func(param_1: type, param_2: type):
"""
[Function Description]
(Optional for any important information) NOTE:
Argument(s)
-----------------------
param_1: [param_1 description] in [parameter type] format. Example (Optional Example Description):
[param_1 Example]
(Optional) NOTE:
param_2: [param_2 description] in [parameter type] format. Example (Optional Example Description):
[param_2 Example]
(Optional) NOTE:
Output
-----------------------
[Output] in [output type] format. Example:
[Example]
(Optional) NOTE:
"""
Besides docstrings, comments should be written throughout your functions for others to understand your code. Similar to a step-by-step guide you created a function. Addtionally, comments should be written on any code that can be ambiguous to readers. A good way to know when to comment is if you had to think for a while to write the code. However, you do not need to comment everything. For example, variable initialization does not need to be commented since it is pretty straightforward. You can add a comment by using the hash sign #
.
If you have code that does not use functions, make sure you write a comment at the top of the code block of what it does. For example:
#--------
# Script that cleans the geocode csv file to split the city geoid and the state geoid
# And makes it to a new csv file.
#--------
Furthermore, if you are unable to finish the code you are working on, make sure to write a #TODO:
comment in the area you need to finish with a description of what needs to be done. This ensures that if other people work on the code, they know what still needs to be done with the code. If there are overall todos, you can leave the code block
# ---------------------------
# THINGS TO DO:
# ---------------------------
# TODO: [description]
at the top of the file with the descriptions of the things you need to do.
- Go to http://www.census.gov/developers/
- Click on REQUEST A KEY on the left side of the page
- Fill out the pop-up window form
- You will get an email with your key code in the message and a link to register it. Enter the organization name as Lowe Institute of Political Economy.
Now we need to use the API key in our Python code in order to make certain requests. While you could simply paste in the API key into your file as a variable, we do not recommend this since this is a public repo and any code you push that explicitly mentions your API key will therefore make your API key open to the public.
Let's make our code more secure by making a file named ".env" in this directory once you've cloned this repo onto your local machine. Because we've added ".env" into our ".gitignore" file, we can each maintain separate ".env" files holding our own API keys. This is useful for us because if we decide to use other API's later, we can use this same codebase and add any additional API keys to this same file.
In the ".env" file you just created, you only need to add one line:
API_KEY_ACS='<your-api-key-goes-here>'
You don't have to name it "API_KEY_ACS" exactly but if you change that, just make sure you change the reference in the code.
We can use this file later to add other API keys that we want to keep private.
In order to use our keys in our scripts, pip install the python-dotenv package then import load_dotenv()
from the dotenv
package. This package will be used in tandem with the os
package to get your environment variable. Note that the dotenv package must be installed with
pip3 install python-dotenv
or
pip install python-dotenv
Then, at the beginning of your script, type in
from dotenv import load_dotenv
load_dotenv()
import os
API_KEY = os.getenv("API_KEY_ACS")
If you named your key something else in the .env file just make sure you use that string in the os.getenv(<keyname>)
function. This method is preferred because there's no need to ever directly copy your API key into your code.
American Community Survey API documentation for 5-year estimates: https://www.census.gov/data/developers/data-sets/acs-5year.html
Geographic codes for ACS: https://api.census.gov/data/2019/acs/acs5/subject/examples.html
Check out the notes.md
file for some more details and useful links!
Whenever you create a new file, please add a description of what it does here!
-
save_webpage.py: This script allows you to save the html file for a website. It is currently configured to save the ACS variable list for 2019, but this can be changed to whatever you need at the time. This is useful for getting html files for screenscraping without having to use
requests
every time. -
acs_var_scrape.py: This is the file that parses the ACS variables for 2019 into a
.json
file calledacs_vars_2019.json
for us to look up later. Note that you have to runsave_webpage.py
in order for this to work, since you need the html file of the site for it to work. If you want to rewrite this to work withrequests
, then go for it :) -
acs_script.py: This is the script that scrapes data from ACS using their API, and cross references the
acs_vars_2019.json
file to rename the columns to something human-readable. -
acs_vars_2019.json: This file is essentially a massive dictionary. The keys are series IDs from ACS, and the values are dictionaries themselves, which contain the keys
Concept
andLabel
. The values corresponding to these two are enough to create an English name for any columns we want to pull. -
notes.md: This file contains useful information on different APIs we are using
If you are interested in learning how to use the prerequisite software, we provide some resources here.
Learn Git in 15 minutes: https://www.youtube.com/watch?v=USjZcfj8yxE This tutorial will help you understand what Git is, as well as how to install it
Basics of using Git and GitHub together: https://www.youtube.com/watch?v=evgZPMWqpHc&list=PLriKzYyLb28nCh3jJLROcYBvj7ZO0l-3G The first 5 videos on this playlist are probably all that you will need. There will be some redundant information. If you already know what Git is but don't know how to use it, then just start here!
Corey Schafer has a great introduction to Python on this YouTube playlist. For our purposes, you'll need to know the contents of about the first 9 videos (basic data types and control flow that we'll work with a lot, as well as functions and modules). We'll teach you the basics of how to use the packages we'll be working with in workshops and all (I want to commit to recording a mini-lecture series, but no promises). Here's the link to the playlist: https://www.youtube.com/watch?v=YYXdXT2l-Gg&list=PL-osiE80TeTskrapNbzXhwoFUiLCjGgY7
CodeCademy has a free course: https://www.codecademy.com/learn/learn-r
The first 4 modules will be essential to what we do, and going through the rest may be useful for learning some of the data processing. But you can go through that stuff as you need it.