[
    "autonomous agent",
    {
        "doc_url": "https://obyte.org/prediction.json",
        "bounce_fees": {
            "base": 10000
        },
        "getters": "{
        $get_reserve = ($yes_amount, $no_amount, $draw_amount) => {
        $coef = var['coef'] OTHERWISE 1;
        $r = ceil($coef * sqrt($yes_amount * $yes_amount + $no_amount * $no_amount + $draw_amount * $draw_amount));
        $r
        };
    }",
        "init": "{
        $lib_aa = "
ODKLBHNUOXRQGVG4WIV5GOCUONOAPTB7" || '';
        // params
        $event = params.event OTHERWISE '';
        require(length($event) >= 3 AND length($event) <= 128, "additional description cannot be over 128 chars");
        $oracle = params.oracle; // for testnet
        require($oracle AND is_valid_address($oracle), "oracle isn't valid");
        $comparison = params.comparison OTHERWISE "==";
        require($comparison == "==" OR $comparison == ">" OR $comparison == ">=" OR $comparison == "<" OR $comparison == "<=" OR $comparison == "!=", "operation is unknown");
        $feed_name = params.feed_name OTHERWISE '';
        require(length($feed_name) > 0 AND $feed_name != "none", "feed_name isn't valid");
        $reserve_asset = params.reserve_asset OTHERWISE 'base';
        require(asset[$reserve_asset].exists, "no such asset: " || $reserve_asset);
        $allow_draw = params.allow_draw OTHERWISE false;
        $datafeed_draw_value = params.datafeed_draw_value OTHERWISE 'none';
        $datafeed_value = params.datafeed_value;
        require(exists($datafeed_value), "datafeed_value does not exist");
        $end_of_trading_period = params.end_of_trading_period OTHERWISE 0;
        require(is_integer($end_of_trading_period) AND $end_of_trading_period > 0, "end_of_trading_period must be integer");
        $waiting_period_length = params.waiting_period_length OTHERWISE 5 * 24 * 3600;
        require(is_integer($waiting_period_length) AND $waiting_period_length >= 0, "trading period must be longer than a day");
        $issue_fee = exists(params.issue_fee) ? params.issue_fee : 0.01;
        require($issue_fee >= 0 AND $issue_fee < 1, "issue_fee isn't valid");
        $redeem_fee = exists(params.redeem_fee) ? params.redeem_fee : 0.02;
        require($redeem_fee >= 0 AND $redeem_fee < 1, "redeem_fee isn't valid");
    
        // helpers
        $ready = var['yes_asset'] AND var['no_asset'] AND (var['draw_asset'] OR !$allow_draw);
        $result = var['result']; // yes/no/draw
        $yes_asset = var['yes_asset'];
        $no_asset = var['no_asset'];
        $draw_asset = var['draw_asset'];
        $end_of_waiting_period = $end_of_trading_period + $waiting_period_length;
        $network_fee = ($reserve_asset == 'base') ? 10000 : 0;
        
        $reserve = var['reserve'] OTHERWISE 0;
        $reserve_amount = trigger.output[[asset=$reserve_asset]];
        $type = trigger.data.type OTHERWISE false;
        
        $supply_yes = var['supply_yes'] OTHERWISE 0;
        $supply_no = var['supply_no'] OTHERWISE 0;
        $supply_draw = var['supply_draw'] OTHERWISE 0;
        $to = trigger.data.to OTHERWISE trigger.address;
        if (trigger.data.to AND !is_valid_address(trigger.data.to))
            bounce("bad to address");
        if ($type) {
            require($type == 'yes' OR $type == 'no' OR $type == 'draw', "unknown type");
        }
    }",
        "messages": {
            "cases": [
                {
                    "if": "{!$ready AND trigger.data.define}",
                    "init": "{
                    $define_type = $yes_asset ? ($no_asset ? "draw_asset" : "no_asset") : "yes_asset";
                    $define_asset_forwarder = 'E4BAASPOCW6WHXSUOY2XEKXKA42RRD5I';
                    require(($define_type == 'draw_asset' AND $allow_draw) OR $define_type == 'no_asset' OR $define_type == 'yes_asset', "unknown define type");
                    require(!var[$define_type], "asset already defined");
                }",
                    "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 AND !$yes_asset}",
                            "app": "data",
                            "payload": {
                                "factory": "{trigger.data.factory}"
                            }
                        },
                        {
                            "if": "{trigger.data.factory AND !$yes_asset}",
                            "app": "payment",
                            "payload": {
                                "asset": "base",
                                "outputs": [
                                    {
                                        "address": "{$define_asset_forwarder}",
                                        "amount": 4000
                                    }
                                ]
                            }
                        },
                        {
                            "if": "{trigger.data.factory AND $yes_asset AND !$no_asset}",
                            "app": "data",
                            "payload": {
                                "yes_asset": "{$yes_asset}",
                                "allow_draw": "{params.allow_draw}"
                            }
                        },
                        {
                            "if": "{trigger.data.factory AND $yes_asset AND !$no_asset}",
                            "app": "payment",
                            "payload": {
                                "asset": "base",
                                "outputs": [
                                    {
                                        "address": "{trigger.data.factory}",
                                        "amount": 4000
                                    }
                                ]
                            }
                        },
                        {
                            "if": "{trigger.data.factory AND $yes_asset AND $no_asset AND !$draw_asset AND $allow_draw}",
                            "app": "data",
                            "payload": {
                                "no_asset": "{$no_asset}"
                            }
                        },
                        {
                            "if": "{trigger.data.factory AND $yes_asset AND $no_asset AND !$draw_asset AND $allow_draw}",
                            "app": "payment",
                            "payload": {
                                "asset": "base",
                                "outputs": [
                                    {
                                        "address": "{trigger.data.factory}",
                                        "amount": 4000
                                    }
                                ]
                            }
                        },
                        {
                            "app": "state",
                            "state": "{
                            var[$define_type] = response_unit;
                            response[$define_type] = response_unit;
                        }"
                        }
                    ]
                },
                {
                    "if": "{$ready AND !exists(trigger.data.claim_profit) AND !exists(trigger.data.add_liquidity) AND (((trigger.output[[asset=$reserve_asset]] > $network_fee) AND (trigger.data.yes_amount OR trigger.data.no_amount OR trigger.data.draw_amount OR trigger.data.type)) OR trigger.output[[asset=$yes_asset]] > 0 OR trigger.output[[asset=$no_asset]] > 0 OR ($allow_draw AND trigger.output[[asset=$draw_asset]] > 0))}",
                    "init": "{
                    require($allow_draw OR !exists(trigger.data.draw_amount), "draw asset does not exist");
                    if ((trigger.data.yes_amount OR trigger.data.no_amount OR trigger.data.draw_amount OR trigger.output[[asset=$yes_asset]] > 0 OR trigger.output[[asset=$no_asset]] > 0 OR ($allow_draw AND trigger.output[[asset=$draw_asset]] > 0)) AND trigger.data.type) {
                        bounce("both type and amount");
                    }
                    
                    $min_expected_amount = trigger.data.min_expected_amount OTHERWISE 0;
                    if ($type AND (!is_integer($min_expected_amount) OR $min_expected_amount < 0)) {
                        bounce("invalid min_expected_amount");
                    }
                    if (!(timestamp <= $end_of_trading_period OR timestamp >= $end_of_waiting_period)){
                        $error = "the trading period is closed";
                        response['error'] = $error;
                        return;
                    }
                    if ($result) {
                        $error = "result already exists";
                        response['error'] = $error;
                        return;
                    }
                
                    if ($type) {
                        $amount_by_type = $lib_aa#11.$get_token_amount(this_address, $type, $reserve_amount);
                        if ($type == 'yes') {
                            $yes_amount = $amount_by_type;
                            $no_amount = 0;
                            $draw_amount = 0;
                        } else if ($type == 'no') { 
                            $yes_amount = 0;
                            $no_amount = $amount_by_type;
                            $draw_amount = 0;
                        } else if ($type == 'no') { 
                            $yes_amount = 0;
                            $no_amount = 0;
                            $draw_amount = $amount_by_type;
                        }
                    } else {
                        $yes_amount = trigger.data.yes_amount OTHERWISE -trigger.output[[asset=$yes_asset]];
                        $no_amount = trigger.data.no_amount OTHERWISE -trigger.output[[asset=$no_asset]];
                        $draw_amount = $allow_draw ? trigger.data.draw_amount OTHERWISE -trigger.output[[asset=$draw_asset]] : 0;
                    }
                    if (trigger.data.yes_amount AND (!is_integer(trigger.data.yes_amount) OR trigger.data.yes_amount <= 0))
                        bounce("invalid number of yes_amount");
                    if (trigger.data.no_amount AND (!is_integer(trigger.data.no_amount) OR trigger.data.no_amount <= 0))
                        bounce("invalid number of no_amount");
                    if ($allow_draw AND trigger.data.draw_amount AND (!is_integer(trigger.data.draw_amount) OR trigger.data.draw_amount <= 0))
                        bounce("invalid number of draw_amount");
                    if (trigger.data.yes_amount AND trigger.output[[asset=$yes_asset]] > 0)
                        bounce("both yes_amount param and amount");
                    if (trigger.data.no_amount AND trigger.output[[asset=$no_asset]] > 0)
                        bounce("both no_amount param and amount");
                    if ($allow_draw AND trigger.data.draw_amount AND trigger.output[[asset=$draw_asset]] > 0)
                        bounce("both draw_amount param and amount");    
                    $res = $lib_aa#12.$get_exchange_result(this_address, $yes_amount, $no_amount, $draw_amount);
                    $payout = floor($res.payout + $reserve_amount - $res.reserve_needed - $res.fee - $network_fee);
                    if ($type) {
                        $amount = $type == 'yes' ? $yes_amount : ($type == 'no' ? $no_amount : $draw_amount);
                        if ($amount < $min_expected_amount) {
                            $error = "amount less than minimum";
                            response['error'] = $error;
                            return;
                        }
                    }
                    if ($res.reserve_needed > 0 AND ($res.reserve_needed + $res.fee + $network_fee > $reserve_amount)) {
                        $error = "expected reserve amount: " || ($res.reserve_needed + $res.fee + $network_fee);
                        response['error'] = $error;
                        return;
                    }
                }",
                    "messages": [
                        {
                            "if": "{$yes_amount AND $yes_amount > 0 AND !$error}",
                            "app": "payment",
                            "payload": {
                                "asset": "{$yes_asset}",
                                "outputs": [
                                    {
                                        "address": "{$to}",
                                        "amount": "{$yes_amount}"
                                    }
                                ]
                            }
                        },
                        {
                            "if": "{$no_amount AND $no_amount > 0 AND !$error}",
                            "app": "payment",
                            "payload": {
                                "asset": "{$no_asset}",
                                "outputs": [
                                    {
                                        "address": "{$to}",
                                        "amount": "{$no_amount}"
                                    }
                                ]
                            }
                        },
                        {
                            "if": "{$allow_draw AND $draw_amount AND $draw_amount > 0 AND !$error}",
                            "app": "payment",
                            "payload": {
                                "asset": "{$draw_asset}",
                                "outputs": [
                                    {
                                        "address": "{$to}",
                                        "amount": "{$draw_amount}"
                                    }
                                ]
                            }
                        },
                        {
                            "if": "{($payout AND $payout > 0) OR $error}",
                            "app": "payment",
                            "payload": {
                                "asset": "{$reserve_asset}",
                                "outputs": [
                                    {
                                        "address": "{$to}",
                                        "amount": "{$error ? $reserve_amount - $network_fee : $payout}"
                                    }
                                ]
                            }
                        },
                        {
                            "if": "{!$error}",
                            "app": "state",
                            "state": "{
                            response['next_reserve'] = $res.new_reserve  OTHERWISE 0;
                            response['next_coef'] = $res.next_coef OTHERWISE 1;
                            response['yes_amount'] = $yes_amount OTHERWISE 0;
                            response['no_amount'] = $no_amount OTHERWISE 0;
                            response['draw_amount'] = $draw_amount OTHERWISE 0;
                            response['yes_price'] = $res.yes_price;
                            response['no_price'] = $res.no_price;
                            response['draw_price'] = $res.draw_price;
                            var['reserve'] = $res.new_reserve;
                            var['coef'] = $res.next_coef;
                            if ($res.fee > 0){
                                response['fee'] = $res.fee;
                            }
                            response['arb_profit_tax'] = $res.arb_profit_tax;
                            var['supply_yes'] += $yes_amount;
                            response['supply_yes'] = var['supply_yes'] OTHERWISE 0;
                            var['supply_no'] += $no_amount;
                            response['supply_no'] = var['supply_no'] OTHERWISE 0;
                            if ($allow_draw) {
                                var['supply_draw'] += $draw_amount;
                                response['supply_draw'] = var['supply_draw'] OTHERWISE 0;
                            }
                        }"
                        }
                    ]
                },
                {
                    "if": "{$ready AND trigger.data.add_liquidity AND $reserve_amount > $network_fee}",
                    "init": "{
                    $fee = $reserve_amount * $issue_fee;
                    $reserve_amount_without_fee = $reserve_amount - $fee - $network_fee;
                    
                    if ((timestamp <= $end_of_trading_period OR timestamp >= $end_of_waiting_period) AND !$result) {
                        if ($supply_yes + $supply_no + $supply_draw == 0) {
                            $yes_amount_ratio = trigger.data.yes_amount_ratio OTHERWISE 0;
                            require($yes_amount_ratio >= 0, 'yes ratio isnt valid');
                            if ($allow_draw) {
                                $no_amount_ratio = trigger.data.no_amount_ratio OTHERWISE 0;
                                require($no_amount_ratio >= 0, 'no ratio isnt valid');
                                $draw_amount_ratio = 1 - $yes_amount_ratio - $no_amount_ratio;
                                require($draw_amount_ratio >= 0, 'ratio isnt valid');
                            } else {
                                $no_amount_ratio = 1 - $yes_amount_ratio;
                                require($no_amount_ratio >= 0, 'yes ratio isnt valid');
                            }
                            require($yes_amount_ratio >= 0 AND $no_amount_ratio >= 0 AND $yes_amount_ratio + $no_amount_ratio + $draw_amount_ratio == 1, 'yes_amount_ratio + no_amount_ratio + draw_amount_ratio must be equal 1');
                            $yes_amount = floor($reserve_amount_without_fee * sqrt($yes_amount_ratio));
                            $no_amount = floor($reserve_amount_without_fee * sqrt($no_amount_ratio));
                            $draw_amount = $allow_draw ? floor($reserve_amount_without_fee * sqrt($draw_amount_ratio)) : 0;
                        } else {
                            $ratio = ($reserve_amount_without_fee + $reserve) / $reserve;
                            $yes_amount = floor($ratio * $supply_yes - $supply_yes);
                            $no_amount = floor($ratio * $supply_no - $supply_no);
                            $draw_amount = floor($ratio * $supply_draw - $supply_draw);
                        }
                    } else {
                        $hasError = true;
                    }
                }",
                    "messages": [
                        {
                            "if": "{!$hasError AND $yes_amount > 0}",
                            "app": "payment",
                            "payload": {
                                "asset": "{$yes_asset}",
                                "outputs": [
                                    {
                                        "address": "{$to}",
                                        "amount": "{$yes_amount}"
                                    }
                                ]
                            }
                        },
                        {
                            "if": "{!$hasError AND $no_amount > 0}",
                            "app": "payment",
                            "payload": {
                                "asset": "{$no_asset}",
                                "outputs": [
                                    {
                                        "address": "{$to}",
                                        "amount": "{$no_amount}"
                                    }
                                ]
                            }
                        },
                        {
                            "if": "{!$hasError AND $allow_draw AND $draw_amount > 0}",
                            "app": "payment",
                            "payload": {
                                "asset": "{$draw_asset}",
                                "outputs": [
                                    {
                                        "address": "{$to}",
                                        "amount": "{$draw_amount}"
                                    }
                                ]
                            }
                        },
                        {
                            "if": "{$hasError}",
                            "app": "payment",
                            "payload": {
                                "asset": "{$reserve_asset}",
                                "outputs": [
                                    {
                                        "address": "{$to}",
                                        "amount": "{$reserve_amount - $network_fee}"
                                    }
                                ]
                            }
                        },
                        {
                            "if": "{!$hasError}",
                            "app": "state",
                            "state": "{
                            var['reserve'] += $reserve_amount;
                            var['supply_yes'] += $yes_amount;
                            response['supply_yes'] = var['supply_yes'] OTHERWISE 0;
                            var['supply_no'] += $no_amount;
                            response['supply_no'] = var['supply_no'] OTHERWISE 0;
                            if ($allow_draw) {
                                var['supply_draw'] += $draw_amount;
                                response['supply_draw'] = var['supply_draw'] OTHERWISE 0;
                            }
                            $new_supply_yes = var['supply_yes'] OTHERWISE 0;
                            $new_supply_no = var['supply_no'] OTHERWISE 0;
                            $new_supply_draw = var['supply_draw'] OTHERWISE 0;
                            $new_reserve = $get_reserve($new_supply_yes, $new_supply_no, $new_supply_draw);
                            $next_coef = (var['coef'] OTHERWISE 1) * (($new_reserve + $fee) / $new_reserve);
                            
                            var['coef'] = $next_coef;
                            // for stats
                            response['yes_amount'] = $yes_amount OTHERWISE 0;
                            response['no_amount'] = $no_amount OTHERWISE 0;
                            response['draw_amount'] = $draw_amount OTHERWISE 0;
                            response['next_reserve'] = var['reserve'];
                            response['next_coef'] = $next_coef;
                            
                            $new_den = sqrt($new_supply_yes * $new_supply_yes + $new_supply_no * $new_supply_no + $new_supply_draw * $new_supply_draw);
                            
                            $new_yes_price = $next_coef * ($new_supply_yes / $new_den);
                            $new_no_price = $next_coef * ($new_supply_no / $new_den);
                            $new_draw_price = $next_coef * ($new_supply_draw / $new_den);
                            response['yes_price'] = $new_yes_price;
                            response['no_price'] = $new_no_price;
                            response['draw_price'] = $new_draw_price;
                        }"
                        }
                    ]
                },
                {
                    "if": "{$ready AND trigger.data.commit AND !$result}",
                    "init": "{
                    require(timestamp > $end_of_trading_period, "trading period has not yet ended");
                    
                    $current_datafeed_value = data_feed[[oracles=$oracle, feed_name=$feed_name, ifnone='none']];
                    require($current_datafeed_value != 'none', "data_feed is empty");
                    require(timestamp > $end_of_trading_period, "trading period has not ended yet");
                    require(timestamp < $end_of_waiting_period, "it's late, the waiting period has passed");
                    if ($comparison == '>')
                            $datafeed_comparison = $current_datafeed_value > $datafeed_value;
                    else if ($comparison == '<')
                            $datafeed_comparison = $current_datafeed_value < $datafeed_value;
                    else if ($comparison == '!=')
                            $datafeed_comparison = $current_datafeed_value != $datafeed_value;
                    else if ($comparison == '==')
                            $datafeed_comparison = $current_datafeed_value == $datafeed_value;
                    else if ($comparison == '>=')
                            $datafeed_comparison = $current_datafeed_value >= $datafeed_value;
                    else if ($comparison == '<=')
                            $datafeed_comparison = $current_datafeed_value <= $datafeed_value;
                    else
                        bounce('Comparison operator not found');
                    
                    if ($datafeed_comparison) {
                        $res = 'yes';
                    } else if ($allow_draw AND $current_datafeed_value == $datafeed_draw_value) {
                        $res = 'draw';
                    } else {
                        $res = 'no';
                    }
                }",
                    "messages": [
                        {
                            "app": "state",
                            "state": "{
                            var['result'] = $res;
                            response['messages'] = "The result is committed";
                            response['result'] = $res;
                        }"
                        }
                    ]
                },
                {
                    "if": "{$ready AND trigger.data.claim_profit}",
                    "init": "{
                    $yes_amount = trigger.output[[asset=$yes_asset]];
                    $no_amount = trigger.output[[asset=$no_asset]];
                    $draw_amount = $allow_draw ? trigger.output[[asset=$draw_asset]] : 0;
                    require(($reserve_asset == 'base' AND $reserve_amount == $network_fee) OR ($reserve_asset != 'base' AND $reserve_amount == 0), "should not send a reserve");
                    require($result, "no results yet");
                    $winner_amount = $result == 'yes' ? $yes_amount : $result == 'no' ? $no_amount : $draw_amount;
                    $winner_supply = $result == 'yes' ? var['supply_yes'] : $result == 'no' ? var['supply_no'] : var['supply_draw'];
                    require($winner_supply > 0 AND $winner_amount > 0, "winner supply < 0 or sending not a winner token");
                    $price_winner_by_reserve = $reserve / $winner_supply;
                    $payout = floor($winner_amount * $price_winner_by_reserve);
                }",
                    "messages": [
                        {
                            "if": "{ $payout > 0 }",
                            "app": "payment",
                            "payload": {
                                "asset": "{$reserve_asset}",
                                "outputs": [
                                    {
                                        "address": "{$to}",
                                        "amount": "{$payout}"
                                    }
                                ]
                            }
                        },
                        {
                            "app": "state",
                            "state": "{
                            response["profit"] = $payout;
                            if ($yes_amount > 0) {
                                var['supply_yes'] -= $yes_amount;
                            }
                            if ($no_amount > 0) {
                                var['supply_no'] -= $no_amount;
                            }
                            if ($allow_draw AND $draw_amount > 0) {
                                var['supply_draw'] -= $draw_amount;
                            }
                            var['reserve'] -= $payout;
                        }"
                        }
                    ]
                }
            ]
        }
    }
]