This repository contains Python code to find arbitrages on the Steam Market.
Arbitrages could consist in:
- purchasing gems to craft booster packs, which are then immediately sold for more than their crafting cost,
- purchasing items, typically foil cards, which are then immediately turned into more gems than they are worth,
- purchasing games in order to produce the corresponding booster packs, with total profit greater than the game price,
- turning normal cards into badges, if the expected value is positive, to create items worth more than the badge cost.
- Install the latest version of Python 3.X (at least version 3.10).
- Install the required packages:
pip install -r requirements.txt
To relax the rate limits enforced by Steam API, fill-in your cookie information in a file called personal_info.json
:
How to fill-in your cookie information
-
To do so, make sure you are connected to your Steam account on a Steam Community page, e.g. Steam Market.
-
Press
<Shift-F9>
in your web browser to access the storage section of the developer tools.
-
Use the filtering option (in the top right of the storage section) to find the cookie value for
steamLoginSecure
.
-
Copy-paste this cookie value into a new file called
personal_info.json
, which will be read bypersonal_info.py
.{ "steamLoginSecure": "PASTE_YOUR_COOKIE_VALUE_HERE" }
NB: In the future, if you notice that the program bugs out due to seemingly very strict rate limits, then it may be a sign that the cookie value tied to your session has changed. In this case, try to fill-in your cookie information with its new value.
NB²: If you want to automate the creation and sale of booster packs, you may need:
- to have a mobile authenticator app running in the background and auto-confirming market transactions,
- to fill-in more cookie information. I have been using the following entries, but you might not need to use all of them. Except for
steamLoginSecure
andsessionid
, the values of the other entries are set in stone and do not need to be updated afterwards.{ "browserid": "PASTE_YOUR_COOKIE_VALUE_HERE", "steamMachineAuth_PASTE_YOUR_STEAM_ID_HERE": "PASTE_YOUR_COOKIE_VALUE_HERE", "steamRememberLogin": "PASTE_YOUR_COOKIE_VALUE_HERE", "sessionid": "PASTE_YOUR_COOKIE_VALUE_HERE", "steamLoginSecure": "PASTE_YOUR_COOKIE_VALUE_HERE" }
To have access to the gem cost for crafting Booster Packs, you will need to manually copy information available here.
There are two solutions:
- solution A is my original solution, but it requires a browser extension called Augmented Steam,
- solution B is a more recent solution, and does not require any third-party browser extension.
How to list craftable packs with *Augmented Steam*
-
Install the browser extension called Augmented Steam, so that the number of gems required to craft a Booster Pack appears in the drop-down menu:
-
Then, right-click the drop-down menu and "inspect" the corresponding HTML code in your browser:
-
Copy the following line and paste it to
data/booster_game_creator.txt
:
-
Add line-breaks, so that the file is formatted in the following way:
For instance, with Visual Studio Code, press
<Ctrl-H>
and run:
-
Strip the following unnecessary lines:
- three lines (including an empty line) at the beginning:
<select id="booster_game_selector"> <option value="">Select a game...</option>
- one line at the end:
</select>
Alternatively, if you wish not to install any browser extension:
How to list craftable packs with the Booster Creation webpage
-
Press
<Ctrl-U>
to display the HTML code of the Booser Creation webpage. -
At the end of the HTML code, find and copy the line below
CBoosterCreatorPage.Init
:
-
Paste the line to
data/booster_game_creator_from_javascript.txt
. -
Strip mentions of packs unavailable because they were crafted less than 24 hours ago. For instance:
{"appid":996580,"name":"Spyro\u2122 Reignited Trilogy","series":1,"price":"400", "unavailable":true,"available_at_time":"4 Sep @ 7:06pm"}
should be replaced with:
{"appid":996580,"name":"Spyro\u2122 Reignited Trilogy","series":1,"price":"400"}
To do so, with Visual Studio Code, press <Ctrl-H>
and remove occurrences of :
,"unavailable":true,"available_at_time":"[\w ]*@[\w :]*"
Basic functions
- To parse all the options to craft 'Booster Packs', for the games you own, run:
python parsing_utils.py
- To retrieve all the listings of 'Booster Packs' on the Steam Market, along with the sell price and volume, run:
python market_search.py
- To retrieve the price which sellers ask for a 'Sack of Gems', run:
python sack_of_gems.py
- To retrieve i) the "item name id" of a listing, and ii) whether a crafted item would really be marketable, run:
python market_listing.py
- To match listing hashes with badge creation details, run:
python market_utils.py
- To retrieve the ask and bid for 'Booster Packs', run:
python market_order.py
- To look for free games which i) feature trading cards (and thus crafting of booster packs), and ii) which I do not own, run:
python free_games_with_trading_cards.py
- To create packs for a manual selection of games, e.g. if you want to create these specific packs every day, run:
python batch_create_packs.py
- To list appIDs of interest for which we can under-cut the lowest sell order and still hope to make a profit:
python list_possible_lures.py
- To look for games which i) are likely to have high bid orders for their booster packs, and ii) which I may not own yet, run:
python market_buzz_detector.py
- To look for potentially profitable gambles on profile backgrounds and emoticons of "Common" rarity, run:
python market_gamble_detector.py
- To find market arbitrages, e.g. sell a pack for more (fee excluded) than the cost to craft it (fee included), run:
python market_arbitrage.py
- To find market arbitrages with foil cards, e.g. buy a foil card to turn it into more gems than its cost, run:
python market_arbitrage_with_foil_cards.py
Caveat: make sure to manually check the goo value of cards with a tool such as this bookmarklet. Indeed, if an arbitrage with foil cards looks too good to be true, it is likely that the goo value was bugged, because of a wrong item type. It can happen for instance if the goo value actually corresponds to emoticon or a profile background, and was then multiplied by 10 to get the value of the non-existent "foil" version of this emoticon or profile background.
javascript:var a=g_rgAssets[Object.keys(g_rgAssets)[0]],b=a[Object.keys(a)[0]],c=b[Object.keys(b)[0]],gem_action=c.owner_actions&&c.owner_actions.filter(function(d){return/javascript:GetGooValue/.test(d.link)})[0];if(gem_action){var matches=gem_action.link.match(/javascript:GetGooValue\( '%contextid%', '%assetid%', (\d+), (\d+), \d+ \)/);fetch("https://steamcommunity.com/auction/ajaxgetgoovalueforitemtype/?appid="+matches[1]+"&item_type="+matches[2]+"&border_color=0").then(function(d){return d.json()}).then(function(d){alert("This is worth "+d.goo_value+" gems")})["catch"](function(d){return console.error(d)})}else alert("This is worth 0 gems");
Caveat²: the bookmarklet linked above does not account for foil versions, so you should multiply by 10 the goo value if you are interested in foil cards. You can check by yourself that the bookmarklet returns the same goo values for normal cards and for foil cards.
For the gamble detector, we are interested in drop-rate estimates, when crafting badges, for items of Common rarity.
Based on the data so far (239 crafed badges):
- the drop-rates for Profile Backgrounds and for Emoticons are different (reject null hypothesis with 95% confidence),
- conditionally to C/UC/R patterns, the drop-rates may be similar (fail to reject null hypothesis with 95% confidence).
where:
- C/UC/R patterns are the numbers of Common/Uncommon/Rare items associated with each appID.
Therefore, we choose to estimate the drop-rates for Common rarity, conditionally to C/UC/R patterns.
The values hard-coded in drop_rate_estimates.py
are the centers of the Wilson score intervals with 95% confidence.
The Wiki shows a ranking of packs with high buy orders.
Rankings are also available for gambles with items of "Common" rarity, obtained after crafting a badge:
- An example of market arbitrage.
- A blog post about scraping Steam Market