Unit ID
1O4mztwuQTeG/D5ka59CQK+Xn2tnJck9L9A44STorI0=
Received
03.10.2019 10:45:21
Confirmation delay (full node)
4 minutes 37 seconds
Confirmation delay (light node)
8 minutes 13 seconds
Messages
Definition
Definition: [ "autonomous agent", { "init": "{ if (trigger.output[[asset!=base]].asset != 'none') bounce('foreign coins'); $fee = 1000; $min_stake = 1e6; $coef = 1.5; $period_length = 3600; // $period_length = 3*24*3600; $feed_name = trigger.data.feed_name; $feed_value = trigger.data.feed_value; }", "messages": { "cases": [ { "if": "{$feed_name AND $feed_value}", "init": "{ $pair = sha256($feed_name || '=' || $feed_value); // long keys not allowed $num = var[$pair || '_number'] otherwise 0; // we need to be prepared that the same name/value pair is submitted again if (!$num AND !trigger.data.new) bounce("new=1 expected now"); $old_key = $pair || '_' || $num; if (trigger.data.new AND (!$num OR var[$old_key] == 'committed') ) $key = $pair || '_' || ($num + 1); else $key = $old_key; }", "messages": { "cases": [ { "if": "{ if (trigger.output[[asset=base]] < $min_stake) return false; $amount = trigger.output[[asset=base]] - $fee; if (var[$key] AND var[$key] == 'committed') bounce('already committed'); $outcome = trigger.data.outcome otherwise 'yes'; if ($outcome != 'yes' AND $outcome != 'no') bounce('neither yes nor no'); $bInitialStake = !var[$key]; if ($bInitialStake AND $outcome == 'no') bounce('initial stake cannot be on No'); if ($bInitialStake){ if (!trigger.data.new) bounce("something went wrong: initial stake without new"); $pool_id = trigger.data.pool_id; if (!$pool_id) bounce('please specify a reward pool id'); $number_of_rewards = var['pool_' || $pool_id || '_number_of_rewards']; if (!$number_of_rewards) bounce("pool " || $pool_id || " doesn't exist or is empty"); $pool_feed_name = var['pool_' || $pool_id || '_feed_name']; if ($pool_feed_name AND $pool_feed_name != $feed_name) bounce("pool " || $pool_id || " is for feed name " || $pool_feed_name || " only"); return true; } // else expect a counterstake if (timestamp - var[$key || '_countdown_start'] > $period_length) bounce('challenging period expired'); $current_outcome = var[$key || '_outcome']; if ($current_outcome == $outcome) bounce('staking on the current outcome is not allowed'); $stake_on_current_outcome = var[$key || '_total_staked_on_' || $current_outcome]; $stake_on_proposed_outcome = var[$key || '_total_staked_on_' || $outcome]; $required_to_challenge = round($coef * $stake_on_current_outcome); $would_override_current_outcome = ($stake_on_proposed_outcome + $amount >= $required_to_challenge); if ($would_override_current_outcome) $excess = $stake_on_proposed_outcome + $amount - $required_to_challenge; true }", "messages": [ { "if": "{$excess}", "app": "payment", "payload": { "asset": "base", "outputs": [ { "address": "{trigger.address}", "amount": "{$excess}" } ] } }, { "app": "state", "state": "{ if ($bInitialStake){ var[$pair || '_number'] += 1; var[$key] = 'onreview'; var[$key || '_initial_reporter'] = trigger.address; var['pool_' || $pool_id || '_number_of_rewards'] -= 1; var[$key || '_pool_id'] = $pool_id; // we'll need it if the feed is not accepted and the reward needs to be returned to the pool response['pool_id'] = $pool_id; response['expected_reward'] = var['pool_' || $pool_id || '_reward_amount']; } if ($bInitialStake OR $would_override_current_outcome){ var[$key || '_countdown_start'] = timestamp; var[$key || '_outcome'] = $outcome; response['countdown_started'] = true; response['outcome'] = $outcome; } else response['outcome'] = $current_outcome; $accepted_amount = $amount - $excess; var[$key || '_total_staked'] += $accepted_amount; var[$key || '_total_staked_on_' || $outcome] += $accepted_amount; var[$key || '_total_staked_on_' || $outcome || '_by_' || trigger.address] += $accepted_amount; response['staked_on_yes'] = var[$key || '_total_staked_on_yes']; response['staked_on_no'] = var[$key || '_total_staked_on_no'] otherwise 0; response['your_stake'] = var[$key || '_total_staked_on_' || $outcome || '_by_' || trigger.address]; }" } ] }, { "if": "{ if (!trigger.data.commit) return false; if (!var[$key]) bounce('unknown feed'); if (var[$key] == 'committed') bounce('already committed'); if (timestamp - var[$key || '_countdown_start'] <= $period_length) bounce('challenge period is still running'); $pool_id = var[$key || '_pool_id']; $outcome = var[$key || '_outcome']; // immediately pay to the initial reporter. Other stakers (if any) will have to manually request withdrawals if ($outcome == 'yes'){ $address = var[$key || '_initial_reporter']; $initial_reporter_stake = var[$key || '_total_staked_on_' || $outcome || '_by_' || $address]; if (!$initial_reporter_stake) bounce('something went wrong: no initial reporter stake'); $reward = var['pool_' || $pool_id || '_reward_amount']; $total_winning_stake = var[$key || '_total_staked_on_' || $outcome]; $total_stake = var[$key || '_total_staked']; $amount = round($initial_reporter_stake / $total_winning_stake * $total_stake); $full_amount = $amount + $reward; } true }", "messages": [ { "if": "{$outcome == 'yes'}", "app": "data_feed", "payload": { "{$feed_name}": "{$feed_value}" } }, { "if": "{$outcome == 'yes'}", "app": "payment", "payload": { "asset": "base", "outputs": [ { "address": "{$address}", "amount": "{$full_amount}" } ] } }, { "app": "state", "state": "{ var[$key] = 'committed'; if ($outcome == 'yes') var[$key || '_paid_out_' || $address] = 1; else // return the reward to the pool var['pool_' || $pool_id || '_number_of_rewards'] += 1; response['message'] = "committed"; }" } ] }, { "if": "{ if (!trigger.data.withdraw) return false; if (!var[$key]) bounce('unknown feed'); if (var[$key] != 'committed') bounce('not committed yet'); $address = trigger.data.address otherwise trigger.address; // withdrawal can be triggered by anybody if (var[$key || '_paid_out_' || $address]) bounce("you were already paid"); $outcome = var[$key || '_outcome']; $my_stake = var[$key || '_total_staked_on_' || $outcome || '_by_' || $address]; if (!$my_stake) bounce("you didn't stake on the winning outcome"); $total_winning_stake = var[$key || '_total_staked_on_' || $outcome]; $total_stake = var[$key || '_total_staked']; $amount = round($my_stake / $total_winning_stake * $total_stake); true }", "messages": [ { "app": "payment", "payload": { "asset": "base", "outputs": [ { "address": "{$address}", "amount": "{$amount}" } ] } }, { "app": "state", "state": "{ var[$key || '_paid_out_' || $address] = 1; response['message'] = "paid " || $amount || " bytes"; }" } ] } ] } }, { "if": "{trigger.data.reward_amount AND trigger.data.number_of_rewards}", "init": "{ $expected_amount = trigger.data.reward_amount * trigger.data.number_of_rewards; if (trigger.output[[asset=base]] != $expected_amount) bounce('wrong amount received, expected: ' || $expected_amount); }", "messages": [ { "app": "state", "state": "{ var['pool_id'] += 1; $pool_id = var['pool_id']; var['pool_' || $pool_id || '_sponsor'] = trigger.address; var['pool_' || $pool_id || '_reward_amount'] = trigger.data.reward_amount; var['pool_' || $pool_id || '_number_of_rewards'] = trigger.data.number_of_rewards; if ($feed_name) // the pool can be for one feed name only or general var['pool_' || $pool_id || '_feed_name'] = $feed_name; response['message'] = "created a reward pool " || $pool_id || " of " || trigger.output[[asset=base]] || " bytes"; }" } ] }, { "if": "{trigger.data.withdraw_pool AND trigger.data.pool_id}", "init": "{ $sponsor = var['pool_' || $pool_id || '_sponsor']; if (!$sponsor) bounce('no such pool: ' || $pool_id); if ($sponsor != trigger.address) bounce('not your pool: ' || $pool_id); $number_of_rewards = var['pool_' || $pool_id || '_number_of_rewards']; // can be less than the initial number as some rewards might be already consumed or locked if (!$number_of_rewards) bounce('pool ' || $pool_id || ' is already empty'); $reward_amount = var['pool_' || $pool_id || '_reward_amount']; }", "messages": [ { "app": "payment", "payload": { "asset": "base", "outputs": [ { "address": "{trigger.address}", "amount": "{$number_of_rewards * $reward_amount}" } ] } }, { "app": "state", "state": "{ var['pool_' || $pool_id || '_number_of_rewards'] = false; response['message'] = "destroyed reward pool " || $pool_id; }" } ] } ] } } ]
Technical information
Fees:
9,114 bytes
(353 headers, 8761 payload)
Level:1089612
Witnessed level:1089605
Main chain index:1088373
Latest included mc index:1088372
Status:stable/confirmed/final