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:
OldNewDifferences
33 {-# CONTENT_TYPE DAPP #-}
44 let k_totalFeesInPeriod = "k_totalFeesInPeriod"
55
6+let k_totalAssetFeesInPeriod = "k_totalAssetFeesInPeriod"
7+
68 let k_traderFeesInPeriod = "k_traderFeesInPeriod"
79
810 let k_lastNotional = "k_lastNotional"
9-
10-let k_lastDateInPeriod = "k_lastDateInPeriod"
1111
1212 let k_totalScoreInPeriod = "k_totalScoreInPeriod"
1313
1717
1818 let k_ammRewardRate = "k_ammRewardRate"
1919
20-let k_ammMaxAmountPerPeriod = "k_ammMaxAmountPerPeriod"
20+let k_assetMaxAmountPerPeriod = "k_assetMaxAmountPerPeriod"
2121
2222 let k_rewardAssetIds = "k_rewardAssetIds"
2323
2828 let k_oracleAddress = "k_oracleAddress"
2929
3030 let k_oraclePrice = "price"
31+
32+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)
132-
133-
134-func getLastDateInPeriodKey (_amm,_trader,_weekId) = ((((((k_lastDateInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
135146
136147
137148 func getTraderScoreInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderScoreInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
146157 func getAmmRewardRateKey (_amm,_assetId) = ((((k_ammRewardRate + "_") + _amm) + "_") + _assetId)
147158
148159
149-func getAmmMaxAmountPerPeriodKey (_amm,_assetId) = ((((k_ammMaxAmountPerPeriod + "_") + _amm) + "_") + _assetId)
160+func getAssetMaxAmountPerPeriodKey (_assetId) = ((k_assetMaxAmountPerPeriod + "_") + _assetId)
150161
151162
152163 func getClaimedTraderAssetPeriodKey (_trader,_assetId,_period) = ((((((k_claimedAssetAndPeriod + "_") + _trader) + "_") + _assetId) + "_") + toString(_period))
155166 func getOraclePriceKey (_assetId,_period) = ((((k_oraclePrice + "_") + toString(_period)) + "_") + _assetId)
156167
157168
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)
163170
164171
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)
173173
174174
175175 func usdnToDecimals (_amount) = (_amount * 100)
185185
186186
187187 func getFeesInPeriod (_amm,_trader,_weekId) = {
188- let key = getFeesInPeriodKey(_amm, _trader, _weekId)
189- valueOrElse(getInteger(this, key), 0)
188+ let key = getTraderFeesInPeriodKey(_amm, _trader, _weekId)
189+ int0(key)
190190 }
191191
192192
193193 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)
196202 }
197203
198204
199205 func getLastNotional (_amm,_trader) = {
200206 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)
208208 }
209209
210210
211211 func getTraderScoreInPeriod (_amm,_trader,_weekId) = {
212212 let key = getTraderScoreInPeriodKey(_amm, _trader, _weekId)
213- valueOrElse(getInteger(this, key), 0)
213+ int0(key)
214214 }
215215
216216
217217 func getTotalScoreInPeriod (_amm,_weekId) = {
218218 let key = getTotalScoreInPeriodKey(_amm, _weekId)
219- valueOrElse(getInteger(this, key), 0)
219+ int0(key)
220220 }
221221
222222
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), ""))
227224
228225
229226 func getAmmRewardRate (_amm,_assetId) = {
230227 let key = getAmmRewardRateKey(_amm, _assetId)
231- valueOrElse(getInteger(this, key), 0)
228+ int0(key)
232229 }
233230
234231
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)
238241 }
239242
240243
241244 func getRewardAssetPrice (_assetId,_weekId) = {
242- let oracleAddressStr = valueOrErrorMessage(getString(this, k_oracleAddress), "Oracle not set")
245+ let oracleAddressStr = valueOrErrorMessage(getString(that, k_oracleAddress), "Oracle not set")
243246 let oracleAddress = valueOrErrorMessage(addressFromString(oracleAddressStr), "Invalid oracle address")
244247 let priceKey = getOraclePriceKey(_assetId, _weekId)
245248 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+ }
246265 }
247266
248267
253272 else {
254273 let totalFeesInPeriod = getTotalFeesInPeriod(_amm, _weekId)
255274 let rewardAssetPrice = getRewardAssetPrice(_assetId, _weekId)
256- let maxAmountOfAssetToDistribute = getAmmMaxAmountPerPeriod(_amm, _assetId)
275+ let maxAmountOfAssetToDistribute = getMaxAmountOfAssetToDistribute(_amm, _assetId, _weekId)
257276 let traderScore = getTraderScoreInPeriod(_amm, _trader, _weekId)
258277 let totalTraderScore = getTotalScoreInPeriod(_amm, _weekId)
259278 if ((totalTraderScore == 0))
260279 then 0
261280 else {
262- let totalAssetToDistribute = muld(divd(totalFeesInPeriod, rewardAssetPrice), rewardAssetRate)
281+ let totalAssetToDistribute = divd(muld(totalFeesInPeriod, rewardAssetRate), rewardAssetPrice)
263282 let actualTotalAssetToDistribute = minv(totalAssetToDistribute, maxAmountOfAssetToDistribute)
264283 let traderShareInRewards = divd(muld(actualTotalAssetToDistribute, traderScore), totalTraderScore)
265284 traderShareInRewards
299318
300319 func $f0_2 ($a,$i) = if (($i >= $s))
301320 then $a
302- else throw("List size exceeds 10")
321+ else throw("List size exceeds 12")
303322
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)
305330 }
306331
307332
308333 func isClaimed (_trader,_assetId,_period) = {
309334 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)]
311436 }
312437
313438
319444
320445
321446 @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)
322471 func view_claimRewards (_trader,_assetId,_period) = throw(toString(totalRewardForTraderInAsset(_trader, _assetId, [_period])))
323472
324473
325474
326475 @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) = {
328490 let trader = toString(i.caller)
329- if (isClaimed(trader, _assetId, _period))
330- then throw("Already claimed")
491+ if (contains(_periods, toString(getWeekId(TIME))))
492+ then throw("Can not claim rewards for current week")
331493 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
341500 }
342501 }
343502
344503
345504
346505 @Callable(i)
347-func attachRewards (_amm,_assetId,_maxAmountPerPeriod,_rewardRate) = if (if (!(initialized()))
506+func attachRewardAsset (_assetId,_maxAmountPerPeriod) = if (if (!(initialized()))
348507 then true
349508 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)))
350527 then throw("Invalid attachRewards params")
351528 else {
352529 let amms = getAmms()
357534 let newRewardAssetIds = if (containsElement(rewardAssetIds, _assetId))
358535 then rewardAssetIds
359536 else (rewardAssetIds :+ _assetId)
360-[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))]
361538 }
362539
363540
371548 then throw("Invalid notifyFees params")
372549 else {
373550 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)
381553 }
382554 }
383555
392564 then throw("Invalid notifyNotional params")
393565 else {
394566 let actualNotional = usdnToDecimals(_notional)
395- let ts = lastBlock.timestamp
396567 let lastNotional = getLastNotional(amm, _trader)
397- let weekId = getWeekId(ts)
568+ let weekId = getWeekId(TIME)
398569 let weekStart = getWeekStart(weekId)
399570 let weekEnd = getWeekEnd(weekId)
400571 let t = (weekEnd - weekStart)
401- let lastDateInPeriod = getLastDateInPeriod(amm, _trader, weekId, weekStart)
402- let dt = (ts - lastDateInPeriod)
572+ let oldRunningAverage = getTraderAverageNotionalInPeriod(amm, _trader, weekId, lastNotional)
573+ let w = divd((weekEnd - TIME), t)
403574 let oldTraderAmmScore = getTraderScoreInPeriod(amm, _trader, weekId)
404575 let oldTotalAmmScore = getTotalScoreInPeriod(amm, weekId)
405576 let fees = getFeesInPeriod(amm, _trader, weekId)
406- let newRunningAverage = (muld(lastNotional, divd(dt, t)) + muld(actualNotional, divd((t - dt), t)))
577+ let newRunningAverage = ((oldRunningAverage + muld(actualNotional, w)) - muld(lastNotional, w))
407578 let newTraderAmmScore = muld(powd(fees, A), powd(newRunningAverage, (DECIMAL_UNIT - A)))
408579 let newTotalAmmScoreInPeriod = ((oldTotalAmmScore - oldTraderAmmScore) + newTraderAmmScore)
409- updateAmmData(amm, _trader, weekId, ts, newRunningAverage, actualNotional, newTraderAmmScore, newTotalAmmScoreInPeriod)
580+ updateAmmData(amm, _trader, weekId, newRunningAverage, actualNotional, newTraderAmmScore, newTotalAmmScoreInPeriod)
410581 }
411582 }
412583
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let k_totalFeesInPeriod = "k_totalFeesInPeriod"
55
6+let k_totalAssetFeesInPeriod = "k_totalAssetFeesInPeriod"
7+
68 let k_traderFeesInPeriod = "k_traderFeesInPeriod"
79
810 let k_lastNotional = "k_lastNotional"
9-
10-let k_lastDateInPeriod = "k_lastDateInPeriod"
1111
1212 let k_totalScoreInPeriod = "k_totalScoreInPeriod"
1313
1414 let k_traderScoreInPeriod = "k_traderScoreInPeriod"
1515
1616 let k_traderAverageNotionalInPeriod = "k_traderAverageNotionalInPeriod"
1717
1818 let k_ammRewardRate = "k_ammRewardRate"
1919
20-let k_ammMaxAmountPerPeriod = "k_ammMaxAmountPerPeriod"
20+let k_assetMaxAmountPerPeriod = "k_assetMaxAmountPerPeriod"
2121
2222 let k_rewardAssetIds = "k_rewardAssetIds"
2323
2424 let k_amms = "k_amms"
2525
2626 let k_claimedAssetAndPeriod = "k_claimedAssetAndPeriod"
2727
2828 let k_oracleAddress = "k_oracleAddress"
2929
3030 let k_oraclePrice = "price"
31+
32+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)
132-
133-
134-func getLastDateInPeriodKey (_amm,_trader,_weekId) = ((((((k_lastDateInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
135146
136147
137148 func getTraderScoreInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderScoreInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
138149
139150
140151 func getTotalScoreInPeriodKey (_amm,_weekId) = ((((k_totalScoreInPeriod + "_") + _amm) + "_") + toString(_weekId))
141152
142153
143154 func getTraderAverageNotionalInPeriodKey (_amm,_trader,_weekId) = ((((((k_traderAverageNotionalInPeriod + "_") + _amm) + "_") + _trader) + "_") + toString(_weekId))
144155
145156
146157 func getAmmRewardRateKey (_amm,_assetId) = ((((k_ammRewardRate + "_") + _amm) + "_") + _assetId)
147158
148159
149-func getAmmMaxAmountPerPeriodKey (_amm,_assetId) = ((((k_ammMaxAmountPerPeriod + "_") + _amm) + "_") + _assetId)
160+func getAssetMaxAmountPerPeriodKey (_assetId) = ((k_assetMaxAmountPerPeriod + "_") + _assetId)
150161
151162
152163 func getClaimedTraderAssetPeriodKey (_trader,_assetId,_period) = ((((((k_claimedAssetAndPeriod + "_") + _trader) + "_") + _assetId) + "_") + toString(_period))
153164
154165
155166 func getOraclePriceKey (_assetId,_period) = ((((k_oraclePrice + "_") + toString(_period)) + "_") + _assetId)
156167
157168
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)
163170
164171
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)
173173
174174
175175 func usdnToDecimals (_amount) = (_amount * 100)
176176
177177
178178 func getWeekStart (_weekId) = (_weekId * MS_IN_WEEK)
179179
180180
181181 func getWeekEnd (_weekId) = ((_weekId + 1) * MS_IN_WEEK)
182182
183183
184184 func getWeekId (_ts) = (_ts / MS_IN_WEEK)
185185
186186
187187 func getFeesInPeriod (_amm,_trader,_weekId) = {
188- let key = getFeesInPeriodKey(_amm, _trader, _weekId)
189- valueOrElse(getInteger(this, key), 0)
188+ let key = getTraderFeesInPeriodKey(_amm, _trader, _weekId)
189+ int0(key)
190190 }
191191
192192
193193 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)
196202 }
197203
198204
199205 func getLastNotional (_amm,_trader) = {
200206 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)
208208 }
209209
210210
211211 func getTraderScoreInPeriod (_amm,_trader,_weekId) = {
212212 let key = getTraderScoreInPeriodKey(_amm, _trader, _weekId)
213- valueOrElse(getInteger(this, key), 0)
213+ int0(key)
214214 }
215215
216216
217217 func getTotalScoreInPeriod (_amm,_weekId) = {
218218 let key = getTotalScoreInPeriodKey(_amm, _weekId)
219- valueOrElse(getInteger(this, key), 0)
219+ int0(key)
220220 }
221221
222222
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), ""))
227224
228225
229226 func getAmmRewardRate (_amm,_assetId) = {
230227 let key = getAmmRewardRateKey(_amm, _assetId)
231- valueOrElse(getInteger(this, key), 0)
228+ int0(key)
232229 }
233230
234231
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)
238241 }
239242
240243
241244 func getRewardAssetPrice (_assetId,_weekId) = {
242- let oracleAddressStr = valueOrErrorMessage(getString(this, k_oracleAddress), "Oracle not set")
245+ let oracleAddressStr = valueOrErrorMessage(getString(that, k_oracleAddress), "Oracle not set")
243246 let oracleAddress = valueOrErrorMessage(addressFromString(oracleAddressStr), "Invalid oracle address")
244247 let priceKey = getOraclePriceKey(_assetId, _weekId)
245248 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+ }
246265 }
247266
248267
249268 func rewardForTraderInAssetInPeriod (_amm,_trader,_assetId,_weekId) = {
250269 let rewardAssetRate = getAmmRewardRate(_amm, _assetId)
251270 if ((rewardAssetRate == 0))
252271 then 0
253272 else {
254273 let totalFeesInPeriod = getTotalFeesInPeriod(_amm, _weekId)
255274 let rewardAssetPrice = getRewardAssetPrice(_assetId, _weekId)
256- let maxAmountOfAssetToDistribute = getAmmMaxAmountPerPeriod(_amm, _assetId)
275+ let maxAmountOfAssetToDistribute = getMaxAmountOfAssetToDistribute(_amm, _assetId, _weekId)
257276 let traderScore = getTraderScoreInPeriod(_amm, _trader, _weekId)
258277 let totalTraderScore = getTotalScoreInPeriod(_amm, _weekId)
259278 if ((totalTraderScore == 0))
260279 then 0
261280 else {
262- let totalAssetToDistribute = muld(divd(totalFeesInPeriod, rewardAssetPrice), rewardAssetRate)
281+ let totalAssetToDistribute = divd(muld(totalFeesInPeriod, rewardAssetRate), rewardAssetPrice)
263282 let actualTotalAssetToDistribute = minv(totalAssetToDistribute, maxAmountOfAssetToDistribute)
264283 let traderShareInRewards = divd(muld(actualTotalAssetToDistribute, traderScore), totalTraderScore)
265284 traderShareInRewards
266285 }
267286 }
268287 }
269288
270289
271290 func totalRewardForTraderInAssetInPeriod (_trader,_assetId,_weekId) = {
272291 let amms = getAmms()
273292 func compute (_acc,_amm) = (_acc + rewardForTraderInAssetInPeriod(_amm, _trader, _assetId, _weekId))
274293
275294 let $l = amms
276295 let $s = size($l)
277296 let $acc0 = 0
278297 func $f0_1 ($a,$i) = if (($i >= $s))
279298 then $a
280299 else compute($a, $l[$i])
281300
282301 func $f0_2 ($a,$i) = if (($i >= $s))
283302 then $a
284303 else throw("List size exceeds 20")
285304
286305 $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)
287306 }
288307
289308
290309 func totalRewardForTraderInAsset (_trader,_assetId,_periods) = {
291310 func compute (_acc,_weekId) = (_acc + totalRewardForTraderInAssetInPeriod(_trader, _assetId, _weekId))
292311
293312 let $l = _periods
294313 let $s = size($l)
295314 let $acc0 = 0
296315 func $f0_1 ($a,$i) = if (($i >= $s))
297316 then $a
298317 else compute($a, $l[$i])
299318
300319 func $f0_2 ($a,$i) = if (($i >= $s))
301320 then $a
302- else throw("List size exceeds 10")
321+ else throw("List size exceeds 12")
303322
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)
305330 }
306331
307332
308333 func isClaimed (_trader,_assetId,_period) = {
309334 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)]
311436 }
312437
313438
314439 @Callable(i)
315440 func initialize (_coordinator,_oracle) = if (initialized())
316441 then throw("Already initialized")
317442 else [StringEntry(k_coordinatorAddress, _coordinator), StringEntry(k_oracleAddress, _oracle), BooleanEntry(k_initialized, true)]
318443
319444
320445
321446 @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)
322471 func view_claimRewards (_trader,_assetId,_period) = throw(toString(totalRewardForTraderInAsset(_trader, _assetId, [_period])))
323472
324473
325474
326475 @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) = {
328490 let trader = toString(i.caller)
329- if (isClaimed(trader, _assetId, _period))
330- then throw("Already claimed")
491+ if (contains(_periods, toString(getWeekId(TIME))))
492+ then throw("Can not claim rewards for current week")
331493 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
341500 }
342501 }
343502
344503
345504
346505 @Callable(i)
347-func attachRewards (_amm,_assetId,_maxAmountPerPeriod,_rewardRate) = if (if (!(initialized()))
506+func attachRewardAsset (_assetId,_maxAmountPerPeriod) = if (if (!(initialized()))
348507 then true
349508 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)))
350527 then throw("Invalid attachRewards params")
351528 else {
352529 let amms = getAmms()
353530 let rewardAssetIds = getRewardAssetIds()
354531 let newAmms = if (containsElement(amms, _amm))
355532 then amms
356533 else (amms :+ _amm)
357534 let newRewardAssetIds = if (containsElement(rewardAssetIds, _assetId))
358535 then rewardAssetIds
359536 else (rewardAssetIds :+ _assetId)
360-[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))]
361538 }
362539
363540
364541
365542 @Callable(i)
366543 func notifyFees (_trader,_fee) = {
367544 let amm = toString(i.caller)
368545 if (if (!(isWhitelist(amm)))
369546 then true
370547 else !(initialized()))
371548 then throw("Invalid notifyFees params")
372549 else {
373550 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)
381553 }
382554 }
383555
384556
385557
386558 @Callable(i)
387559 func notifyNotional (_trader,_notional) = {
388560 let amm = toString(i.caller)
389561 if (if (!(isWhitelist(amm)))
390562 then true
391563 else !(initialized()))
392564 then throw("Invalid notifyNotional params")
393565 else {
394566 let actualNotional = usdnToDecimals(_notional)
395- let ts = lastBlock.timestamp
396567 let lastNotional = getLastNotional(amm, _trader)
397- let weekId = getWeekId(ts)
568+ let weekId = getWeekId(TIME)
398569 let weekStart = getWeekStart(weekId)
399570 let weekEnd = getWeekEnd(weekId)
400571 let t = (weekEnd - weekStart)
401- let lastDateInPeriod = getLastDateInPeriod(amm, _trader, weekId, weekStart)
402- let dt = (ts - lastDateInPeriod)
572+ let oldRunningAverage = getTraderAverageNotionalInPeriod(amm, _trader, weekId, lastNotional)
573+ let w = divd((weekEnd - TIME), t)
403574 let oldTraderAmmScore = getTraderScoreInPeriod(amm, _trader, weekId)
404575 let oldTotalAmmScore = getTotalScoreInPeriod(amm, weekId)
405576 let fees = getFeesInPeriod(amm, _trader, weekId)
406- let newRunningAverage = (muld(lastNotional, divd(dt, t)) + muld(actualNotional, divd((t - dt), t)))
577+ let newRunningAverage = ((oldRunningAverage + muld(actualNotional, w)) - muld(lastNotional, w))
407578 let newTraderAmmScore = muld(powd(fees, A), powd(newRunningAverage, (DECIMAL_UNIT - A)))
408579 let newTotalAmmScoreInPeriod = ((oldTotalAmmScore - oldTraderAmmScore) + newTraderAmmScore)
409- updateAmmData(amm, _trader, weekId, ts, newRunningAverage, actualNotional, newTraderAmmScore, newTotalAmmScoreInPeriod)
580+ updateAmmData(amm, _trader, weekId, newRunningAverage, actualNotional, newTraderAmmScore, newTotalAmmScoreInPeriod)
410581 }
411582 }
412583
413584
414585 @Verifier(tx)
415586 func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], adminPublicKey())
416587

github/deemru/w8io/c3f4982 
89.56 ms