4 changed files with 113 additions and 0 deletions
@ -0,0 +1,5 @@ |
|||||
|
guard |
||||
|
===== |
||||
|
|
||||
|
.. automodule:: steam.guard |
||||
|
:members: |
@ -0,0 +1,88 @@ |
|||||
|
import struct |
||||
|
from binascii import hexlify |
||||
|
from time import time |
||||
|
from steam import webapi |
||||
|
from steam.core.crypto import hmac_sha1, sha1_hash |
||||
|
|
||||
|
def generate_twofactor_code(shared_secret): |
||||
|
"""Generate Steam 2FA code for login with current time |
||||
|
|
||||
|
:param shared_secret: authenticator shared shared_secret |
||||
|
:type shared_secret: bytes |
||||
|
:return: steam two factor code |
||||
|
:rtype: str |
||||
|
""" |
||||
|
return generate_twofactor_code_for_time(shared_secret, time() + get_time_offset()) |
||||
|
|
||||
|
def generate_twofactor_code_for_time(shared_secret, timestamp): |
||||
|
"""Generate Steam 2FA code for timestamp |
||||
|
|
||||
|
:param shared_secret: authenticator shared secret |
||||
|
:type shared_secret: bytes |
||||
|
:param timestamp: timestamp to use, if left out uses current time |
||||
|
:type timestamp: int |
||||
|
:return: steam two factor code |
||||
|
:rtype: str |
||||
|
""" |
||||
|
hmac = hmac_sha1(bytes(shared_secret), |
||||
|
struct.pack('>Q', int(timestamp)//30)) # this will NOT stop working in 2038 |
||||
|
|
||||
|
start = ord(hmac[19:20]) & 0xF |
||||
|
codeint = struct.unpack('>I', hmac[start:start+4])[0] & 0x7fffffff |
||||
|
|
||||
|
charset = '23456789BCDFGHJKMNPQRTVWXY' |
||||
|
code = '' |
||||
|
|
||||
|
for _ in range(5): |
||||
|
codeint, i = divmod(codeint, len(charset)) |
||||
|
code += charset[i] |
||||
|
|
||||
|
return code |
||||
|
|
||||
|
def generate_confirmation_key(identity_secret, timestamp, tag=''): |
||||
|
"""Generate confirmation key for trades. Can only be used once. |
||||
|
|
||||
|
:param identity_secret: authenticator identity secret |
||||
|
:type identity_secret: bytes |
||||
|
:param timestamp: timestamp to use for generating key |
||||
|
:type timestamp: int |
||||
|
:param tag: tag identifies what the request, see list below |
||||
|
:type tag: str |
||||
|
:return: confirmation key |
||||
|
:rtype: bytes |
||||
|
|
||||
|
Tag choices: |
||||
|
|
||||
|
* ``conf`` to load the confirmations page |
||||
|
* ``details`` to load details about a trade |
||||
|
* ``allow`` to confirm a trade |
||||
|
* ``cancel`` to cancel a trade |
||||
|
|
||||
|
""" |
||||
|
data = struct.pack('>Q', int(timestamp)) + tag.encode('ascii') # this will NOT stop working in 2038 |
||||
|
return hmac_sha1(bytes(identity_secret), data) |
||||
|
|
||||
|
def get_time_offset(): |
||||
|
"""Get time offset from steam server time via WebAPI |
||||
|
|
||||
|
:return: time offset |
||||
|
:rtype: int |
||||
|
""" |
||||
|
try: |
||||
|
resp = webapi.post('ITwoFactorService', 'QueryTime', 1, params={'http_timeout': 5}) |
||||
|
except: |
||||
|
return 0 |
||||
|
|
||||
|
ts = int(time()) |
||||
|
return int(resp.get('response', {}).get('server_time', ts)) - ts |
||||
|
|
||||
|
def generate_device_id(steamid): |
||||
|
"""Generate Android device id |
||||
|
|
||||
|
:param steamid: Steam ID |
||||
|
:type steamid: :class:`.SteamID`, :class:`int` |
||||
|
:return: android device id |
||||
|
:rtype: str |
||||
|
""" |
||||
|
h = hexlify(sha1(str(steamid).encode('ascii'))).decode('ascii') |
||||
|
return "android:%s-%s-%s-%s-%s" % (h[:8], h[8:12], h[12:16], h[16:20], h[20:32]) |
@ -0,0 +1,19 @@ |
|||||
|
import unittest |
||||
|
import mock |
||||
|
|
||||
|
from steam import guard as g |
||||
|
|
||||
|
class TCguard(unittest.TestCase): |
||||
|
def test_generate_twofactor_code_for_time(self): |
||||
|
code = g.generate_twofactor_code_for_time(b'superdupersecret', timestamp=3000030) |
||||
|
self.assertEqual(code, 'YRGQJ') |
||||
|
|
||||
|
code = g.generate_twofactor_code_for_time(b'superdupersecret', timestamp=3000029) |
||||
|
self.assertEqual(code, '94R9D') |
||||
|
|
||||
|
def test_generate_confirmation_key(self): |
||||
|
key = g.generate_confirmation_key(b'itsmemario', 100000) |
||||
|
self.assertEqual(key, b'\xed\xb5\xe5\xad\x8f\xf1\x99\x01\xc8-w\xd6\xb5 p\xccz\xd7\xd1\x05') |
||||
|
|
||||
|
key = g.generate_confirmation_key(b'itsmemario', 100000, 'allow') |
||||
|
self.assertEqual(key, b"Q'\x06\x80\xe1g\xa8m$\xb2hV\xe6g\x8b'\x8f\xf1L\xb0") |
Loading…
Reference in new issue