Пример #1
0
    def setUp(self):
        self.db_name = 'fee.db'
        self.address = create_address(AddressPrefix.EOA)
        db = ContextDatabase.from_path(self.db_name)
        self.assertIsNotNone(db)

        self.storage = FeeStorage(db)

        context = IconScoreContext(IconScoreContextType.DIRECT)
        context.tx_batch = TransactionBatch()
        context.block_batch = BlockBatch()
        self.context = context
Пример #2
0
def _patch_service_engine(icon_service_engine, revision):
    # Mocks get_balance so, it returns always 100 icx
    # TODO : patch when use get_balance or transfer
    IconScoreContext.engine = ContextEngine(deploy=DeployEngine(),
                                            fee=FeeEngine(),
                                            icx=IcxEngine(),
                                            iiss=IISSEngine(),
                                            prep=PRepEngine(),
                                            issue=IssueEngine(),
                                            inv=INVEngine())

    db = icon_service_engine._icx_context_db
    IconScoreContext.storage = ContextStorage(deploy=DeployStorage(db),
                                              fee=FeeStorage(db),
                                              icx=IcxStorage(db),
                                              iiss=IISSStorage(db),
                                              prep=PRepStorage(db),
                                              issue=IssueStorage(db),
                                              meta=MetaDBStorage(db),
                                              rc=RewardCalcStorage(),
                                              inv=INVStorage(db))
    IconScoreContext.engine.inv._inv_container = generate_inv_container(
        False, revision)

    return icon_service_engine
Пример #3
0
def storage():
    db_name = 'fee.db'
    db = ContextDatabase.from_path(db_name)
    assert db is not None
    storage = FeeStorage(db)
    yield storage
    rmtree(db_name)
Пример #4
0
def _patch_service_engine(icon_service_engine, revision):
    # Mocks get_balance so, it returns always 100 icx
    # TODO : patch when use get_balance or transfer
    IconScoreContext.engine = ContextEngine(
        deploy=DeployEngine(),
        fee=FeeEngine(),
        icx=IcxEngine(),
        iiss=IISSEngine(),
        prep=PRepEngine(),
        issue=IssueEngine()
    )

    db = icon_service_engine._icx_context_db
    IconScoreContext.storage = ContextStorage(
        deploy=DeployStorage(db),
        fee=FeeStorage(db),
        icx=IcxStorage(db),
        iiss=IISSStorage(db),
        prep=PRepStorage(db),
        issue=IssueStorage(db),
        meta=MetaDBStorage(db),
        rc=RewardCalcStorage()
    )

    # Patch revision
    def set_revision_to_context(context):
        context.revision = revision

    icon_service_engine._set_revision_to_context = \
        Mock(side_effect=set_revision_to_context)

    return icon_service_engine
Пример #5
0
def patch_fee_storage(fee_storage: FeeStorage):
    memory_db = {}

    # noinspection PyUnusedLocal
    def put(context, key, value):
        memory_db[key] = value

    # noinspection PyUnusedLocal
    def put_deposit(context, deposit):
        memory_db[deposit.id] = deposit

    # noinspection PyUnusedLocal
    def get(context, key):
        return memory_db[key] if key in memory_db else None

    # noinspection PyUnusedLocal
    def delete(context, key):
        del memory_db[key]

    fee_storage.put_deposit_meta = put
    fee_storage.get_deposit_meta = get
    fee_storage.delete_deposit_meta = delete
    fee_storage.put_deposit = put_deposit
    fee_storage.get_deposit = get
    fee_storage.delete_deposit = delete
Пример #6
0
    def setUp(self):
        context = IconScoreContext(IconScoreContextType.DIRECT)
        block = Mock(spec=Block)
        block.attach_mock(Mock(return_value=0), 'height')
        context.block = block

        self._sender = Address.from_data(AddressPrefix.EOA, os.urandom(20))
        self._score_address = Address.from_data(AddressPrefix.CONTRACT, os.urandom(20))

        context_db = create_context_db()

        self.deploy_storage = DeployStorage(context_db)
        deploy_info = IconScoreDeployInfo(self._score_address,
                                          DeployState.ACTIVE,
                                          self._sender,
                                          os.urandom(32),
                                          os.urandom(32))
        self.icx_storage = IcxStorage(context_db)
        self._icx_engine = IcxEngine()

        self.fee_storage = FeeStorage(context_db)
        patch_fee_storage(self.fee_storage)

        self.deploy_storage.put_deploy_info(context, deploy_info)
        context.storage = ContextStorage(deploy=self.deploy_storage, fee=self.fee_storage, icx=self.icx_storage,
                                         iiss=None, prep=None, issue=None, rc=None, meta=None)
        context.engine = ContextEngine(deploy=None, fee=None, icx=self._icx_engine, iiss=None, prep=None, issue=None)
        self._icx_engine.open(self.icx_storage)

        self.icx_storage._put_genesis_data_account(context,
                                                   CoinPartType.GENERAL,
                                                   self._sender,
                                                   100000000 * 10 ** 18)
        self.icx_storage._put_genesis_data_account(context, CoinPartType.TREASURY,
                                                   Address.from_data(AddressPrefix.EOA, os.urandom(20)), 0)

        self._engine = FeeEngine()
