Unit ID
jbLk2blEJMaciDeFDV1SA2XxZD+zNE+TiBj/iHpURQY=
Received
08.09.2020 16:17:36
Confirmation delay (full node)
2 minutes 49 seconds
Confirmation delay (light node)
3 minutes 52 seconds
Messages
Definition
Definition: [ "autonomous agent", { "doc_url": "https://ostable.org/stablecoin-t1-arbitrage.json", "getters": "{ $get_curve_aa = () => params.curve_aa; }", "init": "{ $min_reserve_delta = params.min_reserve_delta OTHERWISE 1e5; $curve_aa = $get_curve_aa(); $curve_params = definition[params.curve_aa][1].params; // tokens $reserve_asset = $curve_params.reserve_asset OTHERWISE 'base'; $asset1 = var[$curve_aa]['asset1']; $shares_asset = var['shares_asset']; $get_leverage = () => $curve_params.leverage OTHERWISE 0; $dilution_factor = var[$curve_aa]['dilution_factor']; $get_reserve = ($s1, $s2) => { $r = $dilution_factor * $s1^$curve_params.m * $s2^$curve_params.n; $r }; $get_p2 = ($s1, $s2) => { $p2 = $dilution_factor * $s1^$curve_params.m * $curve_params.n * (is_integer($curve_params.n*2) ? sqrt($s2^(($curve_params.n-1)*2)) : $s2^($curve_params.n-1) ); // derivative $p2 }; $get_p1 = () => { $s1 = var[$curve_aa]['supply1']/10^$curve_params.decimals1; $s2 = var[$curve_aa]['supply2']/10^$curve_params.decimals2; $p1_in_full_units = $dilution_factor * $curve_params.m * $s1^($curve_params.m-1) * $s2^$curve_params.n; // derivative $p1_in_smallest_units = $p1_in_full_units * 10^($curve_params.reserve_asset_decimals - $curve_params.decimals1); $p1_in_smallest_units }; $get_oracles = () => { $oracles = var[$curve_aa]['oracles']; if ($oracles) return $oracles; $initial_oracles = []; if ($curve_params.oracle1 AND $curve_params.feed_name1) $initial_oracles[] = {oracle: $curve_params.oracle1, feed_name: $curve_params.feed_name1, op: $curve_params.op1 OTHERWISE '*'}; if ($curve_params.oracle2 AND $curve_params.feed_name2) $initial_oracles[] = {oracle: $curve_params.oracle2, feed_name: $curve_params.feed_name2, op: $curve_params.op2 OTHERWISE '*'}; if ($curve_params.oracle3 AND $curve_params.feed_name3) $initial_oracles[] = {oracle: $curve_params.oracle3, feed_name: $curve_params.feed_name3, op: $curve_params.op3 OTHERWISE '*'}; $initial_oracles }; $get_initial_interest_rate = () => exists($curve_params.interest_rate) ? $curve_params.interest_rate : 0.1; // 10% $get_interest_rate = () => { $interest_rate_var = var[$curve_aa]['interest_rate']; exists($interest_rate_var) ? $interest_rate_var : $get_initial_interest_rate() }; $get_growth_factor = () => { $interest_rate = $get_interest_rate(); $term = (timestamp - var[$curve_aa]['rate_update_ts']) / (360 * 24 * 3600); // in years $growth_factor = var[$curve_aa]['growth_factor'] * (1 + $interest_rate)^$term; $growth_factor }; $get_oracle_price = () => { $oracles = $get_oracles(); $oracle_price = reduce($oracles, 3, ($price, $oracle_info) => { $df = data_feed[[oracles=$oracle_info.oracle, feed_name=$oracle_info.feed_name]]; ($oracle_info.op == '*') ? $price * $df : $price / $df }, 1); $oracle_price }; $get_target_p2 = () => { $target_p2 = $get_oracle_price()^($get_leverage() - 1) * $get_growth_factor(); $target_p2 }; $get_exchange_data = () => { $target_p2 = $get_target_p2(); $s2 = var[$curve_aa]['supply2']/10^$curve_params.decimals2; $target_s1 = ($target_p2/$curve_params.n * $s2^(1-$curve_params.n))^(1/$curve_params.m); $tokens1_delta = round($target_s1 * 10^$curve_params.decimals1) - var[$curve_aa]['supply1']; $new_s1 = (var[$curve_aa]['supply1'] + $tokens1_delta) / 10^$curve_params.decimals1; $reserve_delta = ceil($get_reserve($new_s1, $s2) * 10^$curve_params.reserve_asset_decimals) - var[$curve_aa]['reserve']; // calc the reward $initial_p2 = var[$curve_aa]['p2']; $distance = abs($initial_p2 - $target_p2) / $target_p2; $p2 = $get_p2($new_s1, $s2); // might be slightly different from target_p2 due to limited precision of s1 $new_distance = abs($p2 - $target_p2) / $target_p2; // zero or very close $reward = floor((1 - $new_distance/$distance) * var[$curve_aa]['fast_capacity']); $reserve_needed = $reserve_delta - $reward + ($reserve_asset == 'base' ? 1000 : 0); { tokens1_delta: $tokens1_delta, reserve_delta: $reserve_delta, reserve_needed: $reserve_needed, reward: $reward, } }; $get_total_assets = () => { balance[$reserve_asset] - trigger.output[[asset=$reserve_asset]] + $get_p1() * (balance[$asset1] - trigger.output[[asset=$asset1]]) }; $get_reserve_share_after = ($delta_reserve, $delta_asset1) => { (balance[$reserve_asset] - $delta_reserve) / (balance[$reserve_asset] - $delta_reserve + $get_p1() * (balance[$asset1] - $delta_asset1)) }; $status = var['status']; $max_reserve_share = params.max_reserve_share OTHERWISE 1; $min_reserve_share = params.min_reserve_share OTHERWISE 0; $triggerer_reward_share = params.triggerer_reward_share OTHERWISE 0; $min_reserve_investment = $reserve_asset == 'base' ? 1e4 : 0; if (trigger.data.to AND !is_valid_address(trigger.data.to)) bounce("bad to address"); $to = trigger.data.to OTHERWISE trigger.address; }", "messages": { "cases": [ { "if": "{ trigger.data.define AND !$shares_asset }", "messages": [ { "app": "asset", "payload": { "is_private": false, "is_transferrable": true, "auto_destroy": false, "fixed_denominations": false, "issued_by_definer_only": true, "cosigned_by_definer": false, "spender_attested": false } }, { "if": "{trigger.data.factory}", "app": "payment", "payload": { "asset": "base", "outputs": [ { "address": "{trigger.data.factory}", "amount": 1000 } ] } }, { "app": "state", "state": "{ var['shares_asset'] = response_unit; response['shares_asset'] = response_unit; }" } ] }, { "if": "{ trigger.data.arb }", "init": "{ $data = $get_exchange_data(); $tokens1 = $data.tokens1_delta; if (abs($tokens1) <= 1) bounce("already on-peg"); if (abs($data.reserve_delta) < $min_reserve_delta) bounce("reserve delta would be too small: " || $data.reserve_delta); $triggerer_reward = floor($triggerer_reward_share * $data.reward); }", "messages": [ { "app": "payment", "payload": { "asset": "{$reserve_asset}", "outputs": [ { "if": "{$tokens1 > 0}", "address": "{$curve_aa}", "amount": "{ $data.reserve_needed }" }, { "if": "{$triggerer_reward > 100}", "address": "{trigger.address}", "amount": "{ $triggerer_reward }" } ] } }, { "if": "{$tokens1 > 0}", "app": "data", "payload": { "tokens1": "{$tokens1}" } }, { "if": "{$tokens1 < 0}", "app": "payment", "payload": { "asset": "{$asset1}", "outputs": [ { "address": "{$curve_aa}", "amount": "{ -$tokens1 }" } ] } }, { "app": "state", "state": "{ var['status'] = 'arbing'; var['expected_asset1_amount'] = $tokens1 > 0 ? $tokens1 : 0; var['expected_reserve_amount'] = $tokens1 < 0 ? -$data.reserve_needed : 0; }" } ] }, { "if": "{ trigger.address == $curve_aa AND (trigger.output[[asset=$asset1]] > 0 OR trigger.output[[asset=$reserve_asset]] > 0) AND $status AND $status == 'arbing' }", "init": "{ $received_asset1_amount = trigger.output[[asset=$asset1]]; $expected_asset1_amount = var['expected_asset1_amount']; if ($expected_asset1_amount != $expected_asset1_amount) bounce("wrong asset1 amount received from curve AA: expected " || $expected_asset1_amount || ", got " || $received_asset1_amount); $received_reserve_amount = trigger.output[[asset=$reserve_asset]]; $expected_reserve_amount = var['expected_reserve_amount']; if ($expected_reserve_amount != $expected_reserve_amount) bounce("wrong reserve amount received from curve AA: expected " || $expected_reserve_amount || ", got " || $received_reserve_amount); }", "messages": [ { "app": "state", "state": "{ var['expected_asset1_amount'] = false; var['expected_reserve_amount'] = false; var['status'] = false; }" } ] }, { "if": "{ $shares_asset AND (trigger.output[[asset=$reserve_asset]] > 0 OR trigger.output[[asset=$asset1]] > 0) AND trigger.output[[asset=$shares_asset]] == 0 }", "init": "{ $received_reserve_amount = trigger.output[[asset=$reserve_asset]] > $min_reserve_investment ? trigger.output[[asset=$reserve_asset]] : 0; $received_asset1_amount = trigger.output[[asset=$asset1]]; $shares_supply = var['shares_supply'] OTHERWISE 0; $balance = $get_total_assets(); // before the purchase $received_asset1_value = $get_p1() * $received_asset1_amount; $received_assets = $received_reserve_amount + $received_asset1_value; if ($shares_supply > 0){ if ($balance == 0) bounce("shares_supply > 0 AND balance == 0"); $reserve_share = $get_reserve_share_after(0, 0); $t1_share = 1 - $reserve_share; if ($received_reserve_amount > 0 AND $received_asset1_amount > 0) $type = ($t1_share AND $received_reserve_amount == round($received_asset1_value * $reserve_share/$t1_share)) ? 'proportional' : 'nonproportional'; else if ($received_reserve_amount > 0) $type = 'reserve'; else if ($received_asset1_amount > 0) $type = 't1'; else bounce("0 contribution"); if ($reserve_share > $max_reserve_share AND $type != 'proportional' AND $type != 't1') bounce("the reserve share is too large and only proportional or T1 contributions are allowed"); if ($reserve_share < $min_reserve_share AND $type != 'proportional' AND $type != 'reserve') bounce("the T1 share is too large and only proportional or reserve contributions are allowed"); } $share_price = $shares_supply ? $balance / $shares_supply : 1; $shares_amount = floor($received_assets / $share_price); }", "messages": [ { "app": "payment", "payload": { "asset": "{$shares_asset}", "outputs": [ { "address": "{$to}", "amount": "{$shares_amount}" } ] } }, { "app": "state", "state": "{ var['shares_supply'] += $shares_amount; }" } ] }, { "if": "{ $shares_asset AND trigger.output[[asset=$shares_asset]] > 0 }", "init": "{ $what = trigger.data.what OTHERWISE 'both'; if ($what != 't1' AND $what != 'reserve' AND $what != 'both') bounce('bad type of asset to withdraw: ' || $what); $received_shares_amount = trigger.output[[asset=$shares_asset]]; $shares_supply = var['shares_supply']; $balance = $get_total_assets(); if ($balance < 0) bounce("balance < 0"); if ($shares_supply > 0 AND $balance == 0) bounce("shares_supply > 0 AND balance == 0"); if ($what == 'both'){ $reserve_amount = floor($received_shares_amount/$shares_supply * balance[$reserve_asset]); $asset1_amount = floor($received_shares_amount/$shares_supply * balance[$asset1]); } else{ // single-token redemptions $share_price = $balance / $shares_supply; if ($what == 'reserve'){ $reserve_amount = floor($received_shares_amount * $share_price); $asset1_amount = 0; } else{ $reserve_amount = 0; $asset1_amount = floor($received_shares_amount * $share_price / $get_p1()); } $reserve_share = $get_reserve_share_after($reserve_amount, $asset1_amount); $t1_share = 1 - $reserve_share; if ($reserve_share < $min_reserve_share AND $what == 'reserve') bounce("the reserve share is too small and only proportional or T1 withdrawals are allowed"); if ($reserve_share > $max_reserve_share AND $what == 't1') bounce("the T1 share is too small and only proportional or reserve withdrawals are allowed"); } }", "messages": [ { "app": "payment", "payload": { "asset": "{$reserve_asset}", "outputs": [ { "address": "{$to}", "amount": "{$reserve_amount}" } ] } }, { "app": "payment", "payload": { "asset": "{$asset1}", "outputs": [ { "address": "{$to}", "amount": "{$asset1_amount}" } ] } }, { "app": "state", "state": "{ var['shares_supply'] -= $received_shares_amount; }" } ] } ] } } ]
Technical information
Fees:
11,920 bytes
(452 headers, 11468 payload)
Level:1605059
Witnessed level:1605052
Main chain index:1597739
Latest included mc index:1597738
Status:stable/confirmed/final