tx · 3cvkTrToEvZvGRr9PsjwQgCDiEBNddD8EnEoPgBoVg4d 3MzDGFvPnXVbA4wCacqcVnfVJPwG952KVAg: -0.01400000 Waves 2021.12.22 14:16 [1845397] smart account 3MzDGFvPnXVbA4wCacqcVnfVJPwG952KVAg > SELF 0.00000000 Waves
{ "type": 13, "id": "3cvkTrToEvZvGRr9PsjwQgCDiEBNddD8EnEoPgBoVg4d", "fee": 1400000, "feeAssetId": null, "timestamp": 1640171815461, "version": 2, "chainId": 84, "sender": "3MzDGFvPnXVbA4wCacqcVnfVJPwG952KVAg", "senderPublicKey": "CViCXhHrygDDAtVc1uFczxe6ANj9tBz5PGQDS1yWZqgy", "proofs": [ "3EVvGThSpLv424eaxDBZ3SLjk3wxPuyrDhhb7EiL9HmWGMmCt8FBefnkLs2P8UJcs46JKw3kaCwcAVEWbRyAFsMs", "54QFGRra5ZWpbyvD3sRYYcoNYDj1m2QaYbTQPggfso4DAyUChedu14n297SQ7233rRk3AAHtpYy6njWpR63CqPQD" ], "script": "base64:AAIFAAAAAAAAADcIAhIECgIBCBIDCgEIEhQKEggBAQgBAQEBAQEBARgEAQEEGBIDCgEBEgMKAQESBAoCAQESABIAAAAASgAAAAAJa2V5QWN0aXZlAgAAAAZhY3RpdmUAAAAAD2tleUFkbWluUHViS2V5MQIAAAALYWRtaW5fcHViXzEAAAAAD2tleUFkbWluUHViS2V5MgIAAAALYWRtaW5fcHViXzIAAAAAD2tleUFkbWluUHViS2V5MwIAAAALYWRtaW5fcHViXzMAAAAAIWtleUxhdW5jaHBhZERhdGFUcmFuc2FjdGlvblN0YXR1cwIAAAAhbGF1bmNocGFkX2RhdGFfdHJhbnNhY3Rpb25fc3RhdHVzAAAAABJrZXlMYXVuY2hwYWROZXh0SWQCAAAAEWxhdW5jaHBhZF9uZXh0X2lkAAAAABRrZXlMYXVuY2hwYWRBY3RpdmVJZAIAAAATbGF1bmNocGFkX2FjdGl2ZV9pZAAAAAANa2V5SW5pdENhbGxlcgIAAAALaW5pdF9jYWxsZXIAAAAACmtleUFkZHJlc3MCAAAACF9hZGRyZXNzAAAAAA1rZXlJbml0SGVpZ2h0AgAAAAxfaW5pdF9oZWlnaHQAAAAADmtleVN0YXJ0SGVpZ2h0AgAAAA1fc3RhcnRfaGVpZ2h0AAAAABFrZXlGaW5hbGlzZUhlaWdodAIAAAAQX2ZpbmFsaXNlX2hlaWdodAAAAAAMa2V5Q29taXNzaW9uAgAAAApfY29taXNzaW9uAAAAAAprZXlBc3NldElkAgAAAAlfYXNzZXRfaWQAAAAAE2tleVRvdGFsVG9rZW5BbW91bnQCAAAAE190b3RhbF90b2tlbl9hbW91bnQAAAAAGmtleVRvdGFsQW1vdW50QnV5QXNzZXRTb2xkAgAAABVfdG90YWxfYnV5X2Fzc2V0X3NvbGQAAAAAEmtleVRva2Vuc1BlclRpY2tldAIAAAASX3Rva2Vuc19wZXJfdGlja2V0AAAAABBrZXlQcmljZVBlclRva2VuAgAAABBfcHJpY2VfcGVyX3Rva2VuAAAAABBrZXlTd29wUGVyVGlja2V0AgAAABBfc3dvcF9wZXJfdGlja2V0AAAAABZrZXlTd29wZmlUaWNrZXRzQW1vdW50AgAAABdfdGlja2V0c19zd29wZmlfbWVtYmVycwAAAAAXa2V5Q2FtcGFpZ25Ub2tlbnNBbW91bnQCAAAAE190b2tlbnNfYWNjZXNzX2xpc3QAAAAAFmtleVRva2Vuc1BlckFsbG9jYXRpb24CAAAAFl90b2tlbnNfcGVyX2FsbG9jYXRpb24AAAAAE2tleVRpY2tldExhc3ROdW1iZXICAAAAE190aWNrZXRfbGFzdF9udW1iZXIAAAAAE2tleUdvdmVybmFuY2VTdGFrZWQCAAAADF9TV09QX2Ftb3VudAAAAAAMa2V5VXNlclJlZklkAgAAAAdfcmVmX2lkAAAAAA1rZXlWZXJpZnlIYXNoAgAAAAxfdmVyaWZ5X2hhc2gAAAAAC2tleUR1cmF0aW9uAgAAAAlfZHVyYXRpb24AAAAAHGtleUFtb3VudEJ1eUFzc2V0SW5NYXJrZXRpbmcCAAAAJ190b3RhbF9wdXJjaGFzZWRfYnV5X2Fzc2V0X2luX21hcmtldGluZwAAAAAla2V5QW1vdW50QnV5QXNzZXRJbk1hcmtldGluZ0ZpbmFsaXplZAIAAAAxX3RvdGFsX3B1cmNoYXNlZF9idXlfYXNzZXRfaW5fbWFya2V0aW5nX2ZpbmFsaXplZAAAAAATa2V5VHJhbnNmZXJCdXlBc3NldAIAAAATX3RyYW5zZmVyX2J1eV9hc3NldAAAAAAUa2V5RnJlZXpTd29wRHVyYXRpb24CAAAAFV9mcmVlemVfc3dvcF9kdXJhdGlvbgAAAAAYa2V5RnJlZXplQWRkaXRpb25hbFRva2VuAgAAABhfZnJlZXplX2FkZGl0aW9uYWxfdG9rZW4AAAAAEGtleUZyZWV6ZVRva2VuSWQCAAAAEF9mcmVlemVfdG9rZW5faWQAAAAAFmtleUZyZWV6ZVRva2VuRHVyYXRpb24CAAAAFl9mcmVlemVfdG9rZW5fZHVyYXRpb24AAAAAF2tleUZyZWV6ZVRva2VuUGVyVGlja2V0AgAAABhfZnJlZXplX3Rva2VuX3Blcl90aWNrZXQAAAAAC2tleUJ1eVRva2VuAgAAAApfYnV5X3Rva2VuAAAAAA5rZXlDbGFpbVBhcmFtcwIAAAANX2NsYWltX3BhcmFtcwAAAAAba2V5SGlnaENvbXBldGl0aW9uQ29taXNzaW9uAgAAABtfaGlnaF9jb21wZXRpdGlvbl9jb21pc3Npb24AAAAAFmtleUhpZ2hDb21wZXRpdGlvblNhbGUCAAAAFl9oaWdoX2NvbXBldGl0aW9uX3NhbGUAAAAAFWtleUhDVGlja2V0TGFzdE51bWJlcgIAAAAkX2hpZ2hfY29tcGV0aXRpb25fdGlja2V0X2xhc3RfbnVtYmVyAAAAABhrZXlIaWdoQ29tcGV0aXRpb25BbW91bnQCAAAAGV90aWNrZXRzX2hpZ2hfY29tcGV0aXRpb24AAAAAHGtleVRvdGFsQW1vdW50QnV5QXNzZXRTb2xkSEMCAAAAJl90b3RhbF9idXlfYXNzZXRfc29sZF9oaWdoX2NvbXBldGl0aW9uAAAAABRrZXlVc2VyQ2xhaW1lZFRva2VucwIAAAAPX2NsYWltZWRfdG9rZW5zAAAAABdrZXlVc2VyQ2xhaW1lZEJ1eVRva2VucwIAAAATX2NsYWltZWRfYnV5X3Rva2VucwAAAAAaa2V5VXNlckJvdWdodFRpY2tldHNBbW91bnQCAAAAD19ib3VnaHRfdGlja2V0cwAAAAAOa2V5VXNlclRpY2tldHMCAAAAD190aWNrZXRzX251bWJlcgAAAAAQa2V5VXNlckhDVGlja2V0cwIAAAAgX3RpY2tldHNfbnVtYmVyX2hpZ2hfY29tcGV0aXRpb24AAAAAHGtleVVzZXJIQ0JvdWdodFRpY2tldHNBbW91bnQCAAAAIF9ib3VnaHRfdGlja2V0c19oaWdoX2NvbXBldGl0aW9uAAAAABtrZXlVc2VyQXZhaWxhYmxlQWxsb2NhdGlvbnMCAAAAHV9hdmFpbGFibGVfcHVyY2hhc2VfbWFya2V0aW5nAAAAACBrZXlVc2VyQW1vdW50QnV5QXNzZXRJbk1hcmtldGluZwIAAAAhX3B1cmNoYXNlZF9idXlfYXNzZXRfaW5fbWFya2V0aW5nAAAAABhrZXlVc2VyQm91Z2h0QWxsb2NhdGlvbnMCAAAAE19ib3VnaHRfYWxsb2NhdGlvbnMAAAAAEmtleVVzZXJDbGFpbVN0YXR1cwIAAAANX2NsYWltX3N0YXR1cwAAAAAaa2V5VXNlcldpbm5lZFRpY2tldHNBbW91bnQCAAAAD190aWNrZXRzX3Jlc3VsdAAAAAApa2V5VXNlcldpbm5lZEhpZ2hDb21wZXRpdGlvblRpY2tl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AAAACGlzQWN0aXZlAAAAAAkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkBAAAAC2lzQWRtaW5DYWxsAAAAAQUAAAABaQkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACBQAAAA1rZXlJbml0Q2FsbGVyBQAAAAdhZGRyZXNzBQAAAANuaWwAAAABaQEAAAAEaW5pdAAAABIAAAAOcHJvamVjdEFkZHJlc3MAAAALc3RhcnRIZWlnaHQAAAAIZHVyYXRpb24AAAAIYnV5VG9rZW4AAAAKY29tbWlzc2lvbgAAAA90b2tlbnNQZXJUaWNrZXQAAAANcHJpY2VQZXJUb2tlbgAAABNzd29wZmlUaWNrZXRzQW1vdW50AAAAFGNhbXBhaWduVG9rZW5zQW1vdW50AAAAE3Rva2Vuc1BlckFsbG9jYXRpb24AAAANc3dvcFBlclRpY2tldAAAABJzd29wRnJlZXplRHVyYXRpb24AAAALY2xhaW1QYXJhbXMAAAATSGlnaENvbXBldGl0aW9uU2FsZQAAABhoaWdoQ29tcGV0aXRpb25Db21pc3Npb24AAAAcaGlnaENvbXBldGl0aW9uVGlja2V0c0Ftb3VudAAAABVmcmVlemVBZGRpdGlvbmFsVG9rZW4AAAAMZnJlZXplUGFyYW1zCQEAAAALdmFsdWVPckVsc2UAAAACCQEAAAAIaXNBY3RpdmUAAAAAAwkBAAAAAiE9AAAAAgkABCUAAAABCAUAAAABaQAAAAZjYWxsZXIJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQdAAAAAgUAAAAEdGhpcwUAAAANa2V5SW5pdENhbGxlcgIAAAAACQAAAgAAAAECAAAAKU9ubHkgcHJvamVjdCBhZG1pbiBjYW4gY2FsbCBpbml0IGZ1bmN0aW9uAwkBAAAAAiE9AAAAAgkAAZAAAAABCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAQkAAAIAAAABAgAAAB1PbmUgYXR0YWNoZWQgcGF5bWVudCBleHBlY3RlZAMJAABmAAAAAgUAAAAGaGVpZ2h0BQAAAAtzdGFydEhlaWdodAkAAAIAAAABAgAAADNTdGFydCBoZWlnaHQgbXVzdCBiZSBncmVhdGVyIHRoYW4gYmxvY2tjaGFpbiBoZWlnaHQDAwUAAAAVZnJlZXplQWRkaXRpb25hbFRva2VuCQEAAAACIT0AAAACCQABkAAAAAEFAAAADGZyZWV6ZVBhcmFtcwAAAAAAAAAAAwcJAAACAAAAAQIAAAAkRnJlZXplIHBhcmFtcyBsaXN0IG11c3QgaGF2ZSAzIGl0ZW1zBAAAAAtsYXVuY2hwYWRJZAkBAAAAEmdldExhdW5jaHBhZE5leHRJZAAAAAAEAAAADmxhdW5jaHBhZElkU3RyCQABpAAAAAEFAAAAC2xhdW5jaHBhZElkBAAAAAskdDA2Mzk3NjQ3MgkABRQAAAACCAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAAAAAZhbW91bnQICQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAAAAAAAB2Fzc2V0SWQEAAAACXBtdEFtb3VudAgFAAAACyR0MDYzOTc2NDcyAAAAAl8xBAAAAApwbXRBc3NldElkCAUAAAALJHQwNjM5NzY0NzIAAAACXzIEAAAACyR0MDY0ODE2NTU0CQEAAAAMZ2V0QXNzZXRJbmZvAAAAAQUAAAAKcG10QXNzZXRJZAQAAAANcG10U3RyQXNzZXRJZAgFAAAACyR0MDY0ODE2NTU0AAAAAl8xBAAAAAxwbXRBc3NldE5hbWUIBQAAAAskdDA2NDgxNjU1NAAAAAJfMgQAAAALcG10RGVjaW1hbHMIBQAAAAskdDA2NDgxNjU1NAAAAAJfMwQAAAAVZnJlZXplQWRkaXRpb25hbEVudHJ5AwUAAAAVZnJlZXplQWRkaXRpb25hbFRva2VuCQAETAAAAAIJAQAAAAxCb29sZWFuRW50cnkAAAACCQABLAAAAAIFAAAADmxhdW5jaHBhZElkU3RyBQAAABhrZXlGcmVlemVBZGRpdGlvbmFsVG9rZW4GCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAAEsAAAAAgUAAAAObGF1bmNocGFkSWRTdHIFAAAAEGtleUZyZWV6ZVRva2VuSWQJAAGRAAAAAgUAAAAMZnJlZXplUGFyYW1zAAAAAAAAAAAACQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQABLAAAAAIFAAAADmxhdW5jaHBhZElkU3RyBQAAABZrZXlGcmVlemVUb2tlbkR1cmF0aW9uCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAMZnJlZXplUGFyYW1zAAAAAAAAAAABCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQABLAAAAAIFAAAADmxhdW5jaHBhZElkU3RyBQAAABdrZXlGcmVlemVUb2tlblBlclRpY2tldAkBAAAADXBhcnNlSW50VmFsdWUAAAABCQABkQAAAAIFAAAADGZyZWV6ZVBhcmFtcwAAAAAAAAAAAgUAAAADbmlsBQAAAANuaWwJAAROAAAAAgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA1wbXRTdHJBc3NldElkAgAAAApfbGF1bmNocGFkBQAAAAtsYXVuY2hwYWRJZAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAASa2V5TGF1bmNocGFkTmV4dElkCQAAZAAAAAIFAAAAC2xhdW5jaHBhZElkAAAAAAAAAAABCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAAEsAAAAAgUAAAAObGF1bmNocGFkSWRTdHIFAAAACmtleUFkZHJlc3MFAAAADnByb2plY3RBZGRyZXNzCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQABLAAAAAIFAAAADmxhdW5jaHBhZElkU3RyBQAAAA1rZXlJbml0SGVpZ2h0BQAAAAZoZWlnaHQJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIJAAEsAAAAAgUAAAAObGF1bmNocGFkSWRTdHIFAAAADmtleVN0YXJ0SGVpZ2h0BQAAAAtzdGFydEhlaWdodAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA5sYXVuY2hwYWRJZFN0cgUAAAALa2V5RHVyYXRpb24FAAAACGR1cmF0aW9uCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAAEsAAAAAgUAAAAObGF1bmNocGFkSWRTdHIFAAAAC2tleUJ1eVRva2VuBQAAAAhidXlUb2tlbgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA5sYXVuY2hwYWRJZFN0cgUAAAAUa2V5RnJlZXpTd29wRHVyYXRpb24FAAAAEnN3b3BGcmVlemVEdXJhdGlvbgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA5sYXVuY2hwYWRJZFN0cgUAAAATa2V5VG90YWxUb2tlbkFtb3VudAUAAAAJcG10QW1vdW50CQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAAEsAAAAAgUAAAAObGF1bmNocGFkSWRTdHIFAAAACmtleUFzc2V0SWQFAAAADXBtdFN0ckFzc2V0SWQJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIJAAEsAAAAAgUAAAAObGF1bmNocGFkSWRTdHIFAAAADGtleUNvbWlzc2lvbgUAAAAKY29tbWlzc2lvbgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA5sYXVuY2hwYWRJZFN0cgUAAAATa2V5VGlja2V0TGFzdE51bWJlcgAAAAAAAAAAAAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA5sYXVuY2hwYWRJZFN0cgUAAAAVa2V5SENUaWNrZXRMYXN0TnVtYmVyAAAAAAAAAAAACQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQABLAAAAAIFAAAADmxhdW5jaHBhZElkU3RyBQAAABJrZXlUb2tlbnNQZXJUaWNrZXQFAAAAD3Rva2Vuc1BlclRpY2tldAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA5sYXVuY2hwYWRJZFN0cgUAAAAQa2V5UHJpY2VQZXJUb2tlbgUAAAANcHJpY2VQZXJUb2tlbgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA5sYXVuY2hwYWRJZFN0cgUAAAAQa2V5U3dvcFBlclRpY2tldAUAAAANc3dvcFBlclRpY2tldAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA5sYXVuY2hwYWRJZFN0cgUAAAAWa2V5U3dvcGZpVGlja2V0c0Ftb3VudAUAAAATc3dvcGZpVGlja2V0c0Ftb3VudAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA5sYXVuY2hwYWRJZFN0cgUAAAAXa2V5Q2FtcGFpZ25Ub2tlbnNBbW91bnQFAAAAFGNhbXBhaWduVG9rZW5zQW1vdW50CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQABLAAAAAIFAAAADmxhdW5jaHBhZElkU3RyBQAAABZrZXlUb2tlbnNQZXJBbGxvY2F0aW9uBQAAABN0b2tlbnNQZXJBbGxvY2F0aW9uCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAAEsAAAAAgUAAAAObGF1bmNocGFkSWRTdHIFAAAADmtleUNsYWltUGFyYW1zCQAEuQAAAAIFAAAAC2NsYWltUGFyYW1zAgAAAAEsCQAETAAAAAIJAQAAAAxCb29sZWFuRW50cnkAAAACCQABLAAAAAIFAAAADmxhdW5jaHBhZElkU3RyBQAAABZrZXlIaWdoQ29tcGV0aXRpb25TYWxlBQAAABNIaWdoQ29tcGV0aXRpb25TYWxlCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQABLAAAAAIFAAAADmxhdW5jaHBhZElkU3RyBQAAABtrZXlIaWdoQ29tcGV0aXRpb25Db21pc3Npb24FAAAAGGhpZ2hDb21wZXRpdGlvbkNvbWlzc2lvbgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACBQAAAA5sYXVuY2hwYWRJZFN0cgUAAAAYa2V5SGlnaENvbXBldGl0aW9uQW1vdW50BQAAABxoaWdoQ29tcGV0aXRpb25UaWNrZXRzQW1vdW50CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAABRrZXlMYXVuY2hwYWRBY3RpdmVJZAUAAAALbGF1bmNocGFkSWQJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAANa2V5SW5pdENhbGxlcgIAAAAABQAAAANuaWwFAAAAFWZyZWV6ZUFkZGl0aW9uYWxFbnRyeQAAAAFpAQAAAAVjbGFpbQAAAAEAAAALbGF1bmNocGFkSWQJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAQAAAAhpc0FjdGl2ZQAAAAAEAAAADWJv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blN0ckFzc2V0SWQIBQAAAA0kdDAxMTE3NjExMzIwAAAAAl8yBAAAAA50b2tlbkFzc2V0TmFtZQgFAAAADSR0MDExMTc2MTEzMjAAAAACXzMEAAAADXRva2VuRGVjaW1hbHMIBQAAAA0kdDAxMTE3NjExMzIwAAAAAl80BAAAAAt0aWNrZXRQcmljZQkAAGsAAAADBQAAAA90b2tlbnNQZXJUaWNrZXQFAAAACnRva2VuUHJpY2UJAABsAAAABgAAAAAAAAAACgAAAAAAAAAAAAUAAAANdG9rZW5EZWNpbWFscwAAAAAAAAAAAAAAAAAAAAAAAAUAAAAERE9XTgQAAAATdG9rZW5zUGVyQWxsb2NhdGlvbgkBAAAAEUBleHRyTmF0aXZlKDEwNTApAAAAAgUAAAAEdGhpcwkAASwAAAACCQABpAAAAAEFAAAAC2xhdW5jaHBhZElkBQAAABZrZXlUb2tlbnNQZXJBbGxvY2F0aW9uBAAAABR0cmFuc2ZlclRva2Vuc0Ftb3VudAkAAGQAAAACCQAAZAAAAAIJAABoAAAAAgUAAAANd29ubmVkVGlja2V0cwUAAAAPdG9rZW5zUGVyVGlja2V0CQAAaAAAAAIFAAAAEmFsbG93ZWRBbGxvY2F0aW9ucwUAAAATdG9rZW5zUGVyQWxsb2NhdGlvbgkAAGgAAAACBQAAAA93b25uZWRIQ1RpY2tldHMFAAAAD3Rva2Vuc1BlclRpY2tldAQAAAAbbm90QWxsb3dlZEFsbG9jYXRpb25zVG9rZW5zCQAAaAAAAAIJAABlAAAAAgUAAAARYm91Z2h0QWxsb2NhdGlvbnMFAAAAEmFsbG93ZWRBbGxvY2F0aW9ucwUAAAATdG9rZW5zUGVyQWxsb2NhdGlvbgQAAAAWYnV5QXNzZXRGb3JBbGxvY2F0aW9ucwkAAGsAAAADCQAAawAAAAMFAAAAG25vdEFsbG93ZWRBbGxvY2F0aW9uc1Rva2VucwUAAAAKdG9rZW5QcmljZQkAAGwAAAAGAAAAAAAAAAAKAAAAAAAAAAAABQAAAA10b2tlbkRlY2ltYWxzAAAAAAAAAAAAAAAAAAAAAAAABQAAAARET1dOCQAAZAAAAAIAAAAAAAAAAGQFAAAACmNvbW1pc3Npb24AAAAAAAAAAGQEAAAAGmJ1eUFzc2V0Rm9yVW53b25uZWRUaWNrZXRzCQAAawAAAAMJAABoAAAAAgkAAGUAAAACBQAAAA1ib3VnaHRUaWNrZXRzBQAAAA13b25uZWRUaWNrZXRzBQAAAAt0aWNrZXRQcmljZQkAAGQAAAACAAAAAAAAAABkBQAAAApjb21taXNzaW9uAAAAAAAAAABkBAAAABxidXlBc3NldEZvckhDVW53b25uZWRUaWNrZXRzCQAAawAAAAMJAABoAAAAAgkAAGUAAAACBQAAAA9ib3VnaHRIQ1RpY2tldHMFAAAAD3dvbm5lZEhDVGlja2V0cwUAAAALdGlja2V0UHJpY2UJAABkAAAAAgAAAAAAAAAAZAUAAAAMY29tbWlzc2lvbkhDAAAAAAAAAABkBAAAAA9jbGFpbWVkQnV5VG9rZW4JAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQaAAAAAgUAAAAEdGhpcwkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkABCUAAAABCAUAAAABaQAAAAZjYWxsZXICAAAAAV8JAAGkAAAAAQUAAAALbGF1bmNocGFkSWQFAAAAF2tleVVzZXJDbGFpbWVkQnV5VG9rZW5zAAAAAAAAAAAAAwMDCQAAZgAAAAIAAAAAAAAAAAAFAAAAFmJ1eUFzc2V0Rm9yQWxsb2NhdGlvbnMGCQAAZgAAAAIAAAAAAAAAAAAFAAAAGmJ1eUFzc2V0Rm9yVW53b25uZWRUaWNrZXRzBgkAAGYAAAACAAAAAAAAAAAABQAAABxidXlBc3NldEZvckhDVW53b25uZWRUaWNrZXRzCQAAAgAAAAECAAAAUkVycm9yIHdpdGggYWxsb3dlZCBhbGxvY2F0aW9uIHBhcmFtIG9yIHdvbm5lZCB0aWNrZXRzIHBhcmFtIHBsZWFzZSBjb250YWN0IHN1cHBvcnQEAAAAFnJldHVybmVkQnV5QXNzZXRBbW91bnQJAABlAAAAAgkAAGQAAAACCQAAZAAAAAIFAAAAGmJ1eUFzc2V0Rm9yVW53b25uZWRUaWNrZXRzBQAAABxidXlBc3NldEZvckhDVW53b25uZWRUaWNrZXRzBQAAABZidXlBc3NldEZvckFsbG9jYXRpb25zBQAAAA9jbGFpbWVkQnV5VG9rZW4EAAAAEHRyYW5zZmVyQnV5QXNzZXQDCQAAZgAAAAIFAAAAFnJl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", "height": 1845397, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 2Y8UoeoSppMRwT3DVT2aFy2n8MS9Px27UVXifkR4gsYG Next: FfjeQD3dZ1HpruzRGKf1AQTQrF6nD7dRpYqpAG8TmVwK Diff:
Old | New | Differences | |
---|---|---|---|
156 | 156 | func getLaunchpadNextId () = valueOrElse(getInteger(this, keyLaunchpadNextId), 1) | |
157 | 157 | ||
158 | 158 | ||
159 | + | func stakedUsdnAmount () = valueOrElse(getInteger(stakingUSDNAddress, ((("rpd_balance_" + toBase58String(USDN)) + "_") + toString(this))), 0) | |
160 | + | ||
161 | + | ||
159 | 162 | let active = valueOrElse(getBoolean(this, keyActive), true) | |
160 | 163 | ||
161 | 164 | func isActive () = if (active) | |
218 | 221 | ||
219 | 222 | ||
220 | 223 | @Callable(i) | |
221 | - | func commitSwopfiSale (launchpadId,refId) = valueOrElse(isActive(), { | |
222 | - | let $t091609235 = $Tuple2(i.payments[0].amount, i.payments[0].assetId) | |
223 | - | let pmtAmount = $t091609235._1 | |
224 | - | let pmtAssetId = $t091609235._2 | |
225 | - | let $t092409390 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
226 | - | let buyAssetId = $t092409390._1 | |
227 | - | let buyAssetStrId = $t092409390._2 | |
228 | - | let buyAssetName = $t092409390._3 | |
229 | - | let buyAssetDecimals = $t092409390._4 | |
230 | - | let stakedSwops = valueOrElse(getInteger(governance, (toString(i.caller) + keyGovernanceStaked)), 0) | |
231 | - | let allowedTicketsAmountAll = (stakedSwops / getIntegerValue(this, (toString(launchpadId) + keySwopPerTicket))) | |
232 | - | let boughtTicketsAmount = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtTicketsAmount)), 0) | |
233 | - | let allowedTicketsAmount = (allowedTicketsAmountAll - boughtTicketsAmount) | |
234 | - | let startHeight = value(getInteger(this, (toString(launchpadId) + keyStartHeight))) | |
235 | - | let duration = value(getInteger(this, (toString(launchpadId) + keyDuration))) | |
236 | - | let freezeAdditionalToken = valueOrElse(getBoolean(this, (toString(launchpadId) + keyFreezeAdditionalToken)), false) | |
237 | - | if ((activeLaunchpadId != launchpadId)) | |
238 | - | then throw("There is no active launchpad or this launchpad is ended") | |
239 | - | else if ((startHeight > height)) | |
240 | - | then throw("Launchpad sale not started yet") | |
241 | - | else if ((height > (startHeight + duration))) | |
242 | - | then throw("Launchpad sale ended") | |
243 | - | else if (if (!(freezeAdditionalToken)) | |
244 | - | then if ((size(i.payments) != 1)) | |
245 | - | then true | |
246 | - | else (pmtAssetId != buyAssetId) | |
247 | - | else false) | |
248 | - | then throw((("One attached payment in " + buyAssetName) + " expected")) | |
249 | - | else if (if (freezeAdditionalToken) | |
250 | - | then if ((size(i.payments) != 2)) | |
251 | - | then true | |
252 | - | else (pmtAssetId != buyAssetId) | |
253 | - | else false) | |
254 | - | then { | |
255 | - | let freezeTokenId = getStringValue(this, (toString(launchpadId) + keyFreezeTokenId)) | |
256 | - | throw(((("Two attached payments expected: First in " + buyAssetName) + ", second in ") + freezeTokenId)) | |
257 | - | } | |
258 | - | else if (if ((0 >= allowedTicketsAmountAll)) | |
259 | - | then true | |
260 | - | else (0 >= allowedTicketsAmount)) | |
261 | - | then throw("Not enought SWOP in staking to buy tickets") | |
262 | - | else { | |
263 | - | let tokenPrice = getIntegerValue(this, (toString(launchpadId) + keyPricePerToken)) | |
264 | - | let tokensPerTicket = getIntegerValue(this, (toString(launchpadId) + keyTokensPerTicket)) | |
265 | - | let $t01120211346 = getAssetInfoFromString(getStringValue(this, (toString(launchpadId) + keyAssetId))) | |
266 | - | let tokenId = $t01120211346._1 | |
267 | - | let tokenStrAssetId = $t01120211346._2 | |
268 | - | let tokenAssetName = $t01120211346._3 | |
269 | - | let tokenDecimals = $t01120211346._4 | |
270 | - | let ticketPrice = fraction(tokensPerTicket, tokenPrice, pow(10, 0, tokenDecimals, 0, 0, DOWN)) | |
271 | - | let commissionPerTicket = fraction(ticketPrice, getIntegerValue(this, (toString(launchpadId) + keyComission)), 100) | |
272 | - | let buyTicketAmount = (pmtAmount / (ticketPrice + commissionPerTicket)) | |
273 | - | let allowedBuyTicketAmount = min([buyTicketAmount, allowedTicketsAmount]) | |
274 | - | let allowedBuyPriceWithComission = (allowedBuyTicketAmount * (ticketPrice + commissionPerTicket)) | |
275 | - | let change = (pmtAmount - allowedBuyPriceWithComission) | |
276 | - | if ((buyTicketAmount == 0)) | |
277 | - | then throw((("Not enought " + buyAssetName) + " to buy tickets")) | |
278 | - | else if ((change != 0)) | |
279 | - | then throw((((("Wrong payment. To buy " + toString(allowedBuyTicketAmount)) + "tickets you need to pay ") + toString(allowedBuyPriceWithComission)) + buyAssetName)) | |
280 | - | else { | |
281 | - | let freezeAdditionalEntry = if (freezeAdditionalToken) | |
282 | - | then { | |
283 | - | let freezeTokenId = getStringValue(this, (toString(launchpadId) + keyFreezeTokenId)) | |
284 | - | let freezeTokenPerTicket = getIntegerValue(this, (toString(launchpadId) + keyFreezeTokenPerTicket)) | |
285 | - | let $t01251012597 = $Tuple2(i.payments[1].amount, i.payments[1].assetId) | |
286 | - | let pmtAmountFreeze = $t01251012597._1 | |
287 | - | let pmtAssetIdFreeze = $t01251012597._2 | |
288 | - | let $t01261412705 = getAssetInfo(pmtAssetId) | |
289 | - | let pmtFreezeStrAssetId = $t01261412705._1 | |
290 | - | let pmtFreezeAssetName = $t01261412705._2 | |
291 | - | let pmtFreezeDecimals = $t01261412705._3 | |
292 | - | let needFreezeTokens = (freezeTokenPerTicket * buyTicketAmount) | |
293 | - | if (if ((pmtFreezeStrAssetId != freezeTokenId)) | |
294 | - | then true | |
295 | - | else (needFreezeTokens != pmtAmountFreeze)) | |
296 | - | then throw((((("You need to add " + toString(needFreezeTokens)) + " ") + pmtFreezeAssetName) + " as a second payment")) | |
297 | - | else [IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserFreezeToken), pmtAmountFreeze)] | |
298 | - | } | |
299 | - | else nil | |
300 | - | let userBoughtTicketsNumbers = valueOrElse(getString(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserTickets)), "") | |
301 | - | let lastBoughtTicketNumber = getIntegerValue(this, (toString(launchpadId) + keyTicketLastNumber)) | |
302 | - | let boughtRange = ((toString((lastBoughtTicketNumber + 1)) + "-") + toString((lastBoughtTicketNumber + allowedBuyTicketAmount))) | |
303 | - | let newUserBoughtTicketsNumbers = (userBoughtTicketsNumbers + (if ((userBoughtTicketsNumbers != "")) | |
304 | - | then ("," + boughtRange) | |
305 | - | else ("" + boughtRange))) | |
306 | - | let inv = if ((buyAssetId == USDN)) | |
307 | - | then invoke(stakingUSDNAddress, "lockNeutrino", nil, [AttachedPayment(pmtAssetId, pmtAmount)]) | |
308 | - | else 0 | |
309 | - | if ((inv == inv)) | |
310 | - | then { | |
311 | - | let baseEntry = ([IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtTicketsAmount), (boughtTicketsAmount + allowedBuyTicketAmount)), IntegerEntry((toString(launchpadId) + keyTicketLastNumber), (lastBoughtTicketNumber + allowedBuyTicketAmount)), StringEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserTickets), newUserBoughtTicketsNumbers)] ++ freezeAdditionalEntry) | |
312 | - | if (!(isDefined(getString(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId))))) | |
313 | - | then (baseEntry ++ [StringEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId), refId)]) | |
314 | - | else baseEntry | |
315 | - | } | |
316 | - | else throw("Strict value is not equal to itself.") | |
317 | - | } | |
318 | - | } | |
319 | - | }) | |
320 | - | ||
321 | - | ||
322 | - | ||
323 | - | @Callable(i) | |
324 | - | func commitAccessListSale (launchpadId,refId) = valueOrElse(isActive(), { | |
325 | - | let $t01477614851 = $Tuple2(i.payments[0].amount, i.payments[0].assetId) | |
326 | - | let pmtAmount = $t01477614851._1 | |
327 | - | let pmtAssetId = $t01477614851._2 | |
328 | - | let $t01485615006 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
329 | - | let buyAssetId = $t01485615006._1 | |
330 | - | let buyAssetStrId = $t01485615006._2 | |
331 | - | let buyAssetName = $t01485615006._3 | |
332 | - | let buyAssetDecimals = $t01485615006._4 | |
333 | - | let userUsdnInMarketing = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserAmountBuyAssetInMarketing)), 0) | |
334 | - | let usdnInMarketing = valueOrElse(getInteger(this, (toString(launchpadId) + keyAmountBuyAssetInMarketing)), 0) | |
335 | - | let boughtAllocations = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtAllocations)), 0) | |
336 | - | let commission = getIntegerValue(this, (toString(launchpadId) + keyComission)) | |
337 | - | let tokenPrice = getIntegerValue(this, (toString(launchpadId) + keyPricePerToken)) | |
338 | - | let tokensPerAllocation = getIntegerValue(this, (toString(launchpadId) + keyTokensPerAllocation)) | |
339 | - | let $t01567515819 = getAssetInfoFromString(getStringValue(this, (toString(launchpadId) + keyAssetId))) | |
340 | - | let tokenId = $t01567515819._1 | |
341 | - | let tokenStrAssetId = $t01567515819._2 | |
342 | - | let tokenAssetName = $t01567515819._3 | |
343 | - | let tokenDecimals = $t01567515819._4 | |
344 | - | let allocationPriceWithComission = fraction(fraction(tokenPrice, tokensPerAllocation, pow(10, 0, tokenDecimals, 0, 0, DOWN)), (100 + commission), 100) | |
345 | - | let startHeight = value(getInteger(this, (toString(launchpadId) + keyStartHeight))) | |
346 | - | let duration = value(getInteger(this, (toString(launchpadId) + keyDuration))) | |
347 | - | if ((activeLaunchpadId != launchpadId)) | |
348 | - | then throw("There is no active launchpad or this launchpad is ended") | |
349 | - | else if ((startHeight > height)) | |
350 | - | then throw("Launchpad sale not started yet") | |
351 | - | else if ((height > (startHeight + duration))) | |
352 | - | then throw("Launchpad sale ended") | |
353 | - | else if (if ((size(i.payments) != 1)) | |
354 | - | then true | |
355 | - | else (pmtAssetId != buyAssetId)) | |
356 | - | then throw((("One attached payment in " + buyAssetName) + " expected")) | |
357 | - | else if ((boughtAllocations >= maxAllocationsAmount)) | |
358 | - | then throw((("You can buy only " + toString(maxAllocationsAmount)) + "allocations")) | |
359 | - | else if (if ((allocationPriceWithComission != pmtAmount)) | |
360 | - | then ((allocationPriceWithComission * 2) != pmtAmount) | |
361 | - | else false) | |
362 | - | then throw(((("Wrong payment. You can buy 1 or 2 allocations for " + toString(allocationPriceWithComission)) + buyAssetName) + " per allocation")) | |
363 | - | else { | |
364 | - | let buyAllocationsAmount = (pmtAmount / allocationPriceWithComission) | |
365 | - | let inv = if ((buyAssetId == USDN)) | |
366 | - | then invoke(stakingUSDNAddress, "lockNeutrino", nil, [AttachedPayment(pmtAssetId, pmtAmount)]) | |
367 | - | else 0 | |
368 | - | if ((inv == inv)) | |
369 | - | then { | |
370 | - | let baseEntry = [IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtAllocations), (boughtAllocations + buyAllocationsAmount)), IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserAmountBuyAssetInMarketing), (userUsdnInMarketing + pmtAmount)), IntegerEntry((toString(launchpadId) + keyAmountBuyAssetInMarketing), (usdnInMarketing + pmtAmount))] | |
371 | - | if (!(isDefined(getString(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId))))) | |
372 | - | then (baseEntry ++ [StringEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId), refId)]) | |
373 | - | else baseEntry | |
374 | - | } | |
375 | - | else throw("Strict value is not equal to itself.") | |
376 | - | } | |
377 | - | }) | |
378 | - | ||
379 | - | ||
380 | - | ||
381 | - | @Callable(i) | |
382 | - | func commitHighCompetitionSale (launchpadId,refId) = valueOrElse(isActive(), { | |
383 | - | let $t01807718152 = $Tuple2(i.payments[0].amount, i.payments[0].assetId) | |
384 | - | let pmtAmount = $t01807718152._1 | |
385 | - | let pmtAssetId = $t01807718152._2 | |
386 | - | let $t01815718307 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
387 | - | let buyAssetId = $t01815718307._1 | |
388 | - | let buyAssetStrId = $t01815718307._2 | |
389 | - | let buyAssetName = $t01815718307._3 | |
390 | - | let buyAssetDecimals = $t01815718307._4 | |
391 | - | let startHeight = value(getInteger(this, (toString(launchpadId) + keyStartHeight))) | |
392 | - | let duration = value(getInteger(this, (toString(launchpadId) + keyDuration))) | |
393 | - | let boughtHCTicketsAmount = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserHCBoughtTicketsAmount)), 0) | |
394 | - | let highCompetitionSale = valueOrElse(getBoolean(this, (toString(launchpadId) + keyHighCompetitionSale)), false) | |
395 | - | if ((activeLaunchpadId != launchpadId)) | |
396 | - | then throw("There is no active launchpad or this launchpad is ended") | |
397 | - | else if (!(highCompetitionSale)) | |
398 | - | then throw("There is no high competition sale for this launchpad") | |
399 | - | else if ((startHeight > height)) | |
400 | - | then throw("Launchpad sale not started yet") | |
401 | - | else if ((height > (startHeight + duration))) | |
402 | - | then throw("Launchpad sale ended") | |
403 | - | else if (if ((size(i.payments) != 1)) | |
404 | - | then true | |
405 | - | else (pmtAssetId != buyAssetId)) | |
406 | - | then throw((("One attached payment in " + buyAssetName) + " expected")) | |
407 | - | else { | |
408 | - | let tokenPrice = getIntegerValue(this, (toString(launchpadId) + keyPricePerToken)) | |
409 | - | let tokensPerTicket = getIntegerValue(this, (toString(launchpadId) + keyTokensPerTicket)) | |
410 | - | let $t01948019624 = getAssetInfoFromString(getStringValue(this, (toString(launchpadId) + keyAssetId))) | |
411 | - | let tokenId = $t01948019624._1 | |
412 | - | let tokenStrAssetId = $t01948019624._2 | |
413 | - | let tokenAssetName = $t01948019624._3 | |
414 | - | let tokenDecimals = $t01948019624._4 | |
415 | - | let ticketPrice = fraction(tokensPerTicket, tokenPrice, pow(10, 0, tokenDecimals, 0, 0, DOWN)) | |
416 | - | let commissionPerTicket = fraction(ticketPrice, getIntegerValue(this, (toString(launchpadId) + keyHighCompetitionComission)), 100) | |
417 | - | let buyTicketAmount = (pmtAmount / (ticketPrice + commissionPerTicket)) | |
418 | - | let buyPriceWithComission = (buyTicketAmount * (ticketPrice + commissionPerTicket)) | |
419 | - | let change = (pmtAmount - buyPriceWithComission) | |
420 | - | if ((buyTicketAmount == 0)) | |
421 | - | then throw((("Not enought " + buyAssetName) + " to buy tickets")) | |
422 | - | else if ((change != 0)) | |
423 | - | then throw((((("Wrong payment. To buy " + toString(buyTicketAmount)) + "tickets you need to pay ") + toString(buyPriceWithComission)) + buyAssetName)) | |
424 | - | else { | |
425 | - | let userBoughtTicketsNumbers = valueOrElse(getString(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserHCTickets)), "") | |
426 | - | let lastBoughtTicketNumber = getIntegerValue(this, (toString(launchpadId) + keyHCTicketLastNumber)) | |
427 | - | let boughtRange = ((toString((lastBoughtTicketNumber + 1)) + "-") + toString((lastBoughtTicketNumber + buyTicketAmount))) | |
428 | - | let newUserBoughtTicketsNumbers = (userBoughtTicketsNumbers + (if ((userBoughtTicketsNumbers != "")) | |
429 | - | then ("," + boughtRange) | |
430 | - | else ("" + boughtRange))) | |
431 | - | let inv = if ((buyAssetId == USDN)) | |
432 | - | then invoke(stakingUSDNAddress, "lockNeutrino", nil, [AttachedPayment(pmtAssetId, pmtAmount)]) | |
433 | - | else 0 | |
434 | - | if ((inv == inv)) | |
435 | - | then { | |
436 | - | let baseEntry = [IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserHCBoughtTicketsAmount), (boughtHCTicketsAmount + buyTicketAmount)), IntegerEntry((toString(launchpadId) + keyHCTicketLastNumber), (lastBoughtTicketNumber + buyTicketAmount)), StringEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserHCTickets), newUserBoughtTicketsNumbers)] | |
437 | - | if (!(isDefined(getString(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId))))) | |
438 | - | then (baseEntry ++ [StringEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId), refId)]) | |
439 | - | else baseEntry | |
440 | - | } | |
441 | - | else throw("Strict value is not equal to itself.") | |
442 | - | } | |
443 | - | } | |
444 | - | }) | |
445 | - | ||
446 | - | ||
447 | - | ||
448 | - | @Callable(i) | |
449 | - | func finalise (launchpadId,vrfHeight,secretWord) = valueOrElse(isActive(), valueOrElse(isAdminCall(i), { | |
450 | - | let savedHash = getStringValue(this, (toString(launchpadId) + keyVerifyHash)) | |
451 | - | let calcHash = toBase58String(sha256((toBytes(vrfHeight) + toBytes(secretWord)))) | |
452 | - | if ((calcHash != savedHash)) | |
453 | - | then throw("vrf Height hash not matching") | |
454 | - | else { | |
455 | - | let $t02223722387 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
456 | - | let buyAssetId = $t02223722387._1 | |
457 | - | let buyAssetStrId = $t02223722387._2 | |
458 | - | let buyAssetName = $t02223722387._3 | |
459 | - | let buyAssetDecimals = $t02223722387._4 | |
460 | - | let projectAddress = addressFromStringValue(getStringValue(this, (toString(launchpadId) + keyAddress))) | |
461 | - | let commission = getIntegerValue(this, (toString(launchpadId) + keyComission)) | |
462 | - | let commissionHC = getIntegerValue(this, (toString(launchpadId) + keyHighCompetitionComission)) | |
463 | - | let transferedBuyAsset = valueOrElse(getInteger(this, (toString(launchpadId) + keyTransferBuyAsset)), 0) | |
464 | - | let swopfiMembersBuyAssetAmount = getIntegerValue(this, (toString(launchpadId) + keyTotalAmountBuyAssetSold)) | |
465 | - | let swopfiHCBuyAssetAmount = valueOrElse(getIntegerValue(this, (toString(launchpadId) + keyTotalAmountBuyAssetSoldHC)), 0) | |
466 | - | let swopfiMembersComission = fraction(swopfiMembersBuyAssetAmount, commission, 100) | |
467 | - | let swopfiHCComission = fraction(swopfiHCBuyAssetAmount, commissionHC, 100) | |
468 | - | let totalAmountBuyAssetInMarketing = getIntegerValue(this, (toString(launchpadId) + keyAmountBuyAssetInMarketingFinalized)) | |
469 | - | let marketingComission = fraction(totalAmountBuyAssetInMarketing, commission, 100) | |
470 | - | let unstakeAmount = (((((swopfiMembersBuyAssetAmount + totalAmountBuyAssetInMarketing) + swopfiMembersComission) + marketingComission) + swopfiHCBuyAssetAmount) + swopfiHCComission) | |
471 | - | if ((transferedBuyAsset > (swopfiMembersBuyAssetAmount + totalAmountBuyAssetInMarketing))) | |
472 | - | then throw("Can't transfer negative value to project") | |
473 | - | else { | |
474 | - | let inv = if ((buyAssetId == USDN)) | |
475 | - | then invoke(stakingUSDNAddress, "unlockNeutrino", [unstakeAmount, toBase58String(USDN)], nil) | |
476 | - | else 0 | |
477 | - | if ((inv == inv)) | |
478 | - | then [IntegerEntry((toString(launchpadId) + keyFinaliseHeight), height), ScriptTransfer(projectAddress, ((swopfiMembersBuyAssetAmount + totalAmountBuyAssetInMarketing) - transferedBuyAsset), buyAssetId), ScriptTransfer(commissionWallet, ((swopfiMembersComission + marketingComission) + swopfiHCComission), buyAssetId)] | |
479 | - | else throw("Strict value is not equal to itself.") | |
480 | - | } | |
481 | - | } | |
482 | - | })) | |
483 | - | ||
484 | - | ||
485 | - | ||
486 | - | @Callable(i) | |
487 | 224 | func claim (launchpadId) = valueOrElse(isActive(), { | |
488 | 225 | let boughtTickets = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtTicketsAmount)), 0) | |
489 | 226 | let boughtHCTickets = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserHCBoughtTicketsAmount)), 0) | |
501 | 238 | else false) | |
502 | 239 | then throw("You can't claim because you don't buy anything") | |
503 | 240 | else { | |
504 | - | let $ | |
505 | - | let buyAssetId = $ | |
506 | - | let buyAssetStrId = $ | |
507 | - | let buyAssetName = $ | |
508 | - | let buyAssetDecimals = $ | |
241 | + | let $t01012010270 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
242 | + | let buyAssetId = $t01012010270._1 | |
243 | + | let buyAssetStrId = $t01012010270._2 | |
244 | + | let buyAssetName = $t01012010270._3 | |
245 | + | let buyAssetDecimals = $t01012010270._4 | |
509 | 246 | let wonnedTickets = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserWinnedTicketsAmount)), 0) | |
510 | 247 | let wonnedHCTickets = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserWinnedHighCompetitionTicketsAmount)), 0) | |
511 | 248 | let awailableAllocations = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserAvailableAllocations)), 0) | |
514 | 251 | let allowedAllocations = min([boughtAllocations, awailableAllocations]) | |
515 | 252 | let tokenPrice = getIntegerValue(this, (toString(launchpadId) + keyPricePerToken)) | |
516 | 253 | let tokensPerTicket = getIntegerValue(this, (toString(launchpadId) + keyTokensPerTicket)) | |
517 | - | let $ | |
518 | - | let tokenId = $ | |
519 | - | let tokenStrAssetId = $ | |
520 | - | let tokenAssetName = $ | |
521 | - | let tokenDecimals = $ | |
254 | + | let $t01117611320 = getAssetInfoFromString(getStringValue(this, (toString(launchpadId) + keyAssetId))) | |
255 | + | let tokenId = $t01117611320._1 | |
256 | + | let tokenStrAssetId = $t01117611320._2 | |
257 | + | let tokenAssetName = $t01117611320._3 | |
258 | + | let tokenDecimals = $t01117611320._4 | |
522 | 259 | let ticketPrice = fraction(tokensPerTicket, tokenPrice, pow(10, 0, tokenDecimals, 0, 0, DOWN)) | |
523 | 260 | let tokensPerAllocation = getIntegerValue(this, (toString(launchpadId) + keyTokensPerAllocation)) | |
524 | 261 | let transferTokensAmount = (((wonnedTickets * tokensPerTicket) + (allowedAllocations * tokensPerAllocation)) + (wonnedHCTickets * tokensPerTicket)) | |
614 | 351 | @Callable(i) | |
615 | 352 | func transferBuyAsset (launchpadId,amount) = valueOrElse(isActive(), valueOrElse(isAdminCall(i), { | |
616 | 353 | let projectAddress = addressFromStringValue(getStringValue(this, (toString(launchpadId) + keyAddress))) | |
617 | - | let $ | |
618 | - | let buyAssetId = $ | |
619 | - | let buyAssetStrId = $ | |
620 | - | let buyAssetName = $ | |
621 | - | let buyAssetDecimals = $ | |
354 | + | let $t01744717597 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
355 | + | let buyAssetId = $t01744717597._1 | |
356 | + | let buyAssetStrId = $t01744717597._2 | |
357 | + | let buyAssetName = $t01744717597._3 | |
358 | + | let buyAssetDecimals = $t01744717597._4 | |
622 | 359 | [IntegerEntry((toString(launchpadId) + keyTransferBuyAsset), amount), ScriptTransfer(projectAddress, amount, buyAssetId)] | |
623 | 360 | })) | |
624 | 361 |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 5 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let keyActive = "active" | |
5 | 5 | ||
6 | 6 | let keyAdminPubKey1 = "admin_pub_1" | |
7 | 7 | ||
8 | 8 | let keyAdminPubKey2 = "admin_pub_2" | |
9 | 9 | ||
10 | 10 | let keyAdminPubKey3 = "admin_pub_3" | |
11 | 11 | ||
12 | 12 | let keyLaunchpadDataTransactionStatus = "launchpad_data_transaction_status" | |
13 | 13 | ||
14 | 14 | let keyLaunchpadNextId = "launchpad_next_id" | |
15 | 15 | ||
16 | 16 | let keyLaunchpadActiveId = "launchpad_active_id" | |
17 | 17 | ||
18 | 18 | let keyInitCaller = "init_caller" | |
19 | 19 | ||
20 | 20 | let keyAddress = "_address" | |
21 | 21 | ||
22 | 22 | let keyInitHeight = "_init_height" | |
23 | 23 | ||
24 | 24 | let keyStartHeight = "_start_height" | |
25 | 25 | ||
26 | 26 | let keyFinaliseHeight = "_finalise_height" | |
27 | 27 | ||
28 | 28 | let keyComission = "_comission" | |
29 | 29 | ||
30 | 30 | let keyAssetId = "_asset_id" | |
31 | 31 | ||
32 | 32 | let keyTotalTokenAmount = "_total_token_amount" | |
33 | 33 | ||
34 | 34 | let keyTotalAmountBuyAssetSold = "_total_buy_asset_sold" | |
35 | 35 | ||
36 | 36 | let keyTokensPerTicket = "_tokens_per_ticket" | |
37 | 37 | ||
38 | 38 | let keyPricePerToken = "_price_per_token" | |
39 | 39 | ||
40 | 40 | let keySwopPerTicket = "_swop_per_ticket" | |
41 | 41 | ||
42 | 42 | let keySwopfiTicketsAmount = "_tickets_swopfi_members" | |
43 | 43 | ||
44 | 44 | let keyCampaignTokensAmount = "_tokens_access_list" | |
45 | 45 | ||
46 | 46 | let keyTokensPerAllocation = "_tokens_per_allocation" | |
47 | 47 | ||
48 | 48 | let keyTicketLastNumber = "_ticket_last_number" | |
49 | 49 | ||
50 | 50 | let keyGovernanceStaked = "_SWOP_amount" | |
51 | 51 | ||
52 | 52 | let keyUserRefId = "_ref_id" | |
53 | 53 | ||
54 | 54 | let keyVerifyHash = "_verify_hash" | |
55 | 55 | ||
56 | 56 | let keyDuration = "_duration" | |
57 | 57 | ||
58 | 58 | let keyAmountBuyAssetInMarketing = "_total_purchased_buy_asset_in_marketing" | |
59 | 59 | ||
60 | 60 | let keyAmountBuyAssetInMarketingFinalized = "_total_purchased_buy_asset_in_marketing_finalized" | |
61 | 61 | ||
62 | 62 | let keyTransferBuyAsset = "_transfer_buy_asset" | |
63 | 63 | ||
64 | 64 | let keyFreezSwopDuration = "_freeze_swop_duration" | |
65 | 65 | ||
66 | 66 | let keyFreezeAdditionalToken = "_freeze_additional_token" | |
67 | 67 | ||
68 | 68 | let keyFreezeTokenId = "_freeze_token_id" | |
69 | 69 | ||
70 | 70 | let keyFreezeTokenDuration = "_freeze_token_duration" | |
71 | 71 | ||
72 | 72 | let keyFreezeTokenPerTicket = "_freeze_token_per_ticket" | |
73 | 73 | ||
74 | 74 | let keyBuyToken = "_buy_token" | |
75 | 75 | ||
76 | 76 | let keyClaimParams = "_claim_params" | |
77 | 77 | ||
78 | 78 | let keyHighCompetitionComission = "_high_competition_comission" | |
79 | 79 | ||
80 | 80 | let keyHighCompetitionSale = "_high_competition_sale" | |
81 | 81 | ||
82 | 82 | let keyHCTicketLastNumber = "_high_competition_ticket_last_number" | |
83 | 83 | ||
84 | 84 | let keyHighCompetitionAmount = "_tickets_high_competition" | |
85 | 85 | ||
86 | 86 | let keyTotalAmountBuyAssetSoldHC = "_total_buy_asset_sold_high_competition" | |
87 | 87 | ||
88 | 88 | let keyUserClaimedTokens = "_claimed_tokens" | |
89 | 89 | ||
90 | 90 | let keyUserClaimedBuyTokens = "_claimed_buy_tokens" | |
91 | 91 | ||
92 | 92 | let keyUserBoughtTicketsAmount = "_bought_tickets" | |
93 | 93 | ||
94 | 94 | let keyUserTickets = "_tickets_number" | |
95 | 95 | ||
96 | 96 | let keyUserHCTickets = "_tickets_number_high_competition" | |
97 | 97 | ||
98 | 98 | let keyUserHCBoughtTicketsAmount = "_bought_tickets_high_competition" | |
99 | 99 | ||
100 | 100 | let keyUserAvailableAllocations = "_available_purchase_marketing" | |
101 | 101 | ||
102 | 102 | let keyUserAmountBuyAssetInMarketing = "_purchased_buy_asset_in_marketing" | |
103 | 103 | ||
104 | 104 | let keyUserBoughtAllocations = "_bought_allocations" | |
105 | 105 | ||
106 | 106 | let keyUserClaimStatus = "_claim_status" | |
107 | 107 | ||
108 | 108 | let keyUserWinnedTicketsAmount = "_tickets_result" | |
109 | 109 | ||
110 | 110 | let keyUserWinnedHighCompetitionTicketsAmount = "_tickets_result_high_competition" | |
111 | 111 | ||
112 | 112 | let keyUserFreezeToken = "_freeze_token_amount" | |
113 | 113 | ||
114 | 114 | let oracle = Address(base58'3NBBWfzZtZtszaXbitTKnrB2xXwv26Bn7H9') | |
115 | 115 | ||
116 | 116 | let governance = Address(base58'3N5W8da2iiijVieA6qLGo7KzCJj8B19smWU') | |
117 | 117 | ||
118 | 118 | let commissionWallet = Address(base58'3N2hBdeDEs7wCNA9EY8qv3B6drjgKD64xQG') | |
119 | 119 | ||
120 | 120 | let stakingUSDNAddress = Address(base58'3N6q7sCGSSLBUXDdjBdYGTJbZGZfhhh8cNg') | |
121 | 121 | ||
122 | 122 | let USDN = base58'8UrfDVd5GreeUwm7uPk7eYz1eMv376kzR52C6sANPkwS' | |
123 | 123 | ||
124 | 124 | let adminPubKeyStaking = base58'Kn7NpzaG12dLZgcHf2ipUftU6hbJygmrhFqQYE4B7ZK' | |
125 | 125 | ||
126 | 126 | let maxAllocationsAmount = 2 | |
127 | 127 | ||
128 | 128 | let activeLaunchpadId = valueOrElse(getIntegerValue(this, keyLaunchpadActiveId), -1) | |
129 | 129 | ||
130 | 130 | func getAdminPub (keyAdminPub) = match getString(oracle, keyAdminPub) { | |
131 | 131 | case string: String => | |
132 | 132 | fromBase58String(string) | |
133 | 133 | case nothing => | |
134 | 134 | throw("Admin public key is empty") | |
135 | 135 | } | |
136 | 136 | ||
137 | 137 | ||
138 | 138 | let adminPubKey1 = getAdminPub(keyAdminPubKey1) | |
139 | 139 | ||
140 | 140 | let adminPubKey2 = getAdminPub(keyAdminPubKey2) | |
141 | 141 | ||
142 | 142 | let adminPubKey3 = getAdminPub(keyAdminPubKey3) | |
143 | 143 | ||
144 | 144 | func getAssetInfo (assetId) = match assetId { | |
145 | 145 | case id: ByteVector => | |
146 | 146 | let stringId = toBase58String(id) | |
147 | 147 | let info = valueOrErrorMessage(assetInfo(id), (("Asset " + stringId) + " doesn't exist")) | |
148 | 148 | $Tuple3(stringId, info.name, info.decimals) | |
149 | 149 | case waves: Unit => | |
150 | 150 | $Tuple3("WAVES", "WAVES", 8) | |
151 | 151 | case _ => | |
152 | 152 | throw("Match error") | |
153 | 153 | } | |
154 | 154 | ||
155 | 155 | ||
156 | 156 | func getLaunchpadNextId () = valueOrElse(getInteger(this, keyLaunchpadNextId), 1) | |
157 | 157 | ||
158 | 158 | ||
159 | + | func stakedUsdnAmount () = valueOrElse(getInteger(stakingUSDNAddress, ((("rpd_balance_" + toBase58String(USDN)) + "_") + toString(this))), 0) | |
160 | + | ||
161 | + | ||
159 | 162 | let active = valueOrElse(getBoolean(this, keyActive), true) | |
160 | 163 | ||
161 | 164 | func isActive () = if (active) | |
162 | 165 | then unit | |
163 | 166 | else throw("DApp is inactive at this moment") | |
164 | 167 | ||
165 | 168 | ||
166 | 169 | func isAdminCall (i) = if (containsElement([adminPubKey1, adminPubKey2, adminPubKey3], i.callerPublicKey)) | |
167 | 170 | then unit | |
168 | 171 | else throw("Only admin can call this function") | |
169 | 172 | ||
170 | 173 | ||
171 | 174 | func getAssetInfoFromString (assetStr) = if ((assetStr == "WAVES")) | |
172 | 175 | then $Tuple4(unit, "WAVES", "WAVES", 8) | |
173 | 176 | else { | |
174 | 177 | let stringId = assetStr | |
175 | 178 | let id = fromBase58String(assetStr) | |
176 | 179 | let info = valueOrErrorMessage(assetInfo(id), (("Asset " + stringId) + " doesn't exist")) | |
177 | 180 | $Tuple4(id, stringId, info.name, info.decimals) | |
178 | 181 | } | |
179 | 182 | ||
180 | 183 | ||
181 | 184 | @Callable(i) | |
182 | 185 | func hashingRandom (launchpadId,hash) = valueOrElse(isActive(), valueOrElse(isAdminCall(i), [StringEntry((toString(launchpadId) + keyVerifyHash), hash)])) | |
183 | 186 | ||
184 | 187 | ||
185 | 188 | ||
186 | 189 | @Callable(i) | |
187 | 190 | func initCaller (address) = valueOrElse(isActive(), valueOrElse(isAdminCall(i), [StringEntry(keyInitCaller, address)])) | |
188 | 191 | ||
189 | 192 | ||
190 | 193 | ||
191 | 194 | @Callable(i) | |
192 | 195 | func init (projectAddress,startHeight,duration,buyToken,commission,tokensPerTicket,pricePerToken,swopfiTicketsAmount,campaignTokensAmount,tokensPerAllocation,swopPerTicket,swopFreezeDuration,claimParams,HighCompetitionSale,highCompetitionComission,highCompetitionTicketsAmount,freezeAdditionalToken,freezeParams) = valueOrElse(isActive(), if ((toString(i.caller) != valueOrElse(getString(this, keyInitCaller), ""))) | |
193 | 196 | then throw("Only project admin can call init function") | |
194 | 197 | else if ((size(i.payments) != 1)) | |
195 | 198 | then throw("One attached payment expected") | |
196 | 199 | else if ((height > startHeight)) | |
197 | 200 | then throw("Start height must be greater than blockchain height") | |
198 | 201 | else if (if (freezeAdditionalToken) | |
199 | 202 | then (size(freezeParams) != 3) | |
200 | 203 | else false) | |
201 | 204 | then throw("Freeze params list must have 3 items") | |
202 | 205 | else { | |
203 | 206 | let launchpadId = getLaunchpadNextId() | |
204 | 207 | let launchpadIdStr = toString(launchpadId) | |
205 | 208 | let $t063976472 = $Tuple2(i.payments[0].amount, i.payments[0].assetId) | |
206 | 209 | let pmtAmount = $t063976472._1 | |
207 | 210 | let pmtAssetId = $t063976472._2 | |
208 | 211 | let $t064816554 = getAssetInfo(pmtAssetId) | |
209 | 212 | let pmtStrAssetId = $t064816554._1 | |
210 | 213 | let pmtAssetName = $t064816554._2 | |
211 | 214 | let pmtDecimals = $t064816554._3 | |
212 | 215 | let freezeAdditionalEntry = if (freezeAdditionalToken) | |
213 | 216 | then [BooleanEntry((launchpadIdStr + keyFreezeAdditionalToken), true), StringEntry((launchpadIdStr + keyFreezeTokenId), freezeParams[0]), IntegerEntry((launchpadIdStr + keyFreezeTokenDuration), parseIntValue(freezeParams[1])), IntegerEntry((launchpadIdStr + keyFreezeTokenPerTicket), parseIntValue(freezeParams[2]))] | |
214 | 217 | else nil | |
215 | 218 | ([IntegerEntry((pmtStrAssetId + "_launchpad"), launchpadId), IntegerEntry(keyLaunchpadNextId, (launchpadId + 1)), StringEntry((launchpadIdStr + keyAddress), projectAddress), IntegerEntry((launchpadIdStr + keyInitHeight), height), IntegerEntry((launchpadIdStr + keyStartHeight), startHeight), IntegerEntry((launchpadIdStr + keyDuration), duration), StringEntry((launchpadIdStr + keyBuyToken), buyToken), IntegerEntry((launchpadIdStr + keyFreezSwopDuration), swopFreezeDuration), IntegerEntry((launchpadIdStr + keyTotalTokenAmount), pmtAmount), StringEntry((launchpadIdStr + keyAssetId), pmtStrAssetId), IntegerEntry((launchpadIdStr + keyComission), commission), IntegerEntry((launchpadIdStr + keyTicketLastNumber), 0), IntegerEntry((launchpadIdStr + keyHCTicketLastNumber), 0), IntegerEntry((launchpadIdStr + keyTokensPerTicket), tokensPerTicket), IntegerEntry((launchpadIdStr + keyPricePerToken), pricePerToken), IntegerEntry((launchpadIdStr + keySwopPerTicket), swopPerTicket), IntegerEntry((launchpadIdStr + keySwopfiTicketsAmount), swopfiTicketsAmount), IntegerEntry((launchpadIdStr + keyCampaignTokensAmount), campaignTokensAmount), IntegerEntry((launchpadIdStr + keyTokensPerAllocation), tokensPerAllocation), StringEntry((launchpadIdStr + keyClaimParams), makeString(claimParams, ",")), BooleanEntry((launchpadIdStr + keyHighCompetitionSale), HighCompetitionSale), IntegerEntry((launchpadIdStr + keyHighCompetitionComission), highCompetitionComission), IntegerEntry((launchpadIdStr + keyHighCompetitionAmount), highCompetitionTicketsAmount), IntegerEntry(keyLaunchpadActiveId, launchpadId), StringEntry(keyInitCaller, "")] ++ freezeAdditionalEntry) | |
216 | 219 | }) | |
217 | 220 | ||
218 | 221 | ||
219 | 222 | ||
220 | 223 | @Callable(i) | |
221 | - | func commitSwopfiSale (launchpadId,refId) = valueOrElse(isActive(), { | |
222 | - | let $t091609235 = $Tuple2(i.payments[0].amount, i.payments[0].assetId) | |
223 | - | let pmtAmount = $t091609235._1 | |
224 | - | let pmtAssetId = $t091609235._2 | |
225 | - | let $t092409390 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
226 | - | let buyAssetId = $t092409390._1 | |
227 | - | let buyAssetStrId = $t092409390._2 | |
228 | - | let buyAssetName = $t092409390._3 | |
229 | - | let buyAssetDecimals = $t092409390._4 | |
230 | - | let stakedSwops = valueOrElse(getInteger(governance, (toString(i.caller) + keyGovernanceStaked)), 0) | |
231 | - | let allowedTicketsAmountAll = (stakedSwops / getIntegerValue(this, (toString(launchpadId) + keySwopPerTicket))) | |
232 | - | let boughtTicketsAmount = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtTicketsAmount)), 0) | |
233 | - | let allowedTicketsAmount = (allowedTicketsAmountAll - boughtTicketsAmount) | |
234 | - | let startHeight = value(getInteger(this, (toString(launchpadId) + keyStartHeight))) | |
235 | - | let duration = value(getInteger(this, (toString(launchpadId) + keyDuration))) | |
236 | - | let freezeAdditionalToken = valueOrElse(getBoolean(this, (toString(launchpadId) + keyFreezeAdditionalToken)), false) | |
237 | - | if ((activeLaunchpadId != launchpadId)) | |
238 | - | then throw("There is no active launchpad or this launchpad is ended") | |
239 | - | else if ((startHeight > height)) | |
240 | - | then throw("Launchpad sale not started yet") | |
241 | - | else if ((height > (startHeight + duration))) | |
242 | - | then throw("Launchpad sale ended") | |
243 | - | else if (if (!(freezeAdditionalToken)) | |
244 | - | then if ((size(i.payments) != 1)) | |
245 | - | then true | |
246 | - | else (pmtAssetId != buyAssetId) | |
247 | - | else false) | |
248 | - | then throw((("One attached payment in " + buyAssetName) + " expected")) | |
249 | - | else if (if (freezeAdditionalToken) | |
250 | - | then if ((size(i.payments) != 2)) | |
251 | - | then true | |
252 | - | else (pmtAssetId != buyAssetId) | |
253 | - | else false) | |
254 | - | then { | |
255 | - | let freezeTokenId = getStringValue(this, (toString(launchpadId) + keyFreezeTokenId)) | |
256 | - | throw(((("Two attached payments expected: First in " + buyAssetName) + ", second in ") + freezeTokenId)) | |
257 | - | } | |
258 | - | else if (if ((0 >= allowedTicketsAmountAll)) | |
259 | - | then true | |
260 | - | else (0 >= allowedTicketsAmount)) | |
261 | - | then throw("Not enought SWOP in staking to buy tickets") | |
262 | - | else { | |
263 | - | let tokenPrice = getIntegerValue(this, (toString(launchpadId) + keyPricePerToken)) | |
264 | - | let tokensPerTicket = getIntegerValue(this, (toString(launchpadId) + keyTokensPerTicket)) | |
265 | - | let $t01120211346 = getAssetInfoFromString(getStringValue(this, (toString(launchpadId) + keyAssetId))) | |
266 | - | let tokenId = $t01120211346._1 | |
267 | - | let tokenStrAssetId = $t01120211346._2 | |
268 | - | let tokenAssetName = $t01120211346._3 | |
269 | - | let tokenDecimals = $t01120211346._4 | |
270 | - | let ticketPrice = fraction(tokensPerTicket, tokenPrice, pow(10, 0, tokenDecimals, 0, 0, DOWN)) | |
271 | - | let commissionPerTicket = fraction(ticketPrice, getIntegerValue(this, (toString(launchpadId) + keyComission)), 100) | |
272 | - | let buyTicketAmount = (pmtAmount / (ticketPrice + commissionPerTicket)) | |
273 | - | let allowedBuyTicketAmount = min([buyTicketAmount, allowedTicketsAmount]) | |
274 | - | let allowedBuyPriceWithComission = (allowedBuyTicketAmount * (ticketPrice + commissionPerTicket)) | |
275 | - | let change = (pmtAmount - allowedBuyPriceWithComission) | |
276 | - | if ((buyTicketAmount == 0)) | |
277 | - | then throw((("Not enought " + buyAssetName) + " to buy tickets")) | |
278 | - | else if ((change != 0)) | |
279 | - | then throw((((("Wrong payment. To buy " + toString(allowedBuyTicketAmount)) + "tickets you need to pay ") + toString(allowedBuyPriceWithComission)) + buyAssetName)) | |
280 | - | else { | |
281 | - | let freezeAdditionalEntry = if (freezeAdditionalToken) | |
282 | - | then { | |
283 | - | let freezeTokenId = getStringValue(this, (toString(launchpadId) + keyFreezeTokenId)) | |
284 | - | let freezeTokenPerTicket = getIntegerValue(this, (toString(launchpadId) + keyFreezeTokenPerTicket)) | |
285 | - | let $t01251012597 = $Tuple2(i.payments[1].amount, i.payments[1].assetId) | |
286 | - | let pmtAmountFreeze = $t01251012597._1 | |
287 | - | let pmtAssetIdFreeze = $t01251012597._2 | |
288 | - | let $t01261412705 = getAssetInfo(pmtAssetId) | |
289 | - | let pmtFreezeStrAssetId = $t01261412705._1 | |
290 | - | let pmtFreezeAssetName = $t01261412705._2 | |
291 | - | let pmtFreezeDecimals = $t01261412705._3 | |
292 | - | let needFreezeTokens = (freezeTokenPerTicket * buyTicketAmount) | |
293 | - | if (if ((pmtFreezeStrAssetId != freezeTokenId)) | |
294 | - | then true | |
295 | - | else (needFreezeTokens != pmtAmountFreeze)) | |
296 | - | then throw((((("You need to add " + toString(needFreezeTokens)) + " ") + pmtFreezeAssetName) + " as a second payment")) | |
297 | - | else [IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserFreezeToken), pmtAmountFreeze)] | |
298 | - | } | |
299 | - | else nil | |
300 | - | let userBoughtTicketsNumbers = valueOrElse(getString(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserTickets)), "") | |
301 | - | let lastBoughtTicketNumber = getIntegerValue(this, (toString(launchpadId) + keyTicketLastNumber)) | |
302 | - | let boughtRange = ((toString((lastBoughtTicketNumber + 1)) + "-") + toString((lastBoughtTicketNumber + allowedBuyTicketAmount))) | |
303 | - | let newUserBoughtTicketsNumbers = (userBoughtTicketsNumbers + (if ((userBoughtTicketsNumbers != "")) | |
304 | - | then ("," + boughtRange) | |
305 | - | else ("" + boughtRange))) | |
306 | - | let inv = if ((buyAssetId == USDN)) | |
307 | - | then invoke(stakingUSDNAddress, "lockNeutrino", nil, [AttachedPayment(pmtAssetId, pmtAmount)]) | |
308 | - | else 0 | |
309 | - | if ((inv == inv)) | |
310 | - | then { | |
311 | - | let baseEntry = ([IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtTicketsAmount), (boughtTicketsAmount + allowedBuyTicketAmount)), IntegerEntry((toString(launchpadId) + keyTicketLastNumber), (lastBoughtTicketNumber + allowedBuyTicketAmount)), StringEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserTickets), newUserBoughtTicketsNumbers)] ++ freezeAdditionalEntry) | |
312 | - | if (!(isDefined(getString(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId))))) | |
313 | - | then (baseEntry ++ [StringEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId), refId)]) | |
314 | - | else baseEntry | |
315 | - | } | |
316 | - | else throw("Strict value is not equal to itself.") | |
317 | - | } | |
318 | - | } | |
319 | - | }) | |
320 | - | ||
321 | - | ||
322 | - | ||
323 | - | @Callable(i) | |
324 | - | func commitAccessListSale (launchpadId,refId) = valueOrElse(isActive(), { | |
325 | - | let $t01477614851 = $Tuple2(i.payments[0].amount, i.payments[0].assetId) | |
326 | - | let pmtAmount = $t01477614851._1 | |
327 | - | let pmtAssetId = $t01477614851._2 | |
328 | - | let $t01485615006 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
329 | - | let buyAssetId = $t01485615006._1 | |
330 | - | let buyAssetStrId = $t01485615006._2 | |
331 | - | let buyAssetName = $t01485615006._3 | |
332 | - | let buyAssetDecimals = $t01485615006._4 | |
333 | - | let userUsdnInMarketing = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserAmountBuyAssetInMarketing)), 0) | |
334 | - | let usdnInMarketing = valueOrElse(getInteger(this, (toString(launchpadId) + keyAmountBuyAssetInMarketing)), 0) | |
335 | - | let boughtAllocations = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtAllocations)), 0) | |
336 | - | let commission = getIntegerValue(this, (toString(launchpadId) + keyComission)) | |
337 | - | let tokenPrice = getIntegerValue(this, (toString(launchpadId) + keyPricePerToken)) | |
338 | - | let tokensPerAllocation = getIntegerValue(this, (toString(launchpadId) + keyTokensPerAllocation)) | |
339 | - | let $t01567515819 = getAssetInfoFromString(getStringValue(this, (toString(launchpadId) + keyAssetId))) | |
340 | - | let tokenId = $t01567515819._1 | |
341 | - | let tokenStrAssetId = $t01567515819._2 | |
342 | - | let tokenAssetName = $t01567515819._3 | |
343 | - | let tokenDecimals = $t01567515819._4 | |
344 | - | let allocationPriceWithComission = fraction(fraction(tokenPrice, tokensPerAllocation, pow(10, 0, tokenDecimals, 0, 0, DOWN)), (100 + commission), 100) | |
345 | - | let startHeight = value(getInteger(this, (toString(launchpadId) + keyStartHeight))) | |
346 | - | let duration = value(getInteger(this, (toString(launchpadId) + keyDuration))) | |
347 | - | if ((activeLaunchpadId != launchpadId)) | |
348 | - | then throw("There is no active launchpad or this launchpad is ended") | |
349 | - | else if ((startHeight > height)) | |
350 | - | then throw("Launchpad sale not started yet") | |
351 | - | else if ((height > (startHeight + duration))) | |
352 | - | then throw("Launchpad sale ended") | |
353 | - | else if (if ((size(i.payments) != 1)) | |
354 | - | then true | |
355 | - | else (pmtAssetId != buyAssetId)) | |
356 | - | then throw((("One attached payment in " + buyAssetName) + " expected")) | |
357 | - | else if ((boughtAllocations >= maxAllocationsAmount)) | |
358 | - | then throw((("You can buy only " + toString(maxAllocationsAmount)) + "allocations")) | |
359 | - | else if (if ((allocationPriceWithComission != pmtAmount)) | |
360 | - | then ((allocationPriceWithComission * 2) != pmtAmount) | |
361 | - | else false) | |
362 | - | then throw(((("Wrong payment. You can buy 1 or 2 allocations for " + toString(allocationPriceWithComission)) + buyAssetName) + " per allocation")) | |
363 | - | else { | |
364 | - | let buyAllocationsAmount = (pmtAmount / allocationPriceWithComission) | |
365 | - | let inv = if ((buyAssetId == USDN)) | |
366 | - | then invoke(stakingUSDNAddress, "lockNeutrino", nil, [AttachedPayment(pmtAssetId, pmtAmount)]) | |
367 | - | else 0 | |
368 | - | if ((inv == inv)) | |
369 | - | then { | |
370 | - | let baseEntry = [IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtAllocations), (boughtAllocations + buyAllocationsAmount)), IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserAmountBuyAssetInMarketing), (userUsdnInMarketing + pmtAmount)), IntegerEntry((toString(launchpadId) + keyAmountBuyAssetInMarketing), (usdnInMarketing + pmtAmount))] | |
371 | - | if (!(isDefined(getString(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId))))) | |
372 | - | then (baseEntry ++ [StringEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId), refId)]) | |
373 | - | else baseEntry | |
374 | - | } | |
375 | - | else throw("Strict value is not equal to itself.") | |
376 | - | } | |
377 | - | }) | |
378 | - | ||
379 | - | ||
380 | - | ||
381 | - | @Callable(i) | |
382 | - | func commitHighCompetitionSale (launchpadId,refId) = valueOrElse(isActive(), { | |
383 | - | let $t01807718152 = $Tuple2(i.payments[0].amount, i.payments[0].assetId) | |
384 | - | let pmtAmount = $t01807718152._1 | |
385 | - | let pmtAssetId = $t01807718152._2 | |
386 | - | let $t01815718307 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
387 | - | let buyAssetId = $t01815718307._1 | |
388 | - | let buyAssetStrId = $t01815718307._2 | |
389 | - | let buyAssetName = $t01815718307._3 | |
390 | - | let buyAssetDecimals = $t01815718307._4 | |
391 | - | let startHeight = value(getInteger(this, (toString(launchpadId) + keyStartHeight))) | |
392 | - | let duration = value(getInteger(this, (toString(launchpadId) + keyDuration))) | |
393 | - | let boughtHCTicketsAmount = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserHCBoughtTicketsAmount)), 0) | |
394 | - | let highCompetitionSale = valueOrElse(getBoolean(this, (toString(launchpadId) + keyHighCompetitionSale)), false) | |
395 | - | if ((activeLaunchpadId != launchpadId)) | |
396 | - | then throw("There is no active launchpad or this launchpad is ended") | |
397 | - | else if (!(highCompetitionSale)) | |
398 | - | then throw("There is no high competition sale for this launchpad") | |
399 | - | else if ((startHeight > height)) | |
400 | - | then throw("Launchpad sale not started yet") | |
401 | - | else if ((height > (startHeight + duration))) | |
402 | - | then throw("Launchpad sale ended") | |
403 | - | else if (if ((size(i.payments) != 1)) | |
404 | - | then true | |
405 | - | else (pmtAssetId != buyAssetId)) | |
406 | - | then throw((("One attached payment in " + buyAssetName) + " expected")) | |
407 | - | else { | |
408 | - | let tokenPrice = getIntegerValue(this, (toString(launchpadId) + keyPricePerToken)) | |
409 | - | let tokensPerTicket = getIntegerValue(this, (toString(launchpadId) + keyTokensPerTicket)) | |
410 | - | let $t01948019624 = getAssetInfoFromString(getStringValue(this, (toString(launchpadId) + keyAssetId))) | |
411 | - | let tokenId = $t01948019624._1 | |
412 | - | let tokenStrAssetId = $t01948019624._2 | |
413 | - | let tokenAssetName = $t01948019624._3 | |
414 | - | let tokenDecimals = $t01948019624._4 | |
415 | - | let ticketPrice = fraction(tokensPerTicket, tokenPrice, pow(10, 0, tokenDecimals, 0, 0, DOWN)) | |
416 | - | let commissionPerTicket = fraction(ticketPrice, getIntegerValue(this, (toString(launchpadId) + keyHighCompetitionComission)), 100) | |
417 | - | let buyTicketAmount = (pmtAmount / (ticketPrice + commissionPerTicket)) | |
418 | - | let buyPriceWithComission = (buyTicketAmount * (ticketPrice + commissionPerTicket)) | |
419 | - | let change = (pmtAmount - buyPriceWithComission) | |
420 | - | if ((buyTicketAmount == 0)) | |
421 | - | then throw((("Not enought " + buyAssetName) + " to buy tickets")) | |
422 | - | else if ((change != 0)) | |
423 | - | then throw((((("Wrong payment. To buy " + toString(buyTicketAmount)) + "tickets you need to pay ") + toString(buyPriceWithComission)) + buyAssetName)) | |
424 | - | else { | |
425 | - | let userBoughtTicketsNumbers = valueOrElse(getString(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserHCTickets)), "") | |
426 | - | let lastBoughtTicketNumber = getIntegerValue(this, (toString(launchpadId) + keyHCTicketLastNumber)) | |
427 | - | let boughtRange = ((toString((lastBoughtTicketNumber + 1)) + "-") + toString((lastBoughtTicketNumber + buyTicketAmount))) | |
428 | - | let newUserBoughtTicketsNumbers = (userBoughtTicketsNumbers + (if ((userBoughtTicketsNumbers != "")) | |
429 | - | then ("," + boughtRange) | |
430 | - | else ("" + boughtRange))) | |
431 | - | let inv = if ((buyAssetId == USDN)) | |
432 | - | then invoke(stakingUSDNAddress, "lockNeutrino", nil, [AttachedPayment(pmtAssetId, pmtAmount)]) | |
433 | - | else 0 | |
434 | - | if ((inv == inv)) | |
435 | - | then { | |
436 | - | let baseEntry = [IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserHCBoughtTicketsAmount), (boughtHCTicketsAmount + buyTicketAmount)), IntegerEntry((toString(launchpadId) + keyHCTicketLastNumber), (lastBoughtTicketNumber + buyTicketAmount)), StringEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserHCTickets), newUserBoughtTicketsNumbers)] | |
437 | - | if (!(isDefined(getString(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId))))) | |
438 | - | then (baseEntry ++ [StringEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserRefId), refId)]) | |
439 | - | else baseEntry | |
440 | - | } | |
441 | - | else throw("Strict value is not equal to itself.") | |
442 | - | } | |
443 | - | } | |
444 | - | }) | |
445 | - | ||
446 | - | ||
447 | - | ||
448 | - | @Callable(i) | |
449 | - | func finalise (launchpadId,vrfHeight,secretWord) = valueOrElse(isActive(), valueOrElse(isAdminCall(i), { | |
450 | - | let savedHash = getStringValue(this, (toString(launchpadId) + keyVerifyHash)) | |
451 | - | let calcHash = toBase58String(sha256((toBytes(vrfHeight) + toBytes(secretWord)))) | |
452 | - | if ((calcHash != savedHash)) | |
453 | - | then throw("vrf Height hash not matching") | |
454 | - | else { | |
455 | - | let $t02223722387 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
456 | - | let buyAssetId = $t02223722387._1 | |
457 | - | let buyAssetStrId = $t02223722387._2 | |
458 | - | let buyAssetName = $t02223722387._3 | |
459 | - | let buyAssetDecimals = $t02223722387._4 | |
460 | - | let projectAddress = addressFromStringValue(getStringValue(this, (toString(launchpadId) + keyAddress))) | |
461 | - | let commission = getIntegerValue(this, (toString(launchpadId) + keyComission)) | |
462 | - | let commissionHC = getIntegerValue(this, (toString(launchpadId) + keyHighCompetitionComission)) | |
463 | - | let transferedBuyAsset = valueOrElse(getInteger(this, (toString(launchpadId) + keyTransferBuyAsset)), 0) | |
464 | - | let swopfiMembersBuyAssetAmount = getIntegerValue(this, (toString(launchpadId) + keyTotalAmountBuyAssetSold)) | |
465 | - | let swopfiHCBuyAssetAmount = valueOrElse(getIntegerValue(this, (toString(launchpadId) + keyTotalAmountBuyAssetSoldHC)), 0) | |
466 | - | let swopfiMembersComission = fraction(swopfiMembersBuyAssetAmount, commission, 100) | |
467 | - | let swopfiHCComission = fraction(swopfiHCBuyAssetAmount, commissionHC, 100) | |
468 | - | let totalAmountBuyAssetInMarketing = getIntegerValue(this, (toString(launchpadId) + keyAmountBuyAssetInMarketingFinalized)) | |
469 | - | let marketingComission = fraction(totalAmountBuyAssetInMarketing, commission, 100) | |
470 | - | let unstakeAmount = (((((swopfiMembersBuyAssetAmount + totalAmountBuyAssetInMarketing) + swopfiMembersComission) + marketingComission) + swopfiHCBuyAssetAmount) + swopfiHCComission) | |
471 | - | if ((transferedBuyAsset > (swopfiMembersBuyAssetAmount + totalAmountBuyAssetInMarketing))) | |
472 | - | then throw("Can't transfer negative value to project") | |
473 | - | else { | |
474 | - | let inv = if ((buyAssetId == USDN)) | |
475 | - | then invoke(stakingUSDNAddress, "unlockNeutrino", [unstakeAmount, toBase58String(USDN)], nil) | |
476 | - | else 0 | |
477 | - | if ((inv == inv)) | |
478 | - | then [IntegerEntry((toString(launchpadId) + keyFinaliseHeight), height), ScriptTransfer(projectAddress, ((swopfiMembersBuyAssetAmount + totalAmountBuyAssetInMarketing) - transferedBuyAsset), buyAssetId), ScriptTransfer(commissionWallet, ((swopfiMembersComission + marketingComission) + swopfiHCComission), buyAssetId)] | |
479 | - | else throw("Strict value is not equal to itself.") | |
480 | - | } | |
481 | - | } | |
482 | - | })) | |
483 | - | ||
484 | - | ||
485 | - | ||
486 | - | @Callable(i) | |
487 | 224 | func claim (launchpadId) = valueOrElse(isActive(), { | |
488 | 225 | let boughtTickets = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtTicketsAmount)), 0) | |
489 | 226 | let boughtHCTickets = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserHCBoughtTicketsAmount)), 0) | |
490 | 227 | let boughtAllocations = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserBoughtAllocations)), 0) | |
491 | 228 | let userClaimStatus = valueOrElse(getBoolean(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserClaimStatus)), false) | |
492 | 229 | let finalizeHeight = valueOrElse(getInteger(this, (toString(launchpadId) + keyFinaliseHeight)), 0) | |
493 | 230 | if ((finalizeHeight == 0)) | |
494 | 231 | then throw("You can't claim because results are not finalized") | |
495 | 232 | else if (userClaimStatus) | |
496 | 233 | then throw("You are already claimed") | |
497 | 234 | else if (if (if ((boughtTickets == 0)) | |
498 | 235 | then (boughtAllocations == 0) | |
499 | 236 | else false) | |
500 | 237 | then (boughtHCTickets == 0) | |
501 | 238 | else false) | |
502 | 239 | then throw("You can't claim because you don't buy anything") | |
503 | 240 | else { | |
504 | - | let $ | |
505 | - | let buyAssetId = $ | |
506 | - | let buyAssetStrId = $ | |
507 | - | let buyAssetName = $ | |
508 | - | let buyAssetDecimals = $ | |
241 | + | let $t01012010270 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
242 | + | let buyAssetId = $t01012010270._1 | |
243 | + | let buyAssetStrId = $t01012010270._2 | |
244 | + | let buyAssetName = $t01012010270._3 | |
245 | + | let buyAssetDecimals = $t01012010270._4 | |
509 | 246 | let wonnedTickets = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserWinnedTicketsAmount)), 0) | |
510 | 247 | let wonnedHCTickets = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserWinnedHighCompetitionTicketsAmount)), 0) | |
511 | 248 | let awailableAllocations = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserAvailableAllocations)), 0) | |
512 | 249 | let commission = getIntegerValue(this, (toString(launchpadId) + keyComission)) | |
513 | 250 | let commissionHC = getIntegerValue(this, (toString(launchpadId) + keyHighCompetitionComission)) | |
514 | 251 | let allowedAllocations = min([boughtAllocations, awailableAllocations]) | |
515 | 252 | let tokenPrice = getIntegerValue(this, (toString(launchpadId) + keyPricePerToken)) | |
516 | 253 | let tokensPerTicket = getIntegerValue(this, (toString(launchpadId) + keyTokensPerTicket)) | |
517 | - | let $ | |
518 | - | let tokenId = $ | |
519 | - | let tokenStrAssetId = $ | |
520 | - | let tokenAssetName = $ | |
521 | - | let tokenDecimals = $ | |
254 | + | let $t01117611320 = getAssetInfoFromString(getStringValue(this, (toString(launchpadId) + keyAssetId))) | |
255 | + | let tokenId = $t01117611320._1 | |
256 | + | let tokenStrAssetId = $t01117611320._2 | |
257 | + | let tokenAssetName = $t01117611320._3 | |
258 | + | let tokenDecimals = $t01117611320._4 | |
522 | 259 | let ticketPrice = fraction(tokensPerTicket, tokenPrice, pow(10, 0, tokenDecimals, 0, 0, DOWN)) | |
523 | 260 | let tokensPerAllocation = getIntegerValue(this, (toString(launchpadId) + keyTokensPerAllocation)) | |
524 | 261 | let transferTokensAmount = (((wonnedTickets * tokensPerTicket) + (allowedAllocations * tokensPerAllocation)) + (wonnedHCTickets * tokensPerTicket)) | |
525 | 262 | let notAllowedAllocationsTokens = ((boughtAllocations - allowedAllocations) * tokensPerAllocation) | |
526 | 263 | let buyAssetForAllocations = fraction(fraction(notAllowedAllocationsTokens, tokenPrice, pow(10, 0, tokenDecimals, 0, 0, DOWN)), (100 + commission), 100) | |
527 | 264 | let buyAssetForUnwonnedTickets = fraction(((boughtTickets - wonnedTickets) * ticketPrice), (100 + commission), 100) | |
528 | 265 | let buyAssetForHCUnwonnedTickets = fraction(((boughtHCTickets - wonnedHCTickets) * ticketPrice), (100 + commissionHC), 100) | |
529 | 266 | let claimedBuyToken = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserClaimedBuyTokens)), 0) | |
530 | 267 | if (if (if ((0 > buyAssetForAllocations)) | |
531 | 268 | then true | |
532 | 269 | else (0 > buyAssetForUnwonnedTickets)) | |
533 | 270 | then true | |
534 | 271 | else (0 > buyAssetForHCUnwonnedTickets)) | |
535 | 272 | then throw("Error with allowed allocation param or wonned tickets param please contact support") | |
536 | 273 | else { | |
537 | 274 | let returnedBuyAssetAmount = (((buyAssetForUnwonnedTickets + buyAssetForHCUnwonnedTickets) + buyAssetForAllocations) - claimedBuyToken) | |
538 | 275 | let transferBuyAsset = if ((returnedBuyAssetAmount > 0)) | |
539 | 276 | then [ScriptTransfer(i.caller, returnedBuyAssetAmount, buyAssetId), IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserClaimedBuyTokens), returnedBuyAssetAmount)] | |
540 | 277 | else nil | |
541 | 278 | let inv = if (if ((returnedBuyAssetAmount > 0)) | |
542 | 279 | then (buyAssetId == USDN) | |
543 | 280 | else false) | |
544 | 281 | then invoke(stakingUSDNAddress, "unlockNeutrino", [returnedBuyAssetAmount, toBase58String(USDN)], nil) | |
545 | 282 | else 0 | |
546 | 283 | if ((inv == inv)) | |
547 | 284 | then { | |
548 | 285 | let claimedParams = split(valueOrElse(getString(this, (toString(launchpadId) + keyClaimParams)), ""), ",") | |
549 | 286 | if (if ((size(claimedParams) == 0)) | |
550 | 287 | then true | |
551 | 288 | else (claimedParams[0] == "claimAll")) | |
552 | 289 | then { | |
553 | 290 | let transferTokens = if ((transferTokensAmount > 0)) | |
554 | 291 | then [ScriptTransfer(i.caller, transferTokensAmount, tokenId)] | |
555 | 292 | else nil | |
556 | 293 | (([BooleanEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserClaimStatus), true), IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserClaimedTokens), transferTokensAmount)] ++ transferTokens) ++ transferBuyAsset) | |
557 | 294 | } | |
558 | 295 | else if ((claimedParams[0] == "claimPeriod")) | |
559 | 296 | then { | |
560 | 297 | let claimedTokens = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserClaimedTokens)), 0) | |
561 | 298 | let unlockPeriod = value(parseInt(claimedParams[2])) | |
562 | 299 | let blockHeight = (value(parseInt(claimedParams[3])) + finalizeHeight) | |
563 | 300 | let unlockHeight = ((unlockPeriod + finalizeHeight) + blockHeight) | |
564 | 301 | let allowedTokensAmount = (fraction(transferTokensAmount, value(parseInt(claimedParams[1])), 100) + (if ((height > blockHeight)) | |
565 | 302 | then fraction(transferTokensAmount, (height - blockHeight), unlockPeriod) | |
566 | 303 | else (0 - claimedTokens))) | |
567 | 304 | let claimStatusEntry = if ((transferTokensAmount >= (claimedTokens + allowedTokensAmount))) | |
568 | 305 | then [BooleanEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserClaimStatus), true)] | |
569 | 306 | else nil | |
570 | 307 | if (if ((allowedTokensAmount > 0)) | |
571 | 308 | then true | |
572 | 309 | else (returnedBuyAssetAmount > 0)) | |
573 | 310 | then { | |
574 | 311 | let transferTokens = if ((allowedTokensAmount > 0)) | |
575 | 312 | then [IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserClaimedTokens), (claimedTokens + allowedTokensAmount)), ScriptTransfer(i.caller, allowedTokensAmount, tokenId)] | |
576 | 313 | else nil | |
577 | 314 | (transferBuyAsset ++ transferTokens) | |
578 | 315 | } | |
579 | 316 | else throw("Nothing to claim") | |
580 | 317 | } | |
581 | 318 | else throw("Unknown claim function") | |
582 | 319 | } | |
583 | 320 | else throw("Strict value is not equal to itself.") | |
584 | 321 | } | |
585 | 322 | } | |
586 | 323 | }) | |
587 | 324 | ||
588 | 325 | ||
589 | 326 | ||
590 | 327 | @Callable(i) | |
591 | 328 | func claimFreezeToken (launchpadId) = valueOrElse(isActive(), { | |
592 | 329 | let freezeAdditionalToken = valueOrElse(getBoolean(this, (toString(launchpadId) + keyFreezeAdditionalToken)), false) | |
593 | 330 | if (!(isDefined(getInteger(this, (toString(launchpadId) + keyFinaliseHeight))))) | |
594 | 331 | then throw("You can't claim because results are not finalized") | |
595 | 332 | else if (!(freezeAdditionalToken)) | |
596 | 333 | then throw("There are no additional token freezing for this launchpad") | |
597 | 334 | else { | |
598 | 335 | let tokenId = fromBase58String(getStringValue(this, (toString(launchpadId) + keyFreezeTokenId))) | |
599 | 336 | let userTokensFreezed = valueOrElse(getInteger(this, (((toString(i.caller) + "_") + toString(launchpadId)) + keyUserFreezeToken)), 0) | |
600 | 337 | let startHeight = valueOrElse(getInteger(this, (toString(launchpadId) + keyStartHeight)), 0) | |
601 | 338 | let duration = valueOrElse(getInteger(this, (toString(launchpadId) + keyDuration)), 0) | |
602 | 339 | let freezeTokenDuration = getIntegerValue(this, (toString(launchpadId) + keyFreezeTokenDuration)) | |
603 | 340 | let blockEndHeight = ((startHeight + duration) + freezeTokenDuration) | |
604 | 341 | if ((userTokensFreezed == 0)) | |
605 | 342 | then throw("You are already claimed your tokens or not paid anything") | |
606 | 343 | else if ((blockEndHeight > height)) | |
607 | 344 | then throw(("Your tokens freezed teel " + toString(blockEndHeight))) | |
608 | 345 | else [ScriptTransfer(i.caller, userTokensFreezed, tokenId), IntegerEntry((((toString(i.caller) + "_") + toString(launchpadId)) + keyUserFreezeToken), 0)] | |
609 | 346 | } | |
610 | 347 | }) | |
611 | 348 | ||
612 | 349 | ||
613 | 350 | ||
614 | 351 | @Callable(i) | |
615 | 352 | func transferBuyAsset (launchpadId,amount) = valueOrElse(isActive(), valueOrElse(isAdminCall(i), { | |
616 | 353 | let projectAddress = addressFromStringValue(getStringValue(this, (toString(launchpadId) + keyAddress))) | |
617 | - | let $ | |
618 | - | let buyAssetId = $ | |
619 | - | let buyAssetStrId = $ | |
620 | - | let buyAssetName = $ | |
621 | - | let buyAssetDecimals = $ | |
354 | + | let $t01744717597 = getAssetInfoFromString(value(getString(this, (toString(launchpadId) + keyBuyToken)))) | |
355 | + | let buyAssetId = $t01744717597._1 | |
356 | + | let buyAssetStrId = $t01744717597._2 | |
357 | + | let buyAssetName = $t01744717597._3 | |
358 | + | let buyAssetDecimals = $t01744717597._4 | |
622 | 359 | [IntegerEntry((toString(launchpadId) + keyTransferBuyAsset), amount), ScriptTransfer(projectAddress, amount, buyAssetId)] | |
623 | 360 | })) | |
624 | 361 | ||
625 | 362 | ||
626 | 363 | ||
627 | 364 | @Callable(i) | |
628 | 365 | func shutdown () = valueOrElse(isAdminCall(i), if (!(active)) | |
629 | 366 | then throw("DApp already inactive") | |
630 | 367 | else [BooleanEntry(keyActive, false)]) | |
631 | 368 | ||
632 | 369 | ||
633 | 370 | ||
634 | 371 | @Callable(i) | |
635 | 372 | func activate () = valueOrElse(isAdminCall(i), if (active) | |
636 | 373 | then throw("DApp already active") | |
637 | 374 | else [BooleanEntry(keyActive, true)]) | |
638 | 375 | ||
639 | 376 | ||
640 | 377 | @Verifier(tx) | |
641 | 378 | func verify () = { | |
642 | 379 | let multiSignedByAdmins = { | |
643 | 380 | let adminPubKey1Signed = if (sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey1)) | |
644 | 381 | then 1 | |
645 | 382 | else 0 | |
646 | 383 | let adminPubKey2Signed = if (sigVerify(tx.bodyBytes, tx.proofs[1], adminPubKey2)) | |
647 | 384 | then 1 | |
648 | 385 | else 0 | |
649 | 386 | let adminPubKey3Signed = if (sigVerify(tx.bodyBytes, tx.proofs[2], adminPubKey3)) | |
650 | 387 | then 1 | |
651 | 388 | else 0 | |
652 | 389 | (((adminPubKey1Signed + adminPubKey2Signed) + adminPubKey3Signed) >= 2) | |
653 | 390 | } | |
654 | 391 | let signedByAdmin = if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey1)) | |
655 | 392 | then true | |
656 | 393 | else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey2)) | |
657 | 394 | then true | |
658 | 395 | else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey3)) | |
659 | 396 | then true | |
660 | 397 | else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKeyStaking) | |
661 | 398 | match tx { | |
662 | 399 | case dtx: DataTransaction => | |
663 | 400 | if (valueOrElse(getBoolean(oracle, keyLaunchpadDataTransactionStatus), false)) | |
664 | 401 | then signedByAdmin | |
665 | 402 | else false | |
666 | 403 | case _ => | |
667 | 404 | multiSignedByAdmins | |
668 | 405 | } | |
669 | 406 | } | |
670 | 407 |
github/deemru/w8io/169f3d6 83.99 ms ◑![]()