Browse Source

replace cryptography with pycryptodomex

pull/166/head
Rossen Georgiev 6 years ago
committed by Rossen Georgiev
parent
commit
b31354f53a
  1. 4
      README.rst
  2. 3
      Vagrantfile
  3. 23
      docs/install.rst
  4. 4
      docs/intro.rst
  5. 2
      requirements.txt
  6. 3
      setup.py
  7. 47
      steam/core/crypto.py
  8. 29
      steam/webauth.py
  9. 10
      tests/test_core_crypto.py

4
README.rst

@ -16,14 +16,14 @@ Key features
* `WebAuth <http://steam.readthedocs.io/en/latest/api/steam.webauth.html>`_ - authentication for access to ``store.steampowered.com`` and ``steamcommunity.com`` * `WebAuth <http://steam.readthedocs.io/en/latest/api/steam.webauth.html>`_ - authentication for access to ``store.steampowered.com`` and ``steamcommunity.com``
Checkout the `User guide <http://steam.readthedocs.io/en/latest/user_guide.html>`_ for examples, Checkout the `User guide <http://steam.readthedocs.io/en/latest/user_guide.html>`_ for examples,
or the `API Reference <http://steam.readthedocs.io/en/latest/api/index.html>`_ for details. or the `API Reference <http://steam.readthedocs.io/en/latest/api/steam.html>`_ for details.
For questions, issues or general curiosity visit the repo at `https://github.com/ValvePython/steam <https://github.com/ValvePython/steam>`_. For questions, issues or general curiosity visit the repo at `https://github.com/ValvePython/steam <https://github.com/ValvePython/steam>`_.
Quick install Quick install
------------- -------------
For details on require system packages, see `Full Installation <http://steam.readthedocs.io/en/latest/install.html>`_. For system specific details, see `Install Details <http://steam.readthedocs.io/en/latest/install.html>`_.
Install latest version from PYPI Install latest version from PYPI

3
Vagrantfile

@ -15,12 +15,13 @@ Vagrant.configure("2") do |config|
# box.vm.synced_folder "../csgo-python/csgo/", "/home/vagrant/csgo" # box.vm.synced_folder "../csgo-python/csgo/", "/home/vagrant/csgo"
box.vm.provision "shell", inline: <<-SHELL box.vm.provision "shell", inline: <<-SHELL
set -x
apt-get update apt-get update
apt-get -y install build-essential libssl-dev libffi-dev python-dev
apt-get -y install python-pip python-virtualenv apt-get -y install python-pip python-virtualenv
SHELL SHELL
box.vm.provision "shell", privileged: false, inline: <<-SHELL box.vm.provision "shell", privileged: false, inline: <<-SHELL
set -x
virtualenv -p python2 venv2 virtualenv -p python2 venv2
source venv2/bin/activate source venv2/bin/activate
pip install -r /vagrant/requirements.txt ipython pip install -r /vagrant/requirements.txt ipython

23
docs/install.rst

@ -14,23 +14,6 @@ Steps assume that ``python`` and ``pip`` are already installed.
in order to keep you system packages untouched. in order to keep you system packages untouched.
Ubuntu/Debian
^^^^^^^^^^^^^
Replace ``python-dev`` with ``python3-dev`` for Python 3.
.. code-block:: console
$ sudo apt-get install build-essential libssl-dev libffi-dev python-dev
RHEL-based
^^^^^^^^^^
.. code-block:: console
$ sudo yum install gcc libffi-devel python-devel openssl-devel
Windows Windows
------- -------
@ -41,13 +24,7 @@ Cygwin
2. During the setup select these additional packages 2. During the setup select these additional packages
- ``python3`` - ``python3``
- ``python3-devel``
- ``python3-setuptools`` - ``python3-setuptools``
- ``gcc-core``
- ``gcc-g++``
- ``libffi6``
- ``libffi-devel``
- ``openssl-devel``
4. Install pip 4. Install pip
- Open cygwin terminal - Open cygwin terminal

4
docs/intro.rst

