A few weeks ago, Chip compared the state of real-time machine learning in China and US: While many Chinese companies have adopted real-time ML, US companies are still assessing its value. She also wrote about ML going real-time here.
Talking to Internet companies in China gives me the impression that their MLOps infra is away ahead.
— Chip Huyen (@chipro) December 10, 2020
Many US Internet co. are still wondering if there's value in real-time ML. Chinese counterparts are already doing real-time inference + online training.
Are my samples biased?
This post continues the thread and shares how real-time ML looks in practice. Drawing from my experience and industry papers/blogs, weāll discuss real-time recommendations.
Note: This discussion assumes basic knowledge of recommendation systems, such as the difference between item-to-item and user-to-item, and the candidate generation & ranking paradigm. Fret not if those terms are unfamiliar. Weāll have two primers to help you get up to speed. Click on the š below to start with the first.
If you understand the difference between collaboration vs. content-based recommendations, and item-to-item vs. user-to-item, feel free to skip this section.
Collaboration-based recommendations are based on user behavior. Assume I like movie X
and dislike movie Y
. To recommend movies to me, we first find users similar to me (i.e., like X
, dislike Y
). Then, from those users, what are movies they liked, but Iāve not watched? Those movies are then recommended to me. With user behavioral data, users can ācollaborateā to create recommendations for each other. Collaborative filtering is probably the most well-known approach.
Content-based recommendations are based on item metadata. Given the movies Iāve watched (and enjoyed), content-based recommenders suggest movies of similar genre, time period, director, etc. Relative to collaboration-based recommenders, content-based recommenders tend to be more effective when the movie is new and we donāt have enough user behavioral data about it yet (i.e., cold-start problem)
In item-to-item (i2i) recommendations, given an item, we recommend other items. Hereās an example of i2i recommendations on IMDB, under the āMore Like Thisā widget. This works well in scenarios where the focus is the item (e.g., item detail page).
In user-to-item (u2i), given a user, we recommend items. We see this on the home page of Netflix, Amazon, Taobao, sometimes with the name of āRecommended For Youā. Our social media feeds (e.g., Twitter, LinkedIn, Facebook) are u2i recommendations too. In such scenarios, the user (and their historical preferences) is the focus.
Taken together, i2i and u2i recommendations provide coverage for the bulk of user traffic via detail pages and home pages.
Before we get too excited, let me first say that most use cases wonāt need real-time recommendations; batch recommendations are good enough.
Relative to real-time recommendations, batch recommendations are computationally cheaper. They are usually generated once a day and benefit from batch processingās economies of scale. The recommendations are then loaded into a key-value store (e.g., Redis, DynamoDB) and served via a key-value lookup.
Batch recommendations are also simpler ops-wise. By caching pre-computed recommendations (in our key-value store), we decouple computation from serving. Thus, even if the compute step fails, thereās no customer-facing impact; we continue to serve the previous batch of (slightly stale) recommendations. The cache provides a buffer and ensures almost 100% uptime, reducing ops burden on the team (e.g., on-call, complaints).
In contrast, real-time recommendations usually require more computation. For example, we might aggregate streamed events (e.g., click, like, purchase) and generate new recommendations on-demand, based on user interactions. (In comparison, batch recommendations only compute a single set daily.) Furthermore, such computation is done individually and does not benefit from economies of scale. (Nonetheless, we save on not generating recommendations for customers who donāt visit our app).
Operating real-time recommendations in production is also far tricker. Instead of using ephemeral Spark clusters (for compute) and a DynamoDB (for serving), weāll need low-latency high-throughput APIs with 24/7 uptime. In the strictest scenario, we wonāt have our key-value store as a buffer. The line between compute and serving disappears. Ops burden will increase (read: weāll be paged in the middle of the night for an outage in another timezone šØ).
Why real-time recommendations then? Theyāre useful when the customer journey is mission-centric and depends on the context. Such missions are often time-sensitive. Real-time demand fades quickly; demand could be met (on a competitor site) or the user might lose interest. Weāll examine this in two examples: shopping and watching a movie.
Shopping is a mission-centric activity. Though a customer might predominately purchase a category of products, their shopping behavior is often punctuated with tangential missions. For example, if I usually shop for clothes, my u2i recommendations will mostly be fashion recommendations. However, if I need a new wide-screen monitor and start browsing for one, my u2i recommendations should update ASAP to help me quickly fulfill my mission (lest I go to a competing app).
In this scenario, batch recommendations donāt react fast enough. And even when the recommendations are updated, due to the data imbalance, such mission-related needs are not met (More in Section 2.1 of this Netflix paper).
The movies we watch depend on context (though our long-term preferences are fairly stable). For example, we might watch different movies depending on whether weāre alone, with friends, with a romantic interest, or with children. It might also depend on our mood, as well as the time of day. Similar to shopping, daily batch recommendations face the same challenges here.
Other than the examples above, real-time recommendations are also handy in:
Real-time recommendations are also useful when the majority of our customers are new (i.e., cold-start). This happens when weāre in the customer acquisition stage, such as when weāve just launched a new product or entered a new market (e.g., e-commerce in Southeast Asia in 2013 - 2015).
Imagine youāve just downloaded an e-commerce app. Since weāre uncertain of your gender, the home page will have a mix of categories catering to each gender, from dresses to menās shirts, from GPUs to makeup. If you click on a dress, we can immediately build a persona (that youāre female) and personalize your shopping experience. In this case, the u2i recommendations on your home page will tilt towards female products.
Given that 1 in 4 users abandon mobile apps after only one use, quickly responding to customer needsāfrom the very first touchpointāhelps with acquisition and retention.
Did you miss the first primer (collaborative vs. content-based, item-to-item vs. user-to-item)? Click on the š here.
Most modern recommenders have two key components: candidate generation and ranking.
Candidate generation is a fastābut coarseāapproach to get (hundreds of) item candidates from millions of items. We trade off precision for efficiency to reduce the search space (e.g., from 100 million to 1,000 candidates, a 99.999% reduction). This is usually done via metadata-based filters (e.g., category, brand) or k-nearest neighbors.
Ranking is a slowerābut more preciseāapproach to sort and select top recommendation candidates. We have leeway to include features that might not have been feasible in the candidate generation step. Such features include user persona (e.g., demographics, price propensity), item metadata (e.g., attributes, engagement statistics), cross features (e.g., interaction between each feature pair), and media embeddings.
Ranking can be framed as either a classification or learning to rank problem. As a classification problem, we can score candidates based on probability of click or purchase. Logistic regression with crossed features is simple to implement and a difficult baseline to beat. Decision trees are also commonly used. As a learning to rank problem, commonly used algorithms include LambdaMart, XGBoost, and LightGBM. Neural networks are also gaining adoption, thanks to gains in model efficiency via distillation, pruning, quantization, etc.
This general reference sheds light on recommendation systems from a Chinese perspective. Itās organized based on the paradigm discussed, with sections for candidate generation (āMatchā) and ranking (āRankā). It also discusses two other components: profile (building user preferences) and post-processing (e.g., excluding previously purchased goods, item-deduplication across recommendation sets).
To see how recommendations can be incrementally updated, weāll discuss two algorithms: the humble collaborative filtering (CF) and Alibabaās Swing algorithm that improves on CF.
Collaborative filtering (for i2i recommendations) is implemented via the three formulas below. Letās try to understand what they do and the intuition behind them.
\[\omega_{user}=\frac{1}{\left(\log _{2}(3+cnt)\right)^{2}}\] \[\omega_{\text {item }}=\sum_{\text {user }} \omega_{\text {user }}\] \[\operatorname{score}(i, j)=\sum_{u \operatorname{ser}} \frac{\omega_{user}}{1+\left(\omega_{i} * \omega_{j}\right)^{0.5}}\]User weight: A userās weight is inversely proportionate to their number of item interactions (e.g., click, like, wish-listed). Intuition: Users who browse a lot tend to be less discriminative and thus have more noisy behavioral data.
Item weight: The more users interact with an item, the higher its weight.
Item similarity score: To calculate similarity between a pair of items, we sum the weight of users who interacted with both items, and divide it by the product of item weights. Intuition: The greater the overlap in user-interactions, the higher the item similarity (similar to collaborative filtering via matrix factorization or alternating least squares).
The Swing algorithm differs in that the weight (of a pair of users) depends on the number of items they both interacted with (in contrast, CF keeps user weights constant). The greater the proportion of items both users interacted with (i.e., intersection), the higher the user weights.
\[\omega_{u 1}=\frac{1}{(cnt+5)^{0.35}}\] \[\omega_{\text {pair }}=\omega_{u 1} * \omega_{u 2}\] \[\operatorname{score}(i, j)=\sum_{\text {pair }} \frac{\omega_{\text {pair }}}{1+\text { intersection }}\]User weight: Similar to collaborative filtering.
User-pair weight: Product of user weights. For n
users, we have n^2
user-pair weights.
Item similarity score: Similar to collaborative filtering. The key difference is that the denominator only considers the intersection of products both users have interacted with. (In contrast, CF considers all products any user has interacted with).
To make things clearer, hereās how we would calculate item similarity in code:
for i in xrange(0, len(u2items)):
wi = math.pow(len(u2items[i]) + 5, -0.35)
for j in xrange(i + 1, len(u2items)):
intersection = u2items[i] & u2items[j]
wj = wi * math.pow(len(u2items[j]) + 5, -0.35)
for product_id in intersection:
i2i[product_id] = i2i.get(product_id, 0.0) + wj / (1 + len(intersection))
# u2items = array of users and their items
# u2items[i] = items user i clicked on
# u2items[j] = items user j clicked on
# intersection = items both user i and user j clicked on
# wj = product-pair score
# i2i is incrementally updated as we loop through users (we won't use a loop in production)
Intuitively, if two users with very different tastes (and product interactions) click on the same product-pair, this suggests a strong relationship between the product-pair. Conversely, if two users have similar tastes (and many product pairs), the product-pair relationship is weaker. This reduces the noise from herd behavior (e.g., Harry Potter problem) and identifies more meaningful i2i recommendations.
From these simple equations, we see how recommendations can be incrementally updated in real-time. Instead of batch matrix multiplication, user and item weights are updated with each customer interaction, perhaps in a key-value store. The weights can then be combinedāvia summationāto update i2i recommendations.
These equations can also be adapted to calculate user affinity towards category, brand, seller, price point, etc. Such affinities are then be used to weigh recommendation candidates. For example, if a user has historical preference for a specific brand, we give items with that brand a higher weight.
Hereās an example of Swing and category affinity used in real-time recommendations at Alibaba. This implementation is for the āRecommended For Youā widget on the home page of 1688, a B2B e-commerce. The widget receives clicks from 72% of users.
A quick rundown of the infra components:
Hereās how recommendations are updated and served. As the user browses on the app:
Tencentās approach is similar. The crux is to break up item-based CF into two aggregates: item count and pair count. Item-to-item similarity is then computed using these aggregates. Thus, item and pair aggregates can be incrementally updated and combined to generate recommendations in real-time. Their system is implemented on Apache Storm and used in Tencent News, Tencent Videos, YiXun (e-commerce), and QQ ads (messaging).
\[\text { itemCount }\left(i_{p}\right)=\sum r_{u, p}\] \[\operatorname{pairCount}\left(i_{p}, i_{q}\right)=\sum_{u \in U} \operatorname{co-rating}\left(i_{p}, i_{q}\right)\] \[\operatorname{sim}\left(i_{p}, i_{q}\right)=\frac{\operatorname{pairCount}\left(i_{p}, i_{q}\right)}{\sqrt{\operatorname{itemCount}\left(i_{p}\right)} \sqrt{\operatorname{itemCount}\left(i_{q}\right)}}\]A/B tests showed that real-time recommendations led to 6 - 8% increase in CTR for news recommendations, and 6 - 18% increase in CTR for e-commerce recommendations.
The paper also shares tricks used to solve several challenges (listed below). Highly recommended read.
Next, we look west and see how US companies have implemented real-time recommendations.
First, hereās YouTubeās video recommendation system. We should be familiar with the overall design paradigm by now.
For candidate generation, approximate kNN is applied on video and search embeddings to select hundreds of videos from millions (probably billions by now).
For ranking, a deep neural network is used to score each video and select the best couple dozen. Additional features (e.g., query and video statistics, previous user interaction) help improve precision. The neural network is trained via weighted logistic regression, where positive labels are weighted by video watch time (negative labels are unweighted). This reflects the business objective of increasing expected watch time per video impression.
Instagramās approach is similar. In the candidate generation stage, account embeddings are used to identify accounts similar to those the user has previously interacted with. From these account candidates, they sample 500 media candidates (e.g., photos, stories).
Then, these media candidates go through a three-pass ranking process which uses a combination of techniques to shrink neural network models:
For our final example, we look at how Netflix re-ranks recommendations in real-time. On most Netflix pages, weāll see several recommendation rows. Each recommendation row has dozens of movies. The goal is to reorder the recommendation rows (i.e., North-South ordering) as well as the movies within each recommendation row (i.e., East-West order).
Horizontal scrolls (on the recommendation rows) are used as user-interaction signals. They showed that video play probability is positively correlated with the number of horizontal scrolls.
In contrast to previous examples, re-ranking is carried out on the client device, avoiding a roundtrip to the server. (However, this also prevents candidate generation.) Given that viewing devices can be low-powered (e.g., TVs), computation needs to be lightweight. Thus, the algorithms used are simpleāyet effectiveāprobabilistic models. This allows recommendation rows below the fold (i.e., not on screen) to be updated on-the-fly as users perform horizontal scrolling above the fold.
Thus, we see that real-time recommendations are increasingly common, both in China and US. However, are they only available to big tech? No. Hereās how to cheaply build an MVP.
After the examples above, youāre might think real-time recommenders require specialized infra, deep learning models, and all that jazz. I hope to convince you otherwise.
Weāll briefly go through how to design and build an MVP, focusing on whatās commonly viewed as the bottleneck of real-time recommenders: compute and serving. In contrast, training is relatively easier and widely discussed.
To begin, I think itās useful to approach DS/ML systems in three broad strokes:
When defining requirements, we should start from the customer. How will real-time recommendations improve the customer experience, and in turn benefit the business? What business metrics are important? Goals and success metrics will vary based on the business and use case, and will not be defined here.
For our MVP, perhaps more important than requirements are constraints (i.e., how not to solve the problem). Here are some constraints for our real-time recommender:
We should also consider other aspects such as availability (aka redundancy), security, privacy, ethics, etc. Nonetheless, for our MVP, the first three constraints are technical and business showstoppers which weāll focus on.
To train item embeddings, we adopt the simple but effective word2vec approach, specifically, the skip-gram model. (This is also used by Instagram, Twitter, and Alibaba.) Iāve previously written about how to create embeddings via word2vec and DeepWalk and wonāt go into details here.
To generate candidates, we apply k-nearest neigbours (Ć la YouTubeās implementation). However, exact kNN is slow and we donāt really need the precision at this stage. Thus, weāll use approximate nearest neighbours (ANN) instead.
There are several open-sourced ANN implementations, such as Facebookās FAISS, Googleās ScANN, and Hierarchical Navigable Small Word Graphs (hnswlib). Weāll benchmark them on the recall/latency trade-off. To mimic production conditions, each query consists of a single embedding (i.e., batch size = 1). The graph below shows ScaNN outperforming the other two implementations. (FAISS, in particular, is optimized for batch queries.)
Installing ScaNN on a Mac is tricky and took a few frustrating hours to figure out. If you run into problems, you might find this step-by-step helpful.
In addition, here are some GitHub issues that might help:
To rank candidates, we start with a single-layer neural network (read: logistic regression) with cross featuresāsimple, yet difficult to beat. Cross features are created by combining all possible pairs of features to capture their interactions. While this blows up the feature space, itās not a problem for logistic regression.
Beyond machine learning metrics (i.e., recall@k, NDCG, AUC), itās probably more important to consider business metrics (circling back to requirements). Different goals call for different metrics:
From experience, business stakeholders will usually have conflicting goals. Marketing wants to make the first sale (regardless of item price), customer experience wants to takedown poor quality products (even if they sell well), and commercial wants to maximize profit (by selling higher-priced items). Getting everyone to agree on key metrics and guardrails can be more difficult than improving our models.
Will our MVP require specialized infrastructure? Not necessarily. A cost-effective approach is to use EC2 instances that can scale horizontally with a load balancer in front. To further simplify things, we can just use AWS SageMaker.
To assess latency and throughput, we have various options including serverless-artillery
(AWS guide) and locust
. Running several load tests showed that a SageMaker endpoint backed by 30 m5.xlarge instances was able to serve 1,200 queries per second without breaking a sweat. At this throughput, median latency was 25ms while the 99th percentile was 65ms. There were zero errors.
With regard to cost, m5.xlarge (16 gb RAM, 4 CPUs) instances in US West (Oregon) have an hourly rate of $0.269. Running 30 instances for 28 days works out to approximately 5.5k, well within our 10k budget. Using reserved instances and/or auto-scaling can help with lowering cost.
Our simple MVP deliberately excludes several considerations. For example, to let other services query our item embeddings, we might want to expose them as a separate service. This will require additional infra (e.g., DynamoDB) which will increase cost and ops burden. The additional service call (for item embeddings) will also add latency (10-30ms) though it can be minimized via a good network setup. Also, how can we expose our candidate generation and ranking services via generic APIs, so other users can mix-and-match as required? Weāll want to consider these in the long-term roadmap.
I hope this improves your understanding of real-time machine learning in the context of recommendation systems, and demonstrates that itās not an insurmountable challenge. Libraries (e.g., ScaNN) and managed services (e.g., AWS SageMaker) abstract away much of the nitty-gritty such as optimization, health checks, auto-scaling, recovery, etc. Building on them allows for effective, low-cost, real-time ML.
Designing and implementing machine learning systems can be difficult. If youāre building one and would like another pair of eyes or feedback, reach out via @eugeneyan or email.
How does real-time ML look in practice?
— Eugene Yan (@eugeneyan) January 12, 2021
We discuss real-time recommenders, drawing on my experience & industry papers:
ā¢ When does it make sense? When does it not?
ā¢ How have China & US companies implemented them?
ā¢ How do we design & build an MVP? https://t.co/hSfRr3q2ie
applied-ml
repositoryThanks to Yang Xinyi, Karl Higley, Will Larson, and Tushar Chandra for reading drafts of this. Thanks to Hammad Khan for a correction.
If you found this useful, please cite this write-up as:
Yan, Ziyou. (Jan 2021). Real-time Machine Learning For Recommendations. eugeneyan.com. https://eugeneyan.com/writing/real-time-recommendations/.
or
@article{yan2021realttime,
title = {Real-time Machine Learning For Recommendations},
author = {Yan, Ziyou},
journal = {eugeneyan.com},
year = {2021},
month = {Jan},
url = {https://eugeneyan.com/writing/real-time-recommendations/}
}
Join 9,800+ readers getting updates on machine learning, RecSys, LLMs, and engineering.