Skip to content

Commit 1274739

Browse files
authored
Merge pull request #76 from opentensor/fix/thewhaleking/disk-cache-logic
Improves the logic of the disk cache so that it doesn't spill over
2 parents a07ca5c + ce1202a commit 1274739

File tree

2 files changed

+103
-22
lines changed

2 files changed

+103
-22
lines changed

async_substrate_interface/utils/cache.py

+29-22
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
import os
33
import pickle
44
import sqlite3
5+
from pathlib import Path
56
import asyncstdlib as a
67

78
USE_CACHE = True if os.getenv("NO_CACHE") != "1" else False
@@ -14,6 +15,12 @@
1415
)
1516

1617

18+
def _ensure_dir():
19+
path = Path(CACHE_LOCATION).parent
20+
if not path.exists():
21+
path.mkdir(parents=True, exist_ok=True)
22+
23+
1724
def _get_table_name(func):
1825
"""Convert "ClassName.method_name" to "ClassName_method_name"""
1926
return func.__qualname__.replace(".", "_")
@@ -74,22 +81,32 @@ def _insert_into_cache(c, conn, table_name, key, result, chain):
7481
pass
7582

7683

77-
def sql_lru_cache(maxsize=None):
78-
def decorator(func):
84+
def _shared_inner_fn_logic(func, self, args, kwargs):
85+
chain = self.url
86+
if not (local_chain := _check_if_local(chain)) or not USE_CACHE:
87+
_ensure_dir()
7988
conn = sqlite3.connect(CACHE_LOCATION)
8089
c = conn.cursor()
8190
table_name = _get_table_name(func)
8291
_create_table(c, conn, table_name)
92+
key = pickle.dumps((args, kwargs))
93+
result = _retrieve_from_cache(c, table_name, key, chain)
94+
else:
95+
result = None
96+
c = None
97+
conn = None
98+
table_name = None
99+
key = None
100+
return c, conn, table_name, key, result, chain, local_chain
101+
83102

103+
def sql_lru_cache(maxsize=None):
104+
def decorator(func):
84105
@functools.lru_cache(maxsize=maxsize)
85106
def inner(self, *args, **kwargs):
86-
c = conn.cursor()
87-
key = pickle.dumps((args, kwargs))
88-
chain = self.url
89-
if not (local_chain := _check_if_local(chain)) or not USE_CACHE:
90-
result = _retrieve_from_cache(c, table_name, key, chain)
91-
if result is not None:
92-
return result
107+
c, conn, table_name, key, result, chain, local_chain = (
108+
_shared_inner_fn_logic(func, self, args, kwargs)
109+
)
93110

94111
# If not in DB, call func and store in DB
95112
result = func(self, *args, **kwargs)
@@ -106,21 +123,11 @@ def inner(self, *args, **kwargs):
106123

107124
def async_sql_lru_cache(maxsize=None):
108125
def decorator(func):
109-
conn = sqlite3.connect(CACHE_LOCATION)
110-
c = conn.cursor()
111-
table_name = _get_table_name(func)
112-
_create_table(c, conn, table_name)
113-
114126
@a.lru_cache(maxsize=maxsize)
115127
async def inner(self, *args, **kwargs):
116-
c = conn.cursor()
117-
key = pickle.dumps((args, kwargs))
118-
chain = self.url
119-
120-
if not (local_chain := _check_if_local(chain)) or not USE_CACHE:
121-
result = _retrieve_from_cache(c, table_name, key, chain)
122-
if result is not None:
123-
return result
128+
c, conn, table_name, key, result, chain, local_chain = (
129+
_shared_inner_fn_logic(func, self, args, kwargs)
130+
)
124131

125132
# If not in DB, call func and store in DB
126133
result = await func(self, *args, **kwargs)
+74
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
import pytest
2+
3+
from async_substrate_interface.async_substrate import (
4+
DiskCachedAsyncSubstrateInterface,
5+
AsyncSubstrateInterface,
6+
)
7+
from async_substrate_interface.sync_substrate import SubstrateInterface
8+
9+
10+
@pytest.mark.asyncio
11+
async def test_disk_cache():
12+
entrypoint = "wss://entrypoint-finney.opentensor.ai:443"
13+
async with DiskCachedAsyncSubstrateInterface(entrypoint) as disk_cached_substrate:
14+
current_block = await disk_cached_substrate.get_block_number(None)
15+
block_hash = await disk_cached_substrate.get_block_hash(current_block)
16+
parent_block_hash = await disk_cached_substrate.get_parent_block_hash(
17+
block_hash
18+
)
19+
block_runtime_info = await disk_cached_substrate.get_block_runtime_info(
20+
block_hash
21+
)
22+
block_runtime_version_for = (
23+
await disk_cached_substrate.get_block_runtime_version_for(block_hash)
24+
)
25+
block_hash_from_cache = await disk_cached_substrate.get_block_hash(
26+
current_block
27+
)
28+
parent_block_hash_from_cache = (
29+
await disk_cached_substrate.get_parent_block_hash(block_hash_from_cache)
30+
)
31+
block_runtime_info_from_cache = (
32+
await disk_cached_substrate.get_block_runtime_info(block_hash_from_cache)
33+
)
34+
block_runtime_version_from_cache = (
35+
await disk_cached_substrate.get_block_runtime_version_for(
36+
block_hash_from_cache
37+
)
38+
)
39+
assert block_hash == block_hash_from_cache
40+
assert parent_block_hash == parent_block_hash_from_cache
41+
assert block_runtime_info == block_runtime_info_from_cache
42+
assert block_runtime_version_for == block_runtime_version_from_cache
43+
async with AsyncSubstrateInterface(entrypoint) as non_cache_substrate:
44+
block_hash_non_cache = await non_cache_substrate.get_block_hash(current_block)
45+
parent_block_hash_non_cache = await non_cache_substrate.get_parent_block_hash(
46+
block_hash_non_cache
47+
)
48+
block_runtime_info_non_cache = await non_cache_substrate.get_block_runtime_info(
49+
block_hash_non_cache
50+
)
51+
block_runtime_version_for_non_cache = (
52+
await non_cache_substrate.get_block_runtime_version_for(
53+
block_hash_non_cache
54+
)
55+
)
56+
assert block_hash == block_hash_non_cache
57+
assert parent_block_hash == parent_block_hash_non_cache
58+
assert block_runtime_info == block_runtime_info_non_cache
59+
assert block_runtime_version_for == block_runtime_version_for_non_cache
60+
with SubstrateInterface(entrypoint) as sync_substrate:
61+
block_hash_sync = sync_substrate.get_block_hash(current_block)
62+
parent_block_hash_sync = sync_substrate.get_parent_block_hash(
63+
block_hash_non_cache
64+
)
65+
block_runtime_info_sync = sync_substrate.get_block_runtime_info(
66+
block_hash_non_cache
67+
)
68+
block_runtime_version_for_sync = sync_substrate.get_block_runtime_version_for(
69+
block_hash_non_cache
70+
)
71+
assert block_hash == block_hash_sync
72+
assert parent_block_hash == parent_block_hash_sync
73+
assert block_runtime_info == block_runtime_info_sync
74+
assert block_runtime_version_for == block_runtime_version_for_sync

0 commit comments

Comments
 (0)