@ -20,14 +20,14 @@ Key features
* :doc:`WebAPI <api/steam.webapi>` - simple API for Steam's Web API with automatic population of interfaces * :doc:`WebAPI <api/steam.webapi>` - simple API for Steam's Web API with automatic population of interfaces
* :doc:`WebAuth <api/steam.webauth>` - authentication for access to ``store.steampowered.com`` and ``steamcommunity.com`` * :doc:`WebAuth <api/steam.webauth>` - authentication for access to ``store.steampowered.com`` and ``steamcommunity.com``
Checkout the :doc:`user_guide` for examples, or the :doc:`api/index` for details. Checkout the :doc:`user_guide` for examples, or the :doc:`api/steam` for details.
For questions, issues, or general curiosity, visit the repo at `https://github.com/ValvePython/steam <https://github.com/ValvePython/steam>`_. For questions, issues, or general curiosity, visit the repo at `https://github.com/ValvePython/steam <https://github.com/ValvePython/steam>`_.
Quick install Quick install
============= =============
For details on require system packages, see :doc:`install`. For system specific details, see :doc:`install`.
Install latest version from PYPI:: Install latest version from PYPI::

2
requirements.txt

@ -1,5 +1,5 @@
six>=1.10.0 six>=1.10.0
cryptography>=1.3 pycryptodomex>=3.7.0
requests>=2.9.1 requests>=2.9.1
vdf>=2.0 vdf>=2.0
gevent>=1.2.0 gevent>=1.2.0

3
setup.py

