tx · DATb5227M6E8p7rs9Bh67aYyvTHz6bYPCCKzx4KaEdfs 3MsxHxruYWoddB4HRiPBYAWtMXMtCF1V9XT: -0.02600000 Waves 2022.08.23 09:22 [2196761] smart account 3MsxHxruYWoddB4HRiPBYAWtMXMtCF1V9XT > SELF 0.00000000 Waves
{ "type": 13, "id": "DATb5227M6E8p7rs9Bh67aYyvTHz6bYPCCKzx4KaEdfs", "fee": 2600000, "feeAssetId": null, "timestamp": 1661234759177, "version": 2, "chainId": 84, "sender": "3MsxHxruYWoddB4HRiPBYAWtMXMtCF1V9XT", "senderPublicKey": "3ijdmxaYrpDFsKVbQH2kvB7i6JzHj9bep9bGWiQPra2D", "proofs": [ "5b7fA5wxCAefy7bSs8NHnUMpKW4yzqghcQTaAQqhdmrkgQSQQSuj6fhzJxpVqPHuQTQcnR9cfRwuqJv1PxuUyvVB", "EXiyG1aFEaCNdxdz7dgmoztHWnRCrtriB9Tgw1E3DAdCahnmLnpbMJRLL1VWDReuMZVfwNmNANf9mkhzaMDa5Ut" ], "script": "base64:BgIlCAISAwoBCBIDCgEIEgMKAQgSAwoBCBIECgIIARIDCgEIEgASAFsADGFkbWluUHViS2V5MQEg4qeMQDuGzRfmtEuH2+Whg6yuKqHsNy5eZQUT8rXs7wQADGFkbWluUHViS2V5MgEg6jisuQG1iDxyo54oPYHUGiJlERON346DjXz9V/GbEVkADGFkbWluUHViS2V5MwEgpzSWgrCjycddMmIBfztFJ08z6r82xAHPELd0cKonzG4AFGtleVNoYXJlVG9rZW5zTG9ja2VkAhpfdG90YWxfc2hhcmVfdG9rZW5zX2xvY2tlZAALa1NoYXJlTGltaXQCHHNoYXJlX2xpbWl0X29uX2ZpcnN0X2hhcnZlc3QACWtleUFjdGl2ZQIGYWN0aXZlAAhrZXlDYXVzZQIOc2h1dGRvd25fY2F1c2UAHGtleVJld2FyZFBvb2xGcmFjdGlvbkN1cnJlbnQCHV9jdXJyZW50X3Bvb2xfZnJhY3Rpb25fcmV3YXJkAB1rZXlSZXdhcmRQb29sRnJhY3Rpb25QcmV2aW91cwIeX3ByZXZpb3VzX3Bvb2xfZnJhY3Rpb25fcmV3YXJkABVrZXlIZWlnaHRQb29sRnJhY3Rpb24CGl9wb29sX3Jld2FyZF91cGRhdGVfaGVpZ2h0AB1rZXlUb3RhbFJld2FyZFBlckJsb2NrQ3VycmVudAIedG90YWxfcmV3YXJkX3Blcl9ibG9ja19jdXJyZW50AB5rZXlUb3RhbFJld2FyZFBlckJsb2NrUHJldmlvdXMCH3RvdGFsX3Jld2FyZF9wZXJfYmxvY2tfcHJldmlvdXMAFWtleVJld2FyZFVwZGF0ZUhlaWdodAIUcmV3YXJkX3VwZGF0ZV9oZWlnaHQAD2tleUxhc3RJbnRlcmVzdAIOX2xhc3RfaW50ZXJlc3QAFWtleUxhc3RJbnRlcmVzdEhlaWdodAIVX2xhc3RfaW50ZXJlc3RfaGVpZ2h0ABhrZXlVc2VyU2hhcmVUb2tlbnNMb2NrZWQCFF9zaGFyZV90b2tlbnNfbG9ja2VkABNrZXlVc2VyTGFzdEludGVyZXN0Ag5fbGFzdF9pbnRlcmVzdAAJa2V5U1dPUGlkAgdTV09QX2lkABhrZXlVc2VyU1dPUENsYWltZWRBbW91bnQCFF9TV09QX2NsYWltZWRfYW1vdW50ABxrZXlVc2VyU1dPUExhc3RDbGFpbWVkQW1vdW50AhlfU1dPUF9sYXN0X2NsYWltZWRfYW1vdW50ABBrZXlBdmFpbGFibGVTV09QAg9fYXZhaWxhYmxlX1NXT1AAFWtleUZhcm1pbmdTdGFydEhlaWdodAIUZmFybWluZ19zdGFydF9oZWlnaHQABmtleUFQWQIDYXB5ABZrUHJldmlvdXNUb3RhbFZvdGVTV09QAhhwcmV2aW91c190b3RhbF92b3RlX1NXT1AAE2tleVN3b3BZZWFyRW1pc3Npb24CEnN3b3BfeWVhcl9lbWlzc2lvbgAPa2V5QmFsYW5jZWNwbW1BAg9BX2Fzc2V0X2JhbGFuY2UAD2tleUJhbGFuY2VjcG1tQgIPQl9hc3NldF9iYWxhbmNlACFrSGFydmVzdFBvb2xBY3RpdmVWb3RlU3RydWNWb3RpbmcCHl9oYXJ2ZXN0X3Bvb2xfYWN0aXZlVm90ZV9zdHJ1YwAla0hhcnZlc3RVc2VyUG9vbEFjdGl2ZVZvdGVTdHJ1Y1ZvdGluZwIjX2hhcnZlc3RfdXNlcl9wb29sX2FjdGl2ZVZvdGVfc3RydWMAGWtleUxpbWl0U2hhcmVGaXJzdEhhcnZlc3QCHHNoYXJlX2xpbWl0X29uX2ZpcnN0X2hhcnZlc3QAC2tleUFzc2V0SWRBAgpBX2Fzc2V0X2lkAAtrZXlBc3NldElkQgIKQl9hc3NldF9pZAAVa2V5Rmlyc3RIYXJ2ZXN0SGVpZ2h0AhRmaXJzdF9oYXJ2ZXN0X2hlaWdodAATa2V5Zmlyc3RIYXJ2ZXN0Q3BtbQINZmlyc3RfaGFydmVzdAAOa2V5VGVtcFByZXZTdW0CE3N1bV9yZXdhcmRfcHJldmlvdXMADWtleVRlbXBDdXJTdW0CEnN1bV9yZXdhcmRfY3VycmVudAARZ292ZXJuYW5jZUFkZHJlc3MJAQdBZGRyZXNzAQEaAVSq+mnegfQTUoXKPy46iFGzGeqYHFnyDS0ABndhbGxldAkBB0FkZHJlc3MBARoBVN89S5eJWFneFeDkUQtW3T16X721f4UhiQANdm90aW5nQWRkcmVzcwkBB0FkZHJlc3MBARoBVBo2ft9Xop0naVz+v/cm1Of7ocNoPiDKfwAcYWRtaW5JbmNyZWFzZUludGVyZXN0QWRkcmVzcwkBB0FkZHJlc3MBARoBVN89S5eJWFneFeDkUQtW3T16X721f4UhiQAOb25lV2Vla0luQmxvY2sA+k4ADnRvdGFsVm90ZVNoYXJlAIDIr6AlAAtzY2FsZVZhbHVlMQAKAAtzY2FsZVZhbHVlMwDoBwALc2NhbGVWYWx1ZTUAoI0GAAtzY2FsZVZhbHVlNgDAhD0AC3NjYWxlVmFsdWU4AIDC1y8ADHNjYWxlVmFsdWUxMQCA0NvD9AIBC3N0ckFzc2V0SWRBAQRwb29sCQERQGV4dHJOYXRpdmUoMTA1MykCBQRwb29sBQtrZXlBc3NldElkQQELc3RyQXNzZXRJZEIBBHBvb2wJARFAZXh0ck5hdGl2ZSgxMDUzKQIFBHBvb2wFC2tleUFzc2V0SWRCAQhhc3NldElkQQEEcG9vbAMJAAACCQELc3RyQXNzZXRJZEEBBQRwb29sAgVXQVZFUwUEdW5pdAkA2QQBCQELc3RyQXNzZXRJZEEBBQRwb29sAQhhc3NldElkQgEEcG9vbAMJAAACCQELc3RyQXNzZXRJZEIBBQRwb29sAgVXQVZFUwUEdW5pdAkA2QQBCQELc3RyQXNzZXRJZEIBBQRwb29sAAtrQmFzZVBlcmlvZAILYmFzZV9wZXJpb2QADWtQZXJpb2RMZW5ndGgCDXBlcmlvZF9sZW5ndGgADGtTdGFydEhlaWdodAIMc3RhcnRfaGVpZ2h0ABNrRmlyc3RIYXJ2ZXN0SGVpZ2h0AhRmaXJzdF9oYXJ2ZXN0X2hlaWdodAAWa0R1cmF0aW9uRnVsbFZvdGVQb3dlcgIYZHVyYXRpb25fZnVsbF92b3RlX3Bvd2VyAA1rTWluVm90ZVBvd2VyAg5taW5fdm90ZV9wb3dlcgAKYmFzZVBlcmlvZAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCaCAIFDXZvdGluZ0FkZHJlc3MFC2tCYXNlUGVyaW9kAhFFbXB0eSBrQmFzZVBlcmlvZAALc3RhcnRIZWlnaHQJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQ12b3RpbmdBZGRyZXNzBQxrU3RhcnRIZWlnaHQCEkVtcHR5IGtTdGFydEhlaWdodAAMcGVyaW9kTGVuZ3RoCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUNdm90aW5nQWRkcmVzcwUNa1BlcmlvZExlbmd0aAITRW1wdHkga1BlcmlvZExlbmd0aAAVZHVyYXRpb25GdWxsVm90ZVBvd2VyCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUNdm90aW5nQWRkcmVzcwUWa0R1cmF0aW9uRnVsbFZvdGVQb3dlcgIcRW1wdHkga0R1cmF0aW9uRnVsbFZvdGVQb3dlcgAMbWluVm90ZVBvd2VyCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUNdm90aW5nQWRkcmVzcwUNa01pblZvdGVQb3dlcgITRW1wdHkga01pblZvdGVQb3dlcgAIaXNBY3RpdmUJARFAZXh0ck5hdGl2ZSgxMDUxKQIFBHRoaXMFCWtleUFjdGl2ZQAKY3VyclBlcmlvZAkAZAIFCmJhc2VQZXJpb2QJAGkCCQBlAgUGaGVpZ2h0BQtzdGFydEhlaWdodAUMcGVyaW9kTGVuZ3RoAQ1nZXRMaW1pdFRva2VuAQRwb29sCQELdmFsdWVPckVsc2UCCQERQGV4dHJOYXRpdmUoMTA1MCkCBQRwb29sBRlrZXlMaW1pdFNoYXJlRmlyc3RIYXJ2ZXN0AAAAA0FQWQkBEUBleHRyTmF0aXZlKDEwNTApAgUEdGhpcwUGa2V5QVBZABBTd29wWWVhckVtaXNzaW9uCQERQGV4dHJOYXRpdmUoMTA1MCkCBQR0aGlzBRNrZXlTd29wWWVhckVtaXNzaW9uAQphc3NldE5hbWVBAQRwb29sBAckbWF0Y2gwCQEIYXNzZXRJZEEBBQRwb29sAwkAAQIFByRtYXRjaDACCkJ5dGVWZWN0b3IEAmlkBQckbWF0Y2gwCAkBBXZhbHVlAQkA7AcBBQJpZARuYW1lAwkAAQIFByRtYXRjaDACBFVuaXQEBXdhdmVzBQckbWF0Y2gwAgVXQVZFUwkAAgECC01hdGNoIGVycm9yAQphc3NldE5hbWVCAQRwb29sBAckbWF0Y2gwCQEIYXNzZXRJZEIBBQRwb29sAwkAAQIFByRtYXRjaDACCkJ5dGVWZWN0b3IEAmlkBQckbWF0Y2gwCAkBBXZhbHVlAQkA7AcBBQJpZARuYW1lAwkAAQIFByRtYXRjaDACBFVuaXQEBXdhdmVzBQckbWF0Y2gwAgVXQVZFUwkAAgECC01hdGNoIGVycm9yAARTV09QCQDZBAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIFBHRoaXMFCWtleVNXT1BpZAEOaXNGaXJzdEhhcnZlc3QBBHBvb2wJAQt2YWx1ZU9yRWxzZQIJAJsIAgUEcG9vbAUTa2V5Zmlyc3RIYXJ2ZXN0Q3BtbQcBFWdldEhlaWdodEZpcnN0SGFydmVzdAEEcG9vbAkBC3ZhbHVlT3JFbHNlAgkAmggCBQRwb29sBRVrZXlGaXJzdEhhcnZlc3RIZWlnaHQAAAELZ2V0QmFsYW5jZUEBBHBvb2wJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQRwb29sBQ9rZXlCYWxhbmNlY3BtbUEJAKwCAgIUTm8gZGF0YSBvbiB0aGUga2V5OiAFD2tleUJhbGFuY2VjcG1tQQELZ2V0QmFsYW5jZUIBBHBvb2wJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQRwb29sBQ9rZXlCYWxhbmNlY3BtbUIJAKwCAgIUTm8gZGF0YSBvbiB0aGUga2V5OiAFD2tleUJhbGFuY2VjcG1tQgESZ2V0U2hhcmVMaW1pdFRva2VuAQRwb29sCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUEcG9vbAULa1NoYXJlTGltaXQJAKwCAgIUTm8gZGF0YSBvbiB0aGUga2V5OiAFC2tTaGFyZUxpbWl0ARhnZXRUb3RhbFNoYXJlVG9rZW5Mb2NrZWQBBHBvb2wJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQR0aGlzCQCsAgIFBHBvb2wFFGtleVNoYXJlVG9rZW5zTG9ja2VkCQCsAgIJAKwCAgIUTm8gZGF0YSBvbiB0aGUga2V5OiAFBHBvb2wFFGtleVNoYXJlVG9rZW5zTG9ja2VkAQ9nZXRTaGFyZUFzc2V0SWQBBHBvb2wJANkEAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBBXZhbHVlAQkApggBBQRwb29sAg5zaGFyZV9hc3NldF9pZAEOYWNjb3VudEJhbGFuY2UBB2Fzc2V0SWQEByRtYXRjaDAFB2Fzc2V0SWQDCQABAgUHJG1hdGNoMAIKQnl0ZVZlY3RvcgQCaWQFByRtYXRjaDAJAPAHAgUEdGhpcwUCaWQDCQABAgUHJG1hdGNoMAIEVW5pdAQFd2F2ZXMFByRtYXRjaDAICQDvBwEFBHRoaXMJYXZhaWxhYmxlCQACAQILTWF0Y2ggZXJyb3IBDGdldEFzc2V0SW5mbwEHYXNzZXRJZAQHJG1hdGNoMAUHYXNzZXRJZAMJAAECBQckbWF0Y2gwAgpCeXRlVmVjdG9yBAJpZAUHJG1hdGNoMAQIc3RyaW5nSWQJANgEAQUCaWQEBGluZm8JARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkA7AcBBQJpZAkArAICCQCsAgICBkFzc2V0IAUIc3RyaW5nSWQCDiBkb2Vzbid0IGV4aXN0CQCVCgMFCHN0cmluZ0lkCAUEaW5mbwRuYW1lCAUEaW5mbwhkZWNpbWFscwMJAAECBQckbWF0Y2gwAgRVbml0BAV3YXZlcwUHJG1hdGNoMAkAlQoDAgVXQVZFUwIFV0FWRVMACAkAAgECC01hdGNoIGVycm9yAQ5jYWxjU2NhbGVWYWx1ZQIIYXNzZXRJZDEIYXNzZXRJZDIEEGFzc2V0SWQxRGVjaW1hbHMICQEFdmFsdWUBCQDsBwEFCGFzc2V0SWQxCGRlY2ltYWxzBBBhc3NldElkMkRlY2ltYWxzCAkBBXZhbHVlAQkA7AcBBQhhc3NldElkMghkZWNpbWFscwQLc2NhbGVEaWdpdHMJAGQCCQBlAgUQYXNzZXRJZDJEZWNpbWFscwUQYXNzZXRJZDFEZWNpbWFscwAICQBsBgAKAAAFC3NjYWxlRGlnaXRzAAAAAAUERE9XTgERdXNlckF2YWlsYWJsZVNXT1ACBHBvb2wEdXNlcgkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQCsAgIJAKwCAgkArAICBQRwb29sAgFfCQClCAEFBHVzZXIFEGtleUF2YWlsYWJsZVNXT1AAAAEKcmV3YXJkSW5mbwEEcG9vbAQadG90YWxSZXdhcmRQZXJCbG9ja0N1cnJlbnQJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBRFnb3Zlcm5hbmNlQWRkcmVzcwUda2V5VG90YWxSZXdhcmRQZXJCbG9ja0N1cnJlbnQJAKwCAgkArAICCQCsAgICFE5vIGRhdGEgb24gdGhlIGtleTogBR1rZXlUb3RhbFJld2FyZFBlckJsb2NrQ3VycmVudAIMIGF0IGFkZHJlc3MgCQClCAEFEWdvdmVybmFuY2VBZGRyZXNzBBt0b3RhbFJld2FyZFBlckJsb2NrUHJldmlvdXMJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBRFnb3Zlcm5hbmNlQWRkcmVzcwUea2V5VG90YWxSZXdhcmRQZXJCbG9ja1ByZXZpb3VzCQCsAgIJAKwCAgkArAICAhRObyBkYXRhIG9uIHRoZSBrZXk6IAUea2V5VG90YWxSZXdhcmRQZXJCbG9ja1ByZXZpb3VzAgwgYXQgYWRkcmVzcyAJAKUIAQURZ292ZXJuYW5jZUFkZHJlc3MEGXJld2FyZFBvb2xGcmFjdGlvbkN1cnJlbnQJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBRFnb3Zlcm5hbmNlQWRkcmVzcwkArAICBQRwb29sBRxrZXlSZXdhcmRQb29sRnJhY3Rpb25DdXJyZW50CQCsAgIJAKwCAgkArAICCQCsAgICFE5vIGRhdGEgb24gdGhlIGtleTogBQRwb29sBRxrZXlSZXdhcmRQb29sRnJhY3Rpb25DdXJyZW50AgwgYXQgYWRkcmVzcyAJAKUIAQURZ292ZXJuYW5jZUFkZHJlc3MEEnJld2FyZFVwZGF0ZUhlaWdodAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCaCAIFEWdvdmVybmFuY2VBZGRyZXNzBRVrZXlSZXdhcmRVcGRhdGVIZWlnaHQJAKwCAgkArAICCQCsAgICFE5vIGRhdGEgb24gdGhlIGtleTogBRVrZXlSZXdhcmRVcGRhdGVIZWlnaHQCDCBhdCBhZGRyZXNzIAkApQgBBRFnb3Zlcm5hbmNlQWRkcmVzcwQWcG9vbFJld2FyZFVwZGF0ZUhlaWdodAkBC3ZhbHVlT3JFbHNlAgkAmggCBRFnb3Zlcm5hbmNlQWRkcmVzcwkArAICBQRwb29sBRVrZXlIZWlnaHRQb29sRnJhY3Rpb24AAAQacmV3YXJkUG9vbEZyYWN0aW9uUHJldmlvdXMJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBRFnb3Zlcm5hbmNlQWRkcmVzcwkArAICBQRwb29sBR1rZXlSZXdhcmRQb29sRnJhY3Rpb25QcmV2aW91cwkArAICCQCsAgIJAKwCAgkArAICAhRObyBkYXRhIG9uIHRoZSBrZXk6IAUEcG9vbAUda2V5UmV3YXJkUG9vbEZyYWN0aW9uUHJldmlvdXMCDCBhdCBhZGRyZXNzIAkApQgBBRFnb3Zlcm5hbmNlQWRkcmVzcwQRcmV3YXJkUG9vbEN1cnJlbnQJAGsDBRp0b3RhbFJld2FyZFBlckJsb2NrQ3VycmVudAUZcmV3YXJkUG9vbEZyYWN0aW9uQ3VycmVudAUOdG90YWxWb3RlU2hhcmUEEnJld2FyZFBvb2xQcmV2aW91cwkAawMFG3RvdGFsUmV3YXJkUGVyQmxvY2tQcmV2aW91cwUacmV3YXJkUG9vbEZyYWN0aW9uUHJldmlvdXMFDnRvdGFsVm90ZVNoYXJlAwMJAGYCBRFyZXdhcmRQb29sQ3VycmVudAUadG90YWxSZXdhcmRQZXJCbG9ja0N1cnJlbnQGCQBmAgUScmV3YXJkUG9vbFByZXZpb3VzBRt0b3RhbFJld2FyZFBlckJsb2NrUHJldmlvdXMJAAIBAmJyZXdhcmRQb29sQ3VycmVudCA+IHRvdGFsUmV3YXJkUGVyQmxvY2tDdXJyZW50IG9yIHJld2FyZFBvb2xQcmV2aW91cyA+IHRvdGFsUmV3YXJkUGVyQmxvY2tQcmV2aW91cwkAlgoEBRFyZXdhcmRQb29sQ3VycmVudAUScmV3YXJkVXBkYXRlSGVpZ2h0BRJyZXdhcmRQb29sUHJldmlvdXMFFnBvb2xSZXdhcmRVcGRhdGVIZWlnaHQBE2dldExhc3RJbnRlcmVzdEluZm8BBHBvb2wEDGxhc3RJbnRlcmVzdAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCaCAIFBHRoaXMJAKwCAgUEcG9vbAUPa2V5TGFzdEludGVyZXN0CQCsAgIJAKwCAgIUTm8gZGF0YSBvbiB0aGUga2V5OiAFBHBvb2wFD2tleUxhc3RJbnRlcmVzdAQSbGFzdEludGVyZXN0SGVpZ2h0CQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAKwCAgUEcG9vbAUVa2V5TGFzdEludGVyZXN0SGVpZ2h0BQZoZWlnaHQJAJQKAgUSbGFzdEludGVyZXN0SGVpZ2h0BQxsYXN0SW50ZXJlc3QBE2dldFVzZXJJbnRlcmVzdEluZm8CBHBvb2wLdXNlckFkZHJlc3MEEHVzZXJMYXN0SW50ZXJlc3QJAJoIAgUEdGhpcwkArAICCQCsAgIJAKwCAgUEcG9vbAIBXwkApQgBBQt1c2VyQWRkcmVzcwUTa2V5VXNlckxhc3RJbnRlcmVzdAQJdXNlclNoYXJlCQCaCAIFBHRoaXMJAKwCAgkArAICCQCsAgIFBHBvb2wCAV8JAKUIAQULdXNlckFkZHJlc3MFGGtleVVzZXJTaGFyZVRva2Vuc0xvY2tlZAQMbGFzdEludGVyZXN0CQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUEdGhpcwkArAICBQRwb29sBQ9rZXlMYXN0SW50ZXJlc3QJAKwCAgkArAICAhRObyBkYXRhIG9uIHRoZSBrZXk6IAUEcG9vbAUPa2V5TGFzdEludGVyZXN0BBV1c2VyTGFzdEludGVyZXN0VmFsdWUEByRtYXRjaDAFEHVzZXJMYXN0SW50ZXJlc3QDCQABAgUHJG1hdGNoMAIDSW50BBB1c2VyTGFzdEludGVyZXN0BQckbWF0Y2gwBRB1c2VyTGFzdEludGVyZXN0BQxsYXN0SW50ZXJlc3QEFXVzZXJTaGFyZVRva2Vuc0Ftb3VudAQHJG1hdGNoMAUJdXNlclNoYXJlAwkAAQIFByRtYXRjaDACA0ludAQJdXNlclNoYXJlBQckbWF0Y2gwBQl1c2VyU2hhcmUAAAkAlAoCBRV1c2VyTGFzdEludGVyZXN0VmFsdWUFFXVzZXJTaGFyZVRva2Vuc0Ftb3VudAEMY2FsY0ludGVyZXN0ChJsYXN0SW50ZXJlc3RIZWlnaHQScmV3YXJkVXBkYXRlSGVpZ2h0FnBvb2xSZXdhcmRVcGRhdGVIZWlnaHQMbGFzdEludGVyZXN0FWN1cnJlbnRSZXdhcmRQZXJCbG9jaxBzaGFyZVRva2VuTG9ja2VkFnByZXZpb3VzUmV3YXJkUGVyQmxvY2sMc2hhcmVBc3NldElkCnNjYWxlVmFsdWUJcG10QW1vdW50AwkAAAIFEHNoYXJlVG9rZW5Mb2NrZWQAAAAAAwkBAiE9AgUWcG9vbFJld2FyZFVwZGF0ZUhlaWdodAAAAwMJAGYCBRJyZXdhcmRVcGRhdGVIZWlnaHQFBmhlaWdodAkAAAIFEnJld2FyZFVwZGF0ZUhlaWdodAUWcG9vbFJld2FyZFVwZGF0ZUhlaWdodAcEBnJld2FyZAkAaAIFFnByZXZpb3VzUmV3YXJkUGVyQmxvY2sJAGUCBQZoZWlnaHQFEmxhc3RJbnRlcmVzdEhlaWdodAkAZAIFDGxhc3RJbnRlcmVzdAkAawMFBnJld2FyZAUKc2NhbGVWYWx1ZQUQc2hhcmVUb2tlbkxvY2tlZAMDCQBmAgUGaGVpZ2h0BRJyZXdhcmRVcGRhdGVIZWlnaHQJAQIhPQIFEnJld2FyZFVwZGF0ZUhlaWdodAUWcG9vbFJld2FyZFVwZGF0ZUhlaWdodAcEBnJld2FyZAkAaAIFFnByZXZpb3VzUmV3YXJkUGVyQmxvY2sJAGUCBQZoZWlnaHQFEmxhc3RJbnRlcmVzdEhlaWdodAkAZAIFDGxhc3RJbnRlcmVzdAkAawMFBnJld2FyZAUKc2NhbGVWYWx1ZQUQc2hhcmVUb2tlbkxvY2tlZAMDAwkAZgIFBmhlaWdodAUScmV3YXJkVXBkYXRlSGVpZ2h0CQAAAgUScmV3YXJkVXBkYXRlSGVpZ2h0BRZwb29sUmV3YXJkVXBkYXRlSGVpZ2h0BwkAZgIFEmxhc3RJbnRlcmVzdEhlaWdodAUScmV3YXJkVXBkYXRlSGVpZ2h0BwQGcmV3YXJkCQBoAgUVY3VycmVudFJld2FyZFBlckJsb2NrCQBlAgUGaGVpZ2h0BRJsYXN0SW50ZXJlc3RIZWlnaHQJAGQCBQxsYXN0SW50ZXJlc3QJAGsDBQZyZXdhcmQFCnNjYWxlVmFsdWUFEHNoYXJlVG9rZW5Mb2NrZWQEKnJld2FyZEFmdGVyTGFzdEludGVyZXN0QmVmb3JlUmVhd2FyZFVwZGF0ZQkAaAIFFnByZXZpb3VzUmV3YXJkUGVyQmxvY2sJAGUCBRJyZXdhcmRVcGRhdGVIZWlnaHQFEmxhc3RJbnRlcmVzdEhlaWdodAQTaW50ZXJlc3RBZnRlclVwZGF0ZQkAZAIFDGxhc3RJbnRlcmVzdAkAawMFKnJld2FyZEFmdGVyTGFzdEludGVyZXN0QmVmb3JlUmVhd2FyZFVwZGF0ZQUKc2NhbGVWYWx1ZQUQc2hhcmVUb2tlbkxvY2tlZAQGcmV3YXJkCQBoAgUVY3VycmVudFJld2FyZFBlckJsb2NrCQBlAgUGaGVpZ2h0BRJyZXdhcmRVcGRhdGVIZWlnaHQJAGQCBRNpbnRlcmVzdEFmdGVyVXBkYXRlCQBrAwUGcmV3YXJkBQpzY2FsZVZhbHVlBRBzaGFyZVRva2VuTG9ja2VkAwkAZgIFEnJld2FyZFVwZGF0ZUhlaWdodAUGaGVpZ2h0BAZyZXdhcmQJAGgCBRZwcmV2aW91c1Jld2FyZFBlckJsb2NrCQBlAgUGaGVpZ2h0BRJsYXN0SW50ZXJlc3RIZWlnaHQJAGQCBQxsYXN0SW50ZXJlc3QJAGsDBQZyZXdhcmQFCnNjYWxlVmFsdWUFEHNoYXJlVG9rZW5Mb2NrZWQDCQBmAgUSbGFzdEludGVyZXN0SGVpZ2h0BRJyZXdhcmRVcGRhdGVIZWlnaHQEBnJld2FyZAkAaAIFFWN1cnJlbnRSZXdhcmRQZXJCbG9jawkAZQIFBmhlaWdodAUSbGFzdEludGVyZXN0SGVpZ2h0CQBkAgUMbGFzdEludGVyZXN0CQBrAwUGcmV3YXJkBQpzY2FsZVZhbHVlBRBzaGFyZVRva2VuTG9ja2VkBCpyZXdhcmRBZnRlckxhc3RJbnRlcmVzdEJlZm9yZVJlYXdhcmRVcGRhdGUJAGgCBRZwcmV2aW91c1Jld2FyZFBlckJsb2NrCQBlAgUScmV3YXJkVXBkYXRlSGVpZ2h0BRJsYXN0SW50ZXJlc3RIZWlnaHQEE2ludGVyZXN0QWZ0ZXJVcGRhdGUJAGQCBQxsYXN0SW50ZXJlc3QJAGsDBSpyZXdhcmRBZnRlckxhc3RJbnRlcmVzdEJlZm9yZVJlYXdhcmRVcGRhdGUFCnNjYWxlVmFsdWUFEHNoYXJlVG9rZW5Mb2NrZWQEBnJld2FyZAkAaAIFFWN1cnJlbnRSZXdhcmRQZXJCbG9jawkAZQIFBmhlaWdodAUScmV3YXJkVXBkYXRlSGVpZ2h0CQBkAgUTaW50ZXJlc3RBZnRlclVwZGF0ZQkAawMFBnJld2FyZAUKc2NhbGVWYWx1ZQUQc2hhcmVUb2tlbkxvY2tlZAEJY2xhaW1DYWxjAwRwb29sBmNhbGxlcglwbXRBbW91bnQEDHNoYXJlQXNzZXRJZAkBD2dldFNoYXJlQXNzZXRJZAEFBHBvb2wECnNjYWxlVmFsdWUJAQ5jYWxjU2NhbGVWYWx1ZQIFBFNXT1AFDHNoYXJlQXNzZXRJZAQQc2hhcmVUb2tlbkxvY2tlZAkBGGdldFRvdGFsU2hhcmVUb2tlbkxvY2tlZAEFBHBvb2wEDSR0MDEzMDkxMTMxNTYJARNnZXRMYXN0SW50ZXJlc3RJbmZvAQUEcG9vbAQSbGFzdEludGVyZXN0SGVpZ2h0CAUNJHQwMTMwOTExMzE1NgJfMQQMbGFzdEludGVyZXN0CAUNJHQwMTMwOTExMzE1NgJfMgQNJHQwMTMxNjExMzI3MwkBCnJld2FyZEluZm8BBQRwb29sBBVjdXJyZW50UmV3YXJkUGVyQmxvY2sIBQ0kdDAxMzE2MTEzMjczAl8xBBJyZXdhcmRVcGRhdGVIZWlnaHQIBQ0kdDAxMzE2MTEzMjczAl8yBBZwcmV2aW91c1Jld2FyZFBlckJsb2NrCAUNJHQwMTMxNjExMzI3MwJfMwQWcG9vbFJld2FyZFVwZGF0ZUhlaWdodAgFDSR0MDEzMTYxMTMyNzMCXzQEDSR0MDEzMjc4MTMzNTcJARNnZXRVc2VySW50ZXJlc3RJbmZvAgUEcG9vbAUGY2FsbGVyBBB1c2VyTGFzdEludGVyZXN0CAUNJHQwMTMyNzgxMzM1NwJfMQQVdXNlclNoYXJlVG9rZW5zQW1vdW50CAUNJHQwMTMyNzgxMzM1NwJfMgQPY3VycmVudEludGVyZXN0CQEMY2FsY0ludGVyZXN0CgUSbGFzdEludGVyZXN0SGVpZ2h0BRJyZXdhcmRVcGRhdGVIZWlnaHQFFnBvb2xSZXdhcmRVcGRhdGVIZWlnaHQFDGxhc3RJbnRlcmVzdAUVY3VycmVudFJld2FyZFBlckJsb2NrBRBzaGFyZVRva2VuTG9ja2VkBRZwcmV2aW91c1Jld2FyZFBlckJsb2NrBQxzaGFyZUFzc2V0SWQFCnNjYWxlVmFsdWUFCXBtdEFtb3VudAQLY2xhaW1BbW91bnQJAGsDBRV1c2VyU2hhcmVUb2tlbnNBbW91bnQJAGUCBQ9jdXJyZW50SW50ZXJlc3QFEHVzZXJMYXN0SW50ZXJlc3QFCnNjYWxlVmFsdWUED3VzZXJOZXdJbnRlcmVzdAUPY3VycmVudEludGVyZXN0CQCWCgQFD3VzZXJOZXdJbnRlcmVzdAUPY3VycmVudEludGVyZXN0BQtjbGFpbUFtb3VudAUVdXNlclNoYXJlVG9rZW5zQW1vdW50ARdjYWxjdWxhdGVQcm90b2NvbFJld2FyZAEEcG9vbAQNJHQwMTM4NzUxMzk0MAkBE2dldExhc3RJbnRlcmVzdEluZm8BBQRwb29sBBJsYXN0SW50ZXJlc3RIZWlnaHQIBQ0kdDAxMzg3NTEzOTQwAl8xBAxsYXN0SW50ZXJlc3QIBQ0kdDAxMzg3NTEzOTQwAl8yBA0kdDAxMzk0NTE0MDU2CQEKcmV3YXJkSW5mbwEFBHBvb2wEFWN1cnJlbnRSZXdhcmRQZXJCbG9jawgFDSR0MDEzOTQ1MTQwNTYCXzEEEnJld2FyZFVwZGF0ZUhlaWdodAgFDSR0MDEzOTQ1MTQwNTYCXzIEFnByZXZpb3VzUmV3YXJkUGVyQmxvY2sIBQ0kdDAxMzk0NTE0MDU2Al8zBBZwb29sUmV3YXJkVXBkYXRlSGVpZ2h0CAUNJHQwMTM5NDUxNDA1NgJfNAQQc2hhcmVUb2tlbkxvY2tlZAkBGGdldFRvdGFsU2hhcmVUb2tlbkxvY2tlZAEFBHBvb2wDAwkAAAIFEHNoYXJlVG9rZW5Mb2NrZWQAAAkAAAIFFnBvb2xSZXdhcmRVcGRhdGVIZWlnaHQAAAcDCQBmAgUScmV3YXJkVXBkYXRlSGVpZ2h0BQZoZWlnaHQEBnJld2FyZAkAaAIFFnByZXZpb3VzUmV3YXJkUGVyQmxvY2sJAGUCBQZoZWlnaHQFEmxhc3RJbnRlcmVzdEhlaWdodAUGcmV3YXJkAwkAZgIFEmxhc3RJbnRlcmVzdEhlaWdodAUScmV3YXJkVXBkYXRlSGVpZ2h0BAZyZXdhcmQJAGgCBRVjdXJyZW50UmV3YXJkUGVyQmxvY2sJAGUCBQZoZWlnaHQFEmxhc3RJbnRlcmVzdEhlaWdodAUGcmV3YXJkBCpyZXdhcmRBZnRlckxhc3RJbnRlcmVzdEJlZm9yZVJlYXdhcmRVcGRhdGUJAGgCBRZwcmV2aW91c1Jld2FyZFBlckJsb2NrCQBlAgUScmV3YXJkVXBkYXRlSGVpZ2h0BRJsYXN0SW50ZXJlc3RIZWlnaHQEBnJld2FyZAkAaAIFFWN1cnJlbnRSZXdhcmRQZXJCbG9jawkAZQIFBmhlaWdodAUScmV3YXJkVXBkYXRlSGVpZ2h0CQBkAgUGcmV3YXJkBSpyZXdhcmRBZnRlckxhc3RJbnRlcmVzdEJlZm9yZVJlYXdhcmRVcGRhdGUDAwkAAAIFEHNoYXJlVG9rZW5Mb2NrZWQAAAkBAiE9AgUWcG9vbFJld2FyZFVwZGF0ZUhlaWdodAAABwMDCQBmAgUScmV3YXJkVXBkYXRlSGVpZ2h0BQZoZWlnaHQJAAACBRJyZXdhcmRVcGRhdGVIZWlnaHQFFnBvb2xSZXdhcmRVcGRhdGVIZWlnaHQHBAZyZXdhcmQJAGgCBRZwcmV2aW91c1Jld2FyZFBlckJsb2NrCQBlAgUGaGVpZ2h0BRJsYXN0SW50ZXJlc3RIZWlnaHQFBnJld2FyZAMDCQBmAgUGaGVpZ2h0BRJyZXdhcmRVcGRhdGVIZWlnaHQJAQIhPQIFEnJld2FyZFVwZGF0ZUhlaWdodAUWcG9vbFJld2FyZFVwZGF0ZUhlaWdodAcEBnJld2FyZAkAaAIFFnByZXZpb3VzUmV3YXJkUGVyQmxvY2sJAGUCBQZoZWlnaHQFEmxhc3RJbnRlcmVzdEhlaWdodAUGcmV3YXJkAwMDCQBmAgUGaGVpZ2h0BRJyZXdhcmRVcGRhdGVIZWlnaHQJAAACBRJyZXdhcmRVcGRhdGVIZWlnaHQFFnBvb2xSZXdhcmRVcGRhdGVIZWlnaHQHCQBmAgUSbGFzdEludGVyZXN0SGVpZ2h0BRJyZXdhcmRVcGRhdGVIZWlnaHQHBAZyZXdhcmQJAGgCBRVjdXJyZW50UmV3YXJkUGVyQmxvY2sJAGUCBQZoZWlnaHQFEmxhc3RJbnRlcmVzdEhlaWdodAUGcmV3YXJkBCpyZXdhcmRBZnRlckxhc3RJbnRlcmVzdEJlZm9yZVJlYXdhcmRVcGRhdGUJAGgCBRZwcmV2aW91c1Jld2FyZFBlckJsb2NrCQBlAgUScmV3YXJkVXBkYXRlSGVpZ2h0BRJsYXN0SW50ZXJlc3RIZWlnaHQEBnJld2FyZAkAaAIFFWN1cnJlbnRSZXdhcmRQZXJCbG9jawkAZQIFBmhlaWdodAUScmV3YXJkVXBkYXRlSGVpZ2h0CQBkAgUGcmV3YXJkBSpyZXdhcmRBZnRlckxhc3RJbnRlcmVzdEJlZm9yZVJlYXdhcmRVcGRhdGUAAAEWY2hlY2tQbXRBc3NldElkQ29ycmVjdAIEcG9vbApwbXRBc3NldElkBBBwb29sU2hhcmVBc3NldElkCQDZBAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQV2YWx1ZQEJAKYIAQUEcG9vbAIOc2hhcmVfYXNzZXRfaWQDCQAAAgUKcG10QXNzZXRJZAUQcG9vbFNoYXJlQXNzZXRJZAYHARhnZXRVc2VyU1dPUENsYWltZWRBbW91bnQCBHBvb2wEdXNlcgkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQCsAgIJAKwCAgkArAICBQRwb29sAgFfCQClCAEFBHVzZXIFGGtleVVzZXJTV09QQ2xhaW1lZEFtb3VudAAAAQdzdXNwZW5kAQVjYXVzZQkAzAgCCQEMQm9vbGVhbkVudHJ5AgUJa2V5QWN0aXZlBwkAzAgCCQELU3RyaW5nRW50cnkCBQhrZXlDYXVzZQUFY2F1c2UFA25pbAgBaQEEaW5pdAEHZWFybHlMUAMJAQlpc0RlZmluZWQBCQCdCAIFBHRoaXMFCWtleVNXT1BpZAkAAgECGFNXT1AgYWxyZWFkeSBpbml0aWFsaXplZAQKaW5pdEFtb3VudACAgOmDsd4WBAlTV09QaXNzdWUJAMIIBQIEU1dPUAITU1dPUCBwcm90b2NvbCB0b2tlbgUKaW5pdEFtb3VudAAIBgQGU1dPUGlkCQC4CAEFCVNXT1Bpc3N1ZQkAzAgCCQEMQm9vbGVhbkVudHJ5AgUJa2V5QWN0aXZlBgkAzAgCCQDCCAUCBFNXT1ACE1NXT1AgcHJvdG9jb2wgdG9rZW4FCmluaXRBbW91bnQACAYJAMwIAgkBC1N0cmluZ0VudHJ5AgUJa2V5U1dPUGlkCQDYBAEFBlNXT1BpZAUDbmlsAWkBFGluaXRQb29sU2hhcmVGYXJtaW5nAQRwb29sAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIrT25seSB0aGUgREFwcCBpdHNlbGYgY2FuIGNhbGwgdGhpcyBmdW5jdGlvbgQNJHQwMTcwMzgxNzE0MQkBCnJld2FyZEluZm8BBQRwb29sBA1jdXJyZW50UmV3YXJkCAUNJHQwMTcwMzgxNzE0MQJfMQQScmV3YXJkVXBkYXRlSGVpZ2h0CAUNJHQwMTcwMzgxNzE0MQJfMgQWcHJldmlvdXNSZXdhcmRQZXJCbG9jawgFDSR0MDE3MDM4MTcxNDECXzMEFnBvb2xSZXdhcmRVcGRhdGVIZWlnaHQIBQ0kdDAxNzAzODE3MTQxAl80CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFFGtleVNoYXJlVG9rZW5zTG9ja2VkAAAJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUEcG9vbAUPa2V5TGFzdEludGVyZXN0AAAJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUEcG9vbAUVa2V5TGFzdEludGVyZXN0SGVpZ2h0BQZoZWlnaHQFA25pbAFpARJ1cGRhdGVQb29sSW50ZXJlc3QBBHBvb2wDCQECIT0CCAUBaQZjYWxsZXIFBndhbGxldAkAAgECLE9ubHkgdGhlIEFkbWluIGl0c2VsZiBjYW4gY2FsbCB0aGlzIGZ1bmN0aW9uAwkBASEBBQhpc0FjdGl2ZQkAAgECH0RBcHAgaXMgaW5hY3RpdmUgYXQgdGhpcyBtb21lbnQEDSR0MDE3NTUwMTc2NzAJAQljbGFpbUNhbGMDBQRwb29sBRxhZG1pbkluY3JlYXNlSW50ZXJlc3RBZGRyZXNzAAAED3VzZXJOZXdJbnRlcmVzdAgFDSR0MDE3NTUwMTc2NzACXzEED2N1cnJlbnRJbnRlcmVzdAgFDSR0MDE3NTUwMTc2NzACXzIEC2NsYWltQW1vdW50CAUNJHQwMTc1NTAxNzY3MAJfMwQVdXNlclNoYXJlVG9rZW5zQW1vdW50CAUNJHQwMTc1NTAxNzY3MAJfNAQNJHQwMTc2NzUxNzc3OAkBCnJld2FyZEluZm8BBQRwb29sBA1jdXJyZW50UmV3YXJkCAUNJHQwMTc2NzUxNzc3OAJfMQQScmV3YXJkVXBkYXRlSGVpZ2h0CAUNJHQwMTc2NzUxNzc3OAJfMgQWcHJldmlvdXNSZXdhcmRQZXJCbG9jawgFDSR0MDE3Njc1MTc3NzgCXzMEFnBvb2xSZXdhcmRVcGRhdGVIZWlnaHQIBQ0kdDAxNzY3NTE3Nzc4Al80CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFD2tleUxhc3RJbnRlcmVzdAUPdXNlck5ld0ludGVyZXN0CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFFWtleUxhc3RJbnRlcmVzdEhlaWdodAUGaGVpZ2h0BQNuaWwBaQEPbG9ja1NoYXJlVG9rZW5zAQRwb29sBA0kdDAxNzk3MDE4MDQ1CQCUCgIICQCRAwIIBQFpCHBheW1lbnRzAAAGYW1vdW50CAkAkQMCCAUBaQhwYXltZW50cwAAB2Fzc2V0SWQECXBtdEFtb3VudAgFDSR0MDE3OTcwMTgwNDUCXzEECnBtdEFzc2V0SWQIBQ0kdDAxNzk3MDE4MDQ1Al8yBA0kdDAxODA1MDE4MTIzCQEMZ2V0QXNzZXRJbmZvAQUKcG10QXNzZXRJZAQNcG10U3RyQXNzZXRJZAgFDSR0MDE4MDUwMTgxMjMCXzEEDHBtdEFzc2V0TmFtZQgFDSR0MDE4MDUwMTgxMjMCXzIEC3BtdERlY2ltYWxzCAUNJHQwMTgwNTAxODEyMwJfMwQNJHQwMTgxMjgxODIzNgkBCWNsYWltQ2FsYwMFBHBvb2wIBQFpBmNhbGxlcgUJcG10QW1vdW50BA91c2VyTmV3SW50ZXJlc3QIBQ0kdDAxODEyODE4MjM2Al8xBA9jdXJyZW50SW50ZXJlc3QIBQ0kdDAxODEyODE4MjM2Al8yBAtjbGFpbUFtb3VudAgFDSR0MDE4MTI4MTgyMzYCXzMEFXVzZXJTaGFyZVRva2Vuc0Ftb3VudAgFDSR0MDE4MTI4MTgyMzYCXzQEEnVzZXJTaGFyZUFtb3VudE5ldwkAZAIFFXVzZXJTaGFyZVRva2Vuc0Ftb3VudAUJcG10QW1vdW50BBFhdmFpbGFibGVGdW5kc05ldwkAZAIJARF1c2VyQXZhaWxhYmxlU1dPUAIFBHBvb2wIBQFpBmNhbGxlcgULY2xhaW1BbW91bnQEEHRvdGFsU2hhcmVBbW91bnQJARhnZXRUb3RhbFNoYXJlVG9rZW5Mb2NrZWQBBQRwb29sBBN0b3RhbFNoYXJlQW1vdW50TmV3CQBkAgUQdG90YWxTaGFyZUFtb3VudAUJcG10QW1vdW50BBF1c2VyQ2xhaW1lZEFtb3VudAkBGGdldFVzZXJTV09QQ2xhaW1lZEFtb3VudAIFBHBvb2wIBQFpBmNhbGxlcgQUdXNlckNsYWltZWRBbW91bnROZXcJAGQCBRF1c2VyQ2xhaW1lZEFtb3VudAULY2xhaW1BbW91bnQECWJhc2VFbnRyeQkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIJAKwCAgUEcG9vbAIBXwkApQgBCAUBaQZjYWxsZXIFE2tleVVzZXJMYXN0SW50ZXJlc3QFD3VzZXJOZXdJbnRlcmVzdAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIJAKwCAgUEcG9vbAIBXwkApQgBCAUBaQZjYWxsZXIFGGtleVVzZXJTaGFyZVRva2Vuc0xvY2tlZAUSdXNlclNoYXJlQW1vdW50TmV3CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFFGtleVNoYXJlVG9rZW5zTG9ja2VkBRN0b3RhbFNoYXJlQW1vdW50TmV3CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFD2tleUxhc3RJbnRlcmVzdAUPY3VycmVudEludGVyZXN0CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFFWtleUxhc3RJbnRlcmVzdEhlaWdodAUGaGVpZ2h0CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgkArAICBQRwb29sAgFfCQClCAEIBQFpBmNhbGxlcgUYa2V5VXNlclNXT1BDbGFpbWVkQW1vdW50BRR1c2VyQ2xhaW1lZEFtb3VudE5ldwkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIJAKwCAgUEcG9vbAIBXwkApQgBCAUBaQZjYWxsZXIFHGtleVVzZXJTV09QTGFzdENsYWltZWRBbW91bnQFC2NsYWltQW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgkArAICBQRwb29sAgFfCQClCAEIBQFpBmNhbGxlcgUQa2V5QXZhaWxhYmxlU1dPUAURYXZhaWxhYmxlRnVuZHNOZXcFA25pbAQNaGFydmVzdFBlcmlvZAkAZQIJAGkCCQBkAgkAZQIJARVnZXRIZWlnaHRGaXJzdEhhcnZlc3QBCQEHQWRkcmVzcwEJANkEAQUEcG9vbAULc3RhcnRIZWlnaHQAAQUMcGVyaW9kTGVuZ3RoAAEEDmFtb3VudE9mVm90aW5nCQC1CQIJARFAZXh0ck5hdGl2ZSgxMDUzKQIFDXZvdGluZ0FkZHJlc3MJAKwCAgkArAICCQCsAgIJAKUIAQgFAWkGY2FsbGVyAgFfBQRwb29sAhBfdXNlcl9wb29sX3N0cnVjAgFfBBBhbW91bnRQb29sU3RyYWN0CQC1CQIJARFAZXh0ck5hdGl2ZSgxMDUzKQIFDXZvdGluZ0FkZHJlc3MJAKwCAgUEcG9vbAILX3Bvb2xfc3RydWMCAV8EHmFtb3VudEFjdGl2ZVZvdGVVc2VyUG9vbFN0cmFjdAkAtQkCCQELdmFsdWVPckVsc2UCCQCdCAIFDXZvdGluZ0FkZHJlc3MJAKwCAgkArAICCQCsAgIJAKUIAQgFAWkGY2FsbGVyAgFfBQRwb29sBSVrSGFydmVzdFVzZXJQb29sQWN0aXZlVm90ZVN0cnVjVm90aW5nAgACAV8EGmFtb3VudFBvb2xBY3RpdmVWb3RlU3RyYWN0CQC1CQIJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUNdm90aW5nQWRkcmVzcwkArAICBQRwb29sBSFrSGFydmVzdFBvb2xBY3RpdmVWb3RlU3RydWNWb3RpbmcCAAIBXwQUdXNlclNoYXJlVG9rZW5Mb2NrZWQFFXVzZXJTaGFyZVRva2Vuc0Ftb3VudAQSdXNlclBvb2xBY3RpdmVWb3RlAwkAAAIJAKQDAQUKY3VyclBlcmlvZAkAkQMCBQ5hbW91bnRPZlZvdGluZwACCQELdmFsdWVPckVsc2UCCQC2CQEJAJEDAgUeYW1vdW50QWN0aXZlVm90ZVVzZXJQb29sU3RyYWN0AAAAAAkBC3ZhbHVlT3JFbHNlAgkAtgkBCQCRAwIFDmFtb3VudE9mVm90aW5nAAEAAAQOcG9vbEFjdGl2ZVZvdGUDCQAAAgkApAMBBQpjdXJyUGVyaW9kCQCRAwIFEGFtb3VudFBvb2xTdHJhY3QAAgkBC3ZhbHVlT3JFbHNlAgkAtgkBCQCRAwIFGmFtb3VudFBvb2xBY3RpdmVWb3RlU3RyYWN0AAAAAAkBC3ZhbHVlT3JFbHNlAgkAtgkBCQCRAwIFEGFtb3VudFBvb2xTdHJhY3QAAQAABA5wcm90b2NvbFJld2FyZAkBF2NhbGN1bGF0ZVByb3RvY29sUmV3YXJkAQUEcG9vbAQPbGltaXRTaGFyZVRva2VuCQESZ2V0U2hhcmVMaW1pdFRva2VuAQkBEUBleHRyTmF0aXZlKDEwNjIpAQUEcG9vbAQKc2hhcmVUb2tlbgkAZQIJAGsDBQ9saW1pdFNoYXJlVG9rZW4FEnVzZXJQb29sQWN0aXZlVm90ZQUOcG9vbEFjdGl2ZVZvdGUFFHVzZXJTaGFyZVRva2VuTG9ja2VkBAphY2NCYWxhbmNlCQEOYWNjb3VudEJhbGFuY2UBBQpwbXRBc3NldElkBARmcmFjCQBrAwBjCQBkAgUKYWNjQmFsYW5jZQUJcG10QW1vdW50AGQDCQBmAgUEZnJhYwUTdG90YWxTaGFyZUFtb3VudE5ldwkAAgECMkJhbGFuY2Ugb2Ygc2hhcmUtdG9rZW4gaXMgZ3JlYXRlciB0aGFuIHRvdGFsQW1vdW50CQACAQIKdGVzdCBlcnJvcgFpARN3aXRoZHJhd1NoYXJlVG9rZW5zAgRwb29sGXNoYXJlVG9rZW5zV2l0aGRyYXdBbW91bnQEDXNoYXJlVG9rZW5zSWQJANkEAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBBXZhbHVlAQkApggBBQRwb29sAg5zaGFyZV9hc3NldF9pZAQNJHQwMjEyOTAyMTM5MAkBCWNsYWltQ2FsYwMFBHBvb2wIBQFpBmNhbGxlcgABBA91c2VyTmV3SW50ZXJlc3QIBQ0kdDAyMTI5MDIxMzkwAl8xBA9jdXJyZW50SW50ZXJlc3QIBQ0kdDAyMTI5MDIxMzkwAl8yBAtjbGFpbUFtb3VudAgFDSR0MDIxMjkwMjEzOTACXzMEFXVzZXJTaGFyZVRva2Vuc0Ftb3VudAgFDSR0MDIxMjkwMjEzOTACXzQEEnVzZXJTaGFyZUFtb3VudE5ldwkAZQIFFXVzZXJTaGFyZVRva2Vuc0Ftb3VudAUZc2hhcmVUb2tlbnNXaXRoZHJhd0Ftb3VudAQRYXZhaWxhYmxlRnVuZHNOZXcJAGQCCQERdXNlckF2YWlsYWJsZVNXT1ACBQRwb29sCAUBaQZjYWxsZXIFC2NsYWltQW1vdW50BBB0b3RhbFNoYXJlQW1vdW50CQEYZ2V0VG90YWxTaGFyZVRva2VuTG9ja2VkAQUEcG9vbAQTdG90YWxTaGFyZUFtb3VudE5ldwkAZQIFEHRvdGFsU2hhcmVBbW91bnQFGXNoYXJlVG9rZW5zV2l0aGRyYXdBbW91bnQEEXVzZXJDbGFpbWVkQW1vdW50CQEYZ2V0VXNlclNXT1BDbGFpbWVkQW1vdW50AgUEcG9vbAgFAWkGY2FsbGVyBBR1c2VyQ2xhaW1lZEFtb3VudE5ldwkAZAIFEXVzZXJDbGFpbWVkQW1vdW50BQtjbGFpbUFtb3VudAMJAGYCBRlzaGFyZVRva2Vuc1dpdGhkcmF3QW1vdW50BRV1c2VyU2hhcmVUb2tlbnNBbW91bnQJAAIBAixXaXRoZHJhdyBhbW91bnQgbW9yZSB0aGVuIHVzZXIgbG9ja2VkIGFtb3VudAMJAQEhAQUIaXNBY3RpdmUJAAIBAh9EQXBwIGlzIGluYWN0aXZlIGF0IHRoaXMgbW9tZW50AwkAZgIFGXNoYXJlVG9rZW5zV2l0aGRyYXdBbW91bnQFFXVzZXJTaGFyZVRva2Vuc0Ftb3VudAkAAgECLFdpdGhkcmF3IGFtb3VudCBtb3JlIHRoZW4gdXNlciBsb2NrZWQgYW1vdW50AwkAZgIJAGsDAGMJAGUCCQEOYWNjb3VudEJhbGFuY2UBBQ1zaGFyZVRva2Vuc0lkBRlzaGFyZVRva2Vuc1dpdGhkcmF3QW1vdW50AGQFE3RvdGFsU2hhcmVBbW91bnROZXcJAAIBAjJCYWxhbmNlIG9mIHNoYXJlLXRva2VuIGlzIGdyZWF0ZXIgdGhhbiB0b3RhbEFtb3VudAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIJAKwCAgUEcG9vbAIBXwkApQgBCAUBaQZjYWxsZXIFE2tleVVzZXJMYXN0SW50ZXJlc3QFD3VzZXJOZXdJbnRlcmVzdAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIJAKwCAgUEcG9vbAIBXwkApQgBCAUBaQZjYWxsZXIFGGtleVVzZXJTaGFyZVRva2Vuc0xvY2tlZAUSdXNlclNoYXJlQW1vdW50TmV3CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFD2tleUxhc3RJbnRlcmVzdAUPY3VycmVudEludGVyZXN0CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFFWtleUxhc3RJbnRlcmVzdEhlaWdodAUGaGVpZ2h0CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFFGtleVNoYXJlVG9rZW5zTG9ja2VkBRN0b3RhbFNoYXJlQW1vdW50TmV3CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgkArAICBQRwb29sAgFfCQClCAEIBQFpBmNhbGxlcgUQa2V5QXZhaWxhYmxlU1dPUAURYXZhaWxhYmxlRnVuZHNOZXcJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICCQCsAgIFBHBvb2wCAV8JAKUIAQgFAWkGY2FsbGVyBRhrZXlVc2VyU1dPUENsYWltZWRBbW91bnQFFHVzZXJDbGFpbWVkQW1vdW50TmV3CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgkArAICBQRwb29sAgFfCQClCAEIBQFpBmNhbGxlcgUca2V5VXNlclNXT1BMYXN0Q2xhaW1lZEFtb3VudAULY2xhaW1BbW91bnQJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyBRlzaGFyZVRva2Vuc1dpdGhkcmF3QW1vdW50BQ1zaGFyZVRva2Vuc0lkBQNuaWwBaQEFY2xhaW0BBHBvb2wEDXNoYXJlVG9rZW5zSWQJANkEAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBBXZhbHVlAQkApggBBQRwb29sAg5zaGFyZV9hc3NldF9pZAQQc2hhcmVUb2tlbkxvY2tlZAkBGGdldFRvdGFsU2hhcmVUb2tlbkxvY2tlZAEFBHBvb2wEDSR0MDIzMzk3MjM0NjIJARNnZXRMYXN0SW50ZXJlc3RJbmZvAQUEcG9vbAQSbGFzdEludGVyZXN0SGVpZ2h0CAUNJHQwMjMzOTcyMzQ2MgJfMQQMbGFzdEludGVyZXN0CAUNJHQwMjMzOTcyMzQ2MgJfMgQNJHQwMjM0NjcyMzU3OQkBCnJld2FyZEluZm8BBQRwb29sBBVjdXJyZW50UmV3YXJkUGVyQmxvY2sIBQ0kdDAyMzQ2NzIzNTc5Al8xBBJyZXdhcmRVcGRhdGVIZWlnaHQIBQ0kdDAyMzQ2NzIzNTc5Al8yBBZwcmV2aW91c1Jld2FyZFBlckJsb2NrCAUNJHQwMjM0NjcyMzU3OQJfMwQWcG9vbFJld2FyZFVwZGF0ZUhlaWdodAgFDSR0MDIzNDY3MjM1NzkCXzQEDSR0MDIzNTg0MjM2ODQJAQljbGFpbUNhbGMDBQRwb29sCAUBaQZjYWxsZXIAAQQPdXNlck5ld0ludGVyZXN0CAUNJHQwMjM1ODQyMzY4NAJfMQQPY3VycmVudEludGVyZXN0CAUNJHQwMjM1ODQyMzY4NAJfMgQLY2xhaW1BbW91bnQIBQ0kdDAyMzU4NDIzNjg0Al8zBBV1c2VyU2hhcmVUb2tlbnNBbW91bnQIBQ0kdDAyMzU4NDIzNjg0Al80BA1hdmFpbGFibGVGdW5kCQBkAgkBEXVzZXJBdmFpbGFibGVTV09QAgUEcG9vbAgFAWkGY2FsbGVyBQtjbGFpbUFtb3VudAQRdXNlckNsYWltZWRBbW91bnQJARhnZXRVc2VyU1dPUENsYWltZWRBbW91bnQCBQRwb29sCAUBaQZjYWxsZXIEFHVzZXJDbGFpbWVkQW1vdW50TmV3CQBkAgURdXNlckNsYWltZWRBbW91bnQFC2NsYWltQW1vdW50AwkAAAIFDWF2YWlsYWJsZUZ1bmQAAAkAAgECGVlvdSBoYXZlIDAgYXZhaWxhYmxlIFNXT1ADCQEBIQEFCGlzQWN0aXZlCQACAQIfREFwcCBpcyBpbmFjdGl2ZSBhdCB0aGlzIG1vbWVudAMJAAACBQ1hdmFpbGFibGVGdW5kAAAJAAIBAhlZb3UgaGF2ZSAwIGF2YWlsYWJsZSBTV09QAwkAZgIJAGsDAGMJAQ5hY2NvdW50QmFsYW5jZQEFDXNoYXJlVG9rZW5zSWQAZAUQc2hhcmVUb2tlbkxvY2tlZAkAAgECMkJhbGFuY2Ugb2Ygc2hhcmUtdG9rZW4gaXMgZ3JlYXRlciB0aGFuIHRvdGFsQW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgkArAICBQRwb29sAgFfCQClCAEIBQFpBmNhbGxlcgUTa2V5VXNlckxhc3RJbnRlcmVzdAUPdXNlck5ld0ludGVyZXN0CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFD2tleUxhc3RJbnRlcmVzdAUPY3VycmVudEludGVyZXN0CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBHBvb2wFFWtleUxhc3RJbnRlcmVzdEhlaWdodAUGaGVpZ2h0CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgkArAICBQRwb29sAgFfCQClCAEIBQFpBmNhbGxlcgUQa2V5QXZhaWxhYmxlU1dPUAAACQDMCAIJAQdSZWlzc3VlAwUEU1dPUAUNYXZhaWxhYmxlRnVuZAYJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICCQCsAgIFBHBvb2wCAV8JAKUIAQgFAWkGY2FsbGVyBRhrZXlVc2VyU1dPUENsYWltZWRBbW91bnQFFHVzZXJDbGFpbWVkQW1vdW50TmV3CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgkArAICBQRwb29sAgFfCQClCAEIBQFpBmNhbGxlcgUca2V5VXNlclNXT1BMYXN0Q2xhaW1lZEFtb3VudAULY2xhaW1BbW91bnQJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyBQ1hdmFpbGFibGVGdW5kBQRTV09QBQNuaWwBaQEIc2h1dGRvd24AAwkBASEBBQhpc0FjdGl2ZQkAAgEJAKwCAgIiREFwcCBpcyBhbHJlYWR5IHN1c3BlbmRlZC4gQ2F1c2U6IAkBC3ZhbHVlT3JFbHNlAgkAnQgCBQR0aGlzBQhrZXlDYXVzZQIadGhlIGNhdXNlIHdhc24ndCBzcGVjaWZpZWQDCQEBIQEJAQ9jb250YWluc0VsZW1lbnQCCQDMCAIFDGFkbWluUHViS2V5MQkAzAgCBQxhZG1pblB1YktleTIJAMwIAgUMYWRtaW5QdWJLZXkzBQNuaWwIBQFpD2NhbGxlclB1YmxpY0tleQkAAgECIU9ubHkgYWRtaW4gY2FuIGNhbGwgdGhpcyBmdW5jdGlvbgkBB3N1c3BlbmQBAg9QYXVzZWQgYnkgYWRtaW4BaQEIYWN0aXZhdGUAAwUIaXNBY3RpdmUJAAIBAhZEQXBwIGlzIGFscmVhZHkgYWN0aXZlAwkBASEBCQEPY29udGFpbnNFbGVtZW50AgkAzAgCBQxhZG1pblB1YktleTEJAMwIAgUMYWRtaW5QdWJLZXkyCQDMCAIFDGFkbWluUHViS2V5MwUDbmlsCAUBaQ9jYWxsZXJQdWJsaWNLZXkJAAIBAiFPbmx5IGFkbWluIGNhbiBjYWxsIHRoaXMgZnVuY3Rpb24JAMwIAgkBDEJvb2xlYW5FbnRyeQIFCWtleUFjdGl2ZQYJAMwIAgkBC0RlbGV0ZUVudHJ5AQUIa2V5Q2F1c2UFA25pbAECdHgBBnZlcmlmeQAEByRtYXRjaDAFAnR4BBJhZG1pblB1YktleTFTaWduZWQDCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwAABQxhZG1pblB1YktleTEAAQAABBJhZG1pblB1YktleTJTaWduZWQDCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwABBQxhZG1pblB1YktleTIAAQAABBJhZG1pblB1YktleTNTaWduZWQDCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwACBQxhZG1pblB1YktleTMAAQAACQBnAgkAZAIJAGQCBRJhZG1pblB1YktleTFTaWduZWQFEmFkbWluUHViS2V5MlNpZ25lZAUSYWRtaW5QdWJLZXkzU2lnbmVkAAImqL2y", "height": 2196761, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: H57c36bdjx5ub7t5EDW2BVLYUmewpWSDr3cXgU41ujgM Next: 5DdpLLmmETV43sHWkzizFEQtwdyoJMJYQLQYYuBTed5M Diff:
Old | New | Differences | |
---|---|---|---|
498 | 498 | let protocolReward = calculateProtocolReward(pool) | |
499 | 499 | let limitShareToken = getShareLimitToken(addressFromStringValue(pool)) | |
500 | 500 | let shareToken = (fraction(limitShareToken, userPoolActiveVote, poolActiveVote) - userShareTokenLocked) | |
501 | - | let frac = fraction(99, (accountBalance(pmtAssetId) + pmtAmount), 100) | |
501 | + | let accBalance = accountBalance(pmtAssetId) | |
502 | + | let frac = fraction(99, (accBalance + pmtAmount), 100) | |
502 | 503 | if ((frac > totalShareAmountNew)) | |
503 | 504 | then throw("Balance of share-token is greater than totalAmount") | |
504 | 505 | else throw("test error") | |
509 | 510 | @Callable(i) | |
510 | 511 | func withdrawShareTokens (pool,shareTokensWithdrawAmount) = { | |
511 | 512 | let shareTokensId = fromBase58String(getStringValue(value(addressFromString(pool)), "share_asset_id")) | |
512 | - | let $ | |
513 | - | let userNewInterest = $ | |
514 | - | let currentInterest = $ | |
515 | - | let claimAmount = $ | |
516 | - | let userShareTokensAmount = $ | |
513 | + | let $t02129021390 = claimCalc(pool, i.caller, 1) | |
514 | + | let userNewInterest = $t02129021390._1 | |
515 | + | let currentInterest = $t02129021390._2 | |
516 | + | let claimAmount = $t02129021390._3 | |
517 | + | let userShareTokensAmount = $t02129021390._4 | |
517 | 518 | let userShareAmountNew = (userShareTokensAmount - shareTokensWithdrawAmount) | |
518 | 519 | let availableFundsNew = (userAvailableSWOP(pool, i.caller) + claimAmount) | |
519 | 520 | let totalShareAmount = getTotalShareTokenLocked(pool) | |
537 | 538 | func claim (pool) = { | |
538 | 539 | let shareTokensId = fromBase58String(getStringValue(value(addressFromString(pool)), "share_asset_id")) | |
539 | 540 | let shareTokenLocked = getTotalShareTokenLocked(pool) | |
540 | - | let $ | |
541 | - | let lastInterestHeight = $ | |
542 | - | let lastInterest = $ | |
543 | - | let $ | |
544 | - | let currentRewardPerBlock = $ | |
545 | - | let rewardUpdateHeight = $ | |
546 | - | let previousRewardPerBlock = $ | |
547 | - | let poolRewardUpdateHeight = $ | |
548 | - | let $ | |
549 | - | let userNewInterest = $ | |
550 | - | let currentInterest = $ | |
551 | - | let claimAmount = $ | |
552 | - | let userShareTokensAmount = $ | |
541 | + | let $t02339723462 = getLastInterestInfo(pool) | |
542 | + | let lastInterestHeight = $t02339723462._1 | |
543 | + | let lastInterest = $t02339723462._2 | |
544 | + | let $t02346723579 = rewardInfo(pool) | |
545 | + | let currentRewardPerBlock = $t02346723579._1 | |
546 | + | let rewardUpdateHeight = $t02346723579._2 | |
547 | + | let previousRewardPerBlock = $t02346723579._3 | |
548 | + | let poolRewardUpdateHeight = $t02346723579._4 | |
549 | + | let $t02358423684 = claimCalc(pool, i.caller, 1) | |
550 | + | let userNewInterest = $t02358423684._1 | |
551 | + | let currentInterest = $t02358423684._2 | |
552 | + | let claimAmount = $t02358423684._3 | |
553 | + | let userShareTokensAmount = $t02358423684._4 | |
553 | 554 | let availableFund = (userAvailableSWOP(pool, i.caller) + claimAmount) | |
554 | 555 | let userClaimedAmount = getUserSWOPClaimedAmount(pool, i.caller) | |
555 | 556 | let userClaimedAmountNew = (userClaimedAmount + claimAmount) |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let adminPubKey1 = base58'GFmKZ2naZFRoCvNbwKAQVGmLb1uBeWGDgFabdGBuZiuy' | |
5 | 5 | ||
6 | 6 | let adminPubKey2 = base58'GmJXRyhRA79g8yUGgKBAVdnFfQFDMjQG98b1MmLDh5kk' | |
7 | 7 | ||
8 | 8 | let adminPubKey3 = base58'CFhbV6h41hVjbGHudGtS3fYUv7QAKRxFQzKNtx4B5PqP' | |
9 | 9 | ||
10 | 10 | let keyShareTokensLocked = "_total_share_tokens_locked" | |
11 | 11 | ||
12 | 12 | let kShareLimit = "share_limit_on_first_harvest" | |
13 | 13 | ||
14 | 14 | let keyActive = "active" | |
15 | 15 | ||
16 | 16 | let keyCause = "shutdown_cause" | |
17 | 17 | ||
18 | 18 | let keyRewardPoolFractionCurrent = "_current_pool_fraction_reward" | |
19 | 19 | ||
20 | 20 | let keyRewardPoolFractionPrevious = "_previous_pool_fraction_reward" | |
21 | 21 | ||
22 | 22 | let keyHeightPoolFraction = "_pool_reward_update_height" | |
23 | 23 | ||
24 | 24 | let keyTotalRewardPerBlockCurrent = "total_reward_per_block_current" | |
25 | 25 | ||
26 | 26 | let keyTotalRewardPerBlockPrevious = "total_reward_per_block_previous" | |
27 | 27 | ||
28 | 28 | let keyRewardUpdateHeight = "reward_update_height" | |
29 | 29 | ||
30 | 30 | let keyLastInterest = "_last_interest" | |
31 | 31 | ||
32 | 32 | let keyLastInterestHeight = "_last_interest_height" | |
33 | 33 | ||
34 | 34 | let keyUserShareTokensLocked = "_share_tokens_locked" | |
35 | 35 | ||
36 | 36 | let keyUserLastInterest = "_last_interest" | |
37 | 37 | ||
38 | 38 | let keySWOPid = "SWOP_id" | |
39 | 39 | ||
40 | 40 | let keyUserSWOPClaimedAmount = "_SWOP_claimed_amount" | |
41 | 41 | ||
42 | 42 | let keyUserSWOPLastClaimedAmount = "_SWOP_last_claimed_amount" | |
43 | 43 | ||
44 | 44 | let keyAvailableSWOP = "_available_SWOP" | |
45 | 45 | ||
46 | 46 | let keyFarmingStartHeight = "farming_start_height" | |
47 | 47 | ||
48 | 48 | let keyAPY = "apy" | |
49 | 49 | ||
50 | 50 | let kPreviousTotalVoteSWOP = "previous_total_vote_SWOP" | |
51 | 51 | ||
52 | 52 | let keySwopYearEmission = "swop_year_emission" | |
53 | 53 | ||
54 | 54 | let keyBalancecpmmA = "A_asset_balance" | |
55 | 55 | ||
56 | 56 | let keyBalancecpmmB = "B_asset_balance" | |
57 | 57 | ||
58 | 58 | let kHarvestPoolActiveVoteStrucVoting = "_harvest_pool_activeVote_struc" | |
59 | 59 | ||
60 | 60 | let kHarvestUserPoolActiveVoteStrucVoting = "_harvest_user_pool_activeVote_struc" | |
61 | 61 | ||
62 | 62 | let keyLimitShareFirstHarvest = "share_limit_on_first_harvest" | |
63 | 63 | ||
64 | 64 | let keyAssetIdA = "A_asset_id" | |
65 | 65 | ||
66 | 66 | let keyAssetIdB = "B_asset_id" | |
67 | 67 | ||
68 | 68 | let keyFirstHarvestHeight = "first_harvest_height" | |
69 | 69 | ||
70 | 70 | let keyfirstHarvestCpmm = "first_harvest" | |
71 | 71 | ||
72 | 72 | let keyTempPrevSum = "sum_reward_previous" | |
73 | 73 | ||
74 | 74 | let keyTempCurSum = "sum_reward_current" | |
75 | 75 | ||
76 | 76 | let governanceAddress = Address(base58'3N5W8da2iiijVieA6qLGo7KzCJj8B19smWU') | |
77 | 77 | ||
78 | 78 | let wallet = Address(base58'3NAGTtZz6WpupSN89NZD5rMZwwziZEg4Kx4') | |
79 | 79 | ||
80 | 80 | let votingAddress = Address(base58'3MrJgdL1GniipErHy44YF9idzLaUL2iX5DQ') | |
81 | 81 | ||
82 | 82 | let adminIncreaseInterestAddress = Address(base58'3NAGTtZz6WpupSN89NZD5rMZwwziZEg4Kx4') | |
83 | 83 | ||
84 | 84 | let oneWeekInBlock = 10106 | |
85 | 85 | ||
86 | 86 | let totalVoteShare = 10000000000 | |
87 | 87 | ||
88 | 88 | let scaleValue1 = 10 | |
89 | 89 | ||
90 | 90 | let scaleValue3 = 1000 | |
91 | 91 | ||
92 | 92 | let scaleValue5 = 100000 | |
93 | 93 | ||
94 | 94 | let scaleValue6 = 1000000 | |
95 | 95 | ||
96 | 96 | let scaleValue8 = 100000000 | |
97 | 97 | ||
98 | 98 | let scaleValue11 = 100000000000 | |
99 | 99 | ||
100 | 100 | func strAssetIdA (pool) = getStringValue(pool, keyAssetIdA) | |
101 | 101 | ||
102 | 102 | ||
103 | 103 | func strAssetIdB (pool) = getStringValue(pool, keyAssetIdB) | |
104 | 104 | ||
105 | 105 | ||
106 | 106 | func assetIdA (pool) = if ((strAssetIdA(pool) == "WAVES")) | |
107 | 107 | then unit | |
108 | 108 | else fromBase58String(strAssetIdA(pool)) | |
109 | 109 | ||
110 | 110 | ||
111 | 111 | func assetIdB (pool) = if ((strAssetIdB(pool) == "WAVES")) | |
112 | 112 | then unit | |
113 | 113 | else fromBase58String(strAssetIdB(pool)) | |
114 | 114 | ||
115 | 115 | ||
116 | 116 | let kBasePeriod = "base_period" | |
117 | 117 | ||
118 | 118 | let kPeriodLength = "period_length" | |
119 | 119 | ||
120 | 120 | let kStartHeight = "start_height" | |
121 | 121 | ||
122 | 122 | let kFirstHarvestHeight = "first_harvest_height" | |
123 | 123 | ||
124 | 124 | let kDurationFullVotePower = "duration_full_vote_power" | |
125 | 125 | ||
126 | 126 | let kMinVotePower = "min_vote_power" | |
127 | 127 | ||
128 | 128 | let basePeriod = valueOrErrorMessage(getInteger(votingAddress, kBasePeriod), "Empty kBasePeriod") | |
129 | 129 | ||
130 | 130 | let startHeight = valueOrErrorMessage(getInteger(votingAddress, kStartHeight), "Empty kStartHeight") | |
131 | 131 | ||
132 | 132 | let periodLength = valueOrErrorMessage(getInteger(votingAddress, kPeriodLength), "Empty kPeriodLength") | |
133 | 133 | ||
134 | 134 | let durationFullVotePower = valueOrErrorMessage(getInteger(votingAddress, kDurationFullVotePower), "Empty kDurationFullVotePower") | |
135 | 135 | ||
136 | 136 | let minVotePower = valueOrErrorMessage(getInteger(votingAddress, kMinVotePower), "Empty kMinVotePower") | |
137 | 137 | ||
138 | 138 | let isActive = getBooleanValue(this, keyActive) | |
139 | 139 | ||
140 | 140 | let currPeriod = (basePeriod + ((height - startHeight) / periodLength)) | |
141 | 141 | ||
142 | 142 | func getLimitToken (pool) = valueOrElse(getIntegerValue(pool, keyLimitShareFirstHarvest), 0) | |
143 | 143 | ||
144 | 144 | ||
145 | 145 | let APY = getIntegerValue(this, keyAPY) | |
146 | 146 | ||
147 | 147 | let SwopYearEmission = getIntegerValue(this, keySwopYearEmission) | |
148 | 148 | ||
149 | 149 | func assetNameA (pool) = match assetIdA(pool) { | |
150 | 150 | case id: ByteVector => | |
151 | 151 | value(assetInfo(id)).name | |
152 | 152 | case waves: Unit => | |
153 | 153 | "WAVES" | |
154 | 154 | case _ => | |
155 | 155 | throw("Match error") | |
156 | 156 | } | |
157 | 157 | ||
158 | 158 | ||
159 | 159 | func assetNameB (pool) = match assetIdB(pool) { | |
160 | 160 | case id: ByteVector => | |
161 | 161 | value(assetInfo(id)).name | |
162 | 162 | case waves: Unit => | |
163 | 163 | "WAVES" | |
164 | 164 | case _ => | |
165 | 165 | throw("Match error") | |
166 | 166 | } | |
167 | 167 | ||
168 | 168 | ||
169 | 169 | let SWOP = fromBase58String(getStringValue(this, keySWOPid)) | |
170 | 170 | ||
171 | 171 | func isFirstHarvest (pool) = valueOrElse(getBoolean(pool, keyfirstHarvestCpmm), false) | |
172 | 172 | ||
173 | 173 | ||
174 | 174 | func getHeightFirstHarvest (pool) = valueOrElse(getInteger(pool, keyFirstHarvestHeight), 0) | |
175 | 175 | ||
176 | 176 | ||
177 | 177 | func getBalanceA (pool) = valueOrErrorMessage(getInteger(pool, keyBalancecpmmA), ("No data on the key: " + keyBalancecpmmA)) | |
178 | 178 | ||
179 | 179 | ||
180 | 180 | func getBalanceB (pool) = valueOrErrorMessage(getInteger(pool, keyBalancecpmmB), ("No data on the key: " + keyBalancecpmmB)) | |
181 | 181 | ||
182 | 182 | ||
183 | 183 | func getShareLimitToken (pool) = valueOrErrorMessage(getInteger(pool, kShareLimit), ("No data on the key: " + kShareLimit)) | |
184 | 184 | ||
185 | 185 | ||
186 | 186 | func getTotalShareTokenLocked (pool) = valueOrErrorMessage(getInteger(this, (pool + keyShareTokensLocked)), (("No data on the key: " + pool) + keyShareTokensLocked)) | |
187 | 187 | ||
188 | 188 | ||
189 | 189 | func getShareAssetId (pool) = fromBase58String(getStringValue(value(addressFromString(pool)), "share_asset_id")) | |
190 | 190 | ||
191 | 191 | ||
192 | 192 | func accountBalance (assetId) = match assetId { | |
193 | 193 | case id: ByteVector => | |
194 | 194 | assetBalance(this, id) | |
195 | 195 | case waves: Unit => | |
196 | 196 | wavesBalance(this).available | |
197 | 197 | case _ => | |
198 | 198 | throw("Match error") | |
199 | 199 | } | |
200 | 200 | ||
201 | 201 | ||
202 | 202 | func getAssetInfo (assetId) = match assetId { | |
203 | 203 | case id: ByteVector => | |
204 | 204 | let stringId = toBase58String(id) | |
205 | 205 | let info = valueOrErrorMessage(assetInfo(id), (("Asset " + stringId) + " doesn't exist")) | |
206 | 206 | $Tuple3(stringId, info.name, info.decimals) | |
207 | 207 | case waves: Unit => | |
208 | 208 | $Tuple3("WAVES", "WAVES", 8) | |
209 | 209 | case _ => | |
210 | 210 | throw("Match error") | |
211 | 211 | } | |
212 | 212 | ||
213 | 213 | ||
214 | 214 | func calcScaleValue (assetId1,assetId2) = { | |
215 | 215 | let assetId1Decimals = value(assetInfo(assetId1)).decimals | |
216 | 216 | let assetId2Decimals = value(assetInfo(assetId2)).decimals | |
217 | 217 | let scaleDigits = ((assetId2Decimals - assetId1Decimals) + 8) | |
218 | 218 | pow(10, 0, scaleDigits, 0, 0, DOWN) | |
219 | 219 | } | |
220 | 220 | ||
221 | 221 | ||
222 | 222 | func userAvailableSWOP (pool,user) = valueOrElse(getInteger(this, (((pool + "_") + toString(user)) + keyAvailableSWOP)), 0) | |
223 | 223 | ||
224 | 224 | ||
225 | 225 | func rewardInfo (pool) = { | |
226 | 226 | let totalRewardPerBlockCurrent = valueOrErrorMessage(getInteger(governanceAddress, keyTotalRewardPerBlockCurrent), ((("No data on the key: " + keyTotalRewardPerBlockCurrent) + " at address ") + toString(governanceAddress))) | |
227 | 227 | let totalRewardPerBlockPrevious = valueOrErrorMessage(getInteger(governanceAddress, keyTotalRewardPerBlockPrevious), ((("No data on the key: " + keyTotalRewardPerBlockPrevious) + " at address ") + toString(governanceAddress))) | |
228 | 228 | let rewardPoolFractionCurrent = valueOrErrorMessage(getInteger(governanceAddress, (pool + keyRewardPoolFractionCurrent)), (((("No data on the key: " + pool) + keyRewardPoolFractionCurrent) + " at address ") + toString(governanceAddress))) | |
229 | 229 | let rewardUpdateHeight = valueOrErrorMessage(getInteger(governanceAddress, keyRewardUpdateHeight), ((("No data on the key: " + keyRewardUpdateHeight) + " at address ") + toString(governanceAddress))) | |
230 | 230 | let poolRewardUpdateHeight = valueOrElse(getInteger(governanceAddress, (pool + keyHeightPoolFraction)), 0) | |
231 | 231 | let rewardPoolFractionPrevious = valueOrErrorMessage(getInteger(governanceAddress, (pool + keyRewardPoolFractionPrevious)), (((("No data on the key: " + pool) + keyRewardPoolFractionPrevious) + " at address ") + toString(governanceAddress))) | |
232 | 232 | let rewardPoolCurrent = fraction(totalRewardPerBlockCurrent, rewardPoolFractionCurrent, totalVoteShare) | |
233 | 233 | let rewardPoolPrevious = fraction(totalRewardPerBlockPrevious, rewardPoolFractionPrevious, totalVoteShare) | |
234 | 234 | if (if ((rewardPoolCurrent > totalRewardPerBlockCurrent)) | |
235 | 235 | then true | |
236 | 236 | else (rewardPoolPrevious > totalRewardPerBlockPrevious)) | |
237 | 237 | then throw("rewardPoolCurrent > totalRewardPerBlockCurrent or rewardPoolPrevious > totalRewardPerBlockPrevious") | |
238 | 238 | else $Tuple4(rewardPoolCurrent, rewardUpdateHeight, rewardPoolPrevious, poolRewardUpdateHeight) | |
239 | 239 | } | |
240 | 240 | ||
241 | 241 | ||
242 | 242 | func getLastInterestInfo (pool) = { | |
243 | 243 | let lastInterest = valueOrErrorMessage(getInteger(this, (pool + keyLastInterest)), (("No data on the key: " + pool) + keyLastInterest)) | |
244 | 244 | let lastInterestHeight = valueOrElse(getInteger(this, (pool + keyLastInterestHeight)), height) | |
245 | 245 | $Tuple2(lastInterestHeight, lastInterest) | |
246 | 246 | } | |
247 | 247 | ||
248 | 248 | ||
249 | 249 | func getUserInterestInfo (pool,userAddress) = { | |
250 | 250 | let userLastInterest = getInteger(this, (((pool + "_") + toString(userAddress)) + keyUserLastInterest)) | |
251 | 251 | let userShare = getInteger(this, (((pool + "_") + toString(userAddress)) + keyUserShareTokensLocked)) | |
252 | 252 | let lastInterest = valueOrErrorMessage(getInteger(this, (pool + keyLastInterest)), (("No data on the key: " + pool) + keyLastInterest)) | |
253 | 253 | let userLastInterestValue = match userLastInterest { | |
254 | 254 | case userLastInterest: Int => | |
255 | 255 | userLastInterest | |
256 | 256 | case _ => | |
257 | 257 | lastInterest | |
258 | 258 | } | |
259 | 259 | let userShareTokensAmount = match userShare { | |
260 | 260 | case userShare: Int => | |
261 | 261 | userShare | |
262 | 262 | case _ => | |
263 | 263 | 0 | |
264 | 264 | } | |
265 | 265 | $Tuple2(userLastInterestValue, userShareTokensAmount) | |
266 | 266 | } | |
267 | 267 | ||
268 | 268 | ||
269 | 269 | func calcInterest (lastInterestHeight,rewardUpdateHeight,poolRewardUpdateHeight,lastInterest,currentRewardPerBlock,shareTokenLocked,previousRewardPerBlock,shareAssetId,scaleValue,pmtAmount) = if ((shareTokenLocked == 0)) | |
270 | 270 | then 0 | |
271 | 271 | else if ((poolRewardUpdateHeight != 0)) | |
272 | 272 | then if (if ((rewardUpdateHeight > height)) | |
273 | 273 | then (rewardUpdateHeight == poolRewardUpdateHeight) | |
274 | 274 | else false) | |
275 | 275 | then { | |
276 | 276 | let reward = (previousRewardPerBlock * (height - lastInterestHeight)) | |
277 | 277 | (lastInterest + fraction(reward, scaleValue, shareTokenLocked)) | |
278 | 278 | } | |
279 | 279 | else if (if ((height > rewardUpdateHeight)) | |
280 | 280 | then (rewardUpdateHeight != poolRewardUpdateHeight) | |
281 | 281 | else false) | |
282 | 282 | then { | |
283 | 283 | let reward = (previousRewardPerBlock * (height - lastInterestHeight)) | |
284 | 284 | (lastInterest + fraction(reward, scaleValue, shareTokenLocked)) | |
285 | 285 | } | |
286 | 286 | else if (if (if ((height > rewardUpdateHeight)) | |
287 | 287 | then (rewardUpdateHeight == poolRewardUpdateHeight) | |
288 | 288 | else false) | |
289 | 289 | then (lastInterestHeight > rewardUpdateHeight) | |
290 | 290 | else false) | |
291 | 291 | then { | |
292 | 292 | let reward = (currentRewardPerBlock * (height - lastInterestHeight)) | |
293 | 293 | (lastInterest + fraction(reward, scaleValue, shareTokenLocked)) | |
294 | 294 | } | |
295 | 295 | else { | |
296 | 296 | let rewardAfterLastInterestBeforeReawardUpdate = (previousRewardPerBlock * (rewardUpdateHeight - lastInterestHeight)) | |
297 | 297 | let interestAfterUpdate = (lastInterest + fraction(rewardAfterLastInterestBeforeReawardUpdate, scaleValue, shareTokenLocked)) | |
298 | 298 | let reward = (currentRewardPerBlock * (height - rewardUpdateHeight)) | |
299 | 299 | (interestAfterUpdate + fraction(reward, scaleValue, shareTokenLocked)) | |
300 | 300 | } | |
301 | 301 | else if ((rewardUpdateHeight > height)) | |
302 | 302 | then { | |
303 | 303 | let reward = (previousRewardPerBlock * (height - lastInterestHeight)) | |
304 | 304 | (lastInterest + fraction(reward, scaleValue, shareTokenLocked)) | |
305 | 305 | } | |
306 | 306 | else if ((lastInterestHeight > rewardUpdateHeight)) | |
307 | 307 | then { | |
308 | 308 | let reward = (currentRewardPerBlock * (height - lastInterestHeight)) | |
309 | 309 | (lastInterest + fraction(reward, scaleValue, shareTokenLocked)) | |
310 | 310 | } | |
311 | 311 | else { | |
312 | 312 | let rewardAfterLastInterestBeforeReawardUpdate = (previousRewardPerBlock * (rewardUpdateHeight - lastInterestHeight)) | |
313 | 313 | let interestAfterUpdate = (lastInterest + fraction(rewardAfterLastInterestBeforeReawardUpdate, scaleValue, shareTokenLocked)) | |
314 | 314 | let reward = (currentRewardPerBlock * (height - rewardUpdateHeight)) | |
315 | 315 | (interestAfterUpdate + fraction(reward, scaleValue, shareTokenLocked)) | |
316 | 316 | } | |
317 | 317 | ||
318 | 318 | ||
319 | 319 | func claimCalc (pool,caller,pmtAmount) = { | |
320 | 320 | let shareAssetId = getShareAssetId(pool) | |
321 | 321 | let scaleValue = calcScaleValue(SWOP, shareAssetId) | |
322 | 322 | let shareTokenLocked = getTotalShareTokenLocked(pool) | |
323 | 323 | let $t01309113156 = getLastInterestInfo(pool) | |
324 | 324 | let lastInterestHeight = $t01309113156._1 | |
325 | 325 | let lastInterest = $t01309113156._2 | |
326 | 326 | let $t01316113273 = rewardInfo(pool) | |
327 | 327 | let currentRewardPerBlock = $t01316113273._1 | |
328 | 328 | let rewardUpdateHeight = $t01316113273._2 | |
329 | 329 | let previousRewardPerBlock = $t01316113273._3 | |
330 | 330 | let poolRewardUpdateHeight = $t01316113273._4 | |
331 | 331 | let $t01327813357 = getUserInterestInfo(pool, caller) | |
332 | 332 | let userLastInterest = $t01327813357._1 | |
333 | 333 | let userShareTokensAmount = $t01327813357._2 | |
334 | 334 | let currentInterest = calcInterest(lastInterestHeight, rewardUpdateHeight, poolRewardUpdateHeight, lastInterest, currentRewardPerBlock, shareTokenLocked, previousRewardPerBlock, shareAssetId, scaleValue, pmtAmount) | |
335 | 335 | let claimAmount = fraction(userShareTokensAmount, (currentInterest - userLastInterest), scaleValue) | |
336 | 336 | let userNewInterest = currentInterest | |
337 | 337 | $Tuple4(userNewInterest, currentInterest, claimAmount, userShareTokensAmount) | |
338 | 338 | } | |
339 | 339 | ||
340 | 340 | ||
341 | 341 | func calculateProtocolReward (pool) = { | |
342 | 342 | let $t01387513940 = getLastInterestInfo(pool) | |
343 | 343 | let lastInterestHeight = $t01387513940._1 | |
344 | 344 | let lastInterest = $t01387513940._2 | |
345 | 345 | let $t01394514056 = rewardInfo(pool) | |
346 | 346 | let currentRewardPerBlock = $t01394514056._1 | |
347 | 347 | let rewardUpdateHeight = $t01394514056._2 | |
348 | 348 | let previousRewardPerBlock = $t01394514056._3 | |
349 | 349 | let poolRewardUpdateHeight = $t01394514056._4 | |
350 | 350 | let shareTokenLocked = getTotalShareTokenLocked(pool) | |
351 | 351 | if (if ((shareTokenLocked == 0)) | |
352 | 352 | then (poolRewardUpdateHeight == 0) | |
353 | 353 | else false) | |
354 | 354 | then if ((rewardUpdateHeight > height)) | |
355 | 355 | then { | |
356 | 356 | let reward = (previousRewardPerBlock * (height - lastInterestHeight)) | |
357 | 357 | reward | |
358 | 358 | } | |
359 | 359 | else if ((lastInterestHeight > rewardUpdateHeight)) | |
360 | 360 | then { | |
361 | 361 | let reward = (currentRewardPerBlock * (height - lastInterestHeight)) | |
362 | 362 | reward | |
363 | 363 | } | |
364 | 364 | else { | |
365 | 365 | let rewardAfterLastInterestBeforeReawardUpdate = (previousRewardPerBlock * (rewardUpdateHeight - lastInterestHeight)) | |
366 | 366 | let reward = (currentRewardPerBlock * (height - rewardUpdateHeight)) | |
367 | 367 | (reward + rewardAfterLastInterestBeforeReawardUpdate) | |
368 | 368 | } | |
369 | 369 | else if (if ((shareTokenLocked == 0)) | |
370 | 370 | then (poolRewardUpdateHeight != 0) | |
371 | 371 | else false) | |
372 | 372 | then if (if ((rewardUpdateHeight > height)) | |
373 | 373 | then (rewardUpdateHeight == poolRewardUpdateHeight) | |
374 | 374 | else false) | |
375 | 375 | then { | |
376 | 376 | let reward = (previousRewardPerBlock * (height - lastInterestHeight)) | |
377 | 377 | reward | |
378 | 378 | } | |
379 | 379 | else if (if ((height > rewardUpdateHeight)) | |
380 | 380 | then (rewardUpdateHeight != poolRewardUpdateHeight) | |
381 | 381 | else false) | |
382 | 382 | then { | |
383 | 383 | let reward = (previousRewardPerBlock * (height - lastInterestHeight)) | |
384 | 384 | reward | |
385 | 385 | } | |
386 | 386 | else if (if (if ((height > rewardUpdateHeight)) | |
387 | 387 | then (rewardUpdateHeight == poolRewardUpdateHeight) | |
388 | 388 | else false) | |
389 | 389 | then (lastInterestHeight > rewardUpdateHeight) | |
390 | 390 | else false) | |
391 | 391 | then { | |
392 | 392 | let reward = (currentRewardPerBlock * (height - lastInterestHeight)) | |
393 | 393 | reward | |
394 | 394 | } | |
395 | 395 | else { | |
396 | 396 | let rewardAfterLastInterestBeforeReawardUpdate = (previousRewardPerBlock * (rewardUpdateHeight - lastInterestHeight)) | |
397 | 397 | let reward = (currentRewardPerBlock * (height - rewardUpdateHeight)) | |
398 | 398 | (reward + rewardAfterLastInterestBeforeReawardUpdate) | |
399 | 399 | } | |
400 | 400 | else 0 | |
401 | 401 | } | |
402 | 402 | ||
403 | 403 | ||
404 | 404 | func checkPmtAssetIdCorrect (pool,pmtAssetId) = { | |
405 | 405 | let poolShareAssetId = fromBase58String(getStringValue(value(addressFromString(pool)), "share_asset_id")) | |
406 | 406 | if ((pmtAssetId == poolShareAssetId)) | |
407 | 407 | then true | |
408 | 408 | else false | |
409 | 409 | } | |
410 | 410 | ||
411 | 411 | ||
412 | 412 | func getUserSWOPClaimedAmount (pool,user) = valueOrElse(getInteger(this, (((pool + "_") + toString(user)) + keyUserSWOPClaimedAmount)), 0) | |
413 | 413 | ||
414 | 414 | ||
415 | 415 | func suspend (cause) = [BooleanEntry(keyActive, false), StringEntry(keyCause, cause)] | |
416 | 416 | ||
417 | 417 | ||
418 | 418 | @Callable(i) | |
419 | 419 | func init (earlyLP) = if (isDefined(getString(this, keySWOPid))) | |
420 | 420 | then throw("SWOP already initialized") | |
421 | 421 | else { | |
422 | 422 | let initAmount = 100000000000000 | |
423 | 423 | let SWOPissue = Issue("SWOP", "SWOP protocol token", initAmount, 8, true) | |
424 | 424 | let SWOPid = calculateAssetId(SWOPissue) | |
425 | 425 | [BooleanEntry(keyActive, true), Issue("SWOP", "SWOP protocol token", initAmount, 8, true), StringEntry(keySWOPid, toBase58String(SWOPid))] | |
426 | 426 | } | |
427 | 427 | ||
428 | 428 | ||
429 | 429 | ||
430 | 430 | @Callable(i) | |
431 | 431 | func initPoolShareFarming (pool) = if ((i.caller != this)) | |
432 | 432 | then throw("Only the DApp itself can call this function") | |
433 | 433 | else { | |
434 | 434 | let $t01703817141 = rewardInfo(pool) | |
435 | 435 | let currentReward = $t01703817141._1 | |
436 | 436 | let rewardUpdateHeight = $t01703817141._2 | |
437 | 437 | let previousRewardPerBlock = $t01703817141._3 | |
438 | 438 | let poolRewardUpdateHeight = $t01703817141._4 | |
439 | 439 | [IntegerEntry((pool + keyShareTokensLocked), 0), IntegerEntry((pool + keyLastInterest), 0), IntegerEntry((pool + keyLastInterestHeight), height)] | |
440 | 440 | } | |
441 | 441 | ||
442 | 442 | ||
443 | 443 | ||
444 | 444 | @Callable(i) | |
445 | 445 | func updatePoolInterest (pool) = if ((i.caller != wallet)) | |
446 | 446 | then throw("Only the Admin itself can call this function") | |
447 | 447 | else if (!(isActive)) | |
448 | 448 | then throw("DApp is inactive at this moment") | |
449 | 449 | else { | |
450 | 450 | let $t01755017670 = claimCalc(pool, adminIncreaseInterestAddress, 0) | |
451 | 451 | let userNewInterest = $t01755017670._1 | |
452 | 452 | let currentInterest = $t01755017670._2 | |
453 | 453 | let claimAmount = $t01755017670._3 | |
454 | 454 | let userShareTokensAmount = $t01755017670._4 | |
455 | 455 | let $t01767517778 = rewardInfo(pool) | |
456 | 456 | let currentReward = $t01767517778._1 | |
457 | 457 | let rewardUpdateHeight = $t01767517778._2 | |
458 | 458 | let previousRewardPerBlock = $t01767517778._3 | |
459 | 459 | let poolRewardUpdateHeight = $t01767517778._4 | |
460 | 460 | [IntegerEntry((pool + keyLastInterest), userNewInterest), IntegerEntry((pool + keyLastInterestHeight), height)] | |
461 | 461 | } | |
462 | 462 | ||
463 | 463 | ||
464 | 464 | ||
465 | 465 | @Callable(i) | |
466 | 466 | func lockShareTokens (pool) = { | |
467 | 467 | let $t01797018045 = $Tuple2(i.payments[0].amount, i.payments[0].assetId) | |
468 | 468 | let pmtAmount = $t01797018045._1 | |
469 | 469 | let pmtAssetId = $t01797018045._2 | |
470 | 470 | let $t01805018123 = getAssetInfo(pmtAssetId) | |
471 | 471 | let pmtStrAssetId = $t01805018123._1 | |
472 | 472 | let pmtAssetName = $t01805018123._2 | |
473 | 473 | let pmtDecimals = $t01805018123._3 | |
474 | 474 | let $t01812818236 = claimCalc(pool, i.caller, pmtAmount) | |
475 | 475 | let userNewInterest = $t01812818236._1 | |
476 | 476 | let currentInterest = $t01812818236._2 | |
477 | 477 | let claimAmount = $t01812818236._3 | |
478 | 478 | let userShareTokensAmount = $t01812818236._4 | |
479 | 479 | let userShareAmountNew = (userShareTokensAmount + pmtAmount) | |
480 | 480 | let availableFundsNew = (userAvailableSWOP(pool, i.caller) + claimAmount) | |
481 | 481 | let totalShareAmount = getTotalShareTokenLocked(pool) | |
482 | 482 | let totalShareAmountNew = (totalShareAmount + pmtAmount) | |
483 | 483 | let userClaimedAmount = getUserSWOPClaimedAmount(pool, i.caller) | |
484 | 484 | let userClaimedAmountNew = (userClaimedAmount + claimAmount) | |
485 | 485 | let baseEntry = [IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserLastInterest), userNewInterest), IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserShareTokensLocked), userShareAmountNew), IntegerEntry((pool + keyShareTokensLocked), totalShareAmountNew), IntegerEntry((pool + keyLastInterest), currentInterest), IntegerEntry((pool + keyLastInterestHeight), height), IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserSWOPClaimedAmount), userClaimedAmountNew), IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserSWOPLastClaimedAmount), claimAmount), IntegerEntry((((pool + "_") + toString(i.caller)) + keyAvailableSWOP), availableFundsNew)] | |
486 | 486 | let harvestPeriod = ((((getHeightFirstHarvest(Address(fromBase58String(pool))) - startHeight) + 1) / periodLength) - 1) | |
487 | 487 | let amountOfVoting = split(getStringValue(votingAddress, (((toString(i.caller) + "_") + pool) + "_user_pool_struc")), "_") | |
488 | 488 | let amountPoolStract = split(getStringValue(votingAddress, (pool + "_pool_struc")), "_") | |
489 | 489 | let amountActiveVoteUserPoolStract = split(valueOrElse(getString(votingAddress, (((toString(i.caller) + "_") + pool) + kHarvestUserPoolActiveVoteStrucVoting)), ""), "_") | |
490 | 490 | let amountPoolActiveVoteStract = split(valueOrElse(getString(votingAddress, (pool + kHarvestPoolActiveVoteStrucVoting)), ""), "_") | |
491 | 491 | let userShareTokenLocked = userShareTokensAmount | |
492 | 492 | let userPoolActiveVote = if ((toString(currPeriod) == amountOfVoting[2])) | |
493 | 493 | then valueOrElse(parseInt(amountActiveVoteUserPoolStract[0]), 0) | |
494 | 494 | else valueOrElse(parseInt(amountOfVoting[1]), 0) | |
495 | 495 | let poolActiveVote = if ((toString(currPeriod) == amountPoolStract[2])) | |
496 | 496 | then valueOrElse(parseInt(amountPoolActiveVoteStract[0]), 0) | |
497 | 497 | else valueOrElse(parseInt(amountPoolStract[1]), 0) | |
498 | 498 | let protocolReward = calculateProtocolReward(pool) | |
499 | 499 | let limitShareToken = getShareLimitToken(addressFromStringValue(pool)) | |
500 | 500 | let shareToken = (fraction(limitShareToken, userPoolActiveVote, poolActiveVote) - userShareTokenLocked) | |
501 | - | let frac = fraction(99, (accountBalance(pmtAssetId) + pmtAmount), 100) | |
501 | + | let accBalance = accountBalance(pmtAssetId) | |
502 | + | let frac = fraction(99, (accBalance + pmtAmount), 100) | |
502 | 503 | if ((frac > totalShareAmountNew)) | |
503 | 504 | then throw("Balance of share-token is greater than totalAmount") | |
504 | 505 | else throw("test error") | |
505 | 506 | } | |
506 | 507 | ||
507 | 508 | ||
508 | 509 | ||
509 | 510 | @Callable(i) | |
510 | 511 | func withdrawShareTokens (pool,shareTokensWithdrawAmount) = { | |
511 | 512 | let shareTokensId = fromBase58String(getStringValue(value(addressFromString(pool)), "share_asset_id")) | |
512 | - | let $ | |
513 | - | let userNewInterest = $ | |
514 | - | let currentInterest = $ | |
515 | - | let claimAmount = $ | |
516 | - | let userShareTokensAmount = $ | |
513 | + | let $t02129021390 = claimCalc(pool, i.caller, 1) | |
514 | + | let userNewInterest = $t02129021390._1 | |
515 | + | let currentInterest = $t02129021390._2 | |
516 | + | let claimAmount = $t02129021390._3 | |
517 | + | let userShareTokensAmount = $t02129021390._4 | |
517 | 518 | let userShareAmountNew = (userShareTokensAmount - shareTokensWithdrawAmount) | |
518 | 519 | let availableFundsNew = (userAvailableSWOP(pool, i.caller) + claimAmount) | |
519 | 520 | let totalShareAmount = getTotalShareTokenLocked(pool) | |
520 | 521 | let totalShareAmountNew = (totalShareAmount - shareTokensWithdrawAmount) | |
521 | 522 | let userClaimedAmount = getUserSWOPClaimedAmount(pool, i.caller) | |
522 | 523 | let userClaimedAmountNew = (userClaimedAmount + claimAmount) | |
523 | 524 | if ((shareTokensWithdrawAmount > userShareTokensAmount)) | |
524 | 525 | then throw("Withdraw amount more then user locked amount") | |
525 | 526 | else if (!(isActive)) | |
526 | 527 | then throw("DApp is inactive at this moment") | |
527 | 528 | else if ((shareTokensWithdrawAmount > userShareTokensAmount)) | |
528 | 529 | then throw("Withdraw amount more then user locked amount") | |
529 | 530 | else if ((fraction(99, (accountBalance(shareTokensId) - shareTokensWithdrawAmount), 100) > totalShareAmountNew)) | |
530 | 531 | then throw("Balance of share-token is greater than totalAmount") | |
531 | 532 | else [IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserLastInterest), userNewInterest), IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserShareTokensLocked), userShareAmountNew), IntegerEntry((pool + keyLastInterest), currentInterest), IntegerEntry((pool + keyLastInterestHeight), height), IntegerEntry((pool + keyShareTokensLocked), totalShareAmountNew), IntegerEntry((((pool + "_") + toString(i.caller)) + keyAvailableSWOP), availableFundsNew), IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserSWOPClaimedAmount), userClaimedAmountNew), IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserSWOPLastClaimedAmount), claimAmount), ScriptTransfer(i.caller, shareTokensWithdrawAmount, shareTokensId)] | |
532 | 533 | } | |
533 | 534 | ||
534 | 535 | ||
535 | 536 | ||
536 | 537 | @Callable(i) | |
537 | 538 | func claim (pool) = { | |
538 | 539 | let shareTokensId = fromBase58String(getStringValue(value(addressFromString(pool)), "share_asset_id")) | |
539 | 540 | let shareTokenLocked = getTotalShareTokenLocked(pool) | |
540 | - | let $ | |
541 | - | let lastInterestHeight = $ | |
542 | - | let lastInterest = $ | |
543 | - | let $ | |
544 | - | let currentRewardPerBlock = $ | |
545 | - | let rewardUpdateHeight = $ | |
546 | - | let previousRewardPerBlock = $ | |
547 | - | let poolRewardUpdateHeight = $ | |
548 | - | let $ | |
549 | - | let userNewInterest = $ | |
550 | - | let currentInterest = $ | |
551 | - | let claimAmount = $ | |
552 | - | let userShareTokensAmount = $ | |
541 | + | let $t02339723462 = getLastInterestInfo(pool) | |
542 | + | let lastInterestHeight = $t02339723462._1 | |
543 | + | let lastInterest = $t02339723462._2 | |
544 | + | let $t02346723579 = rewardInfo(pool) | |
545 | + | let currentRewardPerBlock = $t02346723579._1 | |
546 | + | let rewardUpdateHeight = $t02346723579._2 | |
547 | + | let previousRewardPerBlock = $t02346723579._3 | |
548 | + | let poolRewardUpdateHeight = $t02346723579._4 | |
549 | + | let $t02358423684 = claimCalc(pool, i.caller, 1) | |
550 | + | let userNewInterest = $t02358423684._1 | |
551 | + | let currentInterest = $t02358423684._2 | |
552 | + | let claimAmount = $t02358423684._3 | |
553 | + | let userShareTokensAmount = $t02358423684._4 | |
553 | 554 | let availableFund = (userAvailableSWOP(pool, i.caller) + claimAmount) | |
554 | 555 | let userClaimedAmount = getUserSWOPClaimedAmount(pool, i.caller) | |
555 | 556 | let userClaimedAmountNew = (userClaimedAmount + claimAmount) | |
556 | 557 | if ((availableFund == 0)) | |
557 | 558 | then throw("You have 0 available SWOP") | |
558 | 559 | else if (!(isActive)) | |
559 | 560 | then throw("DApp is inactive at this moment") | |
560 | 561 | else if ((availableFund == 0)) | |
561 | 562 | then throw("You have 0 available SWOP") | |
562 | 563 | else if ((fraction(99, accountBalance(shareTokensId), 100) > shareTokenLocked)) | |
563 | 564 | then throw("Balance of share-token is greater than totalAmount") | |
564 | 565 | else [IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserLastInterest), userNewInterest), IntegerEntry((pool + keyLastInterest), currentInterest), IntegerEntry((pool + keyLastInterestHeight), height), IntegerEntry((((pool + "_") + toString(i.caller)) + keyAvailableSWOP), 0), Reissue(SWOP, availableFund, true), IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserSWOPClaimedAmount), userClaimedAmountNew), IntegerEntry((((pool + "_") + toString(i.caller)) + keyUserSWOPLastClaimedAmount), claimAmount), ScriptTransfer(i.caller, availableFund, SWOP)] | |
565 | 566 | } | |
566 | 567 | ||
567 | 568 | ||
568 | 569 | ||
569 | 570 | @Callable(i) | |
570 | 571 | func shutdown () = if (!(isActive)) | |
571 | 572 | then throw(("DApp is already suspended. Cause: " + valueOrElse(getString(this, keyCause), "the cause wasn't specified"))) | |
572 | 573 | else if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3], i.callerPublicKey))) | |
573 | 574 | then throw("Only admin can call this function") | |
574 | 575 | else suspend("Paused by admin") | |
575 | 576 | ||
576 | 577 | ||
577 | 578 | ||
578 | 579 | @Callable(i) | |
579 | 580 | func activate () = if (isActive) | |
580 | 581 | then throw("DApp is already active") | |
581 | 582 | else if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3], i.callerPublicKey))) | |
582 | 583 | then throw("Only admin can call this function") | |
583 | 584 | else [BooleanEntry(keyActive, true), DeleteEntry(keyCause)] | |
584 | 585 | ||
585 | 586 | ||
586 | 587 | @Verifier(tx) | |
587 | 588 | func verify () = match tx { | |
588 | 589 | case _ => | |
589 | 590 | let adminPubKey1Signed = if (sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey1)) | |
590 | 591 | then 1 | |
591 | 592 | else 0 | |
592 | 593 | let adminPubKey2Signed = if (sigVerify(tx.bodyBytes, tx.proofs[1], adminPubKey2)) | |
593 | 594 | then 1 | |
594 | 595 | else 0 | |
595 | 596 | let adminPubKey3Signed = if (sigVerify(tx.bodyBytes, tx.proofs[2], adminPubKey3)) | |
596 | 597 | then 1 | |
597 | 598 | else 0 | |
598 | 599 | (((adminPubKey1Signed + adminPubKey2Signed) + adminPubKey3Signed) >= 2) | |
599 | 600 | } | |
600 | 601 |
github/deemru/w8io/169f3d6 82.45 ms ◑