Definition: [
    "autonomous agent",
    {
        "bounce_fees": {
            "base": 10000
        },
        "init": "{
        $INVESTMENT_CUT = 0.01;    //We take a 1% cut from all maecenas investments
        $RESERVE_AMOUNT = 20000;   //We will always keep > 200.000 in the AA
        if (NOT trigger.data["method"])
            bounce("method field is mandatory");
            
        $method = trigger.data["method"];
        $spendableFunds = balance["base"] - var["locked"] - storage_size - trigger.output[[asset="base"]].amount;
        $owner = var["owner"];
        $calculateSharesToBuy = ($price, $pool, $myShare)=>ln($price / ($pool * $myShare)) / ln($myShare);
        $_maecenasCurve = $x=>6969420 * $x^2;
        $INVESTING_ENABLED = var["INVESTING_ENABLED"];
    }",
        "getters": "{
        /*
        ** Attested profile (or just the address if not found)
        */
        $profileOf = $address=>{
            $ret = attestation[[attestors=this_address, address=$address, ifseveral='last', type="string"]].data;
            return $ret ? json_parse($ret) : {};
        };
        $licenseExpirationOf = ($user, $license)=>var[$user || "_" || $license];
        $artistInfoOf = $artist=>var["artist_" || $artist];
    }",
        "messages": {
            "cases": [
                {
                    "if": "{
                  NOT $owner
              }",
                    "messages": [
                        {
                            "app": "state",
                            "state": "{
                          var["owner"] = "IUU43O7TS2TBYKAPGKUARDZHOTAE275A";    //Set the Owner to my address
              var["helper"] = "IUU43O7TS2TBYKAPGKUARDZHOTAE275A";
                          var["locked"] = 0; //Bytes locked in investments are not withdrawable by the Owner
                      }"
                        }
                    ]
                },
                {
                    "if": "{
                  trigger.address == $owner
                  AND ($method == "payout"
                  OR $method == "transferOwnership"
                  OR $method == "setHelper"
                    OR $method == "setAA"
                    OR $method == "delAA"
                    OR $method == "enableInvesting"
                    OR $method == "setReseller"
                    OR $method == "delReseller"
                  OR $method == "unverify")
              }",
                    "messages": {
                        "cases": [
                            {
                                "if": "{
                              $method == "payout"
                          }",
                                "messages": [
                                    {
                                        "app": "payment",
                                        "payload": {
                                            "asset": "base",
                                            "outputs": [
                                                {
                                                    "address": "{trigger.address}",
                                                    "amount": "{$spendableFunds}"
                                                }
                                            ]
                                        }
                                    }
                                ]
                            },
                            {
                                "if": "{
                                $method == "unverify"
                            }",
                                "init": "{
                                if (NOT trigger.data["address"])
                                    bounce("address field is mandatory");
                                $_ = $profileOf(trigger.data["address"]);
                                $oldUsername = $_["username"];
                                $prof = $_ || {verified: false, username: false, isReseller: false};
                            }",
                                "messages": [
                                    {
                                        "app": "attestation",
                                        "init": "{
                                        $profileString = json_stringify($prof);
                                        if (length($profileString) > 4096)
                                            bounce("The profile is too long, the user has to remove some fields and retry verification");
                                    }",
                                        "payload": "{
                                        {
                                            address: trigger.data["address"],
                                            profile: {
                                                data: $profileString
                                            }
                                        }
                                    }"
                                    },
                                    {
                                        "app": "state",
                                        "state": "{
                                      var["user_" || $oldUsername] = false; //Free username
                                  }"
                                    }
                                ]
                            },
                            {
                                "if": "{
                                $method == "transferOwnership"
                            }",
                                "init": "{
                                if (NOT trigger.data["newOwner"])
                                    bounce("newOwner field is mandatory");
                                if (NOT is_valid_address(trigger.data["newOwner"]))
                                    bounce("newOwner field is not a valid address");
                            }",
                                "messages": [
                                    {
                                        "if": "{
                                        $spendableFunds > 0
                                    }",
                                        "app": "payment",
                                        "payload": {
                                            "asset": "base",
                                            "outputs": [
                                                {
                                                    "address": "{trigger.address}",
                                                    "amount": "{$spendableFunds}"
                                                }
                                            ]
                                        }
                                    },
                                    {
                                        "app": "state",
                                        "state": "{
                                      var["owner"] = trigger.data["newOwner"];//Ownership is transferred
                                  }"
                                    }
                                ]
                            },
                            {
                                "if": "{
                                $method == "setHelper"
                            }",
                                "init": "{
                                if (NOT trigger.data["helper"])
                                    bounce("helper field is mandatory");
                            }",
                                "messages": [
                                    {
                                        "app": "state",
                                        "state": "{
                                      var["helper"] = trigger.data["helper"];
                                  }"
                                    }
                                ]
                            },
                            {
                                "if": "{
                                trigger.data["method"] == "setAA"
                            }",
                                "init": "{
                                if (NOT is_valid_address(trigger.data["AA"]))
                                    bounce("AA is not a valid address");
                            }",
                                "messages": [
                                    {
                                        "app": "state",
                                        "state": "{
                                        var["AA_" || trigger.data["AA"]] = true;
                                    }"
                                    }
                                ]
                            },
                            {
                                "if": "{
                                trigger.data["method"] == "delAA"
                            }",
                                "init": "{
                                if (NOT var["AA_" || trigger.data["AA"]])
                                    bounce("AA is not a valid address");
                            }",
                                "messages": [
                                    {
                                        "app": "state",
                                        "state": "{
                                        var["AA_" || trigger.data["AA"]] = false;
                                    }"
                                    }
                                ]
                            },
                            {
                                "if": "{
                                $method == "enableInvesting"
                            }",
                                "messages": [
                                    {
                                        "app": "state",
                                        "state": "{
                                        var["INVESTING_ENABLED"] = true;
                                    }"
                                    }
                                ]
                            },
                            {
                                "if": "{
                                $method == "setReseller"
                            }",
                                "init": "{
                                if (NOT is_valid_address(trigger.data["address"]))
                                    bounce("address is not valid");
                                $prof = $profileOf(trigger.data["address"]) || {isReseller: true};
                                $profileString = json_stringify($prof);
                                if (length($profileString) > 4096)
                                    bounce("The profile is too long, the user has to remove some fields and retry verification");
                            }",
                                "messages": [
                                    {
                                        "app": "attestation",
                                        "payload": "{
                                        {
                                            address: trigger.data["address"],
                                            profile: { //Wrapper to be able to query the whole object at once since you can only query attestations by fields
                                                data: $profileString
                                            }
                                        }
                                    }"
                                    }
                                ]
                            },
                            {
                                "if": "{
                                $method == "delReseller"
                            }",
                                "init": "{
                                $prof = $profileOf(trigger.data["address"]) || {isReseller: false};
                                $profileString = json_stringify($prof);
                                if (length($profileString) > 4096)
                                    bounce("The profile is too long, the user has to remove some fields and retry verification");
                            }",
                                "messages": [
                                    {
                                        "app": "attestation",
                                        "payload": "{
                                        {
                                            address: trigger.data["address"],
                                            profile: { //Wrapper to be able to query the whole object at once since you can only query attestations by fields
                                                data: $profileString
                                            }
                                        }
                                    }"
                                    }
                                ]
                            }
                        ]
                    }
                },
                {
                    "if": "{
                    (trigger.address == $owner
                    OR trigger.address == var["helper"])
                    AND $method == "verifyProfile"
                }",
                    "init": "{
                    if (NOT trigger.data["address"])
                        bounce("address field is mandatory");
                    if (NOT is_valid_address(trigger.data["address"]))
                        bounce("That address is not valid");
          $profileObj = $profileOf(trigger.data["address"]) || {verified: true};
                    response["debug"] = json_stringify($profileObj);
                    if (NOT $profileObj.username)
                        bounce("That user has not chosen an username");
                }",
                    "messages": [
                        {
                            "app": "attestation",
                            "payload": "{
              {
                address: trigger.data["address"],
                profile: { //Wrapper to be able to query the whole object at once since you can only query attestations by fields
                                    data: json_stringify($profileObj)
                                }
              }
            }"
                        },
                        {
                            "app": "state",
                            "state": "{
                          var["user_" || $profileObj.username] = trigger.data["address"]; //Username is now taken
                      }"
                        }
                    ]
                },
                {
                    "if": "{
                  $method == "DIVEST"
                }",
                    "init": "{
                    $artist = $profileOf(trigger.data["artist"]) || (var["artist_" || trigger.data["artist"]] OTHERWISE {});
                    if (NOT trigger.data["artist"])
                        bounce("artist field is mandatory");
                    if (NOT is_valid_address(trigger.data["artist"]))
                        bounce("artist's address is invalid");
                    if (NOT $artist)
                        bounce("That artist never accepted maecenas");
                    if (NOT exists(trigger.data["amount"]))
                        bounce("amount field is mandatory");
                    if (NOT is_integer(trigger.data["amount"]) OR trigger.data["amount"] <= 0)
                        bounce("amount must be an integer > 0");
                    $investedAmount = var["maecenas_" || trigger.address || "_" || trigger.data["artist"]];
                    if (NOT $investedAmount)
                        bounce("You have no shares of that artist");
                    if (trigger.data["amount"] > $investedAmount)
                        bounce("You only have " || $investedAmount || " shares of that artist");
                    $yourShare = floor($artist.share * trigger.data["amount"] / $artist.supply, 0);
                    if ($yourShare < 20000)
                        bounce("It is not worth divesting your share");
                    
                        $allowsMaecenas = ($artist.sharePercent >= 0.01) ? true : false;
                        $hasMaecenas = $allowsMaecenas
                            ? $artist.supply > 0
                                ? true
                                : false
                            : false;
                    if ($allowsMaecenas){
                        $maecenasShare = 0;
                        $sharesInThePool = $artist.supply - trigger.data["amount"];    //If divesting we need to subtract divested shares
                        $p = $artist.share - floor((trigger.data["amount"] / $artist.supply) * $artist.share, 0);    //Current share minus what was just divested
                      $k = 1 / ($sharesInThePool + 1);                                                                //The fraction of the pool 1 share represents
                      $price = $_maecenasCurve($sharesInThePool + 1);                                                    //Price of buying one share
                        $totalSpendable = floor($spendableFunds + $price * $INVESTMENT_CUT, 0);
                        $sharesToBuy = $sharesInThePool == 0 ? 0 : ceil($calculateSharesToBuy($price, $p, $k), 0);//Can be 0 and I would still afford 1 share if profitable
                        if ($k == 1){
                            $sharesICanAfford = $price < $totalSpendable ? 1 : 0;
                        }
                        else {
                            if ($totalSpendable > $price AND $sharesToBuy >= 1){
                                $sharesICanAfford = floor($totalSpendable / ($price * $sharesToBuy * (1 - $INVESTMENT_CUT)), 0); //Do not buy more shares than what I can afford
                            }
                            else{
                                $sharesICanAfford = 0;
                            }
                        }
                    }
                    if ($allowsMaecenas AND $sharesICanAfford > 0){
                        $buyShares = true;
                    }
                    else{
                        $buyShares = false;
                    }
                    $shareIWillfinallyBuy = min($sharesToBuy, $sharesICanAfford); //Maybe I can afford more shares than what is profitable to buy
                    $myProfit = floor($buyShares ? $p - $p * $k^$shareIWillfinallyBuy : 0, 0); //0 if no shares are bought
                    $poolFinalValue = $p - $myProfit;            //Not actual profit but what I got out of the pool
                }",
                    "messages": [
                        {
                            "app": "payment",
                            "payload": {
                                "asset": "base",
                                "outputs": [
                                    {
                                        "address": "{trigger.address}",
                                        "amount": "{$yourShare - 10000}"
                                    }
                                ]
                            }
                        },
                        {
                            "app": "state",
                            "state": "{
                          var["artist_" || trigger.data["artist"]] ||= {
                              share: $artist.share - $yourShare,    //Decrease the share by the amount just divested
                              supply: $artist.supply - trigger.data["amount"]    //Decrease the supply by the shares used to divest
                          };
                          var["maecenas_" || trigger.address || "_" || trigger.data["artist"]] -= trigger.data["amount"];
                          var["locked"] -= ($yourShare + $myProfit);
                      }"
                        }
                    ]
                },
                {
                    "if": "{
                  $method == "ENABLE_INVESTING"
                    AND $INVESTING_ENABLED
                }",
                    "init": "{
                    if (var["artist_" || trigger.address])
                        bounce("You are already accepting maecenas");
                    if (NOT trigger.data["share"])
                        bounce("share field is mandatory");
                    if (NOT is_integer(trigger.data["share"]) OR trigger.data["share"] < 1 OR trigger.data["share"] > 50)
                        bounce("share must be an integer between 1 - 50");
                    if (exists(trigger.data["selfBuy"])){
                        if (NOT is_integer(trigger.data["selfBuy"]))
                            bounce("selfBuy field must be an integer");
                        if (trigger.data["selfBuy"] < 0)
                            bounce("selfBuy must be a positive integer");
                      $cost = (1161570 * trigger.data["selfBuy"] * (1 + trigger.data["selfBuy"]) * (1 + 2 * trigger.data["selfBuy"])) * $INVESTMENT_CUT;
                      if (trigger.output[[asset="base"]].amount < $cost)
                          bounce("You need to send at least " || $cost || " bytes");
                    }
                }",
                    "messages": [
                        {
                            "app": "state",
                            "state": "{
                          var["artist_" || trigger.address] = {
                              supply: trigger.data["selfBuy"] OTHERWISE 0,
                              share: 0,   //Pooled amount availabe to investors
                              sharePercent: trigger.data["share"] / 100
                          };
                          if (trigger.data["selfBuy"])
                              var["maecenas_" || trigger.address || "_" || trigger.address] = trigger.data["selfBuy"];
                      }"
                        }
                    ]
                },
                {
                    "if": "{
                  $method == "DISABLE_INVESTING"
                }",
                    "init": "{
                     $info = $profileOf(trigger.address);
                     if (NOT $info)
                         bounce("You are already not accepting maecenas");
                     if ($info.supply > 0)
                         bounce("You cannot stop accepting maecenas right now as you already have some");
                 }",
                    "messages": [
                        {
                            "app": "state",
                            "state": "{
                          var["artist_" || trigger.address] = false;
                      }"
                        }
                    ]
                },
                {
                    "if": "{
                  $method == "INVEST"
                }",
                    "init": "{
                  if (NOT trigger.data["artist"])
                      bounce("artist field is mandatory");
                    $artist = var["artist_" || trigger.data["artist"]];
                    if (NOT $artist)
                        bounce("That artist is not accepting maecenas");
                    if (NOT trigger.data["amount"])
                        bounce("amount field is mandatory");
                    if (NOT is_integer(trigger.data["amount"]) OR trigger.data["amount"] <= 0)
                        bounce("amount must be a positive integer");
                    $x0 = $artist.supply;
                    $x1 = $artist.supply + trigger.data["amount"];
                    $upToFirstToken = 1161570 * $x0 * (1 + $x0) * (1 + 2 * $x0);
                    $upToTargetPurchase = 1161570 * $x1 * (1 + $x1) * (1 + 2 * $x1);
                    $totalCost = round($upToTargetPurchase - $upToFirstToken, 0);
                    if (trigger.output[[asset="base"]].amount < $totalCost)
                        bounce("Your payment is not enough for that purchase");
                }",
                    "messages": [
                        {
                            "app": "payment",
                            "payload": {
                                "asset": "base",
                                "outputs": [
                                    {
                                        "address": "{trigger.data["artist"]}",
                                        "amount": "{floor($totalCost * (1 - $INVESTMENT_CUT) - 10000, 0)}"
                                    }
                                ]
                            }
                        },
                        {
                            "app": "state",
                            "state": "{
                            if (var["artist_" || trigger.data["artist"]])
                                var["artist_" || trigger.data["artist"]] ||= {supply: $artist.supply + trigger.data["amount"]};
                            else
                                var["artist_" || trigger.data["artist"]] = {supply: $artist.supply + trigger.data["amount"]};
                            if (var["maecenas_" || trigger.address || "_" || trigger.data["artist"]])
                                var["maecenas_" || trigger.address || "_" || trigger.data["artist"]] += trigger.data["amount"];
                            else
                                var["maecenas_" || trigger.address || "_" || trigger.data["artist"]] = trigger.data["amount"];
                        }"
                        }
                    ]
                },
                {
                    "if": "{
                  $method == "PROFILE"
                }",
                    "init": "{
                  $prof = $profileOf(trigger.address);
          $oldUsername = $prof.username;
          $newUsername = trigger.data["username"] OTHERWISE $oldUsername;
                    if (NOT $oldUsername){
             if (NOT trigger.data["username"])
                          bounce("username field is mandatory");
          }
            
                    if (var["user_" || $newUsername]){
            if ($oldUsername){
              if ($oldUsername != $newUsername){
                            bounce("username already taken by a verified user");
              }
            }
          }
                    if (exists(trigger.data["username"]) OR exists(trigger.data["name"]) OR exists(trigger.data["company"]) OR exists(trigger.data["isReseller"]) OR exists(trigger.data["verified"]))
                        $unverify = true;
                    else
                        $unverify = false;
          if ($prof)
                      $profileObj = $prof || trigger.data || {username: $newUsername, verified: NOT $unverify};
          else
            $profileObj = trigger.data || {username: $newUsername, verified: NOT $unverify};
                    delete($profileObj, "method");
                }",
                    "messages": [
                        {
                            "app": "attestation",
                            "init": "{
                            $profileString = json_stringify($profileObj);
                            if (length($profileString) > 4000)
                                bounce("Your profile is too big, try removing some fields");
                        }",
                            "payload": "{
              {
                address: trigger.address,
                profile: {
                                    data: $profileString
                                }
              }
            }"
                        },
                        {
                            "app": "state",
                            "state": "{
                          //var["profile_" || trigger.address] = $profileObj;
              if ($oldUsername){
                if ($oldUsername != $newUsername)
                  var["user_" || $oldUsername] = false; //Free old username
              }
            //Do not lock new username here. It will be locked upon verification to prevent username squatting
                      }"
                        }
                    ]
                },
                {
                    "if": "{
                    $method == "addShare"
                }",
                    "init": "{
                    //Failsafe measures against a bug being found on one of our AAs
                    if (NOT var["AA_" || trigger.address])
                        bounce("You have no power here!");
                }",
                    "messages": [
                        {
                            "app": "state",
                            "state": "{
                            $current = $artistInfoOf(trigger.data["artist"]);
                            var["artist_" || trigger.data["artist"]] = $current || {share: ($current.share OTHERWISE 0) + trigger.output[[asset="base"]].amount}; //Update the pool value
                            var["locked"] += trigger.output[[asset="base"]].amount; //Owner cannot withdraw this
                        }"
                        }
                    ]
                },
                {
                    "if": "{
                    $method == "addTime"
                }",
                    "init": "{
                    if (NOT trigger.data["NFT"])
                        bounce("[AA] NFT field is mandatory");
                    if (NOT trigger.data["time"])
                        bounce(["[AA] time field is mandatory"]);
                    if (NOT var["AA_" || trigger.address])
                        bounce("You have no power here!");
                }",
                    "messages": [
                        {
                            "app": "state",
                            "state": "{
                            $key = trigger.initial_address || "_" || trigger.data["NFT"];
                            $previousExpiry = var[$key];
                            if (($previousExpiry AND timestamp < $previousExpiry) OR !$previousExpiry) //License is expired or was never purchased, need to reset the ts
                                var[$key] = timestamp + trigger.data["time"];
                            else //License is active just push the deadline further
                                var[$key] += trigger.data["time"];
                        }"
                        }
                    ]
                }
            ]
        }
    }
]