Пример #7
0
class TestFeeStorage(TestCase):

    def setUp(self):
        self.db_name = 'fee.db'
        self.address = create_address(AddressPrefix.EOA)
        db = ContextDatabase.from_path(self.db_name)
        self.assertIsNotNone(db)

        self.storage = FeeStorage(db)

        context = IconScoreContext(IconScoreContextType.DIRECT)
        context.tx_batch = TransactionBatch()
        context.block_batch = BlockBatch()
        self.context = context

    def tearDown(self):
        context = self.context
        self.storage.close(context)
        rmtree(self.db_name)

    def test_get_put_delete_score_fee(self):
        context = self.context
        score_address = create_address(AddressPrefix.CONTRACT)

        deposit_meta = DepositMeta()
        deposit_meta.head_id = create_tx_hash()
        deposit_meta.tail_id = create_tx_hash()
        deposit_meta.available_head_id_of_deposit = create_tx_hash()
        deposit_meta.available_head_id_of_virtual_step = create_tx_hash()
        self.storage.put_deposit_meta(context, score_address, deposit_meta)

        deposit_meta_2 = self.storage.get_deposit_meta(context, score_address)
        self.assertEqual(deposit_meta, deposit_meta_2)

        self.storage.delete_deposit_meta(context, score_address)
        deposit_meta_2 = self.storage.get_deposit_meta(context, score_address)
        self.assertIsNone(deposit_meta_2)

    def test_get_put_delete_score_fee_with_none_type(self):
        context = self.context
        score_address = create_address(AddressPrefix.CONTRACT)

        deposit_meta = DepositMeta()
        self.storage.put_deposit_meta(context, score_address, deposit_meta)

        deposit_meta_2 = self.storage.get_deposit_meta(context, score_address)
        self.assertEqual(deposit_meta, deposit_meta_2)

        self.storage.delete_deposit_meta(context, score_address)
        deposit_meta_2 = self.storage.get_deposit_meta(context, score_address)
        self.assertIsNone(deposit_meta_2)

    def test_get_put_delete_deposit(self):
        context = self.context

        deposit = Deposit()
        deposit.id = create_tx_hash()
        deposit.score_address = create_address(AddressPrefix.CONTRACT)
        deposit.sender = create_address(AddressPrefix.EOA)
        deposit.deposit_amount = 10000
        deposit.deposit_used = 10000
        deposit.created = 10
        deposit.expires = 1000000
        deposit.virtual_step_issued = 100000000000
        deposit.virtual_step_used = 200000000000
        deposit.prev_id = create_tx_hash()
        deposit.next_id = create_tx_hash()
        deposit.version = 2
        self.storage.put_deposit(context, deposit)

        deposit2 = self.storage.get_deposit(context, deposit.id)
        self.assertEqual(deposit, deposit2)
        self.assertEqual(deposit.id, deposit2.id)

        self.storage.delete_deposit(context, deposit.id)
        deposit2 = self.storage.get_deposit(context, deposit.id)
        self.assertIsNone(deposit2)

    def test_get_put_delete_deposit_with_none_type(self):
        context = self.context

        deposit = Deposit()
        deposit.id = create_tx_hash()
        deposit.score_address = create_address(AddressPrefix.CONTRACT)
        deposit.sender = create_address(AddressPrefix.EOA)
        deposit.deposit_amount = 10000
        deposit.deposit_used = 10000
        deposit.created = 10
        deposit.expires = 1000000
        deposit.virtual_step_issued = 100000000000
        deposit.virtual_step_used = 200000000000
        deposit.version = 1
        self.storage.put_deposit(context, deposit)

        deposit2 = self.storage.get_deposit(context, deposit.id)
        self.assertEqual(deposit, deposit2)
        self.assertEqual(deposit.id, deposit2.id)

        self.storage.delete_deposit(context, deposit.id)
        deposit2 = self.storage.get_deposit(context, deposit.id)
        self.assertIsNone(deposit2)
