tx · 8gijVd7Gv2Fnt3rZZMJ2tT7TMkJ7VEKsu6ZtEKs6xU4C 3N4kXZHGke6yRq3Z57q7BTgCrT2SCvQCYER: -0.01400000 Waves 2022.12.06 13:45 [2348378] smart account 3N4kXZHGke6yRq3Z57q7BTgCrT2SCvQCYER > SELF 0.00000000 Waves
{ "type": 13, "id": "8gijVd7Gv2Fnt3rZZMJ2tT7TMkJ7VEKsu6ZtEKs6xU4C", "fee": 1400000, "feeAssetId": null, "timestamp": 1670323435906, "version": 2, "chainId": 84, "sender": "3N4kXZHGke6yRq3Z57q7BTgCrT2SCvQCYER", "senderPublicKey": "6sxdMr2xYPPACEJccwmgS7nWC2ez1c38Y1kDX49ne42b", "proofs": [ "3srgyW9WWriGcTk5nSFtimJ9JMgs6MQdTeSENp38v9PaW2e1sVoFs1X1mo7wfz4YhUZhikUQt2BG39FvrEpMhAvX" ], "script": "base64:BgIwCAISBgoECAgICBIAEgASABIECgIBARIAEgMKAQgSAwoBCBIECgIBARIGCgQICAEIIgAJU0VQQVJBVE9SAgJfXwAFV0FWRVMCBVdBVkVTAAxLRVlfTVVMVElTSUcCCE1VTFRJU0lHAApLRVlfU1RBVFVTAgZTVEFUVVMADktFWV9CQVNFX0FTU0VUAgpCQVNFX0FTU0VUAAlLRVlfQVNTRVQCBUFTU0VUABNLRVlfU1RBS0lOR19BREFQVEVSAg9TVEFLSU5HX0FEQVBURVIAEktFWV9TVEFLSU5HX0FNT1VOVAIOU1RBS0lOR19BTU9VTlQADUtFWV9MQVNUX1JBVEUCCUxBU1RfUkFURQAQS0VZX0NVUlJFTlRfUkFURQIMQ1VSUkVOVF9SQVRFAA9LRVlfTEFTVF9IRUlHSFQCC0xBU1RfSEVJR0hUABFLRVlfVEFSR0VUX0hFSUdIVAINVEFSR0VUX0hFSUdIVAAXS0VZX1NQT05TT1JTSElQX01BTkFHRVICE1NQT05TT1JTSElQX01BTkFHRVIACkZVTkNfU1RBS0UCBXN0YWtlAAxGVU5DX1VOU1RBS0UCB3Vuc3Rha2UAEUZVTkNfQ0xBSU1fUkVXQVJEAgtjbGFpbVJld2FyZAAOUEVSQ0VOVF9GQUNUT1IJALYCAQCAoJSljR0AC1pFUk9fQklHSU5UCQC2AgEAAAAKT05FX0JJR0lOVAkAtgIBAAEAC0lOSVRfQU1PVU5UAIDC1y8BEF92YWxpZGF0ZUFkZHJlc3MBCGFkZHJlc3NfBAckbWF0Y2gwCQCmCAEFCGFkZHJlc3NfAwkAAQIFByRtYXRjaDACB0FkZHJlc3MEAWEFByRtYXRjaDAGBwEOX3ZhbGlkYXRlQXNzZXQBBmFzc2V0XwMJAAACBQZhc3NldF8FBVdBVkVTBgQHJG1hdGNoMAkA7AcBCQDZBAEFBmFzc2V0XwMJAAECBQckbWF0Y2gwAgVBc3NldAQBYQUHJG1hdGNoMAYHAQ5fdG9Bc3NldFZlY3RvcgEGYXNzZXRfAwkAAAIFBmFzc2V0XwUFV0FWRVMFBHVuaXQJANkEAQUGYXNzZXRfARFfZ2V0QXNzZXREZWNpbWFscwEGYXNzZXRfBAckbWF0Y2gwBQZhc3NldF8DCQABAgUHJG1hdGNoMAIKQnl0ZVZlY3RvcgQBYQUHJG1hdGNoMAQHJG1hdGNoMQkA7AcBBQFhAwkAAQIFByRtYXRjaDECBUFzc2V0BAVhc3NldAUHJG1hdGNoMQgFBWFzc2V0CGRlY2ltYWxzCQACAQIgX2dldEFzc2V0RGVjaW1hbHM6IG5vIHN1Y2ggYXNzZXQACAERX2dldEFzc2V0UXVhbnRpdHkBBmFzc2V0XwQHJG1hdGNoMAkA7AcBBQZhc3NldF8DCQABAgUHJG1hdGNoMAIFQXNzZXQEAWEFByRtYXRjaDAIBQFhCHF1YW50aXR5CQACAQIgX2dldEFzc2V0UXVhbnRpdHk6IG5vIHN1Y2ggYXNzZXQBEF9nZXRBc3NldEJhbGFuY2UCBmFzc2V0XwdhZGRyZXNzBAckbWF0Y2gwBQZhc3NldF8DCQABAgUHJG1hdGNoMAIKQnl0ZVZlY3RvcgQBYQUHJG1hdGNoMAkA8AcCBQdhZGRyZXNzBQFhCAkA7wcBBQdhZGRyZXNzB3JlZ3VsYXIBDl9sb2FkQmFzZUFzc2V0AAQHJG1hdGNoMAkAoggBBQ5LRVlfQkFTRV9BU1NFVAMJAAECBQckbWF0Y2gwAgZTdHJpbmcEAWEFByRtYXRjaDADCQAAAgUBYQUFV0FWRVMFBHVuaXQJANkEAQUBYQkAAgECHV9sb2FkQmFzZUFzc2V0OiBubyBiYXNlIGFzc2V0ARNfbG9hZFN0YWtpbmdBZGFwdGVyAAQHJG1hdGNoMAkAoggBBRNLRVlfU1RBS0lOR19BREFQVEVSAwkAAQIFByRtYXRjaDACBlN0cmluZwQBYQUHJG1hdGNoMAkBEUBleHRyTmF0aXZlKDEwNjIpAQUBYQkAAgECJ19sb2FkU3Rha2luZ0FkYXB0ZXI6IG5vIHN0YWtpbmcgYWRhcHRlcgEXX2xvYWRTcG9uc29yc2hpcE1hbmFnZXIABAckbWF0Y2gwCQCiCAEFF0tFWV9TUE9OU09SU0hJUF9NQU5BR0VSAwkAAQIFByRtYXRjaDACBlN0cmluZwQBYQUHJG1hdGNoMAkBEUBleHRyTmF0aXZlKDEwNjIpAQUBYQkAAgECL19sb2FkU3BvbnNvcnNoaXBNYW5hZ2VyOiBubyBzcG9uc29yc2hpcCBtYW5hZ2VyAQhfbG9hZEludAIEa2V5XwhkZWZhdWx0XwQHJG1hdGNoMAkAnwgBBQRrZXlfAwkAAQIFByRtYXRjaDACA0ludAQBYQUHJG1hdGNoMAUBYQUIZGVmYXVsdF8BC19sb2FkQmlnSW50AgRrZXlfCGRlZmF1bHRfBAckbWF0Y2gwCQChCAEFBGtleV8DCQABAgUHJG1hdGNoMAIKQnl0ZVZlY3RvcgQBYQUHJG1hdGNoMAkAngMBBQFhBQhkZWZhdWx0XwEKX2xvYWRBc3NldAAEByRtYXRjaDAJAKIIAQUJS0VZX0FTU0VUAwkAAQIFByRtYXRjaDACBlN0cmluZwQBYQUHJG1hdGNoMAkA2QQBBQFhAQABDl91cGRhdGVTdGFraW5nAwZhc3NldF8HYW1vdW50XwhhZGFwdGVyXwQNc3Rha2luZ0Ftb3VudAkBCF9sb2FkSW50AgUSS0VZX1NUQUtJTkdfQU1PVU5UAAAEEG5ld1N0YWtpbmdBbW91bnQJAGQCBQ1zdGFraW5nQW1vdW50BQdhbW91bnRfBAppbnZvY2F0aW9uAwkAZgIFB2Ftb3VudF8AAAkA/AcEBQhhZGFwdGVyXwUKRlVOQ19TVEFLRQUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCBQZhc3NldF8FB2Ftb3VudF8FA25pbAkA/AcEBQhhZGFwdGVyXwUMRlVOQ19VTlNUQUtFCQDMCAIJAGgCAP///////////wEFB2Ftb3VudF8FA25pbAUDbmlsAwkAAAIFCmludm9jYXRpb24FCmludm9jYXRpb24JAMwIAgkBDEludGVnZXJFbnRyeQIFEktFWV9TVEFLSU5HX0FNT1VOVAUQbmV3U3Rha2luZ0Ftb3VudAUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAQtfY2hlY2twb2ludAEMY29tcGVuc2F0aW9uBAppbnZvY2F0aW9uCQD8BwQJARNfbG9hZFN0YWtpbmdBZGFwdGVyAAURRlVOQ19DTEFJTV9SRVdBUkQFA25pbAUDbmlsAwkAAAIFCmludm9jYXRpb24FCmludm9jYXRpb24EByRtYXRjaDAFCmludm9jYXRpb24DCQABAgUHJG1hdGNoMAIPKEludCwgSW50LCBJbnQpBAZyZXdhcmQFByRtYXRjaDADBQxjb21wZW5zYXRpb24EBXN0YWtlCQD8BwQFBHRoaXMFCkZVTkNfU1RBS0UJAMwIAggFBnJld2FyZAJfMQkAzAgCCAUGcmV3YXJkAl8zBQNuaWwFA25pbAMJAAACBQVzdGFrZQUFc3Rha2UJAJQKAgYIBQZyZXdhcmQCXzIJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4EBXN0YWtlCQD8BwQFBHRoaXMFCkZVTkNfU1RBS0UJAMwIAgkAZAIIBQZyZXdhcmQCXzEIBQZyZXdhcmQCXzIJAMwIAggFBnJld2FyZAJfMwUDbmlsBQNuaWwDCQAAAgUFc3Rha2UFBXN0YWtlCQCUCgIGAAAJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAJQKAgcAAAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgoBaQEEaW5pdAQKdG9rZW5OYW1lXwt0b2tlbkRlc2NyXwpiYXNlQXNzZXRfD3N0YWtpbmdBZGFwdGVyXwQIbXVsdGlzaWcJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAoggBBQxLRVlfTVVMVElTSUcCFWluaXQ6IG5vIG11bHRpc2lnIHNldAQFYXNzZXQJAQpfbG9hZEFzc2V0AAQJYmFzZUFzc2V0CQEOX3RvQXNzZXRWZWN0b3IBBQpiYXNlQXNzZXRfBANlcnIDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAAIBAhdpbml0OiBwZXJtaXNzaW9uIGRlbmllZAMJAQIhPQIFBWFzc2V0AQAJAAIBAhlpbml0OiBhbHJlYWR5IGluaXRpYWxpemVkAwMJAGYCAAQJALECAQUKdG9rZW5OYW1lXwYJAGYCCQCxAgEFCnRva2VuTmFtZV8AEAkAAgECGGluaXQ6IGludmFsaWQgdG9rZW4gbmFtZQMJAGYCCQCxAgEFC3Rva2VuRGVzY3JfAOgHCQACAQIfaW5pdDogaW52YWxpZCB0b2tlbiBkZXNjcmlwdGlvbgMJAQEhAQkBDl92YWxpZGF0ZUFzc2V0AQUKYmFzZUFzc2V0XwkAAgECGGluaXQ6IGludmFsaWQgYmFzZSBhc3NldAMJAQEhAQkBEF92YWxpZGF0ZUFkZHJlc3MBBQ9zdGFraW5nQWRhcHRlcl8JAAIBAh1pbml0OiBpbnZhbGlkIHN0YWtpbmcgYWRhcHRlcgUEdW5pdAMJAAACBQNlcnIFA2VycgQNdG9rZW5EZWNpbWFscwkBEV9nZXRBc3NldERlY2ltYWxzAQUJYmFzZUFzc2V0BAVpc3N1ZQkAwggFBQp0b2tlbk5hbWVfBQt0b2tlbkRlc2NyXwULSU5JVF9BTU9VTlQFDXRva2VuRGVjaW1hbHMGBAdhc3NldElkCQC4CAEFBWlzc3VlBAxzdGFrZUFjdGlvbnMJAQ5fdXBkYXRlU3Rha2luZwMFCWJhc2VBc3NldAULSU5JVF9BTU9VTlQJARFAZXh0ck5hdGl2ZSgxMDYyKQEFD3N0YWtpbmdBZGFwdGVyXwkAlAoCCQDOCAIJAMwIAgkBC1N0cmluZ0VudHJ5AgUOS0VZX0JBU0VfQVNTRVQFCmJhc2VBc3NldF8JAMwIAgkBC1N0cmluZ0VudHJ5AgUTS0VZX1NUQUtJTkdfQURBUFRFUgUPc3Rha2luZ0FkYXB0ZXJfCQDMCAIJAQtTdHJpbmdFbnRyeQIFCUtFWV9BU1NFVAkA2AQBBQdhc3NldElkCQDMCAIFBWlzc3VlBQNuaWwFDHN0YWtlQWN0aW9ucwUEdW5pdAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQdnZXRSYXRlAAQIbGFzdFJhdGUJAQtfbG9hZEJpZ0ludAIFDUtFWV9MQVNUX1JBVEUFDlBFUkNFTlRfRkFDVE9SBAtjdXJyZW50UmF0ZQkBC19sb2FkQmlnSW50AgUQS0VZX0NVUlJFTlRfUkFURQULWkVST19CSUdJTlQECmxhc3RIZWlnaHQJAQhfbG9hZEludAIFD0tFWV9MQVNUX0hFSUdIVAAABAx0YXJnZXRIZWlnaHQJAJcDAQkAzAgCCQEIX2xvYWRJbnQCBRFLRVlfVEFSR0VUX0hFSUdIVAAACQDMCAIFBmhlaWdodAUDbmlsBARyYXRlCQC3AgIFCGxhc3RSYXRlCQC5AgIFC2N1cnJlbnRSYXRlCQC2AgEJAGUCBQx0YXJnZXRIZWlnaHQFCmxhc3RIZWlnaHQJAJQKAgUDbmlsCQCmAwEFBHJhdGUBaQEKY2hlY2twb2ludAAECmNoZWNrcG9pbnQJAQtfY2hlY2twb2ludAEGAwkAAAIFCmNoZWNrcG9pbnQFCmNoZWNrcG9pbnQDCQEBIQEIBQpjaGVja3BvaW50Al8xCQACAQIVY2hlY2twb2ludDogbm8gcmV3YXJkAwkAZgIIBQpjaGVja3BvaW50Al8yAAAJAJQKAgkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUBaQxvcmlnaW5DYWxsZXIIBQpjaGVja3BvaW50Al8yCQEOX2xvYWRCYXNlQXNzZXQABQNuaWwFBHVuaXQJAJQKAgUDbmlsBQR1bml0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBB2RlcG9zaXQABAZjYWxsZXIJAKUIAQgFAWkGY2FsbGVyBAliYXNlQXNzZXQJAQ5fbG9hZEJhc2VBc3NldAAEBWFzc2V0CQEKX2xvYWRBc3NldAAEDWFzc2V0UXVhbnRpdHkJALYCAQkBEV9nZXRBc3NldFF1YW50aXR5AQUFYXNzZXQEA2VycgMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAQkAAgECFGRlcG9zaXQ6IG5vIHBheW1lbnRzAwkBAiE9AggJAJEDAggFAWkIcGF5bWVudHMAAAdhc3NldElkBQliYXNlQXNzZXQJAAIBAiVkZXBvc2l0OiBwYXltZW50IGlzIG5vdCBpbiBiYXNlIGFzc2V0AwkAZwIAAAgJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQJAAIBAh9kZXBvc2l0OiBpbnZhbGlkIHBheW1lbnQgYW1vdW50BQR1bml0AwkAAAIFA2VycgUDZXJyBApjaGVja3BvaW50CQELX2NoZWNrcG9pbnQBBwMJAAACBQpjaGVja3BvaW50BQpjaGVja3BvaW50BAhsYXN0UmF0ZQkBC19sb2FkQmlnSW50AgUNS0VZX0xBU1RfUkFURQUOUEVSQ0VOVF9GQUNUT1IEC2N1cnJlbnRSYXRlCQELX2xvYWRCaWdJbnQCBRBLRVlfQ1VSUkVOVF9SQVRFBQtaRVJPX0JJR0lOVAQKbGFzdEhlaWdodAkBCF9sb2FkSW50AgUPS0VZX0xBU1RfSEVJR0hUAAAEDHRhcmdldEhlaWdodAkAlwMBCQDMCAIJAQhfbG9hZEludAIFEUtFWV9UQVJHRVRfSEVJR0hUAAAJAMwIAgUGaGVpZ2h0BQNuaWwED2xhc3RSYXRlVXBkYXRlZAkAtwICBQhsYXN0UmF0ZQkAuQICBQtjdXJyZW50UmF0ZQkAtgIBCQBlAgUMdGFyZ2V0SGVpZ2h0BQpsYXN0SGVpZ2h0BAtpc3N1ZUFtb3VudAkAoAMBCQC9AgQJALYCAQgJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQFDlBFUkNFTlRfRkFDVE9SBQ9sYXN0UmF0ZVVwZGF0ZWQFBUZMT09SBBJjdXJyZW50UmF0ZVVwZGF0ZWQJAL0CBAULY3VycmVudFJhdGUFDWFzc2V0UXVhbnRpdHkJALcCAgUNYXNzZXRRdWFudGl0eQkAtgIBBQtpc3N1ZUFtb3VudAUFRkxPT1IEEWxhc3RIZWlnaHRVcGRhdGVkBQx0YXJnZXRIZWlnaHQEDHN0YWtlQWN0aW9ucwkBDl91cGRhdGVTdGFraW5nAwUJYmFzZUFzc2V0CAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAkBE19sb2FkU3Rha2luZ0FkYXB0ZXIACQCUCgIJAM4IAgkAzAgCCQEHUmVpc3N1ZQMFBWFzc2V0BQtpc3N1ZUFtb3VudAYJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyBQtpc3N1ZUFtb3VudAUFYXNzZXQJAMwIAgkBC0JpbmFyeUVudHJ5AgUNS0VZX0xBU1RfUkFURQkAnQMBBQ9sYXN0UmF0ZVVwZGF0ZWQJAMwIAgkBC0JpbmFyeUVudHJ5AgUQS0VZX0NVUlJFTlRfUkFURQkAnQMBBRJjdXJyZW50UmF0ZVVwZGF0ZWQJAMwIAgkBDEludGVnZXJFbnRyeQIFD0tFWV9MQVNUX0hFSUdIVAURbGFzdEhlaWdodFVwZGF0ZWQFA25pbAUMc3Rha2VBY3Rpb25zBQR1bml0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBBXN0YWtlAgdhbW91bnRfDXNwcmVhZEJsb2Nrc18EBmNhbGxlcgkApQgBCAUBaQZjYWxsZXIECWJhc2VBc3NldAkBDl9sb2FkQmFzZUFzc2V0AAQFYXNzZXQJAQpfbG9hZEFzc2V0AAQNYXNzZXRRdWFudGl0eQkAtgIBCQERX2dldEFzc2V0UXVhbnRpdHkBBQVhc3NldAQDZXJyAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIZc3Rha2U6IG9ubHkgdGhpcyBjb250cmFjdAMJAGcCAAAFB2Ftb3VudF8JAAIBAhtzdGFrZTogaW52YWxpZCBzdGFrZSBhbW91bnQDCQBmAgUHYW1vdW50XwkBEF9nZXRBc3NldEJhbGFuY2UCBQliYXNlQXNzZXQFBHRoaXMJAAIBAiBzdGFrZTogaW5zdWZmaWNpZW50IHN0YWtlIGFtb3VudAMJAAACBQ1hc3NldFF1YW50aXR5BQtaRVJPX0JJR0lOVAkAAgECH3N0YWtlOiBubyBkZXBvc2l0cyB0byBzdGFrZSBmb3IFBHVuaXQDCQAAAgUDZXJyBQNlcnIECGxhc3RSYXRlCQELX2xvYWRCaWdJbnQCBQ1LRVlfTEFTVF9SQVRFBQ5QRVJDRU5UX0ZBQ1RPUgQLY3VycmVudFJhdGUJAQtfbG9hZEJpZ0ludAIFEEtFWV9DVVJSRU5UX1JBVEUFC1pFUk9fQklHSU5UBApsYXN0SGVpZ2h0CQEIX2xvYWRJbnQCBQ9LRVlfTEFTVF9IRUlHSFQAAAQMdGFyZ2V0SGVpZ2h0CQEIX2xvYWRJbnQCBRFLRVlfVEFSR0VUX0hFSUdIVAAABA9taW5UYXJnZXRIZWlnaHQJAJcDAQkAzAgCBQx0YXJnZXRIZWlnaHQJAMwIAgUGaGVpZ2h0BQNuaWwEB2FjdGlvbnMDCQAAAgUPbWluVGFyZ2V0SGVpZ2h0BQZoZWlnaHQED2xhc3RSYXRlVXBkYXRlZAkAtwICBQhsYXN0UmF0ZQkAuQICCQC2AgEJAGUCBQ9taW5UYXJnZXRIZWlnaHQFCmxhc3RIZWlnaHQFC2N1cnJlbnRSYXRlBA9yZW1haW5pbmdSZXdhcmQJALkCAgULY3VycmVudFJhdGUJALYCAQkAZQIFDHRhcmdldEhlaWdodAUPbWluVGFyZ2V0SGVpZ2h0BBJjdXJyZW50UmF0ZVVwZGF0ZWQJALwCAwkAtwICCQC5AgIJALYCAQUHYW1vdW50XwUOUEVSQ0VOVF9GQUNUT1IFD3JlbWFpbmluZ1Jld2FyZAUKT05FX0JJR0lOVAkAuQICBQ1hc3NldFF1YW50aXR5CQC2AgEFDXNwcmVhZEJsb2Nrc18EEWxhc3RIZWlnaHRVcGRhdGVkBQZoZWlnaHQEE3RhcmdldEhlaWdodFVwZGF0ZWQJAGQCBRFsYXN0SGVpZ2h0VXBkYXRlZAUNc3ByZWFkQmxvY2tzXwkAzAgCCQELQmluYXJ5RW50cnkCBQ1LRVlfTEFTVF9SQVRFCQCdAwEFD2xhc3RSYXRlVXBkYXRlZAkAzAgCCQELQmluYXJ5RW50cnkCBRBLRVlfQ1VSUkVOVF9SQVRFCQCdAwEFEmN1cnJlbnRSYXRlVXBkYXRlZAkAzAgCCQEMSW50ZWdlckVudHJ5AgUPS0VZX0xBU1RfSEVJR0hUBRFsYXN0SGVpZ2h0VXBkYXRlZAkAzAgCCQEMSW50ZWdlckVudHJ5AgURS0VZX1RBUkdFVF9IRUlHSFQFE3RhcmdldEhlaWdodFVwZGF0ZWQFA25pbAMJAGYCBQx0YXJnZXRIZWlnaHQFCmxhc3RIZWlnaHQED2xhc3RSYXRlVXBkYXRlZAkAtwICBQhsYXN0UmF0ZQkAuQICCQC2AgEJAGUCBQx0YXJnZXRIZWlnaHQFCmxhc3RIZWlnaHQFC2N1cnJlbnRSYXRlBBJjdXJyZW50UmF0ZVVwZGF0ZWQJALwCAwkAtgIBBQdhbW91bnRfBQ5QRVJDRU5UX0ZBQ1RPUgkAuQICBQ1hc3NldFF1YW50aXR5CQC2AgEFDXNwcmVhZEJsb2Nrc18EEWxhc3RIZWlnaHRVcGRhdGVkBQZoZWlnaHQEE3RhcmdldEhlaWdodFVwZGF0ZWQJAGQCBRFsYXN0SGVpZ2h0VXBkYXRlZAUNc3ByZWFkQmxvY2tzXwkAzAgCCQELQmluYXJ5RW50cnkCBQ1LRVlfTEFTVF9SQVRFCQCdAwEFD2xhc3RSYXRlVXBkYXRlZAkAzAgCCQELQmluYXJ5RW50cnkCBRBLRVlfQ1VSUkVOVF9SQVRFCQCdAwEFEmN1cnJlbnRSYXRlVXBkYXRlZAkAzAgCCQEMSW50ZWdlckVudHJ5AgUPS0VZX0xBU1RfSEVJR0hUBRFsYXN0SGVpZ2h0VXBkYXRlZAkAzAgCCQEMSW50ZWdlckVudHJ5AgURS0VZX1RBUkdFVF9IRUlHSFQFE3RhcmdldEhlaWdodFVwZGF0ZWQFA25pbAQSY3VycmVudFJhdGVVcGRhdGVkCQC8AgMJALYCAQUHYW1vdW50XwUOUEVSQ0VOVF9GQUNUT1IJALkCAgUNYXNzZXRRdWFudGl0eQkAtgIBBQ1zcHJlYWRCbG9ja3NfBBFsYXN0SGVpZ2h0VXBkYXRlZAUGaGVpZ2h0BBN0YXJnZXRIZWlnaHRVcGRhdGVkCQBkAgURbGFzdEhlaWdodFVwZGF0ZWQFDXNwcmVhZEJsb2Nrc18JAMwIAgkBC0JpbmFyeUVudHJ5AgUQS0VZX0NVUlJFTlRfUkFURQkAnQMBBRJjdXJyZW50UmF0ZVVwZGF0ZWQJAMwIAgkBDEludGVnZXJFbnRyeQIFD0tFWV9MQVNUX0hFSUdIVAURbGFzdEhlaWdodFVwZGF0ZWQJAMwIAgkBDEludGVnZXJFbnRyeQIFEUtFWV9UQVJHRVRfSEVJR0hUBRN0YXJnZXRIZWlnaHRVcGRhdGVkBQNuaWwEDHN0YWtlQWN0aW9ucwkBDl91cGRhdGVTdGFraW5nAwUJYmFzZUFzc2V0BQdhbW91bnRfCQETX2xvYWRTdGFraW5nQWRhcHRlcgAJAJQKAgkAzggCBQdhY3Rpb25zBQxzdGFrZUFjdGlvbnMFBHVuaXQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEId2l0aGRyYXcABAZjYWxsZXIJAKUIAQgFAWkGY2FsbGVyBAliYXNlQXNzZXQJAQ5fbG9hZEJhc2VBc3NldAAEBWFzc2V0CQEKX2xvYWRBc3NldAAEDWFzc2V0UXVhbnRpdHkJALYCAQkBEV9nZXRBc3NldFF1YW50aXR5AQUFYXNzZXQEA2VycgMJAGYCAAEJAJADAQgFAWkIcGF5bWVudHMJAAIBAhV3aXRoZHJhdzogbm8gcGF5bWVudHMDCQECIT0CCAkAkQMCCAUBaQhwYXltZW50cwAAB2Fzc2V0SWQFBWFzc2V0CQACAQIpd2l0aGRyYXc6IHBheW1lbnQgaXMgbm90IGluIGNvcnJlY3QgYXNzZXQDCQBnAgAACAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAkAAgECIHdpdGhkcmF3OiBpbnZhbGlkIHBheW1lbnQgYW1vdW50BQR1bml0AwkAAAIFA2VycgUDZXJyBApjaGVja3BvaW50CQELX2NoZWNrcG9pbnQBBwMJAAACBQpjaGVja3BvaW50BQpjaGVja3BvaW50BAhsYXN0UmF0ZQkBC19sb2FkQmlnSW50AgUNS0VZX0xBU1RfUkFURQUOUEVSQ0VOVF9GQUNUT1IEC2N1cnJlbnRSYXRlCQELX2xvYWRCaWdJbnQCBRBLRVlfQ1VSUkVOVF9SQVRFBQtaRVJPX0JJR0lOVAQKbGFzdEhlaWdodAkBCF9sb2FkSW50AgUPS0VZX0xBU1RfSEVJR0hUAAAEDHRhcmdldEhlaWdodAkAlwMBCQDMCAIJAQhfbG9hZEludAIFEUtFWV9UQVJHRVRfSEVJR0hUAAAJAMwIAgUGaGVpZ2h0BQNuaWwED2xhc3RSYXRlVXBkYXRlZAkAtwICBQhsYXN0UmF0ZQkAuQICBQtjdXJyZW50UmF0ZQkAtgIBCQBlAgUMdGFyZ2V0SGVpZ2h0BQpsYXN0SGVpZ2h0BApiYXNlQW1vdW50CQCgAwEJAL0CBAkAtgIBCAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAUPbGFzdFJhdGVVcGRhdGVkBQ5QRVJDRU5UX0ZBQ1RPUgUFRkxPT1IEEmN1cnJlbnRSYXRlVXBkYXRlZAkAvQIEBQtjdXJyZW50UmF0ZQUNYXNzZXRRdWFudGl0eQkAuAICBQ1hc3NldFF1YW50aXR5CQC2AgEICQCRAwIIBQFpCHBheW1lbnRzAAAGYW1vdW50BQVGTE9PUgQRbGFzdEhlaWdodFVwZGF0ZWQFDHRhcmdldEhlaWdodAQMc3Rha2VBY3Rpb25zCQEOX3VwZGF0ZVN0YWtpbmcDBQliYXNlQXNzZXQJAGgCAP///////////wEFCmJhc2VBbW91bnQJARNfbG9hZFN0YWtpbmdBZGFwdGVyAAkAlAoCCQDOCAIFDHN0YWtlQWN0aW9ucwkAzAgCCQEEQnVybgIFBWFzc2V0CAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUBaQZjYWxsZXIFCmJhc2VBbW91bnQFBHVuaXQJAMwIAgkBC0JpbmFyeUVudHJ5AgUNS0VZX0xBU1RfUkFURQkAnQMBBQ9sYXN0UmF0ZVVwZGF0ZWQJAMwIAgkBC0JpbmFyeUVudHJ5AgUQS0VZX0NVUlJFTlRfUkFURQkAnQMBBRJjdXJyZW50UmF0ZVVwZGF0ZWQJAMwIAgkBDEludGVnZXJFbnRyeQIFD0tFWV9MQVNUX0hFSUdIVAURbGFzdEhlaWdodFVwZGF0ZWQFA25pbAUEdW5pdAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQtzZXRNdWx0aXNpZwEJbXVsdGlzaWdfBANlcnIDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAAIBAh5zZXRNdWx0aXNpZzogcGVybWlzc2lvbiBkZW5pZWQDCQEBIQEJARBfdmFsaWRhdGVBZGRyZXNzAQUJbXVsdGlzaWdfCQACAQIlc2V0TXVsdGlzaWc6IGludmFsaWQgbXVsdGlzaWcgYWRkcmVzcwUEdW5pdAMJAAACBQNlcnIFA2VycgkAlAoCCQDMCAIJAQtTdHJpbmdFbnRyeQIFDEtFWV9NVUxUSVNJRwUJbXVsdGlzaWdfBQNuaWwFBHVuaXQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEVc2V0U3BvbnNvcnNoaXBNYW5hZ2VyAQhtYW5hZ2VyXwQDZXJyAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIoc2V0U3BvbnNvcnNoaXBNYW5hZ2VyOiBwZXJtaXNzaW9uIGRlbmllZAMJAQEhAQkBEF92YWxpZGF0ZUFkZHJlc3MBBQhtYW5hZ2VyXwkAAgECLnNldFNwb25zb3JzaGlwTWFuYWdlcjogaW52YWxpZCBtYW5hZ2VyIGFkZHJlc3MFBHVuaXQDCQAAAgUDZXJyBQNlcnIJAJQKAgkAzAgCCQELU3RyaW5nRW50cnkCBRdLRVlfU1BPTlNPUlNISVBfTUFOQUdFUgUIbWFuYWdlcl8FA25pbAUEdW5pdAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpARF1cGRhdGVTcG9uc29yc2hpcAIVbWluU3BvbnNvcmVkQXNzZXRGZWVfFHdhdmVzUmVxdWlyZWRBbW91bnRfBANlcnIDCQECIT0CCAUBaQZjYWxsZXIJARdfbG9hZFNwb25zb3JzaGlwTWFuYWdlcgAJAAIBAiR1cGRhdGVTcG9uc29yc2hpcDogcGVybWlzc2lvbiBkZW5pZWQDCQBmAgAABRVtaW5TcG9uc29yZWRBc3NldEZlZV8JAAIBAip1cGRhdGVTcG9uc29yc2hpcDogaW52YWxpZCBzcG9uc29yc2hpcCBmZWUDCQBmAgAABRR3YXZlc1JlcXVpcmVkQW1vdW50XwkAAgECMHVwZGF0ZVNwb25zb3JzaGlwOiBpbnZhbGlkIHdhdmVzIHJlcXVpcmVkIGFtb3VudAUEdW5pdAMJAAACBQNlcnIFA2VycgQMc3BvbnNvckFzc2V0CQEKX2xvYWRBc3NldAAEFG1pblNwb25zb3JlZEFzc2V0RmVlAwkAAAIFFW1pblNwb25zb3JlZEFzc2V0RmVlXwAABQR1bml0BRVtaW5TcG9uc29yZWRBc3NldEZlZV8EB2FjdGlvbnMDCQBmAggJAO8HAQUEdGhpcwdyZWd1bGFyBRR3YXZlc1JlcXVpcmVkQW1vdW50XwkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUBaQZjYWxsZXIJAGUCCAkA7wcBBQR0aGlzB3JlZ3VsYXIFFHdhdmVzUmVxdWlyZWRBbW91bnRfBQR1bml0BQNuaWwFA25pbAkAlAoCCQDOCAIFB2FjdGlvbnMJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyCQDwBwIFBHRoaXMFDHNwb25zb3JBc3NldAUMc3BvbnNvckFzc2V0CQDMCAIJAQpTcG9uc29yRmVlAgUMc3BvbnNvckFzc2V0BRRtaW5TcG9uc29yZWRBc3NldEZlZQUDbmlsBQR1bml0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBD3d4Vm90aW5nU3VnZ2VzdAQOdm90aW5nQWRkcmVzc18Id3hBc3NldF8Jd3hBbW91bnRfC2Fzc2V0SW1hZ2VfBANlcnIDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAAIBAiJ3eFZvdGluZ1N1Z2dlc3Q6IHBlcm1pc3Npb24gZGVuaWVkAwkBASEBCQEQX3ZhbGlkYXRlQWRkcmVzcwEFDnZvdGluZ0FkZHJlc3NfCQACAQInd3hWb3RpbmdTdWdnZXN0OiBpbnZhbGlkIHZvdGluZyBhZGRyZXNzAwkAZgIAAAUJd3hBbW91bnRfCQACAQIid3hWb3RpbmdTdWdnZXN0OiBpbnZhbGlkIHd4IGFtb3VudAUEdW5pdAMJAAACBQNlcnIFA2VycgQPc3BvbnNvckFzc2V0U3RyCQDYBAEJAQpfbG9hZEFzc2V0AAQHd3hBc3NldAkBDl90b0Fzc2V0VmVjdG9yAQUId3hBc3NldF8EC3d4QXNzZXRJbmZvCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAOwHAQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCBQd3eEFzc2V0AiF3eFZvdGluZ1N1Z2dlc3Q6IGludmFsaWQgd3ggYXNzZXQCJnd4Vm90aW5nU3VnZ2VzdDogaW52YWxpZCB3eCBhc3NldCBpbmZvBAppbnZvY2F0aW9uCQD8BwQJARFAZXh0ck5hdGl2ZSgxMDYyKQEFDnZvdGluZ0FkZHJlc3NfAgdzdWdnZXN0CQDMCAIFD3Nwb25zb3JBc3NldFN0cgkAzAgCBQthc3NldEltYWdlXwUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCBQd3eEFzc2V0CQBoAgUJd3hBbW91bnRfCAULd3hBc3NldEluZm8IZGVjaW1hbHMFA25pbAMJAAACBQppbnZvY2F0aW9uBQppbnZvY2F0aW9uCQCUCgIFA25pbAUEdW5pdAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgECdHgBBnZlcmlmeQAEByRtYXRjaDAJAKIIAQUMS0VZX01VTFRJU0lHAwkAAQIFByRtYXRjaDACBlN0cmluZwQIbXVsdGlzaWcFByRtYXRjaDAJAQt2YWx1ZU9yRWxzZQIJAJsIAgkBEUBleHRyTmF0aXZlKDEwNjIpAQUIbXVsdGlzaWcJALkJAgkAzAgCBQpLRVlfU1RBVFVTCQDMCAIJAKUIAQUEdGhpcwkAzAgCCQDYBAEIBQJ0eAJpZAUDbmlsBQlTRVBBUkFUT1IHCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwAACAUCdHgPc2VuZGVyUHVibGljS2V5GLwpbQ==", "height": 2348378, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: HrL5TnfyTHyWT29Avd5GMq1zSMwSkZgStTCWEHUPJ6AN Next: ESTcvu8S9pTZiz2248A3tUjFUKUHsdjdtZPLT5nb8ZAL Diff:
Old | New | Differences | |
---|---|---|---|
418 | 418 | ||
419 | 419 | ||
420 | 420 | @Callable(i) | |
421 | - | func wxVotingSuggest (votingAddress_,wxAsset_,assetImage_) = { | |
422 | - | let votingAddress = valueOrErrorMessage(addressFromString(votingAddress_), "wxVotingSuggest: invalid voting address") | |
423 | - | let sponsorAssetStr = toBase58String(_loadAsset()) | |
424 | - | let wxAsset = _toAssetVector(wxAsset_) | |
425 | - | let wxAssetInfo = valueOrErrorMessage(assetInfo(valueOrErrorMessage(wxAsset, "wxVotingSuggest: invalid wx asset")), "wxVotingSuggest: invalid wx asset info") | |
426 | - | let invocation = invoke(votingAddress, "suggest", [sponsorAssetStr, assetImage_], [AttachedPayment(wxAsset, (100 * wxAssetInfo.decimals))]) | |
427 | - | if ((invocation == invocation)) | |
428 | - | then $Tuple2(nil, unit) | |
421 | + | func wxVotingSuggest (votingAddress_,wxAsset_,wxAmount_,assetImage_) = { | |
422 | + | let err = if ((i.caller != this)) | |
423 | + | then throw("wxVotingSuggest: permission denied") | |
424 | + | else if (!(_validateAddress(votingAddress_))) | |
425 | + | then throw("wxVotingSuggest: invalid voting address") | |
426 | + | else if ((0 > wxAmount_)) | |
427 | + | then throw("wxVotingSuggest: invalid wx amount") | |
428 | + | else unit | |
429 | + | if ((err == err)) | |
430 | + | then { | |
431 | + | let sponsorAssetStr = toBase58String(_loadAsset()) | |
432 | + | let wxAsset = _toAssetVector(wxAsset_) | |
433 | + | let wxAssetInfo = valueOrErrorMessage(assetInfo(valueOrErrorMessage(wxAsset, "wxVotingSuggest: invalid wx asset")), "wxVotingSuggest: invalid wx asset info") | |
434 | + | let invocation = invoke(addressFromStringValue(votingAddress_), "suggest", [sponsorAssetStr, assetImage_], [AttachedPayment(wxAsset, (wxAmount_ * wxAssetInfo.decimals))]) | |
435 | + | if ((invocation == invocation)) | |
436 | + | then $Tuple2(nil, unit) | |
437 | + | else throw("Strict value is not equal to itself.") | |
438 | + | } | |
429 | 439 | else throw("Strict value is not equal to itself.") | |
430 | 440 | } | |
431 | 441 |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let SEPARATOR = "__" | |
5 | 5 | ||
6 | 6 | let WAVES = "WAVES" | |
7 | 7 | ||
8 | 8 | let KEY_MULTISIG = "MULTISIG" | |
9 | 9 | ||
10 | 10 | let KEY_STATUS = "STATUS" | |
11 | 11 | ||
12 | 12 | let KEY_BASE_ASSET = "BASE_ASSET" | |
13 | 13 | ||
14 | 14 | let KEY_ASSET = "ASSET" | |
15 | 15 | ||
16 | 16 | let KEY_STAKING_ADAPTER = "STAKING_ADAPTER" | |
17 | 17 | ||
18 | 18 | let KEY_STAKING_AMOUNT = "STAKING_AMOUNT" | |
19 | 19 | ||
20 | 20 | let KEY_LAST_RATE = "LAST_RATE" | |
21 | 21 | ||
22 | 22 | let KEY_CURRENT_RATE = "CURRENT_RATE" | |
23 | 23 | ||
24 | 24 | let KEY_LAST_HEIGHT = "LAST_HEIGHT" | |
25 | 25 | ||
26 | 26 | let KEY_TARGET_HEIGHT = "TARGET_HEIGHT" | |
27 | 27 | ||
28 | 28 | let KEY_SPONSORSHIP_MANAGER = "SPONSORSHIP_MANAGER" | |
29 | 29 | ||
30 | 30 | let FUNC_STAKE = "stake" | |
31 | 31 | ||
32 | 32 | let FUNC_UNSTAKE = "unstake" | |
33 | 33 | ||
34 | 34 | let FUNC_CLAIM_REWARD = "claimReward" | |
35 | 35 | ||
36 | 36 | let PERCENT_FACTOR = toBigInt(1000000000000) | |
37 | 37 | ||
38 | 38 | let ZERO_BIGINT = toBigInt(0) | |
39 | 39 | ||
40 | 40 | let ONE_BIGINT = toBigInt(1) | |
41 | 41 | ||
42 | 42 | let INIT_AMOUNT = 100000000 | |
43 | 43 | ||
44 | 44 | func _validateAddress (address_) = match addressFromString(address_) { | |
45 | 45 | case a: Address => | |
46 | 46 | true | |
47 | 47 | case _ => | |
48 | 48 | false | |
49 | 49 | } | |
50 | 50 | ||
51 | 51 | ||
52 | 52 | func _validateAsset (asset_) = if ((asset_ == WAVES)) | |
53 | 53 | then true | |
54 | 54 | else match assetInfo(fromBase58String(asset_)) { | |
55 | 55 | case a: Asset => | |
56 | 56 | true | |
57 | 57 | case _ => | |
58 | 58 | false | |
59 | 59 | } | |
60 | 60 | ||
61 | 61 | ||
62 | 62 | func _toAssetVector (asset_) = if ((asset_ == WAVES)) | |
63 | 63 | then unit | |
64 | 64 | else fromBase58String(asset_) | |
65 | 65 | ||
66 | 66 | ||
67 | 67 | func _getAssetDecimals (asset_) = match asset_ { | |
68 | 68 | case a: ByteVector => | |
69 | 69 | match assetInfo(a) { | |
70 | 70 | case asset: Asset => | |
71 | 71 | asset.decimals | |
72 | 72 | case _ => | |
73 | 73 | throw("_getAssetDecimals: no such asset") | |
74 | 74 | } | |
75 | 75 | case _ => | |
76 | 76 | 8 | |
77 | 77 | } | |
78 | 78 | ||
79 | 79 | ||
80 | 80 | func _getAssetQuantity (asset_) = match assetInfo(asset_) { | |
81 | 81 | case a: Asset => | |
82 | 82 | a.quantity | |
83 | 83 | case _ => | |
84 | 84 | throw("_getAssetQuantity: no such asset") | |
85 | 85 | } | |
86 | 86 | ||
87 | 87 | ||
88 | 88 | func _getAssetBalance (asset_,address) = match asset_ { | |
89 | 89 | case a: ByteVector => | |
90 | 90 | assetBalance(address, a) | |
91 | 91 | case _ => | |
92 | 92 | wavesBalance(address).regular | |
93 | 93 | } | |
94 | 94 | ||
95 | 95 | ||
96 | 96 | func _loadBaseAsset () = match getString(KEY_BASE_ASSET) { | |
97 | 97 | case a: String => | |
98 | 98 | if ((a == WAVES)) | |
99 | 99 | then unit | |
100 | 100 | else fromBase58String(a) | |
101 | 101 | case _ => | |
102 | 102 | throw("_loadBaseAsset: no base asset") | |
103 | 103 | } | |
104 | 104 | ||
105 | 105 | ||
106 | 106 | func _loadStakingAdapter () = match getString(KEY_STAKING_ADAPTER) { | |
107 | 107 | case a: String => | |
108 | 108 | addressFromStringValue(a) | |
109 | 109 | case _ => | |
110 | 110 | throw("_loadStakingAdapter: no staking adapter") | |
111 | 111 | } | |
112 | 112 | ||
113 | 113 | ||
114 | 114 | func _loadSponsorshipManager () = match getString(KEY_SPONSORSHIP_MANAGER) { | |
115 | 115 | case a: String => | |
116 | 116 | addressFromStringValue(a) | |
117 | 117 | case _ => | |
118 | 118 | throw("_loadSponsorshipManager: no sponsorship manager") | |
119 | 119 | } | |
120 | 120 | ||
121 | 121 | ||
122 | 122 | func _loadInt (key_,default_) = match getInteger(key_) { | |
123 | 123 | case a: Int => | |
124 | 124 | a | |
125 | 125 | case _ => | |
126 | 126 | default_ | |
127 | 127 | } | |
128 | 128 | ||
129 | 129 | ||
130 | 130 | func _loadBigInt (key_,default_) = match getBinary(key_) { | |
131 | 131 | case a: ByteVector => | |
132 | 132 | toBigInt(a) | |
133 | 133 | case _ => | |
134 | 134 | default_ | |
135 | 135 | } | |
136 | 136 | ||
137 | 137 | ||
138 | 138 | func _loadAsset () = match getString(KEY_ASSET) { | |
139 | 139 | case a: String => | |
140 | 140 | fromBase58String(a) | |
141 | 141 | case _ => | |
142 | 142 | base58'' | |
143 | 143 | } | |
144 | 144 | ||
145 | 145 | ||
146 | 146 | func _updateStaking (asset_,amount_,adapter_) = { | |
147 | 147 | let stakingAmount = _loadInt(KEY_STAKING_AMOUNT, 0) | |
148 | 148 | let newStakingAmount = (stakingAmount + amount_) | |
149 | 149 | let invocation = if ((amount_ > 0)) | |
150 | 150 | then invoke(adapter_, FUNC_STAKE, nil, [AttachedPayment(asset_, amount_)]) | |
151 | 151 | else invoke(adapter_, FUNC_UNSTAKE, [(-1 * amount_)], nil) | |
152 | 152 | if ((invocation == invocation)) | |
153 | 153 | then [IntegerEntry(KEY_STAKING_AMOUNT, newStakingAmount)] | |
154 | 154 | else throw("Strict value is not equal to itself.") | |
155 | 155 | } | |
156 | 156 | ||
157 | 157 | ||
158 | 158 | func _checkpoint (compensation) = { | |
159 | 159 | let invocation = invoke(_loadStakingAdapter(), FUNC_CLAIM_REWARD, nil, nil) | |
160 | 160 | if ((invocation == invocation)) | |
161 | 161 | then match invocation { | |
162 | 162 | case reward: (Int, Int, Int) => | |
163 | 163 | if (compensation) | |
164 | 164 | then { | |
165 | 165 | let stake = invoke(this, FUNC_STAKE, [reward._1, reward._3], nil) | |
166 | 166 | if ((stake == stake)) | |
167 | 167 | then $Tuple2(true, reward._2) | |
168 | 168 | else throw("Strict value is not equal to itself.") | |
169 | 169 | } | |
170 | 170 | else { | |
171 | 171 | let stake = invoke(this, FUNC_STAKE, [(reward._1 + reward._2), reward._3], nil) | |
172 | 172 | if ((stake == stake)) | |
173 | 173 | then $Tuple2(true, 0) | |
174 | 174 | else throw("Strict value is not equal to itself.") | |
175 | 175 | } | |
176 | 176 | case _ => | |
177 | 177 | $Tuple2(false, 0) | |
178 | 178 | } | |
179 | 179 | else throw("Strict value is not equal to itself.") | |
180 | 180 | } | |
181 | 181 | ||
182 | 182 | ||
183 | 183 | @Callable(i) | |
184 | 184 | func init (tokenName_,tokenDescr_,baseAsset_,stakingAdapter_) = { | |
185 | 185 | let multisig = valueOrErrorMessage(getString(KEY_MULTISIG), "init: no multisig set") | |
186 | 186 | let asset = _loadAsset() | |
187 | 187 | let baseAsset = _toAssetVector(baseAsset_) | |
188 | 188 | let err = if ((i.caller != this)) | |
189 | 189 | then throw("init: permission denied") | |
190 | 190 | else if ((asset != base58'')) | |
191 | 191 | then throw("init: already initialized") | |
192 | 192 | else if (if ((4 > size(tokenName_))) | |
193 | 193 | then true | |
194 | 194 | else (size(tokenName_) > 16)) | |
195 | 195 | then throw("init: invalid token name") | |
196 | 196 | else if ((size(tokenDescr_) > 1000)) | |
197 | 197 | then throw("init: invalid token description") | |
198 | 198 | else if (!(_validateAsset(baseAsset_))) | |
199 | 199 | then throw("init: invalid base asset") | |
200 | 200 | else if (!(_validateAddress(stakingAdapter_))) | |
201 | 201 | then throw("init: invalid staking adapter") | |
202 | 202 | else unit | |
203 | 203 | if ((err == err)) | |
204 | 204 | then { | |
205 | 205 | let tokenDecimals = _getAssetDecimals(baseAsset) | |
206 | 206 | let issue = Issue(tokenName_, tokenDescr_, INIT_AMOUNT, tokenDecimals, true) | |
207 | 207 | let assetId = calculateAssetId(issue) | |
208 | 208 | let stakeActions = _updateStaking(baseAsset, INIT_AMOUNT, addressFromStringValue(stakingAdapter_)) | |
209 | 209 | $Tuple2(([StringEntry(KEY_BASE_ASSET, baseAsset_), StringEntry(KEY_STAKING_ADAPTER, stakingAdapter_), StringEntry(KEY_ASSET, toBase58String(assetId)), issue] ++ stakeActions), unit) | |
210 | 210 | } | |
211 | 211 | else throw("Strict value is not equal to itself.") | |
212 | 212 | } | |
213 | 213 | ||
214 | 214 | ||
215 | 215 | ||
216 | 216 | @Callable(i) | |
217 | 217 | func getRate () = { | |
218 | 218 | let lastRate = _loadBigInt(KEY_LAST_RATE, PERCENT_FACTOR) | |
219 | 219 | let currentRate = _loadBigInt(KEY_CURRENT_RATE, ZERO_BIGINT) | |
220 | 220 | let lastHeight = _loadInt(KEY_LAST_HEIGHT, 0) | |
221 | 221 | let targetHeight = min([_loadInt(KEY_TARGET_HEIGHT, 0), height]) | |
222 | 222 | let rate = (lastRate + (currentRate * toBigInt((targetHeight - lastHeight)))) | |
223 | 223 | $Tuple2(nil, toString(rate)) | |
224 | 224 | } | |
225 | 225 | ||
226 | 226 | ||
227 | 227 | ||
228 | 228 | @Callable(i) | |
229 | 229 | func checkpoint () = { | |
230 | 230 | let checkpoint = _checkpoint(true) | |
231 | 231 | if ((checkpoint == checkpoint)) | |
232 | 232 | then if (!(checkpoint._1)) | |
233 | 233 | then throw("checkpoint: no reward") | |
234 | 234 | else if ((checkpoint._2 > 0)) | |
235 | 235 | then $Tuple2([ScriptTransfer(i.originCaller, checkpoint._2, _loadBaseAsset())], unit) | |
236 | 236 | else $Tuple2(nil, unit) | |
237 | 237 | else throw("Strict value is not equal to itself.") | |
238 | 238 | } | |
239 | 239 | ||
240 | 240 | ||
241 | 241 | ||
242 | 242 | @Callable(i) | |
243 | 243 | func deposit () = { | |
244 | 244 | let caller = toString(i.caller) | |
245 | 245 | let baseAsset = _loadBaseAsset() | |
246 | 246 | let asset = _loadAsset() | |
247 | 247 | let assetQuantity = toBigInt(_getAssetQuantity(asset)) | |
248 | 248 | let err = if ((size(i.payments) != 1)) | |
249 | 249 | then throw("deposit: no payments") | |
250 | 250 | else if ((i.payments[0].assetId != baseAsset)) | |
251 | 251 | then throw("deposit: payment is not in base asset") | |
252 | 252 | else if ((0 >= i.payments[0].amount)) | |
253 | 253 | then throw("deposit: invalid payment amount") | |
254 | 254 | else unit | |
255 | 255 | if ((err == err)) | |
256 | 256 | then { | |
257 | 257 | let checkpoint = _checkpoint(false) | |
258 | 258 | if ((checkpoint == checkpoint)) | |
259 | 259 | then { | |
260 | 260 | let lastRate = _loadBigInt(KEY_LAST_RATE, PERCENT_FACTOR) | |
261 | 261 | let currentRate = _loadBigInt(KEY_CURRENT_RATE, ZERO_BIGINT) | |
262 | 262 | let lastHeight = _loadInt(KEY_LAST_HEIGHT, 0) | |
263 | 263 | let targetHeight = min([_loadInt(KEY_TARGET_HEIGHT, 0), height]) | |
264 | 264 | let lastRateUpdated = (lastRate + (currentRate * toBigInt((targetHeight - lastHeight)))) | |
265 | 265 | let issueAmount = toInt(fraction(toBigInt(i.payments[0].amount), PERCENT_FACTOR, lastRateUpdated, FLOOR)) | |
266 | 266 | let currentRateUpdated = fraction(currentRate, assetQuantity, (assetQuantity + toBigInt(issueAmount)), FLOOR) | |
267 | 267 | let lastHeightUpdated = targetHeight | |
268 | 268 | let stakeActions = _updateStaking(baseAsset, i.payments[0].amount, _loadStakingAdapter()) | |
269 | 269 | $Tuple2(([Reissue(asset, issueAmount, true), ScriptTransfer(i.caller, issueAmount, asset), BinaryEntry(KEY_LAST_RATE, toBytes(lastRateUpdated)), BinaryEntry(KEY_CURRENT_RATE, toBytes(currentRateUpdated)), IntegerEntry(KEY_LAST_HEIGHT, lastHeightUpdated)] ++ stakeActions), unit) | |
270 | 270 | } | |
271 | 271 | else throw("Strict value is not equal to itself.") | |
272 | 272 | } | |
273 | 273 | else throw("Strict value is not equal to itself.") | |
274 | 274 | } | |
275 | 275 | ||
276 | 276 | ||
277 | 277 | ||
278 | 278 | @Callable(i) | |
279 | 279 | func stake (amount_,spreadBlocks_) = { | |
280 | 280 | let caller = toString(i.caller) | |
281 | 281 | let baseAsset = _loadBaseAsset() | |
282 | 282 | let asset = _loadAsset() | |
283 | 283 | let assetQuantity = toBigInt(_getAssetQuantity(asset)) | |
284 | 284 | let err = if ((i.caller != this)) | |
285 | 285 | then throw("stake: only this contract") | |
286 | 286 | else if ((0 >= amount_)) | |
287 | 287 | then throw("stake: invalid stake amount") | |
288 | 288 | else if ((amount_ > _getAssetBalance(baseAsset, this))) | |
289 | 289 | then throw("stake: insufficient stake amount") | |
290 | 290 | else if ((assetQuantity == ZERO_BIGINT)) | |
291 | 291 | then throw("stake: no deposits to stake for") | |
292 | 292 | else unit | |
293 | 293 | if ((err == err)) | |
294 | 294 | then { | |
295 | 295 | let lastRate = _loadBigInt(KEY_LAST_RATE, PERCENT_FACTOR) | |
296 | 296 | let currentRate = _loadBigInt(KEY_CURRENT_RATE, ZERO_BIGINT) | |
297 | 297 | let lastHeight = _loadInt(KEY_LAST_HEIGHT, 0) | |
298 | 298 | let targetHeight = _loadInt(KEY_TARGET_HEIGHT, 0) | |
299 | 299 | let minTargetHeight = min([targetHeight, height]) | |
300 | 300 | let actions = if ((minTargetHeight == height)) | |
301 | 301 | then { | |
302 | 302 | let lastRateUpdated = (lastRate + (toBigInt((minTargetHeight - lastHeight)) * currentRate)) | |
303 | 303 | let remainingReward = (currentRate * toBigInt((targetHeight - minTargetHeight))) | |
304 | 304 | let currentRateUpdated = fraction(((toBigInt(amount_) * PERCENT_FACTOR) + remainingReward), ONE_BIGINT, (assetQuantity * toBigInt(spreadBlocks_))) | |
305 | 305 | let lastHeightUpdated = height | |
306 | 306 | let targetHeightUpdated = (lastHeightUpdated + spreadBlocks_) | |
307 | 307 | [BinaryEntry(KEY_LAST_RATE, toBytes(lastRateUpdated)), BinaryEntry(KEY_CURRENT_RATE, toBytes(currentRateUpdated)), IntegerEntry(KEY_LAST_HEIGHT, lastHeightUpdated), IntegerEntry(KEY_TARGET_HEIGHT, targetHeightUpdated)] | |
308 | 308 | } | |
309 | 309 | else if ((targetHeight > lastHeight)) | |
310 | 310 | then { | |
311 | 311 | let lastRateUpdated = (lastRate + (toBigInt((targetHeight - lastHeight)) * currentRate)) | |
312 | 312 | let currentRateUpdated = fraction(toBigInt(amount_), PERCENT_FACTOR, (assetQuantity * toBigInt(spreadBlocks_))) | |
313 | 313 | let lastHeightUpdated = height | |
314 | 314 | let targetHeightUpdated = (lastHeightUpdated + spreadBlocks_) | |
315 | 315 | [BinaryEntry(KEY_LAST_RATE, toBytes(lastRateUpdated)), BinaryEntry(KEY_CURRENT_RATE, toBytes(currentRateUpdated)), IntegerEntry(KEY_LAST_HEIGHT, lastHeightUpdated), IntegerEntry(KEY_TARGET_HEIGHT, targetHeightUpdated)] | |
316 | 316 | } | |
317 | 317 | else { | |
318 | 318 | let currentRateUpdated = fraction(toBigInt(amount_), PERCENT_FACTOR, (assetQuantity * toBigInt(spreadBlocks_))) | |
319 | 319 | let lastHeightUpdated = height | |
320 | 320 | let targetHeightUpdated = (lastHeightUpdated + spreadBlocks_) | |
321 | 321 | [BinaryEntry(KEY_CURRENT_RATE, toBytes(currentRateUpdated)), IntegerEntry(KEY_LAST_HEIGHT, lastHeightUpdated), IntegerEntry(KEY_TARGET_HEIGHT, targetHeightUpdated)] | |
322 | 322 | } | |
323 | 323 | let stakeActions = _updateStaking(baseAsset, amount_, _loadStakingAdapter()) | |
324 | 324 | $Tuple2((actions ++ stakeActions), unit) | |
325 | 325 | } | |
326 | 326 | else throw("Strict value is not equal to itself.") | |
327 | 327 | } | |
328 | 328 | ||
329 | 329 | ||
330 | 330 | ||
331 | 331 | @Callable(i) | |
332 | 332 | func withdraw () = { | |
333 | 333 | let caller = toString(i.caller) | |
334 | 334 | let baseAsset = _loadBaseAsset() | |
335 | 335 | let asset = _loadAsset() | |
336 | 336 | let assetQuantity = toBigInt(_getAssetQuantity(asset)) | |
337 | 337 | let err = if ((1 > size(i.payments))) | |
338 | 338 | then throw("withdraw: no payments") | |
339 | 339 | else if ((i.payments[0].assetId != asset)) | |
340 | 340 | then throw("withdraw: payment is not in correct asset") | |
341 | 341 | else if ((0 >= i.payments[0].amount)) | |
342 | 342 | then throw("withdraw: invalid payment amount") | |
343 | 343 | else unit | |
344 | 344 | if ((err == err)) | |
345 | 345 | then { | |
346 | 346 | let checkpoint = _checkpoint(false) | |
347 | 347 | if ((checkpoint == checkpoint)) | |
348 | 348 | then { | |
349 | 349 | let lastRate = _loadBigInt(KEY_LAST_RATE, PERCENT_FACTOR) | |
350 | 350 | let currentRate = _loadBigInt(KEY_CURRENT_RATE, ZERO_BIGINT) | |
351 | 351 | let lastHeight = _loadInt(KEY_LAST_HEIGHT, 0) | |
352 | 352 | let targetHeight = min([_loadInt(KEY_TARGET_HEIGHT, 0), height]) | |
353 | 353 | let lastRateUpdated = (lastRate + (currentRate * toBigInt((targetHeight - lastHeight)))) | |
354 | 354 | let baseAmount = toInt(fraction(toBigInt(i.payments[0].amount), lastRateUpdated, PERCENT_FACTOR, FLOOR)) | |
355 | 355 | let currentRateUpdated = fraction(currentRate, assetQuantity, (assetQuantity - toBigInt(i.payments[0].amount)), FLOOR) | |
356 | 356 | let lastHeightUpdated = targetHeight | |
357 | 357 | let stakeActions = _updateStaking(baseAsset, (-1 * baseAmount), _loadStakingAdapter()) | |
358 | 358 | $Tuple2((stakeActions ++ [Burn(asset, i.payments[0].amount), ScriptTransfer(i.caller, baseAmount, unit), BinaryEntry(KEY_LAST_RATE, toBytes(lastRateUpdated)), BinaryEntry(KEY_CURRENT_RATE, toBytes(currentRateUpdated)), IntegerEntry(KEY_LAST_HEIGHT, lastHeightUpdated)]), unit) | |
359 | 359 | } | |
360 | 360 | else throw("Strict value is not equal to itself.") | |
361 | 361 | } | |
362 | 362 | else throw("Strict value is not equal to itself.") | |
363 | 363 | } | |
364 | 364 | ||
365 | 365 | ||
366 | 366 | ||
367 | 367 | @Callable(i) | |
368 | 368 | func setMultisig (multisig_) = { | |
369 | 369 | let err = if ((i.caller != this)) | |
370 | 370 | then throw("setMultisig: permission denied") | |
371 | 371 | else if (!(_validateAddress(multisig_))) | |
372 | 372 | then throw("setMultisig: invalid multisig address") | |
373 | 373 | else unit | |
374 | 374 | if ((err == err)) | |
375 | 375 | then $Tuple2([StringEntry(KEY_MULTISIG, multisig_)], unit) | |
376 | 376 | else throw("Strict value is not equal to itself.") | |
377 | 377 | } | |
378 | 378 | ||
379 | 379 | ||
380 | 380 | ||
381 | 381 | @Callable(i) | |
382 | 382 | func setSponsorshipManager (manager_) = { | |
383 | 383 | let err = if ((i.caller != this)) | |
384 | 384 | then throw("setSponsorshipManager: permission denied") | |
385 | 385 | else if (!(_validateAddress(manager_))) | |
386 | 386 | then throw("setSponsorshipManager: invalid manager address") | |
387 | 387 | else unit | |
388 | 388 | if ((err == err)) | |
389 | 389 | then $Tuple2([StringEntry(KEY_SPONSORSHIP_MANAGER, manager_)], unit) | |
390 | 390 | else throw("Strict value is not equal to itself.") | |
391 | 391 | } | |
392 | 392 | ||
393 | 393 | ||
394 | 394 | ||
395 | 395 | @Callable(i) | |
396 | 396 | func updateSponsorship (minSponsoredAssetFee_,wavesRequiredAmount_) = { | |
397 | 397 | let err = if ((i.caller != _loadSponsorshipManager())) | |
398 | 398 | then throw("updateSponsorship: permission denied") | |
399 | 399 | else if ((0 > minSponsoredAssetFee_)) | |
400 | 400 | then throw("updateSponsorship: invalid sponsorship fee") | |
401 | 401 | else if ((0 > wavesRequiredAmount_)) | |
402 | 402 | then throw("updateSponsorship: invalid waves required amount") | |
403 | 403 | else unit | |
404 | 404 | if ((err == err)) | |
405 | 405 | then { | |
406 | 406 | let sponsorAsset = _loadAsset() | |
407 | 407 | let minSponsoredAssetFee = if ((minSponsoredAssetFee_ == 0)) | |
408 | 408 | then unit | |
409 | 409 | else minSponsoredAssetFee_ | |
410 | 410 | let actions = if ((wavesBalance(this).regular > wavesRequiredAmount_)) | |
411 | 411 | then [ScriptTransfer(i.caller, (wavesBalance(this).regular - wavesRequiredAmount_), unit)] | |
412 | 412 | else nil | |
413 | 413 | $Tuple2((actions ++ [ScriptTransfer(i.caller, assetBalance(this, sponsorAsset), sponsorAsset), SponsorFee(sponsorAsset, minSponsoredAssetFee)]), unit) | |
414 | 414 | } | |
415 | 415 | else throw("Strict value is not equal to itself.") | |
416 | 416 | } | |
417 | 417 | ||
418 | 418 | ||
419 | 419 | ||
420 | 420 | @Callable(i) | |
421 | - | func wxVotingSuggest (votingAddress_,wxAsset_,assetImage_) = { | |
422 | - | let votingAddress = valueOrErrorMessage(addressFromString(votingAddress_), "wxVotingSuggest: invalid voting address") | |
423 | - | let sponsorAssetStr = toBase58String(_loadAsset()) | |
424 | - | let wxAsset = _toAssetVector(wxAsset_) | |
425 | - | let wxAssetInfo = valueOrErrorMessage(assetInfo(valueOrErrorMessage(wxAsset, "wxVotingSuggest: invalid wx asset")), "wxVotingSuggest: invalid wx asset info") | |
426 | - | let invocation = invoke(votingAddress, "suggest", [sponsorAssetStr, assetImage_], [AttachedPayment(wxAsset, (100 * wxAssetInfo.decimals))]) | |
427 | - | if ((invocation == invocation)) | |
428 | - | then $Tuple2(nil, unit) | |
421 | + | func wxVotingSuggest (votingAddress_,wxAsset_,wxAmount_,assetImage_) = { | |
422 | + | let err = if ((i.caller != this)) | |
423 | + | then throw("wxVotingSuggest: permission denied") | |
424 | + | else if (!(_validateAddress(votingAddress_))) | |
425 | + | then throw("wxVotingSuggest: invalid voting address") | |
426 | + | else if ((0 > wxAmount_)) | |
427 | + | then throw("wxVotingSuggest: invalid wx amount") | |
428 | + | else unit | |
429 | + | if ((err == err)) | |
430 | + | then { | |
431 | + | let sponsorAssetStr = toBase58String(_loadAsset()) | |
432 | + | let wxAsset = _toAssetVector(wxAsset_) | |
433 | + | let wxAssetInfo = valueOrErrorMessage(assetInfo(valueOrErrorMessage(wxAsset, "wxVotingSuggest: invalid wx asset")), "wxVotingSuggest: invalid wx asset info") | |
434 | + | let invocation = invoke(addressFromStringValue(votingAddress_), "suggest", [sponsorAssetStr, assetImage_], [AttachedPayment(wxAsset, (wxAmount_ * wxAssetInfo.decimals))]) | |
435 | + | if ((invocation == invocation)) | |
436 | + | then $Tuple2(nil, unit) | |
437 | + | else throw("Strict value is not equal to itself.") | |
438 | + | } | |
429 | 439 | else throw("Strict value is not equal to itself.") | |
430 | 440 | } | |
431 | 441 | ||
432 | 442 | ||
433 | 443 | @Verifier(tx) | |
434 | 444 | func verify () = match getString(KEY_MULTISIG) { | |
435 | 445 | case multisig: String => | |
436 | 446 | valueOrElse(getBoolean(addressFromStringValue(multisig), makeString([KEY_STATUS, toString(this), toBase58String(tx.id)], SEPARATOR)), false) | |
437 | 447 | case _ => | |
438 | 448 | sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
439 | 449 | } | |
440 | 450 |
github/deemru/w8io/026f985 45.42 ms ◑