tx · LeAs6EkKcbnkVbEXbQ2oEJU4yZwr5RWcRSaDAC6UdUf 3N1CYgyJqi5s7uHqiAuGnb86PxXAPBEJ2hg: -0.01400000 Waves 2020.12.30 07:49 [1330973] smart account 3N1CYgyJqi5s7uHqiAuGnb86PxXAPBEJ2hg > SELF 0.00000000 Waves
{ "type": 13, "id": "LeAs6EkKcbnkVbEXbQ2oEJU4yZwr5RWcRSaDAC6UdUf", "fee": 1400000, "feeAssetId": null, "timestamp": 1609303793936, "version": 2, "chainId": 84, "sender": "3N1CYgyJqi5s7uHqiAuGnb86PxXAPBEJ2hg", "senderPublicKey": "75wzrSTzcF5qMYeuwuLWeziy4CgffobpM7wFcCFcCTkU", "proofs": [ "5oSTqYU3yVoWroWTojirawYXqHZR79GCRVLRs3VRLYMiJqA5DMKJ12DVj9xNzemx5vQyp6dtvpFn2LbaPaPkxfXP" ], "script": "base64:AAIEAAAAAAAAAEEIAhIOCgwICAgICAgBCAgICAgSDQoLCAgICAgBCAgICAgSBQoDCAgIEgQKAggIEgYKBAgBAQgSBAoCCAgSAwoBCAAAADgAAAAACHVuaXRUZXN0BwAAAAAJb3JhY2xlRmVlCQEAAAAFdmFsdWUAAAABCQAEJgAAAAECAAAAIzNOMnM1UnRhSFBCZW5Dc3gyRUNjb0ZSYllIeDNub1poWFcxAAAAAAhzaWduRGFwcAkBAAAABXZhbHVlAAAAAQkABCYAAAABAgAAACMzTkMyOGhTaXZybXNUVVhhWUQxeDZMMzYySjRacFVub1RkQgAAAAALZmVlUmVjZWl2ZXICAAAAIzNOMUU2dFhkZFJvVmFSZlE5ZFEzdmc1TGFXMmZzZDhIS3ViAAAAAAh1c2VyRGFwcAMFAAAACHVuaXRUZXN0CQEAAAAFdmFsdWUAAAABCQAEJgAAAAECAAAAIzNNdDRSR01FeWpTV1lBb2NUUHFkNTV3ZEhRUUIzUFUyVUNtCQEAAAAFdmFsdWUAAAABCQAEJgAAAAECAAAAIzNOOHhYYVlqRTI3QWE3OWQ1aEhyaEh1OUhhRm9UQm1oREVqAAAAAA13aGl0ZWxpc3REYXBwCQEAAAAFdmFsdWUAAAABCQAEJgAAAAECAAAAIzNNdDZqajFXVVlxZDdEVDdkSEV1YnM5akVUTFVYTWNBQjhrAAAAAAtzaWduQXNzZXRJZAEAAAAg6KVqvMp3QvJwYTI1Sk9Fg7m5HuWZZxfDcerZC6EEresAAAAAC3VzZG5Bc3NldElkAQAAACAP8hwSrOTEPJrsRhqrJaiw7LoHK0bMbYhy8LXikkkBtAAAAAAMd2F2ZXNBc3NldElkAQAAAAAAAAAABWNocmlzAgAAACMzTXNHNmpQTkNyVkpVdFlCN1hKQnhTN3V0V3NYQWY0bjlWcAAAAAAEam9lcAIAAAAjM016bTRWTHdzTjl1WndiVE16UGozWHV4VjZrRWZBUjhVRE4AAAAAB3NpZ25DdXQAAAAAAAAAAAcAAAAAB3VzZG5DdXQAAAAAAAAAAAoAAAAACHdhdmVzQ3V0AAAAAAAAAAAKAAAAAA5kZWZhdWx0TGljZW5jZQIAAABHYmFmeWJlaWdpc2ZxdHlvMnFkZnNjZWg1ZnBjcDdlZXltcnBlZ2xwNmVkYW8yYm15aGlqYmF5dmF1c3kvbGljZW5jZS5wZGYAAAAAEmRlZmF1bHRIYXNoTGljZW5jZQIAAABAM2RmNzlkMzRhYmJjYTk5MzA4ZTc5Y2I5NDQ2MWMxODkzNTgyNjA0ZDY4MzI5YTQxZmQ0YmVjMTg4NWU2YWRiNAAAAAALZGFwcFJ1bm5pbmcEAAAAByRtYXRjaDAJAAQbAAAAAgUAAAAIdXNlckRhcHACAAAAFGNvbmZfZGFwcF9pc19ydW5uaW5nAwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAAdCb29sZWFuBAAAAAFhBQAAAAckbWF0Y2gwBQAAAAFhBgAAAAAObWFpbnRlbmFuY2VNU0cEAAAAByRtYXRjaDAJAAQdAAAAAgUAAAAIdXNlckRhcHACAAAAFGNvbmZfbWFpbnRlbmFuY2VfbXNnAwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAAZTdHJpbmcEAAAAAWEFAAAAByRtYXRjaDAFAAAAAWECAAAAAAAAAAANdXNlclN1c3BlbmRlZAIAAAAJU1VTUEVOREVEAAAAAAt1c2VyUmVtb3ZlZAIAAAAHUkVNT1ZFRAAAAAAQdXNlclVucmVnaXN0ZXJlZAIAAAAMVU5SRUdJU1RFUkVEAAAAAAt1c2VyQWxsb3dlZAIAAAAHQUxMT1dFRAEAAAAXZ2V0U3RyaW5nQnlLZXlGcm9tVXNlcnMAAAABAAAAA2tleQQAAAAHJG1hdGNoMAkABB0AAAACBQAAAAh1c2VyRGFwcAUAAAADa2V5AwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAAZTdHJpbmcEAAAAAWEFAAAAByRtYXRjaDAFAAAAAWECAAAAAAEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABAAAAA2tleQQAAAAHJG1hdGNoMAkABB0AAAACBQAAAAR0aGlzBQAAAANrZXkDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAABlN0cmluZwQAAAABYQUAAAAHJG1hdGNoMAUAAAABYQIAAAAAAQAAABlnZXRJbnRlZ2VyQnlLZXlGcm9tT3JhY2xlAAAAAQAAAANrZXkEAAAAByRtYXRjaDAJAAQaAAAAAgUAAAAJb3JhY2xlRmVlBQAAAANrZXkDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAAA0ludAQAAAABYQUA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AAAABmNhbGxlcgAAAAVieXRlcwQAAAAJdGltZXN0YW1wCAUAAAAJbGFzdEJsb2NrAAAACXRpbWVzdGFtcAQAAAAKbGljZW5jZUNJRAMJAAAAAAAAAgkAATEAAAABBQAAAApjaWRMaWNlbmNlAAAAAAAAAAAABQAAAA5kZWZhdWx0TGljZW5jZQUAAAAKY2lkTGljZW5jZQQAAAALbGljZW5jZUhhc2gDCQAAAAAAAAIJAAExAAAAAQUAAAANc2hhMjU2TGljZW5jZQAAAAAAAAAAAAUAAAASZGVmYXVsdEhhc2hMaWNlbmNlBQAAAA1zaGEyNTZMaWNlbmNlBAAAABB1c2VySXNSZWdpc3RlcmVkBAAAAAckbWF0Y2gwCQAEHQAAAAIFAAAACHVzZXJEYXBwCQABLAAAAAICAAAADHVzZXJfc3RhdHVzXwUAAAAGY2FsbGVyAwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAAZTdHJpbmcEAAAAAXMFAAAAByRtYXRjaDAFAAAAAXMFAAAAEHVzZXJVbnJlZ2lzdGVyZWQDAwkBAAAACWlzRGVmaW5lZAAAAAEFAAAAEHVzZXJJc1JlZ2lzdGVyZWQDCQAAAAAAAAIFAAAAEHVzZXJJc1JlZ2lzdGVyZWQFAAAAEHVzZXJVbnJlZ2lzdGVyZWQGCQAAAAAAAAIFAAAAEHVzZXJJc1JlZ2lzdGVyZWQFAAAAC3VzZXJBbGxvd2VkBwkAAAIAAAABAgAAAC5SZWdpc3RlciB0aGlzIGFjY291bnQgZmlyc3Qgd2l0aCAiQWNjb3VudCIgdGFiAwkAAAAAAAACBQAAABB1c2VySXNSZWdpc3RlcmVkBQAAAA11c2VyU3VzcGVuZGVkCQAAAgAAAAECAAAAEUFjY291bnQgc3VzcGVuZGVkAwkAAAAAAAACBQAAABB1c2VySXNSZWdpc3RlcmVkBQAAAAt1c2VyUmVtb3ZlZAkAAAIAAAABAgAAAA9BY2NvdW50IHJlbW92ZWQDCQEAAAABIQAAAAEJAQAAAA52YWxpZGF0ZUFsbENJRAAAAAMFAAAACmNpZERpc3BsYXkFAAAACWNpZEV4cG9ydAUAAAAKbGljZW5jZUNJRAkAAAIAAAABAgAAABBQcm9ibGVtIHdpdGggQ0lEAwkBAAAAASEAAAABCQEAAAAMdmFsaWRhdGVIYXNoAAAAAQUAAAAKc2hhMjU2SGFzaAkAAAIAAAABAgAAACRIYXNoIHNob3VsZCBiZSA2NCBjaGFyYWN0ZXJzIG1heGltdW0DCQEAAAABIQAAAAEJAQAAAA92YWxpZGF0ZUFsbEhhc2gAAAACBQAAAAxzaGEyNTZFeHBvcnQFAAAAC2xpY2VuY2VIYXNoCQAAAgAAAAECAAAAE1Byb2JsZW0gd2l0aCBIYXNoZXMDCQAAAAAAAAIJAAGQAAAAAQgFAAAABmludm9rZQAAAAhwYXltZW50cwAAAAAAAAAAAAkAAAIAAAABAgAAABNObyBwYXltZW50IGF0dGFjaGVkBAAAAAdwYXltZW50CQEAAAAFdmFsdWUAAAABCQABkQAAAAIIBQAAAAZpbnZva2UAAAAIcGF5bWVudHMAAAAAAAAAAAAEAAAABmFtb3VudAkBAAAABXZhbHVlAAAAAQgFAAAAB3BheW1lbnQAAAAGYW1vdW50BAAAAAdhc3NldElkAwMJAQAAAAlpc0RlZmluZWQAAAABCAUAAAAHcGF5bWVudAAAAAdhc3NldElkCQAAAAAAAAIIBQAAAAdwYXltZW50AAAAB2Fzc2V0SWQFAAAAC3NpZ25Bc3NldElkBwgFAAAAB3BheW1lbnQAAAAHYXNzZXRJZAkAAAIAAAABAgAAABhPbmx5IFNJR04gdG9rZW4gYWNjZXB0ZWQEAAAAGWN1cnJlbnRDZXJ0aWZpY2F0aW9uUHJpY2UJAQAAABlnZXRJbnRlZ2VyQnlLZXlGcm9tT3JhY2xlAAAAAQkAASwAAAACAgAAABJjZXJ0aWZpY2F0aW9uX2ZlZV8JAAJYAAAAAQUAAAALc2lnbkFzc2V0SWQDCQEAAAACIT0AAAACBQAAAAZhbW91bnQFAAAAGWN1cnJlbnRDZXJ0aWZpY2F0aW9uUHJpY2UJAAACAAAAAQkAASwAAAACAgAAABlQYXltZW50IGFtb3VudCBzaG91bGQgYmUgCQABpAAAAAEFAAAAGWN1cnJlbnRDZXJ0aWZpY2F0aW9uUHJpY2UEAAAACmVudHJ5RXhpc3QJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEJAQAAABVrZXlBcnRUeGlkQnlIYXNoT3duZXIAAAACBQAAAApzaGEyNTZIYXNoBQAAAAZjYWxsZXIDCQEAAAACIT0AAAACBQAAAAplbnRyeUV4aXN0AgAAAAAJAAACAAAAAQIAAAAUWW91IGFscmVhZHkgYWRkZWQgaXQEAAAACWhhc2hFeGlzdAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQkBAAAAEWtleUFydE93bmVyQnlIYXNoAAAAAQUAAAAKc2hhMjU2SGFzaAMJAQAAAAIhPQAAAAIFAAAACWhhc2hFeGlzdAIAAAAACQAAAgAAAAECAAAAF0hhc2ggYWxyZWFkeSByZWdpc3RlcmVkBAAAAA9pc1NpZ25DZXJ0aWZpZWQJAQAAABRjaGVja1NpZ25DZXJ0aWZpY2F0ZQAAAAMFAAAABnNpZ25JRAUAAAAGY2FsbGVyBQAAAApzaGEyNTZIYXNoAwkBAAAAASEAAAABBQAAAA9pc1NpZ25DZXJ0aWZpZWQJAAACAAAAAQIAAAAsU2lnbiBDZXJ0aWZpY2F0ZSBub3QgZm91bmQgZm9yIHRoaXMgYWRkcmVzcy4DCQAAAAAAAAIJAAExAAAAAQUAAAAKY2lkRGlzcGxheQAAAAAAAAAAAAkAAAIAAAABAgAAABtEaXNwbGF5IENJRCBjYW5ub3QgYmUgZW1wdHkDCQEAAAABIQAAAAEJAQAAAA52YWxpZGF0ZVN0cmluZwAAAAIFAAAABG5hbWUAAAAAAAAAAGQJAAACAAAAAQIAAAASMTAwIENoYXIuIG1heCBuYW1lAwkBAAAAASEAAAABCQEAAAAOdmFsaWRhdGVTdHJpbmcAAAACBQAAAAtkZXNjcmlwdGlvbgAAAAAAAAAD6AkAAAIAAAABAgAAABoxMDAwIENoYXIuIG1heCBkZXNjcmlwdGlvbgMJAABmAAAAAgkAAZAAAAABCQAEtQAAAAIFAAAABHRhZ3MCAAAAASwAAAAAAAAAAAUJAAACAAAAAQIAAAALNSB0YWdzIG1heC4DCQAAZgAAAAIFAAAAB21heG1pbnQAAAAAAAAAAAoJAAACAAAAAQIAAAAPMTAgZWRpdGlvbnMgbWF4CQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAABFrZXlBcnRPd25lckJ5SGFzaAAAAAEFAAAACnNoYTI1Nkhhc2gFAAAABmNhbGxlcgkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACCQEAAAAVa2V5QXJ0VHhpZEJ5SGFzaE93bmVyAAAAAgUAAAAKc2hhMjU2SGFzaAUAAAAGY2FsbGVyBQAAAAVhcnRJZAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkBAAAACmtleUFydERhdGUAAAACBQAAAAZjYWxsZXIFAAAABWFydElkBQAAAAl0aW1lc3RhbXAJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAACmtleUFydE5hbWUAAAACBQAAAAZjYWxsZXIFAAAABWFydElkBQAAAARuYW1lCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAAAprZXlBcnREZXNjAAAAAgUAAAAGY2FsbGVyBQAAAAVhcnRJZAUAAAALZGVzY3JpcHRpb24JAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAAEGtleUFydERpc3BsYXlDaWQAAAACBQAAAAZjYWxsZXIFAAAABWFydElkBQAAAApjaWREaXNwbGF5CQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAAA9rZXlBcnRFeHBvcnRDaWQAAAACBQAAAAZjYWxsZXIFAAAABWFydElkBQAAAAljaWRFeHBvcnQJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAAEGtleUFydEV4cG9ydEhhc2gAAAACBQAAAAZjYWxsZXIFAAAABWFydElkBQAAAAxzaGEyNTZFeHBvcnQJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAAEWtleUFydExpY2VuY2VIYXNoAAAAAgUAAAAGY2FsbGVyBQAAAAVhcnRJZAUAAAALbGljZW5jZUhhc2gJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAAEGtleUFydExpY2VuY2VDaWQAAAACBQAAAAZjYWxsZXIFAAAABWFydElkBQAAAApsaWNlbmNlQ0lECQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAAAprZXlBcnRUeXBlAAAAAgUAAAAGY2FsbGVyBQAAAAVhcnRJZAUAAAAEdHlwZQkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACCQEAAAAKa2V5QXJ0VGFncwAAAAIFAAAABmNhbGxlcgUAAAAFYXJ0SWQFAAAABHRhZ3MJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIJAQAAAA1rZXlBcnRNYXhNaW50AAAAAgUAAAAGY2FsbGVyBQAAAAVhcnRJZAUAAAAHbWF4bWludAkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACCQEAAAAMa2V5QXJ0U2lnbklEAAAAAgUAAAAGY2FsbGVyBQAAAAVhcnRJZAUAAAAGc2lnbklECQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAMa2V5QXJ0SXNzdWVkAAAAAgUAAAAGY2FsbGVyBQAAAAVhcnRJZAAAAAAAAAAAAAkABEwAAAACCQEAAAAMQm9vbGVhbkVudHJ5AAAAAgkBAAAADGtleUFydE9uU2FsZQAAAAIFAAAABmNhbGxlcgUAAAAFYXJ0SWQHCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAABNrZXlBcnRBcnRpZEJ5U2lnbmlkAAAAAgUAAAAGY2FsbGVyBQAAAAZzaWduSUQFAAAABWFydElkCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAABRrZXlBcnRIYXNoQnlUeGlkQWRkcgAAAAIFAAAABmNhbGxlcgUAAAAFYXJ0SWQFAAAACnNoYTI1Nkhhc2gJAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwkBAAAAB0FkZHJlc3MAAAABCQACWQAAAAEFAAAAC2ZlZVJlY2VpdmVyBQAAAAZhbW91bnQFAAAAB2Fzc2V0SWQFAAAAA25pbAAAAAZpbnZva2UBAAAADXVwZGF0ZUFydHdvcmsAAAALAAAABHR4aWQAAAAEbmFtZQAAAAtkZXNjcmlwdGlvbgAAAAR0YWdzAAAABHR5cGUAAAAHbWF4bWludAAAAApjaWREaXNwbGF5AAAADHNoYTI1NkV4cG9ydAAAAAljaWRFeHBvcnQAAAANc2hhMjU2TGljZW5jZQAAAApjaWRMaWNlbmNlAwkBAAAAASEAAAABBQAAAAtkYXBwUnVubmluZwkAAAIAAAABBQAAAA5tYWludGVuYW5jZU1TRwQAAAAIdXBkYXRlSWQJAAJYAAAAAQgFAAAABmludm9rZQAAAA10cmFuc2FjdGlvbklkBAAAAAZjYWxsZXIJAAJYAAAAAQgIBQAAAAZpbnZva2UAAAAGY2FsbGVyAAAABWJ5dGVzBAAAAApsaWNlbmNlQ0lEAwkAAAAAAAACCQABMQAAAAEFAAAACmNpZExpY2VuY2UAAAAAAAAAAAAFAAAADmRlZmF1bHRMaWNlbmNlBQAAAApjaWRMaWNlbmNlBAAAAAtsaWNlbmNlSGFzaAMJAAAAAAAAAgkAATEAAAABBQAAAA1zaGEyNTZMaWNlbmNlAAAAAAAAAAAABQAAABJkZWZhdWx0SGFzaExpY2VuY2UFAAAADXNoYTI1NkxpY2VuY2UEAAAAEHVzZXJJc1JlZ2lzdGVyZWQJAQAAABdnZXRTdHJpbmdCeUtleUZyb21Vc2VycwAAAAEJAQAAAA1rZXlVc2VyU3RhdHVzAAAAAQUAAAAGY2FsbGVyAwMJAAAAAAAAAgUAAAAQdXNlcklzUmVnaXN0ZXJlZAIAAAAABgkAAAAAAAACBQAAABB1c2VySXNSZWdpc3RlcmVkBQAAAAt1c2VyQWxsb3dlZAkAAAIAAAABAgAAACBSZWdpc3RlciBmaXJzdCB3aXRoICJVc2VyIGluZm9zIgMJAAAAAAAAAgUAAAAQdXNlcklzUmVnaXN0ZXJlZAUAAAANdXNlclN1c3BlbmRlZAkAAAIAAAABAgAAABFBY2NvdW50IHN1c3BlbmRlZAMJAAAAAAAAAgUAAAAQdXNlcklzUmVnaXN0ZXJlZAUAAAALdXNlclJlbW92ZWQJAAACAAAAAQIAAAAPQWNjb3VudCByZW1vdmVkAwkBAAAAASEAAAABCQEAAAAOdmFsaWRhdGVBbGxDSUQAAAADBQAAAApjaWREaXNwbGF5BQAAAAljaWRFeHBvcnQFAAAACmxpY2VuY2VDSUQJAAACAAAAAQIAAAAQUHJvYmxlbSB3aXRoIENJRAMJAQAAAAEhAAAAAQkBAAAAD3ZhbGlkYXRlQWxsSGFzaAAAAAIFAAAADHNoYTI1NkV4cG9ydAUAAAALbGljZW5jZUhhc2gJAAACAAAAAQIAAAATUHJvYmxlbSB3aXRoIEhhc2hlcwQAAAAKZW50cnlFeGlzdAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQkBAAAACmtleUFydE5hbWUAAAACBQAAAAZjYWxs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Zm9yIHRoaXMgYWRkcmVzcwQAAAAGc2lnbklECQEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABCQEAAAAMa2V5QXJ0U2lnbklEAAAAAgUAAAAMYWRkcmVzc1RvVXNlBQAAAAVhcnRJZAMJAAAAAAAAAgUAAAAGc2lnbklEAgAAAAAJAAACAAAAAQIAAAATTm8gU0lHTiBJRCBtYXRjaGluZwQAAAAMZGF0YVRvRGVsZXRlCQAETAAAAAIJAQAAAAtEZWxldGVFbnRyeQAAAAEJAQAAAAprZXlBcnREYXRlAAAAAgUAAAAMYWRkcmVzc1RvVXNlBQAAAAVhcnRJZAkABEwAAAACCQEAAAALRGVsZXRlRW50cnkAAAABCQEAAAAKa2V5QXJ0TmFtZQAAAAIFAAAADGFkZHJlc3NUb1VzZQUAAAAFYXJ0SWQJAARMAAAAAgkBAAAAC0RlbGV0ZUVudHJ5AAAAAQkBAAAACmtleUFydERlc2MAAAACBQAAAAxhZGRyZXNzVG9Vc2UFAAAABWFydElkCQAETAAAAAIJAQAAAAtEZWxldGVFbnRyeQAAAAEJAQAAABBrZXlBcnREaXNwbGF5Q2lkAAAAAgUAAAAMYWRkcmVzc1RvVXNlBQAAAAVhcnRJZAkABEwAAAACCQEAAAALRGVsZXRlRW50cnkAAAABCQEAAAAPa2V5QXJ0RXhwb3J0Q2lkAAAAAgUAAAAMYWRkcmVzc1RvVXNlBQAAAAVhcnRJZAkABEwAAAACCQEAAAALRGVsZXRlRW50cnkAAAABCQEAAAAQa2V5QXJ0RXhwb3J0SGFzaAAAAAIFAAAADGFkZHJlc3NUb1VzZQUAAAAFYXJ0SWQJAARMAAAAAgkBAAAAC0RlbGV0ZUVudHJ5AAAAAQkBAAAAEWtleUFydExpY2VuY2VIYXNoAAAAAgUAAAAMYWRkcmVzc1RvVXNlBQAAAAVhcnRJZAkABEwAAAACCQEAAAALRGVsZXRlRW50cnkAAAABCQEAAAAQa2V5QXJ0TGljZW5jZUNpZAAAAAIFAAAADGFkZHJlc3NUb1VzZQUAAAAFYXJ0SWQJAARMAAAAAgkBAAAAC0RlbGV0ZUVudHJ5AAAAAQkBAAAACmtleUFydFR5cGUAAAACBQAAAAxhZGRyZXNzVG9Vc2UFAAAABWFydElkCQAETAAAAAIJAQAA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dXMJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIJAQAAAAtrZXlBcnRQcmljZQAAAAIFAAAABmNhbGxlcgUAAAAFYXJ0SWQFAAAABXByaWNlCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAANa2V5QXJ0TWF4TWludAAAAAIFAAAABmNhbGxlcgUAAAAFYXJ0SWQFAAAAB21heE1pbnQJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAAFWtleUFydEFzc2V0SWRBY2NlcHRlZAAAAAIFAAAABmNhbGxlcgUAAAAFYXJ0SWQFAAAAB2Fzc2V0SWQFAAAAA25pbAAAAAFpAQAAAApidXlBcnR3b3JrAAAAAgAAAAVhcnRJZAAAAAZpc3N1ZXIDCQEAAAABIQAAAAEFAAAAC2RhcHBSdW5uaW5nCQAAAgAAAAEFAAAADm1haW50ZW5hbmNlTVNHBAAAAAJpZAkAAlgAAAABCAUAAAABaQAAAA10cmFuc2FjdGlvbklkBAAAAAZjYWxsZXIJAAJYAAAAAQgIBQAAAAFpAAAABmNhbGxlcgAAAAVieXRlcwQAAAAIdG90YWxORlQJAQAAAA9nZXRJbnRlZ2VyQnlLZXkAAAABAgAAABB0b3RhbF9uZnRfaXNzdWVkBAAAAAZzaWduSUQJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEJAQAAAAxrZXlBcnRTaWduSUQAAAACBQAAAAZpc3N1ZXIFAAAABWFydElkBAAAAAthcnR3b3JrTmFtZQkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQkBAAAACmtleUFydE5hbWUAAAACBQAAAAZpc3N1ZXIFAAAABWFydElkAwkAAAAAAAACBQAAAAthcnR3b3JrTmFtZQIAAAAACQAAAgAAAAECAAAAEUFydCBkb2Vzbid0IGV4aXN0BAAAAApkaXNwbGF5Q0lECQEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABCQEAAAAQa2V5QXJ0RGlzcGxheUNpZAAAAAIFAAAABmlzc3VlcgUAAAAFYXJ0SWQEAAAACWV4cG9ydENJRAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQkBAAAAD2tleUFydEV4cG9ydENpZAAAAAIFAAAABmlzc3VlcgUAAAAFYXJ0SWQEAAAACmV4cG9ydEhhc2gJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEJAQAAABBrZXlBcnRFeHBvcnRIYXNoAAAAAgUAAAAGaXNzdWVyBQAAAAVhcnRJZAQAAAAKbGljZW5jZUNJRAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQkBAAAAEGtleUFydExpY2VuY2VDaWQAAAACBQAAAAZpc3N1ZXIFAAAABWFydElkBAAAAAtsaWNlbmNlSGFzaAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQkBAAAAEWtleUFydExpY2VuY2VIYXNoAAAAAgUAAAAGaXNzdWVyBQAAAAVhcnRJZAQAAAALZGVzY3JpcHRpb24JAAEvAAAAAgkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQkBAAAACmtleUFydERlc2MAAAACBQAAAAZpc3N1ZXIFAAAABWFydElkAAAAAAAAAAAyBAAAAAphbW91bnRTb2xkCQEAAAAPZ2V0SW50ZWdlckJ5S2V5AAAAAQkBAAAADGtleUFydElzc3VlZAAAAAIFAAAABmlzc3VlcgUAAAAFYXJ0SWQEAAAADGFydHdvcmtQcmljZQkBAAAAD2dldEludGVnZXJCeUtleQAAAAEJAQAAAAtrZXlBcnRQcmljZQAAAAIFAAAABmlzc3VlcgUAAAAFYXJ0SWQEAAAACGlzT25TYWxlCQEAAAAPZ2V0Qm9vbGVhbkJ5S2V5AAAAAQkBAAAADGtleUFydE9uU2FsZQAAAAIFAAAABmlzc3VlcgUAAAAFYXJ0SWQEAAAADHByaWNlQXNzZXRJZAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQkBAAAAFWtleUFydEFzc2V0SWRBY2NlcHRlZAAAAAIFAAAABmlzc3VlcgUAAAAFYXJ0SWQEAAAACnNvdXJjZUhhc2gJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEJAQAAABRrZXlBcnRIYXNoQnlUeGlkQWRkcgAAAAIFAAAABmlzc3VlcgUAAAAFYXJ0SWQEAAAACm1heENhblNlbGwJAQAAAA9nZXRJbnRlZ2VyQnlLZXkAAAABCQEAAAANa2V5QXJ0TWF4TWludAAAAAIFAAAABmlzc3VlcgUAAAAFYXJ0SWQDAwMJAABnAAAAAgAAAAAAAAAAAAUAAAAM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", "height": 1330973, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 7H2Yog2zo6FHjSd5mfaetsHXP9FSqYQaniPVvayKEySK Next: HM8FCLojvoP8XzLNo4w3FuMCGUFfgGgrBXug1yqj8K8P Diff:
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 4 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | - | let userDapp = value(addressFromString("3MzVhsmmU8n1axekFJk1BRNwrmnCMH6Mn8F")) | |
4 | + | let unitTest = false | |
5 | + | ||
6 | + | let oracleFee = value(addressFromString("3N2s5RtaHPBenCsx2ECcoFRbYHx3noZhXW1")) | |
7 | + | ||
8 | + | let signDapp = value(addressFromString("3NC28hSivrmsTUXaYD1x6L362J4ZpUnoTdB")) | |
9 | + | ||
10 | + | let feeReceiver = "3N1E6tXddRoVaRfQ9dQ3vg5LaW2fsd8HKub" | |
11 | + | ||
12 | + | let userDapp = if (unitTest) | |
13 | + | then value(addressFromString("3Mt4RGMEyjSWYAocTPqd55wdHQQB3PU2UCm")) | |
14 | + | else value(addressFromString("3N8xXaYjE27Aa79d5hHrhHu9HaFoTBmhDEj")) | |
15 | + | ||
16 | + | let whitelistDapp = value(addressFromString("3Mt6jj1WUYqd7DT7dHEubs9jETLUXMcAB8k")) | |
17 | + | ||
18 | + | let signAssetId = base58'Gf9t8FA4H3ssoZPCwrg3KwUFCci8zuUFP9ssRsUY3s6a' | |
19 | + | ||
20 | + | let usdnAssetId = base58'25FEqEjRkqK6yCkiT7Lz6SAYz7gUFCtxfCChnrVFD5AT' | |
21 | + | ||
22 | + | let wavesAssetId = base58'' | |
23 | + | ||
24 | + | let chris = "3MsG6jPNCrVJUtYB7XJBxS7utWsXAf4n9Vp" | |
25 | + | ||
26 | + | let joep = "3Mzm4VLwsN9uZwbTMzPj3XuxV6kEfAR8UDN" | |
27 | + | ||
28 | + | let signCut = 7 | |
29 | + | ||
30 | + | let usdnCut = 10 | |
31 | + | ||
32 | + | let wavesCut = 10 | |
5 | 33 | ||
6 | 34 | let defaultLicence = "bafybeigisfqtyo2qdfsceh5fpcp7eeymrpeglp6edao2bmyhijbayvausy/licence.pdf" | |
7 | 35 | ||
8 | 36 | let defaultHashLicence = "3df79d34abbca99308e79cb94461c1893582604d68329a41fd4bec1885e6adb4" | |
37 | + | ||
38 | + | let dappRunning = match getBoolean(userDapp, "conf_dapp_is_running") { | |
39 | + | case a: Boolean => | |
40 | + | a | |
41 | + | case _ => | |
42 | + | true | |
43 | + | } | |
44 | + | ||
45 | + | let maintenanceMSG = match getString(userDapp, "conf_maintenance_msg") { | |
46 | + | case a: String => | |
47 | + | a | |
48 | + | case _ => | |
49 | + | "" | |
50 | + | } | |
51 | + | ||
52 | + | let userSuspended = "SUSPENDED" | |
53 | + | ||
54 | + | let userRemoved = "REMOVED" | |
55 | + | ||
56 | + | let userUnregistered = "UNREGISTERED" | |
57 | + | ||
58 | + | let userAllowed = "ALLOWED" | |
59 | + | ||
60 | + | func getStringByKeyFromUsers (key) = match getString(userDapp, key) { | |
61 | + | case a: String => | |
62 | + | a | |
63 | + | case _ => | |
64 | + | "" | |
65 | + | } | |
66 | + | ||
9 | 67 | ||
10 | 68 | func getStringByKey (key) = match getString(this, key) { | |
11 | 69 | case a: String => | |
13 | 71 | case _ => | |
14 | 72 | "" | |
15 | 73 | } | |
74 | + | ||
75 | + | ||
76 | + | func getIntegerByKeyFromOracle (key) = match getInteger(oracleFee, key) { | |
77 | + | case a: Int => | |
78 | + | a | |
79 | + | case _ => | |
80 | + | throw("Integer undefine or 0 in oracle") | |
81 | + | } | |
82 | + | ||
83 | + | ||
84 | + | func getIntegerByKey (key) = match getInteger(this, key) { | |
85 | + | case i: Int => | |
86 | + | i | |
87 | + | case _ => | |
88 | + | 0 | |
89 | + | } | |
90 | + | ||
91 | + | ||
92 | + | func checkWhitelist (key) = match getInteger(whitelistDapp, key) { | |
93 | + | case a: Int => | |
94 | + | if ((a >= height)) | |
95 | + | then 1 | |
96 | + | else 0 | |
97 | + | case _ => | |
98 | + | 0 | |
99 | + | } | |
100 | + | ||
101 | + | ||
102 | + | func getBooleanByKey (key) = match getBoolean(this, key) { | |
103 | + | case i: Boolean => | |
104 | + | i | |
105 | + | case _ => | |
106 | + | false | |
107 | + | } | |
108 | + | ||
109 | + | ||
110 | + | func checkSignCertificate (signID,Owner,sha256Hash) = match getString(signDapp, ((("data_fc_" + signID) + "_") + Owner)) { | |
111 | + | case a: String => | |
112 | + | if (contains(a, sha256Hash)) | |
113 | + | then true | |
114 | + | else false | |
115 | + | case _ => | |
116 | + | false | |
117 | + | } | |
118 | + | ||
119 | + | ||
120 | + | func validateCID (cid) = if (contains(cid, "/")) | |
121 | + | then if (if ((75 > size(cid))) | |
122 | + | then (60 > size(split(cid, "/")[0])) | |
123 | + | else false) | |
124 | + | then (16 > size(split(cid, "/")[1])) | |
125 | + | else false | |
126 | + | else false | |
127 | + | ||
128 | + | ||
129 | + | func validateHash (hash) = (65 > size(hash)) | |
16 | 130 | ||
17 | 131 | ||
18 | 132 | func keyUserStatus (caller) = ("user_status_" + caller) | |
81 | 195 | func keyArtTxidByHashOwner (sha256Hash,caller) = ("get_txid_by_hash_owner_" + toBase58String(sha256_16Kb(toBytes((sha256Hash + caller))))) | |
82 | 196 | ||
83 | 197 | ||
198 | + | func validateAllCID (cidDisplay,cidExport,cidLicence) = if (!(validateCID(cidDisplay))) | |
199 | + | then throw("Wrong Display CID") | |
200 | + | else if (!(validateCID(cidExport))) | |
201 | + | then throw("Wrong Export CID") | |
202 | + | else if (if ((cidLicence != "")) | |
203 | + | then !(validateCID(cidLicence)) | |
204 | + | else false) | |
205 | + | then throw("Wrong Licence CID") | |
206 | + | else true | |
207 | + | ||
208 | + | ||
209 | + | func validateAllHash (sha256Export,sha256Licence) = if (!(validateHash(sha256Export))) | |
210 | + | then throw("Export Hash 64 char. max") | |
211 | + | else if (!(validateHash(sha256Licence))) | |
212 | + | then throw("Licence Hash 64 char. max") | |
213 | + | else true | |
214 | + | ||
215 | + | ||
216 | + | func validateString (str,max) = if ((size(str) == 0)) | |
217 | + | then throw("Field cannot be is empty") | |
218 | + | else if ((size(str) > max)) | |
219 | + | then throw((str + " is too long")) | |
220 | + | else true | |
221 | + | ||
222 | + | ||
84 | 223 | @Callable(invoke) | |
85 | - | func addArtwork (sha256Hash,signID,name,description,tags,type,maxmint,cidDisplay,sha256Export,cidExport,sha256Licence,cidLicence) = { | |
86 | - | let artId = toBase58String(invoke.transactionId) | |
87 | - | let caller = toBase58String(invoke.caller.bytes) | |
88 | - | let timestamp = lastBlock.timestamp | |
89 | - | let licenceCID = if ((size(cidLicence) == 0)) | |
90 | - | then defaultLicence | |
91 | - | else cidLicence | |
92 | - | let licenceHash = if ((size(sha256Licence) == 0)) | |
93 | - | then defaultHashLicence | |
94 | - | else sha256Licence | |
95 | - | [StringEntry(keyArtOwnerByHash(sha256Hash), caller), StringEntry(keyArtTxidByHashOwner(sha256Hash, caller), artId), IntegerEntry(keyArtDate(caller, artId), timestamp), StringEntry(keyArtName(caller, artId), name), StringEntry(keyArtDesc(caller, artId), description), StringEntry(keyArtDisplayCid(caller, artId), cidDisplay), StringEntry(keyArtExportCid(caller, artId), cidExport), StringEntry(keyArtExportHash(caller, artId), sha256Export), StringEntry(keyArtLicenceHash(caller, artId), licenceHash), StringEntry(keyArtLicenceCid(caller, artId), licenceCID), StringEntry(keyArtType(caller, artId), type), StringEntry(keyArtTags(caller, artId), tags), IntegerEntry(keyArtMaxMint(caller, artId), maxmint), StringEntry(keyArtSignID(caller, artId), signID), IntegerEntry(keyArtIssued(caller, artId), 0), BooleanEntry(keyArtOnSale(caller, artId), false), StringEntry(keyArtArtidBySignid(caller, signID), artId), StringEntry(keyArtHashByTxidAddr(caller, artId), sha256Hash)] | |
224 | + | func addArtwork (sha256Hash,signID,name,description,tags,type,maxmint,cidDisplay,sha256Export,cidExport,sha256Licence,cidLicence) = if (!(dappRunning)) | |
225 | + | then throw(maintenanceMSG) | |
226 | + | else { | |
227 | + | let artId = toBase58String(invoke.transactionId) | |
228 | + | let caller = toBase58String(invoke.caller.bytes) | |
229 | + | let timestamp = lastBlock.timestamp | |
230 | + | let licenceCID = if ((size(cidLicence) == 0)) | |
231 | + | then defaultLicence | |
232 | + | else cidLicence | |
233 | + | let licenceHash = if ((size(sha256Licence) == 0)) | |
234 | + | then defaultHashLicence | |
235 | + | else sha256Licence | |
236 | + | let userIsRegistered = match getString(userDapp, ("user_status_" + caller)) { | |
237 | + | case s: String => | |
238 | + | s | |
239 | + | case _ => | |
240 | + | userUnregistered | |
241 | + | } | |
242 | + | if (if (isDefined(userIsRegistered)) | |
243 | + | then if ((userIsRegistered == userUnregistered)) | |
244 | + | then true | |
245 | + | else (userIsRegistered == userAllowed) | |
246 | + | else false) | |
247 | + | then throw("Register this account first with \"Account\" tab") | |
248 | + | else if ((userIsRegistered == userSuspended)) | |
249 | + | then throw("Account suspended") | |
250 | + | else if ((userIsRegistered == userRemoved)) | |
251 | + | then throw("Account removed") | |
252 | + | else if (!(validateAllCID(cidDisplay, cidExport, licenceCID))) | |
253 | + | then throw("Problem with CID") | |
254 | + | else if (!(validateHash(sha256Hash))) | |
255 | + | then throw("Hash should be 64 characters maximum") | |
256 | + | else if (!(validateAllHash(sha256Export, licenceHash))) | |
257 | + | then throw("Problem with Hashes") | |
258 | + | else if ((size(invoke.payments) == 0)) | |
259 | + | then throw("No payment attached") | |
260 | + | else { | |
261 | + | let payment = value(invoke.payments[0]) | |
262 | + | let amount = value(payment.amount) | |
263 | + | let assetId = if (if (isDefined(payment.assetId)) | |
264 | + | then (payment.assetId == signAssetId) | |
265 | + | else false) | |
266 | + | then payment.assetId | |
267 | + | else throw("Only SIGN token accepted") | |
268 | + | let currentCertificationPrice = getIntegerByKeyFromOracle(("certification_fee_" + toBase58String(signAssetId))) | |
269 | + | if ((amount != currentCertificationPrice)) | |
270 | + | then throw(("Payment amount should be " + toString(currentCertificationPrice))) | |
271 | + | else { | |
272 | + | let entryExist = getStringByKey(keyArtTxidByHashOwner(sha256Hash, caller)) | |
273 | + | if ((entryExist != "")) | |
274 | + | then throw("You already added it") | |
275 | + | else { | |
276 | + | let hashExist = getStringByKey(keyArtOwnerByHash(sha256Hash)) | |
277 | + | if ((hashExist != "")) | |
278 | + | then throw("Hash already registered") | |
279 | + | else { | |
280 | + | let isSignCertified = checkSignCertificate(signID, caller, sha256Hash) | |
281 | + | if (!(isSignCertified)) | |
282 | + | then throw("Sign Certificate not found for this address.") | |
283 | + | else if ((size(cidDisplay) == 0)) | |
284 | + | then throw("Display CID cannot be empty") | |
285 | + | else if (!(validateString(name, 100))) | |
286 | + | then throw("100 Char. max name") | |
287 | + | else if (!(validateString(description, 1000))) | |
288 | + | then throw("1000 Char. max description") | |
289 | + | else if ((size(split(tags, ",")) > 5)) | |
290 | + | then throw("5 tags max.") | |
291 | + | else if ((maxmint > 10)) | |
292 | + | then throw("10 editions max") | |
293 | + | else [StringEntry(keyArtOwnerByHash(sha256Hash), caller), StringEntry(keyArtTxidByHashOwner(sha256Hash, caller), artId), IntegerEntry(keyArtDate(caller, artId), timestamp), StringEntry(keyArtName(caller, artId), name), StringEntry(keyArtDesc(caller, artId), description), StringEntry(keyArtDisplayCid(caller, artId), cidDisplay), StringEntry(keyArtExportCid(caller, artId), cidExport), StringEntry(keyArtExportHash(caller, artId), sha256Export), StringEntry(keyArtLicenceHash(caller, artId), licenceHash), StringEntry(keyArtLicenceCid(caller, artId), licenceCID), StringEntry(keyArtType(caller, artId), type), StringEntry(keyArtTags(caller, artId), tags), IntegerEntry(keyArtMaxMint(caller, artId), maxmint), StringEntry(keyArtSignID(caller, artId), signID), IntegerEntry(keyArtIssued(caller, artId), 0), BooleanEntry(keyArtOnSale(caller, artId), false), StringEntry(keyArtArtidBySignid(caller, signID), artId), StringEntry(keyArtHashByTxidAddr(caller, artId), sha256Hash), ScriptTransfer(Address(fromBase58String(feeReceiver)), amount, assetId)] | |
294 | + | } | |
295 | + | } | |
296 | + | } | |
297 | + | } | |
298 | + | } | |
299 | + | ||
300 | + | ||
301 | + | ||
302 | + | @Callable(invoke) | |
303 | + | func updateArtwork (txid,name,description,tags,type,maxmint,cidDisplay,sha256Export,cidExport,sha256Licence,cidLicence) = if (!(dappRunning)) | |
304 | + | then throw(maintenanceMSG) | |
305 | + | else { | |
306 | + | let updateId = toBase58String(invoke.transactionId) | |
307 | + | let caller = toBase58String(invoke.caller.bytes) | |
308 | + | let licenceCID = if ((size(cidLicence) == 0)) | |
309 | + | then defaultLicence | |
310 | + | else cidLicence | |
311 | + | let licenceHash = if ((size(sha256Licence) == 0)) | |
312 | + | then defaultHashLicence | |
313 | + | else sha256Licence | |
314 | + | let userIsRegistered = getStringByKeyFromUsers(keyUserStatus(caller)) | |
315 | + | if (if ((userIsRegistered == "")) | |
316 | + | then true | |
317 | + | else (userIsRegistered == userAllowed)) | |
318 | + | then throw("Register first with \"User infos\"") | |
319 | + | else if ((userIsRegistered == userSuspended)) | |
320 | + | then throw("Account suspended") | |
321 | + | else if ((userIsRegistered == userRemoved)) | |
322 | + | then throw("Account removed") | |
323 | + | else if (!(validateAllCID(cidDisplay, cidExport, licenceCID))) | |
324 | + | then throw("Problem with CID") | |
325 | + | else if (!(validateAllHash(sha256Export, licenceHash))) | |
326 | + | then throw("Problem with Hashes") | |
327 | + | else { | |
328 | + | let entryExist = getStringByKey(keyArtName(caller, txid)) | |
329 | + | if ((entryExist == "")) | |
330 | + | then throw("Entry not found") | |
331 | + | else if (!(validateString(name, 100))) | |
332 | + | then throw("100 Char. max name") | |
333 | + | else if (!(validateString(description, 1000))) | |
334 | + | then throw("1000 Char. max description") | |
335 | + | else { | |
336 | + | let flag = getStringByKey(keyArtFlag(caller, txid)) | |
337 | + | if ((flag == "ILLEGAL")) | |
338 | + | then throw("Cannot update ILLEGAL artwork") | |
339 | + | else { | |
340 | + | let artworkMinted = match getInteger(this, keyArtIssued(caller, txid)) { | |
341 | + | case b: Int => | |
342 | + | if ((b == 0)) | |
343 | + | then false | |
344 | + | else true | |
345 | + | case _ => | |
346 | + | false | |
347 | + | } | |
348 | + | if ((size(split(tags, ",")) > 5)) | |
349 | + | then throw("5 tags max.") | |
350 | + | else if ((maxmint > 10)) | |
351 | + | then throw("10 editions max per artwork") | |
352 | + | else if (!(artworkMinted)) | |
353 | + | then [StringEntry(keyArtName(caller, txid), name), StringEntry(keyArtDesc(caller, txid), description), StringEntry(keyArtDisplayCid(caller, txid), cidDisplay), StringEntry(keyArtExportCid(caller, txid), cidExport), StringEntry(keyArtExportHash(caller, txid), sha256Export), StringEntry(keyArtLicenceCid(caller, txid), licenceCID), StringEntry(keyArtLicenceHash(caller, txid), licenceHash), IntegerEntry(keyArtMaxMint(caller, txid), maxmint), StringEntry(keyArtTags(caller, txid), tags), StringEntry(keyArtType(caller, txid), type)] | |
354 | + | else throw("Already minted") | |
355 | + | } | |
356 | + | } | |
357 | + | } | |
358 | + | } | |
359 | + | ||
360 | + | ||
361 | + | ||
362 | + | @Callable(i) | |
363 | + | func flagArtwork (artId,addr,flag) = { | |
364 | + | let caller = toString(addressFromPublicKey(i.callerPublicKey)) | |
365 | + | let id = toBase58String(i.transactionId) | |
366 | + | if (containsElement([chris, joep, toString(this)], caller)) | |
367 | + | then if ((flag == "CONSENT")) | |
368 | + | then [StringEntry(keyArtFlag(addr, artId), flag)] | |
369 | + | else if ((flag == "")) | |
370 | + | then [StringEntry(keyArtFlag(addr, artId), flag)] | |
371 | + | else if ((flag == "ILLEGAL")) | |
372 | + | then [StringEntry(keyArtFlag(addr, artId), flag), StringEntry(keyArtName(addr, artId), "ILLEGAL CONTENT"), StringEntry(keyArtDesc(addr, artId), "ILLEGAL CONTENT"), StringEntry(keyArtDisplayCid(addr, artId), ""), StringEntry(keyArtExportCid(addr, artId), ""), StringEntry(keyArtLicenceCid(addr, artId), "")] | |
373 | + | else throw(("Unknow status " + flag)) | |
374 | + | else throw("You are not allowed") | |
375 | + | } | |
376 | + | ||
377 | + | ||
378 | + | ||
379 | + | @Callable(i) | |
380 | + | func deleteArtwork (artId,addr) = { | |
381 | + | let caller = toString(addressFromPublicKey(i.callerPublicKey)) | |
382 | + | let id = toBase58String(i.transactionId) | |
383 | + | let addressToUse = if (containsElement([chris, joep, toString(this)], caller)) | |
384 | + | then addr | |
385 | + | else caller | |
386 | + | let entryExist = getStringByKey(keyArtName(addressToUse, artId)) | |
387 | + | if ((entryExist == "")) | |
388 | + | then throw("No art matching for this address") | |
389 | + | else { | |
390 | + | let artworkMinted = match getInteger(this, keyArtIssued(addressToUse, artId)) { | |
391 | + | case b: Int => | |
392 | + | if ((b != 0)) | |
393 | + | then true | |
394 | + | else false | |
395 | + | case _ => | |
396 | + | false | |
397 | + | } | |
398 | + | let maxMint = getIntegerByKey(keyArtMaxMint(addressToUse, artId)) | |
399 | + | let sha256Hash = getStringByKey(keyArtHashByTxidAddr(addressToUse, artId)) | |
400 | + | if ((sha256Hash == "")) | |
401 | + | then throw("No art hash matching for this address") | |
402 | + | else { | |
403 | + | let signID = getStringByKey(keyArtSignID(addressToUse, artId)) | |
404 | + | if ((signID == "")) | |
405 | + | then throw("No SIGN ID matching") | |
406 | + | else { | |
407 | + | let dataToDelete = [DeleteEntry(keyArtDate(addressToUse, artId)), DeleteEntry(keyArtName(addressToUse, artId)), DeleteEntry(keyArtDesc(addressToUse, artId)), DeleteEntry(keyArtDisplayCid(addressToUse, artId)), DeleteEntry(keyArtExportCid(addressToUse, artId)), DeleteEntry(keyArtExportHash(addressToUse, artId)), DeleteEntry(keyArtLicenceHash(addressToUse, artId)), DeleteEntry(keyArtLicenceCid(addressToUse, artId)), DeleteEntry(keyArtType(addressToUse, artId)), DeleteEntry(keyArtTags(addressToUse, artId)), DeleteEntry(keyArtMaxMint(addressToUse, artId)), DeleteEntry(keyArtSignID(addressToUse, artId)), DeleteEntry(keyArtIssued(addressToUse, artId)), DeleteEntry(keyArtOnSale(addressToUse, artId)), DeleteEntry(keyArtFlag(addr, artId)), DeleteEntry(keyArtOwnerByHash(sha256Hash)), DeleteEntry(keyArtArtidBySignid(addressToUse, signID)), DeleteEntry(keyArtTxidByHashOwner(sha256Hash, addressToUse))] | |
408 | + | if (!(artworkMinted)) | |
409 | + | then if (!(dappRunning)) | |
410 | + | then throw(maintenanceMSG) | |
411 | + | else dataToDelete | |
412 | + | else if (containsElement([chris, toString(this)], caller)) | |
413 | + | then dataToDelete | |
414 | + | else throw("Art already minted, cannot delete") | |
415 | + | } | |
416 | + | } | |
417 | + | } | |
418 | + | } | |
419 | + | ||
420 | + | ||
421 | + | ||
422 | + | @Callable(i) | |
423 | + | func sellArtwork (artId,price,maxMint,assetId) = if (!(dappRunning)) | |
424 | + | then throw(maintenanceMSG) | |
425 | + | else { | |
426 | + | let id = toBase58String(i.transactionId) | |
427 | + | let caller = toBase58String(i.caller.bytes) | |
428 | + | let sellDate = lastBlock.timestamp | |
429 | + | let artworkName = getStringByKey(keyArtName(caller, artId)) | |
430 | + | if ((artworkName == "")) | |
431 | + | then throw("This art doesn't match match your account") | |
432 | + | else { | |
433 | + | let exportCID = getStringByKey(keyArtExportCid(caller, artId)) | |
434 | + | if ((size(split(exportCID, "/")[0]) != 59)) | |
435 | + | then throw("You cannot sell art with no export file") | |
436 | + | else { | |
437 | + | let exportHash = getStringByKey(keyArtExportHash(caller, artId)) | |
438 | + | if ((size(exportHash) != 64)) | |
439 | + | then throw("You cannot sell art with no export hash") | |
440 | + | else if (if (if ((assetId != toBase58String(signAssetId))) | |
441 | + | then (assetId != toBase58String(wavesAssetId)) | |
442 | + | else false) | |
443 | + | then (assetId != toBase58String(usdnAssetId)) | |
444 | + | else false) | |
445 | + | then throw("Only SIGN, USDN or WAVES accepted") | |
446 | + | else { | |
447 | + | let minSellWaves = if (unitTest) | |
448 | + | then 1 | |
449 | + | else getIntegerByKeyFromOracle("waves_min_sell") | |
450 | + | let minSellUsdn = 1000000 | |
451 | + | let minSellSign = if (unitTest) | |
452 | + | then 1 | |
453 | + | else (getIntegerByKeyFromOracle(("certification_fee_" + toBase58String(signAssetId))) * 2) | |
454 | + | if (if (if (if (if ((assetId == toBase58String(usdnAssetId))) | |
455 | + | then (minSellUsdn > price) | |
456 | + | else false) | |
457 | + | then (price != 0) | |
458 | + | else false) | |
459 | + | then true | |
460 | + | else if (if ((assetId == toBase58String(signAssetId))) | |
461 | + | then (minSellSign > price) | |
462 | + | else false) | |
463 | + | then (price != 0) | |
464 | + | else false) | |
465 | + | then true | |
466 | + | else if (if ((assetId == toBase58String(wavesAssetId))) | |
467 | + | then (minSellWaves > price) | |
468 | + | else false) | |
469 | + | then (price != 0) | |
470 | + | else false) | |
471 | + | then throw("Wrong minimum sell price") | |
472 | + | else { | |
473 | + | let userIsRegistered = getStringByKeyFromUsers(keyUserStatus(caller)) | |
474 | + | if (if ((userIsRegistered == "")) | |
475 | + | then true | |
476 | + | else (userIsRegistered == userAllowed)) | |
477 | + | then throw("Register this account first") | |
478 | + | else if ((userIsRegistered == userSuspended)) | |
479 | + | then throw("Account suspended") | |
480 | + | else if ((userIsRegistered == userRemoved)) | |
481 | + | then throw("Account deleted") | |
482 | + | else { | |
483 | + | let amountSold = getIntegerByKey(keyArtIssued(caller, artId)) | |
484 | + | let maxCanSell = getIntegerByKey(keyArtMaxMint(caller, artId)) | |
485 | + | if ((maxMint > 10)) | |
486 | + | then throw("10 editions max per artwork") | |
487 | + | else if (if ((amountSold != 0)) | |
488 | + | then (amountSold == maxCanSell) | |
489 | + | else false) | |
490 | + | then throw("Max edition reached.") | |
491 | + | else if (if ((amountSold > 0)) | |
492 | + | then (maxCanSell != maxMint) | |
493 | + | else false) | |
494 | + | then throw("Cannot change maximum issuable anymore") | |
495 | + | else { | |
496 | + | let sellStatus = if (if ((price > 0)) | |
497 | + | then (maxMint > 0) | |
498 | + | else false) | |
499 | + | then true | |
500 | + | else false | |
501 | + | [BooleanEntry(keyArtOnSale(caller, artId), sellStatus), IntegerEntry(keyArtPrice(caller, artId), price), IntegerEntry(keyArtMaxMint(caller, artId), maxMint), StringEntry(keyArtAssetIdAccepted(caller, artId), assetId)] | |
502 | + | } | |
503 | + | } | |
504 | + | } | |
505 | + | } | |
506 | + | } | |
507 | + | } | |
508 | + | } | |
509 | + | ||
510 | + | ||
511 | + | ||
512 | + | @Callable(i) | |
513 | + | func buyArtwork (artId,issuer) = if (!(dappRunning)) | |
514 | + | then throw(maintenanceMSG) | |
515 | + | else { | |
516 | + | let id = toBase58String(i.transactionId) | |
517 | + | let caller = toBase58String(i.caller.bytes) | |
518 | + | let totalNFT = getIntegerByKey("total_nft_issued") | |
519 | + | let signID = getStringByKey(keyArtSignID(issuer, artId)) | |
520 | + | let artworkName = getStringByKey(keyArtName(issuer, artId)) | |
521 | + | if ((artworkName == "")) | |
522 | + | then throw("Art doesn't exist") | |
523 | + | else { | |
524 | + | let displayCID = getStringByKey(keyArtDisplayCid(issuer, artId)) | |
525 | + | let exportCID = getStringByKey(keyArtExportCid(issuer, artId)) | |
526 | + | let exportHash = getStringByKey(keyArtExportHash(issuer, artId)) | |
527 | + | let licenceCID = getStringByKey(keyArtLicenceCid(issuer, artId)) | |
528 | + | let licenceHash = getStringByKey(keyArtLicenceHash(issuer, artId)) | |
529 | + | let description = take(getStringByKey(keyArtDesc(issuer, artId)), 50) | |
530 | + | let amountSold = getIntegerByKey(keyArtIssued(issuer, artId)) | |
531 | + | let artworkPrice = getIntegerByKey(keyArtPrice(issuer, artId)) | |
532 | + | let isOnSale = getBooleanByKey(keyArtOnSale(issuer, artId)) | |
533 | + | let priceAssetId = getStringByKey(keyArtAssetIdAccepted(issuer, artId)) | |
534 | + | let sourceHash = getStringByKey(keyArtHashByTxidAddr(issuer, artId)) | |
535 | + | let maxCanSell = getIntegerByKey(keyArtMaxMint(issuer, artId)) | |
536 | + | if (if (if ((0 >= artworkPrice)) | |
537 | + | then true | |
538 | + | else !(isOnSale)) | |
539 | + | then true | |
540 | + | else (0 >= maxCanSell)) | |
541 | + | then throw("Art not for sale") | |
542 | + | else { | |
543 | + | let payment = value(i.payments[0]) | |
544 | + | let amount = value(payment.amount) | |
545 | + | let assetId = if (!(isDefined(payment.assetId))) | |
546 | + | then unit | |
547 | + | else if (if ((size(priceAssetId) > 0)) | |
548 | + | then (toBase58String(value(payment.assetId)) == priceAssetId) | |
549 | + | else false) | |
550 | + | then payment.assetId | |
551 | + | else throw("Wrong asset id") | |
552 | + | if (if ((assetId == unit)) | |
553 | + | then (priceAssetId != "") | |
554 | + | else false) | |
555 | + | then throw("Wrong asset id") | |
556 | + | else { | |
557 | + | let isWhitelisted = checkWhitelist(issuer) | |
558 | + | let cut = if ((isWhitelisted == 1)) | |
559 | + | then 0 | |
560 | + | else if ((priceAssetId == toBase58String(signAssetId))) | |
561 | + | then 8 | |
562 | + | else 10 | |
563 | + | let amountForSign = fraction(amount, cut, 100) | |
564 | + | let amountForCreator = (amount - amountForSign) | |
565 | + | if ((amountSold == maxCanSell)) | |
566 | + | then throw("Art sold out") | |
567 | + | else if ((artworkPrice != amount)) | |
568 | + | then throw("Payment don't match") | |
569 | + | else { | |
570 | + | let newAmountSold = (amountSold + 1) | |
571 | + | let entryDate = lastBlock.timestamp | |
572 | + | let issueMeta = ((((((((((((((((((((((((((("Creator: " + issuer) + ", | |
573 | + | ArtID: ") + artId) + ", | |
574 | + | SignID: ") + signID) + ", | |
575 | + | Artwork name: ") + artworkName) + ", | |
576 | + | Artwork description: ") + description) + ", | |
577 | + | Issue: ") + toString(newAmountSold)) + "/") + toString(maxCanSell)) + ", | |
578 | + | Max issuable: ") + toString(maxCanSell)) + ", | |
579 | + | Source hash: ") + sourceHash) + ", | |
580 | + | Display cid: ") + displayCID) + ", | |
581 | + | Export cid: ") + exportCID) + ", | |
582 | + | Export hash: ") + exportHash) + ", | |
583 | + | Licence cid: ") + licenceCID) + ", | |
584 | + | Licence hash: ") + licenceHash) | |
585 | + | let issueNFT = Issue(("SA_" + toString((totalNFT + 1))), issueMeta, 1, 0, false) | |
586 | + | let idNFT = calculateAssetId(issueNFT) | |
587 | + | let sellStatus = if ((newAmountSold == maxCanSell)) | |
588 | + | then false | |
589 | + | else true | |
590 | + | [IntegerEntry(keyArtIssued(issuer, artId), newAmountSold), StringEntry(("nft_" + toBase58String(idNFT)), ((((((("art_sold_" + toString(newAmountSold)) + "_of_") + toString(maxCanSell)) + "_") + artId) + "_") + issuer)), StringEntry(((((((("art_sold_" + toString(newAmountSold)) + "_of_") + toString(maxCanSell)) + "_") + artId) + "_") + issuer), ((((((((((caller + "_") + toString(entryDate)) + "_") + id) + "_") + toString(artworkPrice)) + "_") + priceAssetId) + "_") + toBase58String(idNFT))), IntegerEntry("total_nft_issued", (totalNFT + 1)), BooleanEntry(keyArtOnSale(issuer, artId), sellStatus), issueNFT, ScriptTransfer(Address(fromBase58String(issuer)), amountForCreator, assetId), ScriptTransfer(Address(fromBase58String(feeReceiver)), amountForSign, assetId), ScriptTransfer(i.caller, 1, idNFT)] | |
591 | + | } | |
592 | + | } | |
593 | + | } | |
594 | + | } | |
595 | + | } | |
596 | + | ||
597 | + | ||
598 | + | ||
599 | + | @Callable(i) | |
600 | + | func deleteEntry (entry) = { | |
601 | + | let caller = toString(addressFromPublicKey(i.callerPublicKey)) | |
602 | + | if ((caller == chris)) | |
603 | + | then [DeleteEntry(entry)] | |
604 | + | else throw("no") | |
96 | 605 | } | |
97 | 606 | ||
98 | 607 |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 4 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | - | let userDapp = value(addressFromString("3MzVhsmmU8n1axekFJk1BRNwrmnCMH6Mn8F")) | |
4 | + | let unitTest = false | |
5 | + | ||
6 | + | let oracleFee = value(addressFromString("3N2s5RtaHPBenCsx2ECcoFRbYHx3noZhXW1")) | |
7 | + | ||
8 | + | let signDapp = value(addressFromString("3NC28hSivrmsTUXaYD1x6L362J4ZpUnoTdB")) | |
9 | + | ||
10 | + | let feeReceiver = "3N1E6tXddRoVaRfQ9dQ3vg5LaW2fsd8HKub" | |
11 | + | ||
12 | + | let userDapp = if (unitTest) | |
13 | + | then value(addressFromString("3Mt4RGMEyjSWYAocTPqd55wdHQQB3PU2UCm")) | |
14 | + | else value(addressFromString("3N8xXaYjE27Aa79d5hHrhHu9HaFoTBmhDEj")) | |
15 | + | ||
16 | + | let whitelistDapp = value(addressFromString("3Mt6jj1WUYqd7DT7dHEubs9jETLUXMcAB8k")) | |
17 | + | ||
18 | + | let signAssetId = base58'Gf9t8FA4H3ssoZPCwrg3KwUFCci8zuUFP9ssRsUY3s6a' | |
19 | + | ||
20 | + | let usdnAssetId = base58'25FEqEjRkqK6yCkiT7Lz6SAYz7gUFCtxfCChnrVFD5AT' | |
21 | + | ||
22 | + | let wavesAssetId = base58'' | |
23 | + | ||
24 | + | let chris = "3MsG6jPNCrVJUtYB7XJBxS7utWsXAf4n9Vp" | |
25 | + | ||
26 | + | let joep = "3Mzm4VLwsN9uZwbTMzPj3XuxV6kEfAR8UDN" | |
27 | + | ||
28 | + | let signCut = 7 | |
29 | + | ||
30 | + | let usdnCut = 10 | |
31 | + | ||
32 | + | let wavesCut = 10 | |
5 | 33 | ||
6 | 34 | let defaultLicence = "bafybeigisfqtyo2qdfsceh5fpcp7eeymrpeglp6edao2bmyhijbayvausy/licence.pdf" | |
7 | 35 | ||
8 | 36 | let defaultHashLicence = "3df79d34abbca99308e79cb94461c1893582604d68329a41fd4bec1885e6adb4" | |
37 | + | ||
38 | + | let dappRunning = match getBoolean(userDapp, "conf_dapp_is_running") { | |
39 | + | case a: Boolean => | |
40 | + | a | |
41 | + | case _ => | |
42 | + | true | |
43 | + | } | |
44 | + | ||
45 | + | let maintenanceMSG = match getString(userDapp, "conf_maintenance_msg") { | |
46 | + | case a: String => | |
47 | + | a | |
48 | + | case _ => | |
49 | + | "" | |
50 | + | } | |
51 | + | ||
52 | + | let userSuspended = "SUSPENDED" | |
53 | + | ||
54 | + | let userRemoved = "REMOVED" | |
55 | + | ||
56 | + | let userUnregistered = "UNREGISTERED" | |
57 | + | ||
58 | + | let userAllowed = "ALLOWED" | |
59 | + | ||
60 | + | func getStringByKeyFromUsers (key) = match getString(userDapp, key) { | |
61 | + | case a: String => | |
62 | + | a | |
63 | + | case _ => | |
64 | + | "" | |
65 | + | } | |
66 | + | ||
9 | 67 | ||
10 | 68 | func getStringByKey (key) = match getString(this, key) { | |
11 | 69 | case a: String => | |
12 | 70 | a | |
13 | 71 | case _ => | |
14 | 72 | "" | |
15 | 73 | } | |
74 | + | ||
75 | + | ||
76 | + | func getIntegerByKeyFromOracle (key) = match getInteger(oracleFee, key) { | |
77 | + | case a: Int => | |
78 | + | a | |
79 | + | case _ => | |
80 | + | throw("Integer undefine or 0 in oracle") | |
81 | + | } | |
82 | + | ||
83 | + | ||
84 | + | func getIntegerByKey (key) = match getInteger(this, key) { | |
85 | + | case i: Int => | |
86 | + | i | |
87 | + | case _ => | |
88 | + | 0 | |
89 | + | } | |
90 | + | ||
91 | + | ||
92 | + | func checkWhitelist (key) = match getInteger(whitelistDapp, key) { | |
93 | + | case a: Int => | |
94 | + | if ((a >= height)) | |
95 | + | then 1 | |
96 | + | else 0 | |
97 | + | case _ => | |
98 | + | 0 | |
99 | + | } | |
100 | + | ||
101 | + | ||
102 | + | func getBooleanByKey (key) = match getBoolean(this, key) { | |
103 | + | case i: Boolean => | |
104 | + | i | |
105 | + | case _ => | |
106 | + | false | |
107 | + | } | |
108 | + | ||
109 | + | ||
110 | + | func checkSignCertificate (signID,Owner,sha256Hash) = match getString(signDapp, ((("data_fc_" + signID) + "_") + Owner)) { | |
111 | + | case a: String => | |
112 | + | if (contains(a, sha256Hash)) | |
113 | + | then true | |
114 | + | else false | |
115 | + | case _ => | |
116 | + | false | |
117 | + | } | |
118 | + | ||
119 | + | ||
120 | + | func validateCID (cid) = if (contains(cid, "/")) | |
121 | + | then if (if ((75 > size(cid))) | |
122 | + | then (60 > size(split(cid, "/")[0])) | |
123 | + | else false) | |
124 | + | then (16 > size(split(cid, "/")[1])) | |
125 | + | else false | |
126 | + | else false | |
127 | + | ||
128 | + | ||
129 | + | func validateHash (hash) = (65 > size(hash)) | |
16 | 130 | ||
17 | 131 | ||
18 | 132 | func keyUserStatus (caller) = ("user_status_" + caller) | |
19 | 133 | ||
20 | 134 | ||
21 | 135 | func keyArtDate (caller,artId) = ((("art_date_" + artId) + "_") + caller) | |
22 | 136 | ||
23 | 137 | ||
24 | 138 | func keyArtName (caller,artId) = ((("art_name_" + artId) + "_") + caller) | |
25 | 139 | ||
26 | 140 | ||
27 | 141 | func keyArtDesc (caller,artId) = ((("art_desc_" + artId) + "_") + caller) | |
28 | 142 | ||
29 | 143 | ||
30 | 144 | func keyArtDisplayCid (caller,artId) = ((("art_display_cid_" + artId) + "_") + caller) | |
31 | 145 | ||
32 | 146 | ||
33 | 147 | func keyArtExportHash (caller,artId) = ((("art_export_hash_" + artId) + "_") + caller) | |
34 | 148 | ||
35 | 149 | ||
36 | 150 | func keyArtExportCid (caller,artId) = ((("art_export_cid_" + artId) + "_") + caller) | |
37 | 151 | ||
38 | 152 | ||
39 | 153 | func keyArtMaxMint (caller,artId) = ((("art_maxmint_" + artId) + "_") + caller) | |
40 | 154 | ||
41 | 155 | ||
42 | 156 | func keyArtSignID (caller,artId) = ((("art_signid_" + artId) + "_") + caller) | |
43 | 157 | ||
44 | 158 | ||
45 | 159 | func keyArtIssued (caller,artId) = ((("art_issued_" + artId) + "_") + caller) | |
46 | 160 | ||
47 | 161 | ||
48 | 162 | func keyArtOnSale (caller,artId) = ((("art_onsale_" + artId) + "_") + caller) | |
49 | 163 | ||
50 | 164 | ||
51 | 165 | func keyArtLicenceHash (caller,artId) = ((("art_licence_hash_" + artId) + "_") + caller) | |
52 | 166 | ||
53 | 167 | ||
54 | 168 | func keyArtLicenceCid (caller,artId) = ((("art_licence_cid_" + artId) + "_") + caller) | |
55 | 169 | ||
56 | 170 | ||
57 | 171 | func keyArtTags (caller,artId) = ((("art_tags_" + artId) + "_") + caller) | |
58 | 172 | ||
59 | 173 | ||
60 | 174 | func keyArtType (caller,artId) = ((("art_type_" + artId) + "_") + caller) | |
61 | 175 | ||
62 | 176 | ||
63 | 177 | func keyArtPrice (caller,artId) = ((("art_price_" + artId) + "_") + caller) | |
64 | 178 | ||
65 | 179 | ||
66 | 180 | func keyArtAssetIdAccepted (caller,artId) = ((("art_assetAccepted_" + artId) + "_") + caller) | |
67 | 181 | ||
68 | 182 | ||
69 | 183 | func keyArtFlag (caller,artId) = ((("art_flag_" + artId) + "_") + caller) | |
70 | 184 | ||
71 | 185 | ||
72 | 186 | func keyArtHashByTxidAddr (caller,txid) = ((("get_hashbytxidaddr_" + txid) + "_") + caller) | |
73 | 187 | ||
74 | 188 | ||
75 | 189 | func keyArtOwnerByHash (sha256Hash) = ("get_owner_by_hash_" + sha256Hash) | |
76 | 190 | ||
77 | 191 | ||
78 | 192 | func keyArtArtidBySignid (caller,signId) = ((("get_artidbysignid_" + signId) + "_") + caller) | |
79 | 193 | ||
80 | 194 | ||
81 | 195 | func keyArtTxidByHashOwner (sha256Hash,caller) = ("get_txid_by_hash_owner_" + toBase58String(sha256_16Kb(toBytes((sha256Hash + caller))))) | |
82 | 196 | ||
83 | 197 | ||
198 | + | func validateAllCID (cidDisplay,cidExport,cidLicence) = if (!(validateCID(cidDisplay))) | |
199 | + | then throw("Wrong Display CID") | |
200 | + | else if (!(validateCID(cidExport))) | |
201 | + | then throw("Wrong Export CID") | |
202 | + | else if (if ((cidLicence != "")) | |
203 | + | then !(validateCID(cidLicence)) | |
204 | + | else false) | |
205 | + | then throw("Wrong Licence CID") | |
206 | + | else true | |
207 | + | ||
208 | + | ||
209 | + | func validateAllHash (sha256Export,sha256Licence) = if (!(validateHash(sha256Export))) | |
210 | + | then throw("Export Hash 64 char. max") | |
211 | + | else if (!(validateHash(sha256Licence))) | |
212 | + | then throw("Licence Hash 64 char. max") | |
213 | + | else true | |
214 | + | ||
215 | + | ||
216 | + | func validateString (str,max) = if ((size(str) == 0)) | |
217 | + | then throw("Field cannot be is empty") | |
218 | + | else if ((size(str) > max)) | |
219 | + | then throw((str + " is too long")) | |
220 | + | else true | |
221 | + | ||
222 | + | ||
84 | 223 | @Callable(invoke) | |
85 | - | func addArtwork (sha256Hash,signID,name,description,tags,type,maxmint,cidDisplay,sha256Export,cidExport,sha256Licence,cidLicence) = { | |
86 | - | let artId = toBase58String(invoke.transactionId) | |
87 | - | let caller = toBase58String(invoke.caller.bytes) | |
88 | - | let timestamp = lastBlock.timestamp | |
89 | - | let licenceCID = if ((size(cidLicence) == 0)) | |
90 | - | then defaultLicence | |
91 | - | else cidLicence | |
92 | - | let licenceHash = if ((size(sha256Licence) == 0)) | |
93 | - | then defaultHashLicence | |
94 | - | else sha256Licence | |
95 | - | [StringEntry(keyArtOwnerByHash(sha256Hash), caller), StringEntry(keyArtTxidByHashOwner(sha256Hash, caller), artId), IntegerEntry(keyArtDate(caller, artId), timestamp), StringEntry(keyArtName(caller, artId), name), StringEntry(keyArtDesc(caller, artId), description), StringEntry(keyArtDisplayCid(caller, artId), cidDisplay), StringEntry(keyArtExportCid(caller, artId), cidExport), StringEntry(keyArtExportHash(caller, artId), sha256Export), StringEntry(keyArtLicenceHash(caller, artId), licenceHash), StringEntry(keyArtLicenceCid(caller, artId), licenceCID), StringEntry(keyArtType(caller, artId), type), StringEntry(keyArtTags(caller, artId), tags), IntegerEntry(keyArtMaxMint(caller, artId), maxmint), StringEntry(keyArtSignID(caller, artId), signID), IntegerEntry(keyArtIssued(caller, artId), 0), BooleanEntry(keyArtOnSale(caller, artId), false), StringEntry(keyArtArtidBySignid(caller, signID), artId), StringEntry(keyArtHashByTxidAddr(caller, artId), sha256Hash)] | |
224 | + | func addArtwork (sha256Hash,signID,name,description,tags,type,maxmint,cidDisplay,sha256Export,cidExport,sha256Licence,cidLicence) = if (!(dappRunning)) | |
225 | + | then throw(maintenanceMSG) | |
226 | + | else { | |
227 | + | let artId = toBase58String(invoke.transactionId) | |
228 | + | let caller = toBase58String(invoke.caller.bytes) | |
229 | + | let timestamp = lastBlock.timestamp | |
230 | + | let licenceCID = if ((size(cidLicence) == 0)) | |
231 | + | then defaultLicence | |
232 | + | else cidLicence | |
233 | + | let licenceHash = if ((size(sha256Licence) == 0)) | |
234 | + | then defaultHashLicence | |
235 | + | else sha256Licence | |
236 | + | let userIsRegistered = match getString(userDapp, ("user_status_" + caller)) { | |
237 | + | case s: String => | |
238 | + | s | |
239 | + | case _ => | |
240 | + | userUnregistered | |
241 | + | } | |
242 | + | if (if (isDefined(userIsRegistered)) | |
243 | + | then if ((userIsRegistered == userUnregistered)) | |
244 | + | then true | |
245 | + | else (userIsRegistered == userAllowed) | |
246 | + | else false) | |
247 | + | then throw("Register this account first with \"Account\" tab") | |
248 | + | else if ((userIsRegistered == userSuspended)) | |
249 | + | then throw("Account suspended") | |
250 | + | else if ((userIsRegistered == userRemoved)) | |
251 | + | then throw("Account removed") | |
252 | + | else if (!(validateAllCID(cidDisplay, cidExport, licenceCID))) | |
253 | + | then throw("Problem with CID") | |
254 | + | else if (!(validateHash(sha256Hash))) | |
255 | + | then throw("Hash should be 64 characters maximum") | |
256 | + | else if (!(validateAllHash(sha256Export, licenceHash))) | |
257 | + | then throw("Problem with Hashes") | |
258 | + | else if ((size(invoke.payments) == 0)) | |
259 | + | then throw("No payment attached") | |
260 | + | else { | |
261 | + | let payment = value(invoke.payments[0]) | |
262 | + | let amount = value(payment.amount) | |
263 | + | let assetId = if (if (isDefined(payment.assetId)) | |
264 | + | then (payment.assetId == signAssetId) | |
265 | + | else false) | |
266 | + | then payment.assetId | |
267 | + | else throw("Only SIGN token accepted") | |
268 | + | let currentCertificationPrice = getIntegerByKeyFromOracle(("certification_fee_" + toBase58String(signAssetId))) | |
269 | + | if ((amount != currentCertificationPrice)) | |
270 | + | then throw(("Payment amount should be " + toString(currentCertificationPrice))) | |
271 | + | else { | |
272 | + | let entryExist = getStringByKey(keyArtTxidByHashOwner(sha256Hash, caller)) | |
273 | + | if ((entryExist != "")) | |
274 | + | then throw("You already added it") | |
275 | + | else { | |
276 | + | let hashExist = getStringByKey(keyArtOwnerByHash(sha256Hash)) | |
277 | + | if ((hashExist != "")) | |
278 | + | then throw("Hash already registered") | |
279 | + | else { | |
280 | + | let isSignCertified = checkSignCertificate(signID, caller, sha256Hash) | |
281 | + | if (!(isSignCertified)) | |
282 | + | then throw("Sign Certificate not found for this address.") | |
283 | + | else if ((size(cidDisplay) == 0)) | |
284 | + | then throw("Display CID cannot be empty") | |
285 | + | else if (!(validateString(name, 100))) | |
286 | + | then throw("100 Char. max name") | |
287 | + | else if (!(validateString(description, 1000))) | |
288 | + | then throw("1000 Char. max description") | |
289 | + | else if ((size(split(tags, ",")) > 5)) | |
290 | + | then throw("5 tags max.") | |
291 | + | else if ((maxmint > 10)) | |
292 | + | then throw("10 editions max") | |
293 | + | else [StringEntry(keyArtOwnerByHash(sha256Hash), caller), StringEntry(keyArtTxidByHashOwner(sha256Hash, caller), artId), IntegerEntry(keyArtDate(caller, artId), timestamp), StringEntry(keyArtName(caller, artId), name), StringEntry(keyArtDesc(caller, artId), description), StringEntry(keyArtDisplayCid(caller, artId), cidDisplay), StringEntry(keyArtExportCid(caller, artId), cidExport), StringEntry(keyArtExportHash(caller, artId), sha256Export), StringEntry(keyArtLicenceHash(caller, artId), licenceHash), StringEntry(keyArtLicenceCid(caller, artId), licenceCID), StringEntry(keyArtType(caller, artId), type), StringEntry(keyArtTags(caller, artId), tags), IntegerEntry(keyArtMaxMint(caller, artId), maxmint), StringEntry(keyArtSignID(caller, artId), signID), IntegerEntry(keyArtIssued(caller, artId), 0), BooleanEntry(keyArtOnSale(caller, artId), false), StringEntry(keyArtArtidBySignid(caller, signID), artId), StringEntry(keyArtHashByTxidAddr(caller, artId), sha256Hash), ScriptTransfer(Address(fromBase58String(feeReceiver)), amount, assetId)] | |
294 | + | } | |
295 | + | } | |
296 | + | } | |
297 | + | } | |
298 | + | } | |
299 | + | ||
300 | + | ||
301 | + | ||
302 | + | @Callable(invoke) | |
303 | + | func updateArtwork (txid,name,description,tags,type,maxmint,cidDisplay,sha256Export,cidExport,sha256Licence,cidLicence) = if (!(dappRunning)) | |
304 | + | then throw(maintenanceMSG) | |
305 | + | else { | |
306 | + | let updateId = toBase58String(invoke.transactionId) | |
307 | + | let caller = toBase58String(invoke.caller.bytes) | |
308 | + | let licenceCID = if ((size(cidLicence) == 0)) | |
309 | + | then defaultLicence | |
310 | + | else cidLicence | |
311 | + | let licenceHash = if ((size(sha256Licence) == 0)) | |
312 | + | then defaultHashLicence | |
313 | + | else sha256Licence | |
314 | + | let userIsRegistered = getStringByKeyFromUsers(keyUserStatus(caller)) | |
315 | + | if (if ((userIsRegistered == "")) | |
316 | + | then true | |
317 | + | else (userIsRegistered == userAllowed)) | |
318 | + | then throw("Register first with \"User infos\"") | |
319 | + | else if ((userIsRegistered == userSuspended)) | |
320 | + | then throw("Account suspended") | |
321 | + | else if ((userIsRegistered == userRemoved)) | |
322 | + | then throw("Account removed") | |
323 | + | else if (!(validateAllCID(cidDisplay, cidExport, licenceCID))) | |
324 | + | then throw("Problem with CID") | |
325 | + | else if (!(validateAllHash(sha256Export, licenceHash))) | |
326 | + | then throw("Problem with Hashes") | |
327 | + | else { | |
328 | + | let entryExist = getStringByKey(keyArtName(caller, txid)) | |
329 | + | if ((entryExist == "")) | |
330 | + | then throw("Entry not found") | |
331 | + | else if (!(validateString(name, 100))) | |
332 | + | then throw("100 Char. max name") | |
333 | + | else if (!(validateString(description, 1000))) | |
334 | + | then throw("1000 Char. max description") | |
335 | + | else { | |
336 | + | let flag = getStringByKey(keyArtFlag(caller, txid)) | |
337 | + | if ((flag == "ILLEGAL")) | |
338 | + | then throw("Cannot update ILLEGAL artwork") | |
339 | + | else { | |
340 | + | let artworkMinted = match getInteger(this, keyArtIssued(caller, txid)) { | |
341 | + | case b: Int => | |
342 | + | if ((b == 0)) | |
343 | + | then false | |
344 | + | else true | |
345 | + | case _ => | |
346 | + | false | |
347 | + | } | |
348 | + | if ((size(split(tags, ",")) > 5)) | |
349 | + | then throw("5 tags max.") | |
350 | + | else if ((maxmint > 10)) | |
351 | + | then throw("10 editions max per artwork") | |
352 | + | else if (!(artworkMinted)) | |
353 | + | then [StringEntry(keyArtName(caller, txid), name), StringEntry(keyArtDesc(caller, txid), description), StringEntry(keyArtDisplayCid(caller, txid), cidDisplay), StringEntry(keyArtExportCid(caller, txid), cidExport), StringEntry(keyArtExportHash(caller, txid), sha256Export), StringEntry(keyArtLicenceCid(caller, txid), licenceCID), StringEntry(keyArtLicenceHash(caller, txid), licenceHash), IntegerEntry(keyArtMaxMint(caller, txid), maxmint), StringEntry(keyArtTags(caller, txid), tags), StringEntry(keyArtType(caller, txid), type)] | |
354 | + | else throw("Already minted") | |
355 | + | } | |
356 | + | } | |
357 | + | } | |
358 | + | } | |
359 | + | ||
360 | + | ||
361 | + | ||
362 | + | @Callable(i) | |
363 | + | func flagArtwork (artId,addr,flag) = { | |
364 | + | let caller = toString(addressFromPublicKey(i.callerPublicKey)) | |
365 | + | let id = toBase58String(i.transactionId) | |
366 | + | if (containsElement([chris, joep, toString(this)], caller)) | |
367 | + | then if ((flag == "CONSENT")) | |
368 | + | then [StringEntry(keyArtFlag(addr, artId), flag)] | |
369 | + | else if ((flag == "")) | |
370 | + | then [StringEntry(keyArtFlag(addr, artId), flag)] | |
371 | + | else if ((flag == "ILLEGAL")) | |
372 | + | then [StringEntry(keyArtFlag(addr, artId), flag), StringEntry(keyArtName(addr, artId), "ILLEGAL CONTENT"), StringEntry(keyArtDesc(addr, artId), "ILLEGAL CONTENT"), StringEntry(keyArtDisplayCid(addr, artId), ""), StringEntry(keyArtExportCid(addr, artId), ""), StringEntry(keyArtLicenceCid(addr, artId), "")] | |
373 | + | else throw(("Unknow status " + flag)) | |
374 | + | else throw("You are not allowed") | |
375 | + | } | |
376 | + | ||
377 | + | ||
378 | + | ||
379 | + | @Callable(i) | |
380 | + | func deleteArtwork (artId,addr) = { | |
381 | + | let caller = toString(addressFromPublicKey(i.callerPublicKey)) | |
382 | + | let id = toBase58String(i.transactionId) | |
383 | + | let addressToUse = if (containsElement([chris, joep, toString(this)], caller)) | |
384 | + | then addr | |
385 | + | else caller | |
386 | + | let entryExist = getStringByKey(keyArtName(addressToUse, artId)) | |
387 | + | if ((entryExist == "")) | |
388 | + | then throw("No art matching for this address") | |
389 | + | else { | |
390 | + | let artworkMinted = match getInteger(this, keyArtIssued(addressToUse, artId)) { | |
391 | + | case b: Int => | |
392 | + | if ((b != 0)) | |
393 | + | then true | |
394 | + | else false | |
395 | + | case _ => | |
396 | + | false | |
397 | + | } | |
398 | + | let maxMint = getIntegerByKey(keyArtMaxMint(addressToUse, artId)) | |
399 | + | let sha256Hash = getStringByKey(keyArtHashByTxidAddr(addressToUse, artId)) | |
400 | + | if ((sha256Hash == "")) | |
401 | + | then throw("No art hash matching for this address") | |
402 | + | else { | |
403 | + | let signID = getStringByKey(keyArtSignID(addressToUse, artId)) | |
404 | + | if ((signID == "")) | |
405 | + | then throw("No SIGN ID matching") | |
406 | + | else { | |
407 | + | let dataToDelete = [DeleteEntry(keyArtDate(addressToUse, artId)), DeleteEntry(keyArtName(addressToUse, artId)), DeleteEntry(keyArtDesc(addressToUse, artId)), DeleteEntry(keyArtDisplayCid(addressToUse, artId)), DeleteEntry(keyArtExportCid(addressToUse, artId)), DeleteEntry(keyArtExportHash(addressToUse, artId)), DeleteEntry(keyArtLicenceHash(addressToUse, artId)), DeleteEntry(keyArtLicenceCid(addressToUse, artId)), DeleteEntry(keyArtType(addressToUse, artId)), DeleteEntry(keyArtTags(addressToUse, artId)), DeleteEntry(keyArtMaxMint(addressToUse, artId)), DeleteEntry(keyArtSignID(addressToUse, artId)), DeleteEntry(keyArtIssued(addressToUse, artId)), DeleteEntry(keyArtOnSale(addressToUse, artId)), DeleteEntry(keyArtFlag(addr, artId)), DeleteEntry(keyArtOwnerByHash(sha256Hash)), DeleteEntry(keyArtArtidBySignid(addressToUse, signID)), DeleteEntry(keyArtTxidByHashOwner(sha256Hash, addressToUse))] | |
408 | + | if (!(artworkMinted)) | |
409 | + | then if (!(dappRunning)) | |
410 | + | then throw(maintenanceMSG) | |
411 | + | else dataToDelete | |
412 | + | else if (containsElement([chris, toString(this)], caller)) | |
413 | + | then dataToDelete | |
414 | + | else throw("Art already minted, cannot delete") | |
415 | + | } | |
416 | + | } | |
417 | + | } | |
418 | + | } | |
419 | + | ||
420 | + | ||
421 | + | ||
422 | + | @Callable(i) | |
423 | + | func sellArtwork (artId,price,maxMint,assetId) = if (!(dappRunning)) | |
424 | + | then throw(maintenanceMSG) | |
425 | + | else { | |
426 | + | let id = toBase58String(i.transactionId) | |
427 | + | let caller = toBase58String(i.caller.bytes) | |
428 | + | let sellDate = lastBlock.timestamp | |
429 | + | let artworkName = getStringByKey(keyArtName(caller, artId)) | |
430 | + | if ((artworkName == "")) | |
431 | + | then throw("This art doesn't match match your account") | |
432 | + | else { | |
433 | + | let exportCID = getStringByKey(keyArtExportCid(caller, artId)) | |
434 | + | if ((size(split(exportCID, "/")[0]) != 59)) | |
435 | + | then throw("You cannot sell art with no export file") | |
436 | + | else { | |
437 | + | let exportHash = getStringByKey(keyArtExportHash(caller, artId)) | |
438 | + | if ((size(exportHash) != 64)) | |
439 | + | then throw("You cannot sell art with no export hash") | |
440 | + | else if (if (if ((assetId != toBase58String(signAssetId))) | |
441 | + | then (assetId != toBase58String(wavesAssetId)) | |
442 | + | else false) | |
443 | + | then (assetId != toBase58String(usdnAssetId)) | |
444 | + | else false) | |
445 | + | then throw("Only SIGN, USDN or WAVES accepted") | |
446 | + | else { | |
447 | + | let minSellWaves = if (unitTest) | |
448 | + | then 1 | |
449 | + | else getIntegerByKeyFromOracle("waves_min_sell") | |
450 | + | let minSellUsdn = 1000000 | |
451 | + | let minSellSign = if (unitTest) | |
452 | + | then 1 | |
453 | + | else (getIntegerByKeyFromOracle(("certification_fee_" + toBase58String(signAssetId))) * 2) | |
454 | + | if (if (if (if (if ((assetId == toBase58String(usdnAssetId))) | |
455 | + | then (minSellUsdn > price) | |
456 | + | else false) | |
457 | + | then (price != 0) | |
458 | + | else false) | |
459 | + | then true | |
460 | + | else if (if ((assetId == toBase58String(signAssetId))) | |
461 | + | then (minSellSign > price) | |
462 | + | else false) | |
463 | + | then (price != 0) | |
464 | + | else false) | |
465 | + | then true | |
466 | + | else if (if ((assetId == toBase58String(wavesAssetId))) | |
467 | + | then (minSellWaves > price) | |
468 | + | else false) | |
469 | + | then (price != 0) | |
470 | + | else false) | |
471 | + | then throw("Wrong minimum sell price") | |
472 | + | else { | |
473 | + | let userIsRegistered = getStringByKeyFromUsers(keyUserStatus(caller)) | |
474 | + | if (if ((userIsRegistered == "")) | |
475 | + | then true | |
476 | + | else (userIsRegistered == userAllowed)) | |
477 | + | then throw("Register this account first") | |
478 | + | else if ((userIsRegistered == userSuspended)) | |
479 | + | then throw("Account suspended") | |
480 | + | else if ((userIsRegistered == userRemoved)) | |
481 | + | then throw("Account deleted") | |
482 | + | else { | |
483 | + | let amountSold = getIntegerByKey(keyArtIssued(caller, artId)) | |
484 | + | let maxCanSell = getIntegerByKey(keyArtMaxMint(caller, artId)) | |
485 | + | if ((maxMint > 10)) | |
486 | + | then throw("10 editions max per artwork") | |
487 | + | else if (if ((amountSold != 0)) | |
488 | + | then (amountSold == maxCanSell) | |
489 | + | else false) | |
490 | + | then throw("Max edition reached.") | |
491 | + | else if (if ((amountSold > 0)) | |
492 | + | then (maxCanSell != maxMint) | |
493 | + | else false) | |
494 | + | then throw("Cannot change maximum issuable anymore") | |
495 | + | else { | |
496 | + | let sellStatus = if (if ((price > 0)) | |
497 | + | then (maxMint > 0) | |
498 | + | else false) | |
499 | + | then true | |
500 | + | else false | |
501 | + | [BooleanEntry(keyArtOnSale(caller, artId), sellStatus), IntegerEntry(keyArtPrice(caller, artId), price), IntegerEntry(keyArtMaxMint(caller, artId), maxMint), StringEntry(keyArtAssetIdAccepted(caller, artId), assetId)] | |
502 | + | } | |
503 | + | } | |
504 | + | } | |
505 | + | } | |
506 | + | } | |
507 | + | } | |
508 | + | } | |
509 | + | ||
510 | + | ||
511 | + | ||
512 | + | @Callable(i) | |
513 | + | func buyArtwork (artId,issuer) = if (!(dappRunning)) | |
514 | + | then throw(maintenanceMSG) | |
515 | + | else { | |
516 | + | let id = toBase58String(i.transactionId) | |
517 | + | let caller = toBase58String(i.caller.bytes) | |
518 | + | let totalNFT = getIntegerByKey("total_nft_issued") | |
519 | + | let signID = getStringByKey(keyArtSignID(issuer, artId)) | |
520 | + | let artworkName = getStringByKey(keyArtName(issuer, artId)) | |
521 | + | if ((artworkName == "")) | |
522 | + | then throw("Art doesn't exist") | |
523 | + | else { | |
524 | + | let displayCID = getStringByKey(keyArtDisplayCid(issuer, artId)) | |
525 | + | let exportCID = getStringByKey(keyArtExportCid(issuer, artId)) | |
526 | + | let exportHash = getStringByKey(keyArtExportHash(issuer, artId)) | |
527 | + | let licenceCID = getStringByKey(keyArtLicenceCid(issuer, artId)) | |
528 | + | let licenceHash = getStringByKey(keyArtLicenceHash(issuer, artId)) | |
529 | + | let description = take(getStringByKey(keyArtDesc(issuer, artId)), 50) | |
530 | + | let amountSold = getIntegerByKey(keyArtIssued(issuer, artId)) | |
531 | + | let artworkPrice = getIntegerByKey(keyArtPrice(issuer, artId)) | |
532 | + | let isOnSale = getBooleanByKey(keyArtOnSale(issuer, artId)) | |
533 | + | let priceAssetId = getStringByKey(keyArtAssetIdAccepted(issuer, artId)) | |
534 | + | let sourceHash = getStringByKey(keyArtHashByTxidAddr(issuer, artId)) | |
535 | + | let maxCanSell = getIntegerByKey(keyArtMaxMint(issuer, artId)) | |
536 | + | if (if (if ((0 >= artworkPrice)) | |
537 | + | then true | |
538 | + | else !(isOnSale)) | |
539 | + | then true | |
540 | + | else (0 >= maxCanSell)) | |
541 | + | then throw("Art not for sale") | |
542 | + | else { | |
543 | + | let payment = value(i.payments[0]) | |
544 | + | let amount = value(payment.amount) | |
545 | + | let assetId = if (!(isDefined(payment.assetId))) | |
546 | + | then unit | |
547 | + | else if (if ((size(priceAssetId) > 0)) | |
548 | + | then (toBase58String(value(payment.assetId)) == priceAssetId) | |
549 | + | else false) | |
550 | + | then payment.assetId | |
551 | + | else throw("Wrong asset id") | |
552 | + | if (if ((assetId == unit)) | |
553 | + | then (priceAssetId != "") | |
554 | + | else false) | |
555 | + | then throw("Wrong asset id") | |
556 | + | else { | |
557 | + | let isWhitelisted = checkWhitelist(issuer) | |
558 | + | let cut = if ((isWhitelisted == 1)) | |
559 | + | then 0 | |
560 | + | else if ((priceAssetId == toBase58String(signAssetId))) | |
561 | + | then 8 | |
562 | + | else 10 | |
563 | + | let amountForSign = fraction(amount, cut, 100) | |
564 | + | let amountForCreator = (amount - amountForSign) | |
565 | + | if ((amountSold == maxCanSell)) | |
566 | + | then throw("Art sold out") | |
567 | + | else if ((artworkPrice != amount)) | |
568 | + | then throw("Payment don't match") | |
569 | + | else { | |
570 | + | let newAmountSold = (amountSold + 1) | |
571 | + | let entryDate = lastBlock.timestamp | |
572 | + | let issueMeta = ((((((((((((((((((((((((((("Creator: " + issuer) + ", | |
573 | + | ArtID: ") + artId) + ", | |
574 | + | SignID: ") + signID) + ", | |
575 | + | Artwork name: ") + artworkName) + ", | |
576 | + | Artwork description: ") + description) + ", | |
577 | + | Issue: ") + toString(newAmountSold)) + "/") + toString(maxCanSell)) + ", | |
578 | + | Max issuable: ") + toString(maxCanSell)) + ", | |
579 | + | Source hash: ") + sourceHash) + ", | |
580 | + | Display cid: ") + displayCID) + ", | |
581 | + | Export cid: ") + exportCID) + ", | |
582 | + | Export hash: ") + exportHash) + ", | |
583 | + | Licence cid: ") + licenceCID) + ", | |
584 | + | Licence hash: ") + licenceHash) | |
585 | + | let issueNFT = Issue(("SA_" + toString((totalNFT + 1))), issueMeta, 1, 0, false) | |
586 | + | let idNFT = calculateAssetId(issueNFT) | |
587 | + | let sellStatus = if ((newAmountSold == maxCanSell)) | |
588 | + | then false | |
589 | + | else true | |
590 | + | [IntegerEntry(keyArtIssued(issuer, artId), newAmountSold), StringEntry(("nft_" + toBase58String(idNFT)), ((((((("art_sold_" + toString(newAmountSold)) + "_of_") + toString(maxCanSell)) + "_") + artId) + "_") + issuer)), StringEntry(((((((("art_sold_" + toString(newAmountSold)) + "_of_") + toString(maxCanSell)) + "_") + artId) + "_") + issuer), ((((((((((caller + "_") + toString(entryDate)) + "_") + id) + "_") + toString(artworkPrice)) + "_") + priceAssetId) + "_") + toBase58String(idNFT))), IntegerEntry("total_nft_issued", (totalNFT + 1)), BooleanEntry(keyArtOnSale(issuer, artId), sellStatus), issueNFT, ScriptTransfer(Address(fromBase58String(issuer)), amountForCreator, assetId), ScriptTransfer(Address(fromBase58String(feeReceiver)), amountForSign, assetId), ScriptTransfer(i.caller, 1, idNFT)] | |
591 | + | } | |
592 | + | } | |
593 | + | } | |
594 | + | } | |
595 | + | } | |
596 | + | ||
597 | + | ||
598 | + | ||
599 | + | @Callable(i) | |
600 | + | func deleteEntry (entry) = { | |
601 | + | let caller = toString(addressFromPublicKey(i.callerPublicKey)) | |
602 | + | if ((caller == chris)) | |
603 | + | then [DeleteEntry(entry)] | |
604 | + | else throw("no") | |
96 | 605 | } | |
97 | 606 | ||
98 | 607 |
github/deemru/w8io/169f3d6 52.45 ms ◑