mirror of
https://github.com/RoboSats/robosats.git
synced 2025-01-18 12:11:35 +00:00
Add django utils tests (#911)
* Add util tests * Enable coordinator test workflow
This commit is contained in:
parent
e0fed23377
commit
54a59872fb
@ -1,7 +1,7 @@
|
||||
# Coordinator Alias (Same as longAlias)
|
||||
COORDINATOR_ALIAS="Local Dev"
|
||||
# Lightning node vendor: CLN | LND
|
||||
LNVENDOR='CLN'
|
||||
LNVENDOR='LND'
|
||||
|
||||
# LND directory to read TLS cert and macaroon
|
||||
LND_DIR='/lnd/'
|
||||
|
47
.github/workflows/django-test.yml
vendored
47
.github/workflows/django-test.yml
vendored
@ -6,13 +6,13 @@ on:
|
||||
push:
|
||||
branches: [ "main" ]
|
||||
paths: ["api", "chat", "control", "robosats"]
|
||||
pull_request:
|
||||
pull_request_target:
|
||||
branches: [ "main" ]
|
||||
paths: ["api", "chat", "control", "robosats"]
|
||||
|
||||
# concurrency:
|
||||
# group: '${{ github.workflow }} @ ${{ github.event.pull_request.head.label || github.head_ref || github.ref }}'
|
||||
# cancel-in-progress: true
|
||||
concurrency:
|
||||
group: '${{ github.workflow }} @ ${{ github.event.pull_request.head.label || github.head_ref || github.ref }}'
|
||||
cancel-in-progress: true
|
||||
|
||||
jobs:
|
||||
build:
|
||||
@ -22,24 +22,55 @@ jobs:
|
||||
strategy:
|
||||
max-parallel: 4
|
||||
matrix:
|
||||
python-version: ["3.11"]
|
||||
python-version: ["3.11.6"] # , "3.12"]
|
||||
|
||||
services:
|
||||
db:
|
||||
image: postgres:14.2
|
||||
env:
|
||||
POSTGRES_DB: postgres
|
||||
POSTGRES_USER: postgres
|
||||
POSTGRES_PASSWORD: example
|
||||
ports:
|
||||
- 5432:5432
|
||||
options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5
|
||||
|
||||
steps:
|
||||
- name: 'Checkout'
|
||||
uses: actions/checkout@v4
|
||||
|
||||
- name: 'Set up Python ${{ matrix.python-version }}'
|
||||
uses: actions/setup-python@v4
|
||||
with:
|
||||
python-version: ${{ matrix.python-version }}
|
||||
|
||||
- name: 'Cache pip dependencies'
|
||||
uses: actions/cache@v2
|
||||
with:
|
||||
path: ~/.cache/pip
|
||||
key: ${{ runner.os }}-pip-${{ hashFiles('**/requirements.txt') }}
|
||||
restore-keys: |
|
||||
${{ runner.os }}-pip-
|
||||
|
||||
- name: 'Install Python Dependencies'
|
||||
run: |
|
||||
python -m pip install --upgrade pip
|
||||
pip install -r requirements.txt
|
||||
- name: 'Install LND gRPC Dependencies'
|
||||
|
||||
- name: 'Install LND/CLN gRPC Dependencies'
|
||||
run: bash ./scripts/generate_grpc.sh
|
||||
|
||||
- name: 'Create .env File'
|
||||
run: |
|
||||
mv .env-sample .env
|
||||
- name: 'Tests'
|
||||
|
||||
- name: 'Wait for PostgreSQL to become ready'
|
||||
run: |
|
||||
python manage.py test
|
||||
sudo apt-get install -y postgresql-client
|
||||
until pg_isready -h localhost -p 5432 -U postgres; do sleep 2; done
|
||||
|
||||
- name: 'Run tests with coverage'
|
||||
run: |
|
||||
pip install coverage
|
||||
coverage run manage.py test
|
||||
coverage report
|
2
.github/workflows/py-linter.yml
vendored
2
.github/workflows/py-linter.yml
vendored
@ -24,7 +24,7 @@ jobs:
|
||||
- uses: actions/checkout@v4
|
||||
- uses: actions/setup-python@v4
|
||||
with:
|
||||
python-version: '3.10'
|
||||
python-version: '3.11.6'
|
||||
cache: pip
|
||||
- run: pip install black==22.8.0 flake8==5.0.4 isort==5.10.1
|
||||
- name: Run linters
|
||||
|
6
.github/workflows/release.yml
vendored
6
.github/workflows/release.yml
vendored
@ -38,9 +38,9 @@ jobs:
|
||||
fi
|
||||
|
||||
|
||||
# django-test:
|
||||
# uses: RoboSats/robosats/.github/workflows/django-test.yml@main
|
||||
# needs: check-versions
|
||||
django-test:
|
||||
uses: RoboSats/robosats/.github/workflows/django-test.yml@main
|
||||
needs: check-versions
|
||||
|
||||
frontend-build:
|
||||
uses: RoboSats/robosats/.github/workflows/frontend-build.yml@main
|
||||
|
@ -10,10 +10,10 @@ import ring
|
||||
from decouple import config
|
||||
from django.utils import timezone
|
||||
|
||||
from . import node_pb2 as noderpc
|
||||
from . import node_pb2_grpc as nodestub
|
||||
from . import hold_pb2 as holdrpc
|
||||
from . import hold_pb2_grpc as holdstub
|
||||
from . import node_pb2 as noderpc
|
||||
from . import node_pb2_grpc as nodestub
|
||||
from . import primitives_pb2 as primitives__pb2
|
||||
|
||||
#######
|
||||
|
@ -1,3 +0,0 @@
|
||||
# from django.test import TestCase
|
||||
|
||||
# Create your tests here.
|
0
api/tests/__init__.py
Normal file
0
api/tests/__init__.py
Normal file
18
api/tests/test_enc_priv_key
Normal file
18
api/tests/test_enc_priv_key
Normal file
@ -0,0 +1,18 @@
|
||||
-----BEGIN PGP PRIVATE KEY BLOCK-----
|
||||
|
||||
xYYEZTWJ1xYJKwYBBAHaRw8BAQdAsfdKb90BurKniu+pBPBDHCkzg08S51W0
|
||||
mUR0SKqLmdj+CQMICrS3TNCA/LHgxckC+iTUMxkqQJ9GpXWCDacx1rBQCztu
|
||||
PDgUHNvWdcvW1wWVxU/aJaQLqBTtRVYkJTz332jrKvsSl/LnrfwmUfKgN4nG
|
||||
Oc1MUm9ib1NhdHMgSUQgNTUyZGQxYTY2MWE3YWNhNGE0MWY4ODkxMGZmMzRh
|
||||
YzNiMWFjODBiYjc3OTRlZDlmZDU1ZDhiNzZiOTdhYWQ5N8KMBBAWCgA+BYJl
|
||||
NYnXBAsJBwgJkDc3tq7iCL/OAxUICgQWAAIBAhkBApsDAh4BFiEE7mIEuePQ
|
||||
n8Tqye0QNze2ruIIv84AAORXAQD+0G30U+qvEsc48W3JTKmmmvT+3q+uXdGs
|
||||
vMIuqoTWdQD/R+aGKvYXzwpyzGYInAAOj8WCXBcYWSOtXiw/oWNgEQHHiwRl
|
||||
NYnXEgorBgEEAZdVAQUBAQdALJQh7exBNphr8gU0oY2ZfP/6Gse6Ryi3hM6+
|
||||
LZsP2nwDAQgH/gkDCPPoYWyzm4mT4N/TDBF11GVq0xSEEcubFqjArFKyibRy
|
||||
TDnB8+o8BlkRuGClcfRyKkR5/Rp1v5B0n1BuMsc8nY4Yg4BJv4KhsPfXRp4m
|
||||
31zCeAQYFggAKgWCZTWJ1wmQNze2ruIIv84CmwwWIQTuYgS549CfxOrJ7RA3
|
||||
N7au4gi/zgAAmhQA+wRtxO05FKFirSXuTj808F/mzF8u4Wz0kMNMzPUXvCj9
|
||||
AQCHkxvHdvNHUyut4f7HkACnjI9M3+wGF69ArHL0u39/Aw==
|
||||
=1hCT
|
||||
-----END PGP PRIVATE KEY BLOCK-----
|
14
api/tests/test_pub_key
Normal file
14
api/tests/test_pub_key
Normal file
@ -0,0 +1,14 @@
|
||||
-----BEGIN PGP PUBLIC KEY BLOCK-----
|
||||
|
||||
mDMEZTWJ1xYJKwYBBAHaRw8BAQdAsfdKb90BurKniu+pBPBDHCkzg08S51W0mUR0
|
||||
SKqLmdi0TFJvYm9TYXRzIElEIDU1MmRkMWE2NjFhN2FjYTRhNDFmODg5MTBmZjM0
|
||||
YWMzYjFhYzgwYmI3Nzk0ZWQ5ZmQ1NWQ4Yjc2Yjk3YWFkOTeIjAQQFgoAPgWCZTWJ
|
||||
1wQLCQcICZA3N7au4gi/zgMVCAoEFgACAQIZAQKbAwIeARYhBO5iBLnj0J/E6snt
|
||||
EDc3tq7iCL/OAADkVwEA/tBt9FPqrxLHOPFtyUypppr0/t6vrl3RrLzCLqqE1nUA
|
||||
/0fmhir2F88KcsxmCJwADo/FglwXGFkjrV4sP6FjYBEBuDgEZTWJ1xIKKwYBBAGX
|
||||
VQEFAQEHQCyUIe3sQTaYa/IFNKGNmXz/+hrHukcot4TOvi2bD9p8AwEIB4h4BBgW
|
||||
CAAqBYJlNYnXCZA3N7au4gi/zgKbDBYhBO5iBLnj0J/E6sntEDc3tq7iCL/OAACa
|
||||
FAD7BG3E7TkUoWKtJe5OPzTwX+bMXy7hbPSQw0zM9Re8KP0BAIeTG8d280dTK63h
|
||||
/seQAKeMj0zf7AYXr0CscvS7f38D
|
||||
=+xY8
|
||||
-----END PGP PUBLIC KEY BLOCK-----
|
11
api/tests/test_signed_message
Normal file
11
api/tests/test_signed_message
Normal file
@ -0,0 +1,11 @@
|
||||
-----BEGIN PGP SIGNED MESSAGE-----
|
||||
Hash: SHA512
|
||||
|
||||
test
|
||||
-----BEGIN PGP SIGNATURE-----
|
||||
|
||||
wnUEARYKACcFgmU22/EJkDc3tq7iCL/OFiEE7mIEuePQn8Tqye0QNze2ruII
|
||||
v84AAJDMAP9JXQJNRYUiPaSroIfmfJccPQeaVuHTnl0fJqLToL6GbAD/Rt7c
|
||||
Y67Co6RJi70vytMorPKWmiX6C/mrnKL0auQC8gQ=
|
||||
=1ouc
|
||||
-----END PGP SIGNATURE-----
|
262
api/tests/test_utils.py
Normal file
262
api/tests/test_utils.py
Normal file
@ -0,0 +1,262 @@
|
||||
from unittest.mock import MagicMock, Mock, mock_open, patch
|
||||
|
||||
import numpy as np
|
||||
from decouple import config
|
||||
from django.test import TestCase
|
||||
|
||||
from api.models import Order
|
||||
from api.utils import (
|
||||
base91_to_hex,
|
||||
bitcoind_rpc,
|
||||
compute_premium_percentile,
|
||||
get_cln_version,
|
||||
get_exchange_rates,
|
||||
get_lnd_version,
|
||||
get_robosats_commit,
|
||||
get_session,
|
||||
hex_to_base91,
|
||||
is_valid_token,
|
||||
objects_to_hyperlinks,
|
||||
validate_onchain_address,
|
||||
validate_pgp_keys,
|
||||
verify_signed_message,
|
||||
weighted_median,
|
||||
)
|
||||
|
||||
|
||||
class TestUtils(TestCase):
|
||||
@patch("api.utils.config")
|
||||
@patch("api.utils.requests.session")
|
||||
def test_get_session(self, mock_session, mock_config):
|
||||
mock_config.return_value = True
|
||||
session = get_session()
|
||||
self.assertEqual(session, mock_session.return_value)
|
||||
|
||||
@patch("api.utils.config")
|
||||
@patch("api.utils.requests.post")
|
||||
def test_bitcoind_rpc(self, mock_post, mock_config):
|
||||
mock_config.side_effect = ["url", "user", "password"]
|
||||
mock_post.return_value.json.return_value = {"result": "response"}
|
||||
response = bitcoind_rpc("method", ["params"])
|
||||
self.assertEqual(response, "response")
|
||||
|
||||
@patch("api.utils.bitcoind_rpc")
|
||||
def test_validate_onchain_address(self, mock_bitcoind_rpc):
|
||||
mock_bitcoind_rpc.return_value = {"isvalid": True}
|
||||
result, error = validate_onchain_address("address")
|
||||
self.assertTrue(result)
|
||||
self.assertIsNone(error)
|
||||
|
||||
@patch("api.utils.config")
|
||||
@patch("api.utils.get_session")
|
||||
def test_get_exchange_rates(self, mock_get_session, mock_config):
|
||||
# Mock the config function to return the list of API URLs
|
||||
mock_config.return_value = [
|
||||
"https://api.yadio.io/exrates/BTC",
|
||||
"https://blockchain.info/ticker",
|
||||
]
|
||||
|
||||
# Mock the get_session function to return a mock session object
|
||||
mock_session = mock_get_session.return_value
|
||||
|
||||
# Mock the get method of the session object to return a mock response
|
||||
mock_response_blockchain = Mock()
|
||||
mock_response_yadio = Mock()
|
||||
mock_session.get.side_effect = [mock_response_yadio, mock_response_blockchain]
|
||||
|
||||
# Mock the json method of the response object to return a dictionary of exchange rates
|
||||
mock_response_blockchain.json.return_value = {
|
||||
"USD": {
|
||||
"15m": 10001,
|
||||
"last": 10001,
|
||||
"buy": 10001,
|
||||
"sell": 10001,
|
||||
"symbol": "USD",
|
||||
}
|
||||
}
|
||||
mock_response_yadio.json.return_value = {"BTC": {"USD": 10000}}
|
||||
|
||||
# Call the get_exchange_rates function with a list of currencies
|
||||
currencies = ["USD"]
|
||||
rates = get_exchange_rates(currencies)
|
||||
|
||||
# Assert that the function returns a list of exchange rates
|
||||
self.assertIsInstance(rates, list)
|
||||
self.assertEqual(len(rates), len(currencies))
|
||||
self.assertEqual(
|
||||
rates[0], np.median([10000, 10001])
|
||||
) # Check if the median is correctly calculated
|
||||
|
||||
# Assert that the get method of the session object was called with the correct arguments
|
||||
mock_session.get.assert_any_call("https://blockchain.info/ticker")
|
||||
mock_session.get.assert_any_call("https://api.yadio.io/exrates/BTC")
|
||||
|
||||
# Assert that the json method of the response object was called
|
||||
mock_response_blockchain.json.assert_called_once()
|
||||
mock_response_yadio.json.assert_called_once()
|
||||
|
||||
LNVENDOR = config("LNVENDOR", cast=str, default="LND")
|
||||
|
||||
if LNVENDOR == "LND":
|
||||
|
||||
@patch("api.lightning.lnd.LNDNode.get_version")
|
||||
def test_get_lnd_version(self, mock_get_version):
|
||||
mock_get_version.return_value = "v0.17.0-beta"
|
||||
version = get_lnd_version()
|
||||
self.assertEqual(version, "v0.17.0-beta")
|
||||
|
||||
elif LNVENDOR == "CLN":
|
||||
|
||||
@patch("api.lightning.cln.CLNNode.get_version")
|
||||
def test_get_cln_version(self, mock_get_version):
|
||||
mock_get_version.return_value = "v23.08.1"
|
||||
version = get_cln_version()
|
||||
self.assertEqual(version, "v23.08.1")
|
||||
|
||||
@patch("builtins.open", new_callable=mock_open, read_data="test_commit_hash")
|
||||
def test_get_robosats_commit(self, mock_file):
|
||||
# Call the get_robosats_commit function
|
||||
commit_hash = get_robosats_commit()
|
||||
|
||||
# Assert that the function returns a string
|
||||
self.assertIsInstance(commit_hash, str)
|
||||
|
||||
# Assert that the open function was called with the correct arguments
|
||||
mock_file.assert_called_once_with("commit_sha")
|
||||
|
||||
# Assert that the read method of the file object was called
|
||||
mock_file().read.assert_called_once()
|
||||
|
||||
@patch("api.utils.Order.objects.filter")
|
||||
def test_compute_premium_percentile(self, mock_filter):
|
||||
# Mock the filter method to return a mock queryset
|
||||
mock_queryset = MagicMock()
|
||||
mock_filter.return_value = mock_queryset
|
||||
|
||||
# Mock the exclude method of the queryset to return the same mock queryset
|
||||
mock_queryset.exclude.return_value = mock_queryset
|
||||
|
||||
# Mock the count method of the queryset to return a specific number
|
||||
mock_queryset.count.return_value = 2
|
||||
|
||||
# Mock the order object
|
||||
order = MagicMock()
|
||||
order.currency = "USD"
|
||||
order.status = Order.Status.PUB
|
||||
order.type = "type"
|
||||
order.id = 1
|
||||
order.amount = 1000
|
||||
order.has_range = False
|
||||
order.max_amount = 2000
|
||||
order.last_satoshis = 10000
|
||||
|
||||
# Call the compute_premium_percentile function with the mock order object
|
||||
percentile = compute_premium_percentile(order)
|
||||
|
||||
# Assert that the function returns a float
|
||||
self.assertIsInstance(percentile, float)
|
||||
|
||||
# Assert that the filter method of the queryset was called with the correct arguments
|
||||
mock_filter.assert_called_once_with(
|
||||
currency=order.currency, status=Order.Status.PUB, type=order.type
|
||||
)
|
||||
|
||||
# Assert that the exclude method of the queryset was called with the correct arguments
|
||||
mock_queryset.exclude.assert_called_once_with(id=order.id)
|
||||
|
||||
def test_weighted_median(self):
|
||||
values = [1, 2, 3, 4, 5]
|
||||
weights = [1, 1, 1, 1, 1]
|
||||
median = weighted_median(values, sample_weight=weights)
|
||||
self.assertEqual(median, 3)
|
||||
|
||||
def test_validate_pgp_keys(self):
|
||||
# Example test client generated GPG keys
|
||||
client_pub_key = r"-----BEGIN PGP PUBLIC KEY BLOCK-----\\xjMEZTWJ1xYJKwYBBAHaRw8BAQdAsfdKb90BurKniu+pBPBDHCkzg08S51W0\mUR0SKqLmdjNTFJvYm9TYXRzIElEIDU1MmRkMWE2NjFhN2FjYTRhNDFmODg5\MTBmZjM0YWMzYjFhYzgwYmI3Nzk0ZWQ5ZmQ1NWQ4Yjc2Yjk3YWFkOTfCjAQQ\FgoAPgWCZTWJ1wQLCQcICZA3N7au4gi/zgMVCAoEFgACAQIZAQKbAwIeARYh\BO5iBLnj0J/E6sntEDc3tq7iCL/OAADkVwEA/tBt9FPqrxLHOPFtyUypppr0\/t6vrl3RrLzCLqqE1nUA/0fmhir2F88KcsxmCJwADo/FglwXGFkjrV4sP6Fj\YBEBzjgEZTWJ1xIKKwYBBAGXVQEFAQEHQCyUIe3sQTaYa/IFNKGNmXz/+hrH\ukcot4TOvi2bD9p8AwEIB8J4BBgWCAAqBYJlNYnXCZA3N7au4gi/zgKbDBYh\BO5iBLnj0J/E6sntEDc3tq7iCL/OAACaFAD7BG3E7TkUoWKtJe5OPzTwX+bM\Xy7hbPSQw0zM9Re8KP0BAIeTG8d280dTK63h/seQAKeMj0zf7AYXr0CscvS7\f38D\=h03E\-----END PGP PUBLIC KEY BLOCK-----"
|
||||
client_enc_priv_key = r"-----BEGIN PGP PRIVATE KEY BLOCK-----\\xYYEZTWJ1xYJKwYBBAHaRw8BAQdAsfdKb90BurKniu+pBPBDHCkzg08S51W0\mUR0SKqLmdj+CQMICrS3TNCA/LHgxckC+iTUMxkqQJ9GpXWCDacx1rBQCztu\PDgUHNvWdcvW1wWVxU/aJaQLqBTtRVYkJTz332jrKvsSl/LnrfwmUfKgN4nG\Oc1MUm9ib1NhdHMgSUQgNTUyZGQxYTY2MWE3YWNhNGE0MWY4ODkxMGZmMzRh\YzNiMWFjODBiYjc3OTRlZDlmZDU1ZDhiNzZiOTdhYWQ5N8KMBBAWCgA+BYJl\NYnXBAsJBwgJkDc3tq7iCL/OAxUICgQWAAIBAhkBApsDAh4BFiEE7mIEuePQ\n8Tqye0QNze2ruIIv84AAORXAQD+0G30U+qvEsc48W3JTKmmmvT+3q+uXdGs\vMIuqoTWdQD/R+aGKvYXzwpyzGYInAAOj8WCXBcYWSOtXiw/oWNgEQHHiwRl\NYnXEgorBgEEAZdVAQUBAQdALJQh7exBNphr8gU0oY2ZfP/6Gse6Ryi3hM6+\LZsP2nwDAQgH/gkDCPPoYWyzm4mT4N/TDBF11GVq0xSEEcubFqjArFKyibRy\TDnB8+o8BlkRuGClcfRyKkR5/Rp1v5B0n1BuMsc8nY4Yg4BJv4KhsPfXRp4m\31zCeAQYFggAKgWCZTWJ1wmQNze2ruIIv84CmwwWIQTuYgS549CfxOrJ7RA3\N7au4gi/zgAAmhQA+wRtxO05FKFirSXuTj808F/mzF8u4Wz0kMNMzPUXvCj9\AQCHkxvHdvNHUyut4f7HkACnjI9M3+wGF69ArHL0u39/Aw==\=1hCT\-----END PGP PRIVATE KEY BLOCK-----"
|
||||
|
||||
# Example valid formatted GPG keys
|
||||
with open("api/tests/test_pub_key", "r") as file:
|
||||
# Read the contents of the file
|
||||
pub_key = file.read()
|
||||
with open("api/tests/test_enc_priv_key", "r") as file:
|
||||
# Read the contents of the file
|
||||
enc_priv_key = file.read()
|
||||
|
||||
# Test for success
|
||||
is_valid, error, returned_pub_key, returned_enc_priv_key = validate_pgp_keys(
|
||||
client_pub_key, client_enc_priv_key
|
||||
)
|
||||
self.assertTrue(is_valid)
|
||||
self.assertIsNone(error)
|
||||
self.assertEqual(returned_pub_key, pub_key)
|
||||
self.assertEqual(returned_enc_priv_key, enc_priv_key)
|
||||
|
||||
# Test for failure
|
||||
is_valid, error, returned_pub_key, returned_enc_priv_key = validate_pgp_keys(
|
||||
client_pub_key[:50], client_enc_priv_key + "invalid"
|
||||
)
|
||||
self.assertFalse(is_valid)
|
||||
self.assertIsNotNone(error)
|
||||
self.assertIsNone(returned_pub_key)
|
||||
self.assertIsNone(returned_enc_priv_key)
|
||||
|
||||
def test_verify_signed_message(self):
|
||||
# Call the verify_signed_message function with a mock public key and a mock signed message
|
||||
with open("api/tests/test_pub_key", "r") as file:
|
||||
# Read the contents of the file
|
||||
pub_key = file.read()
|
||||
with open("api/tests/test_signed_message", "r") as file:
|
||||
# Read the contents of the file
|
||||
signed_message = file.read()
|
||||
|
||||
valid, message = verify_signed_message(pub_key, signed_message)
|
||||
|
||||
# Assert that the function returns True and a string
|
||||
self.assertTrue(valid)
|
||||
self.assertIsInstance(message, str)
|
||||
|
||||
unsigned_message = "This message is unsigned cleartext"
|
||||
valid, message = verify_signed_message(pub_key, unsigned_message)
|
||||
|
||||
# Assert that the function returns False and None tuple of a boolean and a string
|
||||
self.assertFalse(valid)
|
||||
self.assertIsNone(message)
|
||||
|
||||
def test_base91_to_hex(self):
|
||||
base91_str = "base91_string"
|
||||
with patch("api.utils.decode") as mock_decode:
|
||||
mock_decode.return_value = b"hex_string"
|
||||
hex_str = base91_to_hex(base91_str)
|
||||
self.assertEqual(hex_str, "6865785f737472696e67") # 'hex_string' in hex
|
||||
|
||||
def test_hex_to_base91(self):
|
||||
hex_str = "6865785f737472696e67" # 'hex_string' in hex
|
||||
with patch("api.utils.encode") as mock_encode:
|
||||
mock_encode.return_value = "base91_string"
|
||||
base91_str = hex_to_base91(hex_str)
|
||||
self.assertEqual(base91_str, "base91_string")
|
||||
|
||||
def test_is_valid_token(self):
|
||||
valid_token_1 = "Tl1S(#SvZ&I$sF9w=qQ|lG<8!JAqT8d}~jnVXX4E"
|
||||
valid_token_2 = '8Wo`Vy*<DiJ"kGO2v1xhJkVW]b{NPITio:s2<JkC'
|
||||
valid_token_3 = "c3g{U#M(.:OIEt)(hj6<Zk3/z`5MMMxu:v,V^R[C"
|
||||
|
||||
self.assertTrue(is_valid_token(valid_token_1))
|
||||
self.assertTrue(is_valid_token(valid_token_2))
|
||||
self.assertTrue(is_valid_token(valid_token_3))
|
||||
|
||||
invalid_token_1 = "Tl1S(#SvZ&I$sF9w=qQ|lG<8!JAqT8d}~jnVXX4EA"
|
||||
invalid_token_2 = "Tl1S(#SvZ&I$sF9w=qQ|lG<8!JAqT8d}~jnVXX"
|
||||
invalid_token_3 = " ,B,,NU|(_Ba"
|
||||
|
||||
self.assertFalse(is_valid_token(invalid_token_1))
|
||||
self.assertFalse(is_valid_token(invalid_token_2))
|
||||
self.assertFalse(is_valid_token(invalid_token_3))
|
||||
|
||||
def test_objects_to_hyperlinks(self):
|
||||
logs = "Robot(1, robot_name)"
|
||||
linked_logs = objects_to_hyperlinks(logs)
|
||||
self.assertEqual(
|
||||
linked_logs, '<b><a href="/coordinator/api/robot/1">robot_name</a></b>'
|
||||
)
|
Loading…
Reference in New Issue
Block a user