231 lines
7.1 KiB
Python
231 lines
7.1 KiB
Python
|
# standard imports
|
||
|
import logging
|
||
|
import json
|
||
|
import base64
|
||
|
import copy
|
||
|
import re
|
||
|
|
||
|
# external imports
|
||
|
import pytest
|
||
|
from hexathon import strip_0x
|
||
|
|
||
|
# local imports
|
||
|
from cic_cache.runnable.daemons.query import *
|
||
|
|
||
|
logg = logging.getLogger()
|
||
|
|
||
|
|
||
|
|
||
|
@pytest.mark.parametrize(
|
||
|
'query_path_prefix, query_role, query_address_index, query_offset, query_offset_index, query_limit, query_limit_index, match_re',
|
||
|
[
|
||
|
('/tx/user/', 'alice', 0, None, 3, None, 5, re_transactions_account_bloom),
|
||
|
('/tx/user/', 'alice', 0, 42, 3, None, 5, re_transactions_account_bloom),
|
||
|
('/tx/user/', 'alice', 0, 42, 3, 13, 5, re_transactions_account_bloom),
|
||
|
('/tx/', None, 0, None, 3, None, 5, re_transactions_all_bloom),
|
||
|
('/tx/', None, 0, 42, 3, None, 5, re_transactions_all_bloom),
|
||
|
('/tx/', None, 0, 42, 3, 13, 5, re_transactions_all_bloom),
|
||
|
('/txa/', None, 0, None, 3, None, 5, re_transactions_all_data),
|
||
|
('/txa/', None, 0, 42, 3, None, 5, re_transactions_all_data),
|
||
|
('/txa/', None, 0, 42, 3, 13, 5, re_transactions_all_data),
|
||
|
],
|
||
|
)
|
||
|
def test_query_regex(
|
||
|
list_actors,
|
||
|
query_path_prefix,
|
||
|
query_role,
|
||
|
query_address_index,
|
||
|
query_offset,
|
||
|
query_offset_index,
|
||
|
query_limit,
|
||
|
query_limit_index,
|
||
|
match_re,
|
||
|
):
|
||
|
|
||
|
paths = []
|
||
|
path = query_path_prefix
|
||
|
query_address = None
|
||
|
if query_role != None:
|
||
|
query_address = strip_0x(list_actors[query_role])
|
||
|
paths.append(path + '0x' + query_address)
|
||
|
paths.append(path + query_address)
|
||
|
if query_offset != None:
|
||
|
if query_limit != None:
|
||
|
for i in range(len(paths)-1):
|
||
|
paths[i] += '/{}/{}'.format(query_offset, query_limit)
|
||
|
else:
|
||
|
for i in range(len(paths)-1):
|
||
|
paths[i] += '/' + str(query_offset)
|
||
|
|
||
|
for i in range(len(paths)):
|
||
|
paths.append(paths[i] + '/')
|
||
|
|
||
|
for p in paths:
|
||
|
logg.debug('testing path {} against {}'.format(p, match_re))
|
||
|
m = re.match(match_re, p)
|
||
|
l = len(m.groups())
|
||
|
logg.debug('laast index match {} groups {}'.format(m.lastindex, l))
|
||
|
for i in range(l+1):
|
||
|
logg.debug('group {} {}'.format(i, m[i]))
|
||
|
if m.lastindex >= query_offset_index:
|
||
|
assert query_offset == int(m[query_offset_index + 1])
|
||
|
if m.lastindex >= query_limit_index:
|
||
|
assert query_limit == int(m[query_limit_index + 1])
|
||
|
if query_address_index != None:
|
||
|
match_address = strip_0x(m[query_address_index + 1])
|
||
|
assert query_address == match_address
|
||
|
|
||
|
|
||
|
|
||
|
@pytest.mark.parametrize(
|
||
|
'role_name, query_offset, query_limit, query_match',
|
||
|
[
|
||
|
('alice', None, None, [(420000, 13), (419999, 42)]),
|
||
|
('alice', None, 1, [(420000, 13)]),
|
||
|
('alice', 1, None, [(419999, 42)]), # 420000 == list_defaults['block']
|
||
|
('alice', 2, None, []), # 420000 == list_defaults['block']
|
||
|
],
|
||
|
)
|
||
|
def test_query_process_txs_account(
|
||
|
init_database,
|
||
|
list_defaults,
|
||
|
list_actors,
|
||
|
list_tokens,
|
||
|
txs,
|
||
|
zero_filter,
|
||
|
role_name,
|
||
|
query_offset,
|
||
|
query_limit,
|
||
|
query_match,
|
||
|
):
|
||
|
|
||
|
actor = None
|
||
|
try:
|
||
|
actor = list_actors[role_name]
|
||
|
except KeyError:
|
||
|
actor = os.urandom(20).hex()
|
||
|
path_info = '/tx/user/0x' + strip_0x(actor)
|
||
|
if query_offset != None:
|
||
|
path_info += '/' + str(query_offset)
|
||
|
if query_limit != None:
|
||
|
if query_offset == None:
|
||
|
path_info += '/0'
|
||
|
path_info += '/' + str(query_limit)
|
||
|
env = {
|
||
|
'PATH_INFO': path_info,
|
||
|
}
|
||
|
logg.debug('using path {}'.format(path_info))
|
||
|
r = process_transactions_account_bloom(init_database, env)
|
||
|
assert r != None
|
||
|
|
||
|
o = json.loads(r[1])
|
||
|
block_filter_data = base64.b64decode(o['block_filter'].encode('utf-8'))
|
||
|
zero_filter_data = zero_filter.to_bytes()
|
||
|
if len(query_match) == 0:
|
||
|
assert block_filter_data == zero_filter_data
|
||
|
return
|
||
|
|
||
|
assert block_filter_data != zero_filter_data
|
||
|
block_filter = copy.copy(zero_filter)
|
||
|
block_filter.merge(block_filter_data)
|
||
|
block_filter_data = block_filter.to_bytes()
|
||
|
assert block_filter_data != zero_filter_data
|
||
|
|
||
|
for (block, tx) in query_match:
|
||
|
block = block.to_bytes(4, byteorder='big')
|
||
|
assert block_filter.check(block)
|
||
|
|
||
|
|
||
|
@pytest.mark.parametrize(
|
||
|
'query_offset, query_limit, query_match',
|
||
|
[
|
||
|
(None, 2, [(420000, 13), (419999, 42)]),
|
||
|
(0, 1, [(420000, 13)]),
|
||
|
(1, 1, [(419999, 42)]),
|
||
|
(2, 0, []),
|
||
|
],
|
||
|
)
|
||
|
def test_query_process_txs_bloom(
|
||
|
init_database,
|
||
|
list_defaults,
|
||
|
list_actors,
|
||
|
list_tokens,
|
||
|
txs,
|
||
|
zero_filter,
|
||
|
query_offset,
|
||
|
query_limit,
|
||
|
query_match,
|
||
|
):
|
||
|
|
||
|
path_info = '/tx'
|
||
|
if query_offset != None:
|
||
|
path_info += '/' + str(query_offset)
|
||
|
if query_limit != None:
|
||
|
if query_offset == None:
|
||
|
path_info += '/0'
|
||
|
path_info += '/' + str(query_limit)
|
||
|
env = {
|
||
|
'PATH_INFO': path_info,
|
||
|
}
|
||
|
logg.debug('using path {}'.format(path_info))
|
||
|
r = process_transactions_all_bloom(init_database, env)
|
||
|
assert r != None
|
||
|
|
||
|
o = json.loads(r[1])
|
||
|
block_filter_data = base64.b64decode(o['block_filter'].encode('utf-8'))
|
||
|
zero_filter_data = zero_filter.to_bytes()
|
||
|
if len(query_match) == 0:
|
||
|
assert block_filter_data == zero_filter_data
|
||
|
return
|
||
|
|
||
|
assert block_filter_data != zero_filter_data
|
||
|
block_filter = copy.copy(zero_filter)
|
||
|
block_filter.merge(block_filter_data)
|
||
|
block_filter_data = block_filter.to_bytes()
|
||
|
assert block_filter_data != zero_filter_data
|
||
|
|
||
|
for (block, tx) in query_match:
|
||
|
block = block.to_bytes(4, byteorder='big')
|
||
|
assert block_filter.check(block)
|
||
|
|
||
|
|
||
|
@pytest.mark.parametrize(
|
||
|
'query_block_start, query_block_end, query_match_count',
|
||
|
[
|
||
|
(None, 42, 0),
|
||
|
(420000, 420001, 1),
|
||
|
(419999, 419999, 1), # matches are inclusive
|
||
|
(419999, 420000, 2),
|
||
|
(419999, 420001, 2),
|
||
|
],
|
||
|
)
|
||
|
def test_query_process_txs_data(
|
||
|
init_database,
|
||
|
list_defaults,
|
||
|
list_actors,
|
||
|
list_tokens,
|
||
|
txs,
|
||
|
zero_filter,
|
||
|
query_block_start,
|
||
|
query_block_end,
|
||
|
query_match_count,
|
||
|
):
|
||
|
|
||
|
path_info = '/txa'
|
||
|
if query_block_start != None:
|
||
|
path_info += '/' + str(query_block_start)
|
||
|
if query_block_end != None:
|
||
|
if query_block_start == None:
|
||
|
path_info += '/0'
|
||
|
path_info += '/' + str(query_block_end)
|
||
|
env = {
|
||
|
'PATH_INFO': path_info,
|
||
|
'HTTP_X_CIC_CACHE_MODE': 'all',
|
||
|
}
|
||
|
logg.debug('using path {}'.format(path_info))
|
||
|
r = process_transactions_all_data(init_database, env)
|
||
|
assert r != None
|
||
|
|
||
|
o = json.loads(r[1])
|
||
|
assert len(o['data']) == query_match_count
|