This is the Capstone Project for Revenue Forecasting for AAVAIL by Bill Dusch. The result is a microservice dockerized into a FastAPI.
To run the API, you can either run run_app.py
locally or dockerize an image of the service using the Dockerfile contained in the service.
This project has been created using FastAPI rather than Flask.
The documentation of the api, runs run locally, is stored at http://0.0.0.0:80/api/docs
. There are five endpoints:
POST /api/vi/model/train/
: train a series of models.GET /api/v1/model/train/
: run performance metrics on a trained model.GET /api/v1/model/forecast_date/
: Forecast revenue 30 days past a chosen date given a trained model.GET /api/v1/model/forecast_range/
: Forecast revenue for a range of dates 30 days past a chosen date range given a trained model.GET /api/v1/model/monitor/
: Run monitoring performance metrics given current training data.
The ultimate model chosen was a Random Forest Regressor model (cross validated for hyperparameter selection) on a number of historical features (previous [7, 14, 28, 70] days of the [unique_invoices, unique_streams, purchases, total_views, revenue]) to predict the total revenue 30 days after the chosen target date. There are models for the top 10 countries in total revenue as well as the total revenue for all countries (all_countries
when selecting a model by country). The pretrained model is named initial_model.db
, stored as a python shelf
object (dictionary of pickled models, essentially).
- Build the docker image and run it.
> cd ai-workflow-capstone
> docker build -t capstone_project .
- Check that the image is there.
> docker image ls
- run the container.
docker run -d --name capstoneproject -p 80:80 capstone-project
- Go to
http://0.0.0.0:80/api/docs
to find the documentation and test it. The endpoints are the ones talked about before, at e.g.
- Unit tests are stored in
src/app/tests
. You can run all the tests atsrc/app/test_all.py
.
Files for the IBM AI Enterprise Workflow Capstone project.
At this point in the project, and in any data science project really, it is best to loosly organize your code as libraries and scripts. Jupyter notebooks are a convenient and powerful tool, but we have mentioned several times that they are not a good place for source code to live. If you decide to use a notebook for this part, we recommend that it is used to run functions that live within a python module.
Overall this part of the case study is meant to tell the story of the data by investigating the relationship between the data and the business opportunity.
(1) Assimilate the business scenario and articulate testable hypotheses.
Take what you have read from the stories and from what you know about the business scenario and, in your own words, carefully re-state the business opportunity. Given the stated opportunity, enumerate the testable hypotheses.
(2) State the ideal data to address the business opportunity and clarify the rationale for needing specific data.
Note that this step is carried out before you read in the data. It helps clarify exactly what your are looking for in the data and it helps provide context for what the feature matrix and targets will look like.
- Create a python script to extract relevant data from multiple data sources, automating the process of data ingestion.
From within a Python module there should be a function that reads in the data, attempts to catch common input errors and returns a feature matrix (NumPy array or Pandas DataFrame) that will subsequently be used as a starting point for EDA and modeling.
- Investigate the relationship between the relevant data, the target and the business metric.
Using the feature matrix and the tools abvailable to you through EDA spend some time to get to know the data.
- Articulate your findings using a deliverable with visualizations.
Summarize what you have learned in your investigations using visualizations.
-
The JSON files may not contain uniformly named features. Be sure to account for this in your data ingestion function.
-
Some of the invoice ids (
invoice
) have letters that can be removed to improve matching. -
One common way to ready time-series data for modeling is to aggregate the transactions by day. Getting the data into this form will help you prepare for part 2.
-
If you have not worked with time-series or time-stamped data before the following two links can be useful.
Time-series analysis is a subject area that has many varied methods and a great potential for customized solutions. We cannot cover the breadth and depth of this important area of data science in a single case study. We do however want to use this as a learning opportunity if time-series analysis is new to you. For those of you who are seasoned practitioners in this area, it may be a useful time to hone your skills or try out a more advanced technique like Gaussian processes. The reference materials for more advanced approaches to time-series analysis will occur in their own section below. If this is your first encounter with time-series data we suggest that that you begin with the supervised learning approach before trying out the other possible methods.
- State the different modeling approaches that you will compare to address the business opportunity.
- Iterate on your suite of possible models by modifying data transformations, pipeline architectures, hyperparameters and other relevant factors.
- Re-train your model on all of the data using the selected approach and prepare it for deployment.
- Articulate your findings in a summary report.
We have used TensorFlow, scikit-learn, and Spark ML as the main ways to implement models. Time-series analysis has been around a long time and there are a number of specialized packages and software to help facilitate model implementation. In the case of our business opportunity, it is required that we predict_date the next point or determine a reasonable value for next month's revenue. If we only had revenue, we could engineer features with revenue for the previous day, previous week, previous month and previous three months, for example. This provides features that machine learning models such as random forests or boosting could use to capture the underlying patterns or trends in the the data. You will likely spend some time optimizing this feature engineering task on a case-by-case basis.
Predicting the next element in a time-series is in line with the other machine learning tasks that we have encountered in this specialization. One caveat to this approach is that sometimes we wish to project further into the future. Although, it is not a specific request of management in the case of this business opportunity, you may want to consider forecasting multiple points into the future, say three months or more. To do this, you have two main categories of methods: 'recursive forecasting' and 'ensemble forecasting'.
In recursive forecasting, you will append your predictions to the feature matrix and roll forward until you get to the desired number of forecasts in the future. In the ensemble approach, you will use separate models for each point. It is possible to use a hybridization of these two ideas as well. If you wish to take your forecasting model to the next level, try to project several months into the future with one or both of these ideas.
Also, be aware that the assumptions of line regression are generally invalidated when using time-series data because of auto-correlation. The engineered features are derived mostly from revenue which often means that there is a high degree of correlation. You will get further with more sophisticated models to in combination with smartly engineered features.
- statsmodels time-series package - one of the most commonly used time-series analysis packages in Python. There are a suite of models including autoregressive models (AR), vector autoregressive models (VAR), univariate autoregressive moving average models (ARMA) and more.
- Tensorflow time series tutorial
- Prophet
- scikit-learn MultiOutputRegressor
- NumPy datetime
- Pandas time-series
- matplotlib time-series plot
- scikit-learn time-series train-test split
- Intro paper to Gaussian Processes in time-series
- Paper for using wavelets to aid time-series forecasts
- Build a draft version of an API with train, predict_date, and logfile endpoints.
- Using Docker, bundle your API, model, and unit tests.
- Using test-driven development iterate on your API in a way that anticipates scale, load, and drift.
- Create a post-production analysis script that investigates the relationship between model performance and the business metric.
- Articulate your summarized findings in a final report.
At a higher level you are being asked to:
- Ready your model for deployment
- Query your API with new data and test your monitoring tools
- Compare your results to the gold standard
To ready your model for deployment you will be required to prepare you model in a way that the Flask API can both train and predict_date. There are some differences when you compare this model to most of those we have discussed throughout this specialization. When it comes to training one solution is that the model train script simply uses all files in a given directory. This way you could set your model up to be re-trained at regular intervals with little overhead.
Prediction in the case of this model requires a little more thought. You are not simply passing a query corresponding to a row in a feature matrix, because this business opportunity requires that the API takes a country name and a date. There are many ways to accommodate these requirements. You model may simply save the forecasts for a range of dates, then the 'predict_date' function serves to looks up the specified 30 day revenue prediction. You model could also transform the target date into an appropriate input vector that is then used as input for a trained model.
You might be tempted to setup the predict_date function to work only with the latest date, which would be appropriate in some circumstances, but in this case we are building a tool to fit the specific needs of individuals. Some people in leadership at AAVAIL make projections at the end of the month and others do this on the 15th so the predict_date function needs to work for all of the end users.
In the case of this project you can safely assume that there are only a few individuals that will be active users of the model so it may not be worth the effort to optimize for speed when it comes to prediction or training. The important thing is to arrive at a working solution.
Once all of your tests pass and your model is being served via Docker you will need to query the API. One suggestion for this part is to use a script to simulate the process. You may want to start with fresh log files and then for every new day make a prediction with the consideration that you have not yet seen the rest of the future data. To may the process more realistic you could 're-train' your model say every week or nightly. At a minimum you should have predictions for each day when you are finished and you should compare them to the known values.
To monitor performance there are several plots that could be made. The time-series plot where X are the day intervals and Y is the 30 day revenue (projected and known) can be particularly useful here. Because we obtain labels for y the performance of your model can be monitored by comparing predicted and known values.