tx · C7rToQQXBRwXA38nMpBW6m6Y8k3VgcqgtsZ8353SrM8

3PG2vMhK5CPqsCDodvLGzQ84QkoHXCJ3oNP:  -0.14000000 Waves

2021.01.15 12:45 [2418140] smart account 3PG2vMhK5CPqsCDodvLGzQ84QkoHXCJ3oNP > SELF 0.00000000 Waves

{ "type": 13, "id": "C7rToQQXBRwXA38nMpBW6m6Y8k3VgcqgtsZ8353SrM8", "fee": 14000000, "feeAssetId": null, "timestamp": 1610706121287, "version": 1, "sender": "3PG2vMhK5CPqsCDodvLGzQ84QkoHXCJ3oNP", "senderPublicKey": "5RM3w4ysmDbtgfswnVNPx7DQkNwVAG3RoxNFHgt6ToNU", "proofs": [ "43eaEwUXBtQikHLURFzMeNwJYaFqfibkNsQt326BZMRwcNLZhHks3tVkx9GSkYvma74Ljsg2qFP6eh7WKaSGqx5J", "Lq7Fn1Yt8Q5xBqZdKKhcwMh778trmvekiLYo8UhzFF6xZxCFyNo3RtVfoMNvU9oWmL8dzjARR9QTfM5JTg4sJ2C", "5jJJkNJF4JEvVQ8WH7tRQ4B2MdvLs6eAfpcHRgFCPpzjthNmWVKyrC5Y1vFe9BmE61vMoCBfkCULC6fFkq5ZSgAN", "52cpgvcukQ5Yzvv5Lz2LU8NTEBs6rbcmTzfnDWQECq4VD7NMUmvVUsECd6tGh8hKitcsMoMs7CatgvTDSAucZCj2" ], "script": "base64:AAIEAAAAAAAAAAkIAhIAEgMKAQgAAABKAQAAAA5nZXROdW1iZXJCeUtleQAAAAEAAAADa2V5CQEAAAALdmFsdWVPckVsc2UAAAACCQAEGgAAAAIFAAAABHRoaXMFAAAAA2tleQAAAAAAAAAAAAEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABAAAAA2tleQkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkABB0AAAACBQAAAAR0aGlzBQAAAANrZXkCAAAAAAEAAAAWZ2V0Qm9vbEJ5QWRkcmVzc0FuZEtleQAAAAIAAAAHYWRkcmVzcwAAAANrZXkJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQbAAAAAgUAAAAHYWRkcmVzcwUAAAADa2V5BwEAAAAYZ2V0U3RyaW5nQnlBZGRyZXNzQW5kS2V5AAAAAgAAAAdhZGRyZXNzAAAAA2tleQkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkABB0AAAACBQAAAAdhZGRyZXNzBQAAAANrZXkCAAAAAAEAAAAYZ2V0TnVtYmVyQnlBZGRyZXNzQW5kS2V5AAAAAgAAAAdhZGRyZXNzAAAAA2tleQkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkABBoAAAACBQAAAAdhZGRyZXNzBQAAAANrZXkAAAAAAAAAAAAAAAAAB1dBVkVMRVQAAAAAAAX14QAAAAAABVBBVUxJAAAAAAAAD0JAAAAAAApQQVVMSVNDQUxFAAAAAAAAAAAGAAAAAAhQUklDRUxFVAAAAAAAAA9CQAAAAAAETVVMVAAAAAAABfXhAAAAAAAJTVVMVFNDQUxFAAAAAAAAAAAIAAAAAA1NSU5PUkRFUlRPVEFMCQAAaAAAAAIAAAAAAAAAAAoFAAAAB1dBVkVMRVQAAAAABk1BWFJPSQAAAAAAAAAAXwAAAAAIQ0FOQ0VMRUQCAAAACGNhbmNlbGVkAAAAAANORVcCAAAAA25ldwAAAAAGRklMTEVEAgAAAAZmaWxsZWQAAAAACFByaWNlS2V5AgAAAAVwcmljZQAAAAAOTnNidEFzc2V0SWRLZXkCAAAADWJvbmRfYXNzZXRfaWQAAAAAEk5ldXRyaW5vQXNzZXRJZEtleQIAAAARbmV1dHJpbm9fYXNzZXRfaWQAAAAAEUJhbGFuY2VMb2NrZWRrS2V5AgAAAA1iYWxhbmNlX2xvY2tfAAAAABVXYXZlc0xvY2tlZEJhbGFuY2VLZXkJAAEsAAAAAgUAAAARQmFsYW5jZUxvY2tlZGtLZXkCAAAABXdhdmVzAAAAABhOZXV0cmlub0xvY2tlZEJhbGFuY2VLZXkJAAEsAAAAAgUAAAARQmFsYW5jZUxvY2tlZGtLZXkCAAAACG5ldXRyaW5vAAAAAA1GaXJzdE9yZGVyS2V5AgAAAAtvcmRlcl9maXJzdAEAAAASZ2V0Um9pQnlPcmRlcklkS2V5AAAAAQAAAAdvcmRlcklkCQABLAAAAAICAAAAEGRlYnVnX29yZGVyX3JvaV8FAAAAB29yZGVySWQBAAAAEGdldE9yZGVyUHJpY2VLZXkAAAABAAAAB29yZGVySWQJAAEsAAAAAgIAAAAMb3JkZXJfcHJpY2VfBQAAAAdvcmRlcklkAQAAABBnZXRPcmRlclRvdGFsS2V5AAAAAQAAAAdvcmRlcklkCQABLAAAAAICAAAADG9yZGVyX3RvdGFsXwUAAAAHb3JkZXJJZAEAAAAQZ2V0T3JkZXJPd25lcktleQAAAAEAAAAHb3JkZXJJZAkAASwAAAACAgAAAAxvcmRlcl9vd25lcl8FAAAAB29yZGVySWQBAAAAEWdldE9yZGVySGVpZ2h0S2V5AAAAAQAAAAdvcmRlcklkCQABLAAAAAICAAAADW9yZGVyX2hlaWdodF8FAAAAB29yZGVySWQBAAAAEWdldE9yZGVyU3RhdHVzS2V5AAAAAQAAAAdvcmRlcklkCQABLAAAAAICAAAADW9yZGVyX3N0YXR1c18FAAAAB29yZGVySWQBAAAAFmdldE9yZGVyRmlsbGVkVG90YWxLZXkAAAABAAAAB29yZGVySWQJAAEsAAAAAgIAAAATb3JkZXJfZmlsbGVkX3RvdGFsXwUAAAAHb3JkZXJJZAEAAAAPZ2V0UHJldk9yZGVyS2V5AAAAAQAAAAdvcmRlcklkCQABLAAAAAICAAAAC29yZGVyX3ByZXZfBQAAAAdvcmRlcklkAQAAAA9nZXROZXh0T3JkZXJLZXkAAAABAAAAB29yZGVySWQJAAEsAAAAAgIAAAALb3JkZXJfbmV4dF8FAAAAB29yZGVySWQBAAAAFmNvbnZlcnROZXV0cmlub1RvV2F2ZXMAAAACAAAABmFtb3VudAAAAAVwcmljZQkAAGsAAAADCQAAawAAAAMFAAAABmFtb3VudAUAAAAIUFJJQ0VMRVQFAAAABXByaWNlBQAAAAdXQVZFTEVUBQAAAAVQQVVMSQEAAAAWY29udmVydFdhdmVzVG9OZXV0cmlubwAAAAIAAAAGYW1vdW50AAAABXByaWNlCQAAawAAAAMJAABrAAAAAwUAAAAGYW1vdW50BQAAAAVwcmljZQUAAAAIUFJJQ0VMRVQFAAAABVBBVUxJBQAAAAdXQVZFTEVUAQAAABJjb252ZXJ0V2F2ZXNUb0JvbmQAAAACAAAABmFtb3VudAAAAAVwcmljZQkBAAAAFmNvbnZlcnRXYXZlc1RvTmV1dHJpbm8AAAACBQAAAAZhbW91bnQFAAAABXByaWNlAQAAABJjb252ZXJ0Qm9uZFRvV2F2ZXMAAAACAAAABmFtb3VudAAAAAVwcmljZQkBAAAAFmNvbnZlcnROZXV0cmlub1RvV2F2ZXMAAAACBQAAAAZhbW91bnQFAAAABXByaWNlAAAAABBuZXV0cmlub0NvbnRyYWN0CQEAAAAHQWRkcmVzcwAAAAEBAAAAGgFXcARipkeb6a1WaJTL74WMMIIgKJoIFJayAAAAAA9jb250cm9sQ29udHJhY3QJAQAAAAdBZGRyZXNzAAAAAQEAAAAaAVcjs60SXJOkyuw5/k9G1s1WTS37EPtjmHoAAAAAE2xpcXVpZGF0aW9uQ29udHJhY3QJAQAAAAdBZGRyZXNzAAAAAQEAAAAaAVca6knL+gp+ygh/KNWflY4Me2m1qTiRH0gAAAAAD25ldXRyaW5vQXNzZXRJZAEAAAAgtiYpwwT1zlORpA5LdSQvZIxRsfrfr1QpvUjSHSqyqtEAAAAAC25zYnRBc3NldElkAQAAACBV7sO+qgvwUOhxyBuqbnCepLHI/kouucHxzMVrD3iXSwAAAAAJaXNCbG9ja2VkCQEAAAAWZ2V0Qm9vbEJ5QWRkcmVzc0FuZEtleQAAAAIFAAAAD2NvbnRyb2xDb250cmFjdAIAAAAKaXNfYmxvY2tlZAAAAAAMY3VycmVudFByaWNlCQEAAAAYZ2V0TnVtYmVyQnlBZGRyZXNzQW5kS2V5AAAAAgUAAAAPY29udHJvbENvbnRyYWN0BQAAAAhQcmljZUtleQAAAAAVbmV1dHJpbm9Mb2NrZWRCYWxhbmNlCQEAAAAYZ2V0TnVtYmVyQnlBZGRyZXNzQW5kS2V5AAAAAgUAAAAQbmV1dHJpbm9Db250cmFjdAUAAAAYTmV1dHJpbm9Mb2NrZWRCYWxhbmNlS2V5AAAAAAdyZXNlcnZlCQAAZQAAAAIICQAD7wAAAAEFAAAAEG5ldXRyaW5vQ29udHJhY3QAAAAHcmVndWxhcgkBAAAAGGdldE51bWJlckJ5QWRkcmVzc0FuZEtleQAAAAIFAAAAEG5ldXRyaW5vQ29udHJhY3QFAAAAFVdhdmVzTG9ja2VkQmFsYW5jZUtleQAAAAAOcmVzZXJ2ZXNJblVzZG4JAQAAABZjb252ZXJ0V2F2ZXNUb05ldXRyaW5vAAAAAgUAAAAHcmVzZXJ2ZQUAAAAMY3VycmVudFByaWNlAAAAAA5uZXV0cmlub1N1cHBseQkAAGUAAAACCQAAZQAAAAIJAABkAAAAAgUAAAAVbmV1dHJpbm9Mb2NrZWRCYWxhbmNlCAkBAAAABXZhbHVlAAAAAQkAA+wAAAABBQAAAA9uZXV0cmlub0Fzc2V0SWQAAAAIcXVhbnRpdHkJAAPwAAAAAgUAAAAQbmV1dHJpbm9Db250cmFjdAUAAAAPbmV1dHJpbm9Bc3NldElkCQAD8AAAAAIFAAAAE2xpcXVpZGF0aW9uQ29udHJhY3QFAAAAD25ldXRyaW5vQXNzZXRJZAAAAAAHZGVmaWNpdAkAAGUAAAACBQAAAA5uZXV0cmlub1N1cHBseQUAAAAOcmVzZXJ2ZXNJblVzZG4AAAAADWN1cnJlbnRNYXhSb2kJAABrAAAAAwUAAAAHZGVmaWNpdAAAAAAAAAAAZAUAAAAObmV1dHJpbm9TdXBwbHkAAAAADWN1cnJlbnRCck11bHQJAABrAAAAAwUAAAAOcmVzZXJ2ZXNJblVzZG4FAAAABE1VTFQFAAAADm5ldXRyaW5vU3VwcGx5AAAAAApmaXJzdE9yZGVyCQEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABBQAAAA1GaXJzdE9yZGVyS2V5AAAAAA1uc2J0U3VwcGx5TUFYCAkBAAAABXZhbHVlAAAAAQkAA+wAAAABBQAAAAtuc2J0QXNzZXRJZAAAAAhxdWFudGl0eQAAAAAKbnNidFN1cHBseQkAAGUAAAACBQAAAA1uc2J0U3VwcGx5TUFYCQAD8AAAAAIFAAAABHRoaXMFAAAAC25zYnRBc3NldElkAQAAAA1nZXRPcmRlclByaWNlAAAAAQAAAAJpZAkBAAAADmdldE51bWJlckJ5S2V5AAAAAQkBAAAAEGdldE9yZGVyUHJpY2VLZXkAAAABBQAAAAJpZAEAAAANZ2V0T3JkZXJUb3RhbAAAAAEAAAACaWQJAQAAAA5nZXROdW1iZXJCeUtleQAAAAEJAQAAABBnZXRPcmRlclRvdGFsS2V5AAAAAQUAAAACaWQBAAAADWdldE9yZGVyT3duZXIAAAABAAAAAmlkCQEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABCQEAAAAQZ2V0T3JkZXJPd25lcktleQAAAAEFAAAAAmlkAQAAAA5nZXRPcmRlclN0YXR1cwAAAAEAAAACaWQJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEJAQAAABFnZXRPcmRlclN0YXR1c0tleQAAAAEFAAAAAmlkAQAAABNnZXRPcmRlckZpbGxlZFRvdGFsAAAAAQAAAAJpZAkBAAAADmdldE51bWJlckJ5S2V5AAAAAQkBAAAAFmdldE9yZGVyRmlsbGVkVG90YWxLZXkAAAABBQAAAAJpZAEAAAAMZ2V0UHJldk9yZGVyAAAAAQAAAAJpZAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQkBAAAAD2dldFByZXZPcmRlcktleQAAAAEFAAAAAmlkAQAAAAxnZXROZXh0T3JkZXIAAAABAAAAAmlkCQEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABCQEAAAAPZ2V0TmV4dE9yZGVyS2V5AAAAAQUAAAACaWQBAAAADmdldFByaWNlRm9yUm9pAAAAAQAAAAtyb2lQZXJjZW50cwkAAGsAAAADCQAAZAAAAAIAAAAAAAAAAGQFAAAAC3JvaVBlcmNlbnRzBQAAAAxjdXJyZW50UHJpY2UAAAAAAAAAAGQBAAAAD2dldFJldmVyc2VQcmljZQAAAAEAAAAFcHJpY2UJAABpAAAAAgkAAGgAAAACBQAAAAhQUklDRUxFVAUAAAAIUFJJQ0VMRVQFAAAABXByaWNlAQAAABZjYWxjTnNidDJXYXZlc1ByaWNlUmF3AAAAAgAAAA1zcGVudFdhdmVzUmF3AAAAD3JlY2VpdmVkTnNidFJhdwkAAGsAAAADBQAAAA1zcGVudFdhdmVzUmF3CQAAaAAAAAIFAAAABVBBVUxJBQAAAAhQUklDRUxFVAUAAAAPcmVjZWl2ZWROc2J0UmF3AQAAAAlvcmRlckRhdGEAAAAHAAAAB29yZGVySWQAAAANdG90YWxXYXZlbGV0cwAAAA5maWxsZWRXYXZlbGV0cwAAAAVvd25lcgAAAAZzdGF0dXMAAAADcm9pAAAABXByaWNlCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAQZ2V0T3JkZXJQcmljZUtleQAAAAEFAAAAB29yZGVySWQFAAAABXByaWNlCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAQZ2V0T3JkZXJUb3RhbEtleQAAAAEFAAAAB29yZGVySWQFAAAADXRvdGFsV2F2ZWxldHMJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIJAQAAABZnZXRPcmRlckZpbGxlZFRvdGFsS2V5AAAAAQUAAAAHb3JkZXJJZAUAAAAOZmlsbGVkV2F2ZWxldHMJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAAEGdldE9yZGVyT3duZXJLZXkAAAABBQAAAAdvcmRlcklkBQAAAAVvd25lcgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkBAAAAEWdldE9yZGVySGVpZ2h0S2V5AAAAAQUAAAAHb3JkZXJJZAUAAAAGaGVpZ2h0CQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAABFnZXRPcmRlclN0YXR1c0tleQAAAAEFAAAAB29yZGVySWQFAAAABnN0YXR1cwkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACAgAAABlkZWJ1Z19vcmRlcl9jdXJyZW50UHJpY2VfBQAAAAdvcmRlcklkBQAAAAxjdXJyZW50UHJpY2UJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIJAQAAABJnZXRSb2lCeU9yZGVySWRLZXkAAAABBQAAAAdvcmRlcklkBQAAAANyb2kFAAAAA25pbAAAAAAESWR4QQAAAAAAAAAAAAAAAAAISWR4UGF1bEIAAAAAAAAAAAEAAAAADUlkeFdS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V2F2ZXNQcmljZVJhdwkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADBQAAAAxvd25lckFkZHJlc3MFAAAACm5zYnRBbW91bnQFAAAAC25zYnRBc3NldElkCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMFAAAAEG5ldXRyaW5vQ29udHJhY3QFAAAACHdhdmVzUGF5BQAAAAR1bml0CQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAICAAAAEWRlYnVnX2N1cnZlUmVzdWx0CQEAAAAFdG9TdHIAAAACAgAAAAtjdXJ2ZVJlc3VsdAUAAAALY3VydmVSZXN1bHQFAAAAA25pbAAAAAFpAQAAAAtjYW5jZWxPcmRlcgAAAAEAAAAHb3JkZXJJZAQAAAAFb3duZXIJAQAAAA1nZXRPcmRlck93bmVyAAAAAQUAAAAHb3JkZXJJZAQAAAAGYW1vdW50CQAAZQAAAAIJAQAAAA1nZXRPcmRlclRvdGFsAAAAAQUAAAAHb3JkZXJJZAkBAAAAE2dldE9yZGVyRmlsbGVkVG90YWwAAAABBQAAAAdvcmRlcklkBAAAAAZjYWxsZXIJAAQlAAAAAQgFAAAAAWkAAAAGY2FsbGVyBAAAAAluZXh0T3JkZXIJAQAAAAxnZXROZXh0T3JkZXIAAAABBQAAAAdvcmRlcklkBAAAAAlwcmV2T3JkZXIJAQAAAAxnZXRQcmV2T3JkZXIAAAABBQAAAAdvcmRlcklkAwkBAAAAAiE9AAAAAgkBAAAADmdldE9yZGVyU3RhdHVzAAAAAQUAAAAHb3JkZXJJZAUAAAADTkVXCQAAAgAAAAECAAAAFGludmFsaWQgb3JkZXIgc3RhdHVzCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIFAAAADUZpcnN0T3JkZXJLZXkDCQAAAAAAAAIFAAAACmZpcnN0T3JkZXIFAAAAB29yZGVySWQFAAAACW5leHRPcmRlcgUAAAAKZmlyc3RPcmRlcgkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACCQEAAAAPZ2V0TmV4dE9yZGVyS2V5AAAAAQUAAAAJcHJldk9yZGVyBQAAAAluZXh0T3JkZXIJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAAD2dldFByZXZPcmRlcktleQAAAAEFAAAACW5leHRPcmRlcgUAAAAJcHJldk9yZGVyCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAABFnZXRPcmRlclN0YXR1c0tleQAAAAEFAAAAB29yZGVySWQFAAAACENBTkNFTEVECQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAAGYW1vdW50BQAAAAR1bml0BQAAAANuaWwAAAABAAAAAnR4AQAAAAZ2ZXJpZnkAAAAABAAAABBwdWJLZXlBZG1pbnNMaXN0CQAETAAAAAICAAAALEdKZExTYUxpdjVLN3h1ZWphYzhtY1JjSG95bzNkUHJFU3J2a3RHM2E2TUFSCQAETAAAAAICAAAALEZXVmZmWXIyQUxtSE1lalptM1dxZUx6NlNkeW0zZ0xGR3RKbjRLVHd5VTV4CQAETAAAAAICAAAALDNXaDJMYVdjYjVnZzdLMnBQY1czRXA2RUF1UkJ6WWtBZ3JkcHQ0M2pUREZhCQAETAAAAAICAAAALDVXUlhGU2p3Y1RiTmZLY0pzOFpxWG1TU1dZc1NWSlV0TXZNcVpqNWhINE5jBQAAAANuaWwEAAAABWNvdW50CQAAZAAAAAIJAABkAAAAAgkAAGQAAAACAwkAAfQAAAADCAUAAAACdHgAAAAJYm9keUJ5dGVzCQABkQAAAAIIBQAAAAJ0eAAAAAZwcm9vZnMAAAAAAAAAAAAJAAJZAAAAAQkAAZEAAAACBQAAABBwdWJLZXlBZG1pbnNMaXN0AAAAAAAAAAAAAAAAAAAAAAABAAAAAAAAAAAAAwkAAfQAAAADCAUAAAACdHgAAAAJYm9keUJ5dGVzCQABkQAAAAIIBQAAAAJ0eAAAAAZwcm9vZnMAAAAAAAAAAAEJAAJZAAAAAQkAAZEAAAACBQAAABBwdWJLZXlBZG1pbnNMaXN0AAAAAAAAAAABAAAAAAAAAAABAAAAAAAAAAAAAwkAAfQAAAADCAUAAAACdHgAAAAJYm9keUJ5dGVzCQABkQAAAAIIBQAAAAJ0eAAAAAZwcm9vZnMAAAAAAAAAAAIJAAJZAAAAAQkAAZEAAAACBQAAABBwdWJLZXlBZG1pbnNMaXN0AAAAAAAAAAACAAAAAAAAAAABAAAAAAAAAAAAAwkAAfQAAAADCAUAAAACdHgAAAAJYm9keUJ5dGVzCQABkQAAAAIIBQAAAAJ0eAAAAAZwcm9vZnMAAAAAAAAAAAMJAAJZAAAAAQkAAZEAAAACBQAAABBwdWJLZXlBZG1pbnNMaXN0AAAAAAAAAAADAAAAAAAAAAACAAAAAAAAAAAACQAAZwAAAAIFAAAABWNvdW50AAAAAAAAAAAD/m3/4Q==", "chainId": 87, "height": 2418140, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 5hCaQEJxVdDKzeDFVMXEuV74mgkEM7r6oPihLHMYc6Re Next: 7vfb4wg6EuGYnjCRTQ6xyys5B8EeS1m1iaqCfudXCQ42 Diff:
OldNewDifferences
2020
2121 let PAULI = 1000000
2222
23+let PAULISCALE = 6
24+
2325 let PRICELET = 1000000
2426
2527 let MULT = 100000000
2628
27-let SCALEMULT = 8
29+let MULTSCALE = 8
2830
2931 let MINORDERTOTAL = (10 * WAVELET)
3032
3840
3941 let PriceKey = "price"
4042
41-let BondAssetIdKey = "bond_asset_id"
43+let NsbtAssetIdKey = "bond_asset_id"
4244
4345 let NeutrinoAssetIdKey = "neutrino_asset_id"
4446
9799
98100 let neutrinoAssetId = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p'
99101
100-let bondAssetId = base58'6nSpVyNH7yM69eg446wrQR94ipbbcmZMU1ENPwanC97g'
102+let nsbtAssetId = base58'6nSpVyNH7yM69eg446wrQR94ipbbcmZMU1ENPwanC97g'
101103
102104 let isBlocked = getBoolByAddressAndKey(controlContract, "is_blocked")
103105
118120 let currentBrMult = fraction(reservesInUsdn, MULT, neutrinoSupply)
119121
120122 let firstOrder = getStringByKey(FirstOrderKey)
123+
124+let nsbtSupplyMAX = value(assetInfo(nsbtAssetId)).quantity
125+
126+let nsbtSupply = (nsbtSupplyMAX - assetBalance(this, nsbtAssetId))
121127
122128 func getOrderPrice (id) = getNumberByKey(getOrderPriceKey(id))
123129
152158 func orderData (orderId,totalWavelets,filledWavelets,owner,status,roi,price) = [IntegerEntry(getOrderPriceKey(orderId), price), IntegerEntry(getOrderTotalKey(orderId), totalWavelets), IntegerEntry(getOrderFilledTotalKey(orderId), filledWavelets), StringEntry(getOrderOwnerKey(orderId), owner), IntegerEntry(getOrderHeightKey(orderId), height), StringEntry(getOrderStatusKey(orderId), status), IntegerEntry(("debug_order_currentPrice_" + orderId), currentPrice), IntegerEntry(getRoiByOrderIdKey(orderId), roi)]
153159
154160
155-func internalSellBond (pFirstOrder,pNextOrder,pFilledTotal,pRoi,pPrice,pPaymentWavelets,orderOwnerAddress,instantOrder) = {
156- let priceWavesByBondCents = getPriceForRoi(pRoi)
157- let remainedTotal = (pPaymentWavelets - pFilledTotal)
158- let fillableOrderAmount = convertWavesToBond(remainedTotal, priceWavesByBondCents)
159- let totalOrderWaveletesRequired = convertBondToWaves(fillableOrderAmount, priceWavesByBondCents)
160- let nbTokensSellCondition = (currentMaxRoi >= pRoi)
161- if (!(nbTokensSellCondition))
162- then throw(("innapropriate roi: " + toString(pRoi)))
163- else if ((totalOrderWaveletesRequired == 0))
164- then throw("cannot fill order at the moment")
165- else {
166- let changeWavelets = (remainedTotal - totalOrderWaveletesRequired)
167- let writeSetData = if (instantOrder)
168- then orderData(pFirstOrder, pPaymentWavelets, (pFilledTotal + totalOrderWaveletesRequired), toBase58String(orderOwnerAddress.bytes), FILLED, pRoi, currentPrice)
169- else [IntegerEntry(getOrderFilledTotalKey(pFirstOrder), (pFilledTotal + totalOrderWaveletesRequired)), StringEntry(getOrderStatusKey(pFirstOrder), FILLED), StringEntry(getPrevOrderKey(pNextOrder), ""), StringEntry(FirstOrderKey, pNextOrder)]
170- (writeSetData ++ [ScriptTransfer(orderOwnerAddress, fillableOrderAmount, bondAssetId), ScriptTransfer(neutrinoContract, totalOrderWaveletesRequired, unit), ScriptTransfer(orderOwnerAddress, changeWavelets, unit)])
171- }
161+let IdxA = 0
162+
163+let IdxPaulB = 1
164+
165+let IdxWResAsUsdn = 2
166+
167+let IdxMultBR = 3
168+
169+let IdxMultPower = 4
170+
171+let IdxMultExpInPower = 5
172+
173+let IdxMultK = 6
174+
175+func calculateK (wRaw,uRaw,price,mRaw,sRaw) = {
176+ let EXP = 271828100
177+ let a = valueOrElse(getInteger(this, "nsbtCurveParam_a"), 3)
178+ let paulB = valueOrElse(getInteger(this, "nsbtCurveParam_b"), ((3 * PAULI) / 10))
179+ let wReservesInUsdn = convertWavesToNeutrino(wRaw, price)
180+ let multBR = fraction(wReservesInUsdn, MULT, uRaw)
181+ let multPower = (a * (multBR - (1 * MULT)))
182+ let multExpInPower = pow(EXP, MULTSCALE, multPower, MULTSCALE, MULTSCALE, DOWN)
183+ let multK = fraction(paulB, multExpInPower, PAULI)
184+[a, paulB, wReservesInUsdn, multBR, multPower, multExpInPower, multK]
172185 }
173186
174187
175-func internalAddBuyBondOrder (roi,price,prevOrder,inv,instantOrder) = {
176- let pmt = value(inv.payments[0])
177- let newOrderId = toBase58String(inv.transactionId)
178- if (isBlocked)
179- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
180- else if ((MINORDERTOTAL > pmt.amount))
181- then throw(("min order total equals " + toString(MINORDERTOTAL)))
182- else if ((roi > MAXROI))
183- then throw("max setOrder ROI is 95%")
184- else if ((0 > roi))
185- then throw("can't place order with negative roi")
186- else if ((roi == 0))
187- then throw("roi should not be equal to 0")
188- else if (isDefined(pmt.assetId))
189- then throw("can use waves only")
190- else if ((getOrderOwner(newOrderId) != ""))
191- then throw("order exists")
192- else if (if ((prevOrder != ""))
193- then (getOrderStatus(prevOrder) != NEW)
194- else false)
195- then throw("prev order status is not new")
196- else {
197- let isNewOrderAtFirstPosition = (prevOrder == "")
198- let owner = toString(inv.caller)
199- let nextOrder = if (isNewOrderAtFirstPosition)
200- then firstOrder
201- else getNextOrder(prevOrder)
202- let nextOrderRoi = getNumberByKey(getRoiByOrderIdKey(nextOrder))
203- let isNextOrderError = if (if ((nextOrder != ""))
204- then (roi >= nextOrderRoi)
205- else false)
206- then true
207- else false
208- let prevOrderRoi = getNumberByKey(getRoiByOrderIdKey(prevOrder))
209- let isPrevOrderError = if (if ((prevOrder != ""))
210- then (prevOrderRoi > roi)
211- else false)
212- then true
213- else false
214- if (if (isNextOrderError)
215- then true
216- else isPrevOrderError)
217- then throw(((("invalid order isPrevOrderError:" + toString(isPrevOrderError)) + " isNextOrderError:") + toString(isNextOrderError)))
218- else if (if (isNewOrderAtFirstPosition)
219- then (currentMaxRoi >= roi)
220- else false)
221- then internalSellBond(newOrderId, nextOrder, 0, roi, price, pmt.amount, inv.caller, true)
222- else if (instantOrder)
223- then throw("Instant order couldn't be added into waiting queue")
224- else ([StringEntry(getPrevOrderKey(newOrderId), prevOrder), StringEntry(getNextOrderKey(newOrderId), nextOrder), StringEntry(getNextOrderKey(prevOrder), if ((prevOrder == ""))
225- then ""
226- else newOrderId), StringEntry(getPrevOrderKey(nextOrder), if ((nextOrder == ""))
227- then ""
228- else newOrderId), StringEntry(FirstOrderKey, if (if ((firstOrder == ""))
229- then true
230- else (firstOrder == nextOrder))
231- then newOrderId
232- else firstOrder)] ++ orderData(newOrderId, pmt.amount, 0, owner, NEW, roi, price))
233- }
188+func curveFunction (wRaw,uRaw,price,mRaw,sRaw,wavesPayRaw) = {
189+ let kCalcArray = calculateK(wRaw, uRaw, price, mRaw, sRaw)
190+ let multK = kCalcArray[IdxMultK]
191+ let usdnPay = convertWavesToNeutrino(wavesPayRaw, price)
192+ let multStepM1 = fraction(multK, mRaw, PAULI)
193+ let multStep2 = fraction(multStepM1, PAULI, (mRaw - sRaw))
194+ let multStepM3 = (fraction(usdnPay, MULT, PAULI) + fraction(multStep2, mRaw, PAULI))
195+ let multStep4 = fraction(multStepM1, MULT, multStepM3)
196+ let stepM5 = fraction(multStep4, mRaw, MULT)
197+ let nsbtAmountRaw = ((mRaw - sRaw) - stepM5)
198+[nsbtAmountRaw, usdnPay, wRaw, uRaw, mRaw, sRaw, kCalcArray[IdxA], kCalcArray[IdxPaulB], kCalcArray[IdxWResAsUsdn], price, kCalcArray[IdxMultBR], kCalcArray[IdxMultPower], kCalcArray[IdxMultExpInPower], multK, multStepM1, multStep2, multStepM3, multStep4, stepM5]
234199 }
235200
236201
237-func curveFunction (wRaw,uRaw,price) = {
238- let EXP = 271828100
239- let nsbtCurveParamA = valueOrElse(getInteger(this, "nsbtCurveParam_a"), 3)
240- let wReservesInUsdn = convertWavesToNeutrino(wRaw, price)
241- let brMult = fraction(wReservesInUsdn, MULT, uRaw)
242- let powerMult = (nsbtCurveParamA * (brMult - (1 * MULT)))
243- let expInPowerMult = pow(EXP, SCALEMULT, powerMult, SCALEMULT, SCALEMULT, DOWN)
244- let constCoeff = fraction(uRaw, PRICELET, (nsbtCurveParamA * price))
245- let finalResultIfPriceInUsdn = fraction(constCoeff, MULT, expInPowerMult)
246- let finalResult = fraction(finalResultIfPriceInUsdn, price, PRICELET)
247- $Tuple6(finalResult, wReservesInUsdn, brMult, powerMult, expInPowerMult, constCoeff)
248- }
249-
250-
251-func toStr (name,l) = (((((((((((((name + "[finalResult=") + toString(l._1)) + "wReservesInUsdn=") + toString(l._2)) + " brMult=") + toString(l._3)) + " powerMult=") + toString(l._4)) + " expInPowerMult=") + toString(l._5)) + " constCoeff=") + toString(l._6)) + "]")
252-
253-
254-func bigComplexity (i) = if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (!(isDefined(transferTransactionById(fromBase58String("PdLw5JK5ppfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k")))))
255- then !(isDefined(transferTransactionById(fromBase58String("adLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
256- else false)
257- then !(isDefined(transferTransactionById(fromBase58String("sdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
258- else false)
259- then !(isDefined(transferTransactionById(fromBase58String("ddLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
260- else false)
261- then !(isDefined(transferTransactionById(fromBase58String("fdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
262- else false)
263- then !(isDefined(transferTransactionById(fromBase58String("gdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
264- else false)
265- then !(isDefined(transferTransactionById(fromBase58String("hdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
266- else false)
267- then !(isDefined(transferTransactionById(fromBase58String("jdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
268- else false)
269- then !(isDefined(transferTransactionById(fromBase58String("kdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
270- else false)
271- then !(isDefined(transferTransactionById(fromBase58String("rdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
272- else false)
273- then !(isDefined(transferTransactionById(fromBase58String("tdLq5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
274- else false)
275- then !(isDefined(transferTransactionById(fromBase58String("tdLy5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
276- else false)
277- then !(isDefined(transferTransactionById(fromBase58String("tdLu5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
278- else false)
279- then !(isDefined(transferTransactionById(fromBase58String("tdLh5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
280- else false)
281- then !(isDefined(transferTransactionById(fromBase58String("1dL4JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
282- else false)
283- then !(isDefined(transferTransactionById(fromBase58String("2dL25JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
284- else false)
285- then !(isDefined(transferTransactionById(fromBase58String("3dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
286- else false)
287- then !(isDefined(transferTransactionById(fromBase58String("4dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
288- else false)
289- then !(isDefined(transferTransactionById(fromBase58String("5dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
290- else false)
291- then !(isDefined(transferTransactionById(fromBase58String("6dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
292- else false)
293- then !(isDefined(transferTransactionById(fromBase58String("7dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
294- else false)
295- then !(isDefined(transferTransactionById(fromBase58String("8dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
296- else false)
297- then !(isDefined(transferTransactionById(fromBase58String("91L95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
298- else false)
299- then !(isDefined(transferTransactionById(fromBase58String("92L95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
300- else false)
301- then !(isDefined(transferTransactionById(fromBase58String("93L95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
302- else false)
303- then !(isDefined(transferTransactionById(fromBase58String("94L95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
304- else false
202+func toStr (name,curveResult) = (((((((((((((((((((((((((((((((((((((((name + "[nsbtAmountRaw=") + toString(curveResult[0])) + " usdnPayment=") + toString(curveResult[1])) + " wRaw=") + toString(curveResult[2])) + " uRaw=") + toString(curveResult[3])) + " mRaw=") + toString(curveResult[4])) + " sRaw=") + toString(curveResult[5])) + " nsbtCurveParam_a=") + toString(curveResult[6])) + " nsbtCurveParam_b=") + toString(curveResult[7])) + " wReservesInUsdn=") + toString(curveResult[8])) + " price=") + toString(curveResult[9])) + " multBR=") + toString(curveResult[10])) + " multPower=") + toString(curveResult[11])) + " multExpInPower=") + toString(curveResult[12])) + " multK=") + toString(curveResult[13])) + " multStepM1=") + toString(curveResult[14])) + " multStep2=") + toString(curveResult[15])) + " multStepM3=") + toString(curveResult[16])) + " multStep4=") + toString(curveResult[17])) + " stepM5=") + toString(curveResult[18])) + "]")
305203
306204
307205 @Callable(i)
308-func buyNsbtInSurplus () = {
206+func buyNsbt () = {
309207 let pmt = value(i.payments[0])
310208 let pmtAmount = pmt.amount
209+ let wavesPay = pmtAmount
311210 let ownerAddress = i.caller
312211 if (isBlocked)
313212 then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
314- else if (((1 * MULT) > currentBrMult))
315- then throw(((("use instantBuyNsbtOrFail or addBuyBondOrder methods to buy nsbt when BR < 1: currentBR=" + toString(currentBrMult)) + "/") + toString(MULT)))
316- else if (isDefined(pmt.assetId))
317- then throw("can use waves only")
213+ else if (isDefined(pmt.assetId))
214+ then throw("can use waves only")
215+ else if (((10 * WAVELET) > pmtAmount))
216+ then throw("min 10 waves expected")
318217 else {
319- let minAmountFail = if (((10 * WAVELET) > pmtAmount))
320- then if ((bigComplexity(i) == true))
321- then true
322- else false
323- else false
324- if (minAmountFail)
325- then throw("big complexity + min 10 waves expected")
326- else if (((10 * WAVELET) > pmtAmount))
327- then throw("min 10 waves expected")
328- else {
329- let f0 = curveFunction(reserve, neutrinoSupply, currentPrice)
330- let f1 = curveFunction((reserve + pmtAmount), neutrinoSupply, currentPrice)
331- let nsbtAmount = (f0._1 - f1._1)
332- let nsbt2WavesPriceRaw = calcNsbt2WavesPriceRaw(pmtAmount, nsbtAmount)
333- let roi = (fraction(getReversePrice(nsbt2WavesPriceRaw), 100, currentPrice) - 100)
334- (orderData(toBase58String(i.transactionId), pmtAmount, pmtAmount, toString(ownerAddress), FILLED, roi, nsbt2WavesPriceRaw) ++ [ScriptTransfer(ownerAddress, nsbtAmount, bondAssetId), ScriptTransfer(neutrinoContract, pmtAmount, unit), StringEntry("debug_f0", toStr("f0", f0)), StringEntry("debug_f1", toStr("f1", f1))])
335- }
218+ let curveResult = curveFunction(reserve, neutrinoSupply, currentPrice, nsbtSupplyMAX, nsbtSupply, wavesPay)
219+ let nsbtAmount = curveResult[0]
220+ if ((0 >= nsbtAmount))
221+ then throw("nsbtAmount <= 0")
222+ else {
223+ let nsbt2WavesPriceRaw = calcNsbt2WavesPriceRaw(wavesPay, nsbtAmount)
224+ let roi = 0
225+ (orderData(toBase58String(i.transactionId), pmtAmount, pmtAmount, toString(ownerAddress), FILLED, roi, nsbt2WavesPriceRaw) ++ [ScriptTransfer(ownerAddress, nsbtAmount, nsbtAssetId), ScriptTransfer(neutrinoContract, wavesPay, unit), StringEntry("debug_curveResult", toStr("curveResult", curveResult))])
226+ }
336227 }
337- }
338-
339-
340-
341-@Callable(i)
342-func instantBuyNsbtOrFail (noLessThenRoi) = {
343- let pmt = value(i.payments[0])
344- let roi = currentMaxRoi
345- if ((noLessThenRoi > roi))
346- then throw(((("Current maxRoi=" + toString(roi)) + " is less then passed parameter noLessThenRoi=") + toString(noLessThenRoi)))
347- else {
348- let priceWavesByBondCents = getPriceForRoi(roi)
349- internalAddBuyBondOrder(roi, getReversePrice(priceWavesByBondCents), "", i, true)
350- }
351- }
352-
353-
354-
355-@Callable(i)
356-func addBuyBondOrder (price,prevOrder) = {
357- let pmt = value(i.payments[0])
358- let priceWavesByBondCents = getReversePrice(price)
359- if ((0 >= price))
360- then throw("price less zero")
361- else internalAddBuyBondOrder(fraction((priceWavesByBondCents - currentPrice), 100, currentPrice), price, prevOrder, i, false)
362228 }
363229
364230
370236 let caller = toString(i.caller)
371237 let nextOrder = getNextOrder(orderId)
372238 let prevOrder = getPrevOrder(orderId)
373- if (isBlocked)
374- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
375- else if ((owner != caller))
376- then throw("permission denied")
377- else if ((getOrderStatus(orderId) != NEW))
378- then throw("invalid order status")
379- else [StringEntry(FirstOrderKey, if ((firstOrder == orderId))
380- then nextOrder
381- else firstOrder), StringEntry(getNextOrderKey(prevOrder), nextOrder), StringEntry(getPrevOrderKey(nextOrder), prevOrder), StringEntry(getOrderStatusKey(orderId), CANCELED), ScriptTransfer(i.caller, amount, unit)]
239+ if ((getOrderStatus(orderId) != NEW))
240+ then throw("invalid order status")
241+ else [StringEntry(FirstOrderKey, if ((firstOrder == orderId))
242+ then nextOrder
243+ else firstOrder), StringEntry(getNextOrderKey(prevOrder), nextOrder), StringEntry(getPrevOrderKey(nextOrder), prevOrder), StringEntry(getOrderStatusKey(orderId), CANCELED), ScriptTransfer(i.caller, amount, unit)]
382244 }
383-
384-
385-
386-@Callable(i)
387-func sellBond () = if (isBlocked)
388- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
389- else if ((firstOrder == ""))
390- then throw("empty orderbook")
391- else {
392- let nextOrder = getNextOrder(firstOrder)
393- let filledTotal = getOrderFilledTotal(firstOrder)
394- let orderPrice = getOrderPrice(firstOrder)
395- let roi = getNumberByKey(getRoiByOrderIdKey(firstOrder))
396- let paymentWavelets = getOrderTotal(firstOrder)
397- let orderOwnerAddress = Address(fromBase58String(getOrderOwner(firstOrder)))
398- internalSellBond(firstOrder, nextOrder, filledTotal, roi, orderPrice, paymentWavelets, orderOwnerAddress, false)
399- }
400245
401246
402247 @Verifier(tx)
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 4 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 func getNumberByKey (key) = valueOrElse(getInteger(this, key), 0)
55
66
77 func getStringByKey (key) = valueOrElse(getString(this, key), "")
88
99
1010 func getBoolByAddressAndKey (address,key) = valueOrElse(getBoolean(address, key), false)
1111
1212
1313 func getStringByAddressAndKey (address,key) = valueOrElse(getString(address, key), "")
1414
1515
1616 func getNumberByAddressAndKey (address,key) = valueOrElse(getInteger(address, key), 0)
1717
1818
1919 let WAVELET = 100000000
2020
2121 let PAULI = 1000000
2222
23+let PAULISCALE = 6
24+
2325 let PRICELET = 1000000
2426
2527 let MULT = 100000000
2628
27-let SCALEMULT = 8
29+let MULTSCALE = 8
2830
2931 let MINORDERTOTAL = (10 * WAVELET)
3032
3133 let MAXROI = 95
3234
3335 let CANCELED = "canceled"
3436
3537 let NEW = "new"
3638
3739 let FILLED = "filled"
3840
3941 let PriceKey = "price"
4042
41-let BondAssetIdKey = "bond_asset_id"
43+let NsbtAssetIdKey = "bond_asset_id"
4244
4345 let NeutrinoAssetIdKey = "neutrino_asset_id"
4446
4547 let BalanceLockedkKey = "balance_lock_"
4648
4749 let WavesLockedBalanceKey = (BalanceLockedkKey + "waves")
4850
4951 let NeutrinoLockedBalanceKey = (BalanceLockedkKey + "neutrino")
5052
5153 let FirstOrderKey = "order_first"
5254
5355 func getRoiByOrderIdKey (orderId) = ("debug_order_roi_" + orderId)
5456
5557
5658 func getOrderPriceKey (orderId) = ("order_price_" + orderId)
5759
5860
5961 func getOrderTotalKey (orderId) = ("order_total_" + orderId)
6062
6163
6264 func getOrderOwnerKey (orderId) = ("order_owner_" + orderId)
6365
6466
6567 func getOrderHeightKey (orderId) = ("order_height_" + orderId)
6668
6769
6870 func getOrderStatusKey (orderId) = ("order_status_" + orderId)
6971
7072
7173 func getOrderFilledTotalKey (orderId) = ("order_filled_total_" + orderId)
7274
7375
7476 func getPrevOrderKey (orderId) = ("order_prev_" + orderId)
7577
7678
7779 func getNextOrderKey (orderId) = ("order_next_" + orderId)
7880
7981
8082 func convertNeutrinoToWaves (amount,price) = fraction(fraction(amount, PRICELET, price), WAVELET, PAULI)
8183
8284
8385 func convertWavesToNeutrino (amount,price) = fraction(fraction(amount, price, PRICELET), PAULI, WAVELET)
8486
8587
8688 func convertWavesToBond (amount,price) = convertWavesToNeutrino(amount, price)
8789
8890
8991 func convertBondToWaves (amount,price) = convertNeutrinoToWaves(amount, price)
9092
9193
9294 let neutrinoContract = Address(base58'3PC9BfRwJWWiw9AREE2B3eWzCks3CYtg4yo')
9395
9496 let controlContract = Address(base58'3P5Bfd58PPfNvBM2Hy8QfbcDqMeNtzg7KfP')
9597
9698 let liquidationContract = Address(base58'3P4PCxsJqMzQBALo8zANHtBDZRRquobHQp7')
9799
98100 let neutrinoAssetId = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p'
99101
100-let bondAssetId = base58'6nSpVyNH7yM69eg446wrQR94ipbbcmZMU1ENPwanC97g'
102+let nsbtAssetId = base58'6nSpVyNH7yM69eg446wrQR94ipbbcmZMU1ENPwanC97g'
101103
102104 let isBlocked = getBoolByAddressAndKey(controlContract, "is_blocked")
103105
104106 let currentPrice = getNumberByAddressAndKey(controlContract, PriceKey)
105107
106108 let neutrinoLockedBalance = getNumberByAddressAndKey(neutrinoContract, NeutrinoLockedBalanceKey)
107109
108110 let reserve = (wavesBalance(neutrinoContract).regular - getNumberByAddressAndKey(neutrinoContract, WavesLockedBalanceKey))
109111
110112 let reservesInUsdn = convertWavesToNeutrino(reserve, currentPrice)
111113
112114 let neutrinoSupply = (((neutrinoLockedBalance + value(assetInfo(neutrinoAssetId)).quantity) - assetBalance(neutrinoContract, neutrinoAssetId)) - assetBalance(liquidationContract, neutrinoAssetId))
113115
114116 let deficit = (neutrinoSupply - reservesInUsdn)
115117
116118 let currentMaxRoi = fraction(deficit, 100, neutrinoSupply)
117119
118120 let currentBrMult = fraction(reservesInUsdn, MULT, neutrinoSupply)
119121
120122 let firstOrder = getStringByKey(FirstOrderKey)
123+
124+let nsbtSupplyMAX = value(assetInfo(nsbtAssetId)).quantity
125+
126+let nsbtSupply = (nsbtSupplyMAX - assetBalance(this, nsbtAssetId))
121127
122128 func getOrderPrice (id) = getNumberByKey(getOrderPriceKey(id))
123129
124130
125131 func getOrderTotal (id) = getNumberByKey(getOrderTotalKey(id))
126132
127133
128134 func getOrderOwner (id) = getStringByKey(getOrderOwnerKey(id))
129135
130136
131137 func getOrderStatus (id) = getStringByKey(getOrderStatusKey(id))
132138
133139
134140 func getOrderFilledTotal (id) = getNumberByKey(getOrderFilledTotalKey(id))
135141
136142
137143 func getPrevOrder (id) = getStringByKey(getPrevOrderKey(id))
138144
139145
140146 func getNextOrder (id) = getStringByKey(getNextOrderKey(id))
141147
142148
143149 func getPriceForRoi (roiPercents) = fraction((100 + roiPercents), currentPrice, 100)
144150
145151
146152 func getReversePrice (price) = ((PRICELET * PRICELET) / price)
147153
148154
149155 func calcNsbt2WavesPriceRaw (spentWavesRaw,receivedNsbtRaw) = fraction(spentWavesRaw, (PAULI * PRICELET), receivedNsbtRaw)
150156
151157
152158 func orderData (orderId,totalWavelets,filledWavelets,owner,status,roi,price) = [IntegerEntry(getOrderPriceKey(orderId), price), IntegerEntry(getOrderTotalKey(orderId), totalWavelets), IntegerEntry(getOrderFilledTotalKey(orderId), filledWavelets), StringEntry(getOrderOwnerKey(orderId), owner), IntegerEntry(getOrderHeightKey(orderId), height), StringEntry(getOrderStatusKey(orderId), status), IntegerEntry(("debug_order_currentPrice_" + orderId), currentPrice), IntegerEntry(getRoiByOrderIdKey(orderId), roi)]
153159
154160
155-func internalSellBond (pFirstOrder,pNextOrder,pFilledTotal,pRoi,pPrice,pPaymentWavelets,orderOwnerAddress,instantOrder) = {
156- let priceWavesByBondCents = getPriceForRoi(pRoi)
157- let remainedTotal = (pPaymentWavelets - pFilledTotal)
158- let fillableOrderAmount = convertWavesToBond(remainedTotal, priceWavesByBondCents)
159- let totalOrderWaveletesRequired = convertBondToWaves(fillableOrderAmount, priceWavesByBondCents)
160- let nbTokensSellCondition = (currentMaxRoi >= pRoi)
161- if (!(nbTokensSellCondition))
162- then throw(("innapropriate roi: " + toString(pRoi)))
163- else if ((totalOrderWaveletesRequired == 0))
164- then throw("cannot fill order at the moment")
165- else {
166- let changeWavelets = (remainedTotal - totalOrderWaveletesRequired)
167- let writeSetData = if (instantOrder)
168- then orderData(pFirstOrder, pPaymentWavelets, (pFilledTotal + totalOrderWaveletesRequired), toBase58String(orderOwnerAddress.bytes), FILLED, pRoi, currentPrice)
169- else [IntegerEntry(getOrderFilledTotalKey(pFirstOrder), (pFilledTotal + totalOrderWaveletesRequired)), StringEntry(getOrderStatusKey(pFirstOrder), FILLED), StringEntry(getPrevOrderKey(pNextOrder), ""), StringEntry(FirstOrderKey, pNextOrder)]
170- (writeSetData ++ [ScriptTransfer(orderOwnerAddress, fillableOrderAmount, bondAssetId), ScriptTransfer(neutrinoContract, totalOrderWaveletesRequired, unit), ScriptTransfer(orderOwnerAddress, changeWavelets, unit)])
171- }
161+let IdxA = 0
162+
163+let IdxPaulB = 1
164+
165+let IdxWResAsUsdn = 2
166+
167+let IdxMultBR = 3
168+
169+let IdxMultPower = 4
170+
171+let IdxMultExpInPower = 5
172+
173+let IdxMultK = 6
174+
175+func calculateK (wRaw,uRaw,price,mRaw,sRaw) = {
176+ let EXP = 271828100
177+ let a = valueOrElse(getInteger(this, "nsbtCurveParam_a"), 3)
178+ let paulB = valueOrElse(getInteger(this, "nsbtCurveParam_b"), ((3 * PAULI) / 10))
179+ let wReservesInUsdn = convertWavesToNeutrino(wRaw, price)
180+ let multBR = fraction(wReservesInUsdn, MULT, uRaw)
181+ let multPower = (a * (multBR - (1 * MULT)))
182+ let multExpInPower = pow(EXP, MULTSCALE, multPower, MULTSCALE, MULTSCALE, DOWN)
183+ let multK = fraction(paulB, multExpInPower, PAULI)
184+[a, paulB, wReservesInUsdn, multBR, multPower, multExpInPower, multK]
172185 }
173186
174187
175-func internalAddBuyBondOrder (roi,price,prevOrder,inv,instantOrder) = {
176- let pmt = value(inv.payments[0])
177- let newOrderId = toBase58String(inv.transactionId)
178- if (isBlocked)
179- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
180- else if ((MINORDERTOTAL > pmt.amount))
181- then throw(("min order total equals " + toString(MINORDERTOTAL)))
182- else if ((roi > MAXROI))
183- then throw("max setOrder ROI is 95%")
184- else if ((0 > roi))
185- then throw("can't place order with negative roi")
186- else if ((roi == 0))
187- then throw("roi should not be equal to 0")
188- else if (isDefined(pmt.assetId))
189- then throw("can use waves only")
190- else if ((getOrderOwner(newOrderId) != ""))
191- then throw("order exists")
192- else if (if ((prevOrder != ""))
193- then (getOrderStatus(prevOrder) != NEW)
194- else false)
195- then throw("prev order status is not new")
196- else {
197- let isNewOrderAtFirstPosition = (prevOrder == "")
198- let owner = toString(inv.caller)
199- let nextOrder = if (isNewOrderAtFirstPosition)
200- then firstOrder
201- else getNextOrder(prevOrder)
202- let nextOrderRoi = getNumberByKey(getRoiByOrderIdKey(nextOrder))
203- let isNextOrderError = if (if ((nextOrder != ""))
204- then (roi >= nextOrderRoi)
205- else false)
206- then true
207- else false
208- let prevOrderRoi = getNumberByKey(getRoiByOrderIdKey(prevOrder))
209- let isPrevOrderError = if (if ((prevOrder != ""))
210- then (prevOrderRoi > roi)
211- else false)
212- then true
213- else false
214- if (if (isNextOrderError)
215- then true
216- else isPrevOrderError)
217- then throw(((("invalid order isPrevOrderError:" + toString(isPrevOrderError)) + " isNextOrderError:") + toString(isNextOrderError)))
218- else if (if (isNewOrderAtFirstPosition)
219- then (currentMaxRoi >= roi)
220- else false)
221- then internalSellBond(newOrderId, nextOrder, 0, roi, price, pmt.amount, inv.caller, true)
222- else if (instantOrder)
223- then throw("Instant order couldn't be added into waiting queue")
224- else ([StringEntry(getPrevOrderKey(newOrderId), prevOrder), StringEntry(getNextOrderKey(newOrderId), nextOrder), StringEntry(getNextOrderKey(prevOrder), if ((prevOrder == ""))
225- then ""
226- else newOrderId), StringEntry(getPrevOrderKey(nextOrder), if ((nextOrder == ""))
227- then ""
228- else newOrderId), StringEntry(FirstOrderKey, if (if ((firstOrder == ""))
229- then true
230- else (firstOrder == nextOrder))
231- then newOrderId
232- else firstOrder)] ++ orderData(newOrderId, pmt.amount, 0, owner, NEW, roi, price))
233- }
188+func curveFunction (wRaw,uRaw,price,mRaw,sRaw,wavesPayRaw) = {
189+ let kCalcArray = calculateK(wRaw, uRaw, price, mRaw, sRaw)
190+ let multK = kCalcArray[IdxMultK]
191+ let usdnPay = convertWavesToNeutrino(wavesPayRaw, price)
192+ let multStepM1 = fraction(multK, mRaw, PAULI)
193+ let multStep2 = fraction(multStepM1, PAULI, (mRaw - sRaw))
194+ let multStepM3 = (fraction(usdnPay, MULT, PAULI) + fraction(multStep2, mRaw, PAULI))
195+ let multStep4 = fraction(multStepM1, MULT, multStepM3)
196+ let stepM5 = fraction(multStep4, mRaw, MULT)
197+ let nsbtAmountRaw = ((mRaw - sRaw) - stepM5)
198+[nsbtAmountRaw, usdnPay, wRaw, uRaw, mRaw, sRaw, kCalcArray[IdxA], kCalcArray[IdxPaulB], kCalcArray[IdxWResAsUsdn], price, kCalcArray[IdxMultBR], kCalcArray[IdxMultPower], kCalcArray[IdxMultExpInPower], multK, multStepM1, multStep2, multStepM3, multStep4, stepM5]
234199 }
235200
236201
237-func curveFunction (wRaw,uRaw,price) = {
238- let EXP = 271828100
239- let nsbtCurveParamA = valueOrElse(getInteger(this, "nsbtCurveParam_a"), 3)
240- let wReservesInUsdn = convertWavesToNeutrino(wRaw, price)
241- let brMult = fraction(wReservesInUsdn, MULT, uRaw)
242- let powerMult = (nsbtCurveParamA * (brMult - (1 * MULT)))
243- let expInPowerMult = pow(EXP, SCALEMULT, powerMult, SCALEMULT, SCALEMULT, DOWN)
244- let constCoeff = fraction(uRaw, PRICELET, (nsbtCurveParamA * price))
245- let finalResultIfPriceInUsdn = fraction(constCoeff, MULT, expInPowerMult)
246- let finalResult = fraction(finalResultIfPriceInUsdn, price, PRICELET)
247- $Tuple6(finalResult, wReservesInUsdn, brMult, powerMult, expInPowerMult, constCoeff)
248- }
249-
250-
251-func toStr (name,l) = (((((((((((((name + "[finalResult=") + toString(l._1)) + "wReservesInUsdn=") + toString(l._2)) + " brMult=") + toString(l._3)) + " powerMult=") + toString(l._4)) + " expInPowerMult=") + toString(l._5)) + " constCoeff=") + toString(l._6)) + "]")
252-
253-
254-func bigComplexity (i) = if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (!(isDefined(transferTransactionById(fromBase58String("PdLw5JK5ppfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k")))))
255- then !(isDefined(transferTransactionById(fromBase58String("adLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
256- else false)
257- then !(isDefined(transferTransactionById(fromBase58String("sdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
258- else false)
259- then !(isDefined(transferTransactionById(fromBase58String("ddLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
260- else false)
261- then !(isDefined(transferTransactionById(fromBase58String("fdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
262- else false)
263- then !(isDefined(transferTransactionById(fromBase58String("gdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
264- else false)
265- then !(isDefined(transferTransactionById(fromBase58String("hdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
266- else false)
267- then !(isDefined(transferTransactionById(fromBase58String("jdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
268- else false)
269- then !(isDefined(transferTransactionById(fromBase58String("kdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
270- else false)
271- then !(isDefined(transferTransactionById(fromBase58String("rdLw5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
272- else false)
273- then !(isDefined(transferTransactionById(fromBase58String("tdLq5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
274- else false)
275- then !(isDefined(transferTransactionById(fromBase58String("tdLy5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
276- else false)
277- then !(isDefined(transferTransactionById(fromBase58String("tdLu5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
278- else false)
279- then !(isDefined(transferTransactionById(fromBase58String("tdLh5JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
280- else false)
281- then !(isDefined(transferTransactionById(fromBase58String("1dL4JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
282- else false)
283- then !(isDefined(transferTransactionById(fromBase58String("2dL25JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
284- else false)
285- then !(isDefined(transferTransactionById(fromBase58String("3dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
286- else false)
287- then !(isDefined(transferTransactionById(fromBase58String("4dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
288- else false)
289- then !(isDefined(transferTransactionById(fromBase58String("5dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
290- else false)
291- then !(isDefined(transferTransactionById(fromBase58String("6dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
292- else false)
293- then !(isDefined(transferTransactionById(fromBase58String("7dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
294- else false)
295- then !(isDefined(transferTransactionById(fromBase58String("8dL95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
296- else false)
297- then !(isDefined(transferTransactionById(fromBase58String("91L95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
298- else false)
299- then !(isDefined(transferTransactionById(fromBase58String("92L95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
300- else false)
301- then !(isDefined(transferTransactionById(fromBase58String("93L95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
302- else false)
303- then !(isDefined(transferTransactionById(fromBase58String("94L95JKdpqfSEqvmFJvchejorAkzd4BrWLvoFwvMGu5k"))))
304- else false
202+func toStr (name,curveResult) = (((((((((((((((((((((((((((((((((((((((name + "[nsbtAmountRaw=") + toString(curveResult[0])) + " usdnPayment=") + toString(curveResult[1])) + " wRaw=") + toString(curveResult[2])) + " uRaw=") + toString(curveResult[3])) + " mRaw=") + toString(curveResult[4])) + " sRaw=") + toString(curveResult[5])) + " nsbtCurveParam_a=") + toString(curveResult[6])) + " nsbtCurveParam_b=") + toString(curveResult[7])) + " wReservesInUsdn=") + toString(curveResult[8])) + " price=") + toString(curveResult[9])) + " multBR=") + toString(curveResult[10])) + " multPower=") + toString(curveResult[11])) + " multExpInPower=") + toString(curveResult[12])) + " multK=") + toString(curveResult[13])) + " multStepM1=") + toString(curveResult[14])) + " multStep2=") + toString(curveResult[15])) + " multStepM3=") + toString(curveResult[16])) + " multStep4=") + toString(curveResult[17])) + " stepM5=") + toString(curveResult[18])) + "]")
305203
306204
307205 @Callable(i)
308-func buyNsbtInSurplus () = {
206+func buyNsbt () = {
309207 let pmt = value(i.payments[0])
310208 let pmtAmount = pmt.amount
209+ let wavesPay = pmtAmount
311210 let ownerAddress = i.caller
312211 if (isBlocked)
313212 then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
314- else if (((1 * MULT) > currentBrMult))
315- then throw(((("use instantBuyNsbtOrFail or addBuyBondOrder methods to buy nsbt when BR < 1: currentBR=" + toString(currentBrMult)) + "/") + toString(MULT)))
316- else if (isDefined(pmt.assetId))
317- then throw("can use waves only")
213+ else if (isDefined(pmt.assetId))
214+ then throw("can use waves only")
215+ else if (((10 * WAVELET) > pmtAmount))
216+ then throw("min 10 waves expected")
318217 else {
319- let minAmountFail = if (((10 * WAVELET) > pmtAmount))
320- then if ((bigComplexity(i) == true))
321- then true
322- else false
323- else false
324- if (minAmountFail)
325- then throw("big complexity + min 10 waves expected")
326- else if (((10 * WAVELET) > pmtAmount))
327- then throw("min 10 waves expected")
328- else {
329- let f0 = curveFunction(reserve, neutrinoSupply, currentPrice)
330- let f1 = curveFunction((reserve + pmtAmount), neutrinoSupply, currentPrice)
331- let nsbtAmount = (f0._1 - f1._1)
332- let nsbt2WavesPriceRaw = calcNsbt2WavesPriceRaw(pmtAmount, nsbtAmount)
333- let roi = (fraction(getReversePrice(nsbt2WavesPriceRaw), 100, currentPrice) - 100)
334- (orderData(toBase58String(i.transactionId), pmtAmount, pmtAmount, toString(ownerAddress), FILLED, roi, nsbt2WavesPriceRaw) ++ [ScriptTransfer(ownerAddress, nsbtAmount, bondAssetId), ScriptTransfer(neutrinoContract, pmtAmount, unit), StringEntry("debug_f0", toStr("f0", f0)), StringEntry("debug_f1", toStr("f1", f1))])
335- }
218+ let curveResult = curveFunction(reserve, neutrinoSupply, currentPrice, nsbtSupplyMAX, nsbtSupply, wavesPay)
219+ let nsbtAmount = curveResult[0]
220+ if ((0 >= nsbtAmount))
221+ then throw("nsbtAmount <= 0")
222+ else {
223+ let nsbt2WavesPriceRaw = calcNsbt2WavesPriceRaw(wavesPay, nsbtAmount)
224+ let roi = 0
225+ (orderData(toBase58String(i.transactionId), pmtAmount, pmtAmount, toString(ownerAddress), FILLED, roi, nsbt2WavesPriceRaw) ++ [ScriptTransfer(ownerAddress, nsbtAmount, nsbtAssetId), ScriptTransfer(neutrinoContract, wavesPay, unit), StringEntry("debug_curveResult", toStr("curveResult", curveResult))])
226+ }
336227 }
337- }
338-
339-
340-
341-@Callable(i)
342-func instantBuyNsbtOrFail (noLessThenRoi) = {
343- let pmt = value(i.payments[0])
344- let roi = currentMaxRoi
345- if ((noLessThenRoi > roi))
346- then throw(((("Current maxRoi=" + toString(roi)) + " is less then passed parameter noLessThenRoi=") + toString(noLessThenRoi)))
347- else {
348- let priceWavesByBondCents = getPriceForRoi(roi)
349- internalAddBuyBondOrder(roi, getReversePrice(priceWavesByBondCents), "", i, true)
350- }
351- }
352-
353-
354-
355-@Callable(i)
356-func addBuyBondOrder (price,prevOrder) = {
357- let pmt = value(i.payments[0])
358- let priceWavesByBondCents = getReversePrice(price)
359- if ((0 >= price))
360- then throw("price less zero")
361- else internalAddBuyBondOrder(fraction((priceWavesByBondCents - currentPrice), 100, currentPrice), price, prevOrder, i, false)
362228 }
363229
364230
365231
366232 @Callable(i)
367233 func cancelOrder (orderId) = {
368234 let owner = getOrderOwner(orderId)
369235 let amount = (getOrderTotal(orderId) - getOrderFilledTotal(orderId))
370236 let caller = toString(i.caller)
371237 let nextOrder = getNextOrder(orderId)
372238 let prevOrder = getPrevOrder(orderId)
373- if (isBlocked)
374- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
375- else if ((owner != caller))
376- then throw("permission denied")
377- else if ((getOrderStatus(orderId) != NEW))
378- then throw("invalid order status")
379- else [StringEntry(FirstOrderKey, if ((firstOrder == orderId))
380- then nextOrder
381- else firstOrder), StringEntry(getNextOrderKey(prevOrder), nextOrder), StringEntry(getPrevOrderKey(nextOrder), prevOrder), StringEntry(getOrderStatusKey(orderId), CANCELED), ScriptTransfer(i.caller, amount, unit)]
239+ if ((getOrderStatus(orderId) != NEW))
240+ then throw("invalid order status")
241+ else [StringEntry(FirstOrderKey, if ((firstOrder == orderId))
242+ then nextOrder
243+ else firstOrder), StringEntry(getNextOrderKey(prevOrder), nextOrder), StringEntry(getPrevOrderKey(nextOrder), prevOrder), StringEntry(getOrderStatusKey(orderId), CANCELED), ScriptTransfer(i.caller, amount, unit)]
382244 }
383-
384-
385-
386-@Callable(i)
387-func sellBond () = if (isBlocked)
388- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
389- else if ((firstOrder == ""))
390- then throw("empty orderbook")
391- else {
392- let nextOrder = getNextOrder(firstOrder)
393- let filledTotal = getOrderFilledTotal(firstOrder)
394- let orderPrice = getOrderPrice(firstOrder)
395- let roi = getNumberByKey(getRoiByOrderIdKey(firstOrder))
396- let paymentWavelets = getOrderTotal(firstOrder)
397- let orderOwnerAddress = Address(fromBase58String(getOrderOwner(firstOrder)))
398- internalSellBond(firstOrder, nextOrder, filledTotal, roi, orderPrice, paymentWavelets, orderOwnerAddress, false)
399- }
400245
401246
402247 @Verifier(tx)
403248 func verify () = {
404249 let pubKeyAdminsList = ["GJdLSaLiv5K7xuejac8mcRcHoyo3dPrESrvktG3a6MAR", "FWVffYr2ALmHMejZm3WqeLz6Sdym3gLFGtJn4KTwyU5x", "3Wh2LaWcb5gg7K2pPcW3Ep6EAuRBzYkAgrdpt43jTDFa", "5WRXFSjwcTbNfKcJs8ZqXmSSWYsSVJUtMvMqZj5hH4Nc"]
405250 let count = ((((if (sigVerify(tx.bodyBytes, tx.proofs[0], fromBase58String(pubKeyAdminsList[0])))
406251 then 1
407252 else 0) + (if (sigVerify(tx.bodyBytes, tx.proofs[1], fromBase58String(pubKeyAdminsList[1])))
408253 then 1
409254 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[2], fromBase58String(pubKeyAdminsList[2])))
410255 then 1
411256 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[3], fromBase58String(pubKeyAdminsList[3])))
412257 then 2
413258 else 0))
414259 (count >= 3)
415260 }
416261

github/deemru/w8io/3ef1775 
134.92 ms