Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/woctezuma/steam-labs-recommender
Tweak the Interactive Recommender of Steam Labs.
https://github.com/woctezuma/steam-labs-recommender
discovery games-list gems hidden-gems interactive-recommender ranking reverse-engineering steam steam-labs
Last synced: about 2 months ago
JSON representation
Tweak the Interactive Recommender of Steam Labs.
- Host: GitHub
- URL: https://github.com/woctezuma/steam-labs-recommender
- Owner: woctezuma
- License: mit
- Created: 2019-07-13T06:39:28.000Z (over 5 years ago)
- Default Branch: master
- Last Pushed: 2024-05-21T03:00:21.000Z (8 months ago)
- Last Synced: 2024-05-21T04:22:05.578Z (8 months ago)
- Topics: discovery, games-list, gems, hidden-gems, interactive-recommender, ranking, reverse-engineering, steam, steam-labs
- Language: Python
- Homepage:
- Size: 610 KB
- Stars: 6
- Watchers: 2
- Forks: 0
- Open Issues: 2
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Steam Labs Recommender
[![Build status with Github Action][build-image-action]][build-action]
[![Code coverage][codecov-image]][codecov]
[![Code Quality][codacy-image]][codacy]![Interactive Recommender](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/banner.png)
This repository contains Python code to tweak the [Interactive Recommender](https://store.steampowered.com/recommender/) of [Steam Labs](https://store.steampowered.com/labs).
## Introduction
The "Interactive Recommender" is a tool provided by Valve on July 11, 2019 to suggest personalized game recommendations. On March 18, 2020, the tool was [officially released](https://steamcommunity.com/games/593110/announcements/detail/1716373422378712841) in production for the whole store.
The input is:
- the user's 50 most played games, along with [normalized](https://steamcommunity.com/groups/SteamLabs/discussions/3/1643168364649465639/) playtime, last play date, counter of the last played games,
- minimal external information about the game ([the release date](https://steamcommunity.com/games/593110/announcements/detail/1612767708821405787)).NB: It is not known whether the input is really limited to the 50 most played games.
It is [possible](https://metacouncil.com/threads/metasteam-july-2019-racing-towards-empty-wallets.1278/page-34#post-66967) that the whole game library is actually taken into account, and that the 50 most played games only appear on the Recommender webpage for cosmetic purposes.A neural network predicts the probability that a game is the next purchase of this user, based on the data available for
millions of other users.The output of the neural network is transformed with an unknown formula to give more or less importance to:
- [popularity](https://steamcommunity.com/groups/SteamLabs/discussions/3/1643168364649665178/),
- release recency.The most basic assumption for this formula would be such as:
> tweaked_output = neural_network_output + popularity_bias * popularity + release_recency_bias * release_recencyIt is likely that the release recency is simply used with a threshold to filter out older games.
In this case, the suggested formula would be instead:
> tweaked_output = neural_network_output + popularity_bias * popularityThe tweaked outputs are then scaled, so that the top recommendation is always assigned the score of 1000. The scale is stored.
> tweaked_output = score_scale * scoreSorting the scores results in a ranking of game recommendations personalized to the user's data.
## Updated information based on a patent
As of December 7, 2021, Valve filed a patent [US 11,194,879 B2][patent-html] ([PDF][patent-pdf] file) which clarifies the situation:
- popularity, "as determined by sales of the game", is a biasing factor of the score,
- release recency is used upstream: different models are trained for different values of the release recency!Moreover, popularity seems quantized into "levels", i.e. games fall in `M=5` bins, which probably correspond to:
1. very popular
2. popular
3. neutral
4. unpopular (or "niche")
5. very unpopular (or very "niche")Therefore, popularity can be one-hot encoded for each game, and popularity bias can be expressed as a M-dimensional vector, which makes it easy to tweak the output and, at the same time, impossible for an observer to estimate popularity as a continuous value in terms of sales.
**Quotes from the patent**
> In addition, the computing system may be configured to bias the score data based on different values of one or more biasing factors. The biasing factors may comprise any number of factors, such as a popularity of each game, a cost of each game, a genre of each game, or the like. In one example, the biasing factor may comprise **a popularity of each game, as determined by sales of the game, an amount of gameplay of the game, and/or the like**. In some instances, each score data (determined by an individual trained model) may be biased at different levels of popularity such that the resulting biased data will more closely match that particular level of popularity if so desired by the user. That is, the computing system may bias the score for each game title (and for each model) at **a number of “M” target levels of popularity ranging from very popular to very unpopular (or very “niche”). As the reader will appreciate, the computed scores of those game titles that are very popular will be boosted for a target level of popularity of “very high” and will be penalized for a target level of popularity of “very low”. Conversely, the computed scores of more niche game titles will be penalized for a target level of popularity of “very high” and will be boosted for a target level of popularity of “very low”.**
>
> Finally, given that the computing system has now computed, for a particular user, score data for each of “N” number of trained models, and has biased each of these scores using “M” number of popularity values, then computing system may now have generated an amount of scores for the user in an amount of “M×N”. For example, if the computing system utilizes six models and five levels of popularity, then the computing system will have generated a total of thirty scores for any particular user (given that each of the six score data generated by the model will biased five times). Of course, it is to be appreciated that the computed score data may have any other number of values and may be sampled in any other (e.g., nonlinear) manner. Further, while this example describes **inputting values of a first parameter prior to training the model (in this case, based on release date) and values of a second after the training (in this case, based on popularity)**, any other number of parameters may be applied as input to the model and/or as part of the postprocessing described with reference to the popularity data. It is also noted that while the postprocessing parameters are described as biasing factors, these biasing factor(s) represent parameter(s) that may be applied as inputs to the models or as part of postprocessing.## Data
Data personalized to my Steam profile is available in the following folders:
- [`data/`](data/),
- [`data_v2/`](data_v2/) after [the update](https://steamcommunity.com/groups/SteamLabs/discussions/3/1643170269567305036/) shipped on July 27, 2019,
- [`data_v3/`](data_v3/) after [the update](https://steamcommunity.com/groups/SteamLabs/discussions/3/1643170903484574354/) shipped on August 8, 2019.## Requirements
- Install the latest version of [Python 3.X](https://www.python.org/downloads/).
- Install the required packages:```bash
pip install -r requirements.txt
```## Usage
### Download the inputs and results of the recommender
You could manually download the inputs and results of the recommender to [`data_v3/`](data_v3/).
Otherwise, this process can be automated after providing the following personal information:1. Fill-in your SteamID in the function `get_steam_id()` found in `utils.py`.
2. Fill-in your cookie information in a new file called `personal_info.json`:
```json
{
"sessionid": "PASTE_YOUR_COOKIE_VALUE_HERE",
"steamLoginSecure": "PASTE_YOUR_COOKIE_VALUE_HERE"
}
```Afterwards, run the following script to download the data:
```bash
python file_utils.py
```## Trivia
### Interpolation
There are 30 pre-computed recommendation rankings of 200 games (`algorithm_options`) for the combinations of:
- 5 popularity biases,
- 6 release recency biases.A combination of sliders of the "Interactive Recommender" leads to [an interpolation](https://github.com/woctezuma/steam-labs-recommender/wiki/Interpolation) of the 4 closest rankings.
If a game appears in a ranking but not in the 3 others, its score is set to 0 for the 3 other rankings, and the interpolation is then computed.It is a nice trick to allow people to play with the sliders:
- without giving too much information away, especially the formula and the "popularity" values for each game,
- without overloading the browser with data. There are 30 rankings instead of 101x101.Moreover, it is important to notice that the interpolation is bilinear, which is consistent with the linear formula
mentioned above for `tweaked_output` with respect to the popularity bias.
However, with respect to the release recency bias, it might be a hack ; further investigations would be required to figure it out.### Misleading sliders
Sliders can be misleading, especially if you filter rare tags.
For instance, recommendations for the past 10 years do NOT necessarily include recommendations for the past 6 months.
![Slider for release recency](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/release_recency_slider.png)
This is a consequence of cutting the pre-computed rankings after 200 entries, and then interpolating rankings.
### How to filter rankings with any tag
In order not to overwhelm people with 337 tags, most tags are not displayed in the drop-down menu.
However, you can manually add them, and the "Interactive Recommender" will work perfectly fine with them!
All you need is the id for the tag of interest in the HTML page of the "Interactive Recommender". For instance:![Manually add a tag id](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/tag_input.png)
![Output filtered by tag](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/tag_output.png)
Caveat: if nothing appears, then do not forget to move the sliders.
Indeed, you might have a recommendation in one of the other pre-computed rankings, and due to ranking interpolation, you
would have to fiddle with the sliders for a new game to pop in.**Edit**: Tag filtering was officially extended to every tag with the update shipped on August 8, 2019.
![Extended tag filtering](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/tag_filtering.png)
## Pieces of advice
- Setting the slider to "popular" is mostly useful for new Steam users.
Looking for "popular" games makes sense for new Steam users.
However, for long-time users, you likely know about the surfaced "popular" games, and discarded them for good reasons.
Looking for more "niche" games increases the chance to discover a game which you might want to purchase and play.![Optimal popularity slider position](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/optimal_popularity_slider.png) ![Optimal popularity slider value](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/optimal_popularity_slider_value.png)
- The recommender might be biased by game bundles.
The recommender might be biased by game bundles.
Indeed, a few games could get recommended solely because they appeared in a bundle along with games which you own.
Focusing on recent releases (max 2-3 years) should alleviate this issue, because recent games are less likely to
have been featured in many bundles.![Optimal release recency slider position](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/optimal_release_recency_slider.png) ![Optimal release recency slider value](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/optimal_release_recency_slider_value.png)
For instance, I suspect that the game "Steel Rats" is recommended to me because it was featured in a Humble Monthly
bundle along with two games (MINIT and Dandara) which I have purchased with [Group Buys](https://steamcommunity.com/groups/groupbuys/) and then played.
That being said, I might be overestimating the impact of this issue, which may be toned down if the Recommender really
only considers my 50 most played games, which do not include any of these two games.![Humble Monthly Bundle](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/monthly.png)
## Extent of the input data
On April 27, 2019, Valve shipped an update which allows to exclude games from the input data.
This allows us to figure out whether the input data is limited to the top 50 games, or if it is larger, potentially
up to the whole user library.I have tried the recommender with a smurf account which has:
- three games (Dota2, Chivarly, Mirage),
- non-zero playtime only for Dota2.As a first observation, the list shown on the left of the recommender only includes Dota2.
This confirms that, at least cosmetically, owned games with zero playtime are ignored.Second, if Dota2 is manually excluded from the input data, then there is no recommendation at all.
This shows that:
- there is no default recommendation,
- games with zero playtime are effectively ignored.![Void response if the input is empty](https://raw.githubusercontent.com/wiki/woctezuma/steam-labs-recommender/img/void_response.png)
Now, if I go back to my main account, and manually exclude all of the top 50 games from the input data,
there are still recommended games.
This implies that the input data is not empty, and thus a bigger part of my library had to be taken into account.In conclusion, the recommender is fed more than your top 50 games, likely your whole library data
(with positive playtime), including all of these idled games.## References
- Valve's [patent US 11,194,879 B2][patent-html] ([PDF][patent-pdf] file)
- Microsoft's ["Best Practices on Recommendation Systems"](https://github.com/microsoft/recommenders),
- Posts on ResetERA to figure out the meaning of:
- [the variables](https://www.resetera.com/threads/pc-gaming-era-july-2019-life-liberty-and-the-pursuit-of-richaaaaaaard.126544/page-31#post-22706382),
- [some interpolation javascript code](https://www.resetera.com/threads/pc-gaming-era-july-2019-life-liberty-and-the-pursuit-of-richaaaaaaard.126544/page-31#post-22707917),
- [the updates shipped on July 27, 2019](https://www.resetera.com/threads/pc-gaming-era-july-2019-life-liberty-and-the-pursuit-of-richaaaaaaard.126544/page-53#post-23105410),
- the definition of popularity ([here](https://www.resetera.com/threads/pc-gaming-era-september-2019-the-girls-are-back-in-town.138427/page-28#post-24498619) and [there](https://www.resetera.com/threads/pc-gaming-era-september-2019-the-girls-are-back-in-town.138427/page-32#post-24574577)): sales and wishlist activity over a time window,
- Explanations regarding the interpolation of rankings: [here](https://www.resetera.com/threads/pc-gaming-era-july-2019-life-liberty-and-the-pursuit-of-richaaaaaaard.126544/page-31#post-22709362) and [there](https://www.resetera.com/threads/pc-gaming-era-july-2019-life-liberty-and-the-pursuit-of-richaaaaaaard.126544/page-31#post-22709744),
- [`Steam-Hype`](https://github.com/woctezuma/steam-hype): a possible measure of the popularity of a game,
- [A tweak](https://www.resetera.com/threads/pc-gaming-era-july-2019-life-liberty-and-the-pursuit-of-richaaaaaaard.126544/page-31#post-22710041) to filter any of the 337 tags with the current Interactive Recommender,
- [An article on PC Gamer](http://www.pcgamer.com/this-algorithm-picks-out-steams-best-hidden-gems/) which provides nice insights regarding my approach to [discover hidden gems](https://github.com/woctezuma/hidden-gems).[build]:
[build-image]:[build-action]:
[build-image-action]:[pyup]:
[dependency-image]:
[python3-image]:[codecov]:
[codecov-image]:[codacy]:
[codacy-image]:[patent-html]:
[patent-pdf-remote]:
[patent-pdf]: