diff --git a/tests/integration/test_reward_stream_state.py b/tests/integration/test_reward_stream_state.py new file mode 100644 index 00000000..2e425dbb --- /dev/null +++ b/tests/integration/test_reward_stream_state.py @@ -0,0 +1,268 @@ +from collections import defaultdict + +import brownie +from brownie import RewardStream, chain +from brownie.test import strategy +from brownie_tokens import ERC20 + +DAY = 60 * 60 * 24 + + +class _State: + """RewardStream.vy encapsulated state.""" + + def __init__(self, owner: str, distributor: str, duration: int) -> None: + # public getters in contract + self.owner = owner + self.distributor = distributor + + # time when reward distribtuion period finishes + self.period_finish = 0 + # rate at which the reward is distributed (per block) + self.reward_rate = 0 + # duration of the reward period (in seconds) + self.reward_duration = duration + # epoch time of last state changing update + self.last_update_time = 0 + # the total amount of rewards a receiver is to receive + self.reward_per_receiver_total = 0 + # total number of receivers + self.receiver_count = 0 + # whether a receiver is approved to get rewards + self.reward_receivers = defaultdict(bool) + + # private storage in contract + # how much reward tokens a receiver has been sent + self._reward_paid = defaultdict(int) + self._reward_start = defaultdict(int) + self._lifetime_earnings = defaultdict(int) + + def _update_per_receiver_total(self, timestamp: int) -> int: + """Globally update the total amount received per receiver. + + This function updates the `self.reward_per_receiver_total` variable in the 4 + external function calls `add_receiver`, `remove_receiver`, `get_reward`, + and `notify_reward_amount`. + + Note: + For users that get added mid-distribution period, this function will + set their `reward_paid` variable to the current `reward_per_receiver_total`, + and then update the global `reward_per_receiver_total` var. This effectively + makes it so rewards are distributed equally from the point onwards which a + user is added. + """ + total = self.reward_per_receiver_total + count = self.receiver_count + if count == 0: + return total + + last_time = min(timestamp, self.period_finish) + total += (last_time - self.last_update_time) * self.reward_rate // count + self.reward_per_receiver_total = total + self.last_update_time = last_time + + return total + + def add_receiver(self, receiver: str, msg_sender: str, timestamp: int): + """Add a new receiver.""" + assert msg_sender == self.owner, "dev: only owner" + assert self.reward_receivers[receiver] is False, "dev: receiver is active" + + total = self._update_per_receiver_total(timestamp) + self.reward_receivers[receiver] = True + self.receiver_count += 1 + self._reward_paid[receiver] = total + self._reward_start[receiver] = total + + def remove_receiver(self, receiver: str, msg_sender: str, timestamp: int): + """Remove a receiver, pay out their reward""" + assert msg_sender == self.owner, "dev: only owner" + assert self.reward_receivers[receiver] is True, "dev: receiver is inactive" + + total = self._update_per_receiver_total(timestamp) + self.reward_receivers[receiver] = False + self.receiver_count -= 1 + amount = total - self._reward_paid[receiver] + if amount > 0: + # send ERC20 reward token to `msg_sender` + self._lifetime_earnings[receiver] += amount + self._reward_paid[receiver] = 0 + self._reward_start[receiver] = 0 + + def get_reward(self, msg_sender: str, timestamp: int): + """Get rewards if any are available""" + assert self.reward_receivers[msg_sender] is True, "dev: caller is not receiver" + + total = self._update_per_receiver_total(timestamp) + amount = total - self._reward_paid[msg_sender] + if amount > 0: + # transfer `amount` of ERC20 tokens to `msg_sender` + # update the total amount paid out to `msg_sender` + self._reward_paid[msg_sender] = total + self._lifetime_earnings[msg_sender] += amount + + def notify_reward_amount(self, amount: int, timestamp: int, msg_sender: str): + """Add rewards to the contract for distribution.""" + assert msg_sender == self.distributor, "dev: only distributor" + + self._update_per_receiver_total(timestamp) + if timestamp >= self.period_finish: + # the reward distribution period has passed + self.reward_rate = amount // self.reward_duration + else: + # reward distribution period currently in progress + remaining_rewards = (self.period_finish - timestamp) * self.reward_rate + self.reward_rate = (amount + remaining_rewards) // self.reward_duration + + self.last_update_time = timestamp + # extend our reward duration period + self.period_finish = timestamp + self.reward_duration + + def set_reward_duration(self, duration: int, timestamp: int, msg_sender: str): + """Adjust the reward distribution duration.""" + assert msg_sender == self.owner, "dev: only owner" + assert timestamp > self.period_finish, "dev: reward period currently active" + + self.reward_duration = duration + + +class StateMachine: + + st_uint = strategy("uint64") + st_receiver = strategy("address") + + def __init__(cls, accounts, owner, distributor, duration, reward_token, reward_stream): + cls.accounts = accounts + cls.owner = str(owner) + cls.distributor = str(distributor) + cls.duration = duration + cls.reward_token = reward_token + cls.reward_stream = reward_stream + + def setup(self): + self.state = _State(self.owner, self.distributor, self.duration) + + def initialize_rewards(self, amount="st_uint"): + distributor_balance = self.reward_token.balanceOf(self.distributor) + if not distributor_balance >= amount: + self.reward_token._mint_for_testing(self.distributor, amount - distributor_balance) + self.reward_token.approve(self.reward_stream, amount, {"from": self.distributor}) + + tx = self.reward_stream.notify_reward_amount(amount, {"from": self.distributor}) + self.state.notify_reward_amount(amount, tx.timestamp, self.distributor) + + def rule_add_receiver(self, st_receiver): + st_receiver = str(st_receiver) + # fail route + if self.state.reward_receivers[st_receiver] is True: + with brownie.reverts("dev: receiver is active"): + self.reward_stream.add_receiver(st_receiver, {"from": self.owner}) + return + elif st_receiver == self.distributor: + return + + # success route + tx = self.reward_stream.add_receiver(st_receiver, {"from": self.owner}) + self.state.add_receiver(st_receiver, self.owner, tx.timestamp) + + def rule_remove_receiver(self, st_receiver): + st_receiver = str(st_receiver) + # fail route + if self.state.reward_receivers[st_receiver] is False: + with brownie.reverts("dev: receiver is inactive"): + self.reward_stream.remove_receiver(st_receiver, {"from": self.owner}) + return + + # success route + tx = self.reward_stream.remove_receiver(st_receiver, {"from": self.owner}) + self.state.remove_receiver(st_receiver, self.owner, tx.timestamp) + + def rule_get_reward(self, caller="st_receiver"): + caller = str(caller) + if self.state.reward_receivers[caller] is False: + with brownie.reverts("dev: caller is not receiver"): + self.reward_stream.get_reward({"from": caller}) + return + + tx = self.reward_stream.get_reward({"from": caller}) + self.state.get_reward(caller, tx.timestamp) + + def rule_notify_reward_amount(self, amount="st_uint"): + distributor_balance = self.reward_token.balanceOf(self.distributor) + if not distributor_balance >= amount: + self.reward_token._mint_for_testing(self.distributor, amount - distributor_balance) + self.reward_token.approve(self.reward_stream, amount, {"from": self.distributor}) + + tx = self.reward_stream.notify_reward_amount(amount, {"from": self.distributor}) + self.state.notify_reward_amount(amount, tx.timestamp, self.distributor) + + def rule_set_reward_duration(self, duration="st_uint"): + if chain.time() < self.state.period_finish: + with brownie.reverts("dev: reward period currently active"): + self.reward_stream.set_reward_duration(duration, {"from": self.owner}) + return + + tx = self.reward_stream.set_reward_duration(duration, {"from": self.owner}) + self.state.set_reward_duration(duration, tx.timestamp, self.owner) + + def invariant_sleep(self): + chain.sleep(10) + + def invariant_state_getters(self): + assert self.reward_stream.period_finish() == self.state.period_finish + + # invariant_reward_rate + assert self.reward_stream.reward_rate() == self.state.reward_rate + + # invariant_reward_duration + assert self.reward_stream.reward_duration() == self.state.reward_duration + + # invariant_last_update_time + assert self.reward_stream.last_update_time() == self.state.last_update_time + + # invariant_reward_per_receiver_total + assert ( + self.reward_stream.reward_per_receiver_total() == self.state.reward_per_receiver_total + ) + + # invariant_receiver_count + assert self.reward_stream.receiver_count() == self.state.receiver_count + + # invariant_reward_receivers + for acct, val in self.state.reward_receivers.items(): + assert self.reward_stream.reward_receivers(acct) is val + + def teardown(self): + chain.sleep(self.state.reward_duration) + + for acct in self.state._lifetime_earnings.keys(): + if self.state.reward_receivers[acct] is True: + tx = self.reward_stream.get_reward({"from": acct}) + self.state.get_reward(acct, tx.timestamp) + assert self.reward_token.balanceOf(acct) == self.state._lifetime_earnings[acct] + + +def test_state_machine(state_machine, accounts): + # setup + owner = accounts[0] + distributor = accounts[1] + duration = DAY * 10 + + # deploy reward token, and mint + reward_token = ERC20() + + # deploy stream and approve + reward_stream = owner.deploy(RewardStream, owner, distributor, reward_token, duration) + + # settings = {"stateful_step_count": 25} + + state_machine( + StateMachine, + accounts, + owner, + distributor, + duration, + reward_token, + reward_stream, + # settings=settings, + ) diff --git a/tests/unitary/RewardStream/conftest.py b/tests/unitary/RewardStream/conftest.py new file mode 100644 index 00000000..c23fff3e --- /dev/null +++ b/tests/unitary/RewardStream/conftest.py @@ -0,0 +1,16 @@ +import pytest +from brownie_tokens import ERC20 + + +@pytest.fixture(scope="module") +def reward_token(bob): + token = ERC20() + token._mint_for_testing(bob, 10 ** 19) + return token + + +@pytest.fixture(scope="module") +def stream(RewardStream, alice, bob, reward_token): + contract = RewardStream.deploy(alice, bob, reward_token, 86400 * 10, {"from": alice}) + reward_token.approve(contract, 2 ** 256 - 1, {"from": bob}) + return contract diff --git a/tests/unitary/RewardStream/test_add_receiver.py b/tests/unitary/RewardStream/test_add_receiver.py new file mode 100644 index 00000000..553dfcf1 --- /dev/null +++ b/tests/unitary/RewardStream/test_add_receiver.py @@ -0,0 +1,27 @@ +import brownie + + +def test_receiver_count_increases(alice, charlie, stream): + pre_receiver_count = stream.receiver_count() + stream.add_receiver(charlie, {"from": alice}) + + assert stream.receiver_count() == pre_receiver_count + 1 + + +def test_receiver_activation(alice, charlie, stream): + pre_activation = stream.reward_receivers(charlie) + stream.add_receiver(charlie, {"from": alice}) + + assert pre_activation is False + assert stream.reward_receivers(charlie) is True + + +def test_reverts_for_non_owner(bob, charlie, stream): + with brownie.reverts("dev: only owner"): + stream.add_receiver(charlie, {"from": bob}) + + +def test_reverts_for_active_receiver(alice, charlie, stream): + stream.add_receiver(charlie, {"from": alice}) + with brownie.reverts("dev: receiver is active"): + stream.add_receiver(charlie, {"from": alice}) diff --git a/tests/unitary/RewardStream/test_amounts.py b/tests/unitary/RewardStream/test_amounts.py new file mode 100644 index 00000000..3d5a2f83 --- /dev/null +++ b/tests/unitary/RewardStream/test_amounts.py @@ -0,0 +1,70 @@ +from brownie import chain + + +def test_single_receiver(stream, alice, bob, charlie, reward_token): + stream.add_receiver(charlie, {"from": alice}) + stream.notify_reward_amount(10 ** 18, {"from": bob}) + chain.sleep(86400 * 10) + + stream.get_reward({"from": charlie}) + + assert 0.9999 <= reward_token.balanceOf(charlie) / 10 ** 18 <= 1 + + +def test_single_receiver_partial_duration(stream, alice, bob, charlie, reward_token): + stream.add_receiver(charlie, {"from": alice}) + tx = stream.notify_reward_amount(10 ** 18, {"from": bob}) + start = tx.timestamp - 1 + + for i in range(1, 10): + chain.mine(timestamp=start + 86400 * i) + stream.get_reward({"from": charlie}) + assert 0.9999 <= reward_token.balanceOf(charlie) / (i * 10 ** 17) <= 1 + + +def test_multiple_receivers(stream, alice, bob, accounts, reward_token): + for i in range(2, 6): + stream.add_receiver(accounts[i], {"from": alice}) + tx = stream.notify_reward_amount(10 ** 18, {"from": bob}) + start = tx.timestamp - 1 + + for i in range(1, 10): + chain.mine(timestamp=start + 86400 * i) + for x in range(2, 6): + stream.get_reward({"from": accounts[x]}) + + balances = [reward_token.balanceOf(i) for i in accounts[2:6]] + assert 0.9999 < min(balances) / max(balances) <= 1 + assert 0.9999 <= sum(balances) / (i * 10 ** 17) <= 1 + + +def test_add_receiver_during_period(stream, alice, bob, charlie, reward_token): + stream.add_receiver(charlie, {"from": alice}) + stream.notify_reward_amount(10 ** 18, {"from": bob}) + chain.sleep(86400 * 5) + stream.add_receiver(alice, {"from": alice}) + chain.sleep(86400 * 5) + + stream.get_reward({"from": alice}) + stream.get_reward({"from": charlie}) + alice_balance = reward_token.balanceOf(alice) + charlie_balance = reward_token.balanceOf(charlie) + + assert 0.9999 <= alice_balance * 3 / charlie_balance <= 1 + assert 0.9999 <= (alice_balance + charlie_balance) / 10 ** 18 <= 1 + + +def test_remove_receiver_during_period(stream, alice, bob, charlie, reward_token): + stream.add_receiver(alice, {"from": alice}) + stream.add_receiver(charlie, {"from": alice}) + stream.notify_reward_amount(10 ** 18, {"from": bob}) + chain.sleep(86400 * 5) + stream.remove_receiver(alice, {"from": alice}) + chain.sleep(86400 * 5) + + stream.get_reward({"from": charlie}) + alice_balance = reward_token.balanceOf(alice) + charlie_balance = reward_token.balanceOf(charlie) + + assert 0.9999 <= alice_balance * 3 / charlie_balance <= 1 + assert 0.9999 <= (alice_balance + charlie_balance) / 10 ** 18 <= 1 diff --git a/tests/unitary/RewardStream/test_get_reward.py b/tests/unitary/RewardStream/test_get_reward.py new file mode 100644 index 00000000..f25ea962 --- /dev/null +++ b/tests/unitary/RewardStream/test_get_reward.py @@ -0,0 +1,6 @@ +import brownie + + +def test_only_active_receiver_can_call(charlie, stream): + with brownie.reverts("dev: caller is not receiver"): + stream.get_reward({"from": charlie}) diff --git a/tests/unitary/RewardStream/test_notify_reward_amount.py b/tests/unitary/RewardStream/test_notify_reward_amount.py new file mode 100644 index 00000000..ad538f40 --- /dev/null +++ b/tests/unitary/RewardStream/test_notify_reward_amount.py @@ -0,0 +1,45 @@ +import brownie +from brownie import chain + + +def test_only_distributor_allowed(alice, stream): + with brownie.reverts("dev: only distributor"): + stream.notify_reward_amount(10 ** 18, {"from": alice}) + + +def test_retrieves_reward_token(bob, stream, reward_token): + stream.notify_reward_amount(10 ** 18, {"from": bob}) + post_notify = reward_token.balanceOf(stream) + + assert post_notify == 10 ** 18 + + +def test_reward_rate_updates(bob, stream): + stream.notify_reward_amount(10 ** 18, {"from": bob}) + post_notify = stream.reward_rate() + + assert post_notify > 0 + assert post_notify == 10 ** 18 / (86400 * 10) + + +def test_reward_rate_updates_mid_duration(bob, stream): + stream.notify_reward_amount(10 ** 18, {"from": bob}) + chain.sleep(86400 * 5) # half of the duration + + # top up the balance to be 10 ** 18 again + stream.notify_reward_amount(10 ** 18 / 2, {"from": bob}) + post_notify = stream.reward_rate() + + assert post_notify == 10 ** 18 / (86400 * 10) + + +def test_period_finish_updates(bob, stream): + tx = stream.notify_reward_amount(10 ** 18, {"from": bob}) + + assert stream.period_finish() == tx.timestamp + 86400 * 10 + + +def test_update_last_update_time(bob, stream): + tx = stream.notify_reward_amount(10 ** 18, {"from": bob}) + + assert stream.last_update_time() == tx.timestamp diff --git a/tests/unitary/RewardStream/test_ownership_change.py b/tests/unitary/RewardStream/test_ownership_change.py new file mode 100644 index 00000000..9a8b0e3a --- /dev/null +++ b/tests/unitary/RewardStream/test_ownership_change.py @@ -0,0 +1,27 @@ +from itertools import product + +import brownie +import pytest + + +@pytest.mark.parametrize("commit,accept", product([0, 1], repeat=2)) +def test_change_ownership(alice, charlie, commit, accept, stream): + if commit: + stream.commit_transfer_ownership(charlie, {"from": alice}) + + if accept: + stream.accept_transfer_ownership({"from": charlie}) + + assert stream.owner() == charlie if commit and accept else alice + + +def test_commit_only_owner(charlie, stream): + with brownie.reverts("dev: only owner"): + stream.commit_transfer_ownership(charlie, {"from": charlie}) + + +def test_accept_only_owner(alice, bob, stream): + stream.commit_transfer_ownership(bob, {"from": alice}) + + with brownie.reverts("dev: only new owner"): + stream.accept_transfer_ownership({"from": alice}) diff --git a/tests/unitary/RewardStream/test_remove_receiver.py b/tests/unitary/RewardStream/test_remove_receiver.py new file mode 100644 index 00000000..e231a530 --- /dev/null +++ b/tests/unitary/RewardStream/test_remove_receiver.py @@ -0,0 +1,54 @@ +import brownie +import pytest +from brownie import chain + + +@pytest.fixture(autouse=True) +def module_setup(alice, bob, charlie, stream, reward_token): + stream.add_receiver(charlie, {"from": alice}) + stream.notify_reward_amount(10 ** 18, {"from": bob}) + chain.sleep(86400 * 10) + + +def test_receiver_deactivates(alice, charlie, stream): + pre_remove = stream.reward_receivers(charlie) + stream.remove_receiver(charlie, {"from": alice}) + + assert pre_remove is True + assert stream.reward_receivers(charlie) is False + + +def test_receiver_count_decreases(alice, charlie, stream): + pre_remove = stream.receiver_count() + stream.remove_receiver(charlie, {"from": alice}) + post_remove = stream.receiver_count() + + assert post_remove == pre_remove - 1 + + +def test_updates_last_update_time(alice, charlie, stream): + pre_remove = stream.last_update_time() + tx = stream.remove_receiver(charlie, {"from": alice}) + post_remove = stream.last_update_time() + + assert pre_remove < post_remove + assert post_remove == tx.timestamp + + +def test_updates_reward_per_receiver_total(alice, charlie, stream): + pre_remove = stream.reward_per_receiver_total() + stream.remove_receiver(charlie, {"from": alice}) + post_remove = stream.reward_per_receiver_total() + + assert pre_remove < post_remove + assert 0.9999 < post_remove / 10 ** 18 <= 1 + + +def test_reverts_for_non_owner(bob, charlie, stream): + with brownie.reverts("dev: only owner"): + stream.remove_receiver(charlie, {"from": bob}) + + +def test_reverts_for_inactive_receiver(alice, bob, stream): + with brownie.reverts("dev: receiver is inactive"): + stream.remove_receiver(bob, {"from": alice}) diff --git a/tests/unitary/RewardStream/test_set_reward_distributor.py b/tests/unitary/RewardStream/test_set_reward_distributor.py new file mode 100644 index 00000000..4c118ddf --- /dev/null +++ b/tests/unitary/RewardStream/test_set_reward_distributor.py @@ -0,0 +1,12 @@ +import brownie + + +def test_successful_change(alice, charlie, stream): + stream.set_reward_distributor(charlie, {"from": alice}) + + assert stream.distributor() == charlie + + +def test_only_owner(bob, charlie, stream): + with brownie.reverts("dev: only owner"): + stream.set_reward_distributor(charlie, {"from": bob}) diff --git a/tests/unitary/RewardStream/test_set_reward_duration.py b/tests/unitary/RewardStream/test_set_reward_duration.py new file mode 100644 index 00000000..66bf97c9 --- /dev/null +++ b/tests/unitary/RewardStream/test_set_reward_duration.py @@ -0,0 +1,21 @@ +import brownie + + +def test_updates_reward_duration(alice, stream): + pre_update = stream.reward_duration() + stream.set_reward_duration(86400 * 365, {"from": alice}) + + assert pre_update == 86400 * 10 + assert stream.reward_duration() == 86400 * 365 + + +def test_only_owner(charlie, stream): + with brownie.reverts("dev: only owner"): + stream.set_reward_duration(10, {"from": charlie}) + + +def test_mid_reward_distribution_period(alice, bob, stream): + stream.notify_reward_amount(10 ** 18, {"from": bob}) + + with brownie.reverts("dev: reward period currently active"): + stream.set_reward_duration(10, {"from": alice})