3 changed files with 269 additions and 0 deletions
@ -0,0 +1,85 @@ |
|||||
|
Simple Web API recipe |
||||
|
--------------------- |
||||
|
|
||||
|
Valve doesn't have a Web API for everything, and they don't need to. |
||||
|
We are going to use the ``steam`` and ``flask`` to build a Web API. |
||||
|
First, we need to install ``flask``. |
||||
|
|
||||
|
.. code:: bash |
||||
|
|
||||
|
(env)$ pip install flask |
||||
|
|
||||
|
``run_webapi.py`` contains our HTTP server app and ``steam_worker.py`` is |
||||
|
a modified version of persistent login recipe that will talk with steam. |
||||
|
|
||||
|
Let's run the app: |
||||
|
|
||||
|
.. code:: bash |
||||
|
|
||||
|
(env)$ python run_webapi.py |
||||
|
2016-11-01 00:00:01,000 | SimpleWebAPI | Simple Web API recipe |
||||
|
2016-11-01 00:00:02,000 | SimpleWebAPI | ------------------------------ |
||||
|
2016-11-01 00:00:03,000 | SimpleWebAPI | Starting Steam worker... |
||||
|
Username: myusername |
||||
|
Password: |
||||
|
2016-11-01 00:00:04,000 | Steam Worker | Connected to (u'1.2.3.4', 27018) |
||||
|
2016-11-01 00:00:05,000 | SimpleWebAPI | Starting HTTP server... |
||||
|
2016-11-01 00:00:06,000 | Steam Worker | ------------------------------ |
||||
|
2016-11-01 00:00:07,000 | Steam Worker | Logged on as: FriendlyGhost |
||||
|
... |
||||
|
127.0.0.1 - - [2016-01-01 00:00:08] "GET /ISteamApps/GetPlayerCount/?appid=0 HTTP/1.1" 200 155 0.262596 |
||||
|
... |
||||
|
|
||||
|
|
||||
|
Here are the available endpoints: |
||||
|
|
||||
|
.. code:: bash |
||||
|
|
||||
|
$ curl -s 127.0.0.1:5000/ISteamApps/GetProductInfo/?appids=570,730 | head -56 |
||||
|
{ |
||||
|
"apps": [ |
||||
|
{ |
||||
|
"appid": 570, |
||||
|
"appinfo": { |
||||
|
"appid": "570", |
||||
|
"common": { |
||||
|
"clienticon": "c0d15684e6c186289b50dfe083f5c562c57e8fb6", |
||||
|
"clienttga": "5ca2b133f8fdf56c3d81dd73d1254f95f0614265", |
||||
|
"community_hub_visible": "1", |
||||
|
"controllervr": { |
||||
|
"steamvr": "1" |
||||
|
}, |
||||
|
"exfgls": "1", |
||||
|
"gameid": "570", |
||||
|
"header_image": { |
||||
|
"english": "header.jpg" |
||||
|
}, |
||||
|
"icon": "0bbb630d63262dd66d2fdd0f7d37e8661a410075", |
||||
|
"linuxclienticon": "e1c520b6a98b1fed674a117e9356cdb9ddc6d40c", |
||||
|
"logo": "d4f836839254be08d8e9dd333ecc9a01782c26d2", |
||||
|
"logo_small": "d4f836839254be08d8e9dd333ecc9a01782c26d2_thumb", |
||||
|
"metacritic_fullurl": "http://www.metacritic.com/game/pc/dota-2?ftag=MCD-06-10aaa1f", |
||||
|
"metacritic_name": "Dota 2", |
||||
|
"metacritic_score": "90", |
||||
|
|
||||
|
|
||||
|
|
||||
|
$ curl -s 127.0.0.1:5000/ISteamApps/GetProductChanges/?since_changenumber=2397700 | head -10 |
||||
|
{ |
||||
|
"app_changes": [ |
||||
|
{ |
||||
|
"appid": 730, |
||||
|
"change_number": 2409212 |
||||
|
}, |
||||
|
{ |
||||
|
"appid": 740, |
||||
|
"change_number": 2409198 |
||||
|
}, |
||||
|
|
||||
|
|
||||
|
|
||||
|
$ curl 127.0.0.1:5000/ISteamApps/GetPlayerCount/?appid=0 |
||||
|
{ |
||||
|
"eresult": 1, |
||||
|
"player_count": 2727080 |
||||
|
} |
@ -0,0 +1,55 @@ |
|||||
|
from getpass import getpass |
||||
|
from gevent.wsgi import WSGIServer |
||||
|
from steam_worker import SteamWorker |
||||
|
from flask import Flask, request, abort, jsonify |
||||
|
|
||||
|
import logging |
||||
|
logging.basicConfig(format="%(asctime)s | %(name)s | %(message)s", level=logging.INFO) |
||||
|
LOG = logging.getLogger('SimpleWebAPI') |
||||
|
|
||||
|
app = Flask('SimpleWebAPI') |
||||
|
|
||||
|
@app.route("/ISteamApps/GetProductInfo/", methods=['GET']) |
||||
|
def GetProductInfo(): |
||||
|
appids = request.args.get('appids', '') |
||||
|
pkgids = request.args.get('packageids', '') |
||||
|
|
||||
|
if not appids and not pkgids: |
||||
|
return jsonify({}) |
||||
|
|
||||
|
appids = map(int, appids.split(',')) if appids else [] |
||||
|
pkgids = map(int, pkgids.split(',')) if pkgids else [] |
||||
|
|
||||
|
return jsonify(worker.get_product_info(appids, pkgids) or {}) |
||||
|
|
||||
|
@app.route("/ISteamApps/GetProductChanges/", methods=['GET']) |
||||
|
def GetProductChanges(): |
||||
|
chgnum = int(request.args.get('since_changenumber', 0)) |
||||
|
return jsonify(worker.get_product_changes(chgnum)) |
||||
|
|
||||
|
@app.route("/ISteamApps/GetPlayerCount/", methods=['GET']) |
||||
|
def GetPlayerCount(): |
||||
|
appid = int(request.args.get('appid', 0)) |
||||
|
return jsonify(worker.get_player_count(appid)) |
||||
|
|
||||
|
|
||||
|
if __name__ == "__main__": |
||||
|
LOG.info("Simple Web API recipe") |
||||
|
LOG.info("-"*30) |
||||
|
LOG.info("Starting Steam worker...") |
||||
|
|
||||
|
worker = SteamWorker() |
||||
|
|
||||
|
try: |
||||
|
worker.start(username=raw_input('Username: '), password=getpass()) |
||||
|
except: |
||||
|
raise SystemExit |
||||
|
|
||||
|
LOG.info("Starting HTTP server...") |
||||
|
http_server = WSGIServer(('', 5000), app) |
||||
|
|
||||
|
try: |
||||
|
http_server.serve_forever() |
||||
|
except KeyboardInterrupt: |
||||
|
LOG.info("Exit requested") |
||||
|
worker.close() |
@ -0,0 +1,129 @@ |
|||||
|
import logging |
||||
|
import gevent |
||||
|
from binascii import hexlify |
||||
|
from steam import SteamClient |
||||
|
from steam.core.msg import MsgProto |
||||
|
from steam.enums.emsg import EMsg |
||||
|
from steam.util import proto_to_dict |
||||
|
import vdf |
||||
|
|
||||
|
LOG = logging.getLogger("Steam Worker") |
||||
|
|
||||
|
|
||||
|
class SteamWorker(object): |
||||
|
def __init__(self): |
||||
|
self.logged_on_once = False |
||||
|
self.logon_details = {} |
||||
|
|
||||
|
self.steam = client = SteamClient() |
||||
|
client.set_credential_location(".") |
||||
|
|
||||
|
@client.on("error") |
||||
|
def handle_error(result): |
||||
|
LOG.info("Logon result: %s", repr(result)) |
||||
|
|
||||
|
@client.on("channel_secured") |
||||
|
def send_login(): |
||||
|
if client.relogin_available: |
||||
|
client.relogin() |
||||
|
else: |
||||
|
client.login(**self.logon_details) |
||||
|
self.logon_details.pop('auth_code', None) |
||||
|
self.logon_details.pop('two_factor_code', None) |
||||
|
|
||||
|
@client.on("connected") |
||||
|
def handle_connected(): |
||||
|
LOG.info("Connected to %s", client.current_server_addr) |
||||
|
|
||||
|
@client.on("reconnect") |
||||
|
def handle_reconnect(delay): |
||||
|
LOG.info("Reconnect in %ds...", delay) |
||||
|
|
||||
|
@client.on("disconnected") |
||||
|
def handle_disconnect(): |
||||
|
LOG.info("Disconnected.") |
||||
|
|
||||
|
if self.logged_on_once: |
||||
|
LOG.info("Reconnecting...") |
||||
|
client.reconnect(maxdelay=30) |
||||
|
|
||||
|
@client.on("auth_code_required") |
||||
|
def auth_code_prompt(is_2fa, mismatch): |
||||
|
if mismatch: |
||||
|
LOG.info("Previous code was incorrect") |
||||
|
|
||||
|
if is_2fa: |
||||
|
code = raw_input("Enter 2FA Code: ") |
||||
|
self.logon_details['two_factor_code'] = code |
||||
|
else: |
||||
|
code = raw_input("Enter Email Code: ") |
||||
|
self.logon_details['auth_code'] = code |
||||
|
|
||||
|
client.connect() |
||||
|
|
||||
|
@client.on("logged_on") |
||||
|
def handle_after_logon(): |
||||
|
self.logged_on_once = True |
||||
|
|
||||
|
LOG.info("-"*30) |
||||
|
LOG.info("Logged on as: %s", client.user.name) |
||||
|
LOG.info("Community profile: %s", client.steam_id.community_url) |
||||
|
LOG.info("Last logon: %s", client.user.last_logon) |
||||
|
LOG.info("Last logoff: %s", client.user.last_logoff) |
||||
|
LOG.info("-"*30) |
||||
|
|
||||
|
|
||||
|
def start(self, username, password): |
||||
|
self.logon_details = { |
||||
|
'username': username, |
||||
|
'password': password, |
||||
|
} |
||||
|
|
||||
|
self.steam.connect() |
||||
|
self.steam.wait_event('logged_on') |
||||
|
|
||||
|
def close(self): |
||||
|
if self.steam.connected: |
||||
|
self.logged_on_once = False |
||||
|
LOG.info("Logout") |
||||
|
self.steam.logout() |
||||
|
|
||||
|
def get_product_info(self, appids=[], packageids=[]): |
||||
|
resp = self.steam.send_job_and_wait(MsgProto(EMsg.ClientPICSProductInfoRequest), |
||||
|
{ |
||||
|
'apps': map(lambda x: {'appid': x}, appids), |
||||
|
'packages': map(lambda x: {'packageid': x}, packageids), |
||||
|
}, |
||||
|
timeout=10 |
||||
|
) |
||||
|
|
||||
|
if not resp: return {} |
||||
|
|
||||
|
resp = proto_to_dict(resp) |
||||
|
|
||||
|
for app in resp.get('apps', []): |
||||
|
app['appinfo'] = vdf.loads(app.pop('buffer').rstrip('\x00'))['appinfo'] |
||||
|
app['sha'] = hexlify(app['sha']) |
||||
|
for pkg in resp.get('packages', []): |
||||
|
pkg['appinfo'] = vdf.binary_loads(pkg.pop('buffer')[4:])[str(pkg['packageid'])] |
||||
|
pkg['sha'] = hexlify(pkg['sha']) |
||||
|
|
||||
|
return resp |
||||
|
|
||||
|
def get_product_changes(self, since_change_number): |
||||
|
resp = self.steam.send_job_and_wait(MsgProto(EMsg.ClientPICSChangesSinceRequest), |
||||
|
{ |
||||
|
'since_change_number': since_change_number, |
||||
|
'send_app_info_changes': True, |
||||
|
'send_package_info_changes': True, |
||||
|
}, |
||||
|
timeout=10 |
||||
|
) |
||||
|
return proto_to_dict(resp) or {} |
||||
|
|
||||
|
def get_player_count(self, appid): |
||||
|
resp = self.steam.send_job_and_wait(MsgProto(EMsg.ClientGetNumberOfCurrentPlayersDP), |
||||
|
{'appid': appid}, |
||||
|
timeout=10 |
||||
|
) |
||||
|
return proto_to_dict(resp) or {} |
Loading…
Reference in new issue