Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Decrypt input secrets if there are some #45

Merged
merged 10 commits into from
Feb 3, 2023
1 change: 1 addition & 0 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@
'websockets ~= 10.4',
'aiofiles ~= 22.1.0',
'aioshutil ~= 1.2',
'cryptography ~= 39.0.0',
],
extras_require={
'dev': [
Expand Down
126 changes: 126 additions & 0 deletions src/apify/_crypto.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,126 @@
import base64
import secrets

from cryptography.exceptions import InvalidTag as InvalidTagException
from cryptography.hazmat.primitives import hashes, serialization
from cryptography.hazmat.primitives.asymmetric import padding, rsa
from cryptography.hazmat.primitives.ciphers import Cipher, algorithms, modes

ENCRYPTION_KEY_LENGTH = 32
ENCRYPTION_IV_LENGTH = 16
ENCRYPTION_AUTH_TAG_LENGTH = 16


def public_encrypt(public_key: rsa.RSAPublicKey, value: str) -> dict:
drobnikj marked this conversation as resolved.
Show resolved Hide resolved
"""Encrypts the given value using AES cipher and the password for encryption using the public key.

The encryption password is a string of encryption key and initial vector used for cipher.
It returns the encrypted password and encrypted value in BASE64 format.

Args:
public_key (RSAPublicKey): Private key to use for decryption.
value (str): Password used to encrypt the private key encoded as base65 string.
drobnikj marked this conversation as resolved.
Show resolved Hide resolved

Returns:
disc: Encrypted password and value.
"""
key_bytes = _crypto_random_object_id(ENCRYPTION_KEY_LENGTH).encode('utf-8')
initialized_vector_bytes = _crypto_random_object_id(ENCRYPTION_IV_LENGTH).encode('utf-8')
value_bytes = value.encode('utf-8')

password_bytes = key_bytes + initialized_vector_bytes

# NOTE: Auth Tag is appended to the end of the encrypted data, it has length of 16 bytes and ensures integrity of the data.
cipher = Cipher(algorithms.AES(key_bytes), modes.GCM(initialized_vector_bytes, min_tag_length=ENCRYPTION_AUTH_TAG_LENGTH))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

min_tag_length (int) – The minimum length tag must be. By default this is 16, meaning tag truncation is not allowed. Allowing tag truncation is strongly discouraged for most applications.

min_tag_length is 16 by default, why override with 16?

Copy link
Member Author

@drobnikj drobnikj Feb 2, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just to be sure that default will change the decryption still work. We are strictly setting tag to 16 chars in length same as in node js version of this method.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ok, fair enough

encryptor = cipher.encryptor()
encrypted_value_bytes = encryptor.update(value_bytes) + encryptor.finalize()
encrypted_password_bytes = public_key.encrypt(
password_bytes,
padding.OAEP(
mgf=padding.MGF1(algorithm=hashes.SHA1()),
algorithm=hashes.SHA1(),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

SHA-1 is a deprecated hash algorithm that has practical known collision attacks. You are strongly discouraged from using it. Existing applications should strongly consider moving away.

https://cryptography.io/en/latest/hazmat/primitives/cryptographic-hashes/#cryptography.hazmat.primitives.hashes.SHA1

Copy link
Member Author

@drobnikj drobnikj Feb 2, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You are right, it is deprecated for the encryption, but in this case, we are using sha1 for padding and it is not the issue. In short, sha1 is there used for generating a hash which fills the block of the message for encryption. It is quite complex stuff, but the message itself together with the hash is then encrypted using the public key(RSA). There is a nice picture and explanation of how OAEP works.
The sha1 is used to This combination(MGF1 + SHA1 for padding) for RSA encryption OAEP (RSA_PKCS1_OAEP_PADDING) is recommended by open SSL, and it is used for example in node js by default. As it is used in node js by default.

label=None,
),
)
return {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why do you .decode('utf-8') here? Do you need the values as strings or?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, I need the value as a string. The function is aligned with the same one from JS for better accountability. Basically, we didn't use this function in python yet. The function is there mainly for testing and maybe for the future.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ok

'encrypted_value': base64.b64encode(encrypted_value_bytes + encryptor.tag).decode('utf-8'),
'encrypted_password': base64.b64encode(encrypted_password_bytes).decode('utf-8'),
}


def private_decrypt(
private_key: rsa.RSAPrivateKey,
*,
encrypted_password: str,
encrypted_value: str,
drobnikj marked this conversation as resolved.
Show resolved Hide resolved
) -> str:
"""Decrypts the given encrypted value using the private key and password.

Args:
private_key (RSAPrivateKey): Private key to use for decryption.
encrypted_password (str): Password used to encrypt the private key encoded as base65 string.
drobnikj marked this conversation as resolved.
Show resolved Hide resolved
encrypted_value (str): Encrypted value to decrypt as base64 string.

Returns:
str: Decrypted value.
"""
encrypted_password_bytes = base64.b64decode(encrypted_password.encode('utf-8'))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think if you didn't decode in encrypt, you can omit encode here?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is true. But the encrypted secret(password and value) is stored in input JSON, and the base64 string is better for handling as it has a subset of characters and we can easily match it.

encrypted_value_bytes = base64.b64decode(encrypted_value.encode('utf-8'))

# Decrypt the password
password_bytes = private_key.decrypt(
encrypted_password_bytes,
padding.OAEP(
mgf=padding.MGF1(algorithm=hashes.SHA1()),
jirimoravcik marked this conversation as resolved.
Show resolved Hide resolved
algorithm=hashes.SHA1(),
label=None,
),
)

if len(password_bytes) != ENCRYPTION_KEY_LENGTH + ENCRYPTION_IV_LENGTH:
raise ValueError('Decryption failed, invalid password length!')

# Slice the encrypted into cypher and authentication tag
authentication_tag_bytes = encrypted_value_bytes[-ENCRYPTION_AUTH_TAG_LENGTH:]
encrypted_data_bytes = encrypted_value_bytes[:len(encrypted_value_bytes) - ENCRYPTION_AUTH_TAG_LENGTH]

print(authentication_tag_bytes)
print(encrypted_data_bytes)
drobnikj marked this conversation as resolved.
Show resolved Hide resolved

encryption_key_bytes = password_bytes[:ENCRYPTION_KEY_LENGTH]
initailization_vector_bytes = password_bytes[ENCRYPTION_KEY_LENGTH:]

try:
cipher = Cipher(algorithms.AES(encryption_key_bytes), modes.GCM(initailization_vector_bytes, authentication_tag_bytes))
drobnikj marked this conversation as resolved.
Show resolved Hide resolved
decryptor = cipher.decryptor()
decipher_bytes = decryptor.update(encrypted_data_bytes) + decryptor.finalize()
except InvalidTagException:
raise ValueError('Decryption failed, malformed encrypted value or password.')
except Exception as err:
raise err

return decipher_bytes.decode('utf-8')


def _load_private_key(private_key_file_base64: str, private_key_password: str) -> rsa.RSAPrivateKey:
private_key = serialization.load_pem_private_key(base64.b64decode(
private_key_file_base64.encode('utf-8')), password=private_key_password.encode('utf-8'))
print(private_key)
if not isinstance(private_key, rsa.RSAPrivateKey):
raise ValueError('Invalid private key.')

return private_key


def _load_public_key(public_key_file_base64: str) -> rsa.RSAPublicKey:
public_key = serialization.load_pem_public_key(base64.b64decode(public_key_file_base64.encode('utf-8')))
if not isinstance(public_key, rsa.RSAPublicKey):
raise ValueError('Invalid public key.')

return public_key


def _crypto_random_object_id(length: int = 17) -> str:
"""Python reimplementation of cryptoRandomObjectId from `@apify/utilities`."""
chars = 'abcdefghijklmnopqrstuvwxyzABCEDFGHIJKLMNOPQRSTUVWXYZ0123456789'
return ''.join(secrets.choice(chars) for _ in range(length))
30 changes: 23 additions & 7 deletions src/apify/_utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@
import mimetypes
import os
import re
import secrets
import sys
import time
from collections import OrderedDict
Expand All @@ -25,9 +24,11 @@
import psutil
from aiofiles import ospath
from aiofiles.os import remove, rename
from cryptography.hazmat.primitives.asymmetric.rsa import RSAPrivateKey

from apify_client import __version__ as client_version

from ._crypto import private_decrypt
from ._version import __version__ as sdk_version
from .consts import (
_BOOL_ENV_VARS_TYPE,
Expand All @@ -36,6 +37,7 @@
_STRING_ENV_VARS_TYPE,
BOOL_ENV_VARS,
DATETIME_ENV_VARS,
ENCRYPTED_INPUT_VALUE_REGEXP,
INTEGER_ENV_VARS,
REQUEST_ID_LENGTH,
ApifyEnvVars,
Expand Down Expand Up @@ -324,12 +326,6 @@ def wrapper(*args: Any, **kwargs: Any) -> Any:
return cast(MetadataType, wrapper)


def _crypto_random_object_id(length: int = 17) -> str:
"""Python reimplementation of cryptoRandomObjectId from `@apify/utilities`."""
chars = 'abcdefghijklmnopqrstuvwxyzABCEDFGHIJKLMNOPQRSTUVWXYZ0123456789'
return ''.join(secrets.choice(chars) for _ in range(length))


T = TypeVar('T')


Expand Down Expand Up @@ -383,3 +379,23 @@ def items(self) -> ItemsView[str, T]: # Needed so we don't mutate the cache by

def _is_running_in_ipython() -> bool:
return getattr(builtins, '__IPYTHON__', False)


def _decrypt_input_secrets(private_key: RSAPrivateKey, input: Any) -> Any:
drobnikj marked this conversation as resolved.
Show resolved Hide resolved
"""Decrypt input secrets."""
if not isinstance(input, dict):
return input

for key, value in input.items():
if isinstance(value, str):
match = ENCRYPTED_INPUT_VALUE_REGEXP.fullmatch(value)
if match:
encrypted_password = match.group(1)
encrypted_value = match.group(2)
input[key] = private_decrypt(
private_key,
encrypted_password=encrypted_password,
encrypted_value=encrypted_value,
)

return input
18 changes: 14 additions & 4 deletions src/apify/actor.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,9 @@
from apify_client import ApifyClientAsync
from apify_client.consts import WebhookEventType

from ._crypto import _load_private_key
from ._utils import (
_decrypt_input_secrets,
_fetch_and_parse_env_var,
_get_cpu_usage_percent,
_get_memory_usage_bytes,
Expand Down Expand Up @@ -584,11 +586,19 @@ async def get_input(cls) -> Any:
async def _get_input_internal(self) -> Any:
self._raise_if_not_initialized()

# TODO: decryption
input_value = await self.get_value(self._config.input_key)
input_secrets_private_key = self._config.input_secrets_private_key_file
input_secrets_key_passphrase = self._config.input_secrets_private_key_passphrase
if input_secrets_private_key and input_secrets_key_passphrase:
private_key = _load_private_key(
input_secrets_private_key,
input_secrets_key_passphrase,
)
input_value = _decrypt_input_secrets(private_key, input_value)

return await self.get_value(self._config.input_key)
return input_value

@classmethod
@ classmethod
drobnikj marked this conversation as resolved.
Show resolved Hide resolved
async def get_value(cls, key: str) -> Any:
"""Get a value from the default key-value store associated with the current actor run.

Expand All @@ -604,7 +614,7 @@ async def _get_value_internal(self, key: str) -> Any:
value = await key_value_store.get_value(key)
return value

@classmethod
@ classmethod
drobnikj marked this conversation as resolved.
Show resolved Hide resolved
async def set_value(
cls,
key: str,
Expand Down
5 changes: 5 additions & 0 deletions src/apify/consts.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import re
from enum import Enum
from typing import List, Literal, get_args

Expand Down Expand Up @@ -159,3 +160,7 @@ class StorageTypes(str, Enum):
REQUEST_QUEUE_HEAD_MAX_LIMIT = 1000

EVENT_LISTENERS_TIMEOUT_SECS = 5

BASE64_REGEXP = '[-A-Za-z0-9+/]*={0,3}'
ENCRYPTED_INPUT_VALUE_PREFIX = 'ENCRYPTED_VALUE'
drobnikj marked this conversation as resolved.
Show resolved Hide resolved
ENCRYPTED_INPUT_VALUE_REGEXP = re.compile(f'^{ENCRYPTED_INPUT_VALUE_PREFIX}:({BASE64_REGEXP}):({BASE64_REGEXP})$')
3 changes: 2 additions & 1 deletion src/apify/storages/request_queue.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,8 @@
from apify_client import ApifyClientAsync
from apify_client.clients import RequestQueueClientAsync

from .._utils import LRUCache, _crypto_random_object_id, _unique_key_to_request_id
from .._crypto import _crypto_random_object_id
from .._utils import LRUCache, _unique_key_to_request_id
from ..config import Configuration
from ..consts import REQUEST_QUEUE_HEAD_MAX_LIMIT
from ..memory_storage import MemoryStorage
Expand Down
2 changes: 1 addition & 1 deletion tests/integration/_utils.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
from apify._utils import _crypto_random_object_id
from apify._crypto import _crypto_random_object_id


def generate_unique_resource_name(label: str) -> str:
Expand Down
2 changes: 1 addition & 1 deletion tests/integration/test_fixtures.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
from datetime import datetime, timezone

from apify import Actor
from apify._utils import _crypto_random_object_id
from apify._crypto import _crypto_random_object_id
from apify_client import ApifyClientAsync

from .conftest import ActorFactory
Expand Down
71 changes: 71 additions & 0 deletions tests/unit/test_crypto.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
import base64

import pytest

from apify._crypto import _crypto_random_object_id, _load_private_key, _load_public_key, private_decrypt, public_encrypt

# NOTE: Uses the same keys as in:
# https://github.com/apify/apify-shared-js/blob/master/test/crypto.test.ts
PRIVATE_KEY_PEM_BASE64 = 'LS0tLS1CRUdJTiBSU0EgUFJJVkFURSBLRVktLS0tLQpQcm9jLVR5cGU6IDQsRU5DUllQVEVECkRFSy1JbmZvOiBERVMtRURFMy1DQkMsNTM1QURERjIzNUQ4QkFGOQoKMXFWUzl0S0FhdkVhVUVFMktESnpjM3plMk1lZkc1dmVEd2o1UVJ0ZkRaMXdWNS9VZmIvcU5sVThTSjlNaGhKaQp6RFdrWExueUUzSW0vcEtITVZkS0czYWZkcFRtcis2TmtidXptd0dVMk0vSWpzRjRJZlpad0lGbGJoY09jUnp4CmZmWVIvTlVyaHNrS1RpNGhGV0lBUDlLb3Z6VDhPSzNZY3h6eVZQWUxYNGVWbWt3UmZzeWkwUU5Xb0tGT3d0ZC8KNm9HYzFnd2piRjI5ZDNnUThZQjFGWmRLa1AyMTJGbkt1cTIrUWgvbE1zTUZrTHlTQTRLTGJ3ZG1RSXExbE1QUwpjbUNtZnppV3J1MlBtNEZoM0dmWlQyaE1JWHlIRFdEVzlDTkxKaERodExOZ2RRamFBUFpVT1E4V2hwSkE5MS9vCjJLZzZ3MDd5Z2RCcVd5dTZrc0pXcjNpZ1JpUEJ5QmVNWEpEZU5HY3NhaUZ3Q2c5eFlja1VORXR3NS90WlRsTjIKSEdZV0NpVU5Ed0F2WllMUHR1SHpIOFRFMGxsZm5HR0VuVC9QQlp1UHV4andlZlRleE1mdzFpbGJRU3lkcy9HMgpOOUlKKzkydms0N0ZXR2NOdGh1Q3lCbklva0NpZ0c1ZlBlV2IwQTdpdjk0UGtwRTRJZ3plc0hGQ0ZFQWoxWldLCnpQdFRBQlkwZlJrUzBNc3UwMHYxOXloTTUrdFUwYkVCZWo2eWpzWHRoYzlwS01hcUNIZWlQTC9TSHRkaWsxNVMKQmU4Sml4dVJxZitUeGlYWWVuNTg2aDlzTFpEYzA3cGpkUGp2NVNYRnBYQjhIMlVxQ0tZY2p4R3RvQWpTV0pjWApMNHc3RHNEby80bVg1N0htR09iamlCN1ZyOGhVWEJDdFh2V0dmQXlmcEFZNS9vOXowdm4zREcxaDc1NVVwdDluCkF2MFZrbm9qcmJVYjM1ZlJuU1lYTVltS01LSnpNRlMrdmFvRlpwV0ZjTG10cFRWSWNzc0JGUEYyZEo3V1c0WHMKK0d2Vkl2eFl3S2wyZzFPTE1TTXRZa09vekdlblBXTzdIdU0yMUVKVGIvbHNEZ25GaTkrYWRGZHBLY3R2cm0zdgpmbW1HeG5pRmhLU05GU0xtNms5YStHL2pjK3NVQVBhb2FZNEQ3NHVGajh0WGp0eThFUHdRRGxVUGRVZld3SE9PClF3bVgyMys1REh4V0VoQy91Tm8yNHNNY2ZkQzFGZUpBV281bUNuVU5vUVVmMStNRDVhMzNJdDhhMmlrNUkxUWoKeSs1WGpRaG0xd3RBMWhWTWE4aUxBR0toT09lcFRuK1VBZHpyS0hvNjVtYzNKbGgvSFJDUXJabnVxWkErK0F2WgpjeWU0dWZGWC8xdmRQSTdLb2Q0MEdDM2dlQnhweFFNYnp1OFNUcGpOcElJRkJvRVc5dFRhemUzeHZXWnV6dDc0CnFjZS8xWURuUHBLeW5lM0xGMk94VWoyYWVYUW5YQkpYcGhTZTBVTGJMcWJtUll4bjJKWkl1d09RNHV5dm94NjUKdG9TWGNac054dUs4QTErZXNXR3JSN3pVc0djdU9QQTFERE9Ja2JjcGtmRUxMNjk4RTJRckdqTU9JWnhrcWdxZQoySE5VNktWRmV2NzdZeEJDbm1VcVdXZEhYMjcyU2NPMUYzdWpUdFVnRVBNWGN0aEdBckYzTWxEaUw1Q0k0RkhqCnhHc3pVemxzalRQTmpiY2MzdUE2MjVZS3VVZEI2c1h1Rk5NUHk5UDgwTzBpRWJGTXl3MWxmN2VpdFhvaUUxWVoKc3NhMDVxTUx4M3pPUXZTLzFDdFpqaFp4cVJMRW5pQ3NWa2JVRlVYclpodEU4dG94bGpWSUtpQ25qbitORmtqdwo2bTZ1anpBSytZZHd2Nk5WMFB4S0gwUk5NYVhwb1lmQk1oUmZ3dGlaS3V3Y2hyRFB5UEhBQ2J3WXNZOXdtUE9rCnpwdDNxWi9JdDVYTmVqNDI0RzAzcGpMbk1sd1B1T1VzYmFQUWQ2VHU4TFhsckZReUVjTXJDNHdjUTA1SzFVN3kKM1NNN3RFaTlnbjV3RjY1YVI5eEFBR0grTUtMMk5WNnQrUmlTazJVaWs1clNmeDE4Mk9wYmpSQ2grdmQ4UXhJdwotLS0tLUVORCBSU0EgUFJJVkFURSBLRVktLS0tLQo=' # noqa: E501
PRIVATE_KEY_PASSWORD = 'pwd1234'
PUBLICK_KEY_PEM_BASE64 = 'LS0tLS1CRUdJTiBQVUJMSUMgS0VZLS0tLS0KTUlJQklqQU5CZ2txaGtpRzl3MEJBUUVGQUFPQ0FROEFNSUlCQ2dLQ0FRRUF0dis3NlNXbklhOFFKWC94RUQxRQpYdnBBQmE3ajBnQnVYenJNUU5adjhtTW1RU0t2VUF0TmpOL2xacUZpQ0haZUQxU2VDcGV1MnFHTm5XbGRxNkhUCnh5cXJpTVZEbFNKaFBNT09QSENISVNVdFI4Tk5lR1Y1MU0wYkxJcENabHcyTU9GUjdqdENWejVqZFRpZ1NvYTIKQWxrRUlRZWQ4UVlDKzk1aGJoOHk5bGcwQ0JxdEdWN1FvMFZQR2xKQ0hGaWNuaWxLVFFZay9MZzkwWVFnUElPbwozbUppeFl5bWFGNmlMZTVXNzg1M0VHWUVFVWdlWmNaZFNjaGVBMEdBMGpRSFVTdnYvMEZjay9adkZNZURJOTVsCmJVQ0JoQjFDbFg4OG4wZUhzUmdWZE5vK0NLMDI4T2IvZTZTK1JLK09VaHlFRVdPTi90alVMdGhJdTJkQWtGcmkKOFFJREFRQUIKLS0tLS1FTkQgUFVCTElDIEtFWS0tLS0tCg==' # noqa: E501
PRIVATE_KEY = _load_private_key(
PRIVATE_KEY_PEM_BASE64,
PRIVATE_KEY_PASSWORD,
)
PUBLIC_KEY = _load_public_key(PUBLICK_KEY_PEM_BASE64)
drobnikj marked this conversation as resolved.
Show resolved Hide resolved


class TestCrypto():

def test_encrypt_decrypt_varions_string(self) -> None:
for value in [_crypto_random_object_id(10), '👍', '!', '@', '#', '$', '%', '^', '&', '*', '(', ')', '-', '_', '=', '+', '[', ']', '{', '}', '|', ';', ':', '"', "'", ',', '.', '<', '>', '?', '/', '~']: # noqa: E501
encrypted = public_encrypt(PUBLIC_KEY, value)
decrypted_value = private_decrypt(PRIVATE_KEY, **encrypted)
assert decrypted_value == value

def test_throw_if_password_is_not_valid(self) -> None:
test_value = 'test'
encrypted = public_encrypt(PUBLIC_KEY, test_value)
encrypted['encrypted_password'] = base64.b64encode(b'invalid_password').decode('utf-8')

with pytest.raises(ValueError):
private_decrypt(PRIVATE_KEY, **encrypted)

def test_throw_error_if_cipher_is_manipulated(self) -> None:
test_value = 'test2'
encrypted = public_encrypt(PUBLIC_KEY, test_value)
encrypted['encrypted_value'] = base64.b64encode(
b'invalid_cipher' + base64.b64decode(encrypted['encrypted_value'].encode('utf-8'))).decode('utf-8')

with pytest.raises(ValueError):
private_decrypt(PRIVATE_KEY, **encrypted)

def test_same_encrypted_value_should_return_deffirent_cipher(self) -> None:
test_value = 'test3'
encrypted1 = public_encrypt(PUBLIC_KEY, test_value)
encrypted2 = public_encrypt(PUBLIC_KEY, test_value)
assert encrypted1['encrypted_value'] != encrypted2['encrypted_value']

# Check if the method is compatible with js version of the same method in:
# https://github.com/apify/apify-shared-js/blob/master/packages/utilities/src/crypto.ts
def test_private_encrypt_node_js_encrypted_value(self) -> None:
value = 'encrypted_with_node_js'
# This was encrypted with nodejs version of the same method.
encrypted_value_with_node_js = {
'encrypted_password': 'lw0ez64/T1UcCQMLfhucZ6VIfMcf/TKni7PmXlL/ZRA4nmdGYz7/YQUzGWzKbLChrpqbG21DHxPIubUIQFDFE1ASkLvoSd0Ks8/wjKHMyhp+hsg5aSh9EZK6pBFpp6FeHoinV80+UURTvJuSVbWd1Orw5Frl41taP6RK3uNJlXikmgs8Xc7mShFEENgkz6y9+Pbe7jpcKkaJ2U/h7FN0eNON189kNFYVuAE1n2N6C3Q7dFnjl2e1btqErvg5Vu7ZS4BbX3wgC2qLYySGnqI3BNI5VGhAnncnQcjHb+85qG+LKoPekgY9I0s0kGMxiz/bmy1mYm9O+Lj1mbVUr7BDjQ==', # noqa: E501
'encrypted_value': 'k8nkZDCi0hRfBc0RRefxeSHeGV0X60N03VCrhRhENKXBjrF/tEg=',
}
encrypted_value = private_decrypt(
PRIVATE_KEY,
**encrypted_value_with_node_js,
)

assert encrypted_value == value
drobnikj marked this conversation as resolved.
Show resolved Hide resolved

def test__crypto_random_object_id(self) -> None:
assert len(_crypto_random_object_id()) == 17
assert len(_crypto_random_object_id(5)) == 5
long_random_object_id = _crypto_random_object_id(1000)
for char in long_random_object_id:
assert char in 'abcdefghijklmnopqrstuvwxyzABCEDFGHIJKLMNOPQRSTUVWXYZ0123456789'
9 changes: 0 additions & 9 deletions tests/unit/test_utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@
from aiofiles.os import mkdir

from apify._utils import (
_crypto_random_object_id,
_fetch_and_parse_env_var,
_filter_out_none_values_recursively,
_filter_out_none_values_recursively_internal,
Expand Down Expand Up @@ -315,11 +314,3 @@ async def test__force_rename(tmp_path: str) -> None:
assert os.path.exists(dst_file) is False
# src_dir.txt should exist in dst_dir
assert os.path.exists(os.path.join(dst_dir, 'src_dir.txt')) is True


def test__crypto_random_object_id() -> None:
assert len(_crypto_random_object_id()) == 17
assert len(_crypto_random_object_id(5)) == 5
long_random_object_id = _crypto_random_object_id(1000)
for char in long_random_object_id:
assert char in 'abcdefghijklmnopqrstuvwxyzABCEDFGHIJKLMNOPQRSTUVWXYZ0123456789'