@ -16,7 +16,7 @@ with open(path.join(here, 'steam/__init__.py'), encoding='utf-8') as f:
install_requires = [ install_requires = [
'six>=1.10', 'six>=1.10',
'cryptography>=1.3', 'pycryptodomex>=3.7.0',
'requests>=2.9.1', 'requests>=2.9.1',
'vdf>=2.0', 'vdf>=2.0',
'gevent>=1.2.0', 'gevent>=1.2.0',
@ -50,6 +50,7 @@ setup(
'Programming Language :: Python :: 3.4', 'Programming Language :: Python :: 3.4',
'Programming Language :: Python :: 3.5', 'Programming Language :: Python :: 3.5',
'Programming Language :: Python :: 3.6', 'Programming Language :: Python :: 3.6',
'Programming Language :: Python :: Implementation :: PyPy',
], ],
keywords='valve steam steamid api webapi steamcommunity', keywords='valve steam steamid api webapi steamcommunity',
packages=['steam'] + ['steam.'+x for x in find_packages(where='steam')], packages=['steam'] + ['steam.'+x for x in find_packages(where='steam')],

47
steam/core/crypto.py

@ -5,20 +5,17 @@ import sys
from os import urandom as random_bytes from os import urandom as random_bytes
from struct import pack from struct import pack
from base64 import b64decode from base64 import b64decode
from cryptography.hazmat.primitives.hmac import HMAC
from cryptography.hazmat.primitives.hashes import Hash, SHA1 from Cryptodome.Hash import SHA1, HMAC
from cryptography.hazmat.primitives.asymmetric.padding import PSS, OAEP, MGF1 from Cryptodome.PublicKey.RSA import import_key as rsa_import_key, construct as rsa_construct
from cryptography.hazmat.primitives.ciphers import Cipher from Cryptodome.Cipher import PKCS1_OAEP, PKCS1_v1_5
from cryptography.hazmat.primitives.ciphers.algorithms import AES from Cryptodome.Cipher import AES as AES
from cryptography.hazmat.primitives.ciphers.modes import CBC, ECB
import cryptography.hazmat.backends
backend = cryptography.hazmat.backends.default_backend()
class UniverseKey(object): class UniverseKey(object):
"""Public keys for Universes""" """Public keys for Universes"""
Public = backend.load_der_public_key(b64decode(""" Public = rsa_import_key(b64decode("""
MIGdMA0GCSqGSIb3DQEBAQUAA4GLADCBhwKBgQDf7BrWLBBmLBc1OhSwfFkRf53T MIGdMA0GCSqGSIb3DQEBAQUAA4GLADCBhwKBgQDf7BrWLBBmLBc1OhSwfFkRf53T
2Ct64+AVzRkeRuh7h3SiGEYxqQMUeYKO6UWiSRKpI2hzic9pobFhRr3Bvr/WARvY 2Ct64+AVzRkeRuh7h3SiGEYxqQMUeYKO6UWiSRKpI2hzic9pobFhRr3Bvr/WARvY
gdTckPv+T1JzZsuVcNfFjrocejN1oWI0Rrtgt4Bo+hOneoo3S57G9F1fOpn5nsQ6 gdTckPv+T1JzZsuVcNfFjrocejN1oWI0Rrtgt4Bo+hOneoo3S57G9F1fOpn5nsQ6
@ -42,9 +39,9 @@ def generate_session_key(hmac_secret=b''):
:rtype: :class:`tuple` :rtype: :class:`tuple`
""" """
session_key = random_bytes(32) session_key = random_bytes(32)
encrypted_session_key = UniverseKey.Public.encrypt(session_key + hmac_secret, encrypted_session_key = PKCS1_OAEP.new(UniverseKey.Public, SHA1)\
OAEP(MGF1(SHA1()), SHA1(), None) .encrypt(session_key + hmac_secret)
)
return (session_key, encrypted_session_key) return (session_key, encrypted_session_key)
def symmetric_encrypt(message, key): def symmetric_encrypt(message, key):
@ -58,13 +55,11 @@ def symmetric_encrypt_HMAC(message, key, hmac_secret):
return symmetric_encrypt_with_iv(message, key, iv) return symmetric_encrypt_with_iv(message, key, iv)
def symmetric_encrypt_iv(iv, key): def symmetric_encrypt_iv(iv, key):
encryptor = Cipher(AES(key), ECB(), backend).encryptor() return AES.new(key, AES.MODE_ECB).encrypt(iv)
return encryptor.update(iv) + encryptor.finalize()
def symmetric_encrypt_with_iv(message, key, iv): def symmetric_encrypt_with_iv(message, key, iv):
encrypted_iv = symmetric_encrypt_iv(iv, key) encrypted_iv = symmetric_encrypt_iv(iv, key)
encryptor = Cipher(AES(key), CBC(iv), backend).encryptor() cyphertext = AES.new(key, AES.MODE_CBC, iv).encrypt(pad(message))
cyphertext = encryptor.update(pad(message)) + encryptor.finalize()
return encrypted_iv + cyphertext return encrypted_iv + cyphertext
def symmetric_decrypt(cyphertext, key): def symmetric_decrypt(cyphertext, key):
@ -84,19 +79,19 @@ def symmetric_decrypt_HMAC(cyphertext, key, hmac_secret):
return message return message
def symmetric_decrypt_iv(cyphertext, key): def symmetric_decrypt_iv(cyphertext, key):
decryptor = Cipher(AES(key), ECB(), backend).decryptor() return AES.new(key, AES.MODE_ECB).decrypt(cyphertext[:BS])
return decryptor.update(cyphertext[:BS]) + decryptor.finalize()
def symmetric_decrypt_with_iv(cyphertext, key, iv): def symmetric_decrypt_with_iv(cyphertext, key, iv):
decryptor = Cipher(AES(key), CBC(iv), backend).decryptor() return unpad(AES.new(key, AES.MODE_CBC, iv).decrypt(cyphertext[BS:]))
return unpad(decryptor.update(cyphertext[BS:]) + decryptor.finalize())
def hmac_sha1(secret, data): def hmac_sha1(secret, data):
hmac = HMAC(secret, SHA1(), backend) return HMAC.new(secret, data, SHA1).digest()
hmac.update(data)
return hmac.finalize()
def sha1_hash(data): def sha1_hash(data):
sha = Hash(SHA1(), backend) return SHA1.new(data).digest()
sha.update(data)
return sha.finalize() def rsa_publickey(mod, exp):
return rsa_construct((mod, exp))
def pkcs1v15_encrypt(key, message):
return PKCS1_v1_5.new(key).encrypt(message)

29
steam/webauth.py

@ -60,12 +60,9 @@ from time import time
from base64 import b64encode from base64 import b64encode
import requests import requests
from cryptography.hazmat.primitives.asymmetric.rsa import RSAPublicNumbers
from cryptography.hazmat.primitives.asymmetric.padding import PKCS1v15
from steam.core.crypto import backend
from steam import SteamID, webapi from steam import SteamID, webapi
from steam.util.web import make_requests_session, generate_session_id from steam.util.web import make_requests_session, generate_session_id
from steam.core.crypto import rsa_publickey, pkcs1v15_encrypt
if sys.version_info < (3,): if sys.version_info < (3,):
intBase = long intBase = long
@ -108,12 +105,12 @@ class WebAuth(object):
""" """
try: try:
resp = self.session.post('https://steamcommunity.com/login/getrsakey/', resp = self.session.post('https://steamcommunity.com/login/getrsakey/',
timeout=15, timeout=15,
data={ data={
'username': username, 'username': username,
'donotchache': int(time() * 1000), 'donotchache': int(time() * 1000),
}, },
).json() ).json()
except requests.exceptions.RequestException as e: except requests.exceptions.RequestException as e:
raise HTTPError(str(e)) raise HTTPError(str(e))
@ -123,17 +120,15 @@ class WebAuth(object):
if not self.key: if not self.key:
resp = self.get_rsa_key(self.username) resp = self.get_rsa_key(self.username)
nums = RSAPublicNumbers(intBase(resp['publickey_exp'], 16), self.key = rsa_publickey(intBase(resp['publickey_mod'], 16),
intBase(resp['publickey_mod'], 16), intBase(resp['publickey_exp'], 16),
) )
self.key = backend.load_rsa_public_numbers(nums)
self.timestamp = resp['timestamp'] self.timestamp = resp['timestamp']
def _send_login(self, captcha='', email_code='', twofactor_code=''): def _send_login(self, captcha='', email_code='', twofactor_code=''):
data = { data = {
'username' : self.username, 'username': self.username,
"password": b64encode(self.key.encrypt(self.password.encode('ascii'), PKCS1v15())), "password": b64encode(pkcs1v15_encrypt(self.key, self.password.encode('ascii'))),
"emailauth": email_code, "emailauth": email_code,
"emailsteamid": str(self.steam_id) if email_code else '', "emailsteamid": str(self.steam_id) if email_code else '',
"twofactorcode": twofactor_code, "twofactorcode": twofactor_code,

10
tests/test_core_crypto.py

@ -45,18 +45,20 @@ class crypto_testcase(unittest.TestCase):
# self.assertEqual(key, expected_key) # self.assertEqual(key, expected_key)
# self.assertEqual(ekey, expected_ekey) # self.assertEqual(ekey, expected_ekey)
def test_encryption(self): def test_encryption_legacy(self):
message = b'My secret message' message = b'My secret message'
key = b'9' * 32 key = b'9' * 32
hmac = b'3' * 16
# legacy
cyphertext = crypto.symmetric_encrypt(message, key) cyphertext = crypto.symmetric_encrypt(message, key)
dmessage = crypto.symmetric_decrypt(cyphertext, key) dmessage = crypto.symmetric_decrypt(cyphertext, key)
self.assertEqual(message, dmessage) self.assertEqual(message, dmessage)
# with HMAC def test_encryption_hmac(self):
message = b'My secret message'
key = b'9' * 32
hmac = b'3' * 16
cyphertext = crypto.symmetric_encrypt_HMAC(message, key, hmac) cyphertext = crypto.symmetric_encrypt_HMAC(message, key, hmac)
dmessage = crypto.symmetric_decrypt_HMAC(cyphertext, key, hmac) dmessage = crypto.symmetric_decrypt_HMAC(cyphertext, key, hmac)

Loading…
Cancel
Save