tx · 8egnpN8ni4wvEeZTEQndwqHJs7sWjcbTqaVqSKCEJc9M 3NBhCjipaSBkkoRSBdcyx2ui7wnY4HnxiVW: -0.01300000 Waves 2023.05.04 13:11 [2562877] smart account 3NBhCjipaSBkkoRSBdcyx2ui7wnY4HnxiVW > SELF 0.00000000 Waves
{ "type": 13, "id": "8egnpN8ni4wvEeZTEQndwqHJs7sWjcbTqaVqSKCEJc9M", "fee": 1300000, "feeAssetId": null, "timestamp": 1683195127225, "version": 2, "chainId": 84, "sender": "3NBhCjipaSBkkoRSBdcyx2ui7wnY4HnxiVW", "senderPublicKey": "FwXakLfcNwZygjHPVBd5TDZ4C5Wrgpoi9U69GufmFkQ", "proofs": [ "4SfLDikSbsx3gnsfQdBn1meYJHtjxzxtDs9cgaujNoRBBn1JFXwktRyHDkURfYcpa1mvYhevcoDAZ1gZQcjWqYFk" ], "script": "base64:BgIlCAISCgoICAgBAQEBAQESAwoBCBIDCgEIEgUKAwgIARIECgIICBIAA1NFUAICX18BE2tleU1hbmFnZXJQdWJsaWNLZXkAAhQlc19fbWFuYWdlclB1YmxpY0tleQEWa2V5TWFuYWdlclZhdWx0QWRkcmVzcwACFyVzX19tYW5hZ2VyVmF1bHRBZGRyZXNzARdnZXRNYW5hZ2VyQWRkcmVzc09yRmFpbAAJARFAZXh0ck5hdGl2ZSgxMDYyKQEJARFAZXh0ck5hdGl2ZSgxMDU4KQEJARZrZXlNYW5hZ2VyVmF1bHRBZGRyZXNzAAEQa2V5V2l0aGRyYXdEZWxheQIGYXNzZXRBBmFzc2V0QgkAuQkCCQDMCAICBiVzJXMlcwkAzAgCAg13aXRoZHJhd0RlbGF5CQDMCAIFBmFzc2V0QQkAzAgCBQZhc3NldEIFA25pbAUDU0VQARVrZXlEZXBvc2l0RmVlUGVybWlsbGUCBmFzc2V0QQZhc3NldEIJALkJAgkAzAgCAgYlcyVzJXMJAMwIAgISZGVwb3NpdEZlZVBlcm1pbGxlCQDMCAIFBmFzc2V0QQkAzAgCBQZhc3NldEIFA25pbAUDU0VQARZrZXlXaXRoZHJhd0ZlZVBlcm1pbGxlAgZhc3NldEEGYXNzZXRCCQC5CQIJAMwIAgIGJXMlcyVzCQDMCAICE3dpdGhkcmF3RmVlUGVybWlsbGUJAMwIAgUGYXNzZXRBCQDMCAIFBmFzc2V0QgUDbmlsBQNTRVABE2tleU1pbkFtb3VudERlcG9zaXQCBmFzc2V0QQZhc3NldEIJALkJAgkAzAgCAgYlcyVzJXMJAMwIAgIQbWluQW1vdW50RGVwb3NpdAkAzAgCBQZhc3NldEEJAMwIAgUGYXNzZXRCBQNuaWwFA1NFUAETa2V5QXNzZXRzUGFpclN0YXR1cwIGYXNzZXRBBmFzc2V0QgkAuQkCCQDMCAICBiVzJXMlcwkAzAgCAhBhc3NldHNQYWlyU3RhdHVzCQDMCAIFBmFzc2V0QQkAzAgCBQZhc3NldEIFA25pbAUDU0VQARRrZXlNaW5BbW91bnRXaXRoZHJhdwIGYXNzZXRBBmFzc2V0QgkAuQkCCQDMCAICBiVzJXMlcwkAzAgCAhFtaW5BbW91bnRXaXRoZHJhdwkAzAgCBQZhc3NldEEJAMwIAgUGYXNzZXRCBQNuaWwFA1NFUAEKa2V5QmFsYW5jZQMGYXNzZXRBBmFzc2V0Qgt1c2VyQWRkcmVzcwkAuQkCCQDMCAICCCVzJXMlcyVzCQDMCAICB2JhbGFuY2UJAMwIAgUGYXNzZXRBCQDMCAIFBmFzc2V0QgkAzAgCBQt1c2VyQWRkcmVzcwUDbmlsBQNTRVABG2tleVRvdGFsRmVlQ29sbGVjdGVkRGVwb3NpdAIGYXNzZXRBBmFzc2V0QgkAuQkCCQDMCAICCCVzJXMlcyVzCQDMCAICEXRvdGFsRmVlQ29sbGVjdGVkCQDMCAICB2RlcG9zaXQJAMwIAgUGYXNzZXRBCQDMCAIFBmFzc2V0QgUDbmlsBQNTRVABFGtleVByb2Nlc3NJblByb2dyZXNzBAt1c2VyQWRkcmVzcwZhc3NldEEGYXNzZXRCC2hlaWdodEluS2V5CQC5CQIJAMwIAgIMJXMlcyVzJXMlcyVkCQDMCAICD3dpdGhkcmF3UHJvY2VzcwkAzAgCAgppblByb2dyZXNzCQDMCAIFC3VzZXJBZGRyZXNzCQDMCAIFBmFzc2V0QQkAzAgCBQZhc3NldEIJAMwIAgkApAMBBQtoZWlnaHRJbktleQUDbmlsBQNTRVABDmtleVByb2Nlc3NEb25lBAt1c2VyQWRkcmVzcwZhc3NldEEGYXNzZXRCC2hlaWdodEluS2V5CQC5CQIJAMwIAgIMJXMlcyVzJXMlcyVkCQDMCAICD3dpdGhkcmF3UHJvY2VzcwkAzAgCAgRkb25lCQDMCAIFC3VzZXJBZGRyZXNzCQDMCAIFBmFzc2V0QQkAzAgCBQZhc3NldEIJAMwIAgkApAMBBQtoZWlnaHRJbktleQUDbmlsBQNTRVABFm1hbmFnZXJQdWJsaWNLZXlPclVuaXQABBNtYW5hZ2VyVmF1bHRBZGRyZXNzCQEXZ2V0TWFuYWdlckFkZHJlc3NPckZhaWwABAckbWF0Y2gwCQCdCAIFE21hbmFnZXJWYXVsdEFkZHJlc3MJARNrZXlNYW5hZ2VyUHVibGljS2V5AAMJAAECBQckbWF0Y2gwAgZTdHJpbmcEAXMFByRtYXRjaDAJANkEAQUBcwMJAAECBQckbWF0Y2gwAgRVbml0BQR1bml0CQACAQILTWF0Y2ggZXJyb3IBC211c3RNYW5hZ2VyAQFpBAJwZAkAAgECElBlcm1pc3Npb24gZGVuaWVkLgQHJG1hdGNoMAkBFm1hbmFnZXJQdWJsaWNLZXlPclVuaXQAAwkAAQIFByRtYXRjaDACCkJ5dGVWZWN0b3IEAnBrBQckbWF0Y2gwAwkAAAIIBQFpD2NhbGxlclB1YmxpY0tleQUCcGsGBQJwZAMJAAECBQckbWF0Y2gwAgRVbml0AwkAAAIIBQFpBmNhbGxlcgUEdGhpcwYFAnBkCQACAQILTWF0Y2ggZXJyb3IBBmZtdEVycgEDbXNnCQC5CQIJAMwIAgIUb3RjX211bHRpYXNzZXQucmlkZToJAMwIAgUDbXNnBQNuaWwCASABCHRocm93RXJyAQNtc2cJAAIBCQEGZm10RXJyAQUDbXNnBQFpAQ1yZWdpc3RlckFzc2V0CAZhc3NldEEGYXNzZXRCDXdpdGhkcmF3RGVsYXkKZGVwb3NpdEZlZQt3aXRoZHJhd0ZlZRBtaW5BbW91bnREZXBvc2l0EW1pbkFtb3VudFdpdGhkcmF3CnBhaXJTdGF0dXMEC2NoZWNrQ2FsbGVyCQELbXVzdE1hbmFnZXIBBQFpAwkAAAIFC2NoZWNrQ2FsbGVyBQtjaGVja0NhbGxlcgQQd2l0aGRyYXdEZWxheUtleQkBEGtleVdpdGhkcmF3RGVsYXkCBQZhc3NldEEFBmFzc2V0QgQVZGVwb3NpdEZlZVBlcm1pbGxlS2V5CQEVa2V5RGVwb3NpdEZlZVBlcm1pbGxlAgUGYXNzZXRBBQZhc3NldEIEFndpdGhkcmF3RmVlUGVybWlsbGVLZXkJARZrZXlXaXRoZHJhd0ZlZVBlcm1pbGxlAgUGYXNzZXRBBQZhc3NldEIEE21pbkFtb3VudERlcG9zaXRLZXkJARNrZXlNaW5BbW91bnREZXBvc2l0AgUGYXNzZXRBBQZhc3NldEIEFG1pbkFtb3VudFdpdGhkcmF3S2V5CQEUa2V5TWluQW1vdW50V2l0aGRyYXcCBQZhc3NldEEFBmFzc2V0QgQNcGFpclN0YXR1c0tleQkBE2tleUFzc2V0c1BhaXJTdGF0dXMCBQZhc3NldEEFBmFzc2V0QgkAzAgCCQEMSW50ZWdlckVudHJ5AgUQd2l0aGRyYXdEZWxheUtleQUNd2l0aGRyYXdEZWxheQkAzAgCCQEMSW50ZWdlckVudHJ5AgUVZGVwb3NpdEZlZVBlcm1pbGxlS2V5BQpkZXBvc2l0RmVlCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRZ3aXRoZHJhd0ZlZVBlcm1pbGxlS2V5BQt3aXRoZHJhd0ZlZQkAzAgCCQEMSW50ZWdlckVudHJ5AgUTbWluQW1vdW50RGVwb3NpdEtleQUQbWluQW1vdW50RGVwb3NpdAkAzAgCCQEMSW50ZWdlckVudHJ5AgUUbWluQW1vdW50V2l0aGRyYXdLZXkFEW1pbkFtb3VudFdpdGhkcmF3CQDMCAIJAQxJbnRlZ2VyRW50cnkCBQ1wYWlyU3RhdHVzS2V5BQpwYWlyU3RhdHVzBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEOc3dhcEFzc2V0c0FUb0IBBmFzc2V0QgQHcGF5bWVudAkBBXZhbHVlAQkAkQMCCAUBaQhwYXltZW50cwAABAZhc3NldEEJANgEAQkBBXZhbHVlAQgFB3BheW1lbnQHYXNzZXRJZAQFYXNzZXQJANkEAQkBBXZhbHVlAQUGYXNzZXRCBAt1c2VyQWRkcmVzcwkApQgBCAUBaQZjYWxsZXIEEG1pbkFtb3VudERlcG9zaXQJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnwgBCQETa2V5TWluQW1vdW50RGVwb3NpdAIFBmFzc2V0QQUGYXNzZXRCCQEGZm10RXJyAQIfVGhpcyBhc3NldCBwYWlyIGRvZXMgbm90IGV4aXN0LgQJdG9EZXBvc2l0CAUHcGF5bWVudAZhbW91bnQECmRlcG9zaXRGZWUJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnwgBCQEVa2V5RGVwb3NpdEZlZVBlcm1pbGxlAgUGYXNzZXRBBQZhc3NldEIJAQZmbXRFcnIBAjNUaGUgZGVwb3NpdCBmZWUgZm9yIHRoaXMgcGFpciBvZiBhc3NldHMgaXMgbm90IHNldC4EA2ZlZQkAaAIJAGkCBQl0b0RlcG9zaXQA6AcFCmRlcG9zaXRGZWUEEmN1cnJlbnRVc2VyQmFsYW5jZQkBC3ZhbHVlT3JFbHNlAgkAnwgBCQEKa2V5QmFsYW5jZQMFBmFzc2V0QQUGYXNzZXRCBQt1c2VyQWRkcmVzcwAABBB0b3RhbENvbW1pc3Npb25zCQELdmFsdWVPckVsc2UCCQCfCAEJARtrZXlUb3RhbEZlZUNvbGxlY3RlZERlcG9zaXQCBQZhc3NldEEFBmFzc2V0QgAABApwYWlyU3RhdHVzCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJ8IAQkBE2tleUFzc2V0c1BhaXJTdGF0dXMCBQZhc3NldEEFBmFzc2V0QgkBBmZtdEVycgECOVRoZSBhc3NldCBwYWlyIHN0YXR1cyBmb3IgdGhpcyBwYWlyIG9mIGFzc2V0cyBpcyBub3Qgc2V0LgQPY2hlY2tQYWlyU3RhdHVzAwkAAAIFCnBhaXJTdGF0dXMAAAYJAQh0aHJvd0VycgECIFRoZSBjb3VwbGUncyBkZXBvc2l0IGlzIGJsb2NrZWQuAwkAAAIFD2NoZWNrUGFpclN0YXR1cwUPY2hlY2tQYWlyU3RhdHVzBAxjaGVja1BheW1lbnQDCQBnAgUJdG9EZXBvc2l0BRBtaW5BbW91bnREZXBvc2l0BgkBCHRocm93RXJyAQIsVGhlIGRlcG9zaXQgYW1vdW50IGlzIGxlc3MgdGhhbiB0aGUgbWluaW11bS4DCQAAAgUMY2hlY2tQYXltZW50BQxjaGVja1BheW1lbnQECm5ld0JhbGFuY2UJAGUCCQBkAgUSY3VycmVudFVzZXJCYWxhbmNlBQl0b0RlcG9zaXQFA2ZlZQQMY2hlY2tCYWxhbmNlAwkAZgIFCm5ld0JhbGFuY2UAAAYJAQh0aHJvd0VycgECLVRoZSBmaW5hbCBiYWxhbmNlIGlzIGxlc3MgdGhhbiBvciBlcXVhbCB0byAwLgMJAAACBQxjaGVja0JhbGFuY2UFDGNoZWNrQmFsYW5jZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkBCmtleUJhbGFuY2UDBQZhc3NldEEFBmFzc2V0QgULdXNlckFkZHJlc3MFCm5ld0JhbGFuY2UJAMwIAgkBDEludGVnZXJFbnRyeQIJARtrZXlUb3RhbEZlZUNvbGxlY3RlZERlcG9zaXQCBQZhc3NldEEFBmFzc2V0QgkAZAIFEHRvdGFsQ29tbWlzc2lvbnMFA2ZlZQkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUBaQZjYWxsZXIJAGUCBQl0b0RlcG9zaXQFA2ZlZQUFYXNzZXQFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpARxpbml0aWFsaXphdGlvblN3YXBBc3NldHNCVG9BAQZhc3NldEEEB3BheW1lbnQJAQV2YWx1ZQEJAJEDAggFAWkIcGF5bWVudHMAAAQKdG9XaXRoZHJhdwgFB3BheW1lbnQGYW1vdW50BAZhc3NldEIJANgEAQkBBXZhbHVlAQgFB3BheW1lbnQHYXNzZXRJZAQLdXNlckFkZHJlc3MJAKUIAQgFAWkGY2FsbGVyBBFtaW5BbW91bnRXaXRoZHJhdwkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCfCAEJARRrZXlNaW5BbW91bnRXaXRoZHJhdwIFBmFzc2V0QQUGYXNzZXRCCQEGZm10RXJyAQJBVGhlIG1pbmltdW0gd2l0aGRyYXdhbCBhbW91bnQgZm9yIHRoaXMgcGFpciBvZiBhc3NldHMgaXMgbm90IHNldC4ED2Fzc2V0TG9ja0hlaWdodAkAZAIFBmhlaWdodAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCfCAEJARBrZXlXaXRoZHJhd0RlbGF5AgUGYXNzZXRBBQZhc3NldEIJAQZmbXRFcnIBAjNXaXRoZHJhd2FsIGRlbGF5IGlzIG5vdCBzZXQgZm9yIHRoZSBzcGVjaWZpZWQgcGFpci4EEmN1cnJlbnRVc2VyQmFsYW5jZQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCfCAEJAQprZXlCYWxhbmNlAwUGYXNzZXRBBQZhc3NldEIFC3VzZXJBZGRyZXNzCQEGZm10RXJyAQIhVGhpcyB1c2VyIGJhbGFuY2UgZG9lcyBub3QgZXhpc3QuBBB0b3RhbENvbW1pc3Npb25zCQELdmFsdWVPckVsc2UCCQCfCAEJARtrZXlUb3RhbEZlZUNvbGxlY3RlZERlcG9zaXQCBQZhc3NldEEFBmFzc2V0QgAABBZ3aXRoZHJhd0ZlZVBlcm1pbGxlS2V5CQEWa2V5V2l0aGRyYXdGZWVQZXJtaWxsZQIFBmFzc2V0QQUGYXNzZXRCBAt3aXRoZHJhd0ZlZQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCfCAEFFndpdGhkcmF3RmVlUGVybWlsbGVLZXkJAQZmbXRFcnIBAjZUaGUgd2l0aGRyYXdhbCBmZWUgZm9yIHRoaXMgcGFpciBvZiBhc3NldHMgaXMgbm90IHNldC4EA2ZlZQkAaAIJAGkCBQp0b1dpdGhkcmF3AOgHBQt3aXRoZHJhd0ZlZQQKbmV3QmFsYW5jZQkAZQIFEmN1cnJlbnRVc2VyQmFsYW5jZQUKdG9XaXRoZHJhdwQMY2hlY2tCYWxhbmNlAwkAZwIFCm5ld0JhbGFuY2UAAAYJAQh0aHJvd0VycgECJVN3YXAgYW1vdW50IGZhaWwsIGFtb3VudCBpcyB0byBzbWFsbC4DCQAAAgUMY2hlY2tCYWxhbmNlBQxjaGVja0JhbGFuY2UEDGNoZWNrUGF5bWVudAMJAGcCBQp0b1dpdGhkcmF3BRFtaW5BbW91bnRXaXRoZHJhdwYJAQh0aHJvd0VycgECLVRoZSB3aXRoZHJhdyBhbW91bnQgaXMgbGVzcyB0aGFuIHRoZSBtaW5pbXVtLgMJAAACBQxjaGVja1BheW1lbnQFDGNoZWNrUGF5bWVudAQWY2hlY2tQcm9jZXNzSW5Qcm9ncmVzcwMJAAACCQCfCAEJARRrZXlQcm9jZXNzSW5Qcm9ncmVzcwQFC3VzZXJBZGRyZXNzBQZhc3NldEEFBmFzc2V0QgUPYXNzZXRMb2NrSGVpZ2h0BQR1bml0BgkBCHRocm93RXJyAQI6QXQgdGhpcyBoZWlnaHQsIHRoZXJlIGlzIGFscmVhZHkgYW4gZXhjaGFuZ2Ugb2YgdGhpcyBwYWlyLgMJAAACBRZjaGVja1Byb2Nlc3NJblByb2dyZXNzBRZjaGVja1Byb2Nlc3NJblByb2dyZXNzCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEKa2V5QmFsYW5jZQMFBmFzc2V0QQUGYXNzZXRCBQt1c2VyQWRkcmVzcwUKbmV3QmFsYW5jZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkBFGtleVByb2Nlc3NJblByb2dyZXNzBAULdXNlckFkZHJlc3MFBmFzc2V0QQUGYXNzZXRCBQ9hc3NldExvY2tIZWlnaHQJAGUCBQp0b1dpdGhkcmF3BQNmZWUJAMwIAgkBDEludGVnZXJFbnRyeQIJARtrZXlUb3RhbEZlZUNvbGxlY3RlZERlcG9zaXQCBQZhc3NldEEFBmFzc2V0QgkAZAIFEHRvdGFsQ29tbWlzc2lvbnMFA2ZlZQUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBDXdpdGhkcmF3QXNzZXQDBmFzc2V0QQZhc3NldEILaGVpZ2h0SW5LZXkEC3VzZXJBZGRyZXNzCQClCAEIBQFpBmNhbGxlcgQKdG9XaXRoZHJhdwkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCfCAEJARRrZXlQcm9jZXNzSW5Qcm9ncmVzcwQFC3VzZXJBZGRyZXNzBQZhc3NldEEFBmFzc2V0QgULaGVpZ2h0SW5LZXkJAQZmbXRFcnIBAkZBdCB0aGlzIGhlaWdodCwgd2l0aGRyYXcgd2FzIG5vdCBpbml0aWFsaXplZCB3aXRoIHRoaXMgcGFpciBvZiBhc3NldHMuBAVhc3NldAkA2QQBCQEFdmFsdWUBBQZhc3NldEEEC2NoZWNrSGVpZ2h0AwkAZwIFBmhlaWdodAULaGVpZ2h0SW5LZXkGCQEIdGhyb3dFcnIBCQCsAgIJAKwCAgIdV2l0aGRyYXdhbCBpcyBwb3NzaWJsZSBhZnRlciAJAKQDAQULaGVpZ2h0SW5LZXkCJiBoZWlnaHQgb3IgeW91IGhhdmUgYWxyZWFkeSB3aXRoZHJhd24uAwkAAAIFC2NoZWNrSGVpZ2h0BQtjaGVja0hlaWdodAkAzAgCCQELRGVsZXRlRW50cnkBCQEUa2V5UHJvY2Vzc0luUHJvZ3Jlc3MEBQt1c2VyQWRkcmVzcwUGYXNzZXRBBQZhc3NldEIFC2hlaWdodEluS2V5CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOa2V5UHJvY2Vzc0RvbmUEBQt1c2VyQWRkcmVzcwUGYXNzZXRBBQZhc3NldEIFC2hlaWdodEluS2V5BQp0b1dpdGhkcmF3CQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMIBQFpBmNhbGxlcgUKdG9XaXRoZHJhdwUFYXNzZXQFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQt3aXRoZHJhd0ZlZQIGYXNzZXRBBmFzc2V0QgQLY2hlY2tDYWxsZXIJAQttdXN0TWFuYWdlcgEFAWkDCQAAAgULY2hlY2tDYWxsZXIFC2NoZWNrQ2FsbGVyBAt0b1dpdGhkcmF3QQkBC3ZhbHVlT3JFbHNlAgkAnwgBCQEba2V5VG90YWxGZWVDb2xsZWN0ZWREZXBvc2l0AgUGYXNzZXRBBQZhc3NldEIAAAQOd2l0aGRyYXdBc3NldEEJANkEAQkBBXZhbHVlAQUGYXNzZXRBCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEba2V5VG90YWxGZWVDb2xsZWN0ZWREZXBvc2l0AgUGYXNzZXRBBQZhc3NldEIAAAkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUBaQZjYWxsZXIFC3RvV2l0aGRyYXdBBQ53aXRoZHJhd0Fzc2V0QQUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAQJ0eAEGdmVyaWZ5AAQPdGFyZ2V0UHVibGljS2V5BAckbWF0Y2gwCQEWbWFuYWdlclB1YmxpY0tleU9yVW5pdAADCQABAgUHJG1hdGNoMAIKQnl0ZVZlY3RvcgQCcGsFByRtYXRjaDAFAnBrAwkAAQIFByRtYXRjaDACBFVuaXQIBQJ0eA9zZW5kZXJQdWJsaWNLZXkJAAIBAgtNYXRjaCBlcnJvcgkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAUPdGFyZ2V0UHVibGljS2V5FVMfmg==", "height": 2562877, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: CLSZFcsU3jobwVaGwq9wqTvsA9R4hKieuD4Fezk7fgAU Next: none Diff:
Old | New | Differences | |
---|---|---|---|
6 | 6 | func keyManagerPublicKey () = "%s__managerPublicKey" | |
7 | 7 | ||
8 | 8 | ||
9 | - | func keyPendingManagerPublicKey () = "%s__pendingManagerPublicKey" | |
9 | + | func keyManagerVaultAddress () = "%s__managerVaultAddress" | |
10 | + | ||
11 | + | ||
12 | + | func getManagerAddressOrFail () = addressFromStringValue(getStringValue(keyManagerVaultAddress())) | |
10 | 13 | ||
11 | 14 | ||
12 | 15 | func keyWithdrawDelay (assetA,assetB) = makeString(["%s%s%s", "withdrawDelay", assetA, assetB], SEP) | |
39 | 42 | func keyProcessDone (userAddress,assetA,assetB,heightInKey) = makeString(["%s%s%s%s%s%d", "withdrawProcess", "done", userAddress, assetA, assetB, toString(heightInKey)], SEP) | |
40 | 43 | ||
41 | 44 | ||
42 | - | func managerPublicKeyOrUnit () = match getString(keyManagerPublicKey()) { | |
43 | - | case s: String => | |
44 | - | fromBase58String(s) | |
45 | - | case _: Unit => | |
46 | - | unit | |
47 | - | case _ => | |
48 | - | throw("Match error") | |
49 | - | } | |
50 | - | ||
51 | - | ||
52 | - | func pendingManagerPublicKeyOrUnit () = match getString(keyPendingManagerPublicKey()) { | |
53 | - | case s: String => | |
54 | - | fromBase58String(s) | |
55 | - | case _: Unit => | |
56 | - | unit | |
57 | - | case _ => | |
58 | - | throw("Match error") | |
59 | - | } | |
45 | + | func managerPublicKeyOrUnit () = { | |
46 | + | let managerVaultAddress = getManagerAddressOrFail() | |
47 | + | match getString(managerVaultAddress, keyManagerPublicKey()) { | |
48 | + | case s: String => | |
49 | + | fromBase58String(s) | |
50 | + | case _: Unit => | |
51 | + | unit | |
52 | + | case _ => | |
53 | + | throw("Match error") | |
54 | + | } | |
55 | + | } | |
60 | 56 | ||
61 | 57 | ||
62 | 58 | func mustManager (i) = { | |
204 | 200 | } | |
205 | 201 | ||
206 | 202 | ||
207 | - | ||
208 | - | @Callable(i) | |
209 | - | func setManager (pendingManagerPublicKey) = { | |
210 | - | let checkCaller = mustManager(i) | |
211 | - | if ((checkCaller == checkCaller)) | |
212 | - | then { | |
213 | - | let checkManagerPublicKey = fromBase58String(pendingManagerPublicKey) | |
214 | - | if ((checkManagerPublicKey == checkManagerPublicKey)) | |
215 | - | then [StringEntry(keyPendingManagerPublicKey(), pendingManagerPublicKey)] | |
216 | - | else throw("Strict value is not equal to itself.") | |
217 | - | } | |
218 | - | else throw("Strict value is not equal to itself.") | |
219 | - | } | |
220 | - | ||
221 | - | ||
222 | - | ||
223 | - | @Callable(i) | |
224 | - | func confirmManager () = { | |
225 | - | let pm = pendingManagerPublicKeyOrUnit() | |
226 | - | let hasPM = if (isDefined(pm)) | |
227 | - | then true | |
228 | - | else throwErr("No pending manager.") | |
229 | - | if ((hasPM == hasPM)) | |
230 | - | then { | |
231 | - | let checkPM = if ((i.callerPublicKey == value(pm))) | |
232 | - | then true | |
233 | - | else throwErr("You are not pending manager.") | |
234 | - | if ((checkPM == checkPM)) | |
235 | - | then [StringEntry(keyManagerPublicKey(), toBase58String(value(pm))), DeleteEntry(keyPendingManagerPublicKey())] | |
236 | - | else throw("Strict value is not equal to itself.") | |
237 | - | } | |
238 | - | else throw("Strict value is not equal to itself.") | |
239 | - | } | |
240 | - | ||
241 | - | ||
242 | 203 | @Verifier(tx) | |
243 | 204 | func verify () = { | |
244 | 205 | let targetPublicKey = match managerPublicKeyOrUnit() { |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let SEP = "__" | |
5 | 5 | ||
6 | 6 | func keyManagerPublicKey () = "%s__managerPublicKey" | |
7 | 7 | ||
8 | 8 | ||
9 | - | func keyPendingManagerPublicKey () = "%s__pendingManagerPublicKey" | |
9 | + | func keyManagerVaultAddress () = "%s__managerVaultAddress" | |
10 | + | ||
11 | + | ||
12 | + | func getManagerAddressOrFail () = addressFromStringValue(getStringValue(keyManagerVaultAddress())) | |
10 | 13 | ||
11 | 14 | ||
12 | 15 | func keyWithdrawDelay (assetA,assetB) = makeString(["%s%s%s", "withdrawDelay", assetA, assetB], SEP) | |
13 | 16 | ||
14 | 17 | ||
15 | 18 | func keyDepositFeePermille (assetA,assetB) = makeString(["%s%s%s", "depositFeePermille", assetA, assetB], SEP) | |
16 | 19 | ||
17 | 20 | ||
18 | 21 | func keyWithdrawFeePermille (assetA,assetB) = makeString(["%s%s%s", "withdrawFeePermille", assetA, assetB], SEP) | |
19 | 22 | ||
20 | 23 | ||
21 | 24 | func keyMinAmountDeposit (assetA,assetB) = makeString(["%s%s%s", "minAmountDeposit", assetA, assetB], SEP) | |
22 | 25 | ||
23 | 26 | ||
24 | 27 | func keyAssetsPairStatus (assetA,assetB) = makeString(["%s%s%s", "assetsPairStatus", assetA, assetB], SEP) | |
25 | 28 | ||
26 | 29 | ||
27 | 30 | func keyMinAmountWithdraw (assetA,assetB) = makeString(["%s%s%s", "minAmountWithdraw", assetA, assetB], SEP) | |
28 | 31 | ||
29 | 32 | ||
30 | 33 | func keyBalance (assetA,assetB,userAddress) = makeString(["%s%s%s%s", "balance", assetA, assetB, userAddress], SEP) | |
31 | 34 | ||
32 | 35 | ||
33 | 36 | func keyTotalFeeCollectedDeposit (assetA,assetB) = makeString(["%s%s%s%s", "totalFeeCollected", "deposit", assetA, assetB], SEP) | |
34 | 37 | ||
35 | 38 | ||
36 | 39 | func keyProcessInProgress (userAddress,assetA,assetB,heightInKey) = makeString(["%s%s%s%s%s%d", "withdrawProcess", "inProgress", userAddress, assetA, assetB, toString(heightInKey)], SEP) | |
37 | 40 | ||
38 | 41 | ||
39 | 42 | func keyProcessDone (userAddress,assetA,assetB,heightInKey) = makeString(["%s%s%s%s%s%d", "withdrawProcess", "done", userAddress, assetA, assetB, toString(heightInKey)], SEP) | |
40 | 43 | ||
41 | 44 | ||
42 | - | func managerPublicKeyOrUnit () = match getString(keyManagerPublicKey()) { | |
43 | - | case s: String => | |
44 | - | fromBase58String(s) | |
45 | - | case _: Unit => | |
46 | - | unit | |
47 | - | case _ => | |
48 | - | throw("Match error") | |
49 | - | } | |
50 | - | ||
51 | - | ||
52 | - | func pendingManagerPublicKeyOrUnit () = match getString(keyPendingManagerPublicKey()) { | |
53 | - | case s: String => | |
54 | - | fromBase58String(s) | |
55 | - | case _: Unit => | |
56 | - | unit | |
57 | - | case _ => | |
58 | - | throw("Match error") | |
59 | - | } | |
45 | + | func managerPublicKeyOrUnit () = { | |
46 | + | let managerVaultAddress = getManagerAddressOrFail() | |
47 | + | match getString(managerVaultAddress, keyManagerPublicKey()) { | |
48 | + | case s: String => | |
49 | + | fromBase58String(s) | |
50 | + | case _: Unit => | |
51 | + | unit | |
52 | + | case _ => | |
53 | + | throw("Match error") | |
54 | + | } | |
55 | + | } | |
60 | 56 | ||
61 | 57 | ||
62 | 58 | func mustManager (i) = { | |
63 | 59 | let pd = throw("Permission denied.") | |
64 | 60 | match managerPublicKeyOrUnit() { | |
65 | 61 | case pk: ByteVector => | |
66 | 62 | if ((i.callerPublicKey == pk)) | |
67 | 63 | then true | |
68 | 64 | else pd | |
69 | 65 | case _: Unit => | |
70 | 66 | if ((i.caller == this)) | |
71 | 67 | then true | |
72 | 68 | else pd | |
73 | 69 | case _ => | |
74 | 70 | throw("Match error") | |
75 | 71 | } | |
76 | 72 | } | |
77 | 73 | ||
78 | 74 | ||
79 | 75 | func fmtErr (msg) = makeString(["otc_multiasset.ride:", msg], " ") | |
80 | 76 | ||
81 | 77 | ||
82 | 78 | func throwErr (msg) = throw(fmtErr(msg)) | |
83 | 79 | ||
84 | 80 | ||
85 | 81 | @Callable(i) | |
86 | 82 | func registerAsset (assetA,assetB,withdrawDelay,depositFee,withdrawFee,minAmountDeposit,minAmountWithdraw,pairStatus) = { | |
87 | 83 | let checkCaller = mustManager(i) | |
88 | 84 | if ((checkCaller == checkCaller)) | |
89 | 85 | then { | |
90 | 86 | let withdrawDelayKey = keyWithdrawDelay(assetA, assetB) | |
91 | 87 | let depositFeePermilleKey = keyDepositFeePermille(assetA, assetB) | |
92 | 88 | let withdrawFeePermilleKey = keyWithdrawFeePermille(assetA, assetB) | |
93 | 89 | let minAmountDepositKey = keyMinAmountDeposit(assetA, assetB) | |
94 | 90 | let minAmountWithdrawKey = keyMinAmountWithdraw(assetA, assetB) | |
95 | 91 | let pairStatusKey = keyAssetsPairStatus(assetA, assetB) | |
96 | 92 | [IntegerEntry(withdrawDelayKey, withdrawDelay), IntegerEntry(depositFeePermilleKey, depositFee), IntegerEntry(withdrawFeePermilleKey, withdrawFee), IntegerEntry(minAmountDepositKey, minAmountDeposit), IntegerEntry(minAmountWithdrawKey, minAmountWithdraw), IntegerEntry(pairStatusKey, pairStatus)] | |
97 | 93 | } | |
98 | 94 | else throw("Strict value is not equal to itself.") | |
99 | 95 | } | |
100 | 96 | ||
101 | 97 | ||
102 | 98 | ||
103 | 99 | @Callable(i) | |
104 | 100 | func swapAssetsAToB (assetB) = { | |
105 | 101 | let payment = value(i.payments[0]) | |
106 | 102 | let assetA = toBase58String(value(payment.assetId)) | |
107 | 103 | let asset = fromBase58String(value(assetB)) | |
108 | 104 | let userAddress = toString(i.caller) | |
109 | 105 | let minAmountDeposit = valueOrErrorMessage(getInteger(keyMinAmountDeposit(assetA, assetB)), fmtErr("This asset pair does not exist.")) | |
110 | 106 | let toDeposit = payment.amount | |
111 | 107 | let depositFee = valueOrErrorMessage(getInteger(keyDepositFeePermille(assetA, assetB)), fmtErr("The deposit fee for this pair of assets is not set.")) | |
112 | 108 | let fee = ((toDeposit / 1000) * depositFee) | |
113 | 109 | let currentUserBalance = valueOrElse(getInteger(keyBalance(assetA, assetB, userAddress)), 0) | |
114 | 110 | let totalCommissions = valueOrElse(getInteger(keyTotalFeeCollectedDeposit(assetA, assetB)), 0) | |
115 | 111 | let pairStatus = valueOrErrorMessage(getInteger(keyAssetsPairStatus(assetA, assetB)), fmtErr("The asset pair status for this pair of assets is not set.")) | |
116 | 112 | let checkPairStatus = if ((pairStatus == 0)) | |
117 | 113 | then true | |
118 | 114 | else throwErr("The couple's deposit is blocked.") | |
119 | 115 | if ((checkPairStatus == checkPairStatus)) | |
120 | 116 | then { | |
121 | 117 | let checkPayment = if ((toDeposit >= minAmountDeposit)) | |
122 | 118 | then true | |
123 | 119 | else throwErr("The deposit amount is less than the minimum.") | |
124 | 120 | if ((checkPayment == checkPayment)) | |
125 | 121 | then { | |
126 | 122 | let newBalance = ((currentUserBalance + toDeposit) - fee) | |
127 | 123 | let checkBalance = if ((newBalance > 0)) | |
128 | 124 | then true | |
129 | 125 | else throwErr("The final balance is less than or equal to 0.") | |
130 | 126 | if ((checkBalance == checkBalance)) | |
131 | 127 | then [IntegerEntry(keyBalance(assetA, assetB, userAddress), newBalance), IntegerEntry(keyTotalFeeCollectedDeposit(assetA, assetB), (totalCommissions + fee)), ScriptTransfer(i.caller, (toDeposit - fee), asset)] | |
132 | 128 | else throw("Strict value is not equal to itself.") | |
133 | 129 | } | |
134 | 130 | else throw("Strict value is not equal to itself.") | |
135 | 131 | } | |
136 | 132 | else throw("Strict value is not equal to itself.") | |
137 | 133 | } | |
138 | 134 | ||
139 | 135 | ||
140 | 136 | ||
141 | 137 | @Callable(i) | |
142 | 138 | func initializationSwapAssetsBToA (assetA) = { | |
143 | 139 | let payment = value(i.payments[0]) | |
144 | 140 | let toWithdraw = payment.amount | |
145 | 141 | let assetB = toBase58String(value(payment.assetId)) | |
146 | 142 | let userAddress = toString(i.caller) | |
147 | 143 | let minAmountWithdraw = valueOrErrorMessage(getInteger(keyMinAmountWithdraw(assetA, assetB)), fmtErr("The minimum withdrawal amount for this pair of assets is not set.")) | |
148 | 144 | let assetLockHeight = (height + valueOrErrorMessage(getInteger(keyWithdrawDelay(assetA, assetB)), fmtErr("Withdrawal delay is not set for the specified pair."))) | |
149 | 145 | let currentUserBalance = valueOrErrorMessage(getInteger(keyBalance(assetA, assetB, userAddress)), fmtErr("This user balance does not exist.")) | |
150 | 146 | let totalCommissions = valueOrElse(getInteger(keyTotalFeeCollectedDeposit(assetA, assetB)), 0) | |
151 | 147 | let withdrawFeePermilleKey = keyWithdrawFeePermille(assetA, assetB) | |
152 | 148 | let withdrawFee = valueOrErrorMessage(getInteger(withdrawFeePermilleKey), fmtErr("The withdrawal fee for this pair of assets is not set.")) | |
153 | 149 | let fee = ((toWithdraw / 1000) * withdrawFee) | |
154 | 150 | let newBalance = (currentUserBalance - toWithdraw) | |
155 | 151 | let checkBalance = if ((newBalance >= 0)) | |
156 | 152 | then true | |
157 | 153 | else throwErr("Swap amount fail, amount is to small.") | |
158 | 154 | if ((checkBalance == checkBalance)) | |
159 | 155 | then { | |
160 | 156 | let checkPayment = if ((toWithdraw >= minAmountWithdraw)) | |
161 | 157 | then true | |
162 | 158 | else throwErr("The withdraw amount is less than the minimum.") | |
163 | 159 | if ((checkPayment == checkPayment)) | |
164 | 160 | then { | |
165 | 161 | let checkProcessInProgress = if ((getInteger(keyProcessInProgress(userAddress, assetA, assetB, assetLockHeight)) == unit)) | |
166 | 162 | then true | |
167 | 163 | else throwErr("At this height, there is already an exchange of this pair.") | |
168 | 164 | if ((checkProcessInProgress == checkProcessInProgress)) | |
169 | 165 | then [IntegerEntry(keyBalance(assetA, assetB, userAddress), newBalance), IntegerEntry(keyProcessInProgress(userAddress, assetA, assetB, assetLockHeight), (toWithdraw - fee)), IntegerEntry(keyTotalFeeCollectedDeposit(assetA, assetB), (totalCommissions + fee))] | |
170 | 166 | else throw("Strict value is not equal to itself.") | |
171 | 167 | } | |
172 | 168 | else throw("Strict value is not equal to itself.") | |
173 | 169 | } | |
174 | 170 | else throw("Strict value is not equal to itself.") | |
175 | 171 | } | |
176 | 172 | ||
177 | 173 | ||
178 | 174 | ||
179 | 175 | @Callable(i) | |
180 | 176 | func withdrawAsset (assetA,assetB,heightInKey) = { | |
181 | 177 | let userAddress = toString(i.caller) | |
182 | 178 | let toWithdraw = valueOrErrorMessage(getInteger(keyProcessInProgress(userAddress, assetA, assetB, heightInKey)), fmtErr("At this height, withdraw was not initialized with this pair of assets.")) | |
183 | 179 | let asset = fromBase58String(value(assetA)) | |
184 | 180 | let checkHeight = if ((height >= heightInKey)) | |
185 | 181 | then true | |
186 | 182 | else throwErr((("Withdrawal is possible after " + toString(heightInKey)) + " height or you have already withdrawn.")) | |
187 | 183 | if ((checkHeight == checkHeight)) | |
188 | 184 | then [DeleteEntry(keyProcessInProgress(userAddress, assetA, assetB, heightInKey)), IntegerEntry(keyProcessDone(userAddress, assetA, assetB, heightInKey), toWithdraw), ScriptTransfer(i.caller, toWithdraw, asset)] | |
189 | 185 | else throw("Strict value is not equal to itself.") | |
190 | 186 | } | |
191 | 187 | ||
192 | 188 | ||
193 | 189 | ||
194 | 190 | @Callable(i) | |
195 | 191 | func withdrawFee (assetA,assetB) = { | |
196 | 192 | let checkCaller = mustManager(i) | |
197 | 193 | if ((checkCaller == checkCaller)) | |
198 | 194 | then { | |
199 | 195 | let toWithdrawA = valueOrElse(getInteger(keyTotalFeeCollectedDeposit(assetA, assetB)), 0) | |
200 | 196 | let withdrawAssetA = fromBase58String(value(assetA)) | |
201 | 197 | [IntegerEntry(keyTotalFeeCollectedDeposit(assetA, assetB), 0), ScriptTransfer(i.caller, toWithdrawA, withdrawAssetA)] | |
202 | 198 | } | |
203 | 199 | else throw("Strict value is not equal to itself.") | |
204 | 200 | } | |
205 | 201 | ||
206 | 202 | ||
207 | - | ||
208 | - | @Callable(i) | |
209 | - | func setManager (pendingManagerPublicKey) = { | |
210 | - | let checkCaller = mustManager(i) | |
211 | - | if ((checkCaller == checkCaller)) | |
212 | - | then { | |
213 | - | let checkManagerPublicKey = fromBase58String(pendingManagerPublicKey) | |
214 | - | if ((checkManagerPublicKey == checkManagerPublicKey)) | |
215 | - | then [StringEntry(keyPendingManagerPublicKey(), pendingManagerPublicKey)] | |
216 | - | else throw("Strict value is not equal to itself.") | |
217 | - | } | |
218 | - | else throw("Strict value is not equal to itself.") | |
219 | - | } | |
220 | - | ||
221 | - | ||
222 | - | ||
223 | - | @Callable(i) | |
224 | - | func confirmManager () = { | |
225 | - | let pm = pendingManagerPublicKeyOrUnit() | |
226 | - | let hasPM = if (isDefined(pm)) | |
227 | - | then true | |
228 | - | else throwErr("No pending manager.") | |
229 | - | if ((hasPM == hasPM)) | |
230 | - | then { | |
231 | - | let checkPM = if ((i.callerPublicKey == value(pm))) | |
232 | - | then true | |
233 | - | else throwErr("You are not pending manager.") | |
234 | - | if ((checkPM == checkPM)) | |
235 | - | then [StringEntry(keyManagerPublicKey(), toBase58String(value(pm))), DeleteEntry(keyPendingManagerPublicKey())] | |
236 | - | else throw("Strict value is not equal to itself.") | |
237 | - | } | |
238 | - | else throw("Strict value is not equal to itself.") | |
239 | - | } | |
240 | - | ||
241 | - | ||
242 | 203 | @Verifier(tx) | |
243 | 204 | func verify () = { | |
244 | 205 | let targetPublicKey = match managerPublicKeyOrUnit() { | |
245 | 206 | case pk: ByteVector => | |
246 | 207 | pk | |
247 | 208 | case _: Unit => | |
248 | 209 | tx.senderPublicKey | |
249 | 210 | case _ => | |
250 | 211 | throw("Match error") | |
251 | 212 | } | |
252 | 213 | sigVerify(tx.bodyBytes, tx.proofs[0], targetPublicKey) | |
253 | 214 | } | |
254 | 215 |
github/deemru/w8io/169f3d6 52.18 ms ◑