Пример #8
0
class TestFeeEngine(unittest.TestCase):

    def setUp(self):
        context = IconScoreContext(IconScoreContextType.DIRECT)
        block = Mock(spec=Block)
        block.attach_mock(Mock(return_value=0), 'height')
        context.block = block

        self._sender = Address.from_data(AddressPrefix.EOA, os.urandom(20))
        self._score_address = Address.from_data(AddressPrefix.CONTRACT, os.urandom(20))

        context_db = create_context_db()

        self.deploy_storage = DeployStorage(context_db)
        deploy_info = IconScoreDeployInfo(self._score_address,
                                          DeployState.ACTIVE,
                                          self._sender,
                                          os.urandom(32),
                                          os.urandom(32))
        self.icx_storage = IcxStorage(context_db)
        self._icx_engine = IcxEngine()

        self.fee_storage = FeeStorage(context_db)
        patch_fee_storage(self.fee_storage)

        self.deploy_storage.put_deploy_info(context, deploy_info)
        context.storage = ContextStorage(deploy=self.deploy_storage, fee=self.fee_storage, icx=self.icx_storage,
                                         iiss=None, prep=None, issue=None, rc=None, meta=None)
        context.engine = ContextEngine(deploy=None, fee=None, icx=self._icx_engine, iiss=None, prep=None, issue=None)
        self._icx_engine.open(self.icx_storage)

        self.icx_storage._put_genesis_data_account(context,
                                                   CoinPartType.GENERAL,
                                                   self._sender,
                                                   100000000 * 10 ** 18)
        self.icx_storage._put_genesis_data_account(context, CoinPartType.TREASURY,
                                                   Address.from_data(AddressPrefix.EOA, os.urandom(20)), 0)

        self._engine = FeeEngine()

    def tearDown(self):
        ContextContainer._clear_context()
        clear_inner_task()
        VirtualStepCalculator.calculate_virtual_step = calculate_virtual_step

    def get_context(self):
        context = IconScoreContext(IconScoreContextType.INVOKE)
        context.step_counter = Mock(spec=IconScoreStepCounter)
        context.step_counter.step_price = 10 ** 10
        context.tx = Mock(spec=Transaction)
        context.tx.to = self._score_address
        block = Mock(spec=Block)
        block.attach_mock(Mock(return_value=0), 'height')
        context.block = block
        context.storage = ContextStorage(deploy=self.deploy_storage,fee=self.fee_storage, icx=self.icx_storage,
                                         iiss=None, prep=None, issue=None, rc=None, meta=None)
        context.engine = ContextEngine(deploy=None, fee=None, icx=self._icx_engine, iiss=None, prep=None, issue=None)
        return context

    def _deposit_bulk(self, count):
        self.context = self.get_context()
        self.block_height = 0
        input_params = []

        for i in range(count):
            tx_hash = os.urandom(32)
            amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
            block_height = randrange(100, 10000)
            term = get_rand_term()

            before_sender_balance = self._icx_engine.get_balance(self.context, self._sender)
            self._engine.add_deposit(
                self.context, tx_hash, self._sender, self._score_address, amount, block_height, term)
            after_sender_balance = self._icx_engine.get_balance(self.context, self._sender)

            self.assertEqual(amount, before_sender_balance - after_sender_balance)
            input_params.append((tx_hash, amount, block_height, term))

        return input_params

    def test_deposit_fee(self):
        context = self.get_context()
        block_height = 0

        size = randrange(10, 100)
        input_param = self._deposit_bulk(size)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, block_height)
        self.assertEqual(size, len(deposit_info.deposits))

        for i in range(size):
            tx_hash, amount, block_height, term = input_param[i]
            deposit = deposit_info.deposits[i]
            self.assertEqual(tx_hash, deposit.id)
            self.assertEqual(self._sender, deposit.sender)
            self.assertEqual(self._score_address, deposit.score_address)
            self.assertEqual(amount, deposit.deposit_amount)
            self.assertEqual(block_height, deposit.created)
            self.assertEqual(block_height + term, deposit.expires)

    def test_deposit_append_and_delete(self):
        size = randrange(10, 100)
        deposit_list = self._deposit_bulk(size)

        for i in range(size):
            index = randrange(0, size)
            size -= 1
            withdrawal_deposit_id = deposit_list.pop(index)[0]
            self._engine.withdraw_deposit(self.context, self._sender, withdrawal_deposit_id, 1)

            deposit_info = self._engine.get_deposit_info(self.context, self._score_address, 1)
            for j in range(size):
                deposit = deposit_info.deposits[j]
                self.assertEqual(deposit.id, deposit_list[j][0])
                self.assertEqual(self._sender, deposit.sender)
                self.assertEqual(self._score_address, deposit.score_address)
                self.assertEqual(deposit.deposit_amount, deposit_list[j][1])
                self.assertEqual(deposit.created, deposit_list[j][2])
                self.assertEqual(deposit.expires, deposit_list[j][2] + deposit_list[j][3])

        input_param = self._deposit_bulk(100)
        deposit_info = self._engine.get_deposit_info(self.context, self._score_address, self.block_height)

        self.assertEqual(100, len(deposit_info.deposits))

        for i in range(size):
            tx_hash, amount, block_height, term = input_param[i]
            deposit = deposit_info.deposits[i]
            self.assertEqual(tx_hash, deposit.id)
            self.assertEqual(self._sender, deposit.sender)
            self.assertEqual(self._score_address, deposit.score_address)
            self.assertEqual(amount, deposit.deposit_amount)
            self.assertEqual(block_height, deposit.created)
            self.assertEqual(block_height + term, deposit.expires)

    def test_deposit_fee_invalid_param(self):
        context = self.get_context()

        tx_hash = os.urandom(32)
        amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
        block_height = randrange(100, 10000)
        term = get_rand_term()

        # invalid amount (underflow)
        # noinspection PyTypeChecker
        with self.assertRaises(InvalidRequestException) as e:
            inv_amount = randrange(0, FeeEngine._MIN_DEPOSIT_AMOUNT - 1)
            self._engine.add_deposit(context, tx_hash, self._sender, self._score_address,
                                     inv_amount, block_height, term)
        # noinspection PyUnresolvedReferences
        self.assertEqual('Invalid deposit amount', e.exception.message)

        # invalid amount (overflow)
        # noinspection PyTypeChecker
        with self.assertRaises(InvalidRequestException) as e:
            inv_amount = \
                randrange(FeeEngine._MAX_DEPOSIT_AMOUNT + 1, FeeEngine._MAX_DEPOSIT_AMOUNT * 10)
            self._engine.add_deposit(context, tx_hash, self._sender, self._score_address,
                                     inv_amount, block_height, term)
        # noinspection PyUnresolvedReferences
        self.assertEqual('Invalid deposit amount', e.exception.message)

        # invalid term (underflow)
        # noinspection PyTypeChecker
        with self.assertRaises(InvalidRequestException) as e:
            inv_term = randrange(0, FeeEngine._MIN_DEPOSIT_TERM - 1)
            self._engine.add_deposit(context, tx_hash, self._sender, self._score_address,
                                     amount, block_height, inv_term)
        # noinspection PyUnresolvedReferences
        self.assertEqual('Invalid deposit term', e.exception.message)

        # invalid term (overflow)
        # noinspection PyTypeChecker
        with self.assertRaises(InvalidRequestException) as e:
            inv_term = \
                randrange(FeeEngine._MAX_DEPOSIT_TERM + 1, FeeEngine._MAX_DEPOSIT_TERM * 10)
            self._engine.add_deposit(context, tx_hash, self._sender, self._score_address,
                                     amount, block_height, inv_term)
        # noinspection PyUnresolvedReferences
        self.assertEqual('Invalid deposit term', e.exception.message)

        # invalid owner
        # noinspection PyTypeChecker
        with self.assertRaises(InvalidRequestException) as e:
            inv_sender = Address.from_data(AddressPrefix.EOA, os.urandom(20))
            self._engine.add_deposit(context, tx_hash, inv_sender, self._score_address,
                                     amount, block_height, term)
        # noinspection PyUnresolvedReferences
        self.assertEqual('Invalid SCORE owner', e.exception.message)

    def test_deposit_fee_out_of_balance(self):
        context = self.get_context()

        self.icx_storage._put_genesis_data_account(
            context, CoinPartType.GENERAL, self._sender, 10000 * 10 ** 18)

        tx_hash = os.urandom(32)
        amount = 10001 * 10 ** 18
        block_height = randrange(100, 10000)
        term = get_rand_term()

        # out of balance
        # noinspection PyTypeChecker
        with self.assertRaises(OutOfBalanceException) as e:
            self._engine.add_deposit(context, tx_hash, self._sender, self._score_address,
                                     amount, block_height, term)
        # noinspection PyUnresolvedReferences
        self.assertEqual('Out of balance', e.exception.message)

    def test_deposit_fee_available_head_ids(self):
        context = self.get_context()
        tx_hash = os.urandom(32)
        amount = 10000 * 10 ** 18
        block_height = 1000

        self.icx_storage._put_genesis_data_account(context,
                                                   CoinPartType.GENERAL,
                                                   self._sender,
                                                   amount)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)

        self.assertEqual(deposit_meta.available_head_id_of_virtual_step, None)
        self.assertEqual(deposit_meta.available_head_id_of_deposit, None)

        self._engine.add_deposit(context, tx_hash, self._sender, self._score_address, amount, block_height,
                                 FeeEngine._MIN_DEPOSIT_TERM)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
        self.assertEqual(deposit_meta.available_head_id_of_virtual_step, tx_hash)
        self.assertEqual(deposit_meta.available_head_id_of_deposit, tx_hash)

    def test_deposit_fee_expires_updated(self):
        context = self.get_context()
        tx_hash = os.urandom(32)
        amount = 10000 * 10 ** 18
        block_height = 1000
        term = FeeEngine._MIN_DEPOSIT_TERM

        self.icx_storage._put_genesis_data_account(context,
                                                   CoinPartType.GENERAL,
                                                   self._sender,
                                                   amount)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)

        self.assertEqual(deposit_meta.expires_of_virtual_step, -1)
        self.assertEqual(deposit_meta.expires_of_deposit, -1)

        self._engine.add_deposit(context, tx_hash, self._sender, self._score_address, amount, block_height, term)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
        self.assertEqual(deposit_meta.expires_of_virtual_step, block_height + term)
        self.assertEqual(deposit_meta.expires_of_deposit, block_height + term)

    def test_withdraw_fee_without_penalty(self):
        context = self.get_context()

        tx_hash = os.urandom(32)
        amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
        block_height = randrange(100, 10000)
        term = get_rand_term()

        self._engine.add_deposit(
            context, tx_hash, self._sender, self._score_address, amount, block_height, term)

        before_sender_balance = self._icx_engine.get_balance(context, self._sender)
        self._engine.withdraw_deposit(context, self._sender, tx_hash, block_height + term + 1)
        after_sender_balance = self._icx_engine.get_balance(context, self._sender)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, block_height)
        self.assertIsNone(deposit_info)
        self.assertEqual(amount, after_sender_balance - before_sender_balance)

    def test_withdraw_fee_with_penalty(self):
        context = self.get_context()

        tx_hash = os.urandom(32)
        amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
        block_height = randrange(100, 10000)
        term = get_rand_term()

        self._engine.add_deposit(
            context, tx_hash, self._sender, self._score_address, amount, block_height, term)

        before_sender_balance = self._icx_engine.get_balance(context, self._sender)
        self._engine.withdraw_deposit(context, self._sender, tx_hash, block_height + term - 1)
        after_sender_balance = self._icx_engine.get_balance(context, self._sender)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, block_height)
        self.assertIsNone(deposit_info)
        self.assertGreater(after_sender_balance - before_sender_balance, 0)
        self.assertLessEqual(after_sender_balance - before_sender_balance, amount)

    def test_withdraw_fee_and_updates_previous_and_next_link_ascending(self):
        """
        Given: There are four deposits.
        When : Withdraws all of them sequentially(ascending).
        Then : Checks if the previous and next link update correctly.
        """
        context = self.get_context()

        cnt_deposit = 4
        block_height = randrange(100, 10000)
        arr_tx_hash = []
        for i in range(cnt_deposit):
            arr_tx_hash.append(os.urandom(32))
            amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
            term = get_rand_term()
            block_height += 1
            self._engine.add_deposit(
                context, arr_tx_hash[i], self._sender, self._score_address, amount, block_height, term)

        for i in range(cnt_deposit):
            target_deposit = self._engine.get_deposit(context, arr_tx_hash[i])
            self._engine.withdraw_deposit(context, self._sender, arr_tx_hash[i], block_height + term // 2)

            if cnt_deposit - 1 == i:
                self.assertIsNone(target_deposit.next_id)
                break

            next_deposit = self._engine.get_deposit(context, target_deposit.next_id)
            self.assertEqual(next_deposit.prev_id, None)

            deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
            self.assertEqual(next_deposit.id, deposit_meta.head_id)

    def test_withdraw_fee_and_updates_previous_and_next_link_descending(self):
        """
        Given: There are four deposits.
        When : Withdraws all of them sequentially(descending).
        Then : Checks if the previous and next link update correctly.
        """
        context = self.get_context()

        cnt_deposit = 4
        block_height = randrange(100, 10000)
        arr_tx_hash = []
        for i in range(cnt_deposit):
            arr_tx_hash.append(os.urandom(32))
            amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
            term = get_rand_term()
            block_height += 1
            self._engine.add_deposit(
                context, arr_tx_hash[i], self._sender, self._score_address, amount, block_height, term)

        for i in range(cnt_deposit - 1, -1, -1):
            target_deposit = self._engine.get_deposit(context, arr_tx_hash[i])
            self._engine.withdraw_deposit(context, self._sender, arr_tx_hash[i], block_height + term // 2)

            if i == 0:
                self.assertIsNone(target_deposit.prev_id)
                break

            prev_deposit = self._engine.get_deposit(context, target_deposit.prev_id)
            self.assertEqual(prev_deposit.next_id, None)

            deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
            self.assertEqual(prev_deposit.id, deposit_meta.tail_id)

    @unittest.skipIf(FIXED_TERM is True, "FIXED_TERM is true")
    def test_withdraw_fee_when_available_head_id_of_virtual_step_is_same_as_deposit_id(self):
        """
        Given: There are four deposits. Only the last deposit has enough to long term.
        When : Available head id of the virtual step is same as deposit id.
        Then : Searches for next deposit id which is available to use virtual step
               and where expires of the deposit is more than current block height.
               In the test, only the last deposit is available.
        """
        context = self.get_context()

        cnt_deposit = 4
        block_height = randrange(100, 10000)
        arr_tx_hash = []
        for i in range(cnt_deposit):
            arr_tx_hash.append(os.urandom(32))
            amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
            block_height += 1

            if i != cnt_deposit - 1:
                term = FeeEngine._MIN_DEPOSIT_TERM
            else:
                term = FeeEngine._MAX_DEPOSIT_TERM

            self._engine.add_deposit(
                context, arr_tx_hash[i], self._sender, self._score_address, amount, block_height, term)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
        self.assertEqual(deposit_meta.available_head_id_of_virtual_step, arr_tx_hash[0])

        self._engine.withdraw_deposit(context, self._sender, arr_tx_hash[0],
                                      block_height + FeeEngine._MAX_DEPOSIT_TERM // 2)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
        self.assertEqual(deposit_meta.available_head_id_of_virtual_step, arr_tx_hash[len(arr_tx_hash) - 1])

    @unittest.skipIf(FIXED_TERM is True, "FIXED_TERM is true")
    def test_withdraw_fee_when_available_head_id_of_deposit_is_same_as_deposit_id(self):
        """
        Given: There are four deposits. Only the third deposit has enough long term.
        When : Available head id of deposit is same as deposit id.
        Then : Searches for next deposit id which is available to use deposit
               and where expires of the deposit is more than current block height.
               In the test, only the third deposit is available.
        """
        context = self.get_context()

        cnt_deposit = 4
        block_height = randrange(100, 10000)
        arr_tx_hash = []
        for i in range(cnt_deposit):
            arr_tx_hash.append(os.urandom(32))
            amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
            block_height += 1

            if i != cnt_deposit - 2:
                term = FeeEngine._MIN_DEPOSIT_TERM
            else:
                term = FeeEngine._MAX_DEPOSIT_TERM

            self._engine.add_deposit(
                context, arr_tx_hash[i], self._sender, self._score_address, amount, block_height, term)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
        self.assertEqual(deposit_meta.available_head_id_of_deposit, arr_tx_hash[0])

        self._engine.withdraw_deposit(context, self._sender, arr_tx_hash[0],
                                      block_height + FeeEngine._MAX_DEPOSIT_TERM // 2)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
        self.assertEqual(deposit_meta.available_head_id_of_deposit, arr_tx_hash[len(arr_tx_hash) - 2])

    @unittest.skipIf(FIXED_TERM is True, "FIXED_TERM is true")
    def test_withdraw_fee_to_check_setting_on_next_max_expires(self):
        """
        Given: There are four deposits.
        When : Expires of the withdrawal deposit is same as expires.
        Then : Searches for max expires which is more than current block height.
        """
        context = self.get_context()

        cnt_deposit = 4
        block_height = randrange(100, 10000)
        arr_tx_hash = []
        last_expires = 0
        org_last_expires = 0
        for i in range(cnt_deposit):
            arr_tx_hash.append(os.urandom(32))
            amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
            block_height += 1

            if i != 0:
                term = FeeEngine._MIN_DEPOSIT_TERM
                if block_height + term > last_expires:
                    last_expires = block_height + term
            else:
                term = FeeEngine._MAX_DEPOSIT_TERM
                org_last_expires = block_height + term
            self._engine.add_deposit(
                context, arr_tx_hash[i], self._sender, self._score_address, amount, block_height, term)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
        self.assertEqual(deposit_meta.available_head_id_of_virtual_step, arr_tx_hash[0])
        self.assertEqual(deposit_meta.expires_of_virtual_step, org_last_expires)
        self.assertEqual(deposit_meta.expires_of_deposit, org_last_expires)

        self._engine.withdraw_deposit(context, self._sender, arr_tx_hash[0],
                                      block_height + FeeEngine._MIN_DEPOSIT_TERM // 2)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
        self.assertEqual(deposit_meta.expires_of_virtual_step, last_expires)
        self.assertEqual(deposit_meta.expires_of_deposit, last_expires)

    @unittest.skipIf(FIXED_TERM is True, "FIXED_TERM is true")
    def test_withdraw_fee_of_last_deposit_to_check_setting_on_next_max_expires(self):
        """
        Given: There are four deposits.
        When : Expires of the withdrawal deposit which is the last one is same as expires.
        Then : Searches for max expires which is more than current block height.
        """
        context = self.get_context()

        cnt_deposit = 4
        block_height = randrange(100, 10000)
        arr_tx_hash = []
        last_expires = 0
        org_last_expires = 0
        for i in range(cnt_deposit):
            arr_tx_hash.append(os.urandom(32))
            amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
            block_height += 1

            if i != cnt_deposit-1:
                term = FeeEngine._MIN_DEPOSIT_TERM
                if block_height + term > last_expires:
                    last_expires = block_height + term
            else:
                term = FeeEngine._MAX_DEPOSIT_TERM
                org_last_expires = block_height + term
            self._engine.add_deposit(
                context, arr_tx_hash[i], self._sender, self._score_address, amount, block_height, term)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
        self.assertEqual(deposit_meta.available_head_id_of_virtual_step, arr_tx_hash[0])
        self.assertEqual(deposit_meta.available_head_id_of_deposit, arr_tx_hash[0])
        self.assertEqual(deposit_meta.expires_of_virtual_step, org_last_expires)
        self.assertEqual(deposit_meta.expires_of_deposit, org_last_expires)

        # Withdraws the last one
        self._engine.withdraw_deposit(context, self._sender, arr_tx_hash[cnt_deposit - 1],
                                      block_height + FeeEngine._MIN_DEPOSIT_TERM // 2)

        deposit_meta = self._engine._get_or_create_deposit_meta(context, self._score_address)
        self.assertEqual(deposit_meta.expires_of_virtual_step, last_expires)
        self.assertEqual(deposit_meta.expires_of_deposit, last_expires)

    def test_get_deposit_info(self):
        context = self.get_context()

        tx_hash = os.urandom(32)
        amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
        block_height = randrange(100, 10000)
        term = get_rand_term()

        before_sender_balance = self._icx_engine.get_balance(context, self._sender)
        self._engine.add_deposit(
            context, tx_hash, self._sender, self._score_address, amount, block_height, term)
        after_sender_balance = self._icx_engine.get_balance(context, self._sender)

        self.assertEqual(amount, before_sender_balance - after_sender_balance)

        deposit = self._engine.get_deposit(context, tx_hash)

        self.assertEqual(tx_hash, deposit.id)
        self.assertEqual(self._score_address, deposit.score_address)
        self.assertEqual(self._sender, deposit.sender)
        self.assertEqual(amount, deposit.deposit_amount)
        self.assertEqual(block_height, deposit.created)
        self.assertEqual(block_height + term, deposit.expires)

    def test_charge_transaction_fee_without_sharing(self):
        context = self.get_context()

        step_price = 10 ** 10
        used_step = 10 ** 10

        tx_hash = os.urandom(32)
        amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
        block_height = randrange(100, 10000)
        term = get_rand_term()

        self._engine.add_deposit(
            context, tx_hash, self._sender, self._score_address, amount, block_height, term)

        before_sender_balance = self._icx_engine.get_balance(context, self._sender)

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, block_height)

        after_sender_balance = self._icx_engine.get_balance(context, self._sender)

        self.assertEqual(step_price * used_step, before_sender_balance - after_sender_balance)

    def test_charge_transaction_fee_sharing_deposit(self):
        context = self.get_context()

        step_price = 10 ** 10
        used_step = 10 ** 10

        tx_hash = os.urandom(32)
        amount = randrange(FeeEngine._MIN_DEPOSIT_AMOUNT, FeeEngine._MAX_DEPOSIT_AMOUNT)
        block_height = randrange(100, 10000)
        term = get_rand_term()
        self._engine.add_deposit(
            context, tx_hash, self._sender, self._score_address, amount, block_height, term)

        ratio = 50
        context.fee_sharing_proportion = ratio

        before_sender_balance = self._icx_engine.get_balance(context, self._sender)

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, block_height)

        after_sender_balance = self._icx_engine.get_balance(context, self._sender)

        score_charging_step = used_step * ratio // 100
        sender_charging_step = used_step - score_charging_step
        self.assertEqual(
            step_price * sender_charging_step, before_sender_balance - after_sender_balance)

    def test_charge_fee_from_score_by_virtual_step_single_deposit(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
        When :  Current  block is 120 so  1st deposit is unavailable
        Then :  Pays fee by virtual step of 2nd.
                update indices to 2nd
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 100),
            (os.urandom(32), 50, 180, 100, 100),
            (os.urandom(32), 70, 150, 100, 100),
            (os.urandom(32), 90, 250, 100, 100),
            (os.urandom(32), 110, 200, 100, 100)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 120
        used_step = 80

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        before_virtual_step = deposit_info.available_virtual_step

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(used_step, before_virtual_step - after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        self.assertEqual(deposits[1][0], deposit_meta.available_head_id_of_virtual_step)

    def test_charge_fee_from_score_by_virtual_step_single_deposit_next_head(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
        When :  Current  block is 120 so  1st deposit is unavailable
        Then :  Pays fee by virtual step of 2nd.
                the virtual steps in 2nd are fully consumed
                update indices to 3rd
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 100),
            (os.urandom(32), 50, 180, 100, 100),
            (os.urandom(32), 70, 150, 100, 100),
            (os.urandom(32), 90, 250, 100, 100),
            (os.urandom(32), 110, 200, 100, 100)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 120
        used_step = 100

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        before_virtual_step = deposit_info.available_virtual_step

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(used_step, before_virtual_step - after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        self.assertEqual(deposits[2][0], deposit_meta.available_head_id_of_virtual_step)

    def test_charge_fee_from_score_by_virtual_step__single_deposit_next_head_next_expire(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
        When :  Current  block is 190 so  4th, 5th deposits are available
        Then :  Pays fee by virtual step of 4th.
                the virtual steps in 4th are fully consumed
                update indices to 5th
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 100),
            (os.urandom(32), 50, 180, 100, 100),
            (os.urandom(32), 70, 150, 100, 100),
            (os.urandom(32), 90, 250, 100, 100),
            (os.urandom(32), 110, 200, 100, 100)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 190
        used_step = 100

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        before_virtual_step = deposit_info.available_virtual_step

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(used_step, before_virtual_step - after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        self.assertEqual(deposits[4][0], deposit_meta.available_head_id_of_virtual_step)
        self.assertEqual(deposits[4][2], deposit_meta.expires_of_virtual_step)

    def test_charge_fee_from_score_by_virtual_step__single_deposit_next_head_next_expire_none(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
        When :  Current  block is 210 so only 4th deposit is available
        Then :  Pays fee by virtual step of 4th.
                the virtual steps in 4th are fully consumed
                should update indices but there are no more available deposits
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 100),
            (os.urandom(32), 50, 180, 100, 100),
            (os.urandom(32), 70, 150, 100, 100),
            (os.urandom(32), 90, 250, 100, 100),
            (os.urandom(32), 110, 200, 100, 100)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 210
        used_step = 100

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        before_virtual_step = deposit_info.available_virtual_step

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(used_step, before_virtual_step - after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        self.assertEqual(None, deposit_meta.available_head_id_of_virtual_step)
        self.assertEqual(-1, deposit_meta.expires_of_virtual_step)

    def test_charge_fee_from_score_by_virtual_step_multiple_deposit(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
        When :  Current  block is 120 so 1st deposit is unavailable
        Then :  Pays fee by virtual step through 2nd, 3rd, 4th.
                the virtual steps in 2nd, 3rd are fully consumed
                update indices to 4th
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 100),
            (os.urandom(32), 50, 180, 100, 100),
            (os.urandom(32), 70, 150, 100, 100),
            (os.urandom(32), 90, 250, 100, 100),
            (os.urandom(32), 110, 200, 100, 100)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 120
        used_step = 250

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        before_virtual_step = deposit_info.available_virtual_step

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(used_step, before_virtual_step - after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        self.assertEqual(deposits[3][0], deposit_meta.available_head_id_of_virtual_step)

    def test_charge_fee_from_score_by_combine_by_single_deposit(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
                Remaining virtual steps are in 5th deposit
        When :  Current  block is 120 so 1st deposit is unavailable
                Remaining virtual steps are not enough to pay fees
        Then :  Pays fee by virtual step first.
                Pays remaining fee by deposit of 2nd
                update indices to 2nd
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 0),
            (os.urandom(32), 50, 180, 100, 0),
            (os.urandom(32), 70, 150, 100, 0),
            (os.urandom(32), 90, 250, 100, 0),
            (os.urandom(32), 110, 200, 100, 50)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 120
        used_step = 70

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(0, after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        self.assertEqual(None, deposit_meta.available_head_id_of_virtual_step)
        self.assertEqual(-1, deposit_meta.expires_of_virtual_step)

        self.assertEqual(deposits[1][0], deposit_meta.available_head_id_of_deposit)

    def test_charge_fee_from_score_by_combine_next_head(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
                Remaining virtual steps are in 5th deposit
        When :  Current  block is 120 so 1st deposit is unavailable
                Remaining virtual steps are not enough to pay fees
        Then :  Pays fee by virtual step first.
                Pays remaining fee by deposit of 2nd
                2nd deposit is fully consumed so update indices to 3rd
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 0),
            (os.urandom(32), 50, 180, 100, 0),
            (os.urandom(32), 70, 150, 100, 0),
            (os.urandom(32), 90, 250, 100, 0),
            (os.urandom(32), 110, 200, 100, 50)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 120
        used_step = 140

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(0, after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        self.assertEqual(None, deposit_meta.available_head_id_of_virtual_step)
        self.assertEqual(-1, deposit_meta.expires_of_virtual_step)

        self.assertEqual(deposits[2][0], deposit_meta.available_head_id_of_deposit)

    def test_charge_fee_from_score_by_combine_next_head_next_expire(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
                Remaining virtual steps are in 5th deposit
        When :  Current  block is 190 so 1st, 2nd, 3rd deposits are unavailable
                Remaining virtual steps are not enough to pay fees
        Then :  Pays fee by virtual step first.
                Pays remaining fee by deposit of 4th
                4th deposit is fully consumed so update indices to 5th
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 0),
            (os.urandom(32), 50, 180, 100, 0),
            (os.urandom(32), 70, 150, 100, 0),
            (os.urandom(32), 90, 250, 100, 0),
            (os.urandom(32), 110, 200, 100, 50)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 190
        used_step = 140

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(0, after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        self.assertEqual(None, deposit_meta.available_head_id_of_virtual_step)
        self.assertEqual(-1, deposit_meta.expires_of_virtual_step)

        self.assertEqual(deposits[4][0], deposit_meta.available_head_id_of_deposit)
        self.assertEqual(deposits[4][2], deposit_meta.expires_of_deposit)

    def test_charge_fee_from_score_by_combine_next_head_next_expire_none(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
                Remaining virtual steps are in 4th and 5th deposit
        When :  Current  block is 220 so 5th deposit is unavailable
                Remaining virtual steps are not enough to pay fees
        Then :  Pays fee by virtual step first.
                Pays remaining fee by deposit of 4th
                All available deposits are consumed so make the SCORE disabled
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 0),
            (os.urandom(32), 50, 180, 100, 0),
            (os.urandom(32), 70, 150, 100, 0),
            (os.urandom(32), 90, 250, 100, 50),
            (os.urandom(32), 110, 200, 100, 100)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 220
        used_step = 140

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(0, after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        self.assertEqual(None, deposit_meta.available_head_id_of_virtual_step)
        self.assertEqual(-1, deposit_meta.expires_of_virtual_step)

        self.assertEqual(None, deposit_meta.available_head_id_of_deposit)
        self.assertEqual(-1, deposit_meta.expires_of_deposit)

    def test_charge_fee_from_score_by_combine_multiple_deposit(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
                Remaining virtual steps are in 5th deposit
        When :  Current  block is 120 so 1st deposit is unavailable
                Remaining virtual steps are not enough to pay fees
        Then :  Pays fee by virtual step first.
                Pays remaining fee by deposit through 2nd and 3rd deposit.
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 0),
            (os.urandom(32), 50, 180, 100, 0),
            (os.urandom(32), 70, 150, 100, 0),
            (os.urandom(32), 90, 250, 100, 0),
            (os.urandom(32), 110, 200, 100, 50)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 120
        used_step = 230

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(0, after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        self.assertEqual(None, deposit_meta.available_head_id_of_virtual_step)
        self.assertEqual(-1, deposit_meta.expires_of_virtual_step)

        # Asserts indices are updated
        self.assertEqual(deposits[3][0], deposit_meta.available_head_id_of_deposit)
        self.assertEqual(deposits[3][2], deposit_meta.expires_of_deposit)

    def test_charge_fee_from_score_by_combine_additional_pay(self):
        """
        Given:  Five deposits. The fourth deposit is the max expire.
                Remaining virtual steps are in 4th and 5th deposit
        When :  Current  block is 220 so 5th deposit is unavailable
                Remaining virtual steps are not enough to pay fees
                Available deposits are also not enough to pay fees
        Then :  Pays fees regardless minimum remaining amount
                and make the SCORE disabled
        """

        context = self.get_context()

        # tx_hash, from_block, to_block, deposit_amount, virtual_step_amount
        deposits = [
            (os.urandom(32), 10, 100, 100, 0),
            (os.urandom(32), 50, 180, 100, 0),
            (os.urandom(32), 70, 150, 100, 0),
            (os.urandom(32), 90, 250, 100, 50),
            (os.urandom(32), 110, 200, 100, 100)
        ]
        self._set_up_deposits(context, deposits)

        step_price = 1
        current_block = 220
        used_step = 150

        self._engine.charge_transaction_fee(
            context, self._sender, self._score_address, step_price, used_step, current_block)

        deposit_info = self._engine.get_deposit_info(context, self._score_address, current_block)
        after_virtual_step = deposit_info.available_virtual_step

        self.assertEqual(0, after_virtual_step)

        deposit_meta = self.fee_storage.get_deposit_meta(context, self._score_address)
        # Asserts virtual step disabled
        self.assertEqual(None, deposit_meta.available_head_id_of_virtual_step)
        self.assertEqual(-1, deposit_meta.expires_of_virtual_step)

        # Asserts deposit disabled
        self.assertEqual(None, deposit_meta.available_head_id_of_deposit)
        self.assertEqual(-1, deposit_meta.expires_of_deposit)

    def _set_up_deposits(self, context, deposits):
        context.fee_sharing_proportion = 100

        self._engine._MIN_DEPOSIT_TERM = 50
        self._engine._MIN_DEPOSIT_AMOUNT = 10

        for deposit in deposits:
            tx_hash = deposit[0]
            amount = deposit[3]
            block_height = deposit[1]
            term = deposit[2] - block_height

            # self._engine._calculate_virtual_step_issuance = Mock(return_value=deposit[4])
            VirtualStepCalculator.calculate_virtual_step = Mock(return_value=deposit[4])

            self._engine.add_deposit(
                context, tx_hash, self._sender, self._score_address, amount, block_height, term)