tx · EDrsrCohnbqjcYv1twZd3muUK2gx5Mm9sA5ifkbWPa1e

3MrMbQwr3cpCvWMWCFjqUZzqRgxg8En6Jd5:  -0.03700000 Waves

2022.10.20 20:24 [2281028] smart account 3MrMbQwr3cpCvWMWCFjqUZzqRgxg8En6Jd5 > SELF 0.00000000 Waves

{ "type": 13, "id": "EDrsrCohnbqjcYv1twZd3muUK2gx5Mm9sA5ifkbWPa1e", "fee": 3700000, "feeAssetId": null, "timestamp": 1666286737683, "version": 2, "chainId": 84, "sender": "3MrMbQwr3cpCvWMWCFjqUZzqRgxg8En6Jd5", "senderPublicKey": "8WTLBzi1s9cZm76TPZ9czV6poaHGWvpZ7sCBeTBaa7uh", "proofs": [ "32Hvm54xBJCK2oRRZbL97PsoUSaTBQU1aA4NJVaDosupgybPeroPYkvfsECJrEVYG1dDmbrmebWDN2H5E27gJHCw" ], "script": "base64:BgJLCAISBAoCCAgSAwoBCBIAEgUKAwgIARIFCgMICAESBgoECAgIARIFCgMICAgSBAoCCAgSBAoCCAESBQoDCAgBEgQKAggBEgQKAggBUgATa190b3RhbEZlZXNJblBlcmlvZAITa190b3RhbEZlZXNJblBlcmlvZAAYa190b3RhbEFzc2V0RmVlc0luUGVyaW9kAhhrX3RvdGFsQXNzZXRGZWVzSW5QZXJpb2QAFGtfdHJhZGVyRmVlc0luUGVyaW9kAhRrX3RyYWRlckZlZXNJblBlcmlvZAAOa19sYXN0Tm90aW9uYWwCDmtfbGFzdE5vdGlvbmFsABRrX3RvdGFsU2NvcmVJblBlcmlvZAIUa190b3RhbFNjb3JlSW5QZXJpb2QAFWtfdHJhZGVyU2NvcmVJblBlcmlvZAIVa190cmFkZXJTY29yZUluUGVyaW9kAB9rX3RyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kAh9rX3RyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kAA9rX2FtbVJld2FyZFJhdGUCD2tfYW1tUmV3YXJkUmF0ZQAZa19hc3NldE1heEFtb3VudFBlclBlcmlvZAIZa19hc3NldE1heEFtb3VudFBlclBlcmlvZAAQa19yZXdhcmRBc3NldElkcwIQa19yZXdhcmRBc3NldElkcwAGa19hbW1zAgZrX2FtbXMAF2tfY2xhaW1lZEFzc2V0QW5kUGVyaW9kAhdrX2NsYWltZWRBc3NldEFuZFBlcmlvZAAPa19vcmFjbGVBZGRyZXNzAg9rX29yYWNsZUFkZHJlc3MADWtfb3JhY2xlUHJpY2UCBXByaWNlAB9rX3RvdGFsQ2xhaW1lZEZvclRyYWRlckFuZEFzc2V0Ah9rX3RvdGFsQ2xhaW1lZEZvclRyYWRlckFuZEFzc2V0ABdrX2FtbU1heEFtb3VudFBlclBlcmlvZAIXa19hbW1NYXhBbW91bnRQZXJQZXJpb2QADWtfaW5pdGlhbGl6ZWQCDWtfaW5pdGlhbGl6ZWQAFGtfY29vcmRpbmF0b3JBZGRyZXNzAhRrX2Nvb3JkaW5hdG9yQWRkcmVzcwASa19nb3Zlcm5hbmNlX2Fzc2V0AgtrX2dvdl9hc3NldAANa19xdW90ZV9hc3NldAINa19xdW90ZV9hc3NldAASa19hZG1pbl9wdWJsaWNfa2V5AhJrX2FkbWluX3B1YmxpY19rZXkAD2tfYWRtaW5fYWRkcmVzcwIPa19hZG1pbl9hZGRyZXNzAAVrX2FtbQIFa19hbW0ADERFQ0lNQUxfVU5JVAkAaAIAAQkAaAIJAGgCCQBoAgkAaAIJAGgCCQBoAgkAaAIACgAKAAoACgAKAAoACgAKAAFBAIC7sCEACk1TX0lOX1dFRUsJAGgCAOgHAID1JAEJbGlzdFRvU3RyAQVfbGlzdAoBBV9qb2luAgthY2N1bXVsYXRvcgN2YWwJAKwCAgkArAICBQthY2N1bXVsYXRvcgUDdmFsAgEsBApuZXdMaXN0U3RyCgACJGwFBV9saXN0CgACJHMJAJADAQUCJGwKAAUkYWNjMAIACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQVfam9pbgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIVTGlzdCBzaXplIGV4Y2VlZHMgMTAwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAAVABYAFwAYABkAGgAbABwAHQAeAB8AIAAhACIAIwAkACUAJgAnACgAKQAqACsALAAtAC4ALwAwADEAMgAzADQANQA2ADcAOAA5ADoAOwA8AD0APgA/AEAAQQBCAEMARABFAEYARwBIAEkASgBLAEwATQBOAE8AUABRAFIAUwBUAFUAVgBXAFgAWQBaAFsAXABdAF4AXwBgAGEAYgBjAGQEC25ld0xpc3RTdHJVCQCzAgIFCm5ld0xpc3RTdHIAAQQLbmV3TGlzdFN0clIDCQAAAgkArwICBQtuZXdMaXN0U3RyVQABAgEsCQCwAgIFC25ld0xpc3RTdHJVAAEFC25ld0xpc3RTdHJVBQtuZXdMaXN0U3RyUgEJc3RyVG9MaXN0AQRfc3RyAwkAAAIFBF9zdHICAAUDbmlsCQC1CQIFBF9zdHICASwBBGRpdmQCAl94Al95CQBuBAUCX3gFDERFQ0lNQUxfVU5JVAUCX3kFCEhBTEZFVkVOAQRtdWxkAgJfeAJfeQkAbgQFAl94BQJfeQUMREVDSU1BTF9VTklUBQhIQUxGRVZFTgEEcG93ZAICX3gCX3kJAGwGBQJfeAAIBQJfeQAIAAgFCEhBTEZFVkVOAQNhYnMBAl94AwkAZgIFAl94AAAFAl94CQEBLQEFAl94AQRtaW52AgJfeAJfeQMJAGYCBQJfeAUCX3kFAl95BQJfeAEOdG9Db21wb3NpdGVLZXkCBF9rZXkIX2FkZHJlc3MJAKwCAgkArAICBQRfa2V5AgFfBQhfYWRkcmVzcwAEdGhhdAUEdGhpcwELY29vcmRpbmF0b3IACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgUEdGhhdAUUa19jb29yZGluYXRvckFkZHJlc3MCE0Nvb3JkaW5hdG9yIG5vdCBzZXQBDmFkbWluUHVibGljS2V5AAkA2QQBCQERQGV4dHJOYXRpdmUoMTA1MykCCQELY29vcmRpbmF0b3IABRJrX2FkbWluX3B1YmxpY19rZXkBDGFkbWluQWRkcmVzcwAJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUPa19hZG1pbl9hZGRyZXNzAQtpc1doaXRlbGlzdAEIX2FkZHJlc3MJAQt2YWx1ZU9yRWxzZQIJAJsIAgkBC2Nvb3JkaW5hdG9yAAkBDnRvQ29tcG9zaXRlS2V5AgUFa19hbW0FCF9hZGRyZXNzBwEDaW50AQFrCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUEdGhhdAUBawkArAICAg1ubyB2YWx1ZSBmb3IgBQFrAQRpbnQwAQFrCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoYXQFAWsAAAELaW5pdGlhbGl6ZWQACQELdmFsdWVPckVsc2UCCQCbCAIFBHRoYXQFDWtfaW5pdGlhbGl6ZWQHARFnZXRSZXdhcmRBc3NldElkcwAJAQlzdHJUb0xpc3QBCQELdmFsdWVPckVsc2UCCQCdCAIFBHRoYXQFEGtfcmV3YXJkQXNzZXRJZHMCAAAEVElNRQgFCWxhc3RCbG9jawl0aW1lc3RhbXABGGdldFRyYWRlckZlZXNJblBlcmlvZEtleQMEX2FtbQdfdHJhZGVyB193ZWVrSWQJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIFFGtfdHJhZGVyRmVlc0luUGVyaW9kAgFfBQRfYW1tAgFfBQdfdHJhZGVyAgFfCQCkAwEFB193ZWVrSWQBGmdldFRvdGFsQW1tRmVlc0luUGVyaW9kS2V5AgRfYW1tB193ZWVrSWQJAKwCAgkArAICCQCsAgIJAKwCAgUTa190b3RhbEZlZXNJblBlcmlvZAIBXwUEX2FtbQIBXwkApAMBBQdfd2Vla0lkARxnZXRUb3RhbEFzc2V0RmVlc0luUGVyaW9kS2V5AghfYXNzZXRJZAdfd2Vla0lkCQCsAgIJAKwCAgkArAICCQCsAgIFGGtfdG90YWxBc3NldEZlZXNJblBlcmlvZAIBXwUIX2Fzc2V0SWQCAV8JAKQDAQUHX3dlZWtJZAESZ2V0TGFzdE5vdGlvbmFsS2V5AgRfYW1tB190cmFkZXIJAKwCAgkArAICCQCsAgIJAKwCAgUOa19sYXN0Tm90aW9uYWwCAV8FBF9hbW0CAV8FB190cmFkZXIBGWdldFRyYWRlclNjb3JlSW5QZXJpb2RLZXkDBF9hbW0HX3RyYWRlcgdfd2Vla0lkCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICBRVrX3RyYWRlclNjb3JlSW5QZXJpb2QCAV8FBF9hbW0CAV8FB190cmFkZXICAV8JAKQDAQUHX3dlZWtJZAEYZ2V0VG90YWxTY29yZUluUGVyaW9kS2V5AgRfYW1tB193ZWVrSWQJAKwCAgkArAICCQCsAgIJAKwCAgUUa190b3RhbFNjb3JlSW5QZXJpb2QCAV8FBF9hbW0CAV8JAKQDAQUHX3dlZWtJZAEjZ2V0VHJhZGVyQXZlcmFnZU5vdGlvbmFsSW5QZXJpb2RLZXkDBF9hbW0HX3RyYWRlcgdfd2Vla0lkCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICBR9rX3RyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kAgFfBQRfYW1tAgFfBQdfdHJhZGVyAgFfCQCkAwEFB193ZWVrSWQBE2dldEFtbVJld2FyZFJhdGVLZXkCBF9hbW0IX2Fzc2V0SWQJAKwCAgkArAICCQCsAgIJAKwCAgUPa19hbW1SZXdhcmRSYXRlAgFfBQRfYW1tAgFfBQhfYXNzZXRJZAEdZ2V0QXNzZXRNYXhBbW91bnRQZXJQZXJpb2RLZXkBCF9hc3NldElkCQCsAgIJAKwCAgUZa19hc3NldE1heEFtb3VudFBlclBlcmlvZAIBXwUIX2Fzc2V0SWQBHmdldENsYWltZWRUcmFkZXJBc3NldFBlcmlvZEtleQMHX3RyYWRlcghfYXNzZXRJZAdfcGVyaW9kCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICBRdrX2NsYWltZWRBc3NldEFuZFBlcmlvZAIBXwUHX3RyYWRlcgIBXwUIX2Fzc2V0SWQCAV8JAKQDAQUHX3BlcmlvZAERZ2V0T3JhY2xlUHJpY2VLZXkCCF9hc3NldElkB19wZXJpb2QJAKwCAgkArAICCQCsAgIJAKwCAgUNa19vcmFjbGVQcmljZQIBXwkApAMBBQdfcGVyaW9kAgFfBQhfYXNzZXRJZAEjZ2V0VG90YWxDbGFpbWVkRm9yVHJhZGVyQW5kQXNzZXRLZXkCB190cmFkZXIIX2Fzc2V0SWQJAKwCAgkArAICCQCsAgIJAKwCAgUfa190b3RhbENsYWltZWRGb3JUcmFkZXJBbmRBc3NldAIBXwUHX3RyYWRlcgIBXwUIX2Fzc2V0SWQBG2dldEFtbU1heEFtb3VudFBlclBlcmlvZEtleQIEX2FtbQhfYXNzZXRJZAkArAICCQCsAgIJAKwCAgkArAICBRdrX2FtbU1heEFtb3VudFBlclBlcmlvZAIBXwUEX2FtbQIBXwUIX2Fzc2V0SWQBDnVzZG5Ub0RlY2ltYWxzAQdfYW1vdW50CQBoAgUHX2Ftb3VudABkAQxnZXRXZWVrU3RhcnQBB193ZWVrSWQJAGgCBQdfd2Vla0lkBQpNU19JTl9XRUVLAQpnZXRXZWVrRW5kAQdfd2Vla0lkCQBoAgkAZAIFB193ZWVrSWQAAQUKTVNfSU5fV0VFSwEJZ2V0V2Vla0lkAQNfdHMJAGkCBQNfdHMFCk1TX0lOX1dFRUsBD2dldEZlZXNJblBlcmlvZAMEX2FtbQdfdHJhZGVyB193ZWVrSWQEA2tleQkBGGdldFRyYWRlckZlZXNJblBlcmlvZEtleQMFBF9hbW0FB190cmFkZXIFB193ZWVrSWQJAQRpbnQwAQUDa2V5ARRnZXRUb3RhbEZlZXNJblBlcmlvZAIEX2FtbQdfd2Vla0lkBANrZXkJARpnZXRUb3RhbEFtbUZlZXNJblBlcmlvZEtleQIFBF9hbW0FB193ZWVrSWQJAQRpbnQwAQUDa2V5ARlnZXRUb3RhbEFzc2V0RmVlc0luUGVyaW9kAghfYXNzZXRJZAdfd2Vla0lkBANrZXkJARxnZXRUb3RhbEFzc2V0RmVlc0luUGVyaW9kS2V5AgUIX2Fzc2V0SWQFB193ZWVrSWQJAQRpbnQwAQUDa2V5AQ9nZXRMYXN0Tm90aW9uYWwCBF9hbW0HX3RyYWRlcgQDa2V5CQESZ2V0TGFzdE5vdGlvbmFsS2V5AgUEX2FtbQUHX3RyYWRlcgkBBGludDABBQNrZXkBFmdldFRyYWRlclNjb3JlSW5QZXJpb2QDBF9hbW0HX3RyYWRlcgdfd2Vla0lkBANrZXkJARlnZXRUcmFkZXJTY29yZUluUGVyaW9kS2V5AwUEX2FtbQUHX3RyYWRlcgUHX3dlZWtJZAkBBGludDABBQNrZXkBFWdldFRvdGFsU2NvcmVJblBlcmlvZAIEX2FtbQdfd2Vla0lkBANrZXkJARhnZXRUb3RhbFNjb3JlSW5QZXJpb2RLZXkCBQRfYW1tBQdfd2Vla0lkCQEEaW50MAEFA2tleQEHZ2V0QW1tcwAJAQlzdHJUb0xpc3QBCQELdmFsdWVPckVsc2UCCQCdCAIFBHRoYXQFBmtfYW1tcwIAARBnZXRBbW1SZXdhcmRSYXRlAgRfYW1tCF9hc3NldElkBANrZXkJARNnZXRBbW1SZXdhcmRSYXRlS2V5AgUEX2FtbQUIX2Fzc2V0SWQJAQRpbnQwAQUDa2V5ARpnZXRBc3NldE1heEFtb3VudFBlclBlcmlvZAEIX2Fzc2V0SWQEA2tleQkBHWdldEFzc2V0TWF4QW1vdW50UGVyUGVyaW9kS2V5AQUIX2Fzc2V0SWQJAQRpbnQwAQUDa2V5ASBnZXRUb3RhbENsYWltZWRGb3JUcmFkZXJBbmRBc3NldAIHX3RyYWRlcghfYXNzZXRJZAQDa2V5CQEjZ2V0VG90YWxDbGFpbWVkRm9yVHJhZGVyQW5kQXNzZXRLZXkCBQdfdHJhZGVyBQhfYXNzZXRJZAkBBGludDABBQNrZXkBE2dldFJld2FyZEFzc2V0UHJpY2UCCF9hc3NldElkB193ZWVrSWQEEG9yYWNsZUFkZHJlc3NTdHIJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQR0aGF0BQ9rX29yYWNsZUFkZHJlc3MCDk9yYWNsZSBub3Qgc2V0BA1vcmFjbGVBZGRyZXNzCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQUQb3JhY2xlQWRkcmVzc1N0cgIWSW52YWxpZCBvcmFjbGUgYWRkcmVzcwQIcHJpY2VLZXkJARFnZXRPcmFjbGVQcmljZUtleQIFCF9hc3NldElkBQdfd2Vla0lkCQEOdXNkblRvRGVjaW1hbHMBCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUNb3JhY2xlQWRkcmVzcwUIcHJpY2VLZXkJAKwCAgkArAICCQCsAgICGk5vIG9yYWNsZSBwcmljZSBmb3IgYXNzZXQgBQhfYXNzZXRJZAIIIHBlcmlvZCAJAKQDAQUHX3dlZWtJZAEfZ2V0TWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQMEX2FtbQhfYXNzZXRJZAdfd2Vla0lkBBF0b3RhbEZlZXNJblBlcmlvZAkBFGdldFRvdGFsRmVlc0luUGVyaW9kAgUEX2FtbQUHX3dlZWtJZAQWdG90YWxBc3NldEZlZXNJblBlcmlvZAkBGWdldFRvdGFsQXNzZXRGZWVzSW5QZXJpb2QCBQhfYXNzZXRJZAUHX3dlZWtJZAMJAAACBRZ0b3RhbEFzc2V0RmVlc0luUGVyaW9kAAAEA2tleQkBG2dldEFtbU1heEFtb3VudFBlclBlcmlvZEtleQIFBF9hbW0FCF9hc3NldElkCQEEaW50MAEFA2tleQQQbWF4QW1vdW50T2ZBc3NldAkBGmdldEFzc2V0TWF4QW1vdW50UGVyUGVyaW9kAQUIX2Fzc2V0SWQEHG1heEFtb3VudE9mQXNzZXRUb0Rpc3RyaWJ1dGUJAQRtdWxkAgkBBGRpdmQCBRF0b3RhbEZlZXNJblBlcmlvZAUWdG90YWxBc3NldEZlZXNJblBlcmlvZAUQbWF4QW1vdW50T2ZBc3NldAUcbWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQEecmV3YXJkRm9yVHJhZGVySW5Bc3NldEluUGVyaW9kBARfYW1tB190cmFkZXIIX2Fzc2V0SWQHX3dlZWtJZAQPcmV3YXJkQXNzZXRSYXRlCQEQZ2V0QW1tUmV3YXJkUmF0ZQIFBF9hbW0FCF9hc3NldElkAwkAAAIFD3Jld2FyZEFzc2V0UmF0ZQAAAAAEEXRvdGFsRmVlc0luUGVyaW9kCQEUZ2V0VG90YWxGZWVzSW5QZXJpb2QCBQRfYW1tBQdfd2Vla0lkBBByZXdhcmRBc3NldFByaWNlCQETZ2V0UmV3YXJkQXNzZXRQcmljZQIFCF9hc3NldElkBQdfd2Vla0lkBBxtYXhBbW91bnRPZkFzc2V0VG9EaXN0cmlidXRlCQEfZ2V0TWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQMFBF9hbW0FCF9hc3NldElkBQdfd2Vla0lkBAt0cmFkZXJTY29yZQkBFmdldFRyYWRlclNjb3JlSW5QZXJpb2QDBQRfYW1tBQdfdHJhZGVyBQdfd2Vla0lkBBB0b3RhbFRyYWRlclNjb3JlCQEVZ2V0VG90YWxTY29yZUluUGVyaW9kAgUEX2FtbQUHX3dlZWtJZAMJAAACBRB0b3RhbFRyYWRlclNjb3JlAAAAAAQWdG90YWxBc3NldFRvRGlzdHJpYnV0ZQkBBGRpdmQCCQEEbXVsZAIFEXRvdGFsRmVlc0luUGVyaW9kBQ9yZXdhcmRBc3NldFJhdGUFEHJld2FyZEFzc2V0UHJpY2UEHGFjdHVhbFRvdGFsQXNzZXRUb0Rpc3RyaWJ1dGUJAQRtaW52AgUWdG90YWxBc3NldFRvRGlzdHJpYnV0ZQUcbWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQQUdHJhZGVyU2hhcmVJblJld2FyZHMJAQRkaXZkAgkBBG11bGQCBRxhY3R1YWxUb3RhbEFzc2V0VG9EaXN0cmlidXRlBQt0cmFkZXJTY29yZQUQdG90YWxUcmFkZXJTY29yZQUUdHJhZGVyU2hhcmVJblJld2FyZHMBI3RvdGFsUmV3YXJkRm9yVHJhZGVySW5Bc3NldEluUGVyaW9kAwdfdHJhZGVyCF9hc3NldElkB193ZWVrSWQEBGFtbXMJAQdnZXRBbW1zAAoBB2NvbXB1dGUCBF9hY2MEX2FtbQkAZAIFBF9hY2MJAR5yZXdhcmRGb3JUcmFkZXJJbkFzc2V0SW5QZXJpb2QEBQRfYW1tBQdfdHJhZGVyBQhfYXNzZXRJZAUHX3dlZWtJZAoAAiRsBQRhbW1zCgACJHMJAJADAQUCJGwKAAUkYWNjMAAACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQdjb21wdXRlAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAyMAkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAANAA4ADwAQABEAEgATABQBG3RvdGFsUmV3YXJkRm9yVHJhZGVySW5Bc3NldAMHX3RyYWRlcghfYXNzZXRJZAhfcGVyaW9kcwoBB2NvbXB1dGUCBF9hY2MHX3dlZWtJZAkAZAIFBF9hY2MJASN0b3RhbFJld2FyZEZvclRyYWRlckluQXNzZXRJblBlcmlvZAMFB190cmFkZXIFCF9hc3NldElkBQdfd2Vla0lkCgACJGwFCF9wZXJpb2RzCgACJHMJAJADAQUCJGwKAAUkYWNjMAAACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQdjb21wdXRlAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxMgkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAEgZ2V0VHJhZGVyQXZlcmFnZU5vdGlvbmFsSW5QZXJpb2QEBF9hbW0HX3RyYWRlcgdfd2Vla0lkDV9kZWZhdWx0VmFsdWUEA2tleQkBI2dldFRyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kS2V5AwUEX2FtbQUHX3RyYWRlcgUHX3dlZWtJZAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGF0BQNrZXkFDV9kZWZhdWx0VmFsdWUBCWlzQ2xhaW1lZAMHX3RyYWRlcghfYXNzZXRJZAdfcGVyaW9kBANrZXkJAR5nZXRDbGFpbWVkVHJhZGVyQXNzZXRQZXJpb2RLZXkDBQdfdHJhZGVyBQhfYXNzZXRJZAUHX3BlcmlvZAkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGF0BQNrZXkHAQZhZGp1c3QCB19hbW91bnQIX2Fzc2V0SWQEBWFzc2V0CQETdmFsdWVPckVycm9yTWVzc2FnZQIJAOwHAQkA2QQBBQhfYXNzZXRJZAIQSW52YWxpZCBhc3NldCBpZAQIZGVjaW1hbHMIBQVhc3NldAhkZWNpbWFscwMJAAACBQhkZWNpbWFscwAGCQBpAgUHX2Ftb3VudABkBQdfYW1vdW50ASBjbGFpbUFsbFJld2FyZEZvclBlcmlvZEZvclRyYWRlcgMHX3RyYWRlcghfYXNzZXRJZAhfcGVyaW9kcwQHcGVyaW9kcwkBCXN0clRvTGlzdAEFCF9wZXJpb2RzCgEQY2hlY2tBbmRGaWx0ZXJGbgIEX2FjYwVfbmV4dAQGcGVyaW9kCQETdmFsdWVPckVycm9yTWVzc2FnZQIJALYJAQUFX25leHQJAKwCAgIQSW52YWxpZCBwZXJpb2Q6IAUFX25leHQDAwkBD2NvbnRhaW5zRWxlbWVudAIFBF9hY2MFBnBlcmlvZAYJAQlpc0NsYWltZWQDBQdfdHJhZGVyBQhfYXNzZXRJZAUGcGVyaW9kBQRfYWNjCQDNCAIFBF9hY2MFBnBlcmlvZAQMdmFsaWRQZXJpb2RzCgACJGwFB3BlcmlvZHMKAAIkcwkAkAMBBQIkbAoABSRhY2MwBQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBEGNoZWNrQW5kRmlsdGVyRm4CBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEyCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMCgESbWFya1BlcmlvZEFzRG9uZUZuAgRfYWNjB19wZXJpb2QJAM0IAgUEX2FjYwkBDEJvb2xlYW5FbnRyeQIJAR5nZXRDbGFpbWVkVHJhZGVyQXNzZXRQZXJpb2RLZXkDBQdfdHJhZGVyBQhfYXNzZXRJZAUHX3BlcmlvZAYEFG1hcmtQZXJpb2RzQXNDbGFpbWVkCgACJGwFDHZhbGlkUGVyaW9kcwoAAiRzCQCQAwEFAiRsCgAFJGFjYzAFA25pbAoBBSRmMV8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQESbWFya1BlcmlvZEFzRG9uZUZuAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYxXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxNgkBBSRmMV8yAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAANAA4ADwAQBAZhbW91bnQJARt0b3RhbFJld2FyZEZvclRyYWRlckluQXNzZXQDBQdfdHJhZGVyBQhfYXNzZXRJZAUMdmFsaWRQZXJpb2RzCQCUCgIFBmFtb3VudAkAzggCCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMJARFAZXh0ck5hdGl2ZSgxMDYyKQEFB190cmFkZXIJAQZhZGp1c3QCBQZhbW91bnQFCF9hc3NldElkCQDZBAEFCF9hc3NldElkCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEjZ2V0VG90YWxDbGFpbWVkRm9yVHJhZGVyQW5kQXNzZXRLZXkCBQdfdHJhZGVyBQhfYXNzZXRJZAkAZAIJASBnZXRUb3RhbENsYWltZWRGb3JUcmFkZXJBbmRBc3NldAIFB190cmFkZXIFCF9hc3NldElkBQZhbW91bnQFA25pbAUUbWFya1BlcmlvZHNBc0NsYWltZWQBDXVwZGF0ZUFtbUZlZXMEBF9hbW0HX3RyYWRlcgdfd2Vla0lkCl9hY3R1YWxGZWUEDXRyYWRlckZlZXNLZXkJARhnZXRUcmFkZXJGZWVzSW5QZXJpb2RLZXkDBQRfYW1tBQdfdHJhZGVyBQdfd2Vla0lkBAx0b3RhbEZlZXNLZXkJARpnZXRUb3RhbEFtbUZlZXNJblBlcmlvZEtleQIFBF9hbW0FB193ZWVrSWQEDHJld2FyZEFzc2V0cwkBEWdldFJld2FyZEFzc2V0SWRzAAQMZmVlc0luUGVyaW9kCQEPZ2V0RmVlc0luUGVyaW9kAwUEX2FtbQUHX3RyYWRlcgUHX3dlZWtJZAQRdG90YWxGZWVzSW5QZXJpb2QJARRnZXRUb3RhbEZlZXNJblBlcmlvZAIFBF9hbW0FB193ZWVrSWQED25ld0ZlZXNJblBlcmlvZAkAZAIFDGZlZXNJblBlcmlvZAUKX2FjdHVhbEZlZQQUbmV3VG90YWxGZWVzSW5QZXJpb2QJAGQCBRF0b3RhbEZlZXNJblBlcmlvZAUKX2FjdHVhbEZlZQoBFWRvSGFuZGxlUmV3YXJkQXNzZXRJZAIEX2FjYwhfYXNzZXRJZAQDa2V5CQETZ2V0QW1tUmV3YXJkUmF0ZUtleQIFBF9hbW0FCF9hc3NldElkAwkAZgIJAQRpbnQwAQUDa2V5AAAEEXRvdGFsQXNzZXRGZWVzS2V5CQEcZ2V0VG90YWxBc3NldEZlZXNJblBlcmlvZEtleQIFCF9hc3NldElkBQdfd2Vla0lkCQDNCAIFBF9hY2MJAQxJbnRlZ2VyRW50cnkCBRF0b3RhbEFzc2V0RmVlc0tleQkAZAIJARlnZXRUb3RhbEFzc2V0RmVlc0luUGVyaW9kAgUIX2Fzc2V0SWQFB193ZWVrSWQFCl9hY3R1YWxGZWUFBF9hY2MED3VwZGF0ZUFzc2V0RmVlcwoAAiRsBQxyZXdhcmRBc3NldHMKAAIkcwkAkAMBBQIkbAoABSRhY2MwBQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBFWRvSGFuZGxlUmV3YXJkQXNzZXRJZAIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMTAJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQ10cmFkZXJGZWVzS2V5BQ9uZXdGZWVzSW5QZXJpb2QJAMwIAgkBDEludGVnZXJFbnRyeQIFDHRvdGFsRmVlc0tleQUUbmV3VG90YWxGZWVzSW5QZXJpb2QFA25pbAUPdXBkYXRlQXNzZXRGZWVzAQ11cGRhdGVBbW1EYXRhBwRfYW1tB190cmFkZXIHX3dlZWtJZA5fdHJhZGVyQXZlcmFnZRNfdHJhZGVyTGFzdE5vdGlvbmFsDF90cmFkZXJTY29yZQtfdG90YWxTY29yZQQVdG90YWxTY29yZUluUGVyaW9kS2V5CQEYZ2V0VG90YWxTY29yZUluUGVyaW9kS2V5AgUEX2FtbQUHX3dlZWtJZAQWdHJhZGVyU2NvcmVJblBlcmlvZEtleQkBGWdldFRyYWRlclNjb3JlSW5QZXJpb2RLZXkDBQRfYW1tBQdfdHJhZGVyBQdfd2Vla0lkBA9sYXN0Tm90aW9uYWxLZXkJARJnZXRMYXN0Tm90aW9uYWxLZXkCBQRfYW1tBQdfdHJhZGVyBCB0cmFkZXJBdmVyYWdlTm90aW9uYWxJblBlcmlvZEtleQkBI2dldFRyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kS2V5AwUEX2FtbQUHX3RyYWRlcgUHX3dlZWtJZAkAzAgCCQEMSW50ZWdlckVudHJ5AgUVdG90YWxTY29yZUluUGVyaW9kS2V5BQtfdG90YWxTY29yZQkAzAgCCQEMSW50ZWdlckVudHJ5AgUWdHJhZGVyU2NvcmVJblBlcmlvZEtleQUMX3RyYWRlclNjb3JlCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQ9sYXN0Tm90aW9uYWxLZXkFE190cmFkZXJMYXN0Tm90aW9uYWwJAMwIAgkBDEludGVnZXJFbnRyeQIFIHRyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kS2V5BQ5fdHJhZGVyQXZlcmFnZQUDbmlsDAFpAQppbml0aWFsaXplAgxfY29vcmRpbmF0b3IHX29yYWNsZQMJAQtpbml0aWFsaXplZAAJAAIBAhNBbHJlYWR5IGluaXRpYWxpemVkCQDMCAIJAQtTdHJpbmdFbnRyeQIFFGtfY29vcmRpbmF0b3JBZGRyZXNzBQxfY29vcmRpbmF0b3IJAMwIAgkBC1N0cmluZ0VudHJ5AgUPa19vcmFjbGVBZGRyZXNzBQdfb3JhY2xlCQDMCAIJAQxCb29sZWFuRW50cnkCBQ1rX2luaXRpYWxpemVkBgUDbmlsAWkBEHNldE9yYWNsZUFkZHJlc3MBB19vcmFjbGUDAwkBASEBCQELaW5pdGlhbGl6ZWQABgkBAiE9AggFAWkGY2FsbGVyCQEMYWRtaW5BZGRyZXNzAAkAAgECI0ludmFsaWQgc2V0T3JhY2xlQWRkcmVzcyBwYXJhbWV0ZXJzCQDMCAIJAQtTdHJpbmdFbnRyeQIFD2tfb3JhY2xlQWRkcmVzcwUHX29yYWNsZQUDbmlsAWkBDnZpZXdfZ2V0UGVyaW9kAAQGd2Vla0lkCQEJZ2V0V2Vla0lkAQUEVElNRQQJd2Vla1N0YXJ0CQEMZ2V0V2Vla1N0YXJ0AQUGd2Vla0lkBAd3ZWVrRW5kCQEKZ2V0V2Vla0VuZAEFBndlZWtJZAkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkApAMBBQl3ZWVrU3RhcnQCASwJAKQDAQUHd2Vla0VuZAIBLAkApAMBBQRUSU1FAWkBJHZpZXdfZ2V0TWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQMEX2FtbQhfYXNzZXRJZAdfd2Vla0lkCQACAQkApAMBCQEfZ2V0TWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQMFBF9hbW0FCF9hc3NldElkBQdfd2Vla0lkAWkBEXZpZXdfY2xhaW1SZXdhcmRzAwdfdHJhZGVyCF9hc3NldElkB19wZXJpb2QJAAIBCQCkAwEJARt0b3RhbFJld2FyZEZvclRyYWRlckluQXNzZXQDBQdfdHJhZGVyBQhfYXNzZXRJZAkAzAgCBQdfcGVyaW9kBQNuaWwBaQEPdmlld19jYWxjUmV3YXJkBAdfdHJhZGVyBF9hbW0IX2Fzc2V0SWQHX3BlcmlvZAkAAgEJAKQDAQkBHnJld2FyZEZvclRyYWRlckluQXNzZXRJblBlcmlvZAQFBF9hbW0FCF9hc3NldElkBQdfdHJhZGVyBQdfcGVyaW9kAWkBFHZpZXdfY2xhaW1BbGxSZXdhcmRzAwdfdHJhZGVyCF9hc3NldElkCF9wZXJpb2RzBAZyZXN1bHQJASBjbGFpbUFsbFJld2FyZEZvclBlcmlvZEZvclRyYWRlcgMFB190cmFkZXIFCF9hc3NldElkBQhfcGVyaW9kcwkAAgEJAKwCAgkArAICCQCkAwEIBQZyZXN1bHQCXzECASwJAKQDAQkBIGdldFRvdGFsQ2xhaW1lZEZvclRyYWRlckFuZEFzc2V0AgUHX3RyYWRlcgUIX2Fzc2V0SWQBaQEPY2xhaW1BbGxSZXdhcmRzAghfYXNzZXRJZAhfcGVyaW9kcwQGdHJhZGVyCQClCAEIBQFpBmNhbGxlcgMJAQhjb250YWlucwIFCF9wZXJpb2RzCQCkAwEJAQlnZXRXZWVrSWQBBQRUSU1FCQACAQImQ2FuIG5vdCBjbGFpbSByZXdhcmRzIGZvciBjdXJyZW50IHdlZWsEDSR0MDE1NDM4MTU1MjEJASBjbGFpbUFsbFJld2FyZEZvclBlcmlvZEZvclRyYWRlcgMFBnRyYWRlcgUIX2Fzc2V0SWQFCF9wZXJpb2RzBAZhbW91bnQIBQ0kdDAxNTQzODE1NTIxAl8xBAZyZXN1bHQIBQ0kdDAxNTQzODE1NTIxAl8yAwkAAAIFBmFtb3VudAAACQACAQIQTm90aGluZyB0byBjbGFpbQUGcmVzdWx0AWkBEWF0dGFjaFJld2FyZEFzc2V0AghfYXNzZXRJZBNfbWF4QW1vdW50UGVyUGVyaW9kAwMJAQEhAQkBC2luaXRpYWxpemVkAAYJAQIhPQIIBQFpBmNhbGxlcgkBDGFkbWluQWRkcmVzcwAJAAIBAiBJbnZhbGlkIGF0dGFjaFJld2FyZEFzc2V0IHBhcmFtcwQEYW1tcwkBB2dldEFtbXMABA5yZXdhcmRBc3NldElkcwkBEWdldFJld2FyZEFzc2V0SWRzAAQRbmV3UmV3YXJkQXNzZXRJZHMDCQEPY29udGFpbnNFbGVtZW50AgUOcmV3YXJkQXNzZXRJZHMFCF9hc3NldElkBQ5yZXdhcmRBc3NldElkcwkAzQgCBQ5yZXdhcmRBc3NldElkcwUIX2Fzc2V0SWQJAMwIAgkBDEludGVnZXJFbnRyeQIJAR1nZXRBc3NldE1heEFtb3VudFBlclBlcmlvZEtleQEFCF9hc3NldElkBRNfbWF4QW1vdW50UGVyUGVyaW9kCQDMCAIJAQtTdHJpbmdFbnRyeQIFEGtfcmV3YXJkQXNzZXRJZHMJAQlsaXN0VG9TdHIBBRFuZXdSZXdhcmRBc3NldElkcwUDbmlsAWkBDWF0dGFjaFJld2FyZHMDBF9hbW0IX2Fzc2V0SWQLX3Jld2FyZFJhdGUDAwMJAQEhAQkBC2luaXRpYWxpemVkAAYJAQIhPQIIBQFpBmNhbGxlcgkBDGFkbWluQWRkcmVzcwAGCQEBIQEJAQtpc1doaXRlbGlzdAEFBF9hbW0JAAIBAhxJbnZhbGlkIGF0dGFjaFJld2FyZHMgcGFyYW1zBARhbW1zCQEHZ2V0QW1tcwAEDnJld2FyZEFzc2V0SWRzCQERZ2V0UmV3YXJkQXNzZXRJZHMABAduZXdBbW1zAwkBD2NvbnRhaW5zRWxlbWVudAIFBGFtbXMFBF9hbW0FBGFtbXMJAM0IAgUEYW1tcwUEX2FtbQQRbmV3UmV3YXJkQXNzZXRJZHMDCQEPY29udGFpbnNFbGVtZW50AgUOcmV3YXJkQXNzZXRJZHMFCF9hc3NldElkBQ5yZXdhcmRBc3NldElkcwkAzQgCBQ5yZXdhcmRBc3NldElkcwUIX2Fzc2V0SWQJAMwIAgkBDEludGVnZXJFbnRyeQIJARNnZXRBbW1SZXdhcmRSYXRlS2V5AgUEX2FtbQUIX2Fzc2V0SWQFC19yZXdhcmRSYXRlCQDMCAIJAQtTdHJpbmdFbnRyeQIFBmtfYW1tcwkBCWxpc3RUb1N0cgEFB25ld0FtbXMJAMwIAgkBC1N0cmluZ0VudHJ5AgUQa19yZXdhcmRBc3NldElkcwkBCWxpc3RUb1N0cgEFEW5ld1Jld2FyZEFzc2V0SWRzBQNuaWwBaQEKbm90aWZ5RmVlcwIHX3RyYWRlcgRfZmVlBANhbW0JAKUIAQgFAWkGY2FsbGVyAwMJAQEhAQkBC2lzV2hpdGVsaXN0AQUDYW1tBgkBASEBCQELaW5pdGlhbGl6ZWQACQACAQIZSW52YWxpZCBub3RpZnlGZWVzIHBhcmFtcwQJYWN0dWFsRmVlCQEOdXNkblRvRGVjaW1hbHMBBQRfZmVlBAZ3ZWVrSWQJAQlnZXRXZWVrSWQBBQRUSU1FCQENdXBkYXRlQW1tRmVlcwQFA2FtbQUHX3RyYWRlcgUGd2Vla0lkBQlhY3R1YWxGZWUBaQEObm90aWZ5Tm90aW9uYWwCB190cmFkZXIJX25vdGlvbmFsBANhbW0JAKUIAQgFAWkGY2FsbGVyAwMJAQEhAQkBC2lzV2hpdGVsaXN0AQUDYW1tBgkBASEBCQELaW5pdGlhbGl6ZWQACQACAQIdSW52YWxpZCBub3RpZnlOb3Rpb25hbCBwYXJhbXMEDmFjdHVhbE5vdGlvbmFsCQEOdXNkblRvRGVjaW1hbHMBBQlfbm90aW9uYWwEDGxhc3ROb3Rpb25hbAkBD2dldExhc3ROb3Rpb25hbAIFA2FtbQUHX3RyYWRlcgQGd2Vla0lkCQEJZ2V0V2Vla0lkAQUEVElNRQQJd2Vla1N0YXJ0CQEMZ2V0V2Vla1N0YXJ0AQUGd2Vla0lkBAd3ZWVrRW5kCQEKZ2V0V2Vla0VuZAEFBndlZWtJZAQBdAkAZQIFB3dlZWtFbmQFCXdlZWtTdGFydAQRb2xkUnVubmluZ0F2ZXJhZ2UJASBnZXRUcmFkZXJBdmVyYWdlTm90aW9uYWxJblBlcmlvZAQFA2FtbQUHX3RyYWRlcgUGd2Vla0lkBQxsYXN0Tm90aW9uYWwEAXcJAQRkaXZkAgkAZQIFB3dlZWtFbmQFBFRJTUUFAXQEEW9sZFRyYWRlckFtbVNjb3JlCQEWZ2V0VHJhZGVyU2NvcmVJblBlcmlvZAMFA2FtbQUHX3RyYWRlcgUGd2Vla0lkBBBvbGRUb3RhbEFtbVNjb3JlCQEVZ2V0VG90YWxTY29yZUluUGVyaW9kAgUDYW1tBQZ3ZWVrSWQEBGZlZXMJAQ9nZXRGZWVzSW5QZXJpb2QDBQNhbW0FB190cmFkZXIFBndlZWtJZAQRbmV3UnVubmluZ0F2ZXJhZ2UJAGUCCQBkAgURb2xkUnVubmluZ0F2ZXJhZ2UJAQRtdWxkAgUOYWN0dWFsTm90aW9uYWwFAXcJAQRtdWxkAgUMbGFzdE5vdGlvbmFsBQF3BBFuZXdUcmFkZXJBbW1TY29yZQkBBG11bGQCCQEEcG93ZAIFBGZlZXMFAUEJAQRwb3dkAgURbmV3UnVubmluZ0F2ZXJhZ2UJAGUCBQxERUNJTUFMX1VOSVQFAUEEGG5ld1RvdGFsQW1tU2NvcmVJblBlcmlvZAkAZAIJAGUCBRBvbGRUb3RhbEFtbVNjb3JlBRFvbGRUcmFkZXJBbW1TY29yZQURbmV3VHJhZGVyQW1tU2NvcmUJAQ11cGRhdGVBbW1EYXRhBwUDYW1tBQdfdHJhZGVyBQZ3ZWVrSWQFEW5ld1J1bm5pbmdBdmVyYWdlBQ5hY3R1YWxOb3Rpb25hbAURbmV3VHJhZGVyQW1tU2NvcmUFGG5ld1RvdGFsQW1tU2NvcmVJblBlcmlvZAECdHgBBnZlcmlmeQAJAPQDAwgFAnR4CWJvZHlCeXRlcwkAkQMCCAUCdHgGcHJvb2ZzAAAJAQ5hZG1pblB1YmxpY0tleQDQl1KH", "height": 2281028, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 9EVN2AZhWwXDWEWkTo6Zfq7NF8RBS1SQPLBfhfhmZ8AC Next: EWbphkhY6NbbuezqiPaYC3aVbUXB9C94t2kLetQdLB8R Diff:
OldNewDifferences
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let k_totalFeesInPeriod = "k_totalFeesInPeriod"
5+
6+let k_totalAssetFeesInPeriod = "k_totalAssetFeesInPeriod"
57
68 let k_traderFeesInPeriod = "k_traderFeesInPeriod"
79
1517
1618 let k_ammRewardRate = "k_ammRewardRate"
1719
18-let k_ammMaxAmountPerPeriod = "k_ammMaxAmountPerPeriod"
20+let k_assetMaxAmountPerPeriod = "k_assetMaxAmountPerPeriod"
1921
2022 let k_rewardAssetIds = "k_rewardAssetIds"
2123
2830 let k_oraclePrice = "price"
2931
3032 let k_totalClaimedForTraderAndAsset = "k_totalClaimedForTraderAndAsset"
33+
34+let k_ammMaxAmountPerPeriod = "k_ammMaxAmountPerPeriod"
3135
3236 let k_initialized = "k_initialized"
3337
101105 func toCompositeKey (_key,_address) = ((_key + "_") + _address)
102106
103107
104-func coordinator () = valueOrErrorMessage(addressFromString(getStringValue(this, k_coordinatorAddress)), "Coordinator not set")
108+let that = this
109+
110+func coordinator () = valueOrErrorMessage(addressFromString(getStringValue(that, k_coordinatorAddress)), "Coordinator not set")
105111
106112
107113 func adminPublicKey () = fromBase58String(getStringValue(coordinator(), k_admin_public_key))
113119 func isWhitelist (_address) = valueOrElse(getBoolean(coordinator(), toCompositeKey(k_amm, _address)), false)
114120
115121
116-func int (k) = valueOrErrorMessage(getInteger(this, k), ("no value for " + k))
122+func int (k) = valueOrErrorMessage(getInteger(that, k), ("no value for " + k))
117123
118124
119-func int0 (k) = valueOrElse(getInteger(this, k), 0)
125+func int0 (k) = valueOrElse(getInteger(that, k), 0)
120126
121127
122-func initialized () = valueOrElse(getBoolean(this, k_initialized), false)
128+func initialized () = valueOrElse(getBoolean(that, k_initialized), false)
123129
124130
125-func getFeesInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderFeesInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
131+func getRewardAssetIds () = strToList(valueOrElse(getString(that, k_rewardAssetIds), ""))
126132
127133
128-func getTotalFeesInPeriodKey (_amm,_weekId) = ((((k_totalFeesInPeriod + "_") + _amm) + "_") + toString(_weekId))
134+let TIME = lastBlock.timestamp
135+
136+func getTraderFeesInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderFeesInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
137+
138+
139+func getTotalAmmFeesInPeriodKey (_amm,_weekId) = ((((k_totalFeesInPeriod + "_") + _amm) + "_") + toString(_weekId))
140+
141+
142+func getTotalAssetFeesInPeriodKey (_assetId,_weekId) = ((((k_totalAssetFeesInPeriod + "_") + _assetId) + "_") + toString(_weekId))
129143
130144
131145 func getLastNotionalKey (_amm,_trader) = ((((k_lastNotional + "_") + _amm) + "_") + _trader)
143157 func getAmmRewardRateKey (_amm,_assetId) = ((((k_ammRewardRate + "_") + _amm) + "_") + _assetId)
144158
145159
146-func getAmmMaxAmountPerPeriodKey (_amm,_assetId) = ((((k_ammMaxAmountPerPeriod + "_") + _amm) + "_") + _assetId)
160+func getAssetMaxAmountPerPeriodKey (_assetId) = ((k_assetMaxAmountPerPeriod + "_") + _assetId)
147161
148162
149163 func getClaimedTraderAssetPeriodKey (_trader,_assetId,_period) = ((((((k_claimedAssetAndPeriod + "_") + _trader) + "_") + _assetId) + "_") + toString(_period))
155169 func getTotalClaimedForTraderAndAssetKey (_trader,_assetId) = ((((k_totalClaimedForTraderAndAsset + "_") + _trader) + "_") + _assetId)
156170
157171
158-func updateAmmFees (_amm,_trader,_weekId,_traderFees,_totalFees) = {
159- let traderFeesKey = getFeesInPeriodKey(_amm, _trader, _weekId)
160- let totalFeesKey = getTotalFeesInPeriodKey(_amm, _weekId)
161-[IntegerEntry(traderFeesKey, _traderFees), IntegerEntry(totalFeesKey, _totalFees)]
162- }
163-
164-
165-func updateAmmData (_amm,_trader,_weekId,_traderAverage,_traderLastNotional,_traderScore,_totalScore) = {
166- let totalScoreInPeriodKey = getTotalScoreInPeriodKey(_amm, _weekId)
167- let traderScoreInPeriodKey = getTraderScoreInPeriodKey(_amm, _trader, _weekId)
168- let lastNotionalKey = getLastNotionalKey(_amm, _trader)
169- let traderAverageNotionalInPeriodKey = getTraderAverageNotionalInPeriodKey(_amm, _trader, _weekId)
170-[IntegerEntry(totalScoreInPeriodKey, _totalScore), IntegerEntry(traderScoreInPeriodKey, _traderScore), IntegerEntry(lastNotionalKey, _traderLastNotional), IntegerEntry(traderAverageNotionalInPeriodKey, _traderAverage)]
171- }
172+func getAmmMaxAmountPerPeriodKey (_amm,_assetId) = ((((k_ammMaxAmountPerPeriod + "_") + _amm) + "_") + _assetId)
172173
173174
174175 func usdnToDecimals (_amount) = (_amount * 100)
184185
185186
186187 func getFeesInPeriod (_amm,_trader,_weekId) = {
187- let key = getFeesInPeriodKey(_amm, _trader, _weekId)
188- valueOrElse(getInteger(this, key), 0)
188+ let key = getTraderFeesInPeriodKey(_amm, _trader, _weekId)
189+ int0(key)
189190 }
190191
191192
192193 func getTotalFeesInPeriod (_amm,_weekId) = {
193- let key = getTotalFeesInPeriodKey(_amm, _weekId)
194- valueOrElse(getInteger(this, key), 0)
194+ let key = getTotalAmmFeesInPeriodKey(_amm, _weekId)
195+ int0(key)
196+ }
197+
198+
199+func getTotalAssetFeesInPeriod (_assetId,_weekId) = {
200+ let key = getTotalAssetFeesInPeriodKey(_assetId, _weekId)
201+ int0(key)
195202 }
196203
197204
198205 func getLastNotional (_amm,_trader) = {
199206 let key = getLastNotionalKey(_amm, _trader)
200- valueOrElse(getInteger(this, key), 0)
207+ int0(key)
201208 }
202209
203210
204211 func getTraderScoreInPeriod (_amm,_trader,_weekId) = {
205212 let key = getTraderScoreInPeriodKey(_amm, _trader, _weekId)
206- valueOrElse(getInteger(this, key), 0)
213+ int0(key)
207214 }
208215
209216
210217 func getTotalScoreInPeriod (_amm,_weekId) = {
211218 let key = getTotalScoreInPeriodKey(_amm, _weekId)
212- valueOrElse(getInteger(this, key), 0)
219+ int0(key)
213220 }
214221
215222
216-func getAmms () = strToList(valueOrElse(getString(this, k_amms), ""))
217-
218-
219-func getRewardAssetIds () = strToList(valueOrElse(getString(this, k_rewardAssetIds), ""))
223+func getAmms () = strToList(valueOrElse(getString(that, k_amms), ""))
220224
221225
222226 func getAmmRewardRate (_amm,_assetId) = {
223227 let key = getAmmRewardRateKey(_amm, _assetId)
224- valueOrElse(getInteger(this, key), 0)
228+ int0(key)
225229 }
226230
227231
228-func getAmmMaxAmountPerPeriod (_amm,_assetId) = {
229- let key = getAmmMaxAmountPerPeriodKey(_amm, _assetId)
230- valueOrElse(getInteger(this, key), 0)
232+func getAssetMaxAmountPerPeriod (_assetId) = {
233+ let key = getAssetMaxAmountPerPeriodKey(_assetId)
234+ int0(key)
231235 }
232236
233237
234238 func getTotalClaimedForTraderAndAsset (_trader,_assetId) = {
235239 let key = getTotalClaimedForTraderAndAssetKey(_trader, _assetId)
236- valueOrElse(getInteger(this, key), 0)
240+ int0(key)
237241 }
238242
239243
240244 func getRewardAssetPrice (_assetId,_weekId) = {
241- let oracleAddressStr = valueOrErrorMessage(getString(this, k_oracleAddress), "Oracle not set")
245+ let oracleAddressStr = valueOrErrorMessage(getString(that, k_oracleAddress), "Oracle not set")
242246 let oracleAddress = valueOrErrorMessage(addressFromString(oracleAddressStr), "Invalid oracle address")
243247 let priceKey = getOraclePriceKey(_assetId, _weekId)
244248 usdnToDecimals(valueOrErrorMessage(getInteger(oracleAddress, priceKey), ((("No oracle price for asset " + _assetId) + " period ") + toString(_weekId))))
249+ }
250+
251+
252+func getMaxAmountOfAssetToDistribute (_amm,_assetId,_weekId) = {
253+ let totalFeesInPeriod = getTotalFeesInPeriod(_amm, _weekId)
254+ let totalAssetFeesInPeriod = getTotalAssetFeesInPeriod(_assetId, _weekId)
255+ if ((totalAssetFeesInPeriod == 0))
256+ then {
257+ let key = getAmmMaxAmountPerPeriodKey(_amm, _assetId)
258+ int0(key)
259+ }
260+ else {
261+ let maxAmountOfAsset = getAssetMaxAmountPerPeriod(_assetId)
262+ let maxAmountOfAssetToDistribute = muld(divd(totalFeesInPeriod, totalAssetFeesInPeriod), maxAmountOfAsset)
263+ maxAmountOfAssetToDistribute
264+ }
245265 }
246266
247267
252272 else {
253273 let totalFeesInPeriod = getTotalFeesInPeriod(_amm, _weekId)
254274 let rewardAssetPrice = getRewardAssetPrice(_assetId, _weekId)
255- let maxAmountOfAssetToDistribute = getAmmMaxAmountPerPeriod(_amm, _assetId)
275+ let maxAmountOfAssetToDistribute = getMaxAmountOfAssetToDistribute(_amm, _assetId, _weekId)
256276 let traderScore = getTraderScoreInPeriod(_amm, _trader, _weekId)
257277 let totalTraderScore = getTotalScoreInPeriod(_amm, _weekId)
258278 if ((totalTraderScore == 0))
259279 then 0
260280 else {
261- let totalAssetToDistribute = muld(divd(totalFeesInPeriod, rewardAssetPrice), rewardAssetRate)
281+ let totalAssetToDistribute = divd(muld(totalFeesInPeriod, rewardAssetRate), rewardAssetPrice)
262282 let actualTotalAssetToDistribute = minv(totalAssetToDistribute, maxAmountOfAssetToDistribute)
263283 let traderShareInRewards = divd(muld(actualTotalAssetToDistribute, traderScore), totalTraderScore)
264284 traderShareInRewards
298318
299319 func $f0_2 ($a,$i) = if (($i >= $s))
300320 then $a
301- else throw("List size exceeds 18")
321+ else throw("List size exceeds 12")
302322
303- $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18)
323+ $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
304324 }
305325
306326
307327 func getTraderAverageNotionalInPeriod (_amm,_trader,_weekId,_defaultValue) = {
308328 let key = getTraderAverageNotionalInPeriodKey(_amm, _trader, _weekId)
309- valueOrElse(getInteger(this, key), _defaultValue)
329+ valueOrElse(getInteger(that, key), _defaultValue)
310330 }
311331
312332
313333 func isClaimed (_trader,_assetId,_period) = {
314334 let key = getClaimedTraderAssetPeriodKey(_trader, _assetId, _period)
315- valueOrElse(getBoolean(this, key), false)
335+ valueOrElse(getBoolean(that, key), false)
336+ }
337+
338+
339+func adjust (_amount,_assetId) = {
340+ let asset = valueOrErrorMessage(assetInfo(fromBase58String(_assetId)), "Invalid asset id")
341+ let decimals = asset.decimals
342+ if ((decimals == 6))
343+ then (_amount / 100)
344+ else _amount
316345 }
317346
318347
337366
338367 func $f0_2 ($a,$i) = if (($i >= $s))
339368 then $a
340- else throw("List size exceeds 18")
369+ else throw("List size exceeds 12")
341370
342- $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18)
371+ $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
343372 }
344373 func markPeriodAsDoneFn (_acc,_period) = (_acc :+ BooleanEntry(getClaimedTraderAssetPeriodKey(_trader, _assetId, _period), true))
345374
353382
354383 func $f1_2 ($a,$i) = if (($i >= $s))
355384 then $a
356- else throw("List size exceeds 18")
385+ else throw("List size exceeds 16")
357386
358- $f1_2($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18)
387+ $f1_2($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16)
359388 }
360389 let amount = totalRewardForTraderInAsset(_trader, _assetId, validPeriods)
361- $Tuple2(amount, ([ScriptTransfer(addressFromStringValue(_trader), amount, fromBase58String(_assetId)), IntegerEntry(getTotalClaimedForTraderAndAssetKey(_trader, _assetId), (getTotalClaimedForTraderAndAsset(_trader, _assetId) + amount))] ++ markPeriodsAsClaimed))
390+ $Tuple2(amount, ([ScriptTransfer(addressFromStringValue(_trader), adjust(amount, _assetId), fromBase58String(_assetId)), IntegerEntry(getTotalClaimedForTraderAndAssetKey(_trader, _assetId), (getTotalClaimedForTraderAndAsset(_trader, _assetId) + amount))] ++ markPeriodsAsClaimed))
391+ }
392+
393+
394+func updateAmmFees (_amm,_trader,_weekId,_actualFee) = {
395+ let traderFeesKey = getTraderFeesInPeriodKey(_amm, _trader, _weekId)
396+ let totalFeesKey = getTotalAmmFeesInPeriodKey(_amm, _weekId)
397+ let rewardAssets = getRewardAssetIds()
398+ let feesInPeriod = getFeesInPeriod(_amm, _trader, _weekId)
399+ let totalFeesInPeriod = getTotalFeesInPeriod(_amm, _weekId)
400+ let newFeesInPeriod = (feesInPeriod + _actualFee)
401+ let newTotalFeesInPeriod = (totalFeesInPeriod + _actualFee)
402+ func doHandleRewardAssetId (_acc,_assetId) = {
403+ let key = getAmmRewardRateKey(_amm, _assetId)
404+ if ((int0(key) > 0))
405+ then {
406+ let totalAssetFeesKey = getTotalAssetFeesInPeriodKey(_assetId, _weekId)
407+ (_acc :+ IntegerEntry(totalAssetFeesKey, (getTotalAssetFeesInPeriod(_assetId, _weekId) + _actualFee)))
408+ }
409+ else _acc
410+ }
411+
412+ let updateAssetFees = {
413+ let $l = rewardAssets
414+ let $s = size($l)
415+ let $acc0 = nil
416+ func $f0_1 ($a,$i) = if (($i >= $s))
417+ then $a
418+ else doHandleRewardAssetId($a, $l[$i])
419+
420+ func $f0_2 ($a,$i) = if (($i >= $s))
421+ then $a
422+ else throw("List size exceeds 10")
423+
424+ $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10)
425+ }
426+ ([IntegerEntry(traderFeesKey, newFeesInPeriod), IntegerEntry(totalFeesKey, newTotalFeesInPeriod)] ++ updateAssetFees)
427+ }
428+
429+
430+func updateAmmData (_amm,_trader,_weekId,_traderAverage,_traderLastNotional,_traderScore,_totalScore) = {
431+ let totalScoreInPeriodKey = getTotalScoreInPeriodKey(_amm, _weekId)
432+ let traderScoreInPeriodKey = getTraderScoreInPeriodKey(_amm, _trader, _weekId)
433+ let lastNotionalKey = getLastNotionalKey(_amm, _trader)
434+ let traderAverageNotionalInPeriodKey = getTraderAverageNotionalInPeriodKey(_amm, _trader, _weekId)
435+[IntegerEntry(totalScoreInPeriodKey, _totalScore), IntegerEntry(traderScoreInPeriodKey, _traderScore), IntegerEntry(lastNotionalKey, _traderLastNotional), IntegerEntry(traderAverageNotionalInPeriodKey, _traderAverage)]
362436 }
363437
364438
366440 func initialize (_coordinator,_oracle) = if (initialized())
367441 then throw("Already initialized")
368442 else [StringEntry(k_coordinatorAddress, _coordinator), StringEntry(k_oracleAddress, _oracle), BooleanEntry(k_initialized, true)]
443+
444+
445+
446+@Callable(i)
447+func setOracleAddress (_oracle) = if (if (!(initialized()))
448+ then true
449+ else (i.caller != adminAddress()))
450+ then throw("Invalid setOracleAddress parameters")
451+ else [StringEntry(k_oracleAddress, _oracle)]
452+
453+
454+
455+@Callable(i)
456+func view_getPeriod () = {
457+ let weekId = getWeekId(TIME)
458+ let weekStart = getWeekStart(weekId)
459+ let weekEnd = getWeekEnd(weekId)
460+ throw(((((toString(weekStart) + ",") + toString(weekEnd)) + ",") + toString(TIME)))
461+ }
462+
463+
464+
465+@Callable(i)
466+func view_getMaxAmountOfAssetToDistribute (_amm,_assetId,_weekId) = throw(toString(getMaxAmountOfAssetToDistribute(_amm, _assetId, _weekId)))
369467
370468
371469
390488 @Callable(i)
391489 func claimAllRewards (_assetId,_periods) = {
392490 let trader = toString(i.caller)
393- let $t01235912442 = claimAllRewardForPeriodForTrader(trader, _assetId, _periods)
394- let amount = $t01235912442._1
395- let result = $t01235912442._2
396- if ((amount == 0))
397- then throw("Nothing to claim")
398- else result
491+ if (contains(_periods, toString(getWeekId(TIME))))
492+ then throw("Can not claim rewards for current week")
493+ else {
494+ let $t01543815521 = claimAllRewardForPeriodForTrader(trader, _assetId, _periods)
495+ let amount = $t01543815521._1
496+ let result = $t01543815521._2
497+ if ((amount == 0))
498+ then throw("Nothing to claim")
499+ else result
500+ }
399501 }
400502
401503
402504
403505 @Callable(i)
404-func attachRewards (_amm,_assetId,_maxAmountPerPeriod,_rewardRate) = if (if (!(initialized()))
506+func attachRewardAsset (_assetId,_maxAmountPerPeriod) = if (if (!(initialized()))
405507 then true
406508 else (i.caller != adminAddress()))
509+ then throw("Invalid attachRewardAsset params")
510+ else {
511+ let amms = getAmms()
512+ let rewardAssetIds = getRewardAssetIds()
513+ let newRewardAssetIds = if (containsElement(rewardAssetIds, _assetId))
514+ then rewardAssetIds
515+ else (rewardAssetIds :+ _assetId)
516+[IntegerEntry(getAssetMaxAmountPerPeriodKey(_assetId), _maxAmountPerPeriod), StringEntry(k_rewardAssetIds, listToStr(newRewardAssetIds))]
517+ }
518+
519+
520+
521+@Callable(i)
522+func attachRewards (_amm,_assetId,_rewardRate) = if (if (if (!(initialized()))
523+ then true
524+ else (i.caller != adminAddress()))
525+ then true
526+ else !(isWhitelist(_amm)))
407527 then throw("Invalid attachRewards params")
408528 else {
409529 let amms = getAmms()
414534 let newRewardAssetIds = if (containsElement(rewardAssetIds, _assetId))
415535 then rewardAssetIds
416536 else (rewardAssetIds :+ _assetId)
417-[IntegerEntry(getAmmRewardRateKey(_amm, _assetId), _rewardRate), IntegerEntry(getAmmMaxAmountPerPeriodKey(_amm, _assetId), _maxAmountPerPeriod), StringEntry(k_amms, listToStr(newAmms)), StringEntry(k_rewardAssetIds, listToStr(newRewardAssetIds))]
537+[IntegerEntry(getAmmRewardRateKey(_amm, _assetId), _rewardRate), StringEntry(k_amms, listToStr(newAmms)), StringEntry(k_rewardAssetIds, listToStr(newRewardAssetIds))]
418538 }
419539
420540
428548 then throw("Invalid notifyFees params")
429549 else {
430550 let actualFee = usdnToDecimals(_fee)
431- let ts = lastBlock.timestamp
432- let weekId = getWeekId(ts)
433- let feesInPeriod = getFeesInPeriod(amm, _trader, weekId)
434- let totalFeesInPeriod = getTotalFeesInPeriod(amm, weekId)
435- let newFeesInPeriod = (feesInPeriod + actualFee)
436- let newTotalFeesInPeriod = (totalFeesInPeriod + actualFee)
437- updateAmmFees(amm, _trader, weekId, newFeesInPeriod, newTotalFeesInPeriod)
551+ let weekId = getWeekId(TIME)
552+ updateAmmFees(amm, _trader, weekId, actualFee)
438553 }
439554 }
440555
449564 then throw("Invalid notifyNotional params")
450565 else {
451566 let actualNotional = usdnToDecimals(_notional)
452- let ts = lastBlock.timestamp
453567 let lastNotional = getLastNotional(amm, _trader)
454- let weekId = getWeekId(ts)
568+ let weekId = getWeekId(TIME)
455569 let weekStart = getWeekStart(weekId)
456570 let weekEnd = getWeekEnd(weekId)
457571 let t = (weekEnd - weekStart)
458572 let oldRunningAverage = getTraderAverageNotionalInPeriod(amm, _trader, weekId, lastNotional)
459- let w = divd((weekEnd - ts), t)
573+ let w = divd((weekEnd - TIME), t)
460574 let oldTraderAmmScore = getTraderScoreInPeriod(amm, _trader, weekId)
461575 let oldTotalAmmScore = getTotalScoreInPeriod(amm, weekId)
462576 let fees = getFeesInPeriod(amm, _trader, weekId)
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let k_totalFeesInPeriod = "k_totalFeesInPeriod"
5+
6+let k_totalAssetFeesInPeriod = "k_totalAssetFeesInPeriod"
57
68 let k_traderFeesInPeriod = "k_traderFeesInPeriod"
79
810 let k_lastNotional = "k_lastNotional"
911
1012 let k_totalScoreInPeriod = "k_totalScoreInPeriod"
1113
1214 let k_traderScoreInPeriod = "k_traderScoreInPeriod"
1315
1416 let k_traderAverageNotionalInPeriod = "k_traderAverageNotionalInPeriod"
1517
1618 let k_ammRewardRate = "k_ammRewardRate"
1719
18-let k_ammMaxAmountPerPeriod = "k_ammMaxAmountPerPeriod"
20+let k_assetMaxAmountPerPeriod = "k_assetMaxAmountPerPeriod"
1921
2022 let k_rewardAssetIds = "k_rewardAssetIds"
2123
2224 let k_amms = "k_amms"
2325
2426 let k_claimedAssetAndPeriod = "k_claimedAssetAndPeriod"
2527
2628 let k_oracleAddress = "k_oracleAddress"
2729
2830 let k_oraclePrice = "price"
2931
3032 let k_totalClaimedForTraderAndAsset = "k_totalClaimedForTraderAndAsset"
33+
34+let k_ammMaxAmountPerPeriod = "k_ammMaxAmountPerPeriod"
3135
3236 let k_initialized = "k_initialized"
3337
3438 let k_coordinatorAddress = "k_coordinatorAddress"
3539
3640 let k_governance_asset = "k_gov_asset"
3741
3842 let k_quote_asset = "k_quote_asset"
3943
4044 let k_admin_public_key = "k_admin_public_key"
4145
4246 let k_admin_address = "k_admin_address"
4347
4448 let k_amm = "k_amm"
4549
4650 let DECIMAL_UNIT = (1 * (((((((10 * 10) * 10) * 10) * 10) * 10) * 10) * 10))
4751
4852 let A = 70000000
4953
5054 let MS_IN_WEEK = (1000 * 604800)
5155
5256 func listToStr (_list) = {
5357 func _join (accumulator,val) = ((accumulator + val) + ",")
5458
5559 let newListStr = {
5660 let $l = _list
5761 let $s = size($l)
5862 let $acc0 = ""
5963 func $f0_1 ($a,$i) = if (($i >= $s))
6064 then $a
6165 else _join($a, $l[$i])
6266
6367 func $f0_2 ($a,$i) = if (($i >= $s))
6468 then $a
6569 else throw("List size exceeds 100")
6670
6771 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
6872 }
6973 let newListStrU = dropRight(newListStr, 1)
7074 let newListStrR = if ((take(newListStrU, 1) == ","))
7175 then drop(newListStrU, 1)
7276 else newListStrU
7377 newListStrR
7478 }
7579
7680
7781 func strToList (_str) = if ((_str == ""))
7882 then nil
7983 else split(_str, ",")
8084
8185
8286 func divd (_x,_y) = fraction(_x, DECIMAL_UNIT, _y, HALFEVEN)
8387
8488
8589 func muld (_x,_y) = fraction(_x, _y, DECIMAL_UNIT, HALFEVEN)
8690
8791
8892 func powd (_x,_y) = pow(_x, 8, _y, 8, 8, HALFEVEN)
8993
9094
9195 func abs (_x) = if ((_x > 0))
9296 then _x
9397 else -(_x)
9498
9599
96100 func minv (_x,_y) = if ((_x > _y))
97101 then _y
98102 else _x
99103
100104
101105 func toCompositeKey (_key,_address) = ((_key + "_") + _address)
102106
103107
104-func coordinator () = valueOrErrorMessage(addressFromString(getStringValue(this, k_coordinatorAddress)), "Coordinator not set")
108+let that = this
109+
110+func coordinator () = valueOrErrorMessage(addressFromString(getStringValue(that, k_coordinatorAddress)), "Coordinator not set")
105111
106112
107113 func adminPublicKey () = fromBase58String(getStringValue(coordinator(), k_admin_public_key))
108114
109115
110116 func adminAddress () = addressFromString(getStringValue(coordinator(), k_admin_address))
111117
112118
113119 func isWhitelist (_address) = valueOrElse(getBoolean(coordinator(), toCompositeKey(k_amm, _address)), false)
114120
115121
116-func int (k) = valueOrErrorMessage(getInteger(this, k), ("no value for " + k))
122+func int (k) = valueOrErrorMessage(getInteger(that, k), ("no value for " + k))
117123
118124
119-func int0 (k) = valueOrElse(getInteger(this, k), 0)
125+func int0 (k) = valueOrElse(getInteger(that, k), 0)
120126
121127
122-func initialized () = valueOrElse(getBoolean(this, k_initialized), false)
128+func initialized () = valueOrElse(getBoolean(that, k_initialized), false)
123129
124130
125-func getFeesInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderFeesInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
131+func getRewardAssetIds () = strToList(valueOrElse(getString(that, k_rewardAssetIds), ""))
126132
127133
128-func getTotalFeesInPeriodKey (_amm,_weekId) = ((((k_totalFeesInPeriod + "_") + _amm) + "_") + toString(_weekId))
134+let TIME = lastBlock.timestamp
135+
136+func getTraderFeesInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderFeesInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
137+
138+
139+func getTotalAmmFeesInPeriodKey (_amm,_weekId) = ((((k_totalFeesInPeriod + "_") + _amm) + "_") + toString(_weekId))
140+
141+
142+func getTotalAssetFeesInPeriodKey (_assetId,_weekId) = ((((k_totalAssetFeesInPeriod + "_") + _assetId) + "_") + toString(_weekId))
129143
130144
131145 func getLastNotionalKey (_amm,_trader) = ((((k_lastNotional + "_") + _amm) + "_") + _trader)
132146
133147
134148 func getTraderScoreInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderScoreInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
135149
136150
137151 func getTotalScoreInPeriodKey (_amm,_weekId) = ((((k_totalScoreInPeriod + "_") + _amm) + "_") + toString(_weekId))
138152
139153
140154 func getTraderAverageNotionalInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderAverageNotionalInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
141155
142156
143157 func getAmmRewardRateKey (_amm,_assetId) = ((((k_ammRewardRate + "_") + _amm) + "_") + _assetId)
144158
145159
146-func getAmmMaxAmountPerPeriodKey (_amm,_assetId) = ((((k_ammMaxAmountPerPeriod + "_") + _amm) + "_") + _assetId)
160+func getAssetMaxAmountPerPeriodKey (_assetId) = ((k_assetMaxAmountPerPeriod + "_") + _assetId)
147161
148162
149163 func getClaimedTraderAssetPeriodKey (_trader,_assetId,_period) = ((((((k_claimedAssetAndPeriod + "_") + _trader) + "_") + _assetId) + "_") + toString(_period))
150164
151165
152166 func getOraclePriceKey (_assetId,_period) = ((((k_oraclePrice + "_") + toString(_period)) + "_") + _assetId)
153167
154168
155169 func getTotalClaimedForTraderAndAssetKey (_trader,_assetId) = ((((k_totalClaimedForTraderAndAsset + "_") + _trader) + "_") + _assetId)
156170
157171
158-func updateAmmFees (_amm,_trader,_weekId,_traderFees,_totalFees) = {
159- let traderFeesKey = getFeesInPeriodKey(_amm, _trader, _weekId)
160- let totalFeesKey = getTotalFeesInPeriodKey(_amm, _weekId)
161-[IntegerEntry(traderFeesKey, _traderFees), IntegerEntry(totalFeesKey, _totalFees)]
162- }
163-
164-
165-func updateAmmData (_amm,_trader,_weekId,_traderAverage,_traderLastNotional,_traderScore,_totalScore) = {
166- let totalScoreInPeriodKey = getTotalScoreInPeriodKey(_amm, _weekId)
167- let traderScoreInPeriodKey = getTraderScoreInPeriodKey(_amm, _trader, _weekId)
168- let lastNotionalKey = getLastNotionalKey(_amm, _trader)
169- let traderAverageNotionalInPeriodKey = getTraderAverageNotionalInPeriodKey(_amm, _trader, _weekId)
170-[IntegerEntry(totalScoreInPeriodKey, _totalScore), IntegerEntry(traderScoreInPeriodKey, _traderScore), IntegerEntry(lastNotionalKey, _traderLastNotional), IntegerEntry(traderAverageNotionalInPeriodKey, _traderAverage)]
171- }
172+func getAmmMaxAmountPerPeriodKey (_amm,_assetId) = ((((k_ammMaxAmountPerPeriod + "_") + _amm) + "_") + _assetId)
172173
173174
174175 func usdnToDecimals (_amount) = (_amount * 100)
175176
176177
177178 func getWeekStart (_weekId) = (_weekId * MS_IN_WEEK)
178179
179180
180181 func getWeekEnd (_weekId) = ((_weekId + 1) * MS_IN_WEEK)
181182
182183
183184 func getWeekId (_ts) = (_ts / MS_IN_WEEK)
184185
185186
186187 func getFeesInPeriod (_amm,_trader,_weekId) = {
187- let key = getFeesInPeriodKey(_amm, _trader, _weekId)
188- valueOrElse(getInteger(this, key), 0)
188+ let key = getTraderFeesInPeriodKey(_amm, _trader, _weekId)
189+ int0(key)
189190 }
190191
191192
192193 func getTotalFeesInPeriod (_amm,_weekId) = {
193- let key = getTotalFeesInPeriodKey(_amm, _weekId)
194- valueOrElse(getInteger(this, key), 0)
194+ let key = getTotalAmmFeesInPeriodKey(_amm, _weekId)
195+ int0(key)
196+ }
197+
198+
199+func getTotalAssetFeesInPeriod (_assetId,_weekId) = {
200+ let key = getTotalAssetFeesInPeriodKey(_assetId, _weekId)
201+ int0(key)
195202 }
196203
197204
198205 func getLastNotional (_amm,_trader) = {
199206 let key = getLastNotionalKey(_amm, _trader)
200- valueOrElse(getInteger(this, key), 0)
207+ int0(key)
201208 }
202209
203210
204211 func getTraderScoreInPeriod (_amm,_trader,_weekId) = {
205212 let key = getTraderScoreInPeriodKey(_amm, _trader, _weekId)
206- valueOrElse(getInteger(this, key), 0)
213+ int0(key)
207214 }
208215
209216
210217 func getTotalScoreInPeriod (_amm,_weekId) = {
211218 let key = getTotalScoreInPeriodKey(_amm, _weekId)
212- valueOrElse(getInteger(this, key), 0)
219+ int0(key)
213220 }
214221
215222
216-func getAmms () = strToList(valueOrElse(getString(this, k_amms), ""))
217-
218-
219-func getRewardAssetIds () = strToList(valueOrElse(getString(this, k_rewardAssetIds), ""))
223+func getAmms () = strToList(valueOrElse(getString(that, k_amms), ""))
220224
221225
222226 func getAmmRewardRate (_amm,_assetId) = {
223227 let key = getAmmRewardRateKey(_amm, _assetId)
224- valueOrElse(getInteger(this, key), 0)
228+ int0(key)
225229 }
226230
227231
228-func getAmmMaxAmountPerPeriod (_amm,_assetId) = {
229- let key = getAmmMaxAmountPerPeriodKey(_amm, _assetId)
230- valueOrElse(getInteger(this, key), 0)
232+func getAssetMaxAmountPerPeriod (_assetId) = {
233+ let key = getAssetMaxAmountPerPeriodKey(_assetId)
234+ int0(key)
231235 }
232236
233237
234238 func getTotalClaimedForTraderAndAsset (_trader,_assetId) = {
235239 let key = getTotalClaimedForTraderAndAssetKey(_trader, _assetId)
236- valueOrElse(getInteger(this, key), 0)
240+ int0(key)
237241 }
238242
239243
240244 func getRewardAssetPrice (_assetId,_weekId) = {
241- let oracleAddressStr = valueOrErrorMessage(getString(this, k_oracleAddress), "Oracle not set")
245+ let oracleAddressStr = valueOrErrorMessage(getString(that, k_oracleAddress), "Oracle not set")
242246 let oracleAddress = valueOrErrorMessage(addressFromString(oracleAddressStr), "Invalid oracle address")
243247 let priceKey = getOraclePriceKey(_assetId, _weekId)
244248 usdnToDecimals(valueOrErrorMessage(getInteger(oracleAddress, priceKey), ((("No oracle price for asset " + _assetId) + " period ") + toString(_weekId))))
249+ }
250+
251+
252+func getMaxAmountOfAssetToDistribute (_amm,_assetId,_weekId) = {
253+ let totalFeesInPeriod = getTotalFeesInPeriod(_amm, _weekId)
254+ let totalAssetFeesInPeriod = getTotalAssetFeesInPeriod(_assetId, _weekId)
255+ if ((totalAssetFeesInPeriod == 0))
256+ then {
257+ let key = getAmmMaxAmountPerPeriodKey(_amm, _assetId)
258+ int0(key)
259+ }
260+ else {
261+ let maxAmountOfAsset = getAssetMaxAmountPerPeriod(_assetId)
262+ let maxAmountOfAssetToDistribute = muld(divd(totalFeesInPeriod, totalAssetFeesInPeriod), maxAmountOfAsset)
263+ maxAmountOfAssetToDistribute
264+ }
245265 }
246266
247267
248268 func rewardForTraderInAssetInPeriod (_amm,_trader,_assetId,_weekId) = {
249269 let rewardAssetRate = getAmmRewardRate(_amm, _assetId)
250270 if ((rewardAssetRate == 0))
251271 then 0
252272 else {
253273 let totalFeesInPeriod = getTotalFeesInPeriod(_amm, _weekId)
254274 let rewardAssetPrice = getRewardAssetPrice(_assetId, _weekId)
255- let maxAmountOfAssetToDistribute = getAmmMaxAmountPerPeriod(_amm, _assetId)
275+ let maxAmountOfAssetToDistribute = getMaxAmountOfAssetToDistribute(_amm, _assetId, _weekId)
256276 let traderScore = getTraderScoreInPeriod(_amm, _trader, _weekId)
257277 let totalTraderScore = getTotalScoreInPeriod(_amm, _weekId)
258278 if ((totalTraderScore == 0))
259279 then 0
260280 else {
261- let totalAssetToDistribute = muld(divd(totalFeesInPeriod, rewardAssetPrice), rewardAssetRate)
281+ let totalAssetToDistribute = divd(muld(totalFeesInPeriod, rewardAssetRate), rewardAssetPrice)
262282 let actualTotalAssetToDistribute = minv(totalAssetToDistribute, maxAmountOfAssetToDistribute)
263283 let traderShareInRewards = divd(muld(actualTotalAssetToDistribute, traderScore), totalTraderScore)
264284 traderShareInRewards
265285 }
266286 }
267287 }
268288
269289
270290 func totalRewardForTraderInAssetInPeriod (_trader,_assetId,_weekId) = {
271291 let amms = getAmms()
272292 func compute (_acc,_amm) = (_acc + rewardForTraderInAssetInPeriod(_amm, _trader, _assetId, _weekId))
273293
274294 let $l = amms
275295 let $s = size($l)
276296 let $acc0 = 0
277297 func $f0_1 ($a,$i) = if (($i >= $s))
278298 then $a
279299 else compute($a, $l[$i])
280300
281301 func $f0_2 ($a,$i) = if (($i >= $s))
282302 then $a
283303 else throw("List size exceeds 20")
284304
285305 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20)
286306 }
287307
288308
289309 func totalRewardForTraderInAsset (_trader,_assetId,_periods) = {
290310 func compute (_acc,_weekId) = (_acc + totalRewardForTraderInAssetInPeriod(_trader, _assetId, _weekId))
291311
292312 let $l = _periods
293313 let $s = size($l)
294314 let $acc0 = 0
295315 func $f0_1 ($a,$i) = if (($i >= $s))
296316 then $a
297317 else compute($a, $l[$i])
298318
299319 func $f0_2 ($a,$i) = if (($i >= $s))
300320 then $a
301- else throw("List size exceeds 18")
321+ else throw("List size exceeds 12")
302322
303- $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18)
323+ $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
304324 }
305325
306326
307327 func getTraderAverageNotionalInPeriod (_amm,_trader,_weekId,_defaultValue) = {
308328 let key = getTraderAverageNotionalInPeriodKey(_amm, _trader, _weekId)
309- valueOrElse(getInteger(this, key), _defaultValue)
329+ valueOrElse(getInteger(that, key), _defaultValue)
310330 }
311331
312332
313333 func isClaimed (_trader,_assetId,_period) = {
314334 let key = getClaimedTraderAssetPeriodKey(_trader, _assetId, _period)
315- valueOrElse(getBoolean(this, key), false)
335+ valueOrElse(getBoolean(that, key), false)
336+ }
337+
338+
339+func adjust (_amount,_assetId) = {
340+ let asset = valueOrErrorMessage(assetInfo(fromBase58String(_assetId)), "Invalid asset id")
341+ let decimals = asset.decimals
342+ if ((decimals == 6))
343+ then (_amount / 100)
344+ else _amount
316345 }
317346
318347
319348 func claimAllRewardForPeriodForTrader (_trader,_assetId,_periods) = {
320349 let periods = strToList(_periods)
321350 func checkAndFilterFn (_acc,_next) = {
322351 let period = valueOrErrorMessage(parseInt(_next), ("Invalid period: " + _next))
323352 if (if (containsElement(_acc, period))
324353 then true
325354 else isClaimed(_trader, _assetId, period))
326355 then _acc
327356 else (_acc :+ period)
328357 }
329358
330359 let validPeriods = {
331360 let $l = periods
332361 let $s = size($l)
333362 let $acc0 = nil
334363 func $f0_1 ($a,$i) = if (($i >= $s))
335364 then $a
336365 else checkAndFilterFn($a, $l[$i])
337366
338367 func $f0_2 ($a,$i) = if (($i >= $s))
339368 then $a
340- else throw("List size exceeds 18")
369+ else throw("List size exceeds 12")
341370
342- $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18)
371+ $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
343372 }
344373 func markPeriodAsDoneFn (_acc,_period) = (_acc :+ BooleanEntry(getClaimedTraderAssetPeriodKey(_trader, _assetId, _period), true))
345374
346375 let markPeriodsAsClaimed = {
347376 let $l = validPeriods
348377 let $s = size($l)
349378 let $acc0 = nil
350379 func $f1_1 ($a,$i) = if (($i >= $s))
351380 then $a
352381 else markPeriodAsDoneFn($a, $l[$i])
353382
354383 func $f1_2 ($a,$i) = if (($i >= $s))
355384 then $a
356- else throw("List size exceeds 18")
385+ else throw("List size exceeds 16")
357386
358- $f1_2($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18)
387+ $f1_2($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16)
359388 }
360389 let amount = totalRewardForTraderInAsset(_trader, _assetId, validPeriods)
361- $Tuple2(amount, ([ScriptTransfer(addressFromStringValue(_trader), amount, fromBase58String(_assetId)), IntegerEntry(getTotalClaimedForTraderAndAssetKey(_trader, _assetId), (getTotalClaimedForTraderAndAsset(_trader, _assetId) + amount))] ++ markPeriodsAsClaimed))
390+ $Tuple2(amount, ([ScriptTransfer(addressFromStringValue(_trader), adjust(amount, _assetId), fromBase58String(_assetId)), IntegerEntry(getTotalClaimedForTraderAndAssetKey(_trader, _assetId), (getTotalClaimedForTraderAndAsset(_trader, _assetId) + amount))] ++ markPeriodsAsClaimed))
391+ }
392+
393+
394+func updateAmmFees (_amm,_trader,_weekId,_actualFee) = {
395+ let traderFeesKey = getTraderFeesInPeriodKey(_amm, _trader, _weekId)
396+ let totalFeesKey = getTotalAmmFeesInPeriodKey(_amm, _weekId)
397+ let rewardAssets = getRewardAssetIds()
398+ let feesInPeriod = getFeesInPeriod(_amm, _trader, _weekId)
399+ let totalFeesInPeriod = getTotalFeesInPeriod(_amm, _weekId)
400+ let newFeesInPeriod = (feesInPeriod + _actualFee)
401+ let newTotalFeesInPeriod = (totalFeesInPeriod + _actualFee)
402+ func doHandleRewardAssetId (_acc,_assetId) = {
403+ let key = getAmmRewardRateKey(_amm, _assetId)
404+ if ((int0(key) > 0))
405+ then {
406+ let totalAssetFeesKey = getTotalAssetFeesInPeriodKey(_assetId, _weekId)
407+ (_acc :+ IntegerEntry(totalAssetFeesKey, (getTotalAssetFeesInPeriod(_assetId, _weekId) + _actualFee)))
408+ }
409+ else _acc
410+ }
411+
412+ let updateAssetFees = {
413+ let $l = rewardAssets
414+ let $s = size($l)
415+ let $acc0 = nil
416+ func $f0_1 ($a,$i) = if (($i >= $s))
417+ then $a
418+ else doHandleRewardAssetId($a, $l[$i])
419+
420+ func $f0_2 ($a,$i) = if (($i >= $s))
421+ then $a
422+ else throw("List size exceeds 10")
423+
424+ $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10)
425+ }
426+ ([IntegerEntry(traderFeesKey, newFeesInPeriod), IntegerEntry(totalFeesKey, newTotalFeesInPeriod)] ++ updateAssetFees)
427+ }
428+
429+
430+func updateAmmData (_amm,_trader,_weekId,_traderAverage,_traderLastNotional,_traderScore,_totalScore) = {
431+ let totalScoreInPeriodKey = getTotalScoreInPeriodKey(_amm, _weekId)
432+ let traderScoreInPeriodKey = getTraderScoreInPeriodKey(_amm, _trader, _weekId)
433+ let lastNotionalKey = getLastNotionalKey(_amm, _trader)
434+ let traderAverageNotionalInPeriodKey = getTraderAverageNotionalInPeriodKey(_amm, _trader, _weekId)
435+[IntegerEntry(totalScoreInPeriodKey, _totalScore), IntegerEntry(traderScoreInPeriodKey, _traderScore), IntegerEntry(lastNotionalKey, _traderLastNotional), IntegerEntry(traderAverageNotionalInPeriodKey, _traderAverage)]
362436 }
363437
364438
365439 @Callable(i)
366440 func initialize (_coordinator,_oracle) = if (initialized())
367441 then throw("Already initialized")
368442 else [StringEntry(k_coordinatorAddress, _coordinator), StringEntry(k_oracleAddress, _oracle), BooleanEntry(k_initialized, true)]
443+
444+
445+
446+@Callable(i)
447+func setOracleAddress (_oracle) = if (if (!(initialized()))
448+ then true
449+ else (i.caller != adminAddress()))
450+ then throw("Invalid setOracleAddress parameters")
451+ else [StringEntry(k_oracleAddress, _oracle)]
452+
453+
454+
455+@Callable(i)
456+func view_getPeriod () = {
457+ let weekId = getWeekId(TIME)
458+ let weekStart = getWeekStart(weekId)
459+ let weekEnd = getWeekEnd(weekId)
460+ throw(((((toString(weekStart) + ",") + toString(weekEnd)) + ",") + toString(TIME)))
461+ }
462+
463+
464+
465+@Callable(i)
466+func view_getMaxAmountOfAssetToDistribute (_amm,_assetId,_weekId) = throw(toString(getMaxAmountOfAssetToDistribute(_amm, _assetId, _weekId)))
369467
370468
371469
372470 @Callable(i)
373471 func view_claimRewards (_trader,_assetId,_period) = throw(toString(totalRewardForTraderInAsset(_trader, _assetId, [_period])))
374472
375473
376474
377475 @Callable(i)
378476 func view_calcReward (_trader,_amm,_assetId,_period) = throw(toString(rewardForTraderInAssetInPeriod(_amm, _assetId, _trader, _period)))
379477
380478
381479
382480 @Callable(i)
383481 func view_claimAllRewards (_trader,_assetId,_periods) = {
384482 let result = claimAllRewardForPeriodForTrader(_trader, _assetId, _periods)
385483 throw(((toString(result._1) + ",") + toString(getTotalClaimedForTraderAndAsset(_trader, _assetId))))
386484 }
387485
388486
389487
390488 @Callable(i)
391489 func claimAllRewards (_assetId,_periods) = {
392490 let trader = toString(i.caller)
393- let $t01235912442 = claimAllRewardForPeriodForTrader(trader, _assetId, _periods)
394- let amount = $t01235912442._1
395- let result = $t01235912442._2
396- if ((amount == 0))
397- then throw("Nothing to claim")
398- else result
491+ if (contains(_periods, toString(getWeekId(TIME))))
492+ then throw("Can not claim rewards for current week")
493+ else {
494+ let $t01543815521 = claimAllRewardForPeriodForTrader(trader, _assetId, _periods)
495+ let amount = $t01543815521._1
496+ let result = $t01543815521._2
497+ if ((amount == 0))
498+ then throw("Nothing to claim")
499+ else result
500+ }
399501 }
400502
401503
402504
403505 @Callable(i)
404-func attachRewards (_amm,_assetId,_maxAmountPerPeriod,_rewardRate) = if (if (!(initialized()))
506+func attachRewardAsset (_assetId,_maxAmountPerPeriod) = if (if (!(initialized()))
405507 then true
406508 else (i.caller != adminAddress()))
509+ then throw("Invalid attachRewardAsset params")
510+ else {
511+ let amms = getAmms()
512+ let rewardAssetIds = getRewardAssetIds()
513+ let newRewardAssetIds = if (containsElement(rewardAssetIds, _assetId))
514+ then rewardAssetIds
515+ else (rewardAssetIds :+ _assetId)
516+[IntegerEntry(getAssetMaxAmountPerPeriodKey(_assetId), _maxAmountPerPeriod), StringEntry(k_rewardAssetIds, listToStr(newRewardAssetIds))]
517+ }
518+
519+
520+
521+@Callable(i)
522+func attachRewards (_amm,_assetId,_rewardRate) = if (if (if (!(initialized()))
523+ then true
524+ else (i.caller != adminAddress()))
525+ then true
526+ else !(isWhitelist(_amm)))
407527 then throw("Invalid attachRewards params")
408528 else {
409529 let amms = getAmms()
410530 let rewardAssetIds = getRewardAssetIds()
411531 let newAmms = if (containsElement(amms, _amm))
412532 then amms
413533 else (amms :+ _amm)
414534 let newRewardAssetIds = if (containsElement(rewardAssetIds, _assetId))
415535 then rewardAssetIds
416536 else (rewardAssetIds :+ _assetId)
417-[IntegerEntry(getAmmRewardRateKey(_amm, _assetId), _rewardRate), IntegerEntry(getAmmMaxAmountPerPeriodKey(_amm, _assetId), _maxAmountPerPeriod), StringEntry(k_amms, listToStr(newAmms)), StringEntry(k_rewardAssetIds, listToStr(newRewardAssetIds))]
537+[IntegerEntry(getAmmRewardRateKey(_amm, _assetId), _rewardRate), StringEntry(k_amms, listToStr(newAmms)), StringEntry(k_rewardAssetIds, listToStr(newRewardAssetIds))]
418538 }
419539
420540
421541
422542 @Callable(i)
423543 func notifyFees (_trader,_fee) = {
424544 let amm = toString(i.caller)
425545 if (if (!(isWhitelist(amm)))
426546 then true
427547 else !(initialized()))
428548 then throw("Invalid notifyFees params")
429549 else {
430550 let actualFee = usdnToDecimals(_fee)
431- let ts = lastBlock.timestamp
432- let weekId = getWeekId(ts)
433- let feesInPeriod = getFeesInPeriod(amm, _trader, weekId)
434- let totalFeesInPeriod = getTotalFeesInPeriod(amm, weekId)
435- let newFeesInPeriod = (feesInPeriod + actualFee)
436- let newTotalFeesInPeriod = (totalFeesInPeriod + actualFee)
437- updateAmmFees(amm, _trader, weekId, newFeesInPeriod, newTotalFeesInPeriod)
551+ let weekId = getWeekId(TIME)
552+ updateAmmFees(amm, _trader, weekId, actualFee)
438553 }
439554 }
440555
441556
442557
443558 @Callable(i)
444559 func notifyNotional (_trader,_notional) = {
445560 let amm = toString(i.caller)
446561 if (if (!(isWhitelist(amm)))
447562 then true
448563 else !(initialized()))
449564 then throw("Invalid notifyNotional params")
450565 else {
451566 let actualNotional = usdnToDecimals(_notional)
452- let ts = lastBlock.timestamp
453567 let lastNotional = getLastNotional(amm, _trader)
454- let weekId = getWeekId(ts)
568+ let weekId = getWeekId(TIME)
455569 let weekStart = getWeekStart(weekId)
456570 let weekEnd = getWeekEnd(weekId)
457571 let t = (weekEnd - weekStart)
458572 let oldRunningAverage = getTraderAverageNotionalInPeriod(amm, _trader, weekId, lastNotional)
459- let w = divd((weekEnd - ts), t)
573+ let w = divd((weekEnd - TIME), t)
460574 let oldTraderAmmScore = getTraderScoreInPeriod(amm, _trader, weekId)
461575 let oldTotalAmmScore = getTotalScoreInPeriod(amm, weekId)
462576 let fees = getFeesInPeriod(amm, _trader, weekId)
463577 let newRunningAverage = ((oldRunningAverage + muld(actualNotional, w)) - muld(lastNotional, w))
464578 let newTraderAmmScore = muld(powd(fees, A), powd(newRunningAverage, (DECIMAL_UNIT - A)))
465579 let newTotalAmmScoreInPeriod = ((oldTotalAmmScore - oldTraderAmmScore) + newTraderAmmScore)
466580 updateAmmData(amm, _trader, weekId, newRunningAverage, actualNotional, newTraderAmmScore, newTotalAmmScoreInPeriod)
467581 }
468582 }
469583
470584
471585 @Verifier(tx)
472586 func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], adminPublicKey())
473587

github/deemru/w8io/873ac7e 
100.19 ms