Advertisement
Guest User

Untitled

a guest
Jan 14th, 2023
52
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Python 14.96 KB | None | 0 0
  1. tad_contract = importlib.import_module('con_tad_tst_001')
  2.  
  3. vaults = Hash(default_value=0)
  4. stability_rate = Hash(default_value=1)
  5. cdp = Hash(default_value=0)
  6. stability_pool = Hash(default_value=0)
  7.  
  8. temporary_var = Variable()
  9.  
  10. @construct
  11. def seed():
  12.     vaults['OWNER'] = ctx.caller
  13.     cdp['current_value'] = 0
  14.     vaults['list'] = [0]
  15.     vaults['current_number'] = 1
  16.  
  17.     vaults['oracle'] = 'oracle'  # dummy for testing purposes
  18.  
  19.     vaults[0, 'collateral_type'] = 'currency'
  20.     vaults[0, 'minimum_collateralization'] = 1.5
  21.     vaults[0, 'minimum_auction_time'] = 259200
  22.     vaults[0, 'cap'] = 100000
  23.     vaults[0, 'weight'] = 10
  24.  
  25.     stability_rate[0] = 1.0000000015469297  # default value, change on deployment
  26.  
  27.  
  28. @export
  29. def get_timestamp():
  30.     # https://developers.lamden.io/docs/smart-contracts/datetime-module/
  31.     td = now - datetime.datetime(1970, 1, 1, 0, 0, 0)
  32.     return fix_decimal(td.seconds)
  33.  
  34.  
  35. @export
  36. def create_vault(vault_type: int, amount_of_tad: float,
  37.                  amount_of_collateral: float):
  38.     assert vault_type in vaults['list'], 'Not an available contract!'
  39.     # interface enforcement is unnecessary because collateral types should be pre-vetted
  40.     collateral = importlib.import_module(
  41.         vaults[vault_type, 'collateral_type'])
  42.     oracle = importlib.import_module(vaults['oracle'])
  43.  
  44.     price = oracle.get_price(vault_type)
  45.  
  46.     assert amount_of_tad > 0, 'Amount of tad must be positive!'
  47.     assert vaults[vault_type, 'total'] + amount_of_tad <= vaults[vault_type,
  48.                                                                  'cap'], 'The allowance is not sufficent!'
  49.  
  50.     assert fix_decimal((amount_of_collateral * price) / \
  51.         amount_of_tad) >= vaults[vault_type,
  52.                                 'minimum_collateralization'], 'Not enough collateral!'
  53.  
  54.     cdp_number = cdp['current_value']
  55.     cdp['current_value'] += 1
  56.  
  57.     cdp[cdp_number, 'owner'] = ctx.caller
  58.     cdp[cdp_number, 'open'] = True
  59.  
  60.     cdp[cdp_number, 'collateral_type'] = vaults[vault_type, 'collateral_type']
  61.     cdp[cdp_number, 'vault_type'] = vault_type
  62.     cdp[cdp_number, 'tad'] = amount_of_tad
  63.     cdp[cdp_number, 'collateral_amount'] = amount_of_collateral
  64.     cdp[cdp_number, 'time'] = get_timestamp()
  65.  
  66.     collateral.approve(amount=amount_of_collateral, to=ctx.this)
  67.     collateral.transfer_from(amount=amount_of_collateral,
  68.                              to=ctx.this, main_account=ctx.caller)
  69.  
  70.     tad_contract.mint(amount=amount_of_tad)
  71.     tad_contract.transfer(amount=amount_of_tad, to=ctx.caller)
  72.  
  73.     vaults[vault_type, 'issued'] += amount_of_tad
  74.     vaults[vault_type, 'total'] += amount_of_tad
  75.  
  76.     return cdp_number
  77.  
  78.  
  79. @export
  80. def close_vault(cdp_number: int):
  81.     assert cdp[cdp_number, 'owner'] == ctx.caller, 'Not the owner!'
  82.     assert cdp[cdp_number, 'open'] == True, 'Vault has already been closed!'
  83.  
  84.     collateral = importlib.import_module(
  85.         vaults[cdp[cdp_number, 'vault_type'], 'collateral_type'])
  86.  
  87.     stability_ratio = fix_decimal(vaults[cdp[cdp_number, 'vault_type'], 'total'] / \
  88.         vaults[cdp[cdp_number, 'vault_type'], 'issued'])
  89.     redemption_cost = cdp[cdp_number, 'tad'] * stability_ratio
  90.     fee = redemption_cost * \
  91.         (stability_rate[cdp[cdp_number, 'vault_type']] **
  92.          (get_timestamp() - cdp[cdp_number, 'time'])) - redemption_cost
  93.  
  94.     amount = redemption_cost + fee
  95.     tad_contract.transfer_from(
  96.         amount=amount, to=ctx.this, main_account=ctx.caller)
  97.     tad_contract.burn(amount=redemption_cost)
  98.  
  99.     stability_pool[cdp[cdp_number, 'vault_type']] += fee
  100.  
  101.     vaults[cdp[cdp_number, 'vault_type'], 'issued'] -= cdp[cdp_number, 'tad']
  102.     # This is only different if the ratio is different
  103.     vaults[cdp[cdp_number, 'vault_type'], 'total'] -= redemption_cost
  104.  
  105.     cdp[cdp_number, 'open'] = False
  106.  
  107.     # Return collateral
  108.     collateral.transfer(
  109.         amount=cdp[cdp_number, 'collateral_amount'], to=ctx.caller)
  110.  
  111.     return amount
  112.  
  113.  
  114. @export
  115. def fast_force_close_vault(cdp_number: int):
  116.     assert_insufficent_collateral(cdp_number=cdp_number)
  117.     assert cdp[cdp_number, 'open'] is True, 'Vault has already been closed!'
  118.  
  119.     collateral = importlib.import_module(
  120.         vaults[cdp[cdp_number, 'vault_type'], 'collateral_type'])
  121.     oracle = importlib.import_module(vaults['oracle'])
  122.  
  123.     stability_ratio = fix_decimal(vaults[cdp[cdp_number, 'vault_type'],
  124.                              'total'] / vaults[cdp[cdp_number, 'vault_type'], 'issued'])
  125.     redemption_cost_without_fee = cdp[cdp_number,
  126.                                       'tad'] * stability_ratio
  127.     redemption_cost = redemption_cost_without_fee * fix_decimal(1.1)
  128.     fee = redemption_cost_without_fee * \
  129.         (stability_rate[cdp[cdp_number, 'vault_type']]
  130.          ** (get_timestamp() - cdp[cdp_number, 'time'])) - redemption_cost_without_fee
  131.     redemption_cost += fee
  132.  
  133.     amount_of_collateral = cdp[cdp_number, 'collateral_amount']
  134.     price = oracle.get_price(cdp[cdp_number, 'vault_type'])
  135.     collateral_percent = fix_decimal((amount_of_collateral * price) / \
  136.         redemption_cost)
  137.  
  138.     if collateral_percent >= fix_decimal(1.03):
  139.         tad_contract.transfer_from(
  140.             amount=redemption_cost, to=ctx.this, main_account=ctx.caller)
  141.         tad_contract.burn(amount=redemption_cost_without_fee)
  142.         amount = fix_decimal((redemption_cost * fix_decimal(1.03)) / price) # Double check this math is correct
  143.  
  144.         collateral.transfer(amount=amount, to=ctx.caller)
  145.         collateral.transfer(amount=amount_of_collateral -
  146.                             amount, to=cdp[cdp_number, 'owner'])
  147.  
  148.         vaults[cdp[cdp_number, 'vault_type'],
  149.                'issued'] -= cdp[cdp_number, 'tad']
  150.         vaults[cdp[cdp_number, 'vault_type'],
  151.                'total'] -= redemption_cost_without_fee
  152.  
  153.     else:
  154.         redemption_cost, redemption_cost_without_fee = redemption_cost * \
  155.             fix_decimal(collateral_percent / fix_decimal(1.03)), redemption_cost_without_fee * \
  156.             fix_decimal(collateral_percent / fix_decimal(1.03))
  157.  
  158.         tad_contract.transfer_from(
  159.             amount=redemption_cost, to=ctx.this, main_account=ctx.caller)
  160.         tad_contract.burn(amount=redemption_cost_without_fee)
  161.  
  162.         amount = cdp[cdp_number, 'collateral_amount']
  163.  
  164.         collateral.transfer(amount=amount, to=ctx.caller)
  165.  
  166.         vaults[cdp[cdp_number, 'vault_type'],
  167.                'issued'] -= cdp[cdp_number, 'tad']
  168.         vaults[cdp[cdp_number, 'vault_type'],
  169.                'total'] -= redemption_cost_without_fee
  170.  
  171.     stability_pool[cdp[cdp_number, 'vault_type']
  172.                    ] += redemption_cost - redemption_cost_without_fee
  173.  
  174.     cdp[cdp_number, 'open'] = False
  175.  
  176.     return amount
  177.  
  178.  
  179. @export
  180. def open_force_close_auction(cdp_number: int):
  181.     assert_insufficent_collateral(cdp_number=cdp_number)
  182.  
  183.     assert cdp[cdp_number, 'owner'] != 0, 'Nonexistent cdp'
  184.     assert cdp[cdp_number, 'auction',
  185.                'open'] is not True, 'Auction is already taking place!' # Probably a redundant check, can be removed
  186.     assert cdp[cdp_number, 'open'] is True, 'Vault has already been closed!'
  187.  
  188.     # This contract may only be bid on, and not closed
  189.     cdp[cdp_number, 'open'] = False
  190.     cdp[cdp_number, 'auction', 'open'] = True
  191.  
  192.     cdp[cdp_number, 'auction', 'highest_bidder'] = ctx.caller
  193.     cdp[cdp_number, 'auction', 'top_bid'] = 0.0
  194.  
  195.     cdp[cdp_number, 'auction', 'time'] = get_timestamp()
  196.  
  197.     return True
  198.  
  199.  
  200. @export
  201. def bid_on_force_close(cdp_number: int, amount: float):
  202.     assert cdp[cdp_number, 'owner'] != 0, 'Nonexistent cdp'
  203.     assert cdp[cdp_number, 'auction',
  204.                'open'] is True, 'Auction is not open!'
  205.     assert amount > cdp[cdp_number, 'auction',
  206.                         'top_bid'], 'There is already a higher bid!'
  207.  
  208.     if cdp[cdp_number, 'auction', ctx.caller, 'bid'] is not None:
  209.         tad_contract.transfer_from(
  210.             amount=amount - cdp[cdp_number, 'auction', ctx.caller, 'bid'],
  211.             to=ctx.this, main_account=ctx.caller)
  212.  
  213.     else:
  214.         tad_contract.transfer_from(
  215.             amount=amount, to=ctx.this, main_account=ctx.caller)
  216.  
  217.     cdp[cdp_number, 'auction', 'highest_bidder'] = ctx.caller
  218.     cdp[cdp_number, 'auction', 'top_bid'] = amount
  219.     cdp[cdp_number, 'auction', ctx.caller, 'bid'] = amount
  220.  
  221.     return True
  222.  
  223.  
  224. @export
  225. def settle_force_close(cdp_number: int):
  226.     assert cdp[cdp_number, 'owner'] != 0, 'Nonexistent cdp'
  227.     assert cdp[cdp_number, 'auction', 'open'] is True, 'Auction is not open!'
  228.  
  229.     assert get_timestamp() - cdp[cdp_number, 'auction', 'time'] > vaults[cdp[cdp_number, 'vault_type'],
  230.                                                                          'minimum_auction_time'], 'Auction is still open!'
  231.  
  232.     collateral = importlib.import_module(
  233.         vaults[cdp[cdp_number, 'vault_type'], 'collateral_type'])
  234.  
  235.     cdp[cdp_number, 'auction', 'settled'] = True
  236.     cdp[cdp_number, 'open'] = False
  237.     cdp[cdp_number, 'auction', 'open'] = False
  238.  
  239.     cdp[cdp_number, 'auction', cdp[cdp_number,
  240.                                    'auction', 'highest_bidder'], 'bid'] = 0
  241.  
  242.     fee = cdp[cdp_number, 'auction', 'top_bid'] * 0.1
  243.     collateral.transfer_from(
  244.         amount=cdp[cdp_number, 'collateral_amount'], to=ctx.caller, main_account=ctx.this)
  245.     tad_contract.burn(amount=cdp[cdp_number, 'auction', 'top_bid'] - fee)
  246.  
  247.     stability_pool[cdp[cdp_number, 'vault_type']] += fee
  248.  
  249.     vaults[cdp[cdp_number, 'vault_type'], 'issued'] -= cdp[cdp_number, 'tad']
  250.     vaults[cdp[cdp_number, 'vault_type'],
  251.            'total'] -= cdp[cdp_number, 'auction', 'top_bid'] - fee  # Fee is not burned, so it does not count
  252.  
  253.     return cdp[cdp_number, 'auction', 'highest_bidder'], cdp[cdp_number,
  254.                                                              'auction', 'top_bid']
  255.  
  256.  
  257. @export
  258. def claim_unwon_bid(cdp_number: int):
  259.     assert cdp[cdp_number, 'owner'] != 0, 'Nonexistent cdp'
  260.     assert cdp[cdp_number, 'auction',
  261.                'settled'] is True, 'Auction is still open or not opened!'
  262.  
  263.     tad_contract.transfer(
  264.         to=ctx.caller, amount=cdp[cdp_number, 'auction', ctx.caller, 'bid'])
  265.     cdp[cdp_number, 'auction', ctx.caller, 'bid'] = 0
  266.  
  267.     return True
  268.  
  269.  
  270. @export
  271. def sync_stability_pool(vault_type: int):
  272.     assert vault_type in vaults['list'], 'Not an available contract!'
  273.  
  274.     default_amount = vaults[vault_type, 'total'] - vaults[vault_type, 'issued']
  275.  
  276.     if default_amount > stability_pool[vault_type]:
  277.         vaults[vault_type, 'issued'] += stability_pool[vault_type]
  278.         stability_pool[vault_type] = 0
  279.         # Return new ratio
  280.         return fix_decimal(vaults[vault_type, 'issued'] / vaults[vault_type, 'total'])
  281.  
  282.     else:  # This also applies to negatives and zeros, although those situations are unlikely
  283.         vaults[vault_type, 'issued'] = vaults[vault_type, 'total']
  284.         stability_pool[vault_type] -= default_amount
  285.  
  286.         return 1.0  # The ratio is perfectly equal
  287.  
  288.  
  289. @export
  290. def export_rewards(vault_type: int, amount: float):
  291.     assert vaults[vault_type, 'DSR', 'owner'] == ctx.caller, 'Not the owner!'
  292.     assert stability_pool[vault_type] >= amount, 'Not enough tad in stability pool to export!'
  293.  
  294.     stability_pool[vault_type] -= amount
  295.     tad_contract.transfer(to=ctx.caller, amount=amount)
  296.  
  297.     return True
  298.  
  299.  
  300. @export
  301. def mint_rewards(amount: float):
  302.     assert vaults['mint', 'DSR', 'owner'] == ctx.caller, 'Not the owner!'
  303.     assert amount > 0, 'Cannot mint negative amount!'
  304.  
  305.     tad_contract.mint(amount=amount)
  306.     tad_contract.transfer(to=ctx.caller, amount=amount)
  307.  
  308.     total_weight = 0
  309.     total_funds = amount
  310.  
  311.     for vault_type in vaults['list']:
  312.         total_weight += vaults[vault_type, 'weight']
  313.  
  314.     # To make the contract more robust, and to prevent floating point errors
  315.     for vault_type in vaults['list']:
  316.         funds_transferred = fix_decimal(
  317.             vaults[vault_type, 'weight'] / total_weight) * total_funds
  318.         vaults[vault_type, 'total'] += funds_transferred
  319.  
  320.         total_funds -= funds_transferred
  321.         total_weight -= vaults[vault_type, 'weight']
  322.  
  323.     return True
  324.  
  325.  
  326. @export
  327. def sync_burn(vault_type: int, amount: float):
  328.     assert vault_type in vaults['list'], 'Not an available contract!'
  329.  
  330.     tad_contract.transfer_from(
  331.         to=ctx.this, amount=amount, main_account=ctx.caller)
  332.     tad_contract.burn(amount=amount)
  333.  
  334.     vaults[vault_type, 'total'] -= amount
  335.  
  336.     return vaults[vault_type, 'total']
  337.  
  338.  
  339. @export
  340. def add_vault(collateral_type: str, collateral_amount: float, auction_time: float,
  341.               max_minted: float, s_rate: float, weight: float):
  342.     assert vaults['OWNER'] == ctx.caller, 'Not the owner!'
  343.  
  344.     vault_number = vaults['current_number']
  345.     vaults['list'].append(vault_number)
  346.     vaults['current_number'] += 1
  347.  
  348.     vaults[vault_number, 'collateral_type'] = collateral_type
  349.     vaults[vault_number, 'minimum_collateralization'] = collateral_amount
  350.     vaults[vault_number, 'minimum_auction_time'] = auction_time
  351.     vaults[vault_number, 'cap'] = max_minted
  352.     vaults[vault_number, 'weight'] = weight
  353.  
  354.     stability_rate[vault_number] = s_rate
  355.  
  356.     return vault_number
  357.  
  358.  
  359. @export
  360. def remove_vault(vault_type: int):
  361.     assert vaults['OWNER'] == ctx.caller, 'Not the owner!'
  362.     vaults['list'].remove(vault_type)
  363.  
  364.  
  365. @export
  366. def change_state(key: str, new_value: str, convert_to_decimal: bool = False):
  367.     assert vaults['OWNER'] == ctx.caller, 'Not the owner!'
  368.     assert type(key) == str, 'Invalid type for key'
  369.     assert type(new_value) == str, 'Invalid type for new value'
  370.  
  371.     if convert_to_decimal:
  372.         new_value = decimal(new_value)
  373.     vaults[key] = new_value
  374.  
  375.     return new_value
  376.  
  377.  
  378. @export
  379. def change_any_state(key: Any, new_value: Any, convert_to_tuple: bool = False):
  380.     assert vaults['OWNER'] == ctx.caller, 'Not the owner!'
  381.  
  382.     if convert_to_tuple:
  383.         key = tuple(key)
  384.  
  385.     vaults[key] = new_value
  386.  
  387.     return new_value
  388.  
  389.  
  390. @export
  391. def change_stability_rate(key: int, new_value: float):
  392.     assert vaults['OWNER'] == ctx.caller, 'Not the owner!'
  393.  
  394.     stability_rate[key] = new_value
  395.  
  396.     return new_value
  397.  
  398.  
  399. @export
  400. def get_collateralization_percent(cdp_number: int):
  401.     assert cdp[cdp_number, 'owner'] != 0, 'Nonexistent cdp'
  402.     oracle = importlib.import_module(vaults['oracle'])
  403.  
  404.     return cdp[cdp_number, 'collateral_amount'] * oracle.get_price(cdp[cdp_number, 'vault_type']) / cdp[cdp_number, 'tad']
  405.  
  406.  
  407. def assert_insufficent_collateral(cdp_number: int):
  408.     assert cdp[cdp_number, 'owner'] != 0, 'Nonexistent cdp'
  409.  
  410.     oracle = importlib.import_module(vaults['oracle'])
  411.  
  412.     assert (cdp[cdp_number, 'collateral_amount'] * oracle.get_price(cdp[cdp_number, 'vault_type']) / cdp[cdp_number, 'tad']) < \
  413.         vaults[cdp[cdp_number, 'collateral_type'], 'minimum_collateralization'], 'Vault above minimum collateralization!'
  414.  
  415.    
  416. def fix_decimal(old_decimal: float):
  417.     temporary_var.set(old_decimal)
  418.     new_decimal = temporary_var.get()
  419.    
  420.     return new_decimal
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement