Welcome to chrisrichardmiles.m5
A submodule built for the 77th place solution for the M5-Accuracy competition on Kaggle.
This package is built with nbdev, so the source code, testing, and documentation are all built in jupyter notebooks. For full solution details, read through documentation, or the 5 notebooks which are used to create the documentation.
Create submission from scratch from the command line
Requirements: 20 GB of RAM and pip installed
1. Install package
pip install chrisrichardmiles
2. Create data folders, download data, and unzip files
If you have your kaggle api info in root/.kaggle/kaggle.json then run:
crm_download_kaggle_data --comp_name m5-forecasting-accuracy
Otherwise, you must run:
crm_mkdirs_data
and manully download the data zipfile from kaggle and upload it into the data/raw folder.
cd data/raw
unzip *
cd ../..
3. Detect out of stock days for products and change sales to NaN.
crm_m5_make_oos_data
4. Create features for training
crm_m5_fe
5. Train models and create submission
crm_m5_lgb_daily
Overview of solution
- Given Walmart sales data, tasked with delivering accurate hierarchical sales forecasts over 28 days, preventing potential losses in the millions resulting from overstocking and understocking, my solution placed 77th of 5,558 teams (top 1.4%) with a simple, explainable tree-based model, using LightGBM. It is also a pip installable software package, with testing, documentation, and continuous integration, found at my github.
- Given a novel scoring metric, I used calculus and numpy to implement a custom objective function utilizable by LightGBM models. After discovering non-convexity, I created another custom function, utilizing insights about the data and metric, which outperformed all other publicly known methods.
- With stockout-days non-differentiable from zero-sales-days, I used probability to detect stockout days by assuming sales were Poisson distributed, finding streaks of zeroes that were unreasonably long, given the average daily sales of a product, leading to enhanced data, more effective features, and superior models.
- With limited memory resources, I utilized principal component analysis from scikit-learn to reduce dimensionality of features, allowing more total features and better model performance.
- Needing feature engineering that did not exist in pandas, I used numpy to calculate novel rolling window features, also giving a 20x speed up, useful to speed up data pipelines.
Some lessons I learned
RAM issues
- Computing rolling window statistics can be very expensive, but can be ok if we do processing in smaller sections like we do with
n_splits
in therolling_window
statistics calculations. - Datatype matter: float32 and float16 datatypes can save a lot of RAM, but be careful for float16, which might cause accuracy problems or completely break a process, like when trying to use
StandardScaler
with float16 datatypes resulted in all zeros. - Be careful with Pandas DataFrames. I had a lot of problems where I was unmindfully creating large copies of data, such as a saving function, looking something like
df[cols].to_csv(....)
which was making an entirely new dataframe in memory before saving. This was remedied by using theusecols
param indf.to_csv
- Be careful with DataFrame names. It seems that it is better to keep the same name of a DataFrame when doing things like concating an existing df with new data. Its like pandas will use memory more intelligently. So this:
df2 = pd.concat([df1, pd.read_csv('data.csv')]; del df2
should be replaced withdf1 = pd.concat([df1, pd.read_csv('data.csv')]
. It seems like it should work the same, but my experience with RAM crashes seems to indicate the second method is much better.
Organization is important
The first iteration of this project was code spread accross hundreds of kaggle notebooks. I ran experiments by running a notebook and just looking at the result. It always felt ok while I was doing it, and it worked out ok, but I didn’t have the results saved in a proper manner. Now I use neptune.ai to track experiments.