tx · Ekyk79jEgCDnwsthiTDu6xBrMNKW2iayZuLy8teSScRU 3N4oVbx9R7qb1SDJE2ZkpqU3oTD28jrnhCF: -0.03700000 Waves 2022.10.20 20:26 [2281029] smart account 3N4oVbx9R7qb1SDJE2ZkpqU3oTD28jrnhCF > SELF 0.00000000 Waves
{ "type": 13, "id": "Ekyk79jEgCDnwsthiTDu6xBrMNKW2iayZuLy8teSScRU", "fee": 3700000, "feeAssetId": null, "timestamp": 1666286781862, "version": 2, "chainId": 84, "sender": "3N4oVbx9R7qb1SDJE2ZkpqU3oTD28jrnhCF", "senderPublicKey": "3XTERkpD7YNa8N8GJwwqD76LZ7vjarkv8pys4mbsPTKg", "proofs": [ "2awwhppvocM4WitnjhVJAe1YMFR3pxDiCTuDC7D5qpuyyYeDzjc48PgzithKL8zYfHLR1QdHofo1HxwYgTJWh2y3" ], "script": "base64:BgJLCAISBAoCCAgSAwoBCBIAEgUKAwgIARIFCgMICAESBgoECAgIARIFCgMICAgSBAoCCAgSBAoCCAESBQoDCAgBEgQKAggBEgQKAggBUgATa190b3RhbEZlZXNJblBlcmlvZAITa190b3RhbEZlZXNJblBlcmlvZAAYa190b3RhbEFzc2V0RmVlc0luUGVyaW9kAhhrX3RvdGFsQXNzZXRGZWVzSW5QZXJpb2QAFGtfdHJhZGVyRmVlc0luUGVyaW9kAhRrX3RyYWRlckZlZXNJblBlcmlvZAAOa19sYXN0Tm90aW9uYWwCDmtfbGFzdE5vdGlvbmFsABRrX3RvdGFsU2NvcmVJblBlcmlvZAIUa190b3RhbFNjb3JlSW5QZXJpb2QAFWtfdHJhZGVyU2NvcmVJblBlcmlvZAIVa190cmFkZXJTY29yZUluUGVyaW9kAB9rX3RyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kAh9rX3RyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kAA9rX2FtbVJld2FyZFJhdGUCD2tfYW1tUmV3YXJkUmF0ZQAZa19hc3NldE1heEFtb3VudFBlclBlcmlvZAIZa19hc3NldE1heEFtb3VudFBlclBlcmlvZAAQa19yZXdhcmRBc3NldElkcwIQa19yZXdhcmRBc3NldElkcwAGa19hbW1zAgZrX2FtbXMAF2tfY2xhaW1lZEFzc2V0QW5kUGVyaW9kAhdrX2NsYWltZWRBc3NldEFuZFBlcmlvZAAPa19vcmFjbGVBZGRyZXNzAg9rX29yYWNsZUFkZHJlc3MADWtfb3JhY2xlUHJpY2UCBXByaWNlAB9rX3RvdGFsQ2xhaW1lZEZvclRyYWRlckFuZEFzc2V0Ah9rX3RvdGFsQ2xhaW1lZEZvclRyYWRlckFuZEFzc2V0ABdrX2FtbU1heEFtb3VudFBlclBlcmlvZAIXa19hbW1NYXhBbW91bnRQZXJQZXJpb2QADWtfaW5pdGlhbGl6ZWQCDWtfaW5pdGlhbGl6ZWQAFGtfY29vcmRpbmF0b3JBZGRyZXNzAhRrX2Nvb3JkaW5hdG9yQWRkcmVzcwASa19nb3Zlcm5hbmNlX2Fzc2V0AgtrX2dvdl9hc3NldAANa19xdW90ZV9hc3NldAINa19xdW90ZV9hc3NldAASa19hZG1pbl9wdWJsaWNfa2V5AhJrX2FkbWluX3B1YmxpY19rZXkAD2tfYWRtaW5fYWRkcmVzcwIPa19hZG1pbl9hZGRyZXNzAAVrX2FtbQIFa19hbW0ADERFQ0lNQUxfVU5JVAkAaAIAAQkAaAIJAGgCCQBoAgkAaAIJAGgCCQBoAgkAaAIACgAKAAoACgAKAAoACgAKAAFBAIC7sCEACk1TX0lOX1dFRUsJAGgCAOgHAID1JAEJbGlzdFRvU3RyAQVfbGlzdAoBBV9qb2luAgthY2N1bXVsYXRvcgN2YWwJAKwCAgkArAICBQthY2N1bXVsYXRvcgUDdmFsAgEsBApuZXdMaXN0U3RyCgACJGwFBV9saXN0CgACJHMJAJADAQUCJGwKAAUkYWNjMAIACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQVfam9pbgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIVTGlzdCBzaXplIGV4Y2VlZHMgMTAwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAAVABYAFwAYABkAGgAbABwAHQAeAB8AIAAhACIAIwAkACUAJgAnACgAKQAqACsALAAtAC4ALwAwADEAMgAzADQANQA2ADcAOAA5ADoAOwA8AD0APgA/AEAAQQBCAEMARABFAEYARwBIAEkASgBLAEwATQBOAE8AUABRAFIAUwBUAFUAVgBXAFgAWQBaAFsAXABdAF4AXwBgAGEAYgBjAGQEC25ld0xpc3RTdHJVCQCzAgIFCm5ld0xpc3RTdHIAAQQLbmV3TGlzdFN0clIDCQAAAgkArwICBQtuZXdMaXN0U3RyVQABAgEsCQCwAgIFC25ld0xpc3RTdHJVAAEFC25ld0xpc3RTdHJVBQtuZXdMaXN0U3RyUgEJc3RyVG9MaXN0AQRfc3RyAwkAAAIFBF9zdHICAAUDbmlsCQC1CQIFBF9zdHICASwBBGRpdmQCAl94Al95CQBuBAUCX3gFDERFQ0lNQUxfVU5JVAUCX3kFCEhBTEZFVkVOAQRtdWxkAgJfeAJfeQkAbgQFAl94BQJfeQUMREVDSU1BTF9VTklUBQhIQUxGRVZFTgEEcG93ZAICX3gCX3kJAGwGBQJfeAAIBQJfeQAIAAgFCEhBTEZFVkVOAQNhYnMBAl94AwkAZgIFAl94AAAFAl94CQEBLQEFAl94AQRtaW52AgJfeAJfeQMJAGYCBQJfeAUCX3kFAl95BQJfeAEOdG9Db21wb3NpdGVLZXkCBF9rZXkIX2FkZHJlc3MJAKwCAgkArAICBQRfa2V5AgFfBQhfYWRkcmVzcwAEdGhhdAUEdGhpcwELY29vcmRpbmF0b3IACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgUEdGhhdAUUa19jb29yZGluYXRvckFkZHJlc3MCE0Nvb3JkaW5hdG9yIG5vdCBzZXQBDmFkbWluUHVibGljS2V5AAkA2QQBCQERQGV4dHJOYXRpdmUoMTA1MykCCQELY29vcmRpbmF0b3IABRJrX2FkbWluX3B1YmxpY19rZXkBDGFkbWluQWRkcmVzcwAJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUPa19hZG1pbl9hZGRyZXNzAQtpc1doaXRlbGlzdAEIX2FkZHJlc3MJAQt2YWx1ZU9yRWxzZQIJAJsIAgkBC2Nvb3JkaW5hdG9yAAkBDnRvQ29tcG9zaXRlS2V5AgUFa19hbW0FCF9hZGRyZXNzBwEDaW50AQFrCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUEdGhhdAUBawkArAICAg1ubyB2YWx1ZSBmb3IgBQFrAQRpbnQwAQFrCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoYXQFAWsAAAELaW5pdGlhbGl6ZWQACQELdmFsdWVPckVsc2UCCQCbCAIFBHRoYXQFDWtfaW5pdGlhbGl6ZWQHARFnZXRSZXdhcmRBc3NldElkcwAJAQlzdHJUb0xpc3QBCQELdmFsdWVPckVsc2UCCQCdCAIFBHRoYXQFEGtfcmV3YXJkQXNzZXRJZHMCAAAEVElNRQgFCWxhc3RCbG9jawl0aW1lc3RhbXABGGdldFRyYWRlckZlZXNJblBlcmlvZEtleQMEX2FtbQdfdHJhZGVyB193ZWVrSWQJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIFFGtfdHJhZGVyRmVlc0luUGVyaW9kAgFfBQRfYW1tAgFfBQdfdHJhZGVyAgFfCQCkAwEFB193ZWVrSWQBGmdldFRvdGFsQW1tRmVlc0luUGVyaW9kS2V5AgRfYW1tB193ZWVrSWQJAKwCAgkArAICCQCsAgIJAKwCAgUTa190b3RhbEZlZXNJblBlcmlvZAIBXwUEX2FtbQIBXwkApAMBBQdfd2Vla0lkARxnZXRUb3RhbEFzc2V0RmVlc0luUGVyaW9kS2V5AghfYXNzZXRJZAdfd2Vla0lkCQCsAgIJAKwCAgkArAICCQCsAgIFGGtfdG90YWxBc3NldEZlZXNJblBlcmlvZAIBXwUIX2Fzc2V0SWQCAV8JAKQDAQUHX3dlZWtJZAESZ2V0TGFzdE5vdGlvbmFsS2V5AgRfYW1tB190cmFkZXIJAKwCAgkArAICCQCsAgIJAKwCAgUOa19sYXN0Tm90aW9uYWwCAV8FBF9hbW0CAV8FB190cmFkZXIBGWdldFRyYWRlclNjb3JlSW5QZXJpb2RLZXkDBF9hbW0HX3RyYWRlcgdfd2Vla0lkCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICBRVrX3RyYWRlclNjb3JlSW5QZXJpb2QCAV8FBF9hbW0CAV8FB190cmFkZXICAV8JAKQDAQUHX3dlZWtJZAEYZ2V0VG90YWxTY29yZUluUGVyaW9kS2V5AgRfYW1tB193ZWVrSWQJAKwCAgkArAICCQCsAgIJAKwCAgUUa190b3RhbFNjb3JlSW5QZXJpb2QCAV8FBF9hbW0CAV8JAKQDAQUHX3dlZWtJZAEjZ2V0VHJhZGVyQXZlcmFnZU5vdGlvbmFsSW5QZXJpb2RLZXkDBF9hbW0HX3RyYWRlcgdfd2Vla0lkCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICBR9rX3RyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kAgFfBQRfYW1tAgFfBQdfdHJhZGVyAgFfCQCkAwEFB193ZWVrSWQBE2dldEFtbVJld2FyZFJhdGVLZXkCBF9hbW0IX2Fzc2V0SWQJAKwCAgkArAICCQCsAgIJAKwCAgUPa19hbW1SZXdhcmRSYXRlAgFfBQRfYW1tAgFfBQhfYXNzZXRJZAEdZ2V0QXNzZXRNYXhBbW91bnRQZXJQZXJpb2RLZXkBCF9hc3NldElkCQCsAgIJAKwCAgUZa19hc3NldE1heEFtb3VudFBlclBlcmlvZAIBXwUIX2Fzc2V0SWQBHmdldENsYWltZWRUcmFkZXJBc3NldFBlcmlvZEtleQMHX3RyYWRlcghfYXNzZXRJZAdfcGVyaW9kCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICBRdrX2NsYWltZWRBc3NldEFuZFBlcmlvZAIBXwUHX3RyYWRlcgIBXwUIX2Fzc2V0SWQCAV8JAKQDAQUHX3BlcmlvZAERZ2V0T3JhY2xlUHJpY2VLZXkCCF9hc3NldElkB19wZXJpb2QJAKwCAgkArAICCQCsAgIJAKwCAgUNa19vcmFjbGVQcmljZQIBXwkApAMBBQdfcGVyaW9kAgFfBQhfYXNzZXRJZAEjZ2V0VG90YWxDbGFpbWVkRm9yVHJhZGVyQW5kQXNzZXRLZXkCB190cmFkZXIIX2Fzc2V0SWQJAKwCAgkArAICCQCsAgIJAKwCAgUfa190b3RhbENsYWltZWRGb3JUcmFkZXJBbmRBc3NldAIBXwUHX3RyYWRlcgIBXwUIX2Fzc2V0SWQBG2dldEFtbU1heEFtb3VudFBlclBlcmlvZEtleQIEX2FtbQhfYXNzZXRJZAkArAICCQCsAgIJAKwCAgkArAICBRdrX2FtbU1heEFtb3VudFBlclBlcmlvZAIBXwUEX2FtbQIBXwUIX2Fzc2V0SWQBDnVzZG5Ub0RlY2ltYWxzAQdfYW1vdW50CQBoAgUHX2Ftb3VudABkAQxnZXRXZWVrU3RhcnQBB193ZWVrSWQJAGgCBQdfd2Vla0lkBQpNU19JTl9XRUVLAQpnZXRXZWVrRW5kAQdfd2Vla0lkCQBoAgkAZAIFB193ZWVrSWQAAQUKTVNfSU5fV0VFSwEJZ2V0V2Vla0lkAQNfdHMJAGkCBQNfdHMFCk1TX0lOX1dFRUsBD2dldEZlZXNJblBlcmlvZAMEX2FtbQdfdHJhZGVyB193ZWVrSWQEA2tleQkBGGdldFRyYWRlckZlZXNJblBlcmlvZEtleQMFBF9hbW0FB190cmFkZXIFB193ZWVrSWQJAQRpbnQwAQUDa2V5ARRnZXRUb3RhbEZlZXNJblBlcmlvZAIEX2FtbQdfd2Vla0lkBANrZXkJARpnZXRUb3RhbEFtbUZlZXNJblBlcmlvZEtleQIFBF9hbW0FB193ZWVrSWQJAQRpbnQwAQUDa2V5ARlnZXRUb3RhbEFzc2V0RmVlc0luUGVyaW9kAghfYXNzZXRJZAdfd2Vla0lkBANrZXkJARxnZXRUb3RhbEFzc2V0RmVlc0luUGVyaW9kS2V5AgUIX2Fzc2V0SWQFB193ZWVrSWQJAQRpbnQwAQUDa2V5AQ9nZXRMYXN0Tm90aW9uYWwCBF9hbW0HX3RyYWRlcgQDa2V5CQESZ2V0TGFzdE5vdGlvbmFsS2V5AgUEX2FtbQUHX3RyYWRlcgkBBGludDABBQNrZXkBFmdldFRyYWRlclNjb3JlSW5QZXJpb2QDBF9hbW0HX3RyYWRlcgdfd2Vla0lkBANrZXkJARlnZXRUcmFkZXJTY29yZUluUGVyaW9kS2V5AwUEX2FtbQUHX3RyYWRlcgUHX3dlZWtJZAkBBGludDABBQNrZXkBFWdldFRvdGFsU2NvcmVJblBlcmlvZAIEX2FtbQdfd2Vla0lkBANrZXkJARhnZXRUb3RhbFNjb3JlSW5QZXJpb2RLZXkCBQRfYW1tBQdfd2Vla0lkCQEEaW50MAEFA2tleQEHZ2V0QW1tcwAJAQlzdHJUb0xpc3QBCQELdmFsdWVPckVsc2UCCQCdCAIFBHRoYXQFBmtfYW1tcwIAARBnZXRBbW1SZXdhcmRSYXRlAgRfYW1tCF9hc3NldElkBANrZXkJARNnZXRBbW1SZXdhcmRSYXRlS2V5AgUEX2FtbQUIX2Fzc2V0SWQJAQRpbnQwAQUDa2V5ARpnZXRBc3NldE1heEFtb3VudFBlclBlcmlvZAEIX2Fzc2V0SWQEA2tleQkBHWdldEFzc2V0TWF4QW1vdW50UGVyUGVyaW9kS2V5AQUIX2Fzc2V0SWQJAQRpbnQwAQUDa2V5ASBnZXRUb3RhbENsYWltZWRGb3JUcmFkZXJBbmRBc3NldAIHX3RyYWRlcghfYXNzZXRJZAQDa2V5CQEjZ2V0VG90YWxDbGFpbWVkRm9yVHJhZGVyQW5kQXNzZXRLZXkCBQdfdHJhZGVyBQhfYXNzZXRJZAkBBGludDABBQNrZXkBE2dldFJld2FyZEFzc2V0UHJpY2UCCF9hc3NldElkB193ZWVrSWQEEG9yYWNsZUFkZHJlc3NTdHIJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQR0aGF0BQ9rX29yYWNsZUFkZHJlc3MCDk9yYWNsZSBub3Qgc2V0BA1vcmFjbGVBZGRyZXNzCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQUQb3JhY2xlQWRkcmVzc1N0cgIWSW52YWxpZCBvcmFjbGUgYWRkcmVzcwQIcHJpY2VLZXkJARFnZXRPcmFjbGVQcmljZUtleQIFCF9hc3NldElkBQdfd2Vla0lkCQEOdXNkblRvRGVjaW1hbHMBCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUNb3JhY2xlQWRkcmVzcwUIcHJpY2VLZXkJAKwCAgkArAICCQCsAgICGk5vIG9yYWNsZSBwcmljZSBmb3IgYXNzZXQgBQhfYXNzZXRJZAIIIHBlcmlvZCAJAKQDAQUHX3dlZWtJZAEfZ2V0TWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQMEX2FtbQhfYXNzZXRJZAdfd2Vla0lkBBF0b3RhbEZlZXNJblBlcmlvZAkBFGdldFRvdGFsRmVlc0luUGVyaW9kAgUEX2FtbQUHX3dlZWtJZAQWdG90YWxBc3NldEZlZXNJblBlcmlvZAkBGWdldFRvdGFsQXNzZXRGZWVzSW5QZXJpb2QCBQhfYXNzZXRJZAUHX3dlZWtJZAMJAAACBRZ0b3RhbEFzc2V0RmVlc0luUGVyaW9kAAAEA2tleQkBG2dldEFtbU1heEFtb3VudFBlclBlcmlvZEtleQIFBF9hbW0FCF9hc3NldElkCQEEaW50MAEFA2tleQQQbWF4QW1vdW50T2ZBc3NldAkBGmdldEFzc2V0TWF4QW1vdW50UGVyUGVyaW9kAQUIX2Fzc2V0SWQEHG1heEFtb3VudE9mQXNzZXRUb0Rpc3RyaWJ1dGUJAQRtdWxkAgkBBGRpdmQCBRF0b3RhbEZlZXNJblBlcmlvZAUWdG90YWxBc3NldEZlZXNJblBlcmlvZAUQbWF4QW1vdW50T2ZBc3NldAUcbWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQEecmV3YXJkRm9yVHJhZGVySW5Bc3NldEluUGVyaW9kBARfYW1tB190cmFkZXIIX2Fzc2V0SWQHX3dlZWtJZAQPcmV3YXJkQXNzZXRSYXRlCQEQZ2V0QW1tUmV3YXJkUmF0ZQIFBF9hbW0FCF9hc3NldElkAwkAAAIFD3Jld2FyZEFzc2V0UmF0ZQAAAAAEEXRvdGFsRmVlc0luUGVyaW9kCQEUZ2V0VG90YWxGZWVzSW5QZXJpb2QCBQRfYW1tBQdfd2Vla0lkBBByZXdhcmRBc3NldFByaWNlCQETZ2V0UmV3YXJkQXNzZXRQcmljZQIFCF9hc3NldElkBQdfd2Vla0lkBBxtYXhBbW91bnRPZkFzc2V0VG9EaXN0cmlidXRlCQEfZ2V0TWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQMFBF9hbW0FCF9hc3NldElkBQdfd2Vla0lkBAt0cmFkZXJTY29yZQkBFmdldFRyYWRlclNjb3JlSW5QZXJpb2QDBQRfYW1tBQdfdHJhZGVyBQdfd2Vla0lkBBB0b3RhbFRyYWRlclNjb3JlCQEVZ2V0VG90YWxTY29yZUluUGVyaW9kAgUEX2FtbQUHX3dlZWtJZAMJAAACBRB0b3RhbFRyYWRlclNjb3JlAAAAAAQWdG90YWxBc3NldFRvRGlzdHJpYnV0ZQkBBGRpdmQCCQEEbXVsZAIFEXRvdGFsRmVlc0luUGVyaW9kBQ9yZXdhcmRBc3NldFJhdGUFEHJld2FyZEFzc2V0UHJpY2UEHGFjdHVhbFRvdGFsQXNzZXRUb0Rpc3RyaWJ1dGUJAQRtaW52AgUWdG90YWxBc3NldFRvRGlzdHJpYnV0ZQUcbWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQQUdHJhZGVyU2hhcmVJblJld2FyZHMJAQRkaXZkAgkBBG11bGQCBRxhY3R1YWxUb3RhbEFzc2V0VG9EaXN0cmlidXRlBQt0cmFkZXJTY29yZQUQdG90YWxUcmFkZXJTY29yZQUUdHJhZGVyU2hhcmVJblJld2FyZHMBI3RvdGFsUmV3YXJkRm9yVHJhZGVySW5Bc3NldEluUGVyaW9kAwdfdHJhZGVyCF9hc3NldElkB193ZWVrSWQEBGFtbXMJAQdnZXRBbW1zAAoBB2NvbXB1dGUCBF9hY2MEX2FtbQkAZAIFBF9hY2MJAR5yZXdhcmRGb3JUcmFkZXJJbkFzc2V0SW5QZXJpb2QEBQRfYW1tBQdfdHJhZGVyBQhfYXNzZXRJZAUHX3dlZWtJZAoAAiRsBQRhbW1zCgACJHMJAJADAQUCJGwKAAUkYWNjMAAACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQdjb21wdXRlAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAyMAkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAANAA4ADwAQABEAEgATABQBG3RvdGFsUmV3YXJkRm9yVHJhZGVySW5Bc3NldAMHX3RyYWRlcghfYXNzZXRJZAhfcGVyaW9kcwoBB2NvbXB1dGUCBF9hY2MHX3dlZWtJZAkAZAIFBF9hY2MJASN0b3RhbFJld2FyZEZvclRyYWRlckluQXNzZXRJblBlcmlvZAMFB190cmFkZXIFCF9hc3NldElkBQdfd2Vla0lkCgACJGwFCF9wZXJpb2RzCgACJHMJAJADAQUCJGwKAAUkYWNjMAAACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQdjb21wdXRlAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxMgkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAEgZ2V0VHJhZGVyQXZlcmFnZU5vdGlvbmFsSW5QZXJpb2QEBF9hbW0HX3RyYWRlcgdfd2Vla0lkDV9kZWZhdWx0VmFsdWUEA2tleQkBI2dldFRyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kS2V5AwUEX2FtbQUHX3RyYWRlcgUHX3dlZWtJZAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGF0BQNrZXkFDV9kZWZhdWx0VmFsdWUBCWlzQ2xhaW1lZAMHX3RyYWRlcghfYXNzZXRJZAdfcGVyaW9kBANrZXkJAR5nZXRDbGFpbWVkVHJhZGVyQXNzZXRQZXJpb2RLZXkDBQdfdHJhZGVyBQhfYXNzZXRJZAUHX3BlcmlvZAkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGF0BQNrZXkHAQZhZGp1c3QCB19hbW91bnQIX2Fzc2V0SWQEBWFzc2V0CQETdmFsdWVPckVycm9yTWVzc2FnZQIJAOwHAQkA2QQBBQhfYXNzZXRJZAIQSW52YWxpZCBhc3NldCBpZAQIZGVjaW1hbHMIBQVhc3NldAhkZWNpbWFscwMJAAACBQhkZWNpbWFscwAGCQBpAgUHX2Ftb3VudABkBQdfYW1vdW50ASBjbGFpbUFsbFJld2FyZEZvclBlcmlvZEZvclRyYWRlcgMHX3RyYWRlcghfYXNzZXRJZAhfcGVyaW9kcwQHcGVyaW9kcwkBCXN0clRvTGlzdAEFCF9wZXJpb2RzCgEQY2hlY2tBbmRGaWx0ZXJGbgIEX2FjYwVfbmV4dAQGcGVyaW9kCQETdmFsdWVPckVycm9yTWVzc2FnZQIJALYJAQUFX25leHQJAKwCAgIQSW52YWxpZCBwZXJpb2Q6IAUFX25leHQDAwkBD2NvbnRhaW5zRWxlbWVudAIFBF9hY2MFBnBlcmlvZAYJAQlpc0NsYWltZWQDBQdfdHJhZGVyBQhfYXNzZXRJZAUGcGVyaW9kBQRfYWNjCQDNCAIFBF9hY2MFBnBlcmlvZAQMdmFsaWRQZXJpb2RzCgACJGwFB3BlcmlvZHMKAAIkcwkAkAMBBQIkbAoABSRhY2MwBQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBEGNoZWNrQW5kRmlsdGVyRm4CBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEyCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMCgESbWFya1BlcmlvZEFzRG9uZUZuAgRfYWNjB19wZXJpb2QJAM0IAgUEX2FjYwkBDEJvb2xlYW5FbnRyeQIJAR5nZXRDbGFpbWVkVHJhZGVyQXNzZXRQZXJpb2RLZXkDBQdfdHJhZGVyBQhfYXNzZXRJZAUHX3BlcmlvZAYEFG1hcmtQZXJpb2RzQXNDbGFpbWVkCgACJGwFDHZhbGlkUGVyaW9kcwoAAiRzCQCQAwEFAiRsCgAFJGFjYzAFA25pbAoBBSRmMV8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQESbWFya1BlcmlvZEFzRG9uZUZuAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYxXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxNgkBBSRmMV8yAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAANAA4ADwAQBAZhbW91bnQJARt0b3RhbFJld2FyZEZvclRyYWRlckluQXNzZXQDBQdfdHJhZGVyBQhfYXNzZXRJZAUMdmFsaWRQZXJpb2RzCQCUCgIFBmFtb3VudAkAzggCCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMJARFAZXh0ck5hdGl2ZSgxMDYyKQEFB190cmFkZXIJAQZhZGp1c3QCBQZhbW91bnQFCF9hc3NldElkCQDZBAEFCF9hc3NldElkCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEjZ2V0VG90YWxDbGFpbWVkRm9yVHJhZGVyQW5kQXNzZXRLZXkCBQdfdHJhZGVyBQhfYXNzZXRJZAkAZAIJASBnZXRUb3RhbENsYWltZWRGb3JUcmFkZXJBbmRBc3NldAIFB190cmFkZXIFCF9hc3NldElkBQZhbW91bnQFA25pbAUUbWFya1BlcmlvZHNBc0NsYWltZWQBDXVwZGF0ZUFtbUZlZXMEBF9hbW0HX3RyYWRlcgdfd2Vla0lkCl9hY3R1YWxGZWUEDXRyYWRlckZlZXNLZXkJARhnZXRUcmFkZXJGZWVzSW5QZXJpb2RLZXkDBQRfYW1tBQdfdHJhZGVyBQdfd2Vla0lkBAx0b3RhbEZlZXNLZXkJARpnZXRUb3RhbEFtbUZlZXNJblBlcmlvZEtleQIFBF9hbW0FB193ZWVrSWQEDHJld2FyZEFzc2V0cwkBEWdldFJld2FyZEFzc2V0SWRzAAQMZmVlc0luUGVyaW9kCQEPZ2V0RmVlc0luUGVyaW9kAwUEX2FtbQUHX3RyYWRlcgUHX3dlZWtJZAQRdG90YWxGZWVzSW5QZXJpb2QJARRnZXRUb3RhbEZlZXNJblBlcmlvZAIFBF9hbW0FB193ZWVrSWQED25ld0ZlZXNJblBlcmlvZAkAZAIFDGZlZXNJblBlcmlvZAUKX2FjdHVhbEZlZQQUbmV3VG90YWxGZWVzSW5QZXJpb2QJAGQCBRF0b3RhbEZlZXNJblBlcmlvZAUKX2FjdHVhbEZlZQoBFWRvSGFuZGxlUmV3YXJkQXNzZXRJZAIEX2FjYwhfYXNzZXRJZAQDa2V5CQETZ2V0QW1tUmV3YXJkUmF0ZUtleQIFBF9hbW0FCF9hc3NldElkAwkAZgIJAQRpbnQwAQUDa2V5AAAEEXRvdGFsQXNzZXRGZWVzS2V5CQEcZ2V0VG90YWxBc3NldEZlZXNJblBlcmlvZEtleQIFCF9hc3NldElkBQdfd2Vla0lkCQDNCAIFBF9hY2MJAQxJbnRlZ2VyRW50cnkCBRF0b3RhbEFzc2V0RmVlc0tleQkAZAIJARlnZXRUb3RhbEFzc2V0RmVlc0luUGVyaW9kAgUIX2Fzc2V0SWQFB193ZWVrSWQFCl9hY3R1YWxGZWUFBF9hY2MED3VwZGF0ZUFzc2V0RmVlcwoAAiRsBQxyZXdhcmRBc3NldHMKAAIkcwkAkAMBBQIkbAoABSRhY2MwBQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBFWRvSGFuZGxlUmV3YXJkQXNzZXRJZAIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMTAJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQ10cmFkZXJGZWVzS2V5BQ9uZXdGZWVzSW5QZXJpb2QJAMwIAgkBDEludGVnZXJFbnRyeQIFDHRvdGFsRmVlc0tleQUUbmV3VG90YWxGZWVzSW5QZXJpb2QFA25pbAUPdXBkYXRlQXNzZXRGZWVzAQ11cGRhdGVBbW1EYXRhBwRfYW1tB190cmFkZXIHX3dlZWtJZA5fdHJhZGVyQXZlcmFnZRNfdHJhZGVyTGFzdE5vdGlvbmFsDF90cmFkZXJTY29yZQtfdG90YWxTY29yZQQVdG90YWxTY29yZUluUGVyaW9kS2V5CQEYZ2V0VG90YWxTY29yZUluUGVyaW9kS2V5AgUEX2FtbQUHX3dlZWtJZAQWdHJhZGVyU2NvcmVJblBlcmlvZEtleQkBGWdldFRyYWRlclNjb3JlSW5QZXJpb2RLZXkDBQRfYW1tBQdfdHJhZGVyBQdfd2Vla0lkBA9sYXN0Tm90aW9uYWxLZXkJARJnZXRMYXN0Tm90aW9uYWxLZXkCBQRfYW1tBQdfdHJhZGVyBCB0cmFkZXJBdmVyYWdlTm90aW9uYWxJblBlcmlvZEtleQkBI2dldFRyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kS2V5AwUEX2FtbQUHX3RyYWRlcgUHX3dlZWtJZAkAzAgCCQEMSW50ZWdlckVudHJ5AgUVdG90YWxTY29yZUluUGVyaW9kS2V5BQtfdG90YWxTY29yZQkAzAgCCQEMSW50ZWdlckVudHJ5AgUWdHJhZGVyU2NvcmVJblBlcmlvZEtleQUMX3RyYWRlclNjb3JlCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQ9sYXN0Tm90aW9uYWxLZXkFE190cmFkZXJMYXN0Tm90aW9uYWwJAMwIAgkBDEludGVnZXJFbnRyeQIFIHRyYWRlckF2ZXJhZ2VOb3Rpb25hbEluUGVyaW9kS2V5BQ5fdHJhZGVyQXZlcmFnZQUDbmlsDAFpAQppbml0aWFsaXplAgxfY29vcmRpbmF0b3IHX29yYWNsZQMJAQtpbml0aWFsaXplZAAJAAIBAhNBbHJlYWR5IGluaXRpYWxpemVkCQDMCAIJAQtTdHJpbmdFbnRyeQIFFGtfY29vcmRpbmF0b3JBZGRyZXNzBQxfY29vcmRpbmF0b3IJAMwIAgkBC1N0cmluZ0VudHJ5AgUPa19vcmFjbGVBZGRyZXNzBQdfb3JhY2xlCQDMCAIJAQxCb29sZWFuRW50cnkCBQ1rX2luaXRpYWxpemVkBgUDbmlsAWkBEHNldE9yYWNsZUFkZHJlc3MBB19vcmFjbGUDAwkBASEBCQELaW5pdGlhbGl6ZWQABgkBAiE9AggFAWkGY2FsbGVyCQEMYWRtaW5BZGRyZXNzAAkAAgECI0ludmFsaWQgc2V0T3JhY2xlQWRkcmVzcyBwYXJhbWV0ZXJzCQDMCAIJAQtTdHJpbmdFbnRyeQIFD2tfb3JhY2xlQWRkcmVzcwUHX29yYWNsZQUDbmlsAWkBDnZpZXdfZ2V0UGVyaW9kAAQGd2Vla0lkCQEJZ2V0V2Vla0lkAQUEVElNRQQJd2Vla1N0YXJ0CQEMZ2V0V2Vla1N0YXJ0AQUGd2Vla0lkBAd3ZWVrRW5kCQEKZ2V0V2Vla0VuZAEFBndlZWtJZAkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkApAMBBQl3ZWVrU3RhcnQCASwJAKQDAQUHd2Vla0VuZAIBLAkApAMBBQRUSU1FAWkBJHZpZXdfZ2V0TWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQMEX2FtbQhfYXNzZXRJZAdfd2Vla0lkCQACAQkApAMBCQEfZ2V0TWF4QW1vdW50T2ZBc3NldFRvRGlzdHJpYnV0ZQMFBF9hbW0FCF9hc3NldElkBQdfd2Vla0lkAWkBEXZpZXdfY2xhaW1SZXdhcmRzAwdfdHJhZGVyCF9hc3NldElkB19wZXJpb2QJAAIBCQCkAwEJARt0b3RhbFJld2FyZEZvclRyYWRlckluQXNzZXQDBQdfdHJhZGVyBQhfYXNzZXRJZAkAzAgCBQdfcGVyaW9kBQNuaWwBaQEPdmlld19jYWxjUmV3YXJkBAdfdHJhZGVyBF9hbW0IX2Fzc2V0SWQHX3BlcmlvZAkAAgEJAKQDAQkBHnJld2FyZEZvclRyYWRlckluQXNzZXRJblBlcmlvZAQFBF9hbW0FCF9hc3NldElkBQdfdHJhZGVyBQdfcGVyaW9kAWkBFHZpZXdfY2xhaW1BbGxSZXdhcmRzAwdfdHJhZGVyCF9hc3NldElkCF9wZXJpb2RzBAZyZXN1bHQJASBjbGFpbUFsbFJld2FyZEZvclBlcmlvZEZvclRyYWRlcgMFB190cmFkZXIFCF9hc3NldElkBQhfcGVyaW9kcwkAAgEJAKwCAgkArAICCQCkAwEIBQZyZXN1bHQCXzECASwJAKQDAQkBIGdldFRvdGFsQ2xhaW1lZEZvclRyYWRlckFuZEFzc2V0AgUHX3RyYWRlcgUIX2Fzc2V0SWQBaQEPY2xhaW1BbGxSZXdhcmRzAghfYXNzZXRJZAhfcGVyaW9kcwQGdHJhZGVyCQClCAEIBQFpBmNhbGxlcgMJAQhjb250YWlucwIFCF9wZXJpb2RzCQCkAwEJAQlnZXRXZWVrSWQBBQRUSU1FCQACAQImQ2FuIG5vdCBjbGFpbSByZXdhcmRzIGZvciBjdXJyZW50IHdlZWsEDSR0MDE1NDM4MTU1MjEJASBjbGFpbUFsbFJld2FyZEZvclBlcmlvZEZvclRyYWRlcgMFBnRyYWRlcgUIX2Fzc2V0SWQFCF9wZXJpb2RzBAZhbW91bnQIBQ0kdDAxNTQzODE1NTIxAl8xBAZyZXN1bHQIBQ0kdDAxNTQzODE1NTIxAl8yAwkAAAIFBmFtb3VudAAACQACAQIQTm90aGluZyB0byBjbGFpbQUGcmVzdWx0AWkBEWF0dGFjaFJld2FyZEFzc2V0AghfYXNzZXRJZBNfbWF4QW1vdW50UGVyUGVyaW9kAwMJAQEhAQkBC2luaXRpYWxpemVkAAYJAQIhPQIIBQFpBmNhbGxlcgkBDGFkbWluQWRkcmVzcwAJAAIBAiBJbnZhbGlkIGF0dGFjaFJld2FyZEFzc2V0IHBhcmFtcwQEYW1tcwkBB2dldEFtbXMABA5yZXdhcmRBc3NldElkcwkBEWdldFJld2FyZEFzc2V0SWRzAAQRbmV3UmV3YXJkQXNzZXRJZHMDCQEPY29udGFpbnNFbGVtZW50AgUOcmV3YXJkQXNzZXRJZHMFCF9hc3NldElkBQ5yZXdhcmRBc3NldElkcwkAzQgCBQ5yZXdhcmRBc3NldElkcwUIX2Fzc2V0SWQJAMwIAgkBDEludGVnZXJFbnRyeQIJAR1nZXRBc3NldE1heEFtb3VudFBlclBlcmlvZEtleQEFCF9hc3NldElkBRNfbWF4QW1vdW50UGVyUGVyaW9kCQDMCAIJAQtTdHJpbmdFbnRyeQIFEGtfcmV3YXJkQXNzZXRJZHMJAQlsaXN0VG9TdHIBBRFuZXdSZXdhcmRBc3NldElkcwUDbmlsAWkBDWF0dGFjaFJld2FyZHMDBF9hbW0IX2Fzc2V0SWQLX3Jld2FyZFJhdGUDAwMJAQEhAQkBC2luaXRpYWxpemVkAAYJAQIhPQIIBQFpBmNhbGxlcgkBDGFkbWluQWRkcmVzcwAGCQEBIQEJAQtpc1doaXRlbGlzdAEFBF9hbW0JAAIBAhxJbnZhbGlkIGF0dGFjaFJld2FyZHMgcGFyYW1zBARhbW1zCQEHZ2V0QW1tcwAEDnJld2FyZEFzc2V0SWRzCQERZ2V0UmV3YXJkQXNzZXRJZHMABAduZXdBbW1zAwkBD2NvbnRhaW5zRWxlbWVudAIFBGFtbXMFBF9hbW0FBGFtbXMJAM0IAgUEYW1tcwUEX2FtbQQRbmV3UmV3YXJkQXNzZXRJZHMDCQEPY29udGFpbnNFbGVtZW50AgUOcmV3YXJkQXNzZXRJZHMFCF9hc3NldElkBQ5yZXdhcmRBc3NldElkcwkAzQgCBQ5yZXdhcmRBc3NldElkcwUIX2Fzc2V0SWQJAMwIAgkBDEludGVnZXJFbnRyeQIJARNnZXRBbW1SZXdhcmRSYXRlS2V5AgUEX2FtbQUIX2Fzc2V0SWQFC19yZXdhcmRSYXRlCQDMCAIJAQtTdHJpbmdFbnRyeQIFBmtfYW1tcwkBCWxpc3RUb1N0cgEFB25ld0FtbXMJAMwIAgkBC1N0cmluZ0VudHJ5AgUQa19yZXdhcmRBc3NldElkcwkBCWxpc3RUb1N0cgEFEW5ld1Jld2FyZEFzc2V0SWRzBQNuaWwBaQEKbm90aWZ5RmVlcwIHX3RyYWRlcgRfZmVlBANhbW0JAKUIAQgFAWkGY2FsbGVyAwMJAQEhAQkBC2lzV2hpdGVsaXN0AQUDYW1tBgkBASEBCQELaW5pdGlhbGl6ZWQACQACAQIZSW52YWxpZCBub3RpZnlGZWVzIHBhcmFtcwQJYWN0dWFsRmVlCQEOdXNkblRvRGVjaW1hbHMBBQRfZmVlBAZ3ZWVrSWQJAQlnZXRXZWVrSWQBBQRUSU1FCQENdXBkYXRlQW1tRmVlcwQFA2FtbQUHX3RyYWRlcgUGd2Vla0lkBQlhY3R1YWxGZWUBaQEObm90aWZ5Tm90aW9uYWwCB190cmFkZXIJX25vdGlvbmFsBANhbW0JAKUIAQgFAWkGY2FsbGVyAwMJAQEhAQkBC2lzV2hpdGVsaXN0AQUDYW1tBgkBASEBCQELaW5pdGlhbGl6ZWQACQACAQIdSW52YWxpZCBub3RpZnlOb3Rpb25hbCBwYXJhbXMEDmFjdHVhbE5vdGlvbmFsCQEOdXNkblRvRGVjaW1hbHMBBQlfbm90aW9uYWwEDGxhc3ROb3Rpb25hbAkBD2dldExhc3ROb3Rpb25hbAIFA2FtbQUHX3RyYWRlcgQGd2Vla0lkCQEJZ2V0V2Vla0lkAQUEVElNRQQJd2Vla1N0YXJ0CQEMZ2V0V2Vla1N0YXJ0AQUGd2Vla0lkBAd3ZWVrRW5kCQEKZ2V0V2Vla0VuZAEFBndlZWtJZAQBdAkAZQIFB3dlZWtFbmQFCXdlZWtTdGFydAQRb2xkUnVubmluZ0F2ZXJhZ2UJASBnZXRUcmFkZXJBdmVyYWdlTm90aW9uYWxJblBlcmlvZAQFA2FtbQUHX3RyYWRlcgUGd2Vla0lkBQxsYXN0Tm90aW9uYWwEAXcJAQRkaXZkAgkAZQIFB3dlZWtFbmQFBFRJTUUFAXQEEW9sZFRyYWRlckFtbVNjb3JlCQEWZ2V0VHJhZGVyU2NvcmVJblBlcmlvZAMFA2FtbQUHX3RyYWRlcgUGd2Vla0lkBBBvbGRUb3RhbEFtbVNjb3JlCQEVZ2V0VG90YWxTY29yZUluUGVyaW9kAgUDYW1tBQZ3ZWVrSWQEBGZlZXMJAQ9nZXRGZWVzSW5QZXJpb2QDBQNhbW0FB190cmFkZXIFBndlZWtJZAQRbmV3UnVubmluZ0F2ZXJhZ2UJAGUCCQBkAgURb2xkUnVubmluZ0F2ZXJhZ2UJAQRtdWxkAgUOYWN0dWFsTm90aW9uYWwFAXcJAQRtdWxkAgUMbGFzdE5vdGlvbmFsBQF3BBFuZXdUcmFkZXJBbW1TY29yZQkBBG11bGQCCQEEcG93ZAIFBGZlZXMFAUEJAQRwb3dkAgURbmV3UnVubmluZ0F2ZXJhZ2UJAGUCBQxERUNJTUFMX1VOSVQFAUEEGG5ld1RvdGFsQW1tU2NvcmVJblBlcmlvZAkAZAIJAGUCBRBvbGRUb3RhbEFtbVNjb3JlBRFvbGRUcmFkZXJBbW1TY29yZQURbmV3VHJhZGVyQW1tU2NvcmUJAQ11cGRhdGVBbW1EYXRhBwUDYW1tBQdfdHJhZGVyBQZ3ZWVrSWQFEW5ld1J1bm5pbmdBdmVyYWdlBQ5hY3R1YWxOb3Rpb25hbAURbmV3VHJhZGVyQW1tU2NvcmUFGG5ld1RvdGFsQW1tU2NvcmVJblBlcmlvZAECdHgBBnZlcmlmeQAJAPQDAwgFAnR4CWJvZHlCeXRlcwkAkQMCCAUCdHgGcHJvb2ZzAAAJAQ5hZG1pblB1YmxpY0tleQDQl1KH", "height": 2281029, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: E18ZCc7xomisbi43u3QHAR9NUBBGgXKTgtFvbFMYe7sG Next: 7vo1Y8ESAWLi8FF8gQbJSyi75EMhPTuxkhTLrUCTgm3k Diff:
Old | New | Differences | |
---|---|---|---|
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let k_totalFeesInPeriod = "k_totalFeesInPeriod" | |
5 | 5 | ||
6 | + | let k_totalAssetFeesInPeriod = "k_totalAssetFeesInPeriod" | |
7 | + | ||
6 | 8 | let k_traderFeesInPeriod = "k_traderFeesInPeriod" | |
7 | 9 | ||
8 | 10 | let k_lastNotional = "k_lastNotional" | |
9 | - | ||
10 | - | let k_lastDateInPeriod = "k_lastDateInPeriod" | |
11 | 11 | ||
12 | 12 | let k_totalScoreInPeriod = "k_totalScoreInPeriod" | |
13 | 13 | ||
17 | 17 | ||
18 | 18 | let k_ammRewardRate = "k_ammRewardRate" | |
19 | 19 | ||
20 | - | let k_ | |
20 | + | let k_assetMaxAmountPerPeriod = "k_assetMaxAmountPerPeriod" | |
21 | 21 | ||
22 | 22 | let k_rewardAssetIds = "k_rewardAssetIds" | |
23 | 23 | ||
28 | 28 | let k_oracleAddress = "k_oracleAddress" | |
29 | 29 | ||
30 | 30 | let k_oraclePrice = "price" | |
31 | + | ||
32 | + | let k_totalClaimedForTraderAndAsset = "k_totalClaimedForTraderAndAsset" | |
33 | + | ||
34 | + | let k_ammMaxAmountPerPeriod = "k_ammMaxAmountPerPeriod" | |
31 | 35 | ||
32 | 36 | let k_initialized = "k_initialized" | |
33 | 37 | ||
101 | 105 | func toCompositeKey (_key,_address) = ((_key + "_") + _address) | |
102 | 106 | ||
103 | 107 | ||
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") | |
105 | 111 | ||
106 | 112 | ||
107 | 113 | func adminPublicKey () = fromBase58String(getStringValue(coordinator(), k_admin_public_key)) | |
113 | 119 | func isWhitelist (_address) = valueOrElse(getBoolean(coordinator(), toCompositeKey(k_amm, _address)), false) | |
114 | 120 | ||
115 | 121 | ||
116 | - | func int (k) = valueOrErrorMessage(getInteger( | |
122 | + | func int (k) = valueOrErrorMessage(getInteger(that, k), ("no value for " + k)) | |
117 | 123 | ||
118 | 124 | ||
119 | - | func int0 (k) = valueOrElse(getInteger( | |
125 | + | func int0 (k) = valueOrElse(getInteger(that, k), 0) | |
120 | 126 | ||
121 | 127 | ||
122 | - | func initialized () = valueOrElse(getBoolean( | |
128 | + | func initialized () = valueOrElse(getBoolean(that, k_initialized), false) | |
123 | 129 | ||
124 | 130 | ||
125 | - | func | |
131 | + | func getRewardAssetIds () = strToList(valueOrElse(getString(that, k_rewardAssetIds), "")) | |
126 | 132 | ||
127 | 133 | ||
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)) | |
129 | 143 | ||
130 | 144 | ||
131 | 145 | func getLastNotionalKey (_amm,_trader) = ((((k_lastNotional + "_") + _amm) + "_") + _trader) | |
132 | - | ||
133 | - | ||
134 | - | func getLastDateInPeriodKey (_amm,_trader,_weekId) = ((((((k_lastDateInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId)) | |
135 | 146 | ||
136 | 147 | ||
137 | 148 | func getTraderScoreInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderScoreInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId)) | |
146 | 157 | func getAmmRewardRateKey (_amm,_assetId) = ((((k_ammRewardRate + "_") + _amm) + "_") + _assetId) | |
147 | 158 | ||
148 | 159 | ||
149 | - | func | |
160 | + | func getAssetMaxAmountPerPeriodKey (_assetId) = ((k_assetMaxAmountPerPeriod + "_") + _assetId) | |
150 | 161 | ||
151 | 162 | ||
152 | 163 | func getClaimedTraderAssetPeriodKey (_trader,_assetId,_period) = ((((((k_claimedAssetAndPeriod + "_") + _trader) + "_") + _assetId) + "_") + toString(_period)) | |
155 | 166 | func getOraclePriceKey (_assetId,_period) = ((((k_oraclePrice + "_") + toString(_period)) + "_") + _assetId) | |
156 | 167 | ||
157 | 168 | ||
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 | - | } | |
169 | + | func getTotalClaimedForTraderAndAssetKey (_trader,_assetId) = ((((k_totalClaimedForTraderAndAsset + "_") + _trader) + "_") + _assetId) | |
163 | 170 | ||
164 | 171 | ||
165 | - | func updateAmmData (_amm,_trader,_weekId,_traderLastDateInPeriod,_traderAverage,_traderLastNotional,_traderScore,_totalScore) = { | |
166 | - | let totalScoreInPeriodKey = getTotalScoreInPeriodKey(_amm, _weekId) | |
167 | - | let traderScoreInPeriodKey = getTraderScoreInPeriodKey(_amm, _trader, _weekId) | |
168 | - | let lastDateInPeriodKey = getLastDateInPeriodKey(_amm, _trader, _weekId) | |
169 | - | let lastNotionalKey = getLastNotionalKey(_amm, _trader) | |
170 | - | let traderAverageNotionalInPeriodKey = getTraderAverageNotionalInPeriodKey(_amm, _trader, _weekId) | |
171 | - | [IntegerEntry(totalScoreInPeriodKey, _totalScore), IntegerEntry(traderScoreInPeriodKey, _traderScore), IntegerEntry(lastDateInPeriodKey, _traderLastDateInPeriod), IntegerEntry(lastNotionalKey, _traderLastNotional), IntegerEntry(traderAverageNotionalInPeriodKey, _traderAverage)] | |
172 | - | } | |
172 | + | func getAmmMaxAmountPerPeriodKey (_amm,_assetId) = ((((k_ammMaxAmountPerPeriod + "_") + _amm) + "_") + _assetId) | |
173 | 173 | ||
174 | 174 | ||
175 | 175 | func usdnToDecimals (_amount) = (_amount * 100) | |
185 | 185 | ||
186 | 186 | ||
187 | 187 | func getFeesInPeriod (_amm,_trader,_weekId) = { | |
188 | - | let key = | |
189 | - | | |
188 | + | let key = getTraderFeesInPeriodKey(_amm, _trader, _weekId) | |
189 | + | int0(key) | |
190 | 190 | } | |
191 | 191 | ||
192 | 192 | ||
193 | 193 | func getTotalFeesInPeriod (_amm,_weekId) = { | |
194 | - | let key = getTotalFeesInPeriodKey(_amm, _weekId) | |
195 | - | 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) | |
196 | 202 | } | |
197 | 203 | ||
198 | 204 | ||
199 | 205 | func getLastNotional (_amm,_trader) = { | |
200 | 206 | let key = getLastNotionalKey(_amm, _trader) | |
201 | - | valueOrElse(getInteger(this, key), 0) | |
202 | - | } | |
203 | - | ||
204 | - | ||
205 | - | func getLastDateInPeriod (_amm,_trader,_weekId,_weekStart) = { | |
206 | - | let key = getLastDateInPeriodKey(_amm, _trader, _weekId) | |
207 | - | valueOrElse(getInteger(this, key), _weekStart) | |
207 | + | int0(key) | |
208 | 208 | } | |
209 | 209 | ||
210 | 210 | ||
211 | 211 | func getTraderScoreInPeriod (_amm,_trader,_weekId) = { | |
212 | 212 | let key = getTraderScoreInPeriodKey(_amm, _trader, _weekId) | |
213 | - | | |
213 | + | int0(key) | |
214 | 214 | } | |
215 | 215 | ||
216 | 216 | ||
217 | 217 | func getTotalScoreInPeriod (_amm,_weekId) = { | |
218 | 218 | let key = getTotalScoreInPeriodKey(_amm, _weekId) | |
219 | - | | |
219 | + | int0(key) | |
220 | 220 | } | |
221 | 221 | ||
222 | 222 | ||
223 | - | func getAmms () = strToList(valueOrElse(getString(this, k_amms), "")) | |
224 | - | ||
225 | - | ||
226 | - | func getRewardAssetIds () = strToList(valueOrElse(getString(this, k_rewardAssetIds), "")) | |
223 | + | func getAmms () = strToList(valueOrElse(getString(that, k_amms), "")) | |
227 | 224 | ||
228 | 225 | ||
229 | 226 | func getAmmRewardRate (_amm,_assetId) = { | |
230 | 227 | let key = getAmmRewardRateKey(_amm, _assetId) | |
231 | - | | |
228 | + | int0(key) | |
232 | 229 | } | |
233 | 230 | ||
234 | 231 | ||
235 | - | func getAmmMaxAmountPerPeriod (_amm,_assetId) = { | |
236 | - | let key = getAmmMaxAmountPerPeriodKey(_amm, _assetId) | |
237 | - | valueOrElse(getInteger(this, key), 0) | |
232 | + | func getAssetMaxAmountPerPeriod (_assetId) = { | |
233 | + | let key = getAssetMaxAmountPerPeriodKey(_assetId) | |
234 | + | int0(key) | |
235 | + | } | |
236 | + | ||
237 | + | ||
238 | + | func getTotalClaimedForTraderAndAsset (_trader,_assetId) = { | |
239 | + | let key = getTotalClaimedForTraderAndAssetKey(_trader, _assetId) | |
240 | + | int0(key) | |
238 | 241 | } | |
239 | 242 | ||
240 | 243 | ||
241 | 244 | func getRewardAssetPrice (_assetId,_weekId) = { | |
242 | - | let oracleAddressStr = valueOrErrorMessage(getString( | |
245 | + | let oracleAddressStr = valueOrErrorMessage(getString(that, k_oracleAddress), "Oracle not set") | |
243 | 246 | let oracleAddress = valueOrErrorMessage(addressFromString(oracleAddressStr), "Invalid oracle address") | |
244 | 247 | let priceKey = getOraclePriceKey(_assetId, _weekId) | |
245 | 248 | 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 | + | } | |
246 | 265 | } | |
247 | 266 | ||
248 | 267 | ||
253 | 272 | else { | |
254 | 273 | let totalFeesInPeriod = getTotalFeesInPeriod(_amm, _weekId) | |
255 | 274 | let rewardAssetPrice = getRewardAssetPrice(_assetId, _weekId) | |
256 | - | let maxAmountOfAssetToDistribute = | |
275 | + | let maxAmountOfAssetToDistribute = getMaxAmountOfAssetToDistribute(_amm, _assetId, _weekId) | |
257 | 276 | let traderScore = getTraderScoreInPeriod(_amm, _trader, _weekId) | |
258 | 277 | let totalTraderScore = getTotalScoreInPeriod(_amm, _weekId) | |
259 | 278 | if ((totalTraderScore == 0)) | |
260 | 279 | then 0 | |
261 | 280 | else { | |
262 | - | let totalAssetToDistribute = muld | |
281 | + | let totalAssetToDistribute = divd(muld(totalFeesInPeriod, rewardAssetRate), rewardAssetPrice) | |
263 | 282 | let actualTotalAssetToDistribute = minv(totalAssetToDistribute, maxAmountOfAssetToDistribute) | |
264 | 283 | let traderShareInRewards = divd(muld(actualTotalAssetToDistribute, traderScore), totalTraderScore) | |
265 | 284 | traderShareInRewards | |
299 | 318 | ||
300 | 319 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
301 | 320 | then $a | |
302 | - | else throw("List size exceeds | |
321 | + | else throw("List size exceeds 12") | |
303 | 322 | ||
304 | - | $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) | |
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) | |
324 | + | } | |
325 | + | ||
326 | + | ||
327 | + | func getTraderAverageNotionalInPeriod (_amm,_trader,_weekId,_defaultValue) = { | |
328 | + | let key = getTraderAverageNotionalInPeriodKey(_amm, _trader, _weekId) | |
329 | + | valueOrElse(getInteger(that, key), _defaultValue) | |
305 | 330 | } | |
306 | 331 | ||
307 | 332 | ||
308 | 333 | func isClaimed (_trader,_assetId,_period) = { | |
309 | 334 | let key = getClaimedTraderAssetPeriodKey(_trader, _assetId, _period) | |
310 | - | 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 | |
345 | + | } | |
346 | + | ||
347 | + | ||
348 | + | func claimAllRewardForPeriodForTrader (_trader,_assetId,_periods) = { | |
349 | + | let periods = strToList(_periods) | |
350 | + | func checkAndFilterFn (_acc,_next) = { | |
351 | + | let period = valueOrErrorMessage(parseInt(_next), ("Invalid period: " + _next)) | |
352 | + | if (if (containsElement(_acc, period)) | |
353 | + | then true | |
354 | + | else isClaimed(_trader, _assetId, period)) | |
355 | + | then _acc | |
356 | + | else (_acc :+ period) | |
357 | + | } | |
358 | + | ||
359 | + | let validPeriods = { | |
360 | + | let $l = periods | |
361 | + | let $s = size($l) | |
362 | + | let $acc0 = nil | |
363 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
364 | + | then $a | |
365 | + | else checkAndFilterFn($a, $l[$i]) | |
366 | + | ||
367 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
368 | + | then $a | |
369 | + | else throw("List size exceeds 12") | |
370 | + | ||
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) | |
372 | + | } | |
373 | + | func markPeriodAsDoneFn (_acc,_period) = (_acc :+ BooleanEntry(getClaimedTraderAssetPeriodKey(_trader, _assetId, _period), true)) | |
374 | + | ||
375 | + | let markPeriodsAsClaimed = { | |
376 | + | let $l = validPeriods | |
377 | + | let $s = size($l) | |
378 | + | let $acc0 = nil | |
379 | + | func $f1_1 ($a,$i) = if (($i >= $s)) | |
380 | + | then $a | |
381 | + | else markPeriodAsDoneFn($a, $l[$i]) | |
382 | + | ||
383 | + | func $f1_2 ($a,$i) = if (($i >= $s)) | |
384 | + | then $a | |
385 | + | else throw("List size exceeds 16") | |
386 | + | ||
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) | |
388 | + | } | |
389 | + | let amount = totalRewardForTraderInAsset(_trader, _assetId, validPeriods) | |
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)] | |
311 | 436 | } | |
312 | 437 | ||
313 | 438 | ||
319 | 444 | ||
320 | 445 | ||
321 | 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))) | |
467 | + | ||
468 | + | ||
469 | + | ||
470 | + | @Callable(i) | |
322 | 471 | func view_claimRewards (_trader,_assetId,_period) = throw(toString(totalRewardForTraderInAsset(_trader, _assetId, [_period]))) | |
323 | 472 | ||
324 | 473 | ||
325 | 474 | ||
326 | 475 | @Callable(i) | |
327 | - | func claimRewards (_assetId,_period) = { | |
476 | + | func view_calcReward (_trader,_amm,_assetId,_period) = throw(toString(rewardForTraderInAssetInPeriod(_amm, _assetId, _trader, _period))) | |
477 | + | ||
478 | + | ||
479 | + | ||
480 | + | @Callable(i) | |
481 | + | func view_claimAllRewards (_trader,_assetId,_periods) = { | |
482 | + | let result = claimAllRewardForPeriodForTrader(_trader, _assetId, _periods) | |
483 | + | throw(((toString(result._1) + ",") + toString(getTotalClaimedForTraderAndAsset(_trader, _assetId)))) | |
484 | + | } | |
485 | + | ||
486 | + | ||
487 | + | ||
488 | + | @Callable(i) | |
489 | + | func claimAllRewards (_assetId,_periods) = { | |
328 | 490 | let trader = toString(i.caller) | |
329 | - | if ( | |
330 | - | then throw(" | |
491 | + | if (contains(_periods, toString(getWeekId(TIME)))) | |
492 | + | then throw("Can not claim rewards for current week") | |
331 | 493 | else { | |
332 | - | let currentWeekId = getWeekId(lastBlock.timestamp) | |
333 | - | if ((_period >= currentWeekId)) | |
334 | - | then throw(("Unable to claim rewards in future periods, current period is " + toString(currentWeekId))) | |
335 | - | else { | |
336 | - | let amount = totalRewardForTraderInAsset(trader, _assetId, [_period]) | |
337 | - | if ((amount == 0)) | |
338 | - | then throw("Nothing to claim") | |
339 | - | else [ScriptTransfer(i.caller, amount, fromBase58String(_assetId)), BooleanEntry(getClaimedTraderAssetPeriodKey(trader, _assetId, _period), true)] | |
340 | - | } | |
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 | |
341 | 500 | } | |
342 | 501 | } | |
343 | 502 | ||
344 | 503 | ||
345 | 504 | ||
346 | 505 | @Callable(i) | |
347 | - | func | |
506 | + | func attachRewardAsset (_assetId,_maxAmountPerPeriod) = if (if (!(initialized())) | |
348 | 507 | then true | |
349 | 508 | 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))) | |
350 | 527 | then throw("Invalid attachRewards params") | |
351 | 528 | else { | |
352 | 529 | let amms = getAmms() | |
357 | 534 | let newRewardAssetIds = if (containsElement(rewardAssetIds, _assetId)) | |
358 | 535 | then rewardAssetIds | |
359 | 536 | else (rewardAssetIds :+ _assetId) | |
360 | - | [IntegerEntry(getAmmRewardRateKey(_amm, _assetId), _rewardRate | |
537 | + | [IntegerEntry(getAmmRewardRateKey(_amm, _assetId), _rewardRate), StringEntry(k_amms, listToStr(newAmms)), StringEntry(k_rewardAssetIds, listToStr(newRewardAssetIds))] | |
361 | 538 | } | |
362 | 539 | ||
363 | 540 | ||
371 | 548 | then throw("Invalid notifyFees params") | |
372 | 549 | else { | |
373 | 550 | let actualFee = usdnToDecimals(_fee) | |
374 | - | let ts = lastBlock.timestamp | |
375 | - | let weekId = getWeekId(ts) | |
376 | - | let feesInPeriod = getFeesInPeriod(amm, _trader, weekId) | |
377 | - | let totalFeesInPeriod = getTotalFeesInPeriod(amm, weekId) | |
378 | - | let newFeesInPeriod = (feesInPeriod + actualFee) | |
379 | - | let newTotalFeesInPeriod = (totalFeesInPeriod + actualFee) | |
380 | - | updateAmmFees(amm, _trader, weekId, newFeesInPeriod, newTotalFeesInPeriod) | |
551 | + | let weekId = getWeekId(TIME) | |
552 | + | updateAmmFees(amm, _trader, weekId, actualFee) | |
381 | 553 | } | |
382 | 554 | } | |
383 | 555 | ||
392 | 564 | then throw("Invalid notifyNotional params") | |
393 | 565 | else { | |
394 | 566 | let actualNotional = usdnToDecimals(_notional) | |
395 | - | let ts = lastBlock.timestamp | |
396 | 567 | let lastNotional = getLastNotional(amm, _trader) | |
397 | - | let weekId = getWeekId( | |
568 | + | let weekId = getWeekId(TIME) | |
398 | 569 | let weekStart = getWeekStart(weekId) | |
399 | 570 | let weekEnd = getWeekEnd(weekId) | |
400 | 571 | let t = (weekEnd - weekStart) | |
401 | - | let | |
402 | - | let | |
572 | + | let oldRunningAverage = getTraderAverageNotionalInPeriod(amm, _trader, weekId, lastNotional) | |
573 | + | let w = divd((weekEnd - TIME), t) | |
403 | 574 | let oldTraderAmmScore = getTraderScoreInPeriod(amm, _trader, weekId) | |
404 | 575 | let oldTotalAmmScore = getTotalScoreInPeriod(amm, weekId) | |
405 | 576 | let fees = getFeesInPeriod(amm, _trader, weekId) | |
406 | - | let newRunningAverage = ( | |
577 | + | let newRunningAverage = ((oldRunningAverage + muld(actualNotional, w)) - muld(lastNotional, w)) | |
407 | 578 | let newTraderAmmScore = muld(powd(fees, A), powd(newRunningAverage, (DECIMAL_UNIT - A))) | |
408 | 579 | let newTotalAmmScoreInPeriod = ((oldTotalAmmScore - oldTraderAmmScore) + newTraderAmmScore) | |
409 | - | updateAmmData(amm, _trader, weekId, | |
580 | + | updateAmmData(amm, _trader, weekId, newRunningAverage, actualNotional, newTraderAmmScore, newTotalAmmScoreInPeriod) | |
410 | 581 | } | |
411 | 582 | } | |
412 | 583 |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let k_totalFeesInPeriod = "k_totalFeesInPeriod" | |
5 | 5 | ||
6 | + | let k_totalAssetFeesInPeriod = "k_totalAssetFeesInPeriod" | |
7 | + | ||
6 | 8 | let k_traderFeesInPeriod = "k_traderFeesInPeriod" | |
7 | 9 | ||
8 | 10 | let k_lastNotional = "k_lastNotional" | |
9 | - | ||
10 | - | let k_lastDateInPeriod = "k_lastDateInPeriod" | |
11 | 11 | ||
12 | 12 | let k_totalScoreInPeriod = "k_totalScoreInPeriod" | |
13 | 13 | ||
14 | 14 | let k_traderScoreInPeriod = "k_traderScoreInPeriod" | |
15 | 15 | ||
16 | 16 | let k_traderAverageNotionalInPeriod = "k_traderAverageNotionalInPeriod" | |
17 | 17 | ||
18 | 18 | let k_ammRewardRate = "k_ammRewardRate" | |
19 | 19 | ||
20 | - | let k_ | |
20 | + | let k_assetMaxAmountPerPeriod = "k_assetMaxAmountPerPeriod" | |
21 | 21 | ||
22 | 22 | let k_rewardAssetIds = "k_rewardAssetIds" | |
23 | 23 | ||
24 | 24 | let k_amms = "k_amms" | |
25 | 25 | ||
26 | 26 | let k_claimedAssetAndPeriod = "k_claimedAssetAndPeriod" | |
27 | 27 | ||
28 | 28 | let k_oracleAddress = "k_oracleAddress" | |
29 | 29 | ||
30 | 30 | let k_oraclePrice = "price" | |
31 | + | ||
32 | + | let k_totalClaimedForTraderAndAsset = "k_totalClaimedForTraderAndAsset" | |
33 | + | ||
34 | + | let k_ammMaxAmountPerPeriod = "k_ammMaxAmountPerPeriod" | |
31 | 35 | ||
32 | 36 | let k_initialized = "k_initialized" | |
33 | 37 | ||
34 | 38 | let k_coordinatorAddress = "k_coordinatorAddress" | |
35 | 39 | ||
36 | 40 | let k_governance_asset = "k_gov_asset" | |
37 | 41 | ||
38 | 42 | let k_quote_asset = "k_quote_asset" | |
39 | 43 | ||
40 | 44 | let k_admin_public_key = "k_admin_public_key" | |
41 | 45 | ||
42 | 46 | let k_admin_address = "k_admin_address" | |
43 | 47 | ||
44 | 48 | let k_amm = "k_amm" | |
45 | 49 | ||
46 | 50 | let DECIMAL_UNIT = (1 * (((((((10 * 10) * 10) * 10) * 10) * 10) * 10) * 10)) | |
47 | 51 | ||
48 | 52 | let A = 70000000 | |
49 | 53 | ||
50 | 54 | let MS_IN_WEEK = (1000 * 604800) | |
51 | 55 | ||
52 | 56 | func listToStr (_list) = { | |
53 | 57 | func _join (accumulator,val) = ((accumulator + val) + ",") | |
54 | 58 | ||
55 | 59 | let newListStr = { | |
56 | 60 | let $l = _list | |
57 | 61 | let $s = size($l) | |
58 | 62 | let $acc0 = "" | |
59 | 63 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
60 | 64 | then $a | |
61 | 65 | else _join($a, $l[$i]) | |
62 | 66 | ||
63 | 67 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
64 | 68 | then $a | |
65 | 69 | else throw("List size exceeds 100") | |
66 | 70 | ||
67 | 71 | $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) | |
68 | 72 | } | |
69 | 73 | let newListStrU = dropRight(newListStr, 1) | |
70 | 74 | let newListStrR = if ((take(newListStrU, 1) == ",")) | |
71 | 75 | then drop(newListStrU, 1) | |
72 | 76 | else newListStrU | |
73 | 77 | newListStrR | |
74 | 78 | } | |
75 | 79 | ||
76 | 80 | ||
77 | 81 | func strToList (_str) = if ((_str == "")) | |
78 | 82 | then nil | |
79 | 83 | else split(_str, ",") | |
80 | 84 | ||
81 | 85 | ||
82 | 86 | func divd (_x,_y) = fraction(_x, DECIMAL_UNIT, _y, HALFEVEN) | |
83 | 87 | ||
84 | 88 | ||
85 | 89 | func muld (_x,_y) = fraction(_x, _y, DECIMAL_UNIT, HALFEVEN) | |
86 | 90 | ||
87 | 91 | ||
88 | 92 | func powd (_x,_y) = pow(_x, 8, _y, 8, 8, HALFEVEN) | |
89 | 93 | ||
90 | 94 | ||
91 | 95 | func abs (_x) = if ((_x > 0)) | |
92 | 96 | then _x | |
93 | 97 | else -(_x) | |
94 | 98 | ||
95 | 99 | ||
96 | 100 | func minv (_x,_y) = if ((_x > _y)) | |
97 | 101 | then _y | |
98 | 102 | else _x | |
99 | 103 | ||
100 | 104 | ||
101 | 105 | func toCompositeKey (_key,_address) = ((_key + "_") + _address) | |
102 | 106 | ||
103 | 107 | ||
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") | |
105 | 111 | ||
106 | 112 | ||
107 | 113 | func adminPublicKey () = fromBase58String(getStringValue(coordinator(), k_admin_public_key)) | |
108 | 114 | ||
109 | 115 | ||
110 | 116 | func adminAddress () = addressFromString(getStringValue(coordinator(), k_admin_address)) | |
111 | 117 | ||
112 | 118 | ||
113 | 119 | func isWhitelist (_address) = valueOrElse(getBoolean(coordinator(), toCompositeKey(k_amm, _address)), false) | |
114 | 120 | ||
115 | 121 | ||
116 | - | func int (k) = valueOrErrorMessage(getInteger( | |
122 | + | func int (k) = valueOrErrorMessage(getInteger(that, k), ("no value for " + k)) | |
117 | 123 | ||
118 | 124 | ||
119 | - | func int0 (k) = valueOrElse(getInteger( | |
125 | + | func int0 (k) = valueOrElse(getInteger(that, k), 0) | |
120 | 126 | ||
121 | 127 | ||
122 | - | func initialized () = valueOrElse(getBoolean( | |
128 | + | func initialized () = valueOrElse(getBoolean(that, k_initialized), false) | |
123 | 129 | ||
124 | 130 | ||
125 | - | func | |
131 | + | func getRewardAssetIds () = strToList(valueOrElse(getString(that, k_rewardAssetIds), "")) | |
126 | 132 | ||
127 | 133 | ||
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)) | |
129 | 143 | ||
130 | 144 | ||
131 | 145 | func getLastNotionalKey (_amm,_trader) = ((((k_lastNotional + "_") + _amm) + "_") + _trader) | |
132 | - | ||
133 | - | ||
134 | - | func getLastDateInPeriodKey (_amm,_trader,_weekId) = ((((((k_lastDateInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId)) | |
135 | 146 | ||
136 | 147 | ||
137 | 148 | func getTraderScoreInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderScoreInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId)) | |
138 | 149 | ||
139 | 150 | ||
140 | 151 | func getTotalScoreInPeriodKey (_amm,_weekId) = ((((k_totalScoreInPeriod + "_") + _amm) + "_") + toString(_weekId)) | |
141 | 152 | ||
142 | 153 | ||
143 | 154 | func getTraderAverageNotionalInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderAverageNotionalInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId)) | |
144 | 155 | ||
145 | 156 | ||
146 | 157 | func getAmmRewardRateKey (_amm,_assetId) = ((((k_ammRewardRate + "_") + _amm) + "_") + _assetId) | |
147 | 158 | ||
148 | 159 | ||
149 | - | func | |
160 | + | func getAssetMaxAmountPerPeriodKey (_assetId) = ((k_assetMaxAmountPerPeriod + "_") + _assetId) | |
150 | 161 | ||
151 | 162 | ||
152 | 163 | func getClaimedTraderAssetPeriodKey (_trader,_assetId,_period) = ((((((k_claimedAssetAndPeriod + "_") + _trader) + "_") + _assetId) + "_") + toString(_period)) | |
153 | 164 | ||
154 | 165 | ||
155 | 166 | func getOraclePriceKey (_assetId,_period) = ((((k_oraclePrice + "_") + toString(_period)) + "_") + _assetId) | |
156 | 167 | ||
157 | 168 | ||
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 | - | } | |
169 | + | func getTotalClaimedForTraderAndAssetKey (_trader,_assetId) = ((((k_totalClaimedForTraderAndAsset + "_") + _trader) + "_") + _assetId) | |
163 | 170 | ||
164 | 171 | ||
165 | - | func updateAmmData (_amm,_trader,_weekId,_traderLastDateInPeriod,_traderAverage,_traderLastNotional,_traderScore,_totalScore) = { | |
166 | - | let totalScoreInPeriodKey = getTotalScoreInPeriodKey(_amm, _weekId) | |
167 | - | let traderScoreInPeriodKey = getTraderScoreInPeriodKey(_amm, _trader, _weekId) | |
168 | - | let lastDateInPeriodKey = getLastDateInPeriodKey(_amm, _trader, _weekId) | |
169 | - | let lastNotionalKey = getLastNotionalKey(_amm, _trader) | |
170 | - | let traderAverageNotionalInPeriodKey = getTraderAverageNotionalInPeriodKey(_amm, _trader, _weekId) | |
171 | - | [IntegerEntry(totalScoreInPeriodKey, _totalScore), IntegerEntry(traderScoreInPeriodKey, _traderScore), IntegerEntry(lastDateInPeriodKey, _traderLastDateInPeriod), IntegerEntry(lastNotionalKey, _traderLastNotional), IntegerEntry(traderAverageNotionalInPeriodKey, _traderAverage)] | |
172 | - | } | |
172 | + | func getAmmMaxAmountPerPeriodKey (_amm,_assetId) = ((((k_ammMaxAmountPerPeriod + "_") + _amm) + "_") + _assetId) | |
173 | 173 | ||
174 | 174 | ||
175 | 175 | func usdnToDecimals (_amount) = (_amount * 100) | |
176 | 176 | ||
177 | 177 | ||
178 | 178 | func getWeekStart (_weekId) = (_weekId * MS_IN_WEEK) | |
179 | 179 | ||
180 | 180 | ||
181 | 181 | func getWeekEnd (_weekId) = ((_weekId + 1) * MS_IN_WEEK) | |
182 | 182 | ||
183 | 183 | ||
184 | 184 | func getWeekId (_ts) = (_ts / MS_IN_WEEK) | |
185 | 185 | ||
186 | 186 | ||
187 | 187 | func getFeesInPeriod (_amm,_trader,_weekId) = { | |
188 | - | let key = | |
189 | - | | |
188 | + | let key = getTraderFeesInPeriodKey(_amm, _trader, _weekId) | |
189 | + | int0(key) | |
190 | 190 | } | |
191 | 191 | ||
192 | 192 | ||
193 | 193 | func getTotalFeesInPeriod (_amm,_weekId) = { | |
194 | - | let key = getTotalFeesInPeriodKey(_amm, _weekId) | |
195 | - | 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) | |
196 | 202 | } | |
197 | 203 | ||
198 | 204 | ||
199 | 205 | func getLastNotional (_amm,_trader) = { | |
200 | 206 | let key = getLastNotionalKey(_amm, _trader) | |
201 | - | valueOrElse(getInteger(this, key), 0) | |
202 | - | } | |
203 | - | ||
204 | - | ||
205 | - | func getLastDateInPeriod (_amm,_trader,_weekId,_weekStart) = { | |
206 | - | let key = getLastDateInPeriodKey(_amm, _trader, _weekId) | |
207 | - | valueOrElse(getInteger(this, key), _weekStart) | |
207 | + | int0(key) | |
208 | 208 | } | |
209 | 209 | ||
210 | 210 | ||
211 | 211 | func getTraderScoreInPeriod (_amm,_trader,_weekId) = { | |
212 | 212 | let key = getTraderScoreInPeriodKey(_amm, _trader, _weekId) | |
213 | - | | |
213 | + | int0(key) | |
214 | 214 | } | |
215 | 215 | ||
216 | 216 | ||
217 | 217 | func getTotalScoreInPeriod (_amm,_weekId) = { | |
218 | 218 | let key = getTotalScoreInPeriodKey(_amm, _weekId) | |
219 | - | | |
219 | + | int0(key) | |
220 | 220 | } | |
221 | 221 | ||
222 | 222 | ||
223 | - | func getAmms () = strToList(valueOrElse(getString(this, k_amms), "")) | |
224 | - | ||
225 | - | ||
226 | - | func getRewardAssetIds () = strToList(valueOrElse(getString(this, k_rewardAssetIds), "")) | |
223 | + | func getAmms () = strToList(valueOrElse(getString(that, k_amms), "")) | |
227 | 224 | ||
228 | 225 | ||
229 | 226 | func getAmmRewardRate (_amm,_assetId) = { | |
230 | 227 | let key = getAmmRewardRateKey(_amm, _assetId) | |
231 | - | | |
228 | + | int0(key) | |
232 | 229 | } | |
233 | 230 | ||
234 | 231 | ||
235 | - | func getAmmMaxAmountPerPeriod (_amm,_assetId) = { | |
236 | - | let key = getAmmMaxAmountPerPeriodKey(_amm, _assetId) | |
237 | - | valueOrElse(getInteger(this, key), 0) | |
232 | + | func getAssetMaxAmountPerPeriod (_assetId) = { | |
233 | + | let key = getAssetMaxAmountPerPeriodKey(_assetId) | |
234 | + | int0(key) | |
235 | + | } | |
236 | + | ||
237 | + | ||
238 | + | func getTotalClaimedForTraderAndAsset (_trader,_assetId) = { | |
239 | + | let key = getTotalClaimedForTraderAndAssetKey(_trader, _assetId) | |
240 | + | int0(key) | |
238 | 241 | } | |
239 | 242 | ||
240 | 243 | ||
241 | 244 | func getRewardAssetPrice (_assetId,_weekId) = { | |
242 | - | let oracleAddressStr = valueOrErrorMessage(getString( | |
245 | + | let oracleAddressStr = valueOrErrorMessage(getString(that, k_oracleAddress), "Oracle not set") | |
243 | 246 | let oracleAddress = valueOrErrorMessage(addressFromString(oracleAddressStr), "Invalid oracle address") | |
244 | 247 | let priceKey = getOraclePriceKey(_assetId, _weekId) | |
245 | 248 | 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 | + | } | |
246 | 265 | } | |
247 | 266 | ||
248 | 267 | ||
249 | 268 | func rewardForTraderInAssetInPeriod (_amm,_trader,_assetId,_weekId) = { | |
250 | 269 | let rewardAssetRate = getAmmRewardRate(_amm, _assetId) | |
251 | 270 | if ((rewardAssetRate == 0)) | |
252 | 271 | then 0 | |
253 | 272 | else { | |
254 | 273 | let totalFeesInPeriod = getTotalFeesInPeriod(_amm, _weekId) | |
255 | 274 | let rewardAssetPrice = getRewardAssetPrice(_assetId, _weekId) | |
256 | - | let maxAmountOfAssetToDistribute = | |
275 | + | let maxAmountOfAssetToDistribute = getMaxAmountOfAssetToDistribute(_amm, _assetId, _weekId) | |
257 | 276 | let traderScore = getTraderScoreInPeriod(_amm, _trader, _weekId) | |
258 | 277 | let totalTraderScore = getTotalScoreInPeriod(_amm, _weekId) | |
259 | 278 | if ((totalTraderScore == 0)) | |
260 | 279 | then 0 | |
261 | 280 | else { | |
262 | - | let totalAssetToDistribute = muld | |
281 | + | let totalAssetToDistribute = divd(muld(totalFeesInPeriod, rewardAssetRate), rewardAssetPrice) | |
263 | 282 | let actualTotalAssetToDistribute = minv(totalAssetToDistribute, maxAmountOfAssetToDistribute) | |
264 | 283 | let traderShareInRewards = divd(muld(actualTotalAssetToDistribute, traderScore), totalTraderScore) | |
265 | 284 | traderShareInRewards | |
266 | 285 | } | |
267 | 286 | } | |
268 | 287 | } | |
269 | 288 | ||
270 | 289 | ||
271 | 290 | func totalRewardForTraderInAssetInPeriod (_trader,_assetId,_weekId) = { | |
272 | 291 | let amms = getAmms() | |
273 | 292 | func compute (_acc,_amm) = (_acc + rewardForTraderInAssetInPeriod(_amm, _trader, _assetId, _weekId)) | |
274 | 293 | ||
275 | 294 | let $l = amms | |
276 | 295 | let $s = size($l) | |
277 | 296 | let $acc0 = 0 | |
278 | 297 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
279 | 298 | then $a | |
280 | 299 | else compute($a, $l[$i]) | |
281 | 300 | ||
282 | 301 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
283 | 302 | then $a | |
284 | 303 | else throw("List size exceeds 20") | |
285 | 304 | ||
286 | 305 | $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) | |
287 | 306 | } | |
288 | 307 | ||
289 | 308 | ||
290 | 309 | func totalRewardForTraderInAsset (_trader,_assetId,_periods) = { | |
291 | 310 | func compute (_acc,_weekId) = (_acc + totalRewardForTraderInAssetInPeriod(_trader, _assetId, _weekId)) | |
292 | 311 | ||
293 | 312 | let $l = _periods | |
294 | 313 | let $s = size($l) | |
295 | 314 | let $acc0 = 0 | |
296 | 315 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
297 | 316 | then $a | |
298 | 317 | else compute($a, $l[$i]) | |
299 | 318 | ||
300 | 319 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
301 | 320 | then $a | |
302 | - | else throw("List size exceeds | |
321 | + | else throw("List size exceeds 12") | |
303 | 322 | ||
304 | - | $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) | |
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) | |
324 | + | } | |
325 | + | ||
326 | + | ||
327 | + | func getTraderAverageNotionalInPeriod (_amm,_trader,_weekId,_defaultValue) = { | |
328 | + | let key = getTraderAverageNotionalInPeriodKey(_amm, _trader, _weekId) | |
329 | + | valueOrElse(getInteger(that, key), _defaultValue) | |
305 | 330 | } | |
306 | 331 | ||
307 | 332 | ||
308 | 333 | func isClaimed (_trader,_assetId,_period) = { | |
309 | 334 | let key = getClaimedTraderAssetPeriodKey(_trader, _assetId, _period) | |
310 | - | 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 | |
345 | + | } | |
346 | + | ||
347 | + | ||
348 | + | func claimAllRewardForPeriodForTrader (_trader,_assetId,_periods) = { | |
349 | + | let periods = strToList(_periods) | |
350 | + | func checkAndFilterFn (_acc,_next) = { | |
351 | + | let period = valueOrErrorMessage(parseInt(_next), ("Invalid period: " + _next)) | |
352 | + | if (if (containsElement(_acc, period)) | |
353 | + | then true | |
354 | + | else isClaimed(_trader, _assetId, period)) | |
355 | + | then _acc | |
356 | + | else (_acc :+ period) | |
357 | + | } | |
358 | + | ||
359 | + | let validPeriods = { | |
360 | + | let $l = periods | |
361 | + | let $s = size($l) | |
362 | + | let $acc0 = nil | |
363 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
364 | + | then $a | |
365 | + | else checkAndFilterFn($a, $l[$i]) | |
366 | + | ||
367 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
368 | + | then $a | |
369 | + | else throw("List size exceeds 12") | |
370 | + | ||
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) | |
372 | + | } | |
373 | + | func markPeriodAsDoneFn (_acc,_period) = (_acc :+ BooleanEntry(getClaimedTraderAssetPeriodKey(_trader, _assetId, _period), true)) | |
374 | + | ||
375 | + | let markPeriodsAsClaimed = { | |
376 | + | let $l = validPeriods | |
377 | + | let $s = size($l) | |
378 | + | let $acc0 = nil | |
379 | + | func $f1_1 ($a,$i) = if (($i >= $s)) | |
380 | + | then $a | |
381 | + | else markPeriodAsDoneFn($a, $l[$i]) | |
382 | + | ||
383 | + | func $f1_2 ($a,$i) = if (($i >= $s)) | |
384 | + | then $a | |
385 | + | else throw("List size exceeds 16") | |
386 | + | ||
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) | |
388 | + | } | |
389 | + | let amount = totalRewardForTraderInAsset(_trader, _assetId, validPeriods) | |
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)] | |
311 | 436 | } | |
312 | 437 | ||
313 | 438 | ||
314 | 439 | @Callable(i) | |
315 | 440 | func initialize (_coordinator,_oracle) = if (initialized()) | |
316 | 441 | then throw("Already initialized") | |
317 | 442 | else [StringEntry(k_coordinatorAddress, _coordinator), StringEntry(k_oracleAddress, _oracle), BooleanEntry(k_initialized, true)] | |
318 | 443 | ||
319 | 444 | ||
320 | 445 | ||
321 | 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))) | |
467 | + | ||
468 | + | ||
469 | + | ||
470 | + | @Callable(i) | |
322 | 471 | func view_claimRewards (_trader,_assetId,_period) = throw(toString(totalRewardForTraderInAsset(_trader, _assetId, [_period]))) | |
323 | 472 | ||
324 | 473 | ||
325 | 474 | ||
326 | 475 | @Callable(i) | |
327 | - | func claimRewards (_assetId,_period) = { | |
476 | + | func view_calcReward (_trader,_amm,_assetId,_period) = throw(toString(rewardForTraderInAssetInPeriod(_amm, _assetId, _trader, _period))) | |
477 | + | ||
478 | + | ||
479 | + | ||
480 | + | @Callable(i) | |
481 | + | func view_claimAllRewards (_trader,_assetId,_periods) = { | |
482 | + | let result = claimAllRewardForPeriodForTrader(_trader, _assetId, _periods) | |
483 | + | throw(((toString(result._1) + ",") + toString(getTotalClaimedForTraderAndAsset(_trader, _assetId)))) | |
484 | + | } | |
485 | + | ||
486 | + | ||
487 | + | ||
488 | + | @Callable(i) | |
489 | + | func claimAllRewards (_assetId,_periods) = { | |
328 | 490 | let trader = toString(i.caller) | |
329 | - | if ( | |
330 | - | then throw(" | |
491 | + | if (contains(_periods, toString(getWeekId(TIME)))) | |
492 | + | then throw("Can not claim rewards for current week") | |
331 | 493 | else { | |
332 | - | let currentWeekId = getWeekId(lastBlock.timestamp) | |
333 | - | if ((_period >= currentWeekId)) | |
334 | - | then throw(("Unable to claim rewards in future periods, current period is " + toString(currentWeekId))) | |
335 | - | else { | |
336 | - | let amount = totalRewardForTraderInAsset(trader, _assetId, [_period]) | |
337 | - | if ((amount == 0)) | |
338 | - | then throw("Nothing to claim") | |
339 | - | else [ScriptTransfer(i.caller, amount, fromBase58String(_assetId)), BooleanEntry(getClaimedTraderAssetPeriodKey(trader, _assetId, _period), true)] | |
340 | - | } | |
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 | |
341 | 500 | } | |
342 | 501 | } | |
343 | 502 | ||
344 | 503 | ||
345 | 504 | ||
346 | 505 | @Callable(i) | |
347 | - | func | |
506 | + | func attachRewardAsset (_assetId,_maxAmountPerPeriod) = if (if (!(initialized())) | |
348 | 507 | then true | |
349 | 508 | 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))) | |
350 | 527 | then throw("Invalid attachRewards params") | |
351 | 528 | else { | |
352 | 529 | let amms = getAmms() | |
353 | 530 | let rewardAssetIds = getRewardAssetIds() | |
354 | 531 | let newAmms = if (containsElement(amms, _amm)) | |
355 | 532 | then amms | |
356 | 533 | else (amms :+ _amm) | |
357 | 534 | let newRewardAssetIds = if (containsElement(rewardAssetIds, _assetId)) | |
358 | 535 | then rewardAssetIds | |
359 | 536 | else (rewardAssetIds :+ _assetId) | |
360 | - | [IntegerEntry(getAmmRewardRateKey(_amm, _assetId), _rewardRate | |
537 | + | [IntegerEntry(getAmmRewardRateKey(_amm, _assetId), _rewardRate), StringEntry(k_amms, listToStr(newAmms)), StringEntry(k_rewardAssetIds, listToStr(newRewardAssetIds))] | |
361 | 538 | } | |
362 | 539 | ||
363 | 540 | ||
364 | 541 | ||
365 | 542 | @Callable(i) | |
366 | 543 | func notifyFees (_trader,_fee) = { | |
367 | 544 | let amm = toString(i.caller) | |
368 | 545 | if (if (!(isWhitelist(amm))) | |
369 | 546 | then true | |
370 | 547 | else !(initialized())) | |
371 | 548 | then throw("Invalid notifyFees params") | |
372 | 549 | else { | |
373 | 550 | let actualFee = usdnToDecimals(_fee) | |
374 | - | let ts = lastBlock.timestamp | |
375 | - | let weekId = getWeekId(ts) | |
376 | - | let feesInPeriod = getFeesInPeriod(amm, _trader, weekId) | |
377 | - | let totalFeesInPeriod = getTotalFeesInPeriod(amm, weekId) | |
378 | - | let newFeesInPeriod = (feesInPeriod + actualFee) | |
379 | - | let newTotalFeesInPeriod = (totalFeesInPeriod + actualFee) | |
380 | - | updateAmmFees(amm, _trader, weekId, newFeesInPeriod, newTotalFeesInPeriod) | |
551 | + | let weekId = getWeekId(TIME) | |
552 | + | updateAmmFees(amm, _trader, weekId, actualFee) | |
381 | 553 | } | |
382 | 554 | } | |
383 | 555 | ||
384 | 556 | ||
385 | 557 | ||
386 | 558 | @Callable(i) | |
387 | 559 | func notifyNotional (_trader,_notional) = { | |
388 | 560 | let amm = toString(i.caller) | |
389 | 561 | if (if (!(isWhitelist(amm))) | |
390 | 562 | then true | |
391 | 563 | else !(initialized())) | |
392 | 564 | then throw("Invalid notifyNotional params") | |
393 | 565 | else { | |
394 | 566 | let actualNotional = usdnToDecimals(_notional) | |
395 | - | let ts = lastBlock.timestamp | |
396 | 567 | let lastNotional = getLastNotional(amm, _trader) | |
397 | - | let weekId = getWeekId( | |
568 | + | let weekId = getWeekId(TIME) | |
398 | 569 | let weekStart = getWeekStart(weekId) | |
399 | 570 | let weekEnd = getWeekEnd(weekId) | |
400 | 571 | let t = (weekEnd - weekStart) | |
401 | - | let | |
402 | - | let | |
572 | + | let oldRunningAverage = getTraderAverageNotionalInPeriod(amm, _trader, weekId, lastNotional) | |
573 | + | let w = divd((weekEnd - TIME), t) | |
403 | 574 | let oldTraderAmmScore = getTraderScoreInPeriod(amm, _trader, weekId) | |
404 | 575 | let oldTotalAmmScore = getTotalScoreInPeriod(amm, weekId) | |
405 | 576 | let fees = getFeesInPeriod(amm, _trader, weekId) | |
406 | - | let newRunningAverage = ( | |
577 | + | let newRunningAverage = ((oldRunningAverage + muld(actualNotional, w)) - muld(lastNotional, w)) | |
407 | 578 | let newTraderAmmScore = muld(powd(fees, A), powd(newRunningAverage, (DECIMAL_UNIT - A))) | |
408 | 579 | let newTotalAmmScoreInPeriod = ((oldTotalAmmScore - oldTraderAmmScore) + newTraderAmmScore) | |
409 | - | updateAmmData(amm, _trader, weekId, | |
580 | + | updateAmmData(amm, _trader, weekId, newRunningAverage, actualNotional, newTraderAmmScore, newTotalAmmScoreInPeriod) | |
410 | 581 | } | |
411 | 582 | } | |
412 | 583 | ||
413 | 584 | ||
414 | 585 | @Verifier(tx) | |
415 | 586 | func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], adminPublicKey()) | |
416 | 587 |
github/deemru/w8io/169f3d6 73.80 ms ◑![]()