-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #10 from opentensor/release/0.1.0
Release/0.1.0
- Loading branch information
Showing
5 changed files
with
284 additions
and
26 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,60 @@ | ||
version: 2.1 | ||
|
||
orbs: | ||
python: circleci/[email protected] | ||
|
||
jobs: | ||
ruff: | ||
resource_class: small | ||
parameters: | ||
python-version: | ||
type: string | ||
docker: | ||
- image: cimg/python:<< parameters.python-version >> | ||
steps: | ||
- checkout | ||
- run: | ||
name: Install Ruff | ||
command: pip install ruff | ||
- run: | ||
name: Run Ruff | ||
command: ruff check . | ||
|
||
build-and-test: | ||
resource_class: medium | ||
parallelism: 2 | ||
parameters: | ||
python-version: | ||
type: string | ||
docker: | ||
- image: cimg/python:<< parameters.python-version >> | ||
steps: | ||
- checkout | ||
- run: | ||
name: Set Up Virtual Environment | ||
command: | | ||
curl https://sh.rustup.rs -sSf | sh -s -- -y | ||
. "$HOME/.cargo/env" | ||
python -m venv .venv | ||
. .venv/bin/activate | ||
python -m pip install --upgrade pip | ||
python -m pip install '.[dev]' | ||
- run: | ||
name: Run Tests | ||
command: | | ||
. .venv/bin/activate | ||
pytest tests/ | ||
- store_test_results: | ||
path: test-results | ||
- store_artifacts: | ||
path: test-results | ||
|
||
workflows: | ||
test-and-lint: | ||
jobs: | ||
- ruff: | ||
python-version: "3.9.13" | ||
- build-and-test: | ||
matrix: | ||
parameters: | ||
python-version: ["3.9", "3.10", "3.11", "3.12"] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,197 @@ | ||
import pytest | ||
import time | ||
from bittensor_commit_reveal import get_encrypted_commit | ||
|
||
SUBTENSOR_PULSE_DELAY = 24 | ||
PERIOD = 3 # Drand period in seconds | ||
GENESIS_TIME = 1692803367 | ||
|
||
|
||
def test_get_encrypted_commits(): | ||
uids = [1, 2] | ||
weights = [11, 22] | ||
version_key = 50 | ||
tempo = 100 | ||
current_block = 1000 | ||
netuid = 1 | ||
reveal_period = 2 | ||
block_time = 12 | ||
|
||
start_time = int(time.time()) | ||
ct_pybytes, reveal_round = get_encrypted_commit( | ||
uids, | ||
weights, | ||
version_key, | ||
tempo, | ||
current_block, | ||
netuid, | ||
reveal_period, | ||
block_time, | ||
) | ||
|
||
# Basic checks | ||
assert ( | ||
ct_pybytes is not None and len(ct_pybytes) > 0 | ||
), "Ciphertext should not be empty" | ||
assert reveal_round > 0, "Reveal round should be positive" | ||
|
||
expected_reveal_round, _, _ = compute_expected_reveal_round( | ||
start_time, tempo, current_block, netuid, reveal_period, block_time | ||
) | ||
|
||
# The reveal_round should be close to what we predict | ||
assert ( | ||
abs(reveal_round - expected_reveal_round) <= 1 | ||
), f"Reveal round {reveal_round} not close to expected {expected_reveal_round}" | ||
|
||
|
||
def test_generate_commit_success(): | ||
uids = [1, 2, 3] | ||
values = [10, 20, 30] | ||
version_key = 42 | ||
tempo = 50 | ||
current_block = 500 | ||
netuid = 100 | ||
subnet_reveal_period_epochs = 2 | ||
block_time = 12 | ||
|
||
start_time = int(time.time()) | ||
ct_pybytes, reveal_round = get_encrypted_commit( | ||
uids, | ||
values, | ||
version_key, | ||
tempo, | ||
current_block, | ||
netuid, | ||
subnet_reveal_period_epochs, | ||
block_time, | ||
) | ||
|
||
assert ( | ||
ct_pybytes is not None and len(ct_pybytes) > 0 | ||
), "Ciphertext should not be empty" | ||
assert reveal_round > 0, "Reveal round should be positive" | ||
|
||
expected_reveal_round, expected_reveal_time, time_until_reveal = ( | ||
compute_expected_reveal_round( | ||
start_time, | ||
tempo, | ||
current_block, | ||
netuid, | ||
subnet_reveal_period_epochs, | ||
block_time, | ||
) | ||
) | ||
|
||
assert ( | ||
abs(reveal_round - expected_reveal_round) <= 1 | ||
), f"Reveal round {reveal_round} differs from expected {expected_reveal_round}" | ||
|
||
required_lead_time = SUBTENSOR_PULSE_DELAY * PERIOD | ||
computed_reveal_time = ( | ||
GENESIS_TIME + (reveal_round + SUBTENSOR_PULSE_DELAY) * PERIOD | ||
) | ||
assert computed_reveal_time - start_time >= required_lead_time, ( | ||
"Not enough lead time before reveal. " | ||
f"computed_reveal_time={computed_reveal_time}, start_time={start_time}, required={required_lead_time}" | ||
) | ||
|
||
assert ( | ||
time_until_reveal >= SUBTENSOR_PULSE_DELAY * PERIOD | ||
), f"time_until_reveal {time_until_reveal} is less than required {SUBTENSOR_PULSE_DELAY * PERIOD}" | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_generate_commit_various_tempos(): | ||
NETUID = 1 | ||
CURRENT_BLOCK = 100_000 | ||
SUBNET_REVEAL_PERIOD_EPOCHS = 1 | ||
BLOCK_TIME = 6 | ||
TEMPOS = [10, 50, 100, 250, 360, 500, 750, 1000] | ||
|
||
uids = [0] | ||
values = [100] | ||
version_key = 1 | ||
|
||
for tempo in TEMPOS: | ||
start_time = int(time.time()) | ||
|
||
ct_pybytes, reveal_round = get_encrypted_commit( | ||
uids, | ||
values, | ||
version_key, | ||
tempo, | ||
CURRENT_BLOCK, | ||
NETUID, | ||
SUBNET_REVEAL_PERIOD_EPOCHS, | ||
BLOCK_TIME, | ||
) | ||
|
||
assert len(ct_pybytes) > 0, f"Ciphertext is empty for tempo {tempo}" | ||
assert reveal_round > 0, f"Reveal round is zero or negative for tempo {tempo}" | ||
|
||
expected_reveal_round, _, time_until_reveal = compute_expected_reveal_round( | ||
start_time, | ||
tempo, | ||
CURRENT_BLOCK, | ||
NETUID, | ||
SUBNET_REVEAL_PERIOD_EPOCHS, | ||
BLOCK_TIME, | ||
) | ||
|
||
assert ( | ||
abs(reveal_round - expected_reveal_round) <= 1 | ||
), f"Tempo {tempo}: reveal_round {reveal_round} not close to expected {expected_reveal_round}" | ||
|
||
computed_reveal_time = ( | ||
GENESIS_TIME + (reveal_round + SUBTENSOR_PULSE_DELAY) * PERIOD | ||
) | ||
required_lead_time = SUBTENSOR_PULSE_DELAY * PERIOD | ||
|
||
assert computed_reveal_time - start_time >= required_lead_time, ( | ||
f"Tempo {tempo}: Not enough lead time: reveal_time={computed_reveal_time}, " | ||
f"start_time={start_time}, required={required_lead_time}" | ||
) | ||
|
||
assert ( | ||
time_until_reveal >= SUBTENSOR_PULSE_DELAY * PERIOD | ||
), f"Tempo {tempo}: time_until_reveal {time_until_reveal} is less than required {SUBTENSOR_PULSE_DELAY * PERIOD}" | ||
|
||
|
||
def compute_expected_reveal_round( | ||
now: int, | ||
tempo: int, | ||
current_block: int, | ||
netuid: int, | ||
subnet_reveal_period_epochs: int, | ||
block_time: int, | ||
): | ||
tempo_plus_one = tempo + 1 | ||
netuid_plus_one = netuid + 1 | ||
block_with_offset = current_block + netuid_plus_one | ||
current_epoch = block_with_offset // tempo_plus_one | ||
|
||
# Initial guess for reveal_epoch | ||
reveal_epoch = current_epoch + subnet_reveal_period_epochs | ||
reveal_block_number = reveal_epoch * tempo_plus_one - netuid_plus_one | ||
|
||
# Compute blocks_until_reveal, ensure non-negative | ||
blocks_until_reveal = reveal_block_number - current_block | ||
if blocks_until_reveal < 0: | ||
blocks_until_reveal = 0 | ||
time_until_reveal = blocks_until_reveal * block_time | ||
|
||
# Adjust until we have enough lead time (at least SUBTENSOR_PULSE_DELAY pulses * period seconds) | ||
while time_until_reveal < SUBTENSOR_PULSE_DELAY * PERIOD: | ||
reveal_epoch += 1 | ||
reveal_block_number = reveal_epoch * tempo_plus_one - netuid_plus_one | ||
blocks_until_reveal = reveal_block_number - current_block | ||
if blocks_until_reveal < 0: | ||
blocks_until_reveal = 0 | ||
time_until_reveal = blocks_until_reveal * block_time | ||
|
||
reveal_time = now + time_until_reveal | ||
reveal_round = ( | ||
(reveal_time - GENESIS_TIME + PERIOD - 1) // PERIOD | ||
) - SUBTENSOR_PULSE_DELAY | ||
return reveal_round, reveal_time, time_until_reveal |