data:image/s3,"s3://crabby-images/82110/8211015e36386902d2a70213ce8527e3df3a0338" alt=""
Analytics has taken over the sports industry, and data is crucial to fantasy sports leagues, including fantasy basketball. While most fantasy sports apps provide you with a wide variety of data and analytics, there are limitations to them, like not being able to customize how you view and interact with the data.
That’s why we found a way to pull the data ourselves so that we can analyze it exactly how we need to. We found an API on RapidAPI that provides NBA player stats and metadata, and we used that data to improve our fantasy basketball team. This article goes over the first step in the process: connecting to the API and setting up our data.
API Setup
The API that we found is this API-NBA one that is hosted on RapidAPI, which is a marketplace that hosts a wide variety of APIs to get data on a large range of topics. You can find this API by navigating to rapidapi.com/hub, searching for “NBA”, and selecting the API-NBA
one in the search results.
data:image/s3,"s3://crabby-images/a3def/a3defcb136a2027042af21d2fafa70f44594d1be" alt=""
This brings you to a page where you can navigate through the different endpoints, view parameters and sample results for each, and get sample code snippets for the different endpoints. The code snippets come in a variety of different programming languages and frameworks, and in this article we’ll be using the Python requests
library.
data:image/s3,"s3://crabby-images/c912a/c912a7cb5d8709a46bebcf7d8a4b3ece0dff1e59" alt=""
If you aren’t signed in yet, you’ll notice it says “Sign Up for Key” where there should be an API key. To get your API key, you’ll need to create a free RapidAPI account and select one of the subscription options. The APIs on RapidAPI are all subscription based, but many of them offer free tiers.
data:image/s3,"s3://crabby-images/78772/787726e049c0735f20cb08b919c064fb331df93c" alt=""
While the API-NBA one does offer a free tier with a limit of 100 requests per day, this is a pretty low limit, so we’d recommend subscribing to either the Pro or Ultra options so that you don’t have to worry about going over.
Once you’re signed in and have selected a subscription option, you’ll your own personal API key in the headers
dictionary in the code snippet. However, since this API key is basically your password for accessing the API, we recommend storing it in a more secure way than written out in your code. One alternative is saving it as an environment variable on your computer.
You can set this up in Windows by searching for “environment” in the windows search bar, hitting the Environment Variables
button, clicking the New
button under the User variables
section, and pasting it into the Variable value
field with NBA_API_KEY
in the Variable name
field.
data:image/s3,"s3://crabby-images/7ff5e/7ff5ee810ac0f71f697b0571e02bdf4685a24cff" alt=""
Python can read that into a variable using the os
package and plug it into our headers dictionary, which we’ll use during the API calls to authenticate access.
data:image/s3,"s3://crabby-images/b836d/b836d8de887ec0f73d90708ec32194d14d4de7f0" alt=""
call_get_endpoint
One thing you’ll notice from the individual code snippets is that the code is very similar across the different endpoints, with just a few differences in the parameters and URL. We can simplify our code with a function that standardizes this consistent code and uses function arguments to specify each endpoint.
data:image/s3,"s3://crabby-images/53b67/53b67fbcf4a8c977828bc539c0b199c6f24314fb" alt=""
With everything setup for the API connection, we’re ready to start calling it and transforming the data.
Dimension Tables
Rather than storing the data in one big flat format, the API structures the data like a relational database with fact and dimension tables. It defines IDs for each game, team, and player, and the full data model looks something like this:
data:image/s3,"s3://crabby-images/2ecdd/2ecdd9644a4b1243a4ae1f4feb6d03cd5be4a5ee" alt=""
We’ll start by recreating those dimension tables using the API and saving them as CSV files.
Teams
First, we’ll have a Teams dimension table that lists out each of the NBA teams. We can get this from the games
endpoint, which doesn’t take any parameters, just pulls all of the current NBA teams with some basic info on each.
Our Teams dimension table will include a TEAM_ID
attribute to identify each NBA team along with other info like the team name, abbreviation, mascot, and city.
data:image/s3,"s3://crabby-images/8d324/8d3240b9af71baa9da7d0da69b757c2e034e1481" alt=""
data:image/s3,"s3://crabby-images/86a14/86a14e799985c7d79a0636102702b4eb4d17705c" alt=""
Players
Next, we’ll have a Players dimension table that lists out the individual NBA players. We can get this from the players
endpoint, which takes a specific TEAM_ID
as a parameter along with the desired season. We’ll use a for loop to iterate through each of the TEAM_ID
values from our Teams dimension table and add everything to one big dataframe of all the players.
Our Players dimension table will include a PLAYER_ID
attribute to identify each player along with other info like the team they belong to, first and last name, position, and jersey number.
data:image/s3,"s3://crabby-images/6b61f/6b61f3f08a3eef117d717fefbd4ca98e6b11ff9d" alt=""
data:image/s3,"s3://crabby-images/f823c/f823c0ee4feb58d689ea3c33ef9107735b41c9ba" alt=""
Games
Lastly, we’ll have a Games dimension table that lists out all of the games on the schedule. We can get this from the games
endpoint, and if we only use the current season as a parameter then it provides a list of all of the games for the whole year.
One attribute that the API doesn’t provide is the fantasy basketball week number that each game corresponds to. Our fantasy basketball league matchups are split up by week, where matchups start on Monday and end the following Sunday. Thus, if we know the date of a particular game, we can calculate the week number by the number of days since the first Monday of the league.
We defined a little function to do that math for us:
data:image/s3,"s3://crabby-images/a8c6d/a8c6d0f1919f1645b0d01668fb9a064546887c79" alt=""
We can then use that function to add the week number attribute based on the date for each game.
Our Games dimension table will include a GAME_ID
attribute to identify each game along with other info like the game date, home and away teams, and that week number that we calculated.
data:image/s3,"s3://crabby-images/43e99/43e99da6cf949120c7ec29b4c35032454d6acc43" alt=""
data:image/s3,"s3://crabby-images/830df/830dfa1ffe469892f642df82a182dbbaeadea659" alt=""
Daily Stats
With all our dimension tables setup, we can put together our actual fact table of player stats by game. There’s a statistics
endpoint that takes a specific GAME_ID
as a parameter and returns all the game stats broken out by player.
Because of how this is setup, we’ll need to make a separate API call for each individual game, which would be a lot of API calls if we tried to re-pull everything every time we update the data. Instead, we’ll set it up to only pull data for games that we haven’t pulled yet.
We’ll probably only refresh the data once a day (at the most), so we decided to group everything by date and make a separate CSV file for each date.
data:image/s3,"s3://crabby-images/22fc4/22fc492332c82606c19d078dd06f4f823efbfdb7" alt=""
That way, whenever we update the data, we just need to pull the dates that we don’t have files for yet.
get_daily_stats
To start, we’ll define a function that pulls all stats for a given date and saves them to a CSV.
data:image/s3,"s3://crabby-images/c29d7/c29d78abbfb5b4f525096921c45e5bc42259c5d9" alt=""
data:image/s3,"s3://crabby-images/c68b3/c68b3270f337824b611be21024714e201b326c98" alt=""
Pull missing dates
As the season goes on, we want to fill in the missing dates without re-pulling dates with existing files, so we can define some code to get a list of missing dates and pull stats for each with our new get_daily_stats
function.
data:image/s3,"s3://crabby-images/5771b/5771bf29e51160e8fb4d56c4119a001b4a6c3799" alt=""
Putting Everything Together
While we’ve pulled all of the stats we need, they’re scattered across the different daily files, so our last step is to combine everything together. This is also a good time to add other attributes like the fantasy points calculation and week number label.
calculate_fantasy_points
The API provides raw NBA stats, so we need to define our own function to calculate fantasy points based on our specific league scoring rules.
data:image/s3,"s3://crabby-images/04aff/04aff4ab621321138a8185d172d09a1590c72928" alt=""
Combine daily files
We’ll put all the daily stats files together with a for loop and apply our new calculate_fantasy_points
function to add the fantasy points column. We can also join in a few attributes from the Games dimension such as game date and week number for easier filtering later, and we’ll end by saving it all to our combined stats CSV file.
data:image/s3,"s3://crabby-images/faac1/faac1b5c367b275a4dd32aa8b5faa6bcbf9d32d0" alt=""
data:image/s3,"s3://crabby-images/088ab/088ab5a641daec72f95f2e0afcbc25635d38b886" alt=""
Conclusion
With the API-NBA API and a little Python code, we were able to setup our own little relational database of NBA stats. In the following posts, this database will be a foundation for improving our fantasy basketball team through more customized analysis and improved projections.
There’s a lot that we covered, so we’ll provide you with the full Jupyter notebook of Python code for you to download and try out yourself:
If you found this guide helpful, feel free to drop a comment below with any questions or feedback. Your input helps us improve future posts and inspires new ideas.
Thanks for joining us today, and we hope to see